|
(load "~/.quicklisp/setup.lisp") |
|
(ql:quickload '("alexandria" "cl-ppcre" "iterate" "spinneret") :silent t) |
|
(use-package :alexandria) |
|
(use-package :iterate) |
|
(require :uiop) |
|
|
|
|
|
(define-constant +markup-to-html+ |
|
'(("^(<.*)$" . (:span :class "pink")) |
|
("^(>.*)$" . (:span :class "green")) |
|
("^==(.*)==$" . (:span :class "heading")) |
|
("\\*{2}(.*?)\\*{2}" . (:span :class "spoiler")) |
|
("(?<!')'{2}([^']*?)'{2}" . (:i)) ;; band-aid solution to differentiate bold and italic regexps |
|
("'{3}(.*?)'{3}" . (:b)) |
|
("_{2}(.*?)_{2}" . (:u)) |
|
("~{2}(.*?)~{2}" . (:s))) |
|
:test #'equal) |
|
|
|
(defstruct (markup-region (:conc-name "")) |
|
(start 0 :type fixnum) |
|
(end 0 :type fixnum) |
|
(ctstart 0 :type fixnum) ;; content start |
|
(ctend 0 :type fixnum) ;; content end |
|
(tag nil :type list)) |
|
|
|
(defun find-markup (text) |
|
"Find all markup in text and return it as a sorted (by start pos) list of regions." |
|
(nreverse |
|
(iter |
|
(for (pat . tag) in +markup-to-html+) |
|
(for scanner = (cl-ppcre::create-scanner pat :multi-line-mode t)) |
|
(for matches = (let ((ret)) |
|
(cl-ppcre:do-scans (match-start match-end reg-starts reg-ends scanner text ret) |
|
(push (make-markup-region :start match-start |
|
:end match-end |
|
:ctstart (first-elt reg-starts) |
|
:ctend (first-elt reg-ends) |
|
:tag tag) |
|
ret)))) |
|
(accumulate matches by (lambda (acc l) (merge 'list acc l #'> :key #'start)))))) |
|
|
|
(defun create-tree (markup-map text &key (root-tag :body)) |
|
"Transform the output of find-markup into a complete spinneret tag tree." |
|
(car |
|
(create-tree/impl |
|
(cons |
|
(make-markup-region |
|
:start 0 |
|
:end (length text) |
|
:ctstart 0 |
|
:ctend (length text) |
|
:tag (list root-tag)) |
|
markup-map) |
|
text))) |
|
|
|
(defun create-tree/impl (markup-map text) |
|
(macrolet ((extract (start end) |
|
`(if (/= ,start ,end) |
|
(push (subseq text ,start ,end) ret)))) |
|
(iter |
|
(with root = (car markup-map)) |
|
(with xstart = (ctstart root)) |
|
(with next-cons = (cdr markup-map)) |
|
(with prev) |
|
(with ret = (reverse (tag root))) |
|
|
|
(while next-cons) |
|
(for next = (car next-cons)) |
|
(while (<= (end next) (ctend root))) |
|
|
|
(extract xstart (start next)) |
|
|
|
(destructuring-bind (child . new-next-cons) (create-tree/impl next-cons text) |
|
(setf prev next |
|
next-cons new-next-cons |
|
next (car next-cons) |
|
xstart (end prev)) |
|
(push child ret)) |
|
(while next-cons) |
|
|
|
(finally |
|
(extract xstart (ctend root)) |
|
(return (cons (nreverse ret) next-cons)))))) |
|
|
|
(defun print-tree (node &key (depth 0) (stream t)) |
|
(flet ((preprocess (obj) |
|
(if (stringp obj) (substitute #\$ #\Newline obj) obj))) |
|
(format stream "~&~0,vT(~S" (* depth 2) (preprocess (car node))) |
|
(dolist (child (cdr node)) |
|
(if (consp child) |
|
(print-tree child :depth (+ depth 1) :stream stream) |
|
(format stream "~&~0,vT~S" (* (+ depth 1) 2) (preprocess child)))) |
|
(format stream ")") |
|
(if (zerop depth) |
|
(format stream "~%")))) |
|
|
|
|
|
(if (equal uiop:*command-line-arguments* '("-h")) |
|
(uiop:die 0 "Usage: ~A [TITLE] <DOC.txt >DOC.html" |
|
(file-namestring #.(or *compile-file-truename* *load-truename*)))) |
|
|
|
(let* ((text (read-stream-content-into-string *standard-input*)) |
|
(markup-map (find-markup text)) |
|
(tree (create-tree markup-map text :root-tag :pre)) |
|
(spinneret:*html-style* :tree) |
|
(*print-pretty* nil)) |
|
;; (print-tree tree :stream *debug-io*) |
|
(spinneret:with-html (:doctype)) |
|
(spinneret:interpret-html-tree |
|
`(:html |
|
(:head |
|
(:title ,(car (or uiop:*command-line-arguments* '("Title")))) |
|
(:link :rel "stylesheet" :href "gtext.css")) |
|
(:body |
|
,tree)))) |