2020-07-19 11:55:54 -04:00
|
|
|
{-# LANGUAGE TransformListComp #-}
|
2020-07-16 10:07:28 -04:00
|
|
|
module GalleryPage (make) where
|
|
|
|
|
2020-09-25 17:08:44 -04:00
|
|
|
import BuilderQQ
|
|
|
|
import Date
|
|
|
|
import Info
|
2020-10-06 16:07:39 -04:00
|
|
|
import qualified NsfwWarning
|
2020-09-25 17:08:44 -04:00
|
|
|
|
2020-08-03 17:07:59 -04:00
|
|
|
import Data.Foldable
|
2020-07-20 16:40:34 -04:00
|
|
|
import Data.Function (on, (&))
|
2020-08-03 17:07:59 -04:00
|
|
|
import qualified Data.HashMap.Strict as HashMap
|
2020-08-03 20:27:19 -04:00
|
|
|
import Data.HashSet (HashSet)
|
|
|
|
import qualified Data.HashSet as HashSet
|
2020-11-16 17:30:56 -05:00
|
|
|
import Data.List (intersperse, groupBy, sortBy, sortOn)
|
2020-07-16 10:07:28 -04:00
|
|
|
import qualified Data.Text.Lazy as Lazy
|
2020-08-11 14:29:54 -04:00
|
|
|
import System.FilePath (takeDirectory, joinPath, splitPath)
|
2020-07-19 11:55:54 -04:00
|
|
|
import GHC.Exts (Down (..), the)
|
2020-07-16 10:07:28 -04:00
|
|
|
|
2020-08-11 14:29:54 -04:00
|
|
|
make :: Text -> GalleryInfo -> [(FilePath, Info)] -> Lazy.Text
|
|
|
|
make root ginfo infos = toLazyText $ make' root ginfo infos
|
2020-07-16 10:07:28 -04:00
|
|
|
|
2020-08-03 20:25:59 -04:00
|
|
|
|
2020-08-11 14:29:54 -04:00
|
|
|
make' :: Text -> GalleryInfo -> [(FilePath, Info)] -> Builder
|
|
|
|
make' root (GalleryInfo {title, desc, prefix, filters, hidden}) infos = [b|@0
|
2020-07-16 10:07:28 -04:00
|
|
|
<!DOCTYPE html>
|
|
|
|
<html lang=en>
|
|
|
|
<meta charset=utf-8>
|
2020-08-04 17:12:58 -04:00
|
|
|
<link rel=stylesheet href=/style/shiny/gallery.css>
|
2020-08-04 18:52:56 -04:00
|
|
|
<link rel=icon href=/style/niss.svg>
|
2020-07-19 12:04:40 -04:00
|
|
|
<link rel=alternate href=rss.xml type=application/rss+xml>
|
2020-07-16 10:07:28 -04:00
|
|
|
|
2020-08-11 14:29:54 -04:00
|
|
|
<meta property=og:type content=og:website>
|
2020-08-30 13:13:40 -04:00
|
|
|
<meta property=og:title content="$title">
|
|
|
|
<meta property=og:site_name content="$title">
|
|
|
|
<meta property=og:description content="$desc">
|
|
|
|
<meta property=og:image content="$url/$imagepath0">
|
2020-08-11 14:29:54 -04:00
|
|
|
<meta property=og:url content="$url">
|
2020-09-19 02:13:13 -04:00
|
|
|
<meta name=twitter:site content=@2_gecs>
|
2020-08-11 14:29:54 -04:00
|
|
|
<meta name=twitter:card content=summary>
|
|
|
|
|
2020-08-04 13:14:12 -04:00
|
|
|
<script src=/script/gallery.js></script>
|
2020-10-06 16:07:39 -04:00
|
|
|
$0.nsfwScript
|
2020-08-03 17:11:15 -04:00
|
|
|
|
2020-08-30 13:13:40 -04:00
|
|
|
<title>$title</title>
|
2020-07-16 10:07:28 -04:00
|
|
|
|
2020-10-06 16:07:39 -04:00
|
|
|
$0.nsfwDialog
|
|
|
|
|
|
|
|
<div class=page>
|
|
|
|
<header>
|
|
|
|
<h1>$title</h1>
|
|
|
|
<h2 class="right corner">
|
|
|
|
<a href=rss.xml>rss</a>
|
|
|
|
</h2>
|
|
|
|
</header>
|
|
|
|
|
|
|
|
<nav id=filters>
|
|
|
|
<details id=filters-details>
|
|
|
|
<summary><h2>filters</h2></summary>
|
|
|
|
<div>
|
|
|
|
<h3>show only</h3>
|
|
|
|
<ul id=require class="buttonbar bb-choice">
|
|
|
|
$10.requireFilters
|
|
|
|
</ul>
|
|
|
|
|
|
|
|
<h3>exclude</h3>
|
|
|
|
<ul id=exclude class="buttonbar bb-choice">
|
|
|
|
$10.excludeFilters
|
|
|
|
</ul>
|
|
|
|
|
|
|
|
<a href=# id=clear>clear</a>
|
|
|
|
<a href=# id=singles>toggle single-use tags</a>
|
|
|
|
</div>
|
|
|
|
</details>
|
|
|
|
</nav>
|
|
|
|
|
|
|
|
<main>
|
|
|
|
<ul class=grid>
|
|
|
|
$6.items
|
|
|
|
</ul>
|
|
|
|
</main>
|
|
|
|
|
|
|
|
<footer>
|
|
|
|
<a href=$undir>all galleries</a>
|
|
|
|
</footer>
|
|
|
|
</div>
|
2020-07-16 10:07:28 -04:00
|
|
|
|]
|
|
|
|
where
|
2020-07-18 05:43:35 -04:00
|
|
|
items = map (uncurry $ makeYearItems nsfw) infosByYear
|
2020-08-03 20:25:59 -04:00
|
|
|
|
2020-07-18 05:43:35 -04:00
|
|
|
infosByYear =
|
2020-07-19 11:55:54 -04:00
|
|
|
[(the year, infopath) |
|
|
|
|
infopath@(_, info) <- infos,
|
2020-11-16 17:30:56 -05:00
|
|
|
then sortInfo by info,
|
|
|
|
let year = #latestYear info nsfw,
|
2020-07-19 11:55:54 -04:00
|
|
|
then group by Down year using groupBy']
|
2020-11-16 17:30:56 -05:00
|
|
|
sortInfo f = sortBy $ flip (compareFor nsfw `on` f)
|
2020-07-19 11:55:54 -04:00
|
|
|
groupBy' f = groupBy ((==) `on` f)
|
2020-08-03 20:25:59 -04:00
|
|
|
|
2020-07-23 13:51:53 -04:00
|
|
|
undir = joinPath (replicate (length (splitPath prefix)) "..")
|
2020-08-03 20:25:59 -04:00
|
|
|
|
2020-08-03 17:07:59 -04:00
|
|
|
allTags = infos
|
|
|
|
& concatMap (map (,1) . tagsFor nsfw . #second)
|
|
|
|
& HashMap.fromListWith (+) & HashMap.toList
|
|
|
|
& sortOn (\(tag, count) -> (Down count, tag))
|
2020-08-03 20:27:19 -04:00
|
|
|
|
|
|
|
requireFilters = map (uncurry $ makeFilter "require" mempty) allTags
|
|
|
|
excludeFilters = map (uncurry $ makeFilter "exclude" hidden) allTags
|
2020-07-18 05:43:35 -04:00
|
|
|
|
2020-08-03 20:25:59 -04:00
|
|
|
nsfw = #nsfw filters /= NoNsfw
|
|
|
|
|
2020-08-30 13:13:40 -04:00
|
|
|
url = [b|$root/$prefix|]
|
2020-08-11 14:29:54 -04:00
|
|
|
imagepath0
|
|
|
|
| (_, (p0, i0) : _) : _ <- infosByYear = getThumb (takeDirectory p0) i0
|
|
|
|
| otherwise = "/style/card.png"
|
|
|
|
|
2020-10-06 16:07:39 -04:00
|
|
|
nsfwScript = NsfwWarning.script nsfw
|
|
|
|
nsfwDialog = NsfwWarning.dialog nsfw
|
|
|
|
|
2020-08-03 20:27:19 -04:00
|
|
|
makeFilter :: Text -> HashSet Text -> Text -> Int -> Builder
|
2020-10-06 16:07:39 -04:00
|
|
|
makeFilter prefix initial tag count = [b|@0
|
2020-08-03 17:10:52 -04:00
|
|
|
<li>
|
2020-08-30 13:13:40 -04:00
|
|
|
<input type=checkbox id="$id'" value="$tag"$checked>
|
2020-09-17 13:12:48 -04:00
|
|
|
<label for="$id'" data-count=$count>$tag</label>
|
2020-08-03 17:10:52 -04:00
|
|
|
|]
|
2020-08-03 20:27:19 -04:00
|
|
|
where
|
2020-08-30 13:13:40 -04:00
|
|
|
id' = [b|$prefix$&_$tag'|]
|
2020-08-03 20:27:19 -04:00
|
|
|
tag' = escId tag
|
2020-08-30 13:13:40 -04:00
|
|
|
checked = if HashSet.member tag initial then [b| checked|] else ""
|
2020-08-03 17:10:52 -04:00
|
|
|
|
2020-07-18 05:43:35 -04:00
|
|
|
makeYearItems :: Bool -- ^ nsfw
|
2020-09-25 17:08:44 -04:00
|
|
|
-> Int -- ^ year
|
2020-07-18 05:43:35 -04:00
|
|
|
-> [(FilePath, Info)]
|
|
|
|
-> Builder
|
2020-10-06 16:07:39 -04:00
|
|
|
makeYearItems nsfw year infos = [b|@0
|
2020-09-26 19:36:20 -04:00
|
|
|
<li class="item year-marker" id="marker-$year">
|
2020-07-20 16:40:34 -04:00
|
|
|
<span class=year-text>$year'</span>
|
2020-10-06 16:07:39 -04:00
|
|
|
$0.items
|
2020-07-18 05:43:35 -04:00
|
|
|
|]
|
2020-07-20 16:40:34 -04:00
|
|
|
where
|
|
|
|
items = map (uncurry $ makeItem nsfw) infos
|
2020-08-30 13:13:40 -04:00
|
|
|
year' = show year & foldMap \c -> [b|<span class=y>$c</span>|]
|
2020-07-16 10:07:28 -04:00
|
|
|
|
2020-07-18 05:40:56 -04:00
|
|
|
makeItem :: Bool -> FilePath -> Info -> Builder
|
2020-10-06 16:07:39 -04:00
|
|
|
makeItem nsfw file info@(Info {title, bg}) = [b|@0
|
2020-09-26 19:36:20 -04:00
|
|
|
<li class="item post$nsfw'" data-date="$date'" data-year=$year'
|
|
|
|
data-updated="$updated'"
|
2020-09-19 01:51:52 -04:00
|
|
|
data-tags="$tags'">
|
2020-07-16 10:07:28 -04:00
|
|
|
<figure>
|
2020-08-30 13:13:40 -04:00
|
|
|
<a href="$dir">
|
2020-09-29 13:58:54 -04:00
|
|
|
<img src="$thumb" loading=lazy$bgStyle>
|
2020-07-16 10:07:28 -04:00
|
|
|
</a>
|
2020-09-19 01:51:52 -04:00
|
|
|
<figcaption>
|
|
|
|
<span class=date>$date'</span>
|
|
|
|
<span class=title>$title</span>
|
|
|
|
</figcaption>
|
2020-07-16 10:07:28 -04:00
|
|
|
</figure>
|
|
|
|
|]
|
|
|
|
where
|
2020-09-19 01:51:52 -04:00
|
|
|
dir = takeDirectory file
|
|
|
|
thumb = getThumb dir info
|
|
|
|
nsfw' = if nsfw && #anyNsfw info then [b| nsfw|] else ""
|
|
|
|
tags' = fold $ intersperse ";" $ map fromText $ tagsFor nsfw info
|
2020-11-16 17:30:56 -05:00
|
|
|
date = #latestDate info nsfw
|
2020-09-26 19:36:20 -04:00
|
|
|
date' = formatShort date
|
|
|
|
year' = #year date
|
2020-11-16 17:30:56 -05:00
|
|
|
updated' = if #updated info nsfw then [b|true|] else [b|false|]
|
2020-09-19 01:51:52 -04:00
|
|
|
bgStyle = ifJust bg \col -> [b| style="background: $col"|]
|