\input pitex % which \input's YaX \setparameter document: author = "Paul Isambert" title = \yax pdftitle = YaX version = \yaxversion date = "\the\month/\the\day/\the\year" display = outlines subject = "The YaX documentation." \setparameter navigator: mode = outlines \setparameter page: left = 5cm right = 3cm height = 26cm \setparameter font : command = \mainfont name = ChaparralPro- roman = Regular italic = Italic bold = Bold \setparameter font : command = \sectionfont size = 13pt name = ChaparralPro- roman = Regular italic = Italic \setparameter font : command = \titlefont size = 26pt name = ChaparralPro- roman = Regular \setparameter font: command = \codefont name = lucon size = 8.1pt type = ttf roman = {} features = "\usevalue metafont:features -trep;" \setparameter section: font = \sectionfont number = none color = ".8 0 0" \setparameter sectionbookmark: meta = navigator up = 15pt \pdfdef\ital#1{#1} \def\yax{% Y\kern-.1em \raise.15em\hbox{\sc a}\kern-.1em \lower.15em\hbox{X}% \ifcatnext a { }{\iffnext({ }}% } \def\defarg#1#2{% \def#1{\arg{#2}\kern1pt\antigobblespace}% } \defarg\param{parameter} \defarg\paralist{list of parameters} \defarg\attr{attribute} \defarg\attrlist{list of attributes} \def\paramatt{\param\kern-1pt\tcode:\attr\antigobblespace} \defarg\attrlist{list of attributes} \defarg\vallist{list of values} \defarg\value{value} \defarg\ospace{optional space} \defarg\andvalue{value exists} \defarg\orvalue{no value} \defarg\true{true} \defarg\false{false} \defarg\codarg{code} \defarg\comarg{command} \newif\ifdescribebookmark \def\describe{% \ifnext*{\describebookmarktrue\gobbleoneand\Describe} {\describebookmarkfalse\Describe}% } \def\Describe{% \ifnext!{\gobbleoneand{\let\COM\ecom\DoDescribe}} {\let\COM\com\DoDescribe}% } \def\ecom#1{{\mainfont\char"E01F}\com{#1}} \long\def\DoDescribe#1#2.{% \ifvmode \vskip\baselineskip \fi \noindent \ifdescribebookmark \outline[color = .4 .4 .4]{3}{\noexpand#1}% \fi \llap{\color{.8 0 0}{\COM#1}\iffemptystring{#2}{\kern1em}}#2% \reverse\iffemptystring{#2}{\par\noindent}\ignorespaces } \newblock*\syntax{\vskip\baselineskip\codefont\setcatcodes{\^^M=13}\obeylines} {\vskip\baselineskip} \extraboxspace=10pt \newbox\codebox \newbox\processbox \newdimen\prevcodedepth \def\processexample{% \setbox\codebox=\vtop{% \let\exampleskip\relax \parindent0pt \hsize.75\hsize \typesetexample \par \global\prevcodedepth=\prevdepth }% \setbox\processbox=\vtop{% \parindent0pt \hsize.5\hsize \doexample }% \exampleskip\exampleskip \moveleft .25\hsize\hbox{\colorbox{.9 .9 .9}{\copy\codebox\copy\processbox}}% \exampleskip\exampleskip \prevdepth=\prevcodedepth } %\output{% % \shipout\vbox to\dimexpr(\vsize+3\baselineskip){% % \box255 % \vfil % \hbox to\hsize{\mainfont\hfil\the\pageno\hfil}% % } % \advancepageno % } % \gates def {headers}{% \setbox255=\vbox to\dimexpr(\outputsize+3\baselineskip){% \box255 \vfil \hbox to\hsize{\mainfont\hfil\the\pageno\hfil}% }% } \gates def {ship}{% \shipout\box255 } \vbox to 2\baselineskip{% \vfil \hbox to\hsize{\hfil\titlefont \usevalue document : title } \vfil} \vskip\baselineskip \it \hfill \usevalue document : author \par \hfill \usevalue document : date \par \hfill v.\usevalue document : version \par \rm \vskip2\baselineskip \noindent\yax is Yet Another Key System, or \yax Ain't Keys, however you want it. It has some peculiarities. First, keys are not keys, but attributes, which means they depend on a parameter. You don't set key individually in \yax (although you can do it), you define parameters, which have attributes, which have values. Second, parameters can have definitions, so that when you set it you can also execute a command, or execute it later; thus, \yax is halfway between key management and macro definition. Finally, the syntax of \yax is not traditional: there are as few braces as possible, which you can find tremendously annoying, but it can be changed. And, of course, \yax is format-independant. Files are provided for La\TeX\ and Con\TeX t, so you can input it with \verb/\usepackage{yax}/ or \verb/\usemodule[yax]/ respectively; anywhere else just use \verb/\input yax/. \describe!\yaxversion. Starting with version 1.01, this macro holds \yax's version. Current is \yaxversion. \section{Setting parameters} \noindent Here's an example of how values are set: \example \setparameter Zappa : firstname = Frank motto = "Music is the best" hairstyle = \moustache \example/ \noindent Here we have defined a \ital{parameter}, `\verb/Zappa/', which has \ital{attributes} `\verb/firstname/', `\verb/motto/' and `\verb/hairstyle/' with \ital{values} `\verb/Frank/', `\verb/Music is the best/' and `\verb/\moustache/' respectively. \describe*\setparameter. Here's how you set parameters more precisely. (This command is not the only way to set values. See below.) Its simplified syntax is as follows: \syntax \noindent\com\setparameter \paralist\ : \attrlist\ = \value \attrlist\ = \value ... \noindent\com\par \syntax/ \noindent First, the \com\par command that terminates the declaration: it was not chosen simply because it echoes \ital{parameter}, but above all because it can be implicit in a blank line, i.e. \com\setparameter can be delimited by a blank line. That's why in the first example there seemed to be nothing delimitating \com\setparameter: it is supposed to be followed by a blank line. I will regularly display parameter setting in this fashion, even though the following would be equally legitimate: \example \setparameter Zappa : firstname = Frank ... \par \example/ \noindent barring readibility, obviously. As for the detail, \paralist is a list of space-delimited parameters, followed by a colon. Any space before the colon is removed, that's why \example \setparameter Zappa Boulez: ... \par \setparameter Zappa Boulez : ... \par \example/ \noindent set the attributes of the same \verb/Zappa/ and \verb/Boulez/ parameters. Which means, of course, that you can set the attributes of as many parameters as you wish by doing so. The name of each parameter should be fully expandable. It can contain spaces, but then it should be enclosed between braces, since space delimits parameters in \paralist. Finally, you can use \com\setparameter several times on the same parameter(s), by default it does the same thing as using one big \com\setparameter. I say \ital{by default}, because things might occur between the two calls, e.g. the parameter might be active and have deleted its own attributes after the first call, which anyway is something we'll see later. Like \paralist, \attrlist is a list of space-delimited attributes. Each \attr is made of the same thing as a \param: i.e. anything expandable. However, at the beginning of a \attrlist, the strings \verb/e:/, \verb/g:/ and \verb/x:/ have a special meaning: these prefixes are used to specify how an attribute or list of attributes is to be defined, and they are similar to \com\edef, \com\gdef and \com\xdef (with no prefix meaning \com\def), i.e.: \example \setparameter foo : one = \whatever e: two = \whatever g: three = \whatever x: four five = \whatever \example/ \noindent defines, for the parameter \verb/foo/, the attributes \verb/one/ and \verb/three/ as \com\whatever and the attributes \verb/two/, \verb/four/ and \verb/five/ as the full expansion of \com\whatever; moreover, \verb/one/ and \verb/two/ are locally defined, whereas \verb/three/, \verb/four/ and \verb/five/ are globally defined. The space between the prefix and the name of the parameter is optional. Finally, \value may be given in three ways. First, it can be delimited by a space, e.g.: \example \setparameter Zappa : firstname = Frank motto = {Music is the best} ... \example/ \noindent here the space is simply the end of the line. The spaces in \verb/Music is the best/ aren't seen, because of the braces (which will be removed from the value). Don't forget that control sequences eats the subsequent space, hence a control sequence can't be at the end of a value supposedly delimited by space without braces. Actually, it can't be at the beginning either (see below). Otherwise, \arg{value} can be given between double quotes: \example \setparameter Zappa : firstname = "Frank" motto = "Music is the best" hairstyle = "\mustache" \example/ \noindent which basically act as braces. The space inserted here by the end of the line is optional and the following can be done: \example \setparameter Zappa:firstname="Frank"motto="Music is the best"...\par \example/ \noindent Finally, there's a special rule. Control sequences gobble the next space, but a \value can be made of one and only one control sequence, e.g.: \example \setparameter Zappa : hairstyle = \mustache ... \example/ \noindent i.e. if \yax sees a control sequence at the beginning of a value it will take this control sequence only as \arg{value}, which means that anything thereafter will be considered as belonging to the next \param name. Hence, all the following are bad ideas: \example \setparameter bad values : % the space stops the value to "bad" one = bad space % the control sequence is taken as the only token two = \control sequence % "\sequence" eats the delimitating space three = control\sequence \example/ \noindent but they'd all be ok with quotes or braces (which still require a space after): \example \setparameter good values : one = "bad space" two = "\control sequence" three = {control\sequence} four = \LonelyCommand five = "\Command\Command" ... \example/ \noindent Finally, they are some exceptions to this rule: first, always put \com\par between braces, otherwise it will be seen as the end of the parameter. Second, always put a character denotation between either braces or quotes, even if it's the only control sequence. If you don't know what character denotation means, never mind. Just remember that \com\bgroup and \com\egroup are character denotations. Now I'm sure you ask: what is this lousy syntax? An answer is I don't like braces. Another answer is \yax is not designed to store complicated strings, although it can do it. Instead it aims at setting simple values in an orderly fasion, e.g.: \example \setparameter page : pagewidth = 30cm pageheight = 32cm top bottom = 2cm lines = 45 whatever = \foo ... \example/ \noindent in which case it is very handy, especially when you're in a hurry: \example \setparameter page: pagewidth=32cm pageheight=30cm whatever=\foo...\par \example/ \noindent Thus, I find commas to delimit values equally superfluous and find the odd quote better. However, good old key-value pairs separated by commas happen to be useful too, for instance when a command takes some options. Hence the following was introduced in v.\yaxversion: \describe*\setparameterlist\paralist\oarg{optional macro}\arg{attribute(s) = value list}. This produces exactly the same thing as \com\setparameter, only the syntax changes. Ignoring the optional argument in the middle, our example just above would be rewritten as: \example \setparameterlist{page}{pagewidth = 32cm, pageheight = 30cm, whatever=\foo} \example/ \noindent Here you can forget about how values are given: the delimiter is the comma, end of story. There can be several parameters in the first argument, and several attributes before each `\verb"="' sign in the second; in both cases, they are separated by spaces, as above. Also, each list of attributes can be prefixed with \verb"e:", \verb"g:" and \verb"x:", again as with \com\setparameter. Finally, as shown in the example, space is trimmed away. But \com\setparameterlist has a feature that \com\setparameter doesn't have. If one of the \attr = \value pair doesn't contain a `\verb"="', i.e. if it isn't a pair at all, then one of the following happens: if there is no \arg{optional macro}, then the default value `\verb"true"' is assigned to all the attributes mentionned; besides, the entry can still be prefixed with \verb"e:", \verb"g:" or \verb"x:". For instance, the following two statements are equivalent: \example \setparameterlist{Zappa}{guitarist composer} \setparameterlist{Zappa}{guitarist composer = true} \example/ On the other hand, if the \arg{optional macro} is present, then it should take one argument, and the entry is passed to it (trimmed, though). What happens then is none of \yax's concern, although of course the macro can itself set parameters. For instance, in the following, \verb"a" and \verb"b" will be set to `\verb"what"' and `\verb"are you kidding"', whereas `\verb"eddie?"' is passed to \com\dosomething. \example \def\dosomething#1{ ... #1 ... } \setparameter{whatever}[\dosomething]{a = what, b = are you kidding, eddie?} \example/ Another way to set a parameter is as follows: \describe*\copyparameter\paralist:\param\arg{space}. \describe\gcopyparameter\paralist:\param\arg{space}. All the attributes of the parameters in \paralist are copied to \param; if the latter already has some attributes, they're aren't deleted (but you can use \com\deleteparameter beforehand, see below) although they might be overwritten. If several parameters in \paralist have the same attribute, the value of the last parameter in the list wins. The difference between the two versions is that the second is global. The space shouldn't be forgotten, it's the same as explained in the \paramatt syntax \jumplink{values}{\ital{below}}. Finally, here's one fast way of setting a single attribute: \describe*\setattribute\paramatt=\value\arg{space}. \describe\esetattribute\paramatt=\value\arg{space}. \describe\gsetattribute\paramatt=\value\arg{space}. \describe\xsetattribute\paramatt=\value\arg{space}. This sets \attr (no list) for \param (no list) to \value; see \jumplink{values}{\ital{below}} for the \paramatt syntax. Note that the \arg{space} is for real, so don't forget it. The `=' sign may be surrounded by optional space. This command can be used instead of \com\setparameter for two reasons: first, it is much faster (because it doesn't take lists into account); second, if \param is active it is \ital{not} executed (see the section \jumplink{defpar}{\ital{Defining parameters}}). The \verb"e-", \verb"g-" and \verb"x-"versions sets the attribute with the \verb"e:", \verb"g:" and \verb"x:" prefixes respectively. \describe*\deleteattribute\paramatt. \describe\gdeleteattribute\paramatt. This deletes \paramatt, which now responds negatively to all previous commands, as if it was never defined. The second version makes this deletion global, the first keeps it local. \describe*\deleteparameter\paralist:. \describe\gdeleteparameter\paralist:. This deletes all paramaters in \paralist; it is equivalent to using the previous command on all the parameters' attributes. The first version is local, the second is global. And yes, the colon is really there, although it might get away. \sectioncommand{section}{The meta attribute}{The \ital{meta} attribute} \noindent You can give any attribute to any parameter (unless they're restricted, but that's not the point for the time being). However, there's one particular attribute which has a special meaning: \verb/meta/. The value of \verb/meta/ should be another parameter. Then, when querying the value of an attribute, say \verb/attr/, for a given parameter, say \verb/param/, \yax will do the following: if \verb/param/ has \verb/attr/, it is returned. Otherwise, if \verb/param/ has a \verb/meta/ attribute, whose value is for instance \verb/metaparam/, then \verb/attr/ is queried for the value of \verb/metaparam/. And if \verb/metaparam/ has no \verb/attr/ but has a \verb/meta/ attribute, this process continues, until either a parameter is found with \verb/attr/ or there are no new \verb/meta/. For instance: \example \setparameter A : attr = value ... \par \setparameter B : meta = A ... \par \setparameter C : meta = B ... \par \example/ \noindent If parameter \verb/C/ has no \verb/attr/, then it retrieves it from parameter \verb/A/ via \verb/B/ (or from \verb/B/ if it has one). On the other hand, if \verb/C/ has \verb/attr/, then its own value is returned. You can also query the value of an attribute for a parameter and forbid the search for \verb/meta/'s, as explained in the next section. Don't be afraid to create loop with \verb/meta/. The following is perfectly legitimate: \example \setparameter A : meta = C ... \par \setparameter B : meta = A ... \par \setparameter C : meta = B ... \par \example/ \noindent and it can even be useful. \yax detects loops when searching \verb/meta/ paths and stops in time (with no value, obviously). Finally, you can set \verb/meta/ to a parameter that doesn't exist. It will simply return no value. The use of \verb/meta/ is useful to create families of parameters and/or to set default values, e.g.: \example \setparameter mammal : egg = no fur = yes ... \setparameter cat whale : meta = mammal ... \setparameter cat : \setparameter whale : foot = clawed fur = no tooth = fang foot = flipper ... tooth = baleen ... \setparameter tiger mykitty : meta = cat stripped = yes ... \setparameter tiger : \setparameter mykitty : foot = "very bad news" foot = "bad news" ... ... \example/ Apart from that, \verb/meta/ behaves as any other attribute, i.e. it can be freely set and queried. \section{Using values}[values] \noindent Once attributes have been set, they can be queried by the macros that follow. But first, one last bit of odd syntax: \paramatt means anything up to the colon as the \param and then anything up to the next space as the \attr. That's the reason why space in attribute names is a bad idea: the space is the main delimitator when using attributes. It is gobbled in the process. On the other hand, any space surrounding the colon is removed, so that `\verb/zappa:hairstyle/ ' and `\verb/zappa : hairstyle/ ' denote the same attribute of the same parameter. Good news, though: if you don't like that syntax, the next section explains how to create commands with the same meaning but a different syntax. In what follows, fully expandable commands are marked with \color{.8 0 0}{\char"E01F}. \describe*!\nometa\arg{command}. All the commands that follow except \com\parameterloop can be prefixed with \com\nometa. In this case, \yax will return the value of the attribute for the specified parameter, as usual, or return no value, i.e. it will not search \verb/meta/ parameters. So, in what follows, `\paramatt is defined' means two things. If \com\nometa isn't used, it means that \param has \attr or it has a metaparameter with \attr; on the other hand, if \com\nometa is used, it means that \param has \attr, end of story. \describe*!\ifattribute\paramatt\true\false. This returns \true if \paramatt is defined, \false otherwise. Since all commands below always check whether \paramatt is defined before trying to do anything with the value, this command can be avoided most of the time. \setparameter musician : job = music \setparameter Zappa : meta = musician firstname = Frank \Example \setparameter musician : job = music\par \setparameter Zappa : meta = musician firstname = Frank % See this space? \ifattribute Zappa : job {Good, it is.}{...}\par \nometa\ifattribute Zappa : job {...}{Too bad.} \Example/ \describe*!\usevalue\paramatt. \describe!\usevalueor\paramatt\orvalue. \describe!\usevalueand\paramatt\andvalue\orvalue. The first macro returns the value of \paramatt if it exists, or does nothing otherwise. Like all the \verb/-or/ and \verb/-and/ variants below, \com\usevalueor executes \orvalue in case \paramatt doesn't exist, while \com\usevalueand returns the value of \paramatt immediately followed by \andvalue (no brace added) if \paramatt exists, otherwise it executes \orvalue. \Example Zappa's job was \usevalueand Zappa : job { (and then some!)} {unknown} and he played the \usevalueor Zappa : instrument {guitar}. \Example/ \iflines4{}{\breakpage} \describe*!\passvalue\codarg\paramatt. \describe!\passvalueor\codarg\paramatt\orvalue. \describe!\passvalueand\codarg\paramatt\andvalue\orvalue. These return \codarg\barg{value} if \paramatt is defined, with the \verb/-or/ and \verb/-and/ variants as above. \Example \def\whichwas#1{(which was #1)} Zappa's job \passvalue\whichwas Zappa : job \ took most of his time, because it's a time-consuming occupation \nometa\passvalueor\whichwas Zappa : job {(you know which)}. \Example/ \describe*!\passvaluenobraces\codarg\paramatt. \describe!\passvaluenobracesor\codarg\paramatt\orvalue. \describe!\passvaluenobracesand\codarg\paramatt\andvalue\orvalue. These are the same as \com\passvalue and variants except the value of the attribute is concatenated to \codarg without braces (which means that no braces are added in the process, not that braces are removed from the value if it has any). \describe*\settovalue\arg{dimen or count}\paramatt. \describe\settovalueor\arg{dimen or count}\paramatt\orvalue. \describe\settovalueand\arg{dimen or count}\paramatt\andvalue\orvalue. This sets the first argument to the value of \paramatt if it exists. If the first argument is more than one token (e.g. \verb/\count0/ vs. \verb/\parindent/), it must be surrounded by braces; and actually it can even be something like \verb/\advance\count0/. Of course \arg{dimen or count} must be a dimension or a count, and the value of \paramatt must be accordingly a dimension or a number (\yax doesn't check either of them). \Example \setparameter para : parskip = 2pt \par Note that \settovalueor\parskip para : parskip {\parskip=1pt\relax} (\the\parskip) is basically the same thing as \parskip=\usevalueor para : foo {1pt\relax} (\the\parskip). \Example/ What the previous example shows is that since \com\usevalue is thoroughly expandable one can say: \example \mydimen=\usevalueor parameter : attribute {0pt} \example/ \noindent and it will set \com\mydimen to the value of \paramatt or to \verb/0pt/. The difference with \com\settovalueor is that in the construction with \com\usevalueor the assignement is obligatorily made (hence the \verb/-or/ variant), whereas with \com\settovalueor the or-clause can do something else (e.g. send an error message). And \com\settovalue insert a prophylactic \com\relax. \describe*\storevalue\comarg\paramatt. \describe\storevalueor\comarg\paramatt\orvalue. \describe\storevalueand\comarg\paramatt\andvalue\orvalue. These define \comarg as the value of \paramatt if it exists. \Example \setparameter Zappa : hairstyle = \moustache\par \storevalue\beard Zappa : hairstyle \meaning\beard \Example/ \describe*!\ifvalue\paramatt=\value\ \true\false. This returns \true if the value of \paramatt is \value and \false otherwise (including unexisting \paramatt). Note that when comparing the value of \paramatt with \value, catcodes aren't part of the picture. Here \value is delimited by the following space, but there might be optional space after the `\verb/=/' sign. Because of this, it is not possible to test for the emptyness of a value with \com\ifvalue, i.e. \example \ifvalue foo : bar = {} {true}{false} \example/ \noindent won't work. Instead, either use \com\ifcasevalue below or \com\passvalue with an emptyness-tester (e.g. \verb/texapi/'s \com\ifemptystring, since \yax is based on \verb/texapi/ (what, me, self-advertising?)). \Example \bgroup \catcode`\Z=13 \setparameter foo : g: bar = Z \par \egroup \edef\foobar{% \ifvalue foo : bar = Z {yes}{no}, even though catcodes are different.} \meaning\foobar \Example/ \iflines2{}{\breakpage} \describe*!\ifcasevalue\paramatt. \describe\val\value\ \codarg. \describe\val\value\ \codarg. \verb/.../ \par\noindent \describe\elseval \codarg. \describe\endval\relax. This executes \codarg following \value matching the value of \paramatt. If \paramatt doesn't exist, or matches no \value, then \com\elseval is executed. Once again catcodes aren't taken into account when values are compared. The exact syntax is: \value is anything from \com\val to the next space, and \codarg is anything that follows up to the next \com\val, \com\elseval or \com\endval (any space on the right is removed, so no need to stick \com\val to \codarg). Apart from \com\endval, everything here is optional: there might be as many \com\val-clauses as needed, including none, and the \com\elseval-clause need not be present (in which case, if no match occurs, nothing happens). Finally, although this is similar to \TeX's primitive \com\ifcase, there's no need to jump before anything with \com\expandafter to avoid bumping into conditional structure. \Example \def\doitalic#1{{\it#1}} \setparameter type : font = italic \par \ifcasevalue type : font \val italic \doitalic \val bold \dobold \endval{Some text.} \edef\foo{% \ifcasevalue type : font \val bold This is bold \elseval This is something else \endval} \meaning\foo \Example/ \describe*!\parameterloop\paralist:\arg{code}. This loops on all the attributes for all the parameters in \paralist, and passes the following thing to \arg{code}: the parameter, the attribute, and the value. Hence \arg{code} can be anything, but it should be prepared to handle three arguments. The order in which parameters are enumerated is the order in \paralist, and for attributes it is the order in which they were set for a given parameter. Only actual attributes are enumerated: if you loop over \verb"ParameterA", which doesn't have \verb"AttributeA" but has a \verb"meta" parameter with that attribute, then \verb"AttributeA" will \emph{not} be enumerated in the loop. \Example \setparameter musician : job = music\par \setparameter zappa : meta = musician instrument = guitar \def\showvalues#1#2#3{#1 : #2 = #3\par} \parameterloop musician zappa : \showvalues \Example/ \section{Using another syntax} \noindent If you don't like \yax's native syntax, and want for instance good old braces to delimit \param and \attr, you might be tempted to do something like: \example \def\myusevalue#1#2{\usevalue #1:#2 } \example/ \noindent On the other hand, if you don't mind \yax's syntax but want other names for the commands, then you'll probably go: \example \let\myusevalue\usevalue \example/ \noindent Both are bad ideas. Indeed, in neither example will \com\myusevalue work properly with \com\nometa. Besides, you have to create the \verb/-or/ and \verb/-and/ variants by hand. Not to mention that in the first example \com\myusevalue wastes time calling \com\usevalue (and what if it is redefined?) when it could be in direct relation with internal code. So here's how to circumvent \yax's syntax and/or create new names. \describe*\newsyntax\arg{syntax}\barg{prefix}. This creates commands whose names are \verb/\/\arg{prefix}\comarg and whose syntax for arguments is \arg{syntax}. The latter is a parameter text which must contain \verb/#1/ and \verb/#2/ (for \param and \attr respectively) with whatever to delimit them. For instance: \example \newsyntax#1#2{x} \newsyntax#1 #2!{y} \example/ \noindent will create among others an \com\xusevalue command whose usage is \com\xusevalue\param\attr and a \com\yusevalue command whose usage is \com\yusevalue\param\ \attr\tcode! and both will do the same thing as \com\usevalue. There must be braces around \arg{prefix} and none around \arg{syntax} (i.e. the latter is delimited by the left brace of the former). If \arg{prefix} is empty, you redefine the default commands, which is dangerous. To be precise, the commands copied are (omitting the \verb/-or/ and \verb/-and/ variants, which are created too, if any): \com\setattribute, \com\esetattribute, \com\gsetattribute, \com\xsetattribute, \com\ifattribute, \com\usevalue, \com\passvalue, \com\passvaluenobraces, \com\settovalue, \com\storevalue, \com\ifvalue, \com\ifcasevalue, \com\deleteattribute and \com\restrictattribute (which you'll learn about in the next section). \describe*\copysyntax\arg{prefix1}\arg{prefix2}. This defines all the commands above with \arg{prefix1} as those same commands with \arg{prefix2}. \describe*\letyaxcommand\arg{command1}\arg{command2}. This at the very least \com\let \arg{command1} to \arg{command2}. Besides, if \arg{command2} can take a \com\nometa prefix, \arg{command1} can too. Finally, if \arg{command2} has \verb/-or/ and \verb/-and/ variants, these are created with \arg{command1}. E.g.: \example \letyaxcommand\defval\storevalue \example/ \noindent defines \com\defval, \com\defvalor and \com\defvaland as \com\storevalue and its variants. If \arg{command2} has been created with \com\newsyntax or \com\copysyntax, \arg{command1} of course has the same syntax. \Example \newsyntax#1#2{x} \letyaxcommand\uv\xusevalue \uvand{noparameter}{noattribute}{yes}{no} \Example/ \section{Restrictions on parameters and attributes} \describe*\restrictparameter\paralist:\attrlist\tcode{\char`\\par}. After this declaration, the \param's in \paralist (where they are separated by space as in \com\setparameter) can take only those \attr's in \attrlist (which are also separated by space). It affects \com\setparameter only, producing an error message when an \attr not belonging to \attrlist is given a value (and the assignment isn't made, of course). Even if it doesn't belong to \attrlist, the \verb/meta/ attribute is always allowed. Several \com\restrictparameter declarations on the same parameter(s) actually accumulate the allowed attributes in \paralist, e.g. after \example \restrictparameter foo : one two three\par \restrictparameter foo : four\par \example/ \noindent one can set the attributes \verb/one/, \verb/two/, \verb/three/ and \verb/four/ for \verb/foo/, and not only \verb/four/. The idea behind \com\restrictparameter is not so much hiding attributes from the user as making the use of a parameter clearer by indicating which attributes are in use with it, especially if it can be executed (see below). \describe*\restrictattribute\paramatt\ \vallist\tcode{\char`\\par}. This restricts \paramatt to take only the values in \vallist (separated by space). \describe*\restrictallattributes\attr\ \vallist\tcode{\char`\\par}. This restricts \attr, whatever the \param in which it appears, to take only the values in \vallist. In this command, \attr is found as anything before the first space, e.g.: \example \restrictallattributes attribute value1 value2 value3\par \restrictallattributes {attri bute} value1 value2 value3\par \example/ \noindent (the second example if you want to have space in attribute names). Note that if an attribute is restricted with both \com\restrictattribute and \com\restrictallattributes, only the former restriction holds. E.g.: \example \restrictattribute foo:bar one\par \restrictallattributes bar two three\par \setparameter foo: bar = two % Will produce an error message. ... \example/ \section{Defining parameters}[defpar] \describe*\defparameter\paralist\barg{definition}. Parameters aren't just a way of organizing attributes. They can have a definition and act as commands whose arguments are the values of their attributes. The \paralist (with parameters once again separated by space) must be braceless, whereas \arg{definition} must be enclosed in braces (like a real definition). No parameter text is allowed. However, \arg{definition} can contain `\verb/#1/', which doesn't refer to any argument but to the parameter being defined instead, so that one can use the commands defined in the previous section without specifying the name of the parameter. E.g. \example \defparameter foo bar {% \usevalue #1 : one \passvalueor\mycomm #1 : whatever {...}% ...} \example/ \noindent defines \verb/foo/ and \verb/bar/ respectively to \example \usevalue foo : one \passvalueor\mycomm foo : whatever {...}% ... \usevalue bar : one \passvalueor\mycomm bar : whatever {...}% ... \example/ \describe*!\executeparameter\param:. This executes the definition of \param with the the latest values of its attributes. If \param hasn't been defined, nothing happens. \Example \defparameter foo {The value is \usevalue#1:bar } \setparameter foo : bar = whatever\par \executeparameter foo: \Example/ \describe*\defactiveparameter\paralist\barg{definition}. This does the same thing as \com\defparameter, i.e. define the parameters in \paralist, but it also set them as `active,' which means that they're automatically executed each time their attributes are defined with \com\setparameter. You can still use \com\executeparameter. \Example \setparameter metasection : skip = 2 font = \it\par \setparameter mysection : % not yet active meta = metasection\par \defactiveparameter mysection {% \vskip \usevalueor #1 : skip 0\baselineskip {\usevalue #1 : font \usevalue #1 : title }% \par} And now we're going to have a new section. \setparameter mysection : title = "A new section"\par Fascinating. Once again? \setparameter mysection : title = "Once again"\par Cool. \Example/ %{\it\noindent See next page for the last example.} \Example \setparameter metasection : skip = 2 inline = false font = \it \setparameter mysection mysubsection : meta = metasection \setparameter mysection : font = \sc \setparameter mysubsection : skip = 1 inline = true \defparameter mysection mysubsection {% \vskip \usevalueor #1 : skip 0\baselineskip {\usevalue #1 : font \usevalue #1 : title }% \ifvalue #1 : inline = true {. \ignorespaces}% \par}% \def\section#1{% \setparameter mysection : title = {#1}\par \executeparameter mysection :} \def\subsection#1{% \setparameter mysubsection : title = {#1}\par \executeparameter mysubsection :} ... and this is the end of our paragraph. \section{New ideas} Here we are going to expose bold new ideas. \subsection{First bold new idea} Lore, aim, hip, sum... what do you think about it? Pig latin, you say? \subsection{Second bold new idea} Perhaps Do lore, aim... then, huh? \section{New new ideas} Etc. \Example/ \finishpdffile \bye