Cleaned up the definition of @chunk and @CHUNK. Fixed issues with subscript numbers on chunks, links, and conflicting tags. Now it's possible to re-print a chunk several times without problem.
This commit is contained in:
parent
148ec8da08
commit
fe2b582def
|
@ -221,10 +221,7 @@
|
||||||
(define-syntax-rule
|
(define-syntax-rule
|
||||||
(unless-preexpanding . b)
|
(unless-preexpanding . b)
|
||||||
(begin . b))
|
(begin . b))
|
||||||
(require (only-in racket/require
|
(require scribble-enhanced/with-manual
|
||||||
subtract-in)
|
|
||||||
(subtract-in scribble/manual
|
|
||||||
scribble-enhanced)
|
|
||||||
scribble-enhanced
|
scribble-enhanced
|
||||||
hyper-literate))))
|
hyper-literate))))
|
||||||
(begn body0 . body))
|
(begn body0 . body))
|
||||||
|
|
257
private/lp.rkt
257
private/lp.rkt
|
@ -18,119 +18,176 @@
|
||||||
chunk-numbers id
|
chunk-numbers id
|
||||||
(+ 1 (free-identifier-mapping-get chunk-numbers id))))
|
(+ 1 (free-identifier-mapping-get chunk-numbers id))))
|
||||||
(define (init-chunk-number id)
|
(define (init-chunk-number id)
|
||||||
(free-identifier-mapping-put! chunk-numbers id 2)))
|
(free-identifier-mapping-put! chunk-numbers id 2))
|
||||||
|
|
||||||
|
(define repeat-chunk-numbers (make-free-identifier-mapping))
|
||||||
|
(define (get+increment-repeat-chunk-number! id)
|
||||||
|
(let ([current (free-identifier-mapping-get repeat-chunk-numbers
|
||||||
|
id
|
||||||
|
(lambda () 1))])
|
||||||
|
(free-identifier-mapping-put! repeat-chunk-numbers id (add1 current))
|
||||||
|
;; note: due to multiple expansions, this does not increase exactly one at
|
||||||
|
;; a time but instead it can skip numbers. Since this is not visible by
|
||||||
|
;; the user, and just used as a token in the URL, it's okay as long as
|
||||||
|
;; compiling the same file twice gives the same numbers (which is
|
||||||
|
;; hopefully the case but hasn't been tested).
|
||||||
|
current)))
|
||||||
|
|
||||||
(require (for-syntax "no-auto-require.rkt"))
|
(require (for-syntax "no-auto-require.rkt"))
|
||||||
(define-for-syntax ((make-chunk racketblock unsyntax?) stx)
|
(define-for-syntax (make-chunk-code unsyntax?)
|
||||||
|
(syntax-parser
|
||||||
|
;; no need for more error checking, using chunk for the code will do that
|
||||||
|
[(_ name:id expr ...)
|
||||||
|
|
||||||
|
;; Lift the code so that it is caught by `extract-chunks` in common.rkt
|
||||||
|
;(syntax-local-lift-expression #'(quote-syntax (a-chunk name expr ...)))
|
||||||
|
|
||||||
|
;; Convoluted trick to allow unsyntax in chunks of code. The unsyntax
|
||||||
|
;; escapes the chunk so that code can be injected at compile-time.
|
||||||
|
;; The identifiers inside the escaped portion need to be available both
|
||||||
|
;; for-syntax i.e. (for-meta 1) and (for-meta 0). This is because the
|
||||||
|
;; underlying @racketblock expands the code at run-time, but the
|
||||||
|
;; extract-chunks function in common.rkt looks at the expanded source
|
||||||
|
;; code.
|
||||||
|
;; For now, only #, i.e. unsyntax is supported, within @chunk.
|
||||||
|
;; Later support for UNSYNTAX within @CHUNK may be added.
|
||||||
|
(define expand-unsyntax
|
||||||
|
(if unsyntax?
|
||||||
|
;; New hack:
|
||||||
|
#'((define-syntax (macro-to-expand-unsyntax _)
|
||||||
|
(define a #'here)
|
||||||
|
(define b (syntax-local-identifier-as-binding
|
||||||
|
(syntax-local-introduce #'here)))
|
||||||
|
(define intr (make-syntax-delta-introducer b a))
|
||||||
|
(syntax-local-lift-expression
|
||||||
|
(intr #'(quote-syntax (a-chunk ((... ...) name)
|
||||||
|
((... ...) expr) ...))
|
||||||
|
'flip))
|
||||||
|
#'(void))
|
||||||
|
(macro-to-expand-unsyntax))
|
||||||
|
;; Default (old) behaviour, does not support escaping via #,
|
||||||
|
(begin (syntax-local-lift-expression
|
||||||
|
#'(quote-syntax (a-chunk name expr ...)))
|
||||||
|
#f)))
|
||||||
|
|
||||||
|
(with-syntax
|
||||||
|
;; Extract require forms
|
||||||
|
([((for-label-mod ...) ...)
|
||||||
|
(if (unbox no-auto-require?)
|
||||||
|
#'()
|
||||||
|
(map (lambda (expr)
|
||||||
|
(syntax-case expr (require)
|
||||||
|
[(require mod ...)
|
||||||
|
(let loop ([mods (syntax->list
|
||||||
|
#'(mod ...))])
|
||||||
|
(cond
|
||||||
|
[(null? mods) null]
|
||||||
|
[else
|
||||||
|
(syntax-case (car mods)
|
||||||
|
(for-syntax quote submod)
|
||||||
|
[(submod ".." . _)
|
||||||
|
(loop (cdr mods))]
|
||||||
|
[(submod "." . _)
|
||||||
|
(loop (cdr mods))]
|
||||||
|
[(quote x)
|
||||||
|
(loop (cdr mods))]
|
||||||
|
[(for-syntax x ...)
|
||||||
|
(append (loop (syntax->list
|
||||||
|
#'(x ...)))
|
||||||
|
(loop (cdr mods)))]
|
||||||
|
[x
|
||||||
|
(cons #'x (loop (cdr mods)))])]))]
|
||||||
|
[else null]))
|
||||||
|
(syntax->list #'(expr ...))))])
|
||||||
|
#`(begin
|
||||||
|
#,@(if expand-unsyntax expand-unsyntax #'())
|
||||||
|
#,@(if (null? (syntax-e #'(for-label-mod ... ...)))
|
||||||
|
#'()
|
||||||
|
#'((require (for-label for-label-mod ... ...))))))]))
|
||||||
|
|
||||||
|
(define-for-syntax ((make-chunk-display racketblock) stx)
|
||||||
(syntax-parse stx
|
(syntax-parse stx
|
||||||
[(_ (~optional (~seq #:save-as save-as)) name expr ...)
|
;; no need for more error checking, using chunk for the code will do that
|
||||||
;; no need for more error checking, using chunk for the code will do that
|
[(_ original-name:id name:id stxn:number expr ...)
|
||||||
(identifier? #'name)
|
(define n (syntax-e #'stxn))
|
||||||
(let* ([n (get-chunk-number (syntax-local-introduce #'name))]
|
(define n-repeat (get+increment-repeat-chunk-number!
|
||||||
[str (symbol->string (syntax-e #'name))]
|
(syntax-local-introduce
|
||||||
[tag (format "chunk:~a:~a" str (or n 1))])
|
(format-id #'name "~a:~a" #'name n))))
|
||||||
|
(define str (string-append (symbol->string (syntax-e #'name))))
|
||||||
|
(define/with-syntax tag (format "chunk:~a:~a:~a" str n n-repeat))
|
||||||
|
(define/with-syntax (rest ...)
|
||||||
|
;; if the would-be-next number for this chunk name is "2", then there is
|
||||||
|
;; only one chunk, whose number is "1". Otherwise, if the number is 3 or
|
||||||
|
;; more, it means that the chunk with number "2" exists, so we should
|
||||||
|
;; display the subscript numbers.
|
||||||
|
(if (let ([c (get-chunk-number #'original-name)])
|
||||||
|
(and c (> c 2)))
|
||||||
|
#`((subscript #,(format "~a" n)))
|
||||||
|
#'()))
|
||||||
|
#`(make-splice
|
||||||
|
(list (make-toc-element
|
||||||
|
#f
|
||||||
|
(list (elemtag '(prefixable tag)
|
||||||
|
(bold (italic (elemref '(prefixable tag)
|
||||||
|
#:underline? #f
|
||||||
|
#,str rest ...))
|
||||||
|
" ::=")))
|
||||||
|
(list (smaller (elemref '(prefixable tag) #:underline? #f
|
||||||
|
#,str
|
||||||
|
rest ...))))
|
||||||
|
(#,racketblock expr ...)))]))
|
||||||
|
|
||||||
(when n
|
(define-for-syntax (make-chunk chunk-code chunk-display)
|
||||||
(inc-chunk-number (syntax-local-introduce #'name)))
|
(syntax-parser
|
||||||
|
;; no need for more error checking, using chunk for the code will do that
|
||||||
|
[(_ (~optional (~seq #:save-as save-as:id)) name:id expr ...)
|
||||||
|
(define n (get-chunk-number (syntax-local-introduce #'name)))
|
||||||
|
(define str (symbol->string (syntax-e #'name)))
|
||||||
|
|
||||||
;; Lift the code so that it is caught by `extract-chunks` in common.rkt
|
(when n
|
||||||
;(syntax-local-lift-expression #'(quote-syntax (a-chunk name expr ...)))
|
(inc-chunk-number (syntax-local-introduce #'name)))
|
||||||
|
|
||||||
;; Convoluted trick to allow unsyntax in chunks of code. The unsyntax
|
(define/with-syntax stx-n (or n 1))
|
||||||
;; escapes the chunk so that code can be injected at compile-time.
|
(define/with-syntax stx-chunk-code chunk-code)
|
||||||
;; The identifiers inside the escaped portion need to be available both
|
(define/with-syntax stx-chunk-display chunk-display)
|
||||||
;; for-syntax i.e. (for-meta 1) and (for-meta 0). This is because the
|
|
||||||
;; underlying @racketblock expands the code at run-time, but the
|
|
||||||
;; extract-chunks function in common.rkt looks at the expanded source
|
|
||||||
;; code.
|
|
||||||
;; For now, only #, i.e. unsyntax is supported, within @chunk.
|
|
||||||
;; Later support for UNSYNTAX within @CHUNK may be added.
|
|
||||||
(define expand-unsyntax
|
|
||||||
(if unsyntax?
|
|
||||||
;; New hack:
|
|
||||||
#'((define-syntax (macro-to-expand-unsyntax _)
|
|
||||||
(define a #'here)
|
|
||||||
(define b (syntax-local-identifier-as-binding
|
|
||||||
(syntax-local-introduce #'here)))
|
|
||||||
(define intr (make-syntax-delta-introducer b a))
|
|
||||||
(syntax-local-lift-expression
|
|
||||||
(intr #`(quote-syntax (a-chunk ((... ...) name)
|
|
||||||
((... ...) expr) ...))
|
|
||||||
'flip))
|
|
||||||
#'(void))
|
|
||||||
(macro-to-expand-unsyntax))
|
|
||||||
;; Default (old) behaviour, which does not support escaping via #,
|
|
||||||
(begin (syntax-local-lift-expression
|
|
||||||
#'(quote-syntax (a-chunk name expr ...)))
|
|
||||||
#f)))
|
|
||||||
|
|
||||||
|
#`(begin
|
||||||
|
(stx-chunk-code name expr ...)
|
||||||
|
#,@(if n
|
||||||
|
#'()
|
||||||
|
#'((define-syntax name (make-element-id-transformer
|
||||||
|
(lambda (stx) #'(chunkref name))))
|
||||||
|
(begin-for-syntax (init-chunk-number #'name))))
|
||||||
|
#,(if (attribute save-as)
|
||||||
|
#'(define-syntax (save-as s)
|
||||||
|
(syntax-case s ()
|
||||||
|
[(_)
|
||||||
|
#`(save-as #,(syntax-local-introduce
|
||||||
|
(quote-syntax name)))]
|
||||||
|
[(_ newname)
|
||||||
|
#`(stx-chunk-display
|
||||||
|
#,(syntax-local-introduce
|
||||||
|
(quote-syntax name))
|
||||||
|
newname
|
||||||
|
stx-n
|
||||||
|
#,@(syntax-local-introduce
|
||||||
|
(quote-syntax (expr ...))))]))
|
||||||
|
;; The (list) here is important, to avoid the code being executed
|
||||||
|
;; multiple times in weird ways, when pre-expanding.
|
||||||
|
#`(list (stx-chunk-display name name stx-n expr ...))))]))
|
||||||
|
|
||||||
;; Extract require forms
|
(define-syntax chunk-code (make-chunk-code #t))
|
||||||
(with-syntax ([tag tag]
|
(define-syntax CHUNK-code (make-chunk-code #f))
|
||||||
[str str]
|
(define-syntax chunk-display (make-chunk-display #'racketblock))
|
||||||
[((for-label-mod ...) ...)
|
(define-syntax CHUNK-display (make-chunk-display #'RACKETBLOCK))
|
||||||
(if (unbox no-auto-require?)
|
(define-syntax chunk (make-chunk #'chunk-code #'chunk-display))
|
||||||
#'()
|
(define-syntax CHUNK (make-chunk #'CHUNK-code #'CHUNK-display))
|
||||||
(map (lambda (expr)
|
|
||||||
(syntax-case expr (require)
|
|
||||||
[(require mod ...)
|
|
||||||
(let loop ([mods (syntax->list
|
|
||||||
#'(mod ...))])
|
|
||||||
(cond
|
|
||||||
[(null? mods) null]
|
|
||||||
[else
|
|
||||||
(syntax-case (car mods)
|
|
||||||
(for-syntax quote submod)
|
|
||||||
[(submod ".." . _)
|
|
||||||
(loop (cdr mods))]
|
|
||||||
[(submod "." . _)
|
|
||||||
(loop (cdr mods))]
|
|
||||||
[(quote x)
|
|
||||||
(loop (cdr mods))]
|
|
||||||
[(for-syntax x ...)
|
|
||||||
(append (loop (syntax->list
|
|
||||||
#'(x ...)))
|
|
||||||
(loop (cdr mods)))]
|
|
||||||
[x
|
|
||||||
(cons #'x (loop (cdr mods)))])]))]
|
|
||||||
[else null]))
|
|
||||||
(syntax->list #'(expr ...))))]
|
|
||||||
|
|
||||||
[(rest ...) (if n
|
|
||||||
#`((subscript #,(format "~a" n)))
|
|
||||||
#`())])
|
|
||||||
(define/with-syntax pre-content
|
|
||||||
#`(make-splice
|
|
||||||
(list (make-toc-element
|
|
||||||
#f
|
|
||||||
(list (elemtag '(prefixable tag)
|
|
||||||
(bold (italic (racket name)) " ::=")))
|
|
||||||
(list (smaller (elemref '(prefixable tag) #:underline? #f
|
|
||||||
str
|
|
||||||
rest ...))))
|
|
||||||
(#,racketblock expr ...))))
|
|
||||||
#`(begin
|
|
||||||
#,@(if expand-unsyntax expand-unsyntax #'())
|
|
||||||
#,@(if (null? (syntax-e #'(for-label-mod ... ...)))
|
|
||||||
#'()
|
|
||||||
#'((require (for-label for-label-mod ... ...))))
|
|
||||||
#,@(if n
|
|
||||||
#'()
|
|
||||||
#'((define-syntax name (make-element-id-transformer
|
|
||||||
(lambda (stx) #'(chunkref name))))
|
|
||||||
(begin-for-syntax (init-chunk-number #'name))))
|
|
||||||
#,(if (attribute save-as)
|
|
||||||
#'(define-syntax (save-as s) (quote-syntax pre-content))
|
|
||||||
#'pre-content))))]))
|
|
||||||
|
|
||||||
(define-syntax chunk (make-chunk #'racketblock #t))
|
|
||||||
(define-syntax CHUNK (make-chunk #'RACKETBLOCK #f))
|
|
||||||
|
|
||||||
(define-syntax (chunkref stx)
|
(define-syntax (chunkref stx)
|
||||||
(syntax-case stx ()
|
(syntax-case stx ()
|
||||||
[(_ id)
|
[(_ id)
|
||||||
(identifier? #'id)
|
(identifier? #'id)
|
||||||
(with-syntax ([tag (format "chunk:~a:1" (syntax-e #'id))]
|
(with-syntax ([tag (format "chunk:~a:1:1" (syntax-e #'id))]
|
||||||
[str (format "~a" (syntax-e #'id))])
|
[str (format "~a" (syntax-e #'id))])
|
||||||
#'(elemref '(prefixable tag) #:underline? #f str))]))
|
#'(elemref '(prefixable tag) #:underline? #f str))]))
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue
Block a user