implement tag pages
[~bandali/bndl.org] / haunt.scm
index 27f2719..03af1aa 100644 (file)
--- a/haunt.scm
+++ b/haunt.scm
              (ice-9 match)
              (srfi srfi-19))
 
+(define my-scheme 'https)
+(define my-domain "bandali.eu.org")
+(define my-url
+  (string-append (symbol->string my-scheme) "://" my-domain))
+(define my-tag-prefix "tags")
+(define my-date-format "~B ~e, ~Y")
+
 (define (stylesheet name)
   `(link (@ (rel "stylesheet")
             (href ,(string-append "/" name ".css")))))
@@ -19,7 +26,7 @@
 (define* (aa content #:optional (uri content) . title)
   `(a (@ (href ,uri) (title ,(apply string-append title))) ,content))
 
-(define* (base-layout site body #:key title)
+(define* (base-layout site body #:key title copy license-page?)
   `((doctype "html")
     (html
      (head
      (body
       (main ,body)
       (footer
-       (p
-        "Copyright 2016–2019 Amin Bandali.  See " ,(aa "license.html")
-        " for license conditions. Please copy and share."))))))
+       (p "Copyright © "
+          ,(if copy copy "2016–2019")
+          " Amin Bandali.  See "
+          ,(if license-page? "the above"
+               (aa "license.html" "/license.html"))
+          " for license conditions.  Please copy and share."))))))
+
+(define* (tag-uri prefix tag #:optional (ext ".html"))
+  "Return a URI relative to the site's root for a page listing entries
+in PREFIX that are tagged with TAG."
+  (string-append "/" my-tag-prefix "/" tag ext))
+
+(define* (tag-pages #:key
+                    (theme bandali-theme)
+                    (prefix "")
+                    (filter posts/reverse-chronological))
+  "Return a builder procedure that renders a list page for every tag
+used in a post.  All arguments are optional:
+
+PREFIX: The directory in which to write the posts
+FILTER: The procedure called to manipulate the posts list before rendering"
+  (lambda (site posts)
+    (define (tag-list tag posts all-posts)
+      (define (render-list title posts prefix)
+        (let ((body ((theme-collection-template theme)
+                     site title posts prefix all-posts tag)))
+          ((theme-layout theme) site title body)))
+      (make-page (tag-uri my-tag-prefix tag)
+                 (render-list (string-append "Notes tagged ‘" tag "’")
+                              (filter posts)
+                              prefix)
+                 sxml->html))
+    (let ((tag-groups (posts/group-by-tag posts)))
+      (map (match-lambda
+             ((tag . tagged-posts) (tag-list tag tagged-posts posts)))
+           tag-groups))))
 
-(define my-date-format "~B ~d, ~Y")
+(define (tag-links posts)
+  "Generate an alphabetically sorted list of links to tagged posts.
+The link text consists of the tag name and the number of tagged posts
+in parentheses."
+  `(ul (@ (class "tag-list"))
+       ,(map (match-lambda
+               ((tag . posts)
+                `(li
+                  ,(aa (string-append tag
+                                      " ("
+                                      (number->string (length posts))
+                                      ")")
+                       (tag-uri my-tag-prefix tag)))))
+             ;; sort by tag
+             (sort (posts/group-by-tag posts)
+                   (lambda (a b) (string<? (car a) (car b)))))))
+
+(register-metadata-parser! 'updated string->date*)
+
+(define (intersperse lst delim)
+  "Return the elements of LST delimited by DELIM, such that the
+resulting list is of an odd length and every second element is DELIM."
+  (if (<= (length lst) 1)
+      lst
+      (cons* (car lst)
+             delim
+             (intersperse (cdr lst) delim))))
 
 (define (my-post-template post)
   `((header
      (address "By " ,(aa (post-ref post 'author) "/")
               " <" ,(post-ref post 'email) ">")
      (p (@ (class "date"))
-        ,(date->string (post-date post) my-date-format)))
+        "Published "
+        ,(date->string (post-date post) my-date-format))
+     ,(if (post-ref post 'updated)
+          `(p (@ (class "updated"))
+              "Updated "
+              ,(date->string (post-ref post 'updated)
+                             my-date-format)) '())
+     ,(if (post-ref post 'tags)
+          `(p (@ (class "tags"))
+              "Tagged "
+              ,@(intersperse
+                 (map (lambda (tag)
+                        (aa tag (tag-uri my-tag-prefix tag)))
+                      (post-ref post 'tags))
+                 ", "))
+          '()))
     ,(post-sxml post)
     (p (@ (class "muted inbox"))
        "Have a question or comment?  Start a discussion in my "
                   "https://man.sr.ht/lists.sr.ht/etiquette.md") "]")
        ".")))
 
-(define (my-collection-template site title posts prefix)
+(define* (my-collection-template site title posts prefix
+                                 #:optional all-posts tag)
   (define (post-uri post)
     (string-append (or prefix "") "/"
                    (site-post-slug site post) ".html"))
 
-  `((h3 ,title)
-    (ul
-     ,@(map (lambda (post)
-              `(li
-                ,(aa (post-ref post 'title) (post-uri post))
-                " — "
-                ,(date->string (post-date post) my-date-format)))
-            posts))))
+  `((h2 ,title
+        ,(if tag
+             (aa `(img (@ (class "feed-icon")
+                          (src "/feed.svg")
+                          (alt "subscribe to atom feed")))
+                 (tag-uri my-tag-prefix tag ".xml"))
+             '()))
+    (table
+     (@ (class "post-list"))
+     (tbody
+      ,@(map (lambda (post)
+               `(tr
+                 (td ,(aa (post-ref post 'title) (post-uri post)))
+                 (td (@ (style "font-size: 0.875em;"))
+                     ,(date->string (post-date post)
+                                    my-date-format))))
+             posts)))
+    (h2 "Tags")
+    ,(tag-links (or all-posts posts))
+    ,(if tag
+         '(a (@ (href "/notes.html"))
+             "← all posts")
+         '())))
 
 (define bandali-theme
   (theme #:name "bandali"
          #:post-template my-post-template
          #:collection-template my-collection-template))
 
-(define (static-page title file-name body)
+(define* (static-page title file-name body copy #:key license?)
   (lambda (site posts)
     (make-page file-name
-               (with-layout bandali-theme site title body)
+               (base-layout site body #:title title #:copy copy
+                            #:license-page? license?)
                sxml->html)))
 
 (define (index-material site posts)
-  `(div
-    (h1 (@ (style "font-size: 0;"))
+  `((h1 (@ (style "font-size: 0;"))
         "Amin Bandali")
     (p (@ (style "margin-top: 0;"))
        "Hi, I’m "
            " [ " ,(aa "pdf" "papers/modre2018-declarative.pdf") " | "
            ,(aa "bib" "papers/modre2018-declarative.bib") " ]"))
      (dd "Ali Abbassi, "
-         ,(aa "Amin Bandali" "https://bandali.eu.org") ", "
+         ,(aa "Amin Bandali" my-url) ", "
          ,(aa "Nancy A. Day" "https://cs.uwaterloo.ca/~nday/") ", "
          "Jose Serna"
          (br)
             " Research Conference,\n"
             "York University, Toronto, Canada, August 15, 2017")
        " ]"))
-     (dd ,(aa "Amin Bandali" "https://bandali.eu.org") ", "
+     (dd ,(aa "Amin Bandali" my-url) ", "
          ,(aa "Simon Hudon" "https://github.com/cipher1024") ", "
          ,(aa "Jonathan S. Ostroff"
               "http://www.cse.yorku.ca/~jonathan/")))
      (dd "Library and CLI for converting TeX and LaTeX to PNG "
          "images"))
     (h2 (@ (id "notes")) "Notes")
+    (p "Here are notes about a variety of topics and issues I care "
+       "about.  They’re also available via " ,(aa "Atom" "notes.atom")
+       " and " ,(aa "RSS" "notes.rss") " feeds.")
     (table
      (@ (class "post-list"))
      (tbody
-     ,@(map
-        (lambda (post)
-          (define (post-uri post)
-            (string-append "/"
-                           (site-post-slug site post) ".html"))
-          `(tr
-            (td ,(aa (post-ref post 'title) (post-uri post)))
-            (td (@ (style "font-size: 0.875em;"))
-             ,(date->string (post-date post) my-date-format))))
-        (take-up-to 10 (posts/reverse-chronological posts)))))))
+      ,@(map
+         (lambda (post)
+           (define (post-uri post)
+             (string-append "/"
+                            (site-post-slug site post) ".html"))
+           `(tr
+             (td ,(aa (post-ref post 'title) (post-uri post)))
+             (td (small
+                  ,(date->string (post-date post) my-date-format)))))
+         (take-up-to 10 (posts/reverse-chronological posts)))))))
 
 (define (index-page site posts)
   (make-page
   (static-page
    "Licensing Information"
    "license.html"
-   `((h1 "License information for bandali.eu.org")
+   `((h1 "License information for "
+         ,(aa my-domain my-url))
      (p "I strongly believe in "
         ,(aa "free culture"
              "https://questioncopyright.org/what_is_free_culture")
       (li ,(aa "Various Licenses and Comments about Them"
                "https://www.gnu.org/licenses/license-list.html"))
       (li ,(aa "Proprietary Software Is Often Malware"
-               "https://www.gnu.org/proprietary/proprietary.html"))))))
+               "https://www.gnu.org/proprietary/proprietary.html"))))
+      "2019"
+      #:license? #t))
 
-(define my-domain "bandali.eu.org")
+(define contact-page
+  (static-page
+   "Contact Information"
+   "contact.html"
+   `((h1 "Contact information")
+     (p "Email is by far my preferred method of communication.  I may"
+        " be contacted at any of the following addresses (choose the"
+        " most closely related):")
+     (ul
+      (li "bandali@gnu.org")
+      (li "bandali@uwaterloo.ca")
+      (li "bandali@csclub.uwaterloo.ca"))
+     (p "If you want to send me GPG-encrypted mail, you can use my "
+        ,(aa "public key" "bandali-pubkey.txt") " with the"
+        " fingerprint "
+        (code "BE62 7373 8E61 6D6D 1B3A  08E8 A21A 0202 4881 6103")
+        ".")
+     (table
+      (tbody
+       (tr
+        (td "IRC")
+        (td "bandali on " ,(aa "freenode" "https://freenode.net") ", "
+            ,(aa "moznet" "https://wiki.mozilla.org/IRC") ", and "
+            ,(aa "oftc" "https://www.oftc.net")))
+       (tr
+        (td "XMPP")
+        (td ,(aa "bandali@member.fsf.org"
+                 "xmpp:bandali@member.fsf.org")))
+       (tr
+        (td "Matrix")
+        (td ,(aa "@bandali:matrix.org"
+                 "https://matrix.to/#/@bandali:matrix.org")))
+       (tr
+        (td "Fediverse")
+        (td ,(aa "@bandali@pleroma.site"
+                 "https://pleroma.site/bandali")))))
+     (h2 "Elsewhere")
+     (p "You may also find me at a few other places online.  Stricken"
+        " through accounts are those I don’t use anymore, unless"
+        " absolutely necessary.")
+     (ul
+      (li ,(aa "bandali" "https://libreplanet.org/wiki/User:Bandali")
+          " on LibrePlanet")
+      (li ,(aa "bandali" "https://emacsconf.org/bandali")
+          " on EmacsConf")
+      (li ,(aa "bandali" "https://savannah.gnu.org/users/bandali")
+          " on Savannah")
+      (li ,(aa "bandali" "https://git.sr.ht/~bandali")
+          " on Sourcehut")
+      (li ,(aa "bandali" "https://lobste.rs/u/bandali")
+          " on Lobsters")
+      (li ,(aa "bandali" "https://hackage.haskell.org/user/bandali")
+          " on Hackage")
+      (li ,(aa "bandali" "https://gitlab.com/bandali")
+          " on GitLab")
+      (li ,(aa "bandali"
+               "https://news.ycombinator.com/user?id=bandali")
+          " on HN")
+      (li ,(aa "bandali" "https://www.reddit.com/u/bandali")
+          " on reddit")
+      (li (del ,(aa "bandali0" "https://github.com/bandali0")
+               " on GitHub"))
+      (li (del ,(aa "bandali0" "https://twitter.com/bandali0")
+               " on Twitter"))))
+   "2019"))
+
+(define cv-page
+  (static-page
+   "Curriculum vitae"
+   "bandali-cv.html"
+   `((h1 "Curriculum vitae (" ,(aa "PDF" "bandali-cv.pdf") ")")
+     (table
+      (tbody
+       (tr
+        (td "Site")
+        (td ,(aa my-domain my-url)))
+       (tr
+        (td "Email")
+        (td "bandali@uwaterloo.ca"))
+       (tr
+        (td "Phone")
+        (td "available upon request via email"))))
+     (h2 "Education")
+     (h3 "Master of Mathematics (Computer Science) | 2018–present")
+     (p "University of Waterloo, Canada")
+     (p "Supervised by Dr. Nancy Day | GPA: 3.7/4.0 | "
+        "Expected completion: April 2020")
+     (p "Research focusing on formal logic, model checking, and "
+        "verification.")
+     (h3 "B.Sc. Honours Computer Science | 2013–2017")
+     (p "York University, Toronto, Canada")
+     (p "GPA: 7.84/9.0")
+     (p "Relevant courses: System Specification & Refinement, "
+        "Software Requirements Eng., Software Design, "
+        "Operating Systems, Computational Complexity, "
+        "Design & Analysis of Algorithms.")
+     (p "Finished first year (2013-14) at " (em "Carleton University")
+        " with a GPA of 11.0/12.0, then transferred to "
+        (em "York University") " in Fall 2014.")
+     (h2 "Publications")
+     (p "Listed on my " ,(aa "homepage" "/#papers"))
+     (h2 "Work & Research Experience")
+     (h3 "Cheriton School of Computer Science, University of Waterloo"
+         " | 2018–present")
+     (p "Instructional Apprentice, Teaching Assistant, "
+        "Research Assistant")
+     (ul
+      (li (abbr (@ (title "Logic and Computation")) "SE 212") ": "
+          (abbr (@ (title "Instructional Apprentice")) "IA") " in "
+          "Fall 2019, "
+          (abbr (@ (title "Teaching Assistant")) "TA") " in "
+          "Fall 2018")
+      (li (abbr (@ (title ,(string-append
+                            "Software Requirements Specification and "
+                            "Analysis"))) "SE 463")
+          ": TA in Summer 2019 and 2018")
+      (li (abbr (@ (title ,(string-append
+                            "Elementary Algorithm Design and "
+                            "Data Abstraction"))) "CS 136")
+          ": TA in Winter 2018"))
+     (h3 (abbr (@ (title
+                   ,(string-append
+                     "Electrical Engineering & Computer Science")))
+               "EECS")
+         " Department, York University | Fall 2017")
+     (p "Teaching Assistant")
+     (p (abbr (@ (title "Net-Centric Introduction to Computing"))
+              "EECS 1012")
+        ": TA in Fall 2017"))
+   "2019"))
+
+(define se212-f19-page
+  (static-page
+   "SE 212 Material"
+   "se212-f19/index.html"
+   `((h1 "Material from SE 212 tutorials")
+     (p "This page contains slides and other material from "
+        ,(aa "SE 212 tutorials"
+             "https://www.student.cs.uwaterloo.ca/~se212/times.html")
+        " held by me in Fall 2019. "
+        (del "If you have any questions, concerns, or suggestions "
+             "about the presented material, please email me at "
+             "bandali@uwaterloo.ca or come see me during my "
+             ,(aa "Friday office hours"
+                  "https://www.student.cs.uwaterloo.ca/~se212/personnel.html")
+             "."))
+     (ul
+      (li "Tutorial 1:"
+          (ul
+           (li ,(aa "TUT 101 slides" "se212-t01-101.pdf"))
+           (li ,(aa "TUT 102 slides" "se212-t01-102.pdf"))
+           (li ,(aa "Org beamer sources" "se212-t01.org"))))
+      (li "Tutorial 2:"
+          (ul
+           (li ,(aa "Homework 2 q04d solution"
+                    "se212-h02q04d-soln.grg"))))
+      (li "Tutorial 3: —")
+      (li "Tutorial 4: —")
+      (li "Tutorial 5:"
+          (ul
+           (li ,(aa "Slides" "se212-t05.pdf"))
+           (li ,(aa "Org beamer sources" "se212-t05.org"))))
+      (li "Tutorial 6: —")
+      (li "Tutorial 7: worked through questions 1–5 of Homework 7")
+      (li "Tutorial 8: —")
+      (li "Tutorial 9: —")
+      (li "Tutorial 10: worked through questions 1–10 of "
+          "Homework 10")))
+   "2019"))
+
+(module-define!
+ (resolve-module '(haunt builder blog))
+ 'render-post
+ (lambda (theme site post)
+   (let ((title (post-ref post 'title))
+         (body ((theme-post-template theme) post))
+         (copy (post-ref post 'copyright)))
+     (base-layout site body #:title title #:copy copy))))
 
 (site #:title "Amin Bandali"
+      ;; TODO: uncomment after new haunt release
+      ;; #:scheme my-scheme
       #:domain my-domain
       #:default-metadata
       '((author . "Amin Bandali")
                              #:collections
                              `(("Notes" "notes.html"
                                 ,posts/reverse-chronological)))
+                       (tag-pages)
                        index-page
                        (atom-feed
-                        #:file-name "feed.atom")
+                        #:file-name "notes.atom")
                        (atom-feeds-by-tag
-                        #:prefix "tags")
+                        #:prefix my-tag-prefix)
                        (rss-feed
-                        #:file-name "feed.rss")
+                        #:file-name "notes.rss")
+                       contact-page
+                       cv-page
                        license-page
+                       se212-f19-page
                        (static-directory "static" "")))