-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathwww-lowdown-colorize.scm
99 lines (84 loc) · 3.87 KB
/
www-lowdown-colorize.scm
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
;; This code is ported from Vasilij Schneidermann's gist:
;; https://gist.github.com/wasamasa/e49e66e050255a8973270e0a52d68818
;; This is an extension to the Chicken 5 `lowdown` egg to recognize
;; GitHub Flavored Markdown code blocks. These code blocks may contain
;; blank lines, and the line with the opening ``` can say the name of
;; the language being used in the block. We also add syntax coloring.
;; Example:
;; ```Scheme
;; (display "Hello world")
;; (newline)
;; ```
;; TODO:
;; - (html-colorize lang*) returns a HTML string, which is parsed back
;; into SXML by (html->sxml), then turned into HTML again by the
;; code that calls this module. Send a patch to the `colorize` egg
;; so it can return SXML directly and this module does not have to
;; depend on html-parser.
;; - Send a patch to add the features from this module into the
;; `lowdown` egg. The (lowdown extra) module would be a good place.
(define (fenced-code-block-end fence)
(any-of end-of-input
(skip non-indent-space
(char-seq fence)
(zero-or-more (is (string-ref fence 0)))
space*
line-end)))
(define (fenced-code-block-lines indent code-block-end)
(zero-or-more
(preceded-by (none-of code-block-end)
(repeated (is #\space) max: (length indent))
line)))
(define fenced-code-block-info-string
(as-string (zero-or-more (none-of* (is #\`) normal-line-end item))))
(define fenced-code-block
(sequence* ((indent non-indent-space)
(fence (as-string (repeated (in #\` #\~) min: 3)))
(_ space*)
(info fenced-code-block-info-string)
(_ normal-line-end))
(let ((code-block-end (fenced-code-block-end fence)))
(sequence* ((code-lines (fenced-code-block-lines
indent code-block-end))
(_ code-block-end))
(result `(verbatim (info ,(string-trim-both info))
(code ,@code-lines)))))))
(define (dashes->spaces string)
(string-map (lambda (c) (if (char=? c #\-) #\space c)) string))
(define (spaces->dashes string)
(string-map (lambda (c) (if (char=? c #\space) #\- c)) string))
(define (coloring-type-string->symbol string)
(let ((string (dashes->spaces string)))
(let loop ((names (coloring-type-names)))
(and (not (null? names))
(if (string-ci= string (cdar names))
(caar names)
(loop (cdr names)))))))
(define fenced-code-block-conversion-rules*
`((verbatim
. ,(lambda (_ contents)
(or (and-let* (((pair? contents))
((pair? (car contents)))
(info (alist-ref 'info contents))
(code (alist-ref 'code contents))
(code* (string-intersperse code ""))
(raw-lang (car info))
(lang-sym (coloring-type-string->symbol raw-lang))
(lang-dashed (spaces->dashes
(string-downcase raw-lang))))
(if (coloring-type-exists? lang-sym)
`(pre (code (@ (class ,(string-append
"colorize"
" language-" lang-dashed)))
,@(->> code*
(html-colorize lang-sym)
(html->sxml)
(cdr))))
`(pre (code ,code*))))
`(pre (code ,@contents)))))))
(define (enable-www-lowdown-colorize!)
(block-hook (cons fenced-code-block (block-hook)))
(markdown-html-conversion-rules*
(append fenced-code-block-conversion-rules*
(markdown-html-conversion-rules*)))
(void))