package require expander 1.0
% set greeting "Howdy" Howdy % proc place {} {return "World"} % subst {$greeting, [place]!} Howdy, World! %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:
% package require textutil::expander 1.0 % ::textutil::expander myexp ::myexp %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:
% set greeting "Howdy" Howdy % proc place {} {return "World"} % ::myexp expand {[set greeting], [place]!} Howdy, World! %
% set greetings {Howdy Hi "What's up"} Howdy Hi "What's up" % ::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!} There are many ways to say "Hello, World!": Howdy, World! Hi, World! What's up, World! And that's just a small sample! %
% proc bold {} {return "<b>"} % proc /bold {} {return "</b>"} % ::myexp expand {Some of this text is in [bold]boldface[/bold]} Some of this text is in <b>boldface</b> %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.
lb
and 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 setbrackets
command changes the brackets permanently.
For example, you can write pseudo-html by change them to "<" and ">":
% ::myexp setbrackets < > % ::myexp expand {<bold>This is boldface</bold>} <b>This is boldface</b>Alternatively, you can change the expansion brackets temporarily by passing the desired brackets to the
expand
command:
% ::myexp setbrackets "\[" "\]" % ::myexp expand {<bold>This is boldface</bold>} {< >} <b>This is boldface</b> %
evalcmd
; this allows the application to use a safe
interpreter, for example, or even to evaluated something other than
Tcl code. There is one caveat: to be recognized as valid, a macro
must return 1 when passed to Tcl's "info complete" command.For example, the following code "evaluates" each macro by returning the macro text itself.
proc identity {macro} {return $macro} ::myexp evalcmd identity
Dr. Pangloss, however, thinks that this is the best of all possible worlds.[footnote "See Candide, by Voltaire"]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:
Dr. Pangloss, however, thinks that this is the best of all possible worlds.[footnote]See [bookTitle "Candide"], by [authorsName "Voltaire"], for more information.[/footnote]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:
proc footnote {} { ::myexp cpush footnote } proc /footnote {} { set footnoteText [::myexp cpop footnote] # Save the footnote text, and return an appropriate footnote # number and link. }The
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, 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
cpush
, the first macro can define one or more context
variables; the second macro can retrieve their values any time before
calling cpop
. For example, suppose the document must
specify the footnote number explicitly:
proc footnote {footnoteNumber} { ::myexp cpush footnote ::myexp csave num $footnoteNumber # Return an appropriate link } proc /footnote {} { set footnoteNumber [::myexp cget num] set footnoteText [::myexp cpop footnote] # Save the footnote text and its footnoteNumber for future # output. }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
cis
or cname
.
expander name
cappend text
cget varname
cis cname
cname
cpop cname
cpush cname
cpop
before expansion
ends or an error results.
cset varname value
cvar varname
errmode ?newErrmode?
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.
evalcmd ?newEvalCmd?
expand inputString ?brackets?
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.
lb ?newbracket?
rb ?newbracket?
reset
expand
.
setbrackets lbrack rbrack
textcmd ?newTextCmd?]
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, by William H. Duquette. All rights reserved.