diff --git a/content/assets/Asanb.woff2 b/content/assets/Asanb.woff2 new file mode 100644 index 0000000..606fb6d Binary files /dev/null and b/content/assets/Asanb.woff2 differ diff --git a/content/assets/mononoki-Regular.woff2 b/content/assets/mononoki-Regular.woff2 new file mode 100644 index 0000000..63ecf58 Binary files /dev/null and b/content/assets/mononoki-Regular.woff2 differ diff --git a/content/assets/theme.css b/content/assets/theme.css index d600025..cd39755 100755 --- a/content/assets/theme.css +++ b/content/assets/theme.css @@ -1,8 +1,20 @@ :root { --large-width: 1080px; --small-width: 600px; + --mono-font: mononoki; } +@font-face { + font-family: mononoki; + src: url("/assets/mononoki-Regular.woff2") format("woff2"); +} + +@font-face { + font-family: Asanb; + src: url("/assets/Asanb.woff2") format("woff2"); +} + + body { font-family: sans-serif; font-size: 14px; @@ -15,13 +27,13 @@ body { line-height: 1.54; min-height: 100vh; background: #dce0df; - grid-template: "a m e" 1fr + grid-template: "a m m" 1fr ". f f" / 180px 1fr 180px; gap: 2em; } h1, h2, h3, nav a { - font-family: monospace; + font-family: var(--mono-font), monospace; font-weight: 500; text-transform: uppercase; letter-spacing: 0.1rem; @@ -39,13 +51,6 @@ nav ul { padding: 0 } -@media (max-width: 840px) { - body { - grid-template: "a" "m" 1fr "f"; - } - nav {padding: 0} - nav ul { position: static; } -} nav a { display: block; @@ -74,11 +79,34 @@ ul { } main *+h2 {margin-top: 2rem} -main {margin-bottom: 2rem;} +main { + margin-bottom: 2rem; + grid-area: m; +} +main > * { + max-width: var(--small-width); +} + +main > pre, main > details, div.sourceCode { + max-width: 100%; + box-sizing: border-box; + clear: both; +} +main > ol.footnotes { + margin: 0; + padding: 0; + float: right; + font-size: .9em; + opacity: .8; +} + +main > ol.footnotes li p:first-child { margin-top: 0; } +main > ol.footnotes li p:last-child { margin-bottom: 0; } + main h2:first-child { margin-top: 0 } /* aside info */ -#contact + table {font-family: monospace} +#contact + table {font-family: var(--mono-font), monospace} #contact + table td:first-child { padding-right: 1em; text-transform: uppercase; @@ -95,7 +123,7 @@ main h2:first-child { margin-top: 0 } .pubs ul::before { content: attr(data-year); position: absolute; - font-family: monospace; + font-family: var(--mono-font), monospace; writing-mode: vertical-rl; text-orientation: upright; left: 0; @@ -131,3 +159,43 @@ main h2:first-child { margin-top: 0 } } footer {grid-area: f} + +pre { + white-space: pre-wrap; + background: #fff; + border-radius: 4px; + padding: .5em 1em; +} + +details summary { + border: 1px solid #000; + border-radius: 4px; + padding: .5em 1em; + cursor: pointer; +} + +details[open] { + border-bottom: 1px solid #000; + border-radius: 4px; +} + +code {font-family: var(--mono-font), Asanb, monospace} + +pre.Agda a { text-decoration: none; font-family: var(--mono-font), Asanb, monospace; } +.Keyword { color: #d08770 } +.Primitive { color: #5e81ac } +.InductiveConstructor { color: #a3be8c } +.Comment { color: #d8dee9 } +.Hole { + background: #88c0d0; + border-radius: 4px; +} + +@media (max-width: 840px) { + body { + grid-template: "a" "m" 1fr "f"; + } + nav {padding: 0} + nav ul { position: static; } + main > ol.footnotes + * { clear: both; } +} diff --git a/content/index.html b/content/index.html index fac4a05..da8b6af 100644 --- a/content/index.html +++ b/content/index.html @@ -7,7 +7,7 @@ - Lucas Escot + {{ $title }} diff --git a/content/index.md b/content/index.md index 4996759..da4830d 100755 --- a/content/index.md +++ b/content/index.md @@ -5,7 +5,13 @@ in the [Programming Languages Group](https://pl.ewi.tudelft.nl/), under the supervision of Jesper Cockx. My work revolves around generic programming in dependently-typed languages --- namely, [Agda](https://github.com/agda/agda). -## Miscellaneous +## Notes {#notes} + +```{=html} +{{ $notes }} +``` + +## Miscellaneous {#misc} On my spare time, I do a fair bit of drawing. Some of it may be found over at [acatalepsie.fr](https://acatalepsie.fr). Along with a group of friends, I am part of the [sbi.re](https://sbi.re) network, @@ -13,13 +19,15 @@ under which we self-host a bunch of services. ## Contact/Links {#contact} -------- --------------------------------------- -Mail [lucas@escot.me](mailto:lucas@escot.me) +------- ------------------------------------------------------------------------------------------------ +Mail [lucas@escot.me](mailto:lucas@escot.me), [l.f.b.escot@tudelft.nl](mailto:l.f.b.escot@tudelft.nl) GPG [lescot.gpg](/lescot.gpg) GH [flupe](https://github.com/flupe) SRHT [flupe](https://sr.ht/~flupe) -------- --------------------------------------- +------- ------------------------------------------------------------------------------------------------ ## Publications +```{=html} {{ $publications }} +``` diff --git a/escot.cabal b/escot.cabal index f9e618a..18447f6 100755 --- a/escot.cabal +++ b/escot.cabal @@ -10,6 +10,7 @@ executable escot hs-source-dirs: src other-modules: Config , Template + , Markdown build-depends: base , filepath , achille @@ -31,6 +32,7 @@ executable escot , process , directory , megaparsec + , mtl default-extensions: BlockArguments , TupleSections , OverloadedStrings diff --git a/src/Main.hs b/src/Main.hs index 4a1a868..1acb9d5 100755 --- a/src/Main.hs +++ b/src/Main.hs @@ -8,7 +8,9 @@ import Data.Aeson (FromJSON) import Data.Binary (Binary) import Control.Monad ((>=>)) import Data.Function ((&)) -import Data.Text (Text) +import Data.Text (Text, pack) +import Data.Time (UTCTime, defaultTimeLocale) +import System.Directory qualified as System import Data.List (intersperse) import Data.Maybe (fromJust) import Data.Yaml @@ -26,11 +28,13 @@ import Achille import Achille.Writable (Writable) import Achille.Writable qualified as Writable import Achille.Internal.IO (AchilleIO) +import Achille.Internal.IO qualified as AchilleIO import Achille.Task.Pandoc import Template (Template, Context, parseTemplate) import Template qualified import Config (config, ropts, wopts, SiteConfig(title)) +import Markdown qualified -- Bibliography info @@ -75,38 +79,88 @@ render template ctx body = KeyMap.insert "body" (String body) ctx & Template.render template - -- writing Html to disk (efficiently) instance AchilleIO m => Writable m (Html ()) where write to = Writable.write to . renderBS +data Note = Note + { date :: UTCTime + , title :: Text + } + deriving (Eq, Binary, Generic, FromJSON) main = achille do index <- readTemplate "index.html" - match_ "assets/*" $ copyFile + match_ "assets/*" copyFile + match_ "papers/*" copyFile + match_ "static/*" $ copyFileAs (makeRelative "static/") - match_ "papers/*" $ copyFile - pubs :: [Publication] <- matchFile "publications.yaml" parseYaml - - watch pubs $ match_ "index.md" \src -> do + (nav, index') <- matchFile "index.md" \src -> do doc <- readPandocWith def {readerExtensions = pandocExtensions} src let Pandoc _ blocks = doc let headers = filter isHeaderBlock blocks -- parsing rendered md as template - template :: Template <- renderPandoc doc <&> parseTemplate <&> fromJust + let template = renderText (Markdown.render doc) & LT.toStrict & parseTemplate & fromJust - let ctx = KeyMap.insert "nav" (String (LT.toStrict $ renderText $ renderNav headers)) $ + return (LT.toStrict $ renderText $ renderNav headers, template) + + + pubs :: [Publication] <- matchFile "publications.yaml" parseYaml + + notes :: [(FilePath, Note)] <- match "notes/*.md" \src -> do + if ".lagda.md" /= takeExtensions src then do + (meta@Note{..}, doc) <- readPandocMetadataWith def {readerExtensions = pandocExtensions} src + + path <- LT.toStrict (renderText (Markdown.render doc)) + & write (src -<.> "html") + . render index (KeyMap.insert "title" (String title) $ KeyMap.insert "nav" (String nav) $ KeyMap.empty) + return (path, meta) + else do + spath <- toAbsolute src + odir <- getOutputDir <&> ( dropExtensions src) + debug odir + tmp <- liftIO System.getTemporaryDirectory <&> ( "escot") + + liftIO $ System.createDirectoryIfMissing False tmp + liftIO $ System.createDirectoryIfMissing True odir + liftIO $ AchilleIO.copyFile spath (tmp "index.lagda.md") + + dir <- liftIO System.getCurrentDirectory + liftIO $ System.setCurrentDirectory tmp + + callCommand $ + "agda --html " + <> "--html-dir=. " + <> "--html-highlight=auto " + <> "index.lagda.md" + + liftIO $ System.setCurrentDirectory dir + callCommand $ "cp " <> tmp <> "/* " <> odir + + (meta@Note{..}, doc) <- readAbsPandocMetadataWith def {readerExtensions = pandocExtensions} (tmp "index.md") + debug doc + path <- renderPandoc doc >>= write (dropExtensions src "index.html") + . render index (KeyMap.insert "title" (String title) $ KeyMap.insert "nav" (String nav) $ KeyMap.empty) + return (takeDirectory path, meta) + + let notesList :: Html () = + ul_ $ foldMap (\(src, Note{..}) -> li_ $ a_ [ href_ (pack src) ] $ toHtmlRaw title) notes + + watch pubs $ watch notes do + let ctx = KeyMap.insert "nav" (String nav) $ + KeyMap.insert "notes" (String (LT.toStrict $ renderText notesList)) $ + KeyMap.insert "title" (String "Lucas Escot") $ KeyMap.insert "publications" (String (LT.toStrict $ renderText $ renderPublications pubs)) KeyMap.empty - write (src -<.> "html") $ render index ctx (LT.toStrict $ Template.render template ctx) + write "index.html" $ render index ctx (LT.toStrict $ Template.render index' ctx) where renderNav :: [Block] -> Html () - renderNav = ul_ . foldMap (\(Header _ (id, _, _) _) -> li_ $ a_ [href_ $ "#" <> id] (toHtmlRaw id)) + renderNav = ul_ . foldMap (\(Header _ (id, _, _) _) -> li_ $ a_ [href_ $ "/#" <> id] (toHtmlRaw id)) readTemplate :: FilePath -> Task IO Template readTemplate src = readText src <&> parseTemplate >>= \case diff --git a/src/Markdown.hs b/src/Markdown.hs new file mode 100644 index 0000000..4d6d2e0 --- /dev/null +++ b/src/Markdown.hs @@ -0,0 +1,113 @@ +module Markdown where + +import Lucid +import Lucid.Base (makeElement) +import Data.Text (pack) +import Text.Pandoc.Builder +import Control.Monad (foldM) +import Data.Functor ((<&>)) +import Data.Bifoldable (Bifoldable, bifoldMap, bifoldrM) +import Control.Monad.State.Strict +import Control.Monad.Identity + +data MDState = MDState + { stNoteCount :: Int -- count of notes that HAVE been printed + , stNotes :: [Html ()] -- notes that haven't be printed yet + , stLevel :: Int -- block depth + } + +defaultMDState :: MDState +defaultMDState = MDState + { stNoteCount = 0 + , stNotes = [] + , stLevel = 0 + } + +render :: Pandoc -> Html () +render (Pandoc meta blocks) = + fst $ runIdentity $ runStateT (blocksToHtml blocks) defaultMDState + +inlineToHtml :: Monad m => Inline -> StateT MDState m (Html ()) +inlineToHtml (Str str) = pure $ toHtml str +inlineToHtml (Emph ins) = em_ <$> inlinesToHtml ins +inlineToHtml (Underline ins) = makeElement "u" <$> inlinesToHtml ins +inlineToHtml (Strong ins) = strong_ <$> inlinesToHtml ins +inlineToHtml (Strikeout ins) = makeElement "s" <$> inlinesToHtml ins +inlineToHtml (Superscript ins) = sup_ <$> inlinesToHtml ins +inlineToHtml (Subscript ins) = sub_ <$> inlinesToHtml ins +inlineToHtml (SmallCaps ins) = span_ [class_ "sc"] <$> inlinesToHtml ins +inlineToHtml (Quoted typ ins) = + let (l, r) = case typ of + SingleQuote -> ("‘", "’") + DoubleQuote -> ("“", "”") + in pure l >> inlinesToHtml ins >> pure r +inlineToHtml (Cite cits ins) = undefined -- TODO: citations +inlineToHtml (Code attrs str) = pure $ code_ $ toHtml str +inlineToHtml Space = pure " " +inlineToHtml SoftBreak = pure "\n" +inlineToHtml LineBreak = pure $ br_ [] +inlineToHtml (Math _ math) = pure $ span_ [class_ "math"] $ toHtml math -- TODO: compile-time katex math +inlineToHtml (RawInline f str) | f == "html" = pure $ toHtmlRaw str +inlineToHtml (RawInline f str) = pure mempty +inlineToHtml (Link attrs ins (src, title)) = a_ [href_ src, title_ title] <$> inlinesToHtml ins +inlineToHtml (Image attrs ins (src, title)) = pure $ img_ [href_ src, title_ title] -- TODO: handle image alt +inlineToHtml (Note blocks) = do + MDState {..} <- get + let currentIdx = stNoteCount + length stNotes + 1 + note <- blocksToHtml blocks + modify \s -> s { stNotes = note : stNotes } + pure $ a_ [href_ $ "#note" <> pack (show currentIdx)] $ "[" <> toHtml (show currentIdx) <> "]" +inlineToHtml (Span attrs ins) = span_ <$> inlinesToHtml ins + +inlinesToHtml :: Monad m => [Inline] -> StateT MDState m (Html ()) +inlinesToHtml = foldMapM inlineToHtml + +blockToHtml :: Monad m => Block -> StateT MDState m (Html ()) +blockToHtml (Plain ins) = pre_ <$> inlinesToHtml ins +blockToHtml (Para ins) = p_ <$> inlinesToHtml ins +blockToHtml (LineBlock ins) = undefined +blockToHtml (CodeBlock attrs code) = pure $ pre_ $ code_ $ toHtmlRaw code -- TODO: highlighting +blockToHtml (RawBlock f str) | f == "html" = pure $ toHtmlRaw str +blockToHtml (RawBlock f str) = pure mempty +blockToHtml (BlockQuote blocks) = blockquote_ <$> blocksToHtml blocks +blockToHtml (OrderedList attrs items) = ol_ <$> foldMapM (fmap li_ . blocksToHtml) items -- TODO: handle attrs +blockToHtml (BulletList items) = ul_ <$> foldMapM (fmap li_ . blocksToHtml) items +blockToHtml (DefinitionList items) = + dl_ <$> foldMapM (bifoldMapM inlinesToHtml (foldMapM (fmap dd_ . blocksToHtml))) items +blockToHtml (Header k attrs ins) = h_ k <$> inlinesToHtml ins -- TODO: handle attrs (especially ID) + where h_ :: Applicative m => Int -> HtmlT m a -> HtmlT m a + h_ 1 = h1_ + h_ 2 = h2_ + h_ 3 = h3_ + h_ 4 = h4_ + h_ 5 = h5_ + h_ _ = h6_ +blockToHtml HorizontalRule = pure $ hr_ [] +blockToHtml (Table attrs caption _ head rows foot) = pure mempty +blockToHtml (Div attrs blocks) = pure mempty +blockToHtml Null = pure mempty + +blocksToHtml :: Monad m => [Block] -> StateT MDState m (Html ()) +blocksToHtml = fmap mconcat . mapM renderBlock + where renderBlock :: Monad m => Block -> StateT MDState m (Html ()) + renderBlock block = do + modify \s -> s { stLevel = stLevel s + 1 } + html <- blockToHtml block + MDState {..} <- get + let printingNotes = not (null stNotes) && stLevel == 1 + + let notes = when printingNotes (ol_ [class_ "footnotes", start_ $ pack (show $ stNoteCount + 1)] $ foldMap li_ (reverse stNotes)) + when printingNotes $ modify \s -> s {stNoteCount = stNoteCount + length stNotes, stNotes = [] } + + modify \s -> s { stLevel = stLevel - 1 } + + pure $ html <> notes + +foldMapM :: (Monad m, Monoid w, Foldable t) => (a -> m w) -> t a -> m w +foldMapM f = foldM (\acc a -> do + w <- f a + return $! mappend acc w) + mempty + +bifoldMapM :: (Monad m, Monoid w, Bifoldable t) => (a -> m w) -> (b -> m w) -> t a b -> m w +bifoldMapM f g = bifoldrM (\a b -> f a <&> (`mappend` b)) (\a b -> g a <&> (`mappend` b)) mempty