#!r6rs ;; Copyright 2009 Derick Eddington. My MIT-style license is in the file named ;; LICENSE from the original collection this file is distributed with. (library (yuni lib ssax private to-html) (export SXML->HTML string->goodHTML enattr entag) (import (rnrs) (yuni lib ssax tree-trans) (yuni lib ssax private output) (yuni lib ssax private util)) ; HTML Authoring in SXML ; ; The present file defines and demonstrates a function SXML->HTML, the ; most generic transformation of SXML into the corresponding HTML ; document. The SXML tree is traversed post-order and converted into ; another tree, which, written in a depth-first fashion, results in a ; HTML document. The function SXML->HTML can generate an arbitrary ; HTML markup, for any existing or yet to be introduced HTML ; tag. Furthermore, the function supports one higher-level tag, ; 'html:begin'. As the source code below indicates, SXML->HTML can be ; trivially extended to support other higher-level tags. ; ; The proper HTML markup is being created by a set of node ; handlers. An iterator 'post-order' executes these functions while it ; traverses an SXML tree. ; ; Each node handler takes a tag (the head of an SXML node) and the ; list of children nodes, if any. A handler returns a fragment or a ; list of HTML fragments -- which become arguments to a handler of a ; parent SXML node. A function SRV:send-reply takes the resulting ; tree of fragments and writes out the fragments in a depth-first ; order. The output is an HTML document that corresponds to the ; original SXML tree. ; ; This pretty-printing operation makes it possible to author and ; compose HTML documents in their SXML form. SXML is more concise and ; expressive than a raw markup language. SXML representing regular ; Scheme code can be entered in any Scheme-sensitive editor. SXML as a ; data structure -- a list -- can likewise be composed as a literal or ; quasi-literal expression. Furthermore, SXML can be produced by regular ; Scheme functions, which may make authoring more succinct, advanced, ; and less tedious, as the code below illustrates. ; ; IMPORT ; A prelude appropriate for your Scheme system ; (myenv-bigloo.scm, myenv-mit.scm, etc.) ; util.scm for make-char-quotator ; SXML-tree-trans.scm for post-order ; ; EXPORT ; SXML->HTML enattr entag string->goodHTML ; ; All these files are available in the same directory as this file. ; See vSXML-to-HTML.scm for the validation code, which also ; serves as usage examples. ; ; See http://pobox.com/~oleg/ftp/Scheme/xml.html#XML-authoring ; for more examples and explanation. ; ; $Id: SXML-to-HTML.scm,v 1.5 2003/04/25 19:17:37 oleg Exp $ ; The following procedure is the most generic transformation of SXML ; into the corresponding HTML document. The SXML tree is traversed ; post-oder (depth-first) and transformed into another tree, which, ; written in a depth-first fashion, results in an HTML document. (define (SXML->HTML tree) (SRV:send-reply (pre-post-order tree ; Universal transformation rules. Work for every HTML, ; present and future `((^ ((*DEFAULT* ; local override for attributes . ,(lambda (attr-key . value) (enattr attr-key value)))) . ,(lambda (trigger . value) (cons '^ value))) (*DEFAULT* . ,(lambda (tag . elems) (entag tag elems))) (*TEXT* . ,(lambda (trigger str) (if (string? str) (string->goodHTML str) str))) ; Handle a nontraditional but convenient top-level element: ; (html:begin title ) element (html:begin . ,(lambda (tag title . elems) (list "Content-type: text/html" ; HTTP headers nl nl ; two nl end the headers "" title "" elems "")))) ))) ; The following two functions create the HTML markup for tags and attributes. ; They are being used in the node handlers for the post-order function, see ; above. (define (entag tag elems) (if (and (pair? elems) (pair? (car elems)) (eq? '^ (caar elems))) (list #\newline #\< tag (cdar elems) #\> (and (pair? (cdr elems)) (list (cdr elems) "))) (list #\newline #\< tag #\> (and (pair? elems) (list elems ")) ))) (define (enattr attr-key value) (if (null? value) (list #\space attr-key) (list #\space attr-key "=\"" value #\"))) ; Given a string, check to make sure it does not contain characters ; such as '<' or '&' that require encoding. Return either the original ; string, or a list of string fragments with special characters ; replaced by appropriate character entities. (define string->goodHTML (make-char-quotator '((#\< . "<") (#\> . ">") (#\& . "&") (#\" . """)))) )