;;;; pages.lisp -- html generation functions for dnd
(in-package :dnd)
(defmacro with-page ((&key title) &body body)
`(with-html-string
(:doctype)
(:html
(:head
(:title ,title))
(:body ,@body))))
(defun godess-shrine ()
(with-page (:title "A Sacred Shrine")
(:header
(:h1 "Pray and become a hero..."))
(:form :method "POST" :action "/godess-shrine"
(:label :for "NAME" "Enter the epithet by which the ages shall know thy hero:")
(:input :name "NAME")
(:button :type "submit" "Pray To The Goddess"))))
(defun doorkeeper (&key (message "Come ye player, Wot's yer name?"))
(with-page (:title "Tavern Door")
(:h1 message)
(:form :method "POST" :action "/tavern-door"
(:label :for "NICK" "Wut's yer handle?:")
(:input :name "NICK")
(:button :type "submit" "Announce Thyself"))
(:h2 "Eh? Ye need to announce thyeself?")
(:a :href "/register" "Follow me...")))
(defun register ()
(with-page (:title "Register Player")
(:header
(:h1 "Choose a Nickname Player"))
(:form :method "POST" :action "/register"
(:label :for "NICK" "Choose a nickname. No spaces. Letters, Numbers, and -._")
(:input :name "NICK" :placeholder "superbob")
(:button :type "submit" "Register"))))
(defun tavern (player)
(with-page (:title "A Bustling Tavern")
(navbar)
(:div
:class "heroes-container"
(:h2 "Heroes of rampant renown:")
(hall-of-heroes))))
(defun navbar ()
(with-html
(:nav :class "navbar" :aria-label "Navigation"
(:div :class "logo" :aria-label "DND logo" "DND")
(:ul :class "nav-links" :aria-label "Nav links"
(:li (:a :href "/hero" :aria-label "Hero profile" "🧝"))
(:li (:a :href "/inventory" :aria-label "Inventory" "🎒"))
(:li (:a :href "/quests" :aria-label "Quests" "📜"))
(:li (:a :href "/tavern" :aria-label "Tavern" "🍺"))))))
(defun hall-of-heroes ()
(with-html
(:ul :class "hall-of-heroes"
(dolist (hero (all-heroes))
(:li (hero-name hero)
"the"
(hero-class hero)
(hero-title hero))))))
(defgeneric render (element)
(:documentation "Render an element as HTML"))
(defclass page ()
((title
:reader title
:initarg :title
:documentation "Page title ")
(main
:reader main
:initarg :main
:initform ""
:documentation "Main content for a page.")))
(defmethod render ((doc page))
(with-html
(:doctype)
(:html
(:head (:title (title doc)))
(:body
(main doc)))))
(defclass navable ()
((nav-links
:reader nav-links
:initarg :nav-links
:type list))
(:documentation "Mixin for a PAGE class with navigation."))
(defclass nav-link ()
((target :initarg :target)
(text :initarg :text)
(icon :initarg :icon :initform nil)
(label :initarg :label :initform nil)))
(defmethod render ((element nav-link))
(with-slots (target icon label text) element
(with-html
(:li (:a :href target :aria-label label icon text)))))
(defclass page-with-nav (page navable)
())
(defmethod render ((doc page-with-nav))
(with-html
(:doctype)
(:html
(:head (:title (title doc)))
(:body
(:nav (:ul (mapc #'render (nav-links doc))))
(:main (main doc))))))
;; (let* ((link (make-instance
;; 'nav-link :target "/moo" :label "goo" :icon "🍺" :text "Tavern"))
;; (foo (make-instance
;; 'page-with-nav :title "goober" :main "Content" :nav-links (list link link))))
;; (render foo))