(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")
(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 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) (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 "
(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 " ,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")
(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
(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")
(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
#: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" "")))