From d893680ac26bdf13e60f6fef9966632aaf477ddc Mon Sep 17 00:00:00 2001 From: Matthew Flatt Date: Thu, 24 May 2007 02:20:28 +0000 Subject: [PATCH] new documentation source moved to the trunk collects svn: r6249 original commit: 858c8024efa41ebfdb5f05383c70be5412448570 --- collects/scribblings/scribble/basic.scrbl | 151 ++++++ collects/scribblings/scribble/decode.scrbl | 111 ++++ collects/scribblings/scribble/doclang.scrbl | 18 + collects/scribblings/scribble/docreader.scrbl | 13 + collects/scribblings/scribble/eval.scrbl | 56 ++ collects/scribblings/scribble/manual.scrbl | 328 ++++++++++++ collects/scribblings/scribble/reader.scrbl | 499 ++++++++++++++++++ collects/scribblings/scribble/renderer.scrbl | 31 ++ collects/scribblings/scribble/scribble.scrbl | 90 ++++ collects/scribblings/scribble/struct.scrbl | 212 ++++++++ collects/scribblings/scribble/utils.ss | 44 ++ 11 files changed, 1553 insertions(+) create mode 100644 collects/scribblings/scribble/basic.scrbl create mode 100644 collects/scribblings/scribble/decode.scrbl create mode 100644 collects/scribblings/scribble/doclang.scrbl create mode 100644 collects/scribblings/scribble/docreader.scrbl create mode 100644 collects/scribblings/scribble/eval.scrbl create mode 100644 collects/scribblings/scribble/manual.scrbl create mode 100644 collects/scribblings/scribble/reader.scrbl create mode 100644 collects/scribblings/scribble/renderer.scrbl create mode 100644 collects/scribblings/scribble/scribble.scrbl create mode 100644 collects/scribblings/scribble/struct.scrbl create mode 100644 collects/scribblings/scribble/utils.ss diff --git a/collects/scribblings/scribble/basic.scrbl b/collects/scribblings/scribble/basic.scrbl new file mode 100644 index 00000000..f89f2691 --- /dev/null +++ b/collects/scribblings/scribble/basic.scrbl @@ -0,0 +1,151 @@ +#reader"../docreader.ss" +@require["../manual.ss"] +@require["utils.ss"] +@require-for-syntax[mzscheme] + +@define-syntax[def-title-like + (syntax-rules () + [(_ id result/c x ...) (defproc (id [#:tag tag (or/c false/c string?) #f] + [pre-content any/c] ...0) + result/c + x ...)])] + +@define-syntax[def-elem-proc + (syntax-rules () + [(_ id x ...) (defproc (id [pre-content any/c] ...0) + element? + x ...)])] +@define-syntax[def-style-proc + (syntax-rules () + [(_ id) @def-elem-proc[id]{Like @scheme[elem], but with style @scheme['id]}])] + +@title[#:tag "basic"]{Basic Document Forms} + +The @file{basic.ss} libraryprovides functions and forms that can be +used from code written either in Scheme or with @elem["@"] +expressions. For example, the @scheme[title] and @scheme[italic] +functions might be called from Scheme as + +@schemeblock[ +(title #:tag "how-to" "How to Design " (italic "Great") " Programs") +] + +or with an @elem["@"] expression as + +@verbatim[ +#<string] on the parsed @scheme[pre-content] list. + +} + + +@section{Tables of Contents} + +@defproc[(table-of-contents) delayed-flow-element?]{ + Returns a flow element that expands to a table of contents for the + enclosing section. For LaTeX output, however, the table of contents + currently spans the entire enclosing document. +} diff --git a/collects/scribblings/scribble/decode.scrbl b/collects/scribblings/scribble/decode.scrbl new file mode 100644 index 00000000..1e8e2c8e --- /dev/null +++ b/collects/scribblings/scribble/decode.scrbl @@ -0,0 +1,111 @@ +#reader"../docreader.ss" +@require["../manual.ss"] +@require["utils.ss"] + +@title[#:tag "decode"]{Text Decoder} + +The @file{decode.ss} library helps you write document content in a +natural way---more like plain text, except for @elem["@"] escapes. +Roughly, it processes a stream of strings to produces instances of the +@file{struct.ss} datatypes (see @secref["struct"]). + +At the flow level, decoding recognizes a blank line as a paragraph +separator. At the paragraph-content level, decoding makes just a few +special text conversions: + +@itemize{ + + @item{@litchar{---}: converted to @scheme['mdash], which the HTML render + outputs as an en-dash surrounded by space (so don't put spaces around + @litchar{---} in a document)} + + @item{@litchar{--}: converted to @scheme['ndash]} + + @item{@litchar{``}: converted to @scheme['ldquo], which is fancy open quotes: ``} + + @item{@litchar{''}: converted to @scheme['rdquo], which is fancy closing quotes: ''} + + @item{@litchar{'}: converted to @scheme['rsquo], which is a fancy apostrophe: '} + +} + +@defproc[(decode [lst list?]) part?]{ + +Decodes a document, producing a part. In @scheme[lst], instances of +@scheme[splice] are inlined into the list. An instance of +@scheme[title-decl] supplies the title for the part. Instances of +@scheme[part-start] at level 0 trigger sub-part parsing. Instances of +@scheme[section] trigger are used as-is as subsections, and instances +of @scheme[paragraph] and other flow-element datatypes are used as-is +in the enclosing flow. + +} + +@defproc[(decode-part [lst list?] + [tag string?] + [title (or/c false/c list?)] + [depth excat-nonnegative-integer?]) + part?]{ + +Like @scheme[decode], but given a tag for the section, a title (if +@scheme[#f], then a @scheme[title-decl] instance is used if found), +and a depth for @scheme[part-start]s to trigger sub-part parsing. + +} + +@defproc[(decode-flow [lst list?]) (listof flow-element?)]{ + +Decodes a flow. A sequence of two or more newlines separated only by +whitespace counts is parsed as a paragraph separator. In @scheme[lst], +instances of @scheme[splice] are inlined into the list. Instances of +@scheme[paragraph] and other flow-element datatypes are used as-is in +the enclosing flow. + +} + +@defproc[(decode-paragraph [lst list?]) paragraph?]{ + +Decodes a paragraph. + +} + +@defproc[(decode-content [lst list?]) list?]{ + +Decodes a sequence of elements. + +} + +@defproc[(decode-string [s string?]) list?]{ + +Decodes a single string to produce a list of elements. + +} + +@defproc[(whitespace? [s string?]) boolean?]{ + +Returns @scheme[#t] if @scheme[s] contains only whitespace, @scheme[#f] +otherwise. + +} + +@defstruct[title-decl ([tag any/c] + [content list?])]{ + +See @scheme[decode] and @scheme[decode-part]. + +} + +@defstruct[part-start ([depth integer?] + [tag (or/c false/c string?)] + [title list?])]{ + +See @scheme[decode] and @scheme[decode-part]. + +} + +@defstruct[splice ([run list?])]{ + +See @scheme[decode], @scheme[decode-part], and @scheme[decode-flow]. + +} + diff --git a/collects/scribblings/scribble/doclang.scrbl b/collects/scribblings/scribble/doclang.scrbl new file mode 100644 index 00000000..53ee0ed6 --- /dev/null +++ b/collects/scribblings/scribble/doclang.scrbl @@ -0,0 +1,18 @@ +#reader"../docreader.ss" +@require["../manual.ss"] +@require["utils.ss"] + +@title[#:tag "doclang"]{Document Module Languegs} + +The @file{doclang.ss} module is suitable for use as a module +language. It provides everything from @scheme[mzscheme], except that +it replaces the @scheme[#%module-begin] form. + +The @file{doclang.ss} @scheme[#%module-begin] essentially packages the +body of the module into a call to @scheme[decode], binds the result to +@scheme[doc], and exports @scheme[doc]. + +Any module-level form other than an expression (e.g., a +@scheme[require] or @scheme[define]) is remains at the top level, and +the @scheme[doc] binding is put at the end of the module. As usual, a +module-top-level @scheme[begin] slices into the module top level. diff --git a/collects/scribblings/scribble/docreader.scrbl b/collects/scribblings/scribble/docreader.scrbl new file mode 100644 index 00000000..38064814 --- /dev/null +++ b/collects/scribblings/scribble/docreader.scrbl @@ -0,0 +1,13 @@ +#reader"../docreader.ss" +@require["../manual.ss"] +@require["../bnf.ss"] +@require["utils.ss"] + +@title[#:tag "docreader"]{Document Reader} + +The @file{docreader.ss} module is suitable for use with +@schemefont{#reader} at the beginning of a file. It reads the entire +file with @scheme[read-inside-syntax] from Scribble's +@file{reader.ss}, and then wraps the result with @scheme[(module #, +@nonterm{name} (lib "doclang.ss" "scribble") ...)], where +@nonterm{name} is derived from the enclosing file's name. diff --git a/collects/scribblings/scribble/eval.scrbl b/collects/scribblings/scribble/eval.scrbl new file mode 100644 index 00000000..3feff19b --- /dev/null +++ b/collects/scribblings/scribble/eval.scrbl @@ -0,0 +1,56 @@ +#reader"../docreader.ss" +@require["../manual.ss"] +@require["utils.ss"] + +@title[#:tag "eval"]{Evaluation and Examples} + +The @file{eval.ss} library provides utilities for evaluating code at +document-build time and incorporating the results in the document, +especially to show example uses of defined procedures and syntax. + +@defform[(interaction datum ...)]{Like @scheme[schemeinput], except +that the result for each input @scheme[datum] is shown on the next +line. The result is determined by evaluating the quoted form of the +datum. + +Uses of @scheme[code:comment] and @schemeidfont{code:blank} are +stipped from each @scheme[datum] before evaluation. + +If a datum has the form @scheme[(#,(scheme code:line) #,(svar datum) +(#,(scheme code:comment) ...))], then only the @svar[datum] is +evaluated. + +If a datum has the form @scheme[(eval:alts #,(svar show-datum) #,(svar +eval-datum))], then @svar[show-datum] is typeset, while +@svar[eval-datum] is evaluated.} + +@defform[(interaction-eval datum)]{Evaluates the quoted form of +each @scheme[datum] via @scheme[do-eval] and returns the empty string.} + +@defform[(interaction-eval-show datum)]{Evaluates the quoted form of +@scheme[datum] and produces an element represeting the printed form of +the result.} + +@defform[(schemeblock+eval datum ...)]{Combines @scheme[schemeblock] +and @scheme[interaction-eval].} + +@defform[(schememod+eval name datum ...)]{Combines @scheme[schememod] +and @scheme[interaction-eval].} + +@defform[(def+int defn-datum expr-datum ...)]{Like +@scheme[interaction], except the the @scheme[defn-datum] is typeset as +for @scheme[schemeblock] (i.e., no prompt) with a line of space +between the definition and the interactions.} + +@defform[(defs+int (defn-datum ...) expr-datum ...)]{Like +@scheme[def+int], but for multiple leading definitions.} + +@defform[(examples datum ...)]{Like @scheme[interaction], but with an +``Examples:'' label prefixed.} + +@defform[(defexamples datum ...)]{Like @scheme[examples], but each +definition using @scheme[define] among the @scheme[datum]s is typeset +without a prompt, and with space after it.} + +@defthing[current-int-namespace parameter?]{A parameter to hold the +namespace used by @scheme[interaction], etc.} diff --git a/collects/scribblings/scribble/manual.scrbl b/collects/scribblings/scribble/manual.scrbl new file mode 100644 index 00000000..221fad0d --- /dev/null +++ b/collects/scribblings/scribble/manual.scrbl @@ -0,0 +1,328 @@ +#reader"../docreader.ss" +@require["../manual.ss"] +@require["utils.ss"] +@require-for-syntax[mzscheme] + +@title[#:tag "manual"]{PLT Manual Forms} + +The @file{manual.ss} module provides all of @file{basic.ss}, and +more... + +@; ------------------------------------------------------------------------ +@section{Typesetting Code} + +@defform[(schemeblock datum ...)]{ + +Typesets the @scheme[datum] sequence as a table of Scheme code inset +by two spaces. The source locations of the @scheme[datum]s determines +the generated layout. For example, + +@schemeblock[ +(schemeblock + (define (loop x) + (loop (not x)))) +] + +produces the output + +@schemeblock[ +(define (loop x) + (loop (not x))) +] + +with the @scheme[(loop (not x))] indented under @scheme[define], +because that's the way it is idented the use of @scheme[schemeblock]. +Furthermore, @scheme[define] is typeset as a keyword (bold and black) +and as a hyperlink to @scheme[define]'s definition in the reference +manual, because this document was built using information about the +MzScheme manual. Similarly, @scheme[not] is a hyperlink to the its +definition in the reference manual. + +Use @scheme[unsyntax] to escape back to an expression that produces an +@scheme[element]. For example, + +@let[([unsyntax #f]) +(schemeblock + (schemeblock + (+ 1 (unsyntax (elem (scheme x) (subscript "2")))))) +] + +produces + +@schemeblock[ +(+ 1 (unsyntax (elem (scheme x) (subscript "2")))) +] + +The @scheme[unsyntax] form is regonized via +@scheme[module-identifier=?], so if you want to typeset code that +includes @scheme[unsyntax], you can simply hide the usual binding: + +@SCHEMEBLOCK[ +(schemeblock + (let ([(UNSYNTAX (scheme unsyntax)) #f]) + (schemeblock + (syntax (+ 1 (unsyntax x)))))) +] + +Or use @scheme[SCHEMEBLOCK], whose escape form is @scheme[UNSYNTAX] +instead of @scheme[unsyntax]. See also @scheme[define-code] from +@file{scheme.ss}. + +A few other escapes are recognized symbolically: + +@itemize{ + + @item{@scheme[(#,(scheme code:line) datum ...)] typesets as the + sequence of @scheme[datum]s (i.e., without the + @scheme[code:line] wrapper.} + + @item{@scheme[(#,(scheme code:comment) content-expr)] typesets as a + comment whose content (i.e., sequence of elements) is produced + by @scheme[content-expr].} + + @item{@schemeidfont{code:blank} typesets as a blank line.} + +} + +} + +@defform[(SCHEMEBLOCK datum ...)]{Like @scheme[schemeblock], but with +the expression escape @scheme[UNSYNTAX] instead of @scheme[unsyntax].} + +@defform[(schemeblock0 datum ...)]{Like @scheme[schemeblock], but +without insetting the code.} + +@defform[(SCHEMEBLOCK0 datum ...)]{Like @scheme[SCHEMEBLOCK], but +without insetting the code.} + +@defform[(schemeinput datum ...)]{Like @scheme[schemeblock], but the +@scheme[datum] are typeset after a prompt representing a REPL.} + +@defform[(schememod lang datum ...)]{Like @scheme[schemeblock], but +the @scheme[datum] are typeset inside a @schemefont{#module}-form +module whose language is @scheme[lang].} + +@defform[(scheme datum ...)]{Like @scheme[schemeblock], but typeset on +a single line and wrapped with its enclosing paragraph, independent of +the formatting of @scheme[datum].} + +@defform[(schemeresult datum ...)]{Like @scheme[scheme], but typeset +as a REPL value (i.e., a single color with no hperlinks).} + +@defform[(schemeid datum ...)]{Like @scheme[scheme], but typeset +as an unbound identifier (i.e., no coloring or hyperlink).} + +@defform[(schememodname datum ...)]{Like @scheme[scheme], but typeset +as a @schemefont{#module} language name.} + +@defproc[(litchar [str string?]) element?]{Typesets @scheme[str] as a +representation of literal text. Use this when you have to talk about +the individual characters in a stream of text, as as when documenting +a reader extension.} + +@defproc[(verbatim [str string?]) flow-element?]{Typesets @scheme[str] +as a table/paragraph in typewriter font with the linebreaks specified +by newline characters in @scheme[str]. ``Here strings'' are often +useful with @scheme[verbatim].} + +@defproc[(schemefont [pre-content any/c] ...0) element?]{Typesets the given +content as uncolored, unhyperlinked Scheme. This procedure is useful +for typesetting thngs like @scheme{#module}, which are not +@scheme[read]able by themselves.} + +@defproc[(schemevalfont [pre-content any/c] ...0) element?]{Like +@scheme[schemefont], but colored as a value.} + +@defproc[(schemeresultfont [pre-content any/c] ...0) element?]{Like +@scheme[schemefont], but colored as a REPL result.} + +@defproc[(schemeidfont [pre-content any/c] ...0) element?]{Like +@scheme[schemefont], but colored as an identifier.} + +@defproc[(schemekeywordfont [pre-content any/c] ...0) element?]{Like +@scheme[schemefont], but colored as a syntactic form name.} + +@defproc[(procedure [pre-content any/c] ...0) element?]{Typesets the given +content as a procedure name in a REPL result (e.g., in typewrite font +with a @schemefont{#} suffix.).} + +@defform[(var datum)]{Typesets @scheme[var] as an identifier that is +an argument or sub-form in a procedure being +documented. Normally, the @scheme[defproc] and @scheme[defform] +arrange for @scheme[scheme] to format such identifiers automatically +in the description of the procedure, but use @scheme[var] if that +cannot work for some reason.} + +@defform[(svar datum)]{Like @scheme[var], but for subform non-terminals +in a form definition.} + +@; ------------------------------------------------------------------------ +@section{Definition Reference} + +@defform[(defproc (identifier arg-spec ...) result-contract-expr-datum pre-flow ...)]{Produces +a sequence of flow elements (encaptured in a @scheme[splice]) to +document a procedure named @scheme[identifier]. The +@scheme[identifier] is registered so that @scheme[scheme]-typeset uses +of the identifier are hyperlinked to this documentation. + +Each @scheme[arg-spec] must have one of the following forms: + +@itemize{ + + @item{@specsubform/inline[(arg-identifier contract-expr-datum)]{--- + an argument whose contract is specified by + @scheme[contract-expr-datum] which is typeset via + @scheme[scheme].}} + + @item{@specsubform/inline[(arg-identifier contract-expr-datum + default-expr)]{ --- like the previous case, but with a default + value. All arguments with a default value must be grouped + together, but they can be in the middle of required + arguments.}} + + @item{@specsubform/inline[(keyword arg-identifier + contract-expr-datum)]{ --- like the first case, but for a + keyword-based argument.}} + + @item{@specsubform/inline[(keyword arg-identifier contract-expr-datum + default-expr)]{ --- like the previous case, but with a default + value.}} + + @item{@scheme[...0] --- any number of the preceding argument + (normally at the end)} + + @item{@scheme[...1] --- one or more of the preceding argument + (normally at the end)} + +} + +The @scheme[result-contract-expr-datum] is typeset via +@scheme[scheme], and it represents a contract on the procedure's +result. + +The @scheme[pre-flow]s list is parsed as a flow that documents the +procedure. In this description, references to @svar[arg-identifier]s +are typeset as procedure arguments. + +The typesetting of all data before the @scheme[pre-flow]s ignores the +source layout.} + + +@defform[(defproc* (((identifier arg-spec ...) result-contract-expr-datum) ...) pre-flow ...)]{Like +@scheme[defproc], but for multiple cases with the same @scheme[identifier]. +} + + +@defform[(defform (identifier . datum) pre-flow ...)]{Produces a +a sequence of flow elements (encaptured in a @scheme[splice]) to +document a syntaic form named by @scheme[identifier]. The +@scheme[identifier] is registered so that @scheme[scheme]-typeset uses +of the identifier are hyperlinked to this documentation. + +The @scheme[pre-flow]s list is parsed as a flow that documents the +procedure. In this description, a reference to any identifier in +@scheme[datum] is typeset as a sub-form non-terminal. + +The typesetting of @scheme[(identifier . datum)] preserves the source +layout, like @scheme[scheme], and unlike @scheme[defproc].} + + +@defform[(specsubform/inline datum pre-flow ...)]{Similar to +@scheme[defform], but without any specific identifier being defined, +without the output format that highlights a definition, and with +@scheme[datum] as an element rather than a table. This form is +intended for use when refining the syntax of a non-terminal used in a +@scheme[defform], @scheme[specsubform], or other +@scheme[specsubform/inline]. For example, it is used in the +documentation for @scheme[defproc] in the itemization of possible +shapes for @svar[arg-spec]. + +The @scheme[pre-flow]s list is parsed as a flow that documents the +procedure. In this description, a reference to any identifier in +@scheme[datum] is typeset as a sub-form non-terminal.} + + +@defform[(specsubform datum pre-flow ...)]{Like +@scheme[specsubform/inline], but the @scheme[datum] is typeset in the +resulting flow as a table instead of an element.} + + +@defform[(defthing identifier contract-expr-datum pre-flow ...)]{Like +@scheme[defproc], but for a non-procedure binding.} + +@defform[(defstruct struct-name ([field-name contract-expr-datum] ...) pre-flow ...)]{Similar +to @scheme[defform], but for a structure definition. + +The @scheme[struct-name] can be either of the following: + +@itemize{ + + @item{@specsubform/inline[identifier]{--- a structure type with no + specified supertype.}} + + @item{@specsubform/inline[(identifier super-identifier)]{ --- a structure + type with indicated supertype.}} + +}} + + +@; ------------------------------------------------------------------------ +@section{Various String Forms} + +@defproc[(defterm [pre-content any/c] ...0) element?]{Typesets the given +content as a defined term (e.g., in italic).} + +@defproc[(onscreen [pre-content any/c] ...0) element?]{ Typesets the given +content as a string that appears in a GUI, such as the name of a +button.} + +@defproc[(menuitem [menu-name string?] [item-name string?]) element?]{ +Typesets the given combination of a GUI's menu and item name.} + +@defproc[(file [pre-content any/c] ...0) element?]{Typesets the given content +as a file name (e.g., in typewriter font and in in quotes).} + +@defproc[(exec [pre-content any/c] ...0) element?]{Typesets the given content +as a command line (e.g., in typewriter font).} + +@; ------------------------------------------------------------------------ +@section{Section Links} + +@defproc[(secref [tag string?]) element?]{Inserts the hyperlinked +title of the section tagged @scheme[tag].} + +@defproc[(seclink [tag string?] [pre-content any/c] ...0) element?]{The content from +@scheme[pre-content] is hyperlinked to the section tagged @scheme[tag].} + +@defproc[(schemelink [id symbol?] [pre-content any/c] ...0) element?]{The content from +@scheme[pre-content] is hyperlinked to the definition of @scheme[id].} + + + +@; ------------------------------------------------------------------------ +@section{Indexing} + +@defproc[(idefterm [pre-content any/c] ...0) element?]{Combines +@scheme[as-index] and @scheme[defterm]. The content normally should be +plurarl, rather than singular.} + +@defproc[(pidefterm [pre-content any/c] ...0) element?]{Like +@scheme[idefterm], but plural: adds an ``s'' on the end of the content +for the index entry.} + +@; ------------------------------------------------------------------------ +@section{Miscellaneous} + +@defthing[PLaneT string?]{@scheme["PLaneT"] (to help make sure you get +the letters in the right case).} + +@defproc[(void-const) any/c]{Returns @scheme["void"], as opposed to +@scheme[(scheme void)]---but we may eventually find a clearer way to +refer to @void-const in prose.} + +@defproc[(centerline [pre-flow any/c] ...0) table?]{Produces a +centered table with the @scheme[pre-flow] parsed by +@scheme[decode-flow].} + +@defproc[(commandline [pre-content any/c] ...0) paragraph?]{Produces a +an inset command-line example (e.g., in typewriter font).} diff --git a/collects/scribblings/scribble/reader.scrbl b/collects/scribblings/scribble/reader.scrbl new file mode 100644 index 00000000..4e1f570d --- /dev/null +++ b/collects/scribblings/scribble/reader.scrbl @@ -0,0 +1,499 @@ +#reader"../docreader.ss" +@require["../manual.ss"] +@require["../bnf.ss"] +@require["utils.ss"] + +@title[#:tag "reader"]{Scribble Reader} + +The Scribble @|at|-reader is designed to be a convenient facility for +using free-form text in Scheme code, where ``@at'' is chosen as one of +the least-used characters in Scheme code. + +You can use the reader via MzScheme's @schemefont{#reader} form: + +@schemeblock[ + #, @schemefont[#<|" +] + +More simply, if you get into too much trouble with special characters +in a body, then it's often a good idea to use the Scheme part, +instead. + +@scribble-examples[ + "@foo[\"}\"]" + "@foo[\"@literally{}\"]" +] + +@; - - - - - - - - - - - - - - - - - - - - - - - - +@subsubsub*section{Quoting in Body Texts} + +To quote braces or @at, precede them with a backslash. Note that this +is an irregular use of backslash quoting! To use @litchar["\\@"] in +your text, simply precede it with a backslash. The general rule is +that to use N backslashes-and-a-special-character, you should precede +it with one extra backslash. Any other use of a backslash (one that +is not followed by more back-slashes and a special character) is +preserved in the text as usual. + +@scribble-examples[ + "@foo{b\\@ar}" + "@foo{b\\\\@ar}" + "@foo{b\\\\\\@ar}" + "@foo{b\\{\\@\\@ar}" + "@foo{b\\ar}" + "@foo{b\\\\ar}" +] + +@; - - - - - - - - - - - - - - - - - - - - - - - - +@subsubsub*section{Newlines and Indentation} + +When indentation is used, all-space indentation string syntaxes are +perpended to the beginning of each line. The rule for adding these +string is: + +@itemize{ + + @item{A spaces-string is added to each line according to its distance from + the leftmost syntax object;} + + @item{The first string is not prepended with indentation if it appears on + the first line of output.} + +} + +@scribble-examples[ +#< '' prompt, which might lead to confusing results.} + +} + +@italic{The following is likely to change.} + +For situations where spaces at the beginning of lines matter (various +verbatim environments), you should begin a line with a @litchar["|"]. +It has no other special meaning -- so to use a @litchar["|"] as the +first character in the text, simply use another before it. + +@scribble-examples[ +#< (define (important . text) @`b{@u{@big{@,@text}}}) + > (important @`p{This is an important announcement! + Read it!}) + (b (u (big (p "This is an important announcement!" "\n" "Read it!")))) +EOS +] diff --git a/collects/scribblings/scribble/renderer.scrbl b/collects/scribblings/scribble/renderer.scrbl new file mode 100644 index 00000000..0d3a19fb --- /dev/null +++ b/collects/scribblings/scribble/renderer.scrbl @@ -0,0 +1,31 @@ +#reader"../docreader.ss" +@require["../manual.ss"] +@require["utils.ss"] + +@title[#:tag "renderer"]{Renderer} + +A renderer is an object that provides two main methods: +@scheme[collect] and @scheme[render]. The first method is called to +collect global information about the document, including information +that spans multiple documents rendered together; the collection pass +tends to be format-independent, and it usually implemented completely +by the base renderer. The latter method generates the actual output, +which is naturally specific to a particular format. + +The @file{base-render.ss} module provides @scheme[render%], which +implements the core of a renderer. The @file{html-renderer.ss}, +@file{latex-renderer.ss}, and @file{text-renderer.ss} modules each +provide @scheme[renderer-mixin] to extend the base. The +@file{html-renderer.ss} module also provides +@scheme[multi-renderer-mixin] to produce multi-file HTML instead +instead of single-file HTML. + +The mixin structure is meant to support document-specific extensions +to the renderers. For example, the @exec{scribble} command-line tool +might, in the future, extract rendering mixins from a document module +(in addition to the document proper). + +See @file{base-render.ss} for more information about the methods of +the renderer. Documents built with higher layers, such as +@file{manual.ss}, generally do not call the render object's methods +directly. diff --git a/collects/scribblings/scribble/scribble.scrbl b/collects/scribblings/scribble/scribble.scrbl new file mode 100644 index 00000000..5eaab854 --- /dev/null +++ b/collects/scribblings/scribble/scribble.scrbl @@ -0,0 +1,90 @@ +#reader"../docreader.ss" +@require["../manual.ss"] +@require["../bnf.ss"] +@require["utils.ss"] + +@title{PLT Scribble} + +The @file{scribble} collection provides libraries that can be used to +create documents from Scheme. + +@table-of-contents[] + +@; ------------------------------------------------------------------------ +@section{Scribble Layers} + +Scribble is made of independently usable parts. For example, the +Scribble reader can be used in any situation that requires lots of +free-form text. You can also skip Scribble's special reader support, +and instead use the document-generation structure directly. + +The layers are: + +@itemize{ + + @item{@file{reader.ss}: a reader that extends the syntax of Scheme + with @at forms for conveniently embedding a mixin of text and + escapes. See @secref["reader"].} + + @item{@file{struct.ss}: a set of document datatypes, which define the + basic layout of a document. See @secref["struct"].} + + @item{@file{base-render.ss} with @file{html-render.ss}, + @file{latex-render.ss}, or @file{text-render.ss}: A base + renderer and mixins that generate documents in various formats + from instances of the @file{struct.ss} datatype. See + @secref["renderer"].} + + @item{@file{decode.ss}: Processes a stream of text, section-start + markers, etc. to produce instances of the @file{struct.ss} + datatype. See @secref["decode"].} + + @item{@file{doclang.ss}: to be used for the initial import of a + module; processes the module top level through + @file{decode.ss}, and otherwise provides all of + @scheme[mzscheme]. See @secref["doclang"].} + + @item{@file{docreader.ss}: a reader that is meant to tbe used to + process an entire file; it essentially combines + @file{reader.ss} with @file{doclang.ss}. See + @secref["docreader"].} + + @item{@file{basic.ss}: a library of basic document operators---such + as @scheme[title], @scheme[section], and @scheme[secref]---for + use with @file{decode.ss} and a renderer. See + @secref["basic"].} + + @item{@file{scheme.ss}: a library of support functions for + typesetting Scheme code.} + + @item{@file{manual.ss}: a library of support functions for writing + PLT Scheme documentation; re-exports @file{basic.ss}. See + @secref["manual"].} + + @item{@file{eval.ss}: a library of support functions for ealuating + code at document-build time, especially for showing + examples. See @secref["eval"].} + + @item{@file{bnf.ss}: a library of support functions for writing + grammars.} + +} + +The @exec{scribble} command-line utility works with a module that +exports a @scheme{struct.ss}-based document, generating output with a +specified renderer. More specifically, the executable installs a +renderer, loads the specified modules and extracts the @scheme[doc] +export of each (which must be an instance of @scheme[section] from +@file{struct.ss}), and renders each. Use @exec{scribble -h} for more +information. + +@; ------------------------------------------------------------------------ +@include-section["reader.scrbl"] +@include-section["struct.scrbl"] +@include-section["renderer.scrbl"] +@include-section["decode.scrbl"] +@include-section["doclang.scrbl"] +@include-section["docreader.scrbl"] +@include-section["basic.scrbl"] +@include-section["manual.scrbl"] +@include-section["eval.scrbl"] diff --git a/collects/scribblings/scribble/struct.scrbl b/collects/scribblings/scribble/struct.scrbl new file mode 100644 index 00000000..9371830b --- /dev/null +++ b/collects/scribblings/scribble/struct.scrbl @@ -0,0 +1,212 @@ +#reader"../docreader.ss" +@require["../manual.ss"] +@require["utils.ss"] + +@title[#:tag "struct"]{Document Structures} + +A single document is reprsented as a @defterm{part}: + +@itemize{ + + @item{A @defterm{part} is an instance of @scheme[part]; it has a + title @defterm{content}, an initial @defterm{flow}, and a list + of subsection @defterm{part}s. After the ``collect'' phase of + rendering, it also has @defterm{collected info}. An + @scheme[unnumbered-part] is the same as a @scheme[part], but it + isn't numbered.} + + @item{A @defterm{flow} is an instance of @scheme[flow]; it has a list + of @defterm{flow element}s.} + + @item{A @defterm{flow element} is either a @defterm{table}, an + @defterm{itemization}, @defterm{paragraph}, or a + @defterm{delayed flow element}. + + @itemize{ + + @item{A @defterm{table} is an instance of @scheme[table]; it has a + list of list of @defterm{flow}s with a particular style.} + + @item{A @defterm{itemization} is an instance of @scheme[itemization]; + it has a list of flows.} + + @item{A @defterm{paragraph} is an instance of @scheme[paragraph]; it + has a list of @defterm{element}s. + + @itemize{ + + @item{An element can be a string, one of a few symbols, an instance of + @scheme[element] (possibly @scheme[link-element], + @scheme[target-element], or + @scheme[index-element]), a @defterm{delayed + element}, or anything else allowed by the current + renderer. + + @itemize{ + + @item{A string element is included in the result + document verbatim.} + + @item{A symbol element is either @scheme['mdash], + @scheme['ndash], @scheme['ldquo], + @scheme['lsquo], @scheme['rsquo], or + @scheme['rarr]; it is drawn as the + corresponding HTML entity.} + + @item{An instance of @scheme[element] has a list of + @defterm{element}s plus a style. The style's + interpretation depends on the rendrer; it can + be one of a few special symbols that are + recognized by all renderers: @scheme['tt], + @scheme['italic], @scheme['bold], + @scheme['sf], @scheme['subscript], + @scheme['superscript], or @scheme['hspace]. + A string corresponds to a CSS class, LaTeX + macro, or something else renderer-specific. + Instances of @scheme[target-url] and + @scheme[image-file] may also be supported.} + + @item{An instance of @scheme[link-element] has a + @defterm{tag} for the target of the link.} + + @item{An instance of @scheme[target-element] has a + @defterm{tag} to be referenced by + @scheme[link-element]s.} + + @item{An instance of @scheme[index-element] has a + @defterm{tag} (as a target), a list of + strings for the keywords (for sorting and + search), and a list of @defterm{element}s to + appear in the end-of-document index.} + + @item{A @defterm{delayed element} is an instance of + @scheme[delayed-element], which has a + procedure that produces a + @defterm{element}. The ``collect'' phase of + rendering ignores delayed flow elements.} + + }}}} + + @item{A @defterm{delayed flow element} is an instance of + @scheme[delayed-flow-element], which has a procedure that + produces a @defterm{flow element}. The ``collect'' phase + of rendering ignores delayed flow elements.} + + }} + + @item{The @defterm{collected info} of a part includes its number, its + parent part (or @scheme[#f]), and information about link + targets and index entries within the part.} + + @item{A @defterm{tag} is eiter a string or a list containing a symbol + and a string.} + +} + +Note that there's no difference between a part and a full document. A +particular source module just as easily defines a subsection +(incoprated via @scheme[include-section]) as a document. + +@defstruct[part ([tag (or/c false/c tag?)] + [title-content (or/c false/c list?)] + [collected-info (or/c false/c collected-info?)] + [flow flow?] + [parts (listof part?)])]{ + +} + + +@defstruct[(unnumbered-part part) ()]{ + +} + +@defstruct[flow ([paragraphs (listof flow-element?)])]{ + +} + +@defstruct[paragraph ([content list?])]{ + +} + + +@defstruct[table ([style any/c] + [flowss (listof (listof flow?))])]{ + +} + + +@defstruct[delayed-flow-element ([render (any/c part? any/c . -> . flow-element?)])]{ + +For the @scheme[render] procedure, the first argument corresponds to +the rendering context, the second to the immediately enclosing +section, and the last argument correspond to global information +(possibly psanning multiple documents). + +} + + +@defstruct[itemization ([flows (listof flow?)])]{ + +} + +@defstruct[element ([style any/c] + [content list?])]{ + +} + +@defstruct[(target-element element) ([tag tag?])]{ + +} + +@defstruct[(link-element element) ([tag any/c] + [complain-if-fail? boolean?])]{ + +} + + +@defstruct[(index-element element) ([tag tag?] + [plain-seq (listof string?)] + [entry-seq list?])]{ + +} + +@defstruct[delayed-element ([render (any/c part? any/c . -> . list?)])]{ + +The @scheme[render] procedure's arguments are the same as for +@scheme[delayed-flow-element]. Unlike @scheme[delayed-flow-element], +the result of the @scheme[render] procedure's argument is remembered +on the first call. Furthemore, the element can be marshelled (e.g., +for an index entry or a section-title entry) only if it has been +rendered first. + +} + +@defstruct[collected-info ([number (listof (or/c false/c integer?))] + [parent (or/c false/c part?)] + [info any/c])]{ + +} + +@defproc[(flow-element? [v any/c]) boolean?]{ + +Returns @scheme[#t] if @scheme[v] is a @scheme[paragraph], +@scheme[table], @scheme[itemization], or +@scheme[delayed-flow-element], @scheme[#f] otherwise. + +} + +@defproc[(tag? [v any/c]) boolean?]{ + +Returns @scheme[#t] if @scheme[v] is acceptable as a link tag, +@scheme[#f], otherwise. Currently, an acceptable tag is either a +string or a list containing a symbol and a string. + +} + +@defproc[(content->string (content list?)) string?]{ + +Converts a list of elements to a single string (essentially +rendering the content as ``plain text''). + +} + diff --git a/collects/scribblings/scribble/utils.ss b/collects/scribblings/scribble/utils.ss new file mode 100644 index 00000000..15795a82 --- /dev/null +++ b/collects/scribblings/scribble/utils.ss @@ -0,0 +1,44 @@ + +(module utils mzscheme + (require "../struct.ss" + "../manual.ss" + (prefix scheme: "../scheme.ss") + (prefix scribble: "../reader.ss") + (lib "string.ss")) + + (provide at + litchar/lines + scribble-examples) + + (define at "@") + + (define (litchar/lines s) + (let ([strs (regexp-split #rx"\n" s)]) + (if (= 1 (length strs)) + (litchar s) + (make-table + #f + (map (lambda (s) + (list (make-flow (list (make-paragraph (list (litchar s))))))) + strs))))) + + (define (as-flow e) + (make-flow (list (if (flow-element? e) + e + (make-paragraph (list e)))))) + + (define spacer (hspace 2)) + + (define (scribble-examples . lines) + (make-table + #f + (map (lambda (line) + (let ([line (if (string? line) + (list (litchar/lines line) + (scheme:to-element (scribble:read (open-input-string line)))) + line)]) + (list (as-flow spacer) + (as-flow (if line (car line) "")) + (as-flow (if line (make-paragraph (list spacer "reads as" spacer)) "")) + (as-flow (if line (cadr line) ""))))) + lines))))