X-Git-Url: https://git.shemshak.org/~bandali/bndl.org/blobdiff_plain/8c2dfb465c044b15b38ce0f4cc48ee9431fa2a4f..08d93db6b9653bd4e5a4dbe6f5d6e54ea469248d:/haunt.scm diff --git a/haunt.scm b/haunt.scm index c51ecf8..7b2d740 100644 --- a/haunt.scm +++ b/haunt.scm @@ -16,6 +16,8 @@ (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") @@ -24,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 @@ -36,11 +38,70 @@ (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 my-date-format "~B ~d, ~Y") +(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 (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) (stringdate*) + +(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 @@ -48,7 +109,22 @@ (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 " @@ -61,19 +137,35 @@ "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-h2") + (src "/icon-16px.png") + (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 (small + ,(date->string (post-date post) + my-date-format))))) + posts))) + (h2 "Tags") + ,(tag-links (or all-posts posts)) + ,(if tag + '(a (@ (href "/notes.html")) + "← all notes") + '()))) (define bandali-theme (theme #:name "bandali" @@ -84,15 +176,15 @@ #: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 " @@ -193,19 +285,31 @@ (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 `((img (@ (class "feed-icon") + (src "/icon-12px.png") + (alt "subscribe to atom feed"))) + "Atom") + "notes.atom") + " and " + ,(aa `((img (@ (class "feed-icon") + (src "/icon-12px.png") + (alt "subscribe to rss feed"))) + "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 @@ -217,7 +321,8 @@ (static-page "Licensing Information" "license.html" - `((h1 "License information for " ,my-domain) + `((h1 "License information for " + ,(aa my-domain my-url)) (p "I strongly believe in " ,(aa "free culture" "https://questioncopyright.org/what_is_free_culture") @@ -235,7 +340,9 @@ (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 contact-page (static-page @@ -291,7 +398,7 @@ (li ,(aa "bandali" "https://hackage.haskell.org/user/bandali") " on Hackage") (li ,(aa "bandali" "https://gitlab.com/bandali") - " on GitLab.com") + " on GitLab") (li ,(aa "bandali" "https://news.ycombinator.com/user?id=bandali") " on HN") @@ -300,7 +407,121 @@ (li (del ,(aa "bandali0" "https://github.com/bandali0") " on GitHub")) (li (del ,(aa "bandali0" "https://twitter.com/bandali0") - " on Twitter")))))) + " 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 @@ -315,13 +536,16 @@ #: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" "")))