Workaround to be able to use @include-section to include several hyper-literate programs into a single document.

This commit is contained in:
Georges Dupéron 2016-09-30 16:13:58 +02:00
parent a7f15cdfd5
commit 397680acb0
4 changed files with 81 additions and 11 deletions

View File

@ -0,0 +1,4 @@
#lang racket/base
(provide chunks-toc-prefix)
(define chunks-toc-prefix (make-parameter '()))

View File

@ -159,7 +159,8 @@
(let ([expanded (let ([expanded
(expand `(,#'module (expand `(,#'module
scribble-lp-tmp-name hyper-literate/private/lp scribble-lp-tmp-name hyper-literate/private/lp
(require (for-syntax racket/base (require hyper-literate/private/chunks-toc-prefix
(for-syntax racket/base
hyper-literate/private/no-auto-require)) hyper-literate/private/no-auto-require))
(begin-for-syntax (set-box! no-auto-require? (begin-for-syntax (set-box! no-auto-require?
,(if (attribute no-auto-require) #t #f))) ,(if (attribute no-auto-require) #t #f)))
@ -210,7 +211,8 @@
;; and make these identifiers exported by ;; and make these identifiers exported by
;; hyper-literate ;; hyper-literate
(strip-context (strip-context
#`((require (for-syntax racket/base #`((require hyper-literate/private/chunks-toc-prefix
(for-syntax racket/base
hyper-literate/private/no-auto-require)) hyper-literate/private/no-auto-require))
(begin-for-syntax (set-box! no-auto-require? (begin-for-syntax (set-box! no-auto-require?
#,(if (attribute no-auto-require) #t #f))) #,(if (attribute no-auto-require) #t #f)))

View File

@ -40,7 +40,8 @@
(free-identifier-mapping-put! repeat-chunk-numbers id (add1 current)) (free-identifier-mapping-put! repeat-chunk-numbers id (add1 current))
current))) current)))
(require (for-syntax "no-auto-require.rkt")) (require (for-syntax "no-auto-require.rkt")
"chunks-toc-prefix.rkt")
(define-for-syntax (make-chunk-code unsyntax?) (define-for-syntax (make-chunk-code unsyntax?)
(syntax-parser (syntax-parser
;; 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
@ -118,10 +119,13 @@
[(_ original-name:id name:id stxn:number expr ...) [(_ original-name:id name:id stxn:number expr ...)
(define n (syntax-e #'stxn)) (define n (syntax-e #'stxn))
(define original-name:n (syntax-local-introduce (define original-name:n (syntax-local-introduce
(format-id #'original-name "~a:~a" #'original-name n))) (format-id #'original-name
"~a:~a"
#'original-name
n)))
(define n-repeat (get+increment-repeat-chunk-number! (define n-repeat (get+increment-repeat-chunk-number!
original-name:n)) original-name:n))
(define str (string-append (symbol->string (syntax-e #'name)))) (define str (symbol->string (syntax-e #'name)))
(define/with-syntax tag (format "chunk:~a:~a:~a" str n n-repeat)) (define/with-syntax tag (format "chunk:~a:~a:~a" str n n-repeat))
(define/with-syntax (rest ...) (define/with-syntax (rest ...)
;; if the would-be-next number for this chunk name is "2", then there is ;; if the would-be-next number for this chunk name is "2", then there is
@ -143,9 +147,12 @@
#:underline? #f #:underline? #f
#,str rest ...)) #,str rest ...))
" ::="))) " ::=")))
(list (smaller (elemref '(prefixable tag) #:underline? #f (list (smaller
#,str (make-link-element "plainlink"
rest ...)))) (decode-content (list #,str rest ...))
`(elem (prefixable
,@(chunks-toc-prefix)
tag))))))
(#,racketblock expr ...))))])) (#,racketblock expr ...))))]))
(define-for-syntax (make-chunk chunk-code chunk-display) (define-for-syntax (make-chunk chunk-code chunk-display)
@ -230,4 +237,6 @@
(provide (all-from-out scheme/base (provide (all-from-out scheme/base
scribble-enhanced/with-manual) scribble-enhanced/with-manual)
chunk CHUNK) chunk
CHUNK
chunks-toc-prefix)

View File

@ -1,6 +1,9 @@
#lang scribble/manual #lang scribble/manual
@require[@for-label[;hyper-literate @require[racket/require
racket/base]] @for-label[hyper-literate
racket/base
(subtract-in scribble/manual hyper-literate)
racket/contract]]
@title{hyper-literate} @title{hyper-literate}
@author{Georges Dupéron} @author{Georges Dupéron}
@ -198,3 +201,55 @@ for the first time or not (in which case the
@defform[(unless-preexpanding . body)]{ @defform[(unless-preexpanding . body)]{
Expands to @racket[(begin . body)] if the @racket[(submod "..")] Expands to @racket[(begin . body)] if the @racket[(submod "..")]
module can be used, and expands to @racket[(begin)] otherwise.} module can be used, and expands to @racket[(begin)] otherwise.}
@section{A note on literate programs as subsections of another document}
To use @racket[include-section] on hyper-literate programs, a couple of
workarounds are required to avoid issues with duplicate tags for
identically-named chunks (like @racket[<*>], which is likely to always be
present).
@defparam[chunks-toc-prefix prefix-list (listof string?)]{
We give an example for two files which are part of a hypothetical
@elem[#:style 'tt "pkg"] package:
@itemlist[
@item{The main scribble file @filepath{main.scrbl} in the
@filepath{scribblings} sub-directory includes the hyper-literate file
@filepath{program.hl.rkt} located in the package's root directory, one
directory level above:
@codeblock[#:keep-lang-line? #t
"#lang scribble/manual\n"
"@title{Main document title}\n"
"@include-section{../program.hl.rkt}\n"
"@; could include other hyper-literat programs here\n"]}
@item{To avoid issues with duplicate tag names, it is necessary to use the
@racket[#:tag-prefix] option on the hyper literate program's @racket[title].
Unfortunately, this breaks links to chunks in the table of contents, because
scribble does not automatically add the correct prefix to them. To ensure
that the links correctly work in the table of contents, it is necessary to
tell hyper-literate what is the chain of document includes. The whole
@filepath{program.hl.rkt} file will be:
@codeblock[#:keep-lang-line? #t
"#lang hyper-literate racket/base\n"
"@title[#:tag-prefix '(lib \"pkg/program.hl.rkt\")]{Program title}\n"
"@(chunks-toc-prefix '(\"(lib pkg/scribblings/main.scrbl)\"\n"
" \"(lib pkg/program.hl.rkt)\"))\n"
"@chunk[<*>\n"
" 'program-code-here]\n"]
Note that the argument for the @racket[chunks-toc-prefix] parameter is a list
of string, and the strings are representations of module paths. The
occurrences of @racket[lib] above are not symbols, they are just part of the
string. Compare this with the following, which would be incorrect:
@codeblock[#:keep-lang-line? #t
"#lang hyper-literate racket/base\n"
"@title[#:tag-prefix '(lib \"pkg/program.hl.rkt\")]{Program title}\n"
"@; This is incorrect:\n"
"@(chunks-toc-prefix '((lib \"pkg/scribblings/main.scrbl\")\n"
" (lib \"pkg/program.hl.rkt\")))\n"
"@chunk[<*>\n"
" 'program-code-here]\n"]}]}