scribble: use 'pdf-bytes+bounds8, etc.

Use 'pdf-bytes+bounds8, 'png-bytes+bounds8, etc. to convert images
for Latex/PDF and HTML output. The new conversions particularly
help with picts that draw text, since the baseline of the text
can be handled properly in HTML output.

The new conversion also helps hide the difference between a bounding
box and the "inked" area of a pict, since picts normally ink only
slightly outside the bounding box, and the default for pict conversion
pads the image with a few pixels/units in each direction.

Also, add a `++convert <fmt>` command-line option to select a
preferred target for image conversions. (This preference can be
overridden by the existing `render-convertible-as` style property
for HTML rendering.)

original commit: 05760a12f690b7efcd58e65ee2b723a05133e229
This commit is contained in:
Matthew Flatt 2014-06-27 17:30:46 +01:00
parent a272f5a5ca
commit 271ec59ad2
9 changed files with 201 additions and 52 deletions

View File

@ -49,6 +49,7 @@ function to render a document.
[#:style-file style-file (or/c #f path-string?) #f]
[#:style-extra-files style-extra-files (listof path-string?) #f]
[#:extra-files extra-files (listof path-string?) #f]
[#:image-preferences image-preferences (listof (or/c 'ps 'pdf 'png 'svg 'gif)) null]
[#:xrefs xrefs (listof xref?) null]
[#:info-in-files info-in-files (listof path-string?) null]
[#:info-out-file info-out-file (or/c #f path-string?) #f]
@ -76,6 +77,12 @@ The @racket[helper-file-prefix], @racket[prefix-file],
@racket[extra-files] arguments are passed on to the @racket[render%]
constructor.
The @racket[image-preferences] argument specified preferred formats
for image files and conversion, where formats listed earlier in the
list are more preferred. The renderer specified by
@racket[render-mixin] may not support all of the formats listed in
@racket[image-preferences].
The @racket[xrefs] argument provides extra cross-reference information
to be used during the documents' @tech{resolve pass}. The
@racket[info-in-files] arguments supply additional cross-reference

View File

@ -36,7 +36,7 @@ its file suffix:
via @exec{pdflatex}}
@item{@DFlag{dvipdf} --- PDF @filepath{@|fn|.pdf} that is generated
via @exec{latex} and @exec{dvipdf}}
via @exec{latex}, @exec{dvips}, and @exec{pstopdf}}
@item{@DFlag{latex-section} @nonterm{n} --- LaTeX source
@filepath{@|fn|.tex} plus additional @filepath{.tex} files to
@ -168,6 +168,26 @@ builds @filepath{c.html} with cross-reference links into
@filepath{a.html} and @filepath{b.html}.
@section{Selecting an Image Format}
Use the @DPFlag{convert} @nonterm{fmt} flag to select @nonterm{fmt} as
a preferred image format to use when rendering a document that
includes values that can be converted to different image formats. The
@nonterm{fmt} argument can be @exec{pdf}, @exec{ps}, @exec{png},
@exec{svg}, or @exec{gif}, but a renderer typically supports only a
subset of those formats.
Use @DPFlag{convert} @nonterm{fmt} multiple times to specify multiple
preferred formats, where a @nonterm{fmt} earlier in the command line
take precedence over @nonterm{fmt}s specified later.
For example, to generate Latex sources with images in Encapsulated
PostScript format (so that the result works with @exec{latex} instead
of @exec{pdflatex}), combine @DFlag{latex} with @exec{@DPFlag{convert}
ps}. To generate HTML pages with images converted to SVG format
instead of PNG format, combine @DFlag{html} with
@exec{@DPFlag{convert} svg}.
@section{Passing Command-Line Arguments to Documents}
When @exec{scribble} loads and renders a document module, by default

View File

@ -53,6 +53,7 @@
[style-file #f]
[style-extra-files null]
[extra-files null]
[image-preferences null]
[helper-file-prefix #f])
(define/public (current-render-mode)
@ -759,6 +760,23 @@
;; ----------------------------------------
;; render methods
(define/public (sort-image-requests reqs prefs)
(for/fold ([reqs reqs]) ([pref (in-list (reverse prefs))])
(define matches
(for/list ([req (in-list reqs)]
#:when (case pref
[(png) (or (eq? req 'png@2x-bytes)
(eq? req 'png-bytes))]
[(svg) (eq? req 'svg-bytes)]
[(pdf) (eq? req 'pdf-bytes)]
[(ps) (eq? req 'eps-bytes)]
[(gif) (eq? req 'gif-bytes)]
[else #f]))
req))
(if (null? matches)
reqs
(append matches (remove* matches reqs)))))
(define/public (auto-extra-files? v) #f)
(define/public (auto-extra-files-paths v) null)
(define/public (skip-extra-file? v) #f)

View File

@ -277,7 +277,7 @@
extract-version
extract-authors
extract-pretitle)
(inherit-field prefix-file style-file style-extra-files)
(inherit-field prefix-file style-file style-extra-files image-preferences)
(init-field [alt-paths null]
;; `up-path' is either a link "up", or #t which goes
@ -547,6 +547,16 @@
(uri-unreserved-encode
(anchor-name (dest-anchor dest)))))))
(inherit sort-image-requests)
(define/override (render ds fns ri)
(parameterize ([current-render-convertible-requests
(sort-image-requests (current-render-convertible-requests)
image-preferences)])
(render-top ds fns ri)))
(define/public (render-top ds fns ri)
(super render ds fns ri))
(define/public (render-toc-view d ri)
(define has-sub-parts?
(pair? (part-parts d)))
@ -1370,29 +1380,76 @@
(define/private (render-as-convertible e requests)
(for/or ([request (in-list requests)])
(cond
[(and (or (equal? request 'png-bytes)
(equal? request 'png@2x-bytes))
(convert e request))
[(case request
[(png-bytes)
(or (convert e 'png-bytes+bounds8)
(convert e 'png-bytes+bounds)
(convert e 'png-bytes))]
[(png@2x-bytes)
(or (convert e 'png@2x-bytes+bounds8)
(convert e 'png@2x-bytes+bounds)
(convert e 'png@2x-bytes))]
[else #f])
=>
(lambda (bstr)
(let ([w (integer-bytes->integer (subbytes bstr 16 20) #f #t)]
[h (integer-bytes->integer (subbytes bstr 20 24) #f #t)]
[scale (lambda (v)
(if (equal? request 'png@2x-bytes)
(/ v 2.0)
v))])
`((img ([src ,(install-file "pict.png" bstr)]
[alt "image"]
[width ,(number->string (scale w))]
[height ,(number->string (scale h))])))))]
[(and (equal? request 'svg-bytes)
(convert e 'svg-bytes))
=> (lambda (bstr)
`((object
([data ,(install-file "pict.svg" bstr)]
[type "image/svg+xml"]))))]
(lambda (cvt)
(let* ([bstr (if (list? cvt) (first cvt) cvt)]
[w (if (list? cvt)
(list-ref cvt 1)
(integer-bytes->integer (subbytes bstr 16 20) #f #t))]
[h (if (list? cvt)
(list-ref cvt 2)
(integer-bytes->integer (subbytes bstr 20 24) #f #t))]
[scale (lambda (v)
(if (and (not (list? cvt))
(equal? request 'png@2x-bytes))
(/ v 2.0)
v))])
(list
(add-padding
cvt
`(img ([src ,(install-file "pict.png" bstr)]
[alt "image"]
[width ,(number->string (scale w))]
[height ,(number->string (scale h))]))))))]
[(case request
[(svg-bytes)
(or (convert e 'svg-bytes+bounds8)
(convert e 'svg-bytes))]
[else #f])
=> (lambda (cvt)
(let* ([bstr (if (list? cvt) (first cvt) cvt)])
(list
(add-padding
cvt
`(object
([data ,(install-file "pict.svg" bstr)]
[type "image/svg+xml"]))))))]
[else #f])))
;; Add padding for a bounding-box conversion reply:
(define/private (add-padding cvt e)
(define descent (and (list? cvt)
((length cvt) . >= . 5)
(list-ref cvt 3)))
(define padding (and (list? cvt)
((length cvt) . >= . 9)
(take (list-tail cvt 5) 4)))
(cond
[(and (or (not descent)
(zero? descent))
(or (not padding)
(andmap zero? padding)))
e]
[else
;; Descent and padding:
(define-values (left top right bottom) (apply values padding))
`(,(car e) ,(cons `[style ,(format "vertical-align: ~apx; margin: ~apx ~apx ~apx ~apx;"
(- (- descent bottom))
(- top) (- right)
(- bottom) (- left))]
(cadr e))
,@(cddr e))]))
(define/private (render-plain-content e part ri)
(define (attribs) (content-attribs e))
(let* ([properties (let ([s (content-style e)])
@ -1843,7 +1900,7 @@
(super collect-part d parent ci number sub-init-number)))
(super collect-part d parent ci number sub-init-number)))))
(define/override (render ds fns ri)
(define/override (render-top ds fns ri)
(map (lambda (d fn)
(when (report-output?)
(printf " [Output to ~a/index.html]\n" fn))

View File

@ -43,13 +43,26 @@
(define-runtime-path skull-tex "scribble-skull.tex")
(define skull-style (make-style #f (list (tex-addition skull-tex))))
(define (render-mixin % #:convert-as-ps-not-pdf? [convert-as-ps-not-pdf? #f])
(define (render-mixin % #:image-mode [image-mode #f])
(class %
(inherit-field prefix-file style-file style-extra-files)
(super-new)
(inherit-field prefix-file style-file style-extra-files image-preferences)
(define/override (current-render-mode)
'(latex))
(inherit sort-image-requests)
(define image-reqs
(sort-image-requests (cond
[(eq? image-mode 'pdf)
'(pdf-bytes png@2x-bytes png-bytes)]
[(eq? image-mode 'ps)
'(eps-bytes)]
[else
'(pdf-bytes png@2x-bytes png-bytes eps-bytes)])
image-preferences))
(define/override (get-suffix) #".tex")
(inherit render-block
@ -353,30 +366,47 @@
(image-element-scale e) fn))]
[(and (convertible? e)
(not (disable-images))
(let ([ftag (lambda (v suffix) (and v (list v suffix)))]
[xlist (lambda (v) (and v (list v #f #f #f #f)))])
(if convert-as-ps-not-pdf?
(or (ftag (convert e 'eps-bytes+bounds) ".ps")
(ftag (xlist (convert e 'eps-bytes)) ".ps")
(ftag (xlist (convert e 'png-bytes)) ".png"))
(or (ftag (convert e 'pdf-bytes+bounds) ".pdf")
(ftag (xlist (convert e 'pdf-bytes)) ".pdf")
(ftag (xlist (convert e 'eps-bytes)) ".ps")
(ftag (xlist (convert e 'png-bytes)) ".png")))))
(let ([ftag (lambda (v suffix [scale 1]) (and v (list v suffix scale)))]
[xxlist (lambda (v) (and v (list v #f #f #f #f #f #f #f #f)))]
[xlist (lambda (v) (and v (append v (list 0 0 0 0))))])
(for/or ([req (in-list image-reqs)])
(case req
[(eps-bytes)
(or (ftag (convert e 'eps-bytes+bounds8) ".ps")
(ftag (xlist (convert e 'eps-bytes+bounds)) ".ps")
(ftag (xxlist (convert e 'eps-bytes)) ".ps"))]
[(pdf-bytes)
(or (ftag (convert e 'pdf-bytes+bounds8) ".pdf")
(ftag (xlist (convert e 'pdf-bytes+bounds)) ".pdf")
(ftag (xxlist (convert e 'pdf-bytes)) ".pdf"))]
[(png@2x-bytes)
(or (ftag (convert e 'png@2x-bytes+bounds8) ".png" 0.5)
(ftag (xxlist (convert e 'png@2x-bytes)) ".png" 0.5))]
[(png-bytes)
(or (ftag (convert e 'png-bytes+bounds8) ".png")
(ftag (xxlist (convert e 'png-bytes)) ".png"))]))))
=> (lambda (bstr+info+suffix)
(check-render)
(let* ([bstr (list-ref (list-ref bstr+info+suffix 0) 0)]
[suffix (list-ref bstr+info+suffix 1)]
[width (list-ref (list-ref bstr+info+suffix 0) 1)]
[scale (list-ref bstr+info+suffix 2)]
[height (list-ref (list-ref bstr+info+suffix 0) 2)]
[pad-left (or (list-ref (list-ref bstr+info+suffix 0) 5) 0)]
[pad-top (or (list-ref (list-ref bstr+info+suffix 0) 6) 0)]
[pad-right (or (list-ref (list-ref bstr+info+suffix 0) 7) 0)]
[pad-bottom (or (list-ref (list-ref bstr+info+suffix 0) 8) 0)]
[descent (and height
(+ (list-ref (list-ref bstr+info+suffix 0) 3)
(- (ceiling height) height)))]
(- (+ (list-ref (list-ref bstr+info+suffix 0) 3)
(- (ceiling height) height))
pad-bottom))]
[width (- (list-ref (list-ref bstr+info+suffix 0) 1) pad-left pad-right)]
[fn (install-file (format "pict~a" suffix) bstr)])
(if descent
(printf "\\raisebox{-~abp}{\\makebox[~abp][l]{\\includegraphics{~a}}}"
(printf "\\raisebox{-~abp}{\\makebox[~abp][l]{\\includegraphics[~atrim=~a ~a ~a ~a]{~a}}}"
descent
width
width
(if (= scale 1) "" (format "scale=~a," scale))
(/ pad-left scale) (/ pad-bottom scale) (/ pad-right scale) (/ pad-top scale)
fn)
(printf "\\includegraphics{~a}" fn))))]
[else
@ -1264,8 +1294,4 @@
(make-toc-paragraph plain null))
(define/override (local-table-of-contents part ri style)
(make-paragraph plain null))
;; ----------------------------------------
(super-new)))
(make-paragraph plain null))))

View File

@ -8,10 +8,10 @@
(define render-mixin
(make-indirect-renderer-mixin
latex:render-mixin #".tex" #".pdf"
(λ (%) (latex:render-mixin % #:image-mode 'pdf)) #".tex" #".pdf"
run-pdflatex))
(define dvi-render-mixin
(make-indirect-renderer-mixin
(λ (%) (latex:render-mixin % #:convert-as-ps-not-pdf? #t)) #".tex" #".pdf"
(λ (%) (latex:render-mixin % #:image-mode 'ps)) #".tex" #".pdf"
run-dvipdf-latex))

View File

@ -45,11 +45,21 @@
file)]))
(when via-dvipdf?
(define dvi-file (path-replace-suffix file #".dvi"))
(define ps-file (path-replace-suffix file #".ps"))
(unless (file-exists? dvi-file) (err "didn't find .dvi file"))
(define dvipdf (get-latex-binary "dvipdf"))
(notify "running dvipdf on ~a" dvi-file)
(unless (parameterize ([current-output-port (open-output-nowhere)])
(system* dvipdf dvi-file))
(define dvips (get-latex-binary "dvips"))
(define pstopdf (get-latex-binary "pstopdf"))
(notify "running dvips on ~a" dvi-file)
(define stderr (open-output-bytes))
(unless (parameterize ([current-output-port (open-output-nowhere)]
[current-error-port stderr])
(system* dvips dvi-file))
(displayln (get-output-bytes stderr))
(err "got error exit code"))
(unless (parameterize ([current-output-port (open-output-nowhere)]
[current-error-port stderr])
(system* pstopdf ps-file))
(displayln (get-output-bytes stderr))
(err "got error exit code")))
(path-replace-suffix file #".pdf"))

View File

@ -19,6 +19,7 @@
#:style-file (or/c #f path-string?)
#:style-extra-files (listof path-string?)
#:extra-files (listof path-string?)
#:image-preferences (listof (or/c 'ps 'pdf 'png 'svg 'gif))
#:redirect (or/c #f string?)
#:redirect-main (or/c #f string?)
#:directory-depth exact-nonnegative-integer?
@ -38,6 +39,7 @@
#:style-file [style-file #f]
#:style-extra-files [style-extra-files null]
#:extra-files [extra-files null]
#:image-preferences [image-preferences null]
#:redirect [redirect #f]
#:redirect-main [redirect-main #f]
#:directory-depth [directory-depth 0]
@ -53,6 +55,7 @@
[style-file style-file]
[style-extra-files style-extra-files]
[extra-files extra-files]
[image-preferences image-preferences]
[helper-file-prefix helper-file-prefix])])
(when redirect
(send renderer set-external-tag-path redirect))

View File

@ -31,6 +31,7 @@
(define current-quiet (make-parameter #f))
(define helper-file-prefix (make-parameter #f))
(define doc-command-line-arguments (make-parameter null))
(define current-image-prefs (make-parameter null)) ; reverse order
(define (read-one str)
(let ([i (open-input-string str)])
@ -65,7 +66,7 @@
[("--pdf") "generate PDF-format output (via PDFLaTeX)"
(current-html #f)
(current-render-mixin pdf:render-mixin)]
[("--dvipdf") "generate PDF-format output (via LaTeX and DVIPDF)"
[("--dvipdf") "generate PDF-format output (via LaTeX, dvips, and pstopdf)"
(current-html #f)
(current-render-mixin pdf:dvi-render-mixin)]
[("--latex-section") n "generate LaTeX-format output for section depth <n>"
@ -88,6 +89,12 @@
[("--dest-base") prefix "start support-file names with <prefix>"
(helper-file-prefix prefix)]
#:multi
[("++convert") fmt ("prefer image conversion to <fmt> (in given order)"
" <fmt> as one of: ps pdf svg png gif")
(define sym (string->symbol fmt))
(unless (member sym '(ps pdf svg png gif))
(raise-user-error 'scribble "bad format for ++convert: ~s" fmt))
(current-image-prefs (cons sym (current-image-prefs)))]
[("++style") file "add given .css/.tex file after others"
(current-style-extra-files (cons file (current-style-extra-files)))]
#:once-each
@ -152,6 +159,7 @@
files)
#:dest-dir (current-dest-directory)
#:render-mixin (current-render-mixin)
#:image-preferences (reverse (current-image-prefs))
#:prefix-file (current-prefix-file)
#:style-file (current-style-file)
#:style-extra-files (reverse (current-style-extra-files))