templates everywhere
This commit is contained in:
parent
7e6b53458b
commit
9f9e955524
|
@ -1,80 +1,133 @@
|
||||||
|
:root {
|
||||||
|
--large-width: 1080px;
|
||||||
|
--small-width: 600px;
|
||||||
|
}
|
||||||
|
|
||||||
body {
|
body {
|
||||||
font-family: sans-serif;
|
font-family: sans-serif;
|
||||||
font-size: 13px;
|
font-size: 14px;
|
||||||
max-width: 800px;
|
max-width: var(--large-width);
|
||||||
margin: 0 auto;
|
margin: 0 auto;
|
||||||
padding: 1em;
|
padding: 2em;
|
||||||
display: grid;
|
display: grid;
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
line-height: 1.54;
|
line-height: 1.54;
|
||||||
min-height: 100vh;
|
min-height: 100vh;
|
||||||
background: #dce0df;
|
background: #dce0df;
|
||||||
grid-template: "a m" 1fr
|
grid-template: "a m e" 1fr
|
||||||
"f f" / 240px 1fr;
|
". f f" / 180px 1fr 180px;
|
||||||
gap: 2em;
|
gap: 2em;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
h1, h2, h3, nav a {
|
||||||
|
font-family: monospace;
|
||||||
|
font-weight: 500;
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
nav {
|
||||||
|
padding: .4em 0 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
nav ul {
|
||||||
|
top: 2em;
|
||||||
|
position: sticky;
|
||||||
|
list-style: none;
|
||||||
|
margin: 0;
|
||||||
|
padding: 0
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 840px) {
|
||||||
|
body {
|
||||||
|
grid-template: "a" "m" 1fr "f";
|
||||||
|
}
|
||||||
|
nav {padding: 0}
|
||||||
|
nav ul { position: static; }
|
||||||
|
}
|
||||||
|
|
||||||
|
nav a {
|
||||||
|
display: block;
|
||||||
|
float: right;
|
||||||
|
clear: right;
|
||||||
|
text-decoration: none;
|
||||||
|
opacity: .7;
|
||||||
|
}
|
||||||
|
nav a:hover {opacity: 1}
|
||||||
|
|
||||||
a {
|
a {
|
||||||
|
color: inherit;
|
||||||
|
/* font-weight: 400; */
|
||||||
/* text-decoration-line: overline; */
|
/* text-decoration-line: overline; */
|
||||||
text-decoration-thickness: 1px;
|
text-decoration-thickness: 1px;
|
||||||
text-decoration-color: rgba(0, 0, 0, 0.2);
|
text-decoration-color: rgba(0, 0, 0, 0.4);
|
||||||
transition: .5s text-decoration-color;
|
transition: .5s text-decoration-color;
|
||||||
}
|
}
|
||||||
|
|
||||||
a:hover {
|
a:hover {
|
||||||
text-decoration-color: rgba(0, 0, 0, 0.5);
|
text-decoration-color: rgba(0, 0, 0, 0.8);
|
||||||
}
|
}
|
||||||
|
|
||||||
ul {
|
ul {
|
||||||
padding-left: 1em;
|
padding-left: 1em;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
main *+h2 {margin-top: 2rem}
|
||||||
|
main {margin-bottom: 2rem;}
|
||||||
|
main h2:first-child { margin-top: 0 }
|
||||||
|
|
||||||
/* aside info */
|
/* aside info */
|
||||||
#summary img {
|
#contact + table {font-family: monospace}
|
||||||
border-radius: 2px;
|
#contact + table td:first-child {
|
||||||
max-width: 100%;
|
|
||||||
display: block;
|
|
||||||
}
|
|
||||||
#summary figure { margin: 0 0 1em; }
|
|
||||||
#summary table td:first-child {
|
|
||||||
padding-right: 1em;
|
padding-right: 1em;
|
||||||
font-weight: 600;
|
text-transform: uppercase;
|
||||||
|
font-weight: 500;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* pubs list */
|
/* pubs list */
|
||||||
#publications ul {
|
.pubs ul {
|
||||||
list-style: none;
|
list-style: none;
|
||||||
margin: 0;
|
margin: 0;
|
||||||
padding: 0 0 0 2em;
|
padding: 0 0 0 2em;
|
||||||
position: relative;
|
position: relative;
|
||||||
}
|
}
|
||||||
#publications ul::before {
|
.pubs ul::before {
|
||||||
content: attr(data-year);
|
content: attr(data-year);
|
||||||
position: absolute;
|
position: absolute;
|
||||||
|
font-family: monospace;
|
||||||
writing-mode: vertical-rl;
|
writing-mode: vertical-rl;
|
||||||
text-orientation: upright;
|
text-orientation: upright;
|
||||||
left: 0;
|
left: 0;
|
||||||
top: .4em;
|
top: .35em;
|
||||||
line-height: 1em;
|
line-height: .8em;
|
||||||
opacity: .5;
|
opacity: .5;
|
||||||
}
|
}
|
||||||
#publications p { margin: 0; }
|
.pubs p { margin: 0; }
|
||||||
#publications li+li {margin: 1em 0 0}
|
.pubs li+li {margin: 1em 0 0}
|
||||||
#publications .title a {
|
.pubs .title a {
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
color: #000;
|
color: #000;
|
||||||
font-style: italic;
|
font-style: italic;
|
||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
}
|
}
|
||||||
#publications .buttons a {
|
.pubs .buttons {
|
||||||
|
margin-top: .5em;
|
||||||
|
}
|
||||||
|
.pubs .buttons a {
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
border: 1px solid #666;
|
border: 1px solid #777;
|
||||||
font-size: .8em;
|
font-size: .8em;
|
||||||
padding: 0 .5em;
|
padding: 0 .5em;
|
||||||
border-radius: 4px;
|
border-radius: 4px;
|
||||||
|
transition: .5s border-color;
|
||||||
}
|
}
|
||||||
#publications .buttons a+a {
|
.pubs .buttons a:hover {
|
||||||
|
border-color: #555;
|
||||||
|
}
|
||||||
|
.pubs .buttons a+a {
|
||||||
margin: 0 0 0 .5em;
|
margin: 0 0 0 .5em;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
footer {grid-area: f}
|
||||||
|
|
|
@ -0,0 +1,17 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0, user-scalable=yes">
|
||||||
|
<meta name="theme-color" content="#000000">
|
||||||
|
<meta name="robots" content="index, follow">
|
||||||
|
<link rel="stylesheet" href="/assets/theme.css">
|
||||||
|
<link rel="shortcut icon" type="image/svg" href="/assets/favicon.svg">
|
||||||
|
<title>Lucas Escot</title>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<nav>{{ $nav }}</nav>
|
||||||
|
<main>{{ $body }}</main>
|
||||||
|
<footer>2022 · <a href="https://creativecommons.org/licenses/by-nc/2.0/">CC BY-NC 2.0</a></footer>
|
||||||
|
</body>
|
||||||
|
</html>
|
|
@ -1,13 +1,25 @@
|
||||||
## Lucas Escot
|
## Lucas Escot {#me}
|
||||||
|
|
||||||
As of 2022, I am a PhD student at [TU Delft](https://tudelft.nl),
|
As of 2022, I am a PhD student at [TU Delft](https://tudelft.nl),
|
||||||
in the [Programming Languages Group](https://pl.ewi.tudelft.nl/),
|
in the [Programming Languages Group](https://pl.ewi.tudelft.nl/),
|
||||||
under the supervision of Jesper Cockx. My work revolves around generic programming
|
under the supervision of Jesper Cockx. My work revolves around generic programming
|
||||||
in dependently-typed languages.
|
in dependently-typed languages --- namely, [Agda](https://github.com/agda/agda).
|
||||||
|
|
||||||
## Misc
|
## Miscellaneous
|
||||||
|
|
||||||
I use to enjoy drawing, and you can find some of my works on
|
On my spare time, I do a fair bit of drawing. Some of it may be found over at [acatalepsie.fr](https://acatalepsie.fr).
|
||||||
[acatalepsie.fr](https://acatalepsie.fr). With a group of friends we run the
|
Along with a group of friends, I am part of the [sbi.re](https://sbi.re) network,
|
||||||
[sbi.re](https://sbi.re) network, for which we self-host a bunch of services.
|
under which we self-host a bunch of services.
|
||||||
It is here that I host my [personal blog](https://sbi.re/~lucas) where I ramble on the most --- which is, not a lot --- in French.
|
|
||||||
|
## Contact/Links {#contact}
|
||||||
|
|
||||||
|
------- ---------------------------------------
|
||||||
|
Mail [lucas@escot.me](mailto:lucas@escot.me)
|
||||||
|
GPG [lescot.gpg](/lescot.gpg)
|
||||||
|
GH [flupe](https://github.com/flupe)
|
||||||
|
SRHT [flupe](https://sr.ht/~flupe)
|
||||||
|
------- ---------------------------------------
|
||||||
|
|
||||||
|
## Publications
|
||||||
|
|
||||||
|
{{ $publications }}
|
||||||
|
|
|
@ -24,5 +24,5 @@
|
||||||
file: papers/agda2hs-haskell22.pdf
|
file: papers/agda2hs-haskell22.pdf
|
||||||
doi: "10.1145/3546189.3549920"
|
doi: "10.1145/3546189.3549920"
|
||||||
venue:
|
venue:
|
||||||
name: Haskell 2022
|
name: Haskell Symposium 2022
|
||||||
url: https://www.haskell.org/haskell-symposium/2022/
|
url: https://www.haskell.org/haskell-symposium/2022/
|
||||||
|
|
|
@ -9,6 +9,7 @@ executable escot
|
||||||
main-is: Main.hs
|
main-is: Main.hs
|
||||||
hs-source-dirs: src
|
hs-source-dirs: src
|
||||||
other-modules: Config
|
other-modules: Config
|
||||||
|
, Template
|
||||||
build-depends: base
|
build-depends: base
|
||||||
, filepath
|
, filepath
|
||||||
, achille
|
, achille
|
||||||
|
@ -29,6 +30,7 @@ executable escot
|
||||||
, optparse-applicative
|
, optparse-applicative
|
||||||
, process
|
, process
|
||||||
, directory
|
, directory
|
||||||
|
, megaparsec
|
||||||
default-extensions: BlockArguments
|
default-extensions: BlockArguments
|
||||||
, TupleSections
|
, TupleSections
|
||||||
, OverloadedStrings
|
, OverloadedStrings
|
||||||
|
|
103
src/Main.hs
103
src/Main.hs
|
@ -1,29 +1,36 @@
|
||||||
{-# LANGUAGE LambdaCase, TypeSynonymInstances, FlexibleInstances, MultiParamTypeClasses, OverloadedStrings #-}
|
{-# LANGUAGE LambdaCase, TypeSynonymInstances, FlexibleInstances, MultiParamTypeClasses, OverloadedStrings #-}
|
||||||
{-# LANGUAGE DuplicateRecordFields #-}
|
{-# LANGUAGE DuplicateRecordFields, ImportQualifiedPost #-}
|
||||||
|
|
||||||
module Main where
|
module Main where
|
||||||
|
|
||||||
import GHC.Generics (Generic)
|
import GHC.Generics (Generic)
|
||||||
import Data.Aeson (FromJSON)
|
import Data.Aeson (FromJSON)
|
||||||
import Data.Binary (Binary)
|
import Data.Binary (Binary)
|
||||||
import Control.Monad ((>=>))
|
import Control.Monad ((>=>))
|
||||||
import System.FilePath
|
import Data.Function ((&))
|
||||||
import Data.Text (Text)
|
import Data.Text (Text)
|
||||||
import Data.List (intersperse)
|
import Data.List (intersperse)
|
||||||
import Lucid
|
import Data.Maybe (fromJust)
|
||||||
import Data.Yaml
|
import Data.Yaml
|
||||||
|
import System.FilePath
|
||||||
|
import Lucid
|
||||||
import Control.Applicative ((<|>))
|
import Control.Applicative ((<|>))
|
||||||
|
import Data.Aeson.KeyMap qualified as KeyMap
|
||||||
|
import Data.Text.Lazy qualified as LT
|
||||||
|
import Data.Functor
|
||||||
|
import Text.Pandoc.Options
|
||||||
|
import Text.Pandoc.Shared (isHeaderBlock)
|
||||||
|
import Text.Pandoc.Definition (Pandoc(Pandoc), Block(Header))
|
||||||
|
|
||||||
import Achille
|
import Achille
|
||||||
import Achille.Writable (Writable)
|
import Achille.Writable (Writable)
|
||||||
import qualified Achille.Writable as Writable
|
import Achille.Writable qualified as Writable
|
||||||
import Achille.Internal.IO (AchilleIO)
|
import Achille.Internal.IO (AchilleIO)
|
||||||
|
|
||||||
import Achille.Task.Pandoc
|
import Achille.Task.Pandoc
|
||||||
|
|
||||||
import Data.Functor
|
import Template (Template, Context, parseTemplate)
|
||||||
|
import Template qualified
|
||||||
import Config (config, ropts, wopts, SiteConfig(title))
|
import Config (config, ropts, wopts, SiteConfig(title))
|
||||||
import Text.Pandoc.Options
|
|
||||||
|
|
||||||
|
|
||||||
-- Bibliography info
|
-- Bibliography info
|
||||||
|
@ -63,9 +70,11 @@ parseYaml p = do
|
||||||
Left err -> fail (prettyPrintParseException err)
|
Left err -> fail (prettyPrintParseException err)
|
||||||
Right ok -> return ok
|
Right ok -> return ok
|
||||||
|
|
||||||
|
render :: Template -> Context -> Text -> LT.Text
|
||||||
|
render template ctx body =
|
||||||
|
KeyMap.insert "body" (String body) ctx
|
||||||
|
& Template.render template
|
||||||
|
|
||||||
-- compile markdown with custom options and extensions
|
|
||||||
compileMD = compilePandocWith def { readerExtensions = pandocExtensions } def
|
|
||||||
|
|
||||||
-- writing Html to disk (efficiently)
|
-- writing Html to disk (efficiently)
|
||||||
instance AchilleIO m => Writable m (Html ()) where
|
instance AchilleIO m => Writable m (Html ()) where
|
||||||
|
@ -73,23 +82,42 @@ instance AchilleIO m => Writable m (Html ()) where
|
||||||
|
|
||||||
|
|
||||||
main = achille do
|
main = achille do
|
||||||
|
index <- readTemplate "index.html"
|
||||||
|
|
||||||
match_ "assets/*" $ copyFile
|
match_ "assets/*" $ copyFile
|
||||||
match_ "static/*" $ copyFileAs (makeRelative "static/")
|
match_ "static/*" $ copyFileAs (makeRelative "static/")
|
||||||
match_ "papers/*" $ copyFile
|
match_ "papers/*" $ copyFile
|
||||||
|
|
||||||
summary <- matchFile "summary.md" compileMD
|
|
||||||
pubs :: [Publication] <- matchFile "publications.yaml" parseYaml
|
pubs :: [Publication] <- matchFile "publications.yaml" parseYaml
|
||||||
|
|
||||||
watch summary $ watch pubs $ match_ "index.md" \src -> do
|
watch pubs $ match_ "index.md" \src -> do
|
||||||
compileMD src
|
doc <- readPandocWith def {readerExtensions = pandocExtensions} src
|
||||||
<&> renderIndex summary (renderPublications pubs)
|
|
||||||
>>= write (src -<.> "html")
|
let Pandoc _ blocks = doc
|
||||||
|
let headers = filter isHeaderBlock blocks
|
||||||
|
|
||||||
|
-- parsing rendered md as template
|
||||||
|
template :: Template <- renderPandoc doc <&> parseTemplate <&> fromJust
|
||||||
|
|
||||||
|
let ctx = KeyMap.insert "nav" (String (LT.toStrict $ renderText $ renderNav headers)) $
|
||||||
|
KeyMap.insert "publications" (String (LT.toStrict $ renderText $ renderPublications pubs)) KeyMap.empty
|
||||||
|
|
||||||
|
write (src -<.> "html") $ render index ctx (LT.toStrict $ Template.render template ctx)
|
||||||
|
|
||||||
|
where
|
||||||
|
renderNav :: [Block] -> Html ()
|
||||||
|
renderNav = ul_ . foldMap (\(Header _ (id, _, _) _) -> li_ $ a_ [href_ $ "#" <> id] (toHtmlRaw id))
|
||||||
|
|
||||||
|
readTemplate :: FilePath -> Task IO Template
|
||||||
|
readTemplate src = readText src <&> parseTemplate >>= \case
|
||||||
|
Just t -> return t
|
||||||
|
Nothing -> fail $ "Could not parse template: " ++ src
|
||||||
|
|
||||||
-- TODO: fix year
|
-- TODO: fix year
|
||||||
renderPublications :: [Publication] -> Html ()
|
renderPublications :: [Publication] -> Html ()
|
||||||
renderPublications pubs = section_ [ id_ "publications" ] do
|
renderPublications pubs = section_ [class_ "pubs"] do
|
||||||
h2_ "Publications"
|
|
||||||
ul_ [ data_ "year" "2022" ] $ foldMap renderPub pubs
|
ul_ [ data_ "year" "2022" ] $ foldMap renderPub pubs
|
||||||
|
|
||||||
where renderPub :: Publication -> Html ()
|
where renderPub :: Publication -> Html ()
|
||||||
renderPub Publication {..} = li_ [id_ slug] do
|
renderPub Publication {..} = li_ [id_ slug] do
|
||||||
p_ [class_ "title"] $ a_ [href_ ("#" <> slug)] $ toHtmlRaw title
|
p_ [class_ "title"] $ a_ [href_ ("#" <> slug)] $ toHtmlRaw title
|
||||||
|
@ -111,34 +139,3 @@ renderPublications pubs = section_ [ id_ "publications" ] do
|
||||||
renderAuthors [one] = one
|
renderAuthors [one] = one
|
||||||
renderAuthors (last:others) =
|
renderAuthors (last:others) =
|
||||||
mconcat (intersperse ", " (reverse others)) <> " and " <> last
|
mconcat (intersperse ", " (reverse others)) <> " and " <> last
|
||||||
|
|
||||||
renderIndex :: Text -> Html () -> Text -> Html ()
|
|
||||||
renderIndex summary pubs body =
|
|
||||||
doctypehtml_ do
|
|
||||||
head_ do
|
|
||||||
meta_ [ name_ "viewport"
|
|
||||||
, content_ "width=device-width, initial-scale=1.0, user-scalable=yes"
|
|
||||||
]
|
|
||||||
meta_ [ name_ "theme-color", content_ "#000000" ]
|
|
||||||
meta_ [ name_ "robots", content_ "index, follow" ]
|
|
||||||
meta_ [ charset_ "utf-8" ]
|
|
||||||
link_ [ rel_ "stylesheet", href_ "/assets/theme.css" ]
|
|
||||||
link_ [ rel_ "shortcut icon"
|
|
||||||
, type_ "image/svg"
|
|
||||||
, href_ "/assets/favicon.svg"
|
|
||||||
]
|
|
||||||
-- meta_ [ property_ "og:title", content_ title ]
|
|
||||||
-- meta_ [ property_ "og:type", content_ "website" ]
|
|
||||||
-- meta_ [ property_ "og:image", content_ image ]
|
|
||||||
-- meta_ [ property_ "og:description", content_ description ]
|
|
||||||
title_ "Lucas Escot"
|
|
||||||
|
|
||||||
body_ do
|
|
||||||
aside_ [id_ "summary"] $ toHtmlRaw summary
|
|
||||||
main_ do
|
|
||||||
section_ $ toHtmlRaw body
|
|
||||||
pubs
|
|
||||||
footer_ do
|
|
||||||
"2021 · "
|
|
||||||
a_ [ href_ "https://creativecommons.org/licenses/by-nc/2.0/" ]
|
|
||||||
"CC BY-NC 2.0"
|
|
||||||
|
|
|
@ -0,0 +1,106 @@
|
||||||
|
-- | Tiny templating engine
|
||||||
|
module Template where
|
||||||
|
|
||||||
|
import Data.Text
|
||||||
|
import Data.Text.Lazy.Encoding (encodeUtf8)
|
||||||
|
import Data.ByteString.Lazy (ByteString)
|
||||||
|
import Data.Binary
|
||||||
|
import Data.Functor (void)
|
||||||
|
import Data.Void
|
||||||
|
import GHC.Generics
|
||||||
|
import Text.Megaparsec
|
||||||
|
import Text.Megaparsec.Char
|
||||||
|
import Data.Aeson (Value(..))
|
||||||
|
|
||||||
|
import qualified Data.Text.Lazy as LT
|
||||||
|
import qualified Data.Aeson as Aeson
|
||||||
|
import qualified Data.Aeson.KeyMap as KeyMap
|
||||||
|
import qualified Data.Aeson.Key as Key
|
||||||
|
-- import qualified Data.HashMap.Strict as HashMap
|
||||||
|
|
||||||
|
|
||||||
|
data Chunk
|
||||||
|
= Raw Text
|
||||||
|
| Var [Text]
|
||||||
|
| For Text [Text] Template
|
||||||
|
| If [Text] Template
|
||||||
|
deriving (Eq, Show, Generic, Binary)
|
||||||
|
|
||||||
|
type Template = [Chunk]
|
||||||
|
type Context = Aeson.Object
|
||||||
|
|
||||||
|
type Parser = Parsec Void Text
|
||||||
|
|
||||||
|
|
||||||
|
-- RENDERING
|
||||||
|
|
||||||
|
render :: Template -> Context -> LT.Text
|
||||||
|
render chunks ctx = foldMap (renderChunk ctx) chunks
|
||||||
|
|
||||||
|
lookupCtx :: [Text] -> Context -> Maybe Aeson.Value
|
||||||
|
lookupCtx keys = aux keys . Aeson.Object
|
||||||
|
where
|
||||||
|
aux :: [Text] -> Aeson.Value -> Maybe Aeson.Value
|
||||||
|
aux [] v = Just v
|
||||||
|
aux (k:ks) (Object h) = KeyMap.lookup (Key.fromText k) h >>= aux ks
|
||||||
|
aux _ _ = Nothing
|
||||||
|
|
||||||
|
renderChunk :: Context -> Chunk -> LT.Text
|
||||||
|
renderChunk ctx (Raw t) = LT.fromStrict t
|
||||||
|
renderChunk ctx (Var ks) =
|
||||||
|
case Aeson.fromJSON <$> lookupCtx ks ctx of
|
||||||
|
Just (Aeson.Success v) -> v
|
||||||
|
_ -> mempty
|
||||||
|
renderChunk ctx (For kn ks c) =
|
||||||
|
case lookupCtx ks ctx of
|
||||||
|
Just (Aeson.Array arr) ->
|
||||||
|
foldMap (\v -> foldMap (renderChunk (KeyMap.insert (Key.fromText kn) v ctx)) c) arr
|
||||||
|
_ -> mempty
|
||||||
|
renderChunk ctx (If ks c) =
|
||||||
|
case lookupCtx ks ctx of
|
||||||
|
Just _ -> foldMap (renderChunk ctx) c
|
||||||
|
_ -> mempty
|
||||||
|
|
||||||
|
|
||||||
|
-- PARSING
|
||||||
|
|
||||||
|
parseTemplate :: Text -> Maybe Template
|
||||||
|
parseTemplate = parseMaybe templateP
|
||||||
|
|
||||||
|
templateP :: Parser Template
|
||||||
|
templateP = many chunkP
|
||||||
|
|
||||||
|
chunkP :: Parser Chunk
|
||||||
|
chunkP = choice [ varP , forP , ifP , rawP ]
|
||||||
|
where
|
||||||
|
identP :: Parser Text
|
||||||
|
identP = pack <$> ((:) <$> letterChar <*> many alphaNumChar)
|
||||||
|
|
||||||
|
keysP :: Parser [Text]
|
||||||
|
keysP = try $ "$" *> sepBy1 identP "."
|
||||||
|
|
||||||
|
varP :: Parser Chunk
|
||||||
|
varP = try $ Var <$> between ldelim rdelim
|
||||||
|
(space *> keysP <* space)
|
||||||
|
|
||||||
|
rawP :: Parser Chunk
|
||||||
|
rawP = Raw . pack <$>
|
||||||
|
someTill anySingle (eof <|> lookAhead ldelim)
|
||||||
|
|
||||||
|
forP :: Parser Chunk
|
||||||
|
forP = try do
|
||||||
|
ldelim *> space *> string "for" *> space1
|
||||||
|
key <- identP <* space1 <* "in" <* space1
|
||||||
|
val <- keysP <* space <* rdelim
|
||||||
|
chunks <- manyTill chunkP (try (between ldelim rdelim (space *> "end" *> space)))
|
||||||
|
return $ For key val chunks
|
||||||
|
|
||||||
|
ifP :: Parser Chunk
|
||||||
|
ifP = try do
|
||||||
|
ldelim *> space *> string "if" *> space1
|
||||||
|
val <- keysP <* space <* rdelim
|
||||||
|
chunks <- manyTill chunkP (try (between ldelim rdelim (space *> "end" *> space)))
|
||||||
|
return $ If val chunks
|
||||||
|
|
||||||
|
ldelim = void "{{"
|
||||||
|
rdelim = "}}"
|
Loading…
Reference in New Issue