[pageheader "expander"] [section SYNOPSIS]
package require expander 1.0
[section DESCRIPTION] The Tcl "subst" command is often used to support a kind of template processing. Given a string with embedded variables or function calls, "subst" will interpolate the variable and function values, returning the new string:
[listing] [tclsh {set greeting "Howdy"}] [tclsh {proc place {} {return "World"}}] [tclsh {subst {$greeting, [place]!}}] % [/listing] By defining a suitable set of Tcl commands, "subst" can be used to implement a markup language similar to HTML.
The "subst" command is efficient, but it has three drawbacks for this kind of template processing:
To begin, create an expander object:
[listing] [tclsh {package require textutil::expander}] [tclsh {::textutil::expander myexp}] % [/listing] The created "::myexp" object can be used to expand text strings containing embedded Tcl commands. By default, embedded commands are delimited by square brackets. Note that expander doesn't attempt to interpolate variables, since variables can be referenced by embedded commands:
[listing] [tclsh {set greeting "Howdy"}] [tclsh {proc place {} {return "World"}}] [tclsh {::myexp expand {[set greeting], [place]!}}] % [/listing] [subsection "Embedding Macros"] An expander macro is simply a Tcl script embedded within a text string. Expander evaluates the script in the global context, and replaces it with its result string. For example, [listing] [tclsh {set greetings {Howdy Hi "What's up"}}] [tclsh {::myexp expand {There are many ways to say "Hello, World!": [set result {} foreach greeting $greetings { append result "$greeting, World!\n" } set result] And that's just a small sample!}}] % [/listing] [subsection "Writing Macro Commands"] More typically, "macro commands" are used to create a markup language. A macro command is just a Tcl command that returns an output string. For example, expand can be used to implement a generic document markup language that can be retargeted to HTML or any other output format: [listing] [tclsh {proc bold {} {return ""}}] [tclsh {proc /bold {} {return ""}}] [tclsh {::myexp expand {Some of this text is in [bold]boldface[/bold]}}] % [/listing] The above definition of "bold" and "/bold" returns HTML, but such commands can be as complicated as needed; they could, for example, decide what to return based on the desired output format.
[subsection "Changing the Expansion Brackets"] By default, embedded macros are enclosed in square brackets, "[lb]" and "[rb]". If square brackets need to be included in the output, the input can contain the [command lb] and [command rb] commands. Alternatively, or if square brackets are objectionable for some other reason, the macro expansion brackets can be changed to any pair of non-empty strings.
The [command setbrackets] command changes the brackets permanently. For example, you can write pseudo-html by change them to "<" and ">":
[listing]
[tclsh {::myexp setbrackets < >}]
[tclsh {::myexp expand {
[listing]
[tclsh {::myexp setbrackets "\[" "\]"}]
[tclsh {::myexp expand {
For example, the following code "evaluates" each macro by returning the macro text itself.
[listing] proc identity {macro} {return $macro} ::myexp evalcmd identity [/listing] [subsection "Using the Context Stack"] Often it's desirable to define a pair of macros which operate in some way on the plain text between them. Consider a set of macros for adding footnotes to a web page: one could have implement something like this:
[listing]
Dr. Pangloss, however, thinks that this is the best of all
possible worlds.[lb]footnote "See Candide, by Voltaire"[rb]
[/listing]
The footnote
macro would, presumably, assign a number to
this footnote and save the text to be formatted later on. However,
this solution is ugly if the footnote text is long or should contain
additional markup. Consider the following instead:
[listing]
Dr. Pangloss, however, thinks that this is the best of all
possible worlds.[lb]footnote[rb]See [lb]bookTitle "Candide"[rb], by
[lb]authorsName "Voltaire"[rb], for more information.[lb]/footnote[rb]
[/listing]
Here the footnote text is contained between footnote
and
/footnote
macros, continues onto a second line, and
contains several macros of its own. This is both clearer and more
flexible; however, with the features presented so far there's no easy
way to do it. That's the purpose of the context stack.
All macro expansion takes place in a particular context.
Here, the footnote
macro pushes a new
context onto the context stack. Then, all expanded text gets placed
in that new context. /footnote
retrieves it by popping
the context. Here's a skeleton implementation of these two macros:
[listing] proc footnote {} { ::myexp cpush footnote } proc /footnote {} { set footnoteText [lb]::myexp cpop footnote[rb] # Save the footnote text, and return an appropriate footnote # number and link. } [/listing] The [command cpush] command pushes a new context onto the stack; the argument is the context's name. It can be any string, but would typically be the name of the macro itself. Then, [command cpop] verifies that the current context has the expected name, pops it off of the stack, and returns the accumulated text.
Expand provides several other tools related to the context stack. Suppose the first macro in a context pair takes arguments or computes values which the second macro in the pair needs. After calling [command cpush], the first macro can define one or more context variables; the second macro can retrieve their values any time before calling [command cpop]. For example, suppose the document must specify the footnote number explicitly:
[listing] proc footnote {footnoteNumber} { ::myexp cpush footnote ::myexp csave num $footnoteNumber # Return an appropriate link } proc /footnote {} { set footnoteNumber [lb]::myexp cget num[rb] set footnoteText [lb]::myexp cpop footnote[rb] # Save the footnote text and its footnoteNumber for future # output. } [/listing] At times, it might be desirable to define macros that are valid only within a particular context pair; such macros should verify that they are only called within the correct context using either [command cis] or [command cname].
[section "TCL COMMANDS"] The package defines the following Tcl commands:
If the error mode is "fail", the error propagates normally and can be caught or ignored by the application.
If the error mode is "error", the macro expands into a detailed error message, and expansion continues.
If the error mode is "macro", the macro expands to itself; that is, it is passed along to the output unchanged.
If the error mode is "nothing", the macro expands to the empty string, and is effectively ignored.
If brackets is given, it must be a list of two strings; the items will be used as the left and right macro expansion bracket sequences for this expansion only.
Note that the combination of textcmd plaintext is run through the evalcmd for the actual evaluation. In other words, the textcmd is treated as a special macro implicitly surrounding all plain text in the template.
[copyright 2001 "William H. Duquette"]