#lang racket/base (require scribble/manual scribble/core scribble/html-properties scribble/latex-properties scriblib/render-cond racket/runtime-path setup/collects "katex-convert-unicode.rkt" "mathjax-convert-unicode.rkt" racket/list (only-in xml cdata) (only-in racket/match match) (only-in racket/system process) (only-in racket/port port->string) (for-syntax racket/base)) (provide $ $$ $-html-handler $$-html-handler $-katex $$-katex $-mathjax $$-mathjax use-katex use-mathjax with-html5 use-external-katex use-external-mathjax) (define-syntax (if-version≥6.12 stx) (syntax-case stx () [(_ . rest) (if (and (not (regexp-match #px"^6\\.11\\.0\\.900$" (version))) (or (regexp-match #px"^6(\\.([0123456789]|10|11)(\\..*|)|)$" (version)) (regexp-match #px"^[123245]\\..*$" (version)))) #'(begin) #'(begin . rest))])) (if-version≥6.12 (provide $-tex2svg $$-tex2svg use-tex2svg current-tex2svg-path)) (define use-external-mathjax (make-parameter #f)) (define use-external-katex (make-parameter #f)) ;; KaTeX does not work well with the HTML 4.01 Transitional loose DTD, ;; so we define a style modifier which replaces the prefix for HTML rendering. (define (with-html5 doc-style) (define has-html-defaults? (memf html-defaults? (style-properties doc-style))) (define new-properties (if has-html-defaults? (map (λ (s) (if (html-defaults? s) (html-defaults (path->collects-relative (collection-file-path "html5-prefix.html" "scribble-math")) (html-defaults-style-path s) (html-defaults-extra-files s)) s)) (style-properties doc-style)) (cons (html-defaults (path->collects-relative (collection-file-path "html5-prefix.html" "scribble-math")) #f '())))) (style (style-name doc-style) new-properties)) ;; Other possible sources for MathJax: ;"http://cdn.mathjax.org/mathjax/latest/MathJax.js?config=TeX-AMS-MML_HTMLorMML" ;"http://c328740.r40.cf1.rackcdn.com/mathjax/latest/MathJax.js?config=default" ;"http://cdn.mathjax.org/mathjax/latest/MathJax.js?config=TeX-AMS-HTML" (define-runtime-path mathjax-dir "MathJax") (define-runtime-path katex-dir "katex") #| (define mathjax-dir (path->collects-relative (collection-file-path "MathJax" "scribble-math"))) |# ;; take into account last scroll event, to avoid locking up the page ;; TODO: should actually pause the whole MathJax queue, as it still locks ;; up the browser between "processing Math" and "rendering math". ;; + set the #MathJax_Message CSS to opacity: 0.5. (define gradually-rename-texMath-to-texMathX-js #< 1000) { if (e.id == "") { e.id = "texMathElement" + pos; } e[pos++].className = "texMathX"; MathJax.Hub.Queue(["Typeset",MathJax.Hub,e.id]); MathJax.Hub.Queue(["next",o]); } else { window.setTimeout(process, 100); } } else { window.removeEventListener("scroll", scrollEventHandler); } }; o.next = function() { window.setTimeout(process, 100); } process(); })(); EOJS ) (define (load-script-string src [async-defer #f]) (string-append #<');})(); EOJS )) (define (load-style-string src) (string-append #<'); })(); EOJS )) (define load-mathjax-code (string->bytes/utf-8 ;; To avoid the need to alter the MathJax configuration, add: ;; (load-script-string (or (use-external-mathjax) "MathJax/MathJax.js?config=default")))) #;(define load-mathjax-code (string->bytes/utf-8 (string-append (or (use-external-mathjax) "MathJax/MathJax.js?config=default") #<bytes/utf-8 (string-append (load-style-string (if (use-external-katex) (cadr (use-external-katex)) "katex/katex.min.css")) (load-script-string (if (use-external-katex) (car (use-external-katex)) "katex/katex.min.js")) #<bytes/utf-8 #<string stdout)]) (error 'tex2svg "Cannot find tex2svg in path or common places; set path manually with current-tex2svg-path."))))) (define ($-tex2svg strs) (elem #:style (style #f (list (xexpr-property (cdata #f #f (tex2svg #:inline #t (flatten strs))) (cdata #f #f ""))))))) (define ($$-mathjax strs) (elem #:style math-display-style-mathjax strs)) (define ($$-katex strs) (elem #:style math-display-style-katex (map (λ (s) (katex-convert-unicode s #t)) (flatten strs)))) (if-version≥6.12 (define ($$-tex2svg strs) (elem #:style (style #f (list (xexpr-property (cdata #f #f (tex2svg (flatten strs))) (cdata #f #f ""))))))) (define $-html-handler (make-parameter $-katex)) (define $$-html-handler (make-parameter $$-katex)) (define (use-katex) ($-html-handler $-katex) ($$-html-handler $$-katex) (void)) (define (use-mathjax) ($-html-handler $-mathjax) ($$-html-handler $$-mathjax) (void)) (if-version≥6.12 (define (use-tex2svg) ($-html-handler $-tex2svg) ($$-html-handler $$-tex2svg) (void))) (define ($ . strs) (let ([$- ($-html-handler)]) (cond-element [html ($- strs)] [latex (elem #:style math-inline-style-latex strs)] ;; TODO: use a unicode representation of math, e.g. x^2 becomes x² [else strs]))) (define ($$ #:latex-style [latex-style math-display-style-latex] . strs) (let ([$$- ($$-html-handler)]) (cond-element [html ($$- strs)] [latex (elem #:style latex-style strs)] ;; TODO: use a spatial representation of display math, e.g. ;; \sum_{i=0}^n x_i^2 ;; becomes: ;; n ;; ─── ;; ╲ 2 ;; 〉 x ;; ╱ i ;; ─── ;; i=0 ;; Or use a spatial unicode representation, so that the above becomes: ;; n ;; ∑ xᵢ² ;; i=0 [else strs])))