diff options
Diffstat (limited to 'compiler/src/Processors.hs')
-rw-r--r-- | compiler/src/Processors.hs | 144 |
1 files changed, 73 insertions, 71 deletions
diff --git a/compiler/src/Processors.hs b/compiler/src/Processors.hs index 159a425..02db325 100644 --- a/compiler/src/Processors.hs +++ b/compiler/src/Processors.hs | |||
@@ -25,20 +25,20 @@ module Processors | |||
25 | 25 | ||
26 | 26 | ||
27 | import Control.Exception (Exception, throwIO) | 27 | import Control.Exception (Exception, throwIO) |
28 | import Control.Monad (when) | ||
28 | import Data.Function ((&)) | 29 | import Data.Function ((&)) |
29 | import Data.Ratio ((%)) | ||
30 | import Data.Char (toLower) | 30 | import Data.Char (toLower) |
31 | import Text.Read (readMaybe) | ||
31 | 32 | ||
32 | import System.Directory hiding (copyFile) | 33 | import System.Directory hiding (copyFile) |
33 | import qualified System.Directory | 34 | import qualified System.Directory |
34 | import System.FilePath | 35 | import System.FilePath |
35 | 36 | ||
36 | import Codec.Picture | 37 | import System.Process (callProcess, readProcess) |
37 | import Codec.Picture.Extra -- TODO: compare DCT and bilinear (and Lanczos, but it's not implemented) | ||
38 | 38 | ||
39 | import Resource | 39 | import Resource |
40 | ( ItemProcessor, ThumbnailProcessor | 40 | ( ItemProcessor, ThumbnailProcessor |
41 | , GalleryItemProps(..), Resolution(..) ) | 41 | , GalleryItemProps(..), Resolution(..), Resource(..), Thumbnail(..) ) |
42 | 42 | ||
43 | import Files | 43 | import Files |
44 | 44 | ||
@@ -47,10 +47,8 @@ data ProcessingException = ProcessingException FilePath String deriving Show | |||
47 | instance Exception ProcessingException | 47 | instance Exception ProcessingException |
48 | 48 | ||
49 | 49 | ||
50 | data PictureFileFormat = Bmp | Jpg | Png | Tiff | Hdr | Gif | ||
51 | |||
52 | -- TODO: handle video, music, text... | 50 | -- TODO: handle video, music, text... |
53 | data Format = PictureFormat PictureFileFormat | Unknown | 51 | data Format = PictureFormat | Unknown |
54 | 52 | ||
55 | formatFromPath :: Path -> Format | 53 | formatFromPath :: Path -> Format |
56 | formatFromPath = | 54 | formatFromPath = |
@@ -60,14 +58,15 @@ formatFromPath = | |||
60 | . fileName | 58 | . fileName |
61 | where | 59 | where |
62 | fromExt :: String -> Format | 60 | fromExt :: String -> Format |
63 | fromExt ".bmp" = PictureFormat Bmp | 61 | fromExt ext = case ext of |
64 | fromExt ".jpg" = PictureFormat Jpg | 62 | ".bmp" -> PictureFormat |
65 | fromExt ".jpeg" = PictureFormat Jpg | 63 | ".jpg" -> PictureFormat |
66 | fromExt ".png" = PictureFormat Png | 64 | ".jpeg" -> PictureFormat |
67 | fromExt ".tiff" = PictureFormat Tiff | 65 | ".png" -> PictureFormat |
68 | fromExt ".hdr" = PictureFormat Hdr | 66 | ".tiff" -> PictureFormat |
69 | fromExt ".gif" = PictureFormat Gif | 67 | ".hdr" -> PictureFormat |
70 | fromExt _ = Unknown | 68 | ".gif" -> PictureFormat |
69 | _ -> Unknown | ||
71 | 70 | ||
72 | 71 | ||
73 | type FileProcessor = | 72 | type FileProcessor = |
@@ -80,43 +79,20 @@ copyFileProcessor inputPath outputPath = | |||
80 | (putStrLn $ "Copying:\t" ++ outputPath) | 79 | (putStrLn $ "Copying:\t" ++ outputPath) |
81 | >> ensureParentDir (flip System.Directory.copyFile) outputPath inputPath | 80 | >> ensureParentDir (flip System.Directory.copyFile) outputPath inputPath |
82 | 81 | ||
83 | resizeStaticImageUpTo :: PictureFileFormat -> Resolution -> FileProcessor | 82 | resizePictureUpTo :: Resolution -> FileProcessor |
84 | resizeStaticImageUpTo Bmp = resizeStaticGeneric readBitmap saveBmpImage | 83 | resizePictureUpTo maxResolution inputPath outputPath = |
85 | -- TODO: parameterise export quality for jpg | ||
86 | resizeStaticImageUpTo Jpg = resizeStaticGeneric readJpeg (saveJpgImage 80) | ||
87 | resizeStaticImageUpTo Png = resizeStaticGeneric readPng savePngImage | ||
88 | resizeStaticImageUpTo Tiff = resizeStaticGeneric readTiff saveTiffImage | ||
89 | resizeStaticImageUpTo Hdr = resizeStaticGeneric readHDR saveRadianceImage | ||
90 | resizeStaticImageUpTo Gif = resizeStaticGeneric readGif saveGifImage' | ||
91 | where | ||
92 | saveGifImage' :: StaticImageWriter | ||
93 | saveGifImage' outputPath image = | ||
94 | saveGifImage outputPath image | ||
95 | & either (throwIO . ProcessingException outputPath) id | ||
96 | |||
97 | |||
98 | type StaticImageReader = FilePath -> IO (Either String DynamicImage) | ||
99 | type StaticImageWriter = FilePath -> DynamicImage -> IO () | ||
100 | |||
101 | resizeStaticGeneric :: StaticImageReader -> StaticImageWriter -> Resolution -> FileProcessor | ||
102 | resizeStaticGeneric reader writer maxRes inputPath outputPath = | ||
103 | (putStrLn $ "Generating:\t" ++ outputPath) | 84 | (putStrLn $ "Generating:\t" ++ outputPath) |
104 | >> reader inputPath | 85 | >> ensureParentDir (flip resize) outputPath inputPath |
105 | >>= either (throwIO . ProcessingException inputPath) return | ||
106 | >>= return . (fitDynamicImage maxRes) | ||
107 | >>= ensureParentDir writer outputPath | ||
108 | |||
109 | fitDynamicImage :: Resolution -> DynamicImage -> DynamicImage | ||
110 | fitDynamicImage (Resolution boxWidth boxHeight) image = | ||
111 | convertRGBA8 image | ||
112 | & scaleBilinear targetWidth targetHeight | ||
113 | & ImageRGBA8 | ||
114 | where | 86 | where |
115 | picWidth = dynamicMap imageWidth image | 87 | maxSize :: Resolution -> String |
116 | picHeight = dynamicMap imageHeight image | 88 | maxSize Resolution{width, height} = show width ++ "x" ++ show height ++ ">" |
117 | resizeRatio = min (boxWidth % picWidth) (boxHeight % picHeight) | 89 | |
118 | targetWidth = floor $ resizeRatio * (picWidth % 1) | 90 | resize :: FileName -> FileName -> IO () |
119 | targetHeight = floor $ resizeRatio * (picHeight % 1) | 91 | resize input output = callProcess "magick" |
92 | [ input | ||
93 | , "-auto-orient" | ||
94 | , "-resize", maxSize maxResolution | ||
95 | , output ] | ||
120 | 96 | ||
121 | 97 | ||
122 | type Cache = FileProcessor -> FileProcessor | 98 | type Cache = FileProcessor -> FileProcessor |
@@ -130,7 +106,7 @@ withCached :: Cache | |||
130 | withCached processor inputPath outputPath = | 106 | withCached processor inputPath outputPath = |
131 | do | 107 | do |
132 | isDir <- doesDirectoryExist outputPath | 108 | isDir <- doesDirectoryExist outputPath |
133 | if isDir then removePathForcibly outputPath else noop | 109 | when isDir $ removePathForcibly outputPath |
134 | 110 | ||
135 | fileExists <- doesFileExist outputPath | 111 | fileExists <- doesFileExist outputPath |
136 | if fileExists then | 112 | if fileExists then |
@@ -141,11 +117,39 @@ withCached processor inputPath outputPath = | |||
141 | update | 117 | update |
142 | 118 | ||
143 | where | 119 | where |
144 | noop = return () | ||
145 | update = processor inputPath outputPath | 120 | update = processor inputPath outputPath |
146 | skip = putStrLn $ "Skipping:\t" ++ outputPath | 121 | skip = putStrLn $ "Skipping:\t" ++ outputPath |
147 | 122 | ||
148 | 123 | ||
124 | resourceAt :: FilePath -> Path -> IO Resource | ||
125 | resourceAt fsPath resPath = getModificationTime fsPath >>= return . Resource resPath | ||
126 | |||
127 | getImageResolution :: FilePath -> IO Resolution | ||
128 | getImageResolution fsPath = | ||
129 | readProcess "magick" ["identify", "-format", "%w %h", firstFrame] [] | ||
130 | >>= parseResolution . break (== ' ') | ||
131 | where | ||
132 | firstFrame :: FilePath | ||
133 | firstFrame = fsPath ++ "[0]" | ||
134 | |||
135 | parseResolution :: (String, String) -> IO Resolution | ||
136 | parseResolution (widthString, heightString) = | ||
137 | case (readMaybe widthString, readMaybe heightString) of | ||
138 | (Just w, Just h) -> return $ Resolution w h | ||
139 | _ -> throwIO $ ProcessingException fsPath "Unable to read image resolution." | ||
140 | |||
141 | getPictureProps :: ItemDescriber | ||
142 | getPictureProps fsPath resource = | ||
143 | getImageResolution fsPath | ||
144 | >>= return . Picture resource | ||
145 | |||
146 | |||
147 | type ItemDescriber = | ||
148 | FilePath | ||
149 | -> Resource | ||
150 | -> IO GalleryItemProps | ||
151 | |||
152 | |||
149 | type ItemFileProcessor = | 153 | type ItemFileProcessor = |
150 | FileName -- ^ Input base path | 154 | FileName -- ^ Input base path |
151 | -> FileName -- ^ Output base path | 155 | -> FileName -- ^ Output base path |
@@ -154,23 +158,20 @@ type ItemFileProcessor = | |||
154 | 158 | ||
155 | itemFileProcessor :: Maybe Resolution -> Cache -> ItemFileProcessor | 159 | itemFileProcessor :: Maybe Resolution -> Cache -> ItemFileProcessor |
156 | itemFileProcessor maxResolution cached inputBase outputBase resClass inputRes = | 160 | itemFileProcessor maxResolution cached inputBase outputBase resClass inputRes = |
157 | cached processor inPath outPath | 161 | cached processor inPath outPath |
158 | >> return (props relOutPath) | 162 | >> resourceAt outPath relOutPath |
163 | >>= descriptor outPath | ||
159 | where | 164 | where |
160 | relOutPath = resClass /> inputRes | 165 | relOutPath = resClass /> inputRes |
161 | inPath = localPath $ inputBase /> inputRes | 166 | inPath = localPath $ inputBase /> inputRes |
162 | outPath = localPath $ outputBase /> relOutPath | 167 | outPath = localPath $ outputBase /> relOutPath |
163 | (processor, props) = processorFor maxResolution $ formatFromPath inputRes | 168 | (processor, descriptor) = processorFor (formatFromPath inputRes) maxResolution |
164 | 169 | ||
165 | processorFor :: Maybe Resolution -> Format -> (FileProcessor, Path -> GalleryItemProps) | 170 | processorFor :: Format -> Maybe Resolution -> (FileProcessor, ItemDescriber) |
166 | processorFor Nothing _ = | 171 | processorFor PictureFormat (Just maxRes) = (resizePictureUpTo maxRes, getPictureProps) |
167 | (copyFileProcessor, Other) | 172 | processorFor PictureFormat Nothing = (copyFileProcessor, getPictureProps) |
168 | processorFor _ (PictureFormat Gif) = | 173 | -- TODO: handle video reencoding and others? |
169 | (copyFileProcessor, Picture) -- TODO: handle animated gif resizing | 174 | processorFor Unknown _ = (copyFileProcessor, const $ return . Other) |
170 | processorFor (Just maxRes) (PictureFormat picFormat) = | ||
171 | (resizeStaticImageUpTo picFormat maxRes, Picture) | ||
172 | processorFor _ Unknown = | ||
173 | (copyFileProcessor, Other) -- TODO: handle video reencoding and others? | ||
174 | 175 | ||
175 | 176 | ||
176 | type ThumbnailFileProcessor = | 177 | type ThumbnailFileProcessor = |
@@ -188,14 +189,15 @@ thumbnailFileProcessor maxRes cached inputBase outputBase resClass inputRes = | |||
188 | inPath = localPath $ inputBase /> inputRes | 189 | inPath = localPath $ inputBase /> inputRes |
189 | outPath = localPath $ outputBase /> relOutPath | 190 | outPath = localPath $ outputBase /> relOutPath |
190 | 191 | ||
191 | process :: Maybe FileProcessor -> IO (Maybe Path) | 192 | process :: Maybe FileProcessor -> IO (Maybe Thumbnail) |
192 | process Nothing = return Nothing | 193 | process Nothing = return Nothing |
193 | process (Just proc) = | 194 | process (Just proc) = |
194 | proc inPath outPath | 195 | do |
195 | >> return (Just relOutPath) | 196 | proc inPath outPath |
197 | resource <- resourceAt outPath relOutPath | ||
198 | resolution <- getImageResolution outPath | ||
199 | return $ Just $ Thumbnail resource resolution | ||
196 | 200 | ||
197 | processorFor :: Format -> Maybe FileProcessor | 201 | processorFor :: Format -> Maybe FileProcessor |
198 | processorFor (PictureFormat picFormat) = | 202 | processorFor PictureFormat = Just $ resizePictureUpTo maxRes |
199 | Just $ resizeStaticImageUpTo picFormat maxRes | 203 | processorFor _ = Nothing |
200 | processorFor _ = |