summaryrefslogtreecommitdiff
path: root/syntax-highlight.scm
blob: 265ff37babf8b740eb806296013488eea20b0def (plain)
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
;;; guile-syntax-highlight -- General-purpose syntax highlighter
;;; Copyright © 2015 David Thompson <davet@gnu.org>
;;;
;;; Guile-syntax-highlight is free software; you can redistribute it
;;; and/or modify it under the terms of the GNU Lesser General Public
;;; License as published by the Free Software Foundation; either
;;; version 3 of the License, or (at your option) any later version.
;;;
;;; Guile-syntax-highlight is distributed in the hope that it will be
;;; useful, but WITHOUT ANY WARRANTY; without even the implied
;;; warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
;;; See the GNU Lesser General Public License for more details.
;;;
;;; You should have received a copy of the GNU Lesser General Public
;;; License along with guile-syntax-highlight.  If not, see
;;; <http://www.gnu.org/licenses/>.

;;; Commentary:
;;
;; General-purpose syntax highlighting framework.
;;
;;; Code:

(define-module (syntax-highlight)
  #:use-module (ice-9 match)
  #:use-module (ice-9 rdelim)
  #:use-module (srfi srfi-1)
  #:use-module (srfi srfi-11)
  #:use-module (srfi srfi-26)
  #:use-module (syntax-highlight lexers)
  #:export (highlight
            highlights->sxml))

(define (object->string obj)
  (cond
   ((input-port? obj)
    (read-string obj))
   ((string? obj)
    obj)
   (else
    (error "not an input port or string: "
           obj))))

(define object->cursor
  (compose string->cursor object->string))

(define* (highlight lexer #:optional (string-or-port (current-input-port)))
  "Apply LEXER to STRING-OR-PORT, a string or an open input port.  If
STRING-OR-PORT is not specified, characters are read from the current
input port."
  (let-values (((result remainder)
                (lexer empty-tokens (object->cursor string-or-port))))
    (and result (tokens->list result))))

(define (highlights->sxml highlights)
  "Convert HIGHLIGHTS, a list of syntax highlighting expressions, into
a list of SXML 'span' nodes.  Each 'span' node has a 'class' attribute
corresponding to the highlighting tag name."
  (define (tag->class tag)
    (string-append "syntax-" (symbol->string tag)))

  (map (match-lambda
         ((? string? str) str)
         ((content ...)
          (let loop ((tags '()) (text "") (content content))
            (match content
             (() `(span (@ (class ,(string-join (map tag->class tags) " "))) ,text))
             (((? symbol? tag) content ...)
              (loop (cons tag tags) text content))
             (((? string? s) content ...)
              (loop tags (string-append text s) content))))))
       highlights))