diff options
Diffstat (limited to 'compiler/src/Gallery.hs')
-rw-r--r-- | compiler/src/Gallery.hs | 123 |
1 files changed, 123 insertions, 0 deletions
diff --git a/compiler/src/Gallery.hs b/compiler/src/Gallery.hs new file mode 100644 index 0000000..3be62ad --- /dev/null +++ b/compiler/src/Gallery.hs | |||
@@ -0,0 +1,123 @@ | |||
1 | {-# LANGUAGE DuplicateRecordFields, DeriveGeneric, DeriveAnyClass #-} | ||
2 | |||
3 | -- ldgallery - A static generator which turns a collection of tagged | ||
4 | -- pictures into a searchable web gallery. | ||
5 | -- | ||
6 | -- Copyright (C) 2019 Pacien TRAN-GIRARD | ||
7 | -- | ||
8 | -- This program is free software: you can redistribute it and/or modify | ||
9 | -- it under the terms of the GNU Affero General Public License as | ||
10 | -- published by the Free Software Foundation, either version 3 of the | ||
11 | -- License, or (at your option) any later version. | ||
12 | -- | ||
13 | -- This program is distributed in the hope that it will be useful, | ||
14 | -- but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
15 | -- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
16 | -- GNU Affero General Public License for more details. | ||
17 | -- | ||
18 | -- You should have received a copy of the GNU Affero General Public License | ||
19 | -- along with this program. If not, see <https://www.gnu.org/licenses/>. | ||
20 | |||
21 | |||
22 | module Gallery | ||
23 | ( GalleryItem(..), buildGalleryTree | ||
24 | ) where | ||
25 | |||
26 | |||
27 | import GHC.Generics (Generic) | ||
28 | import Data.Char (toLower) | ||
29 | import Data.Function ((&)) | ||
30 | import Data.Maybe (fromMaybe) | ||
31 | |||
32 | import Data.Aeson (ToJSON, genericToJSON, genericToEncoding) | ||
33 | import qualified Data.Aeson as JSON | ||
34 | |||
35 | importĀ Utils | ||
36 | import Files | ||
37 | import Input | ||
38 | import Resource | ||
39 | |||
40 | |||
41 | encodingOptions :: JSON.Options | ||
42 | encodingOptions = JSON.defaultOptions | ||
43 | { JSON.fieldLabelModifier = map toLower | ||
44 | , JSON.constructorTagModifier = map toLower | ||
45 | , JSON.sumEncoding = JSON.defaultTaggedObject | ||
46 | { JSON.tagFieldName = "type" | ||
47 | , JSON.contentsFieldName = "contents" | ||
48 | } | ||
49 | } | ||
50 | |||
51 | |||
52 | type ResourcePath = String | ||
53 | type Tag = String | ||
54 | type FileSizeKB = Int | ||
55 | |||
56 | |||
57 | data Resolution = Resolution | ||
58 | { width :: Int | ||
59 | , height :: Int | ||
60 | } deriving (Generic, Show) | ||
61 | |||
62 | instance ToJSON Resolution where | ||
63 | toJSON = genericToJSON encodingOptions | ||
64 | toEncoding = genericToEncoding encodingOptions | ||
65 | |||
66 | |||
67 | data GalleryItemProps = | ||
68 | Directory { items :: [GalleryItem] } | ||
69 | -- | Image { resolution :: Resolution, filesize :: FileSizeKB } | ||
70 | -- | Video { filesize :: FileSizeKB } | ||
71 | | Unknown | ||
72 | deriving (Generic, Show) | ||
73 | |||
74 | instance ToJSON GalleryItemProps where | ||
75 | toJSON = genericToJSON encodingOptions | ||
76 | toEncoding = genericToEncoding encodingOptions | ||
77 | |||
78 | |||
79 | -- TODO: fuse GalleryItem and GalleryItemProps | ||
80 | data GalleryItem = GalleryItem | ||
81 | { title :: String | ||
82 | , date :: String -- TODO: checked ISO8601 date | ||
83 | , description :: String | ||
84 | , tags :: [Tag] | ||
85 | , path :: ResourcePath | ||
86 | , thumbnail :: Maybe ResourcePath | ||
87 | , properties :: GalleryItemProps | ||
88 | } deriving (Generic, Show) | ||
89 | |||
90 | instance ToJSON GalleryItem where | ||
91 | toJSON = genericToJSON encodingOptions | ||
92 | toEncoding = genericToEncoding encodingOptions | ||
93 | |||
94 | |||
95 | buildGalleryTree :: ResourceTree -> GalleryItem | ||
96 | buildGalleryTree (ItemResource sidecar path@(filename:_) thumbnailPath) = | ||
97 | GalleryItem | ||
98 | { title = optMeta title filename | ||
99 | , date = optMeta date "" -- TODO: check and normalise dates | ||
100 | , description = optMeta description "" | ||
101 | , tags = optMeta tags [] | ||
102 | , path = webPath path | ||
103 | , thumbnail = Just $ webPath thumbnailPath | ||
104 | , properties = Unknown } -- TODO | ||
105 | where | ||
106 | optMeta :: (Sidecar -> Maybe a) -> a -> a | ||
107 | optMeta get fallback = fromMaybe fallback $ get sidecar | ||
108 | |||
109 | buildGalleryTree (DirResource dirItems path@(dirname:_) thumbnailPath) = | ||
110 | map buildGalleryTree dirItems | ||
111 | & \items -> GalleryItem | ||
112 | { title = dirname | ||
113 | -- TODO: consider using the most recent item's date? what if empty? | ||
114 | , date = "" | ||
115 | -- TODO: consider allowing metadata sidecars for directories too | ||
116 | , description = "" | ||
117 | , tags = aggregateChildTags items | ||
118 | , path = webPath path | ||
119 | , thumbnail = fmap webPath thumbnailPath | ||
120 | , properties = Directory items } | ||
121 | where | ||
122 | aggregateChildTags :: [GalleryItem] -> [Tag] | ||
123 | aggregateChildTags = unique . concatMap (\item -> tags (item::GalleryItem)) | ||