[en] LaTeX-package 'exp-testopt' - expandable variant of \@testopt Author: Paul Ebermann <Paul-Ebermann@gmx.de> License: LPPL 1.3b or later, maintained (see http://www.latex-project.org/lppl/). Version: 0.3 All Documentation (other than this file) is in Esperanto. This package is meant for use by other packages. This package provides an expandable variant of the LaTeX kernel command \@testopt, named \@expandable@testopt, and a more general \@expandable@ifopt, both intended for package writers. Also we have a variant of \newcommand which uses these macros to check for optional arguments. This Package needs no other packages for running. The typesetting of the documentation needs additionally class scrartcl (KOMA-script), gmdoc-enhance (from me) and gmdoc (from Grzegorz `Natror' Murzynowski), gmdoc's dependencies, and some required LaTeX packages. The package comes as .dtx + .ins. Run "tex exp-testopt.ins" to create the style file (and maybe put it to texmf/tex/latex/paul/, if your docstrip is configured accordingly), run (after that and maybe updating your TeX hash) "latex exp-testopt.dtx" to create the documentation. (Docstrip also creates an exp-testopt.test in the same directory, which is to be used by the documentation.) Usage ------- There are two main commands for package writers: \@expandable@ifopt{<YES>}{<NO>} \@expandable@testopt{<COMMAND>}{<DEFAULT>} Both of them try to check, wether following them is a optional argument (this is, some stuff enclosed in []). In this case, they expand to: <YES> <COMMAND> In the other case, they expand to: <NO> <COMMAND>[<DEFAULT>] So, \@expandable@ifopt is a general branch command, while \@expandable@testopt is meant as a replacement for \@testopt, where <COMMAND> is "any sequence of commands that 'expects' to be followed by [." (Quoted from ltdefns.dtx.) This replacement has the advantage of being expandable (if <COMMAND> is so), but also has quite some limitations. A third command is: \expnewcommand*{<COMMAND>}[<argnum>][<DEFAULT>]{<code>} This works like \newcommand, but when a command with an optional argument is defined, it uses our expandable check for the existence of the optional argument. (Without the default argument it should work exactly the same.) Examples ---------- The main usage is in definitions of commands, which are to accept optional arguments. \expnewcommand*{\example}[1][Default]{% \fbox{#1}% } Thus \example outputs an box with "Default" in it, while \example[Test] hat "Test" in it. From another package (to be published soon): \expnewcommand*\monthName[2][\languagename]{% \@format@month@name{#1}{\value{#2}}% } This outputs the value of a given LaTeX counter as a name of the like-numbered month, in the given language, the default language being the current babel language. (And here the expansibility is important, since we don't want the value of the counter when this is printed, but now, also when this ends up in the table of contents.) When you want more control, you may use \@expandable@ifopt{<YES>}{<NO>}: \def\exampleOne{% \@expandable@ifopt{% \exampleOneOpt}{% \exampleOneNoOpt}% } \def\exampleOneOpt[#1]{#2}{ % do something with #1 and #2 } \def\exampleOneNoOpt#1{% % do some other thing with #1 } With \@expandable@testopt{<COMMAND>}{<DEFAULT>} you may also create commands which accept more than one optional argument. Benefits ---------- The advantages of \expnewcommand and \@expandable@testopt to \newcommand and \@testopt are in the expandibility of the created commands - you can use them in moving arguments (like inside of \protected@edef, \protected@write and so on), and the accessed variables then have their current meanings (or the meaning at shipout time, for write), not the meanings when the thing is finally typeset. \newcommand, on the other hand, creates protected commands - you can use them also in moving arguments, but they are evaluated only at typesetting time. What to use when depends heavily on the use case. Limitations ------------- Since we want to be expandable, we really can't check the next token, but only the next "macro argument". And comparing also is not so easy when you can't assign something to a macro. So, we use as a workaround making a control sequence from the next argument (trying to neutralize a bit things like control sequences), and comparing them to a known control sequence. This has the following consequences: - we cant distinguish between [ and {[} - the second one is also regarded as a "start of optional argument". - the next argument (when it is not simple '[') has to be of a "friendly" kind - it should not contain braces, and any "dangerous" token (anything not of category 10, 11, 12) should come as the first token (only then it is neutralized). (Often the next argument is only one token, then this is no problem. Or it will be simple text, also no problem.) Otherwise, there will be strange errormessages. - multiple space tokens before the [ may irritate our macros. So, before using this macros, make sure the limitations don't affect you. Have fun! ------------- [de] LaTeX-Paket 'exp-testopt' - Expandierbare Variante von \@testopt. Autor: Paul Ebermann (Paul-Ebermann@gmx.de). Lizenz: LPPL 1.3b oder später, mit Maintenance-Status "author-maintained". Siehe http://www.latex-project.org/lppl/. Version 0.3 Geschrieben für eigenen Gebrauch (d.h. für Nutzung in anderen LaTeX-Paketen von Paul Ebermann), aber vielleicht ist es auch für andere von Nutzen. Die Dokumentation (außer dieser Datei) ist nur auf Esperanto. Dieses Paket ist für Verwendung in anderen LaTeX-Paketen gedacht. Es enthält eine expandierbare Variante des LaTeX-Kernel-Kommandos \@testopt unter dem Namen \@expandable@testopt, eine Verallgemeinerung davon als \@expandable@ifopt sowie eine Variante von \newcommand, welche diese Makros nutzt, um die Existenz optionaler Argumente festzustellen. Das Paket kommt als .dtx + .ins. Mit "latex exp-testopt.ins" wird die .sty-Datei erstellt (und eventuell gleich nach texmf/tex/latex/paul/ installiert, wenn docstrip entsprechend eingerichtet ist), mit "latex exp-testopt.dtx" kann (danach und eventuell nach einer Aktualisierung der TeX-Dateidatenbank) die Dokumentation neu erstellt werden. Docstrip erstellt auch eine Datei exp-testopt.test, welche nur von der Dokumentation verwendet wird (vom Paket nicht verwendet). Anwendung ---------- Die beiden Haupt-Kommandos für Package-Autoren: \@expandable@ifopt{<JA>}{<NEIN>} \@expandable@testopt{<BEFEHL>}{<DEFAULT>} Beide versuchen herauszufinden, ob nach ihnen ein optionales Argument folgt (das ist irgend etwas, was in [...] eingeschlossen ist). In diesem Fall expandieren diese Makros zu: <JA> <BEFEHL> Im anderen Fall expandieren sie zu: <NEIN> <BEFEHL>[<DEFAULT>] \@expandable@ifopt ist also ein genereller Verzweigungsbefehl, während \@expandable@testopt (welches dem Paket den Namen gab) ein Ersatz für \@testopt aus dem LaTeX-Kernes ist: hier sollte <BEFEHL> eine beliebige Sequenz von Kommandos sein, welche es 'erwartet', von [ gefolgt zu werden. (Frei übersetzt aus ltdefns.dtx.) Damit kann man Makros mit optionalen Argumenten und Defaultwerten für diese bauen. Dieser Ersatz hat den Vorteil, expandierbar zu sein (falls <BEFEHL> dies ist), hat aber auch einige Beschränkungen (siehe unten). Ein drittes Kommando ist: \expnewcommand*{<MAKRONAME>}[<argnum>][<DEFAULT>]{<Code>} Es funktioniert analog zu \newcommand, mit dem Unterschied, dass (falls <DEFAULT> gegeben ist) die Überprüfung auf optionale Argumente mit unserem \@expandable@testopt anstatt mit \@testopt (bzw. seiner protected-Variante) geschieht. (Ohne diesen Default-Wert sollten beide Definitionen exakt identisch sein.) Beispiele ----------- Die Hauptverwendung wird die Definition von Kommandos mit optionalen Argumenten sein: \expnewcommand*{\example}[1][Default]{% \fbox{#1}% } Hier gibt \example einen Kasten mit "Default" drin aus, wobei \example[Test] ein "Test" im Kasten hat. Aus einem meiner anderen Packages (wird bald veröffentlicht) sinngemäß: \expnewcommand*\monthName[2][\languagename]{% \@format@month@name{#1}{\value{#2}}% } Das gibt den Wert eines LaTeX-Zählers in der Form eines Monatsnamens in der gewählten Sprache aus, wobei der Default die aktuelle Babel-Sprache ist. (Hierbei braucht man die Expansibilität wirklich, denn wir wollen ja den aktuellen Wert des Zählers (und der Sprache) haben, auch wenn das ganze nachher im Inhaltsverzeichnis landet, wo eventuell andere Werte aktuell sind.) Mehr Kontrolle bekommt man mit \@expandable@ifopt: \def\exampleOne{% \@expandable@ifopt{% \exampleOneOpt}{% \exampleOneNoOpt}% } \def\exampleOneOpt[#1]#2{ % mache etwas mit #1 und #2 } \def\exampleOneNoOpt#1{% % mache etwas ganz anderes mit #1 } Mit \@expandable@testopt kann man auch Kommandos basteln, welche mehr als ein optionales Argument enthalten. Vorteile --------- Die Vorteile von \expnewcommand sowie \@expandable@testopt im Vergleich zu \newcommand und \@testopt aus dem Kernel liegen in der Expandierbarkeit der erstellten Kommandos - man kann sie innerhalb von bewegten Argumenten nutzen (wie \protexted@edef, \@protected@write etc.) und die darin verwendeten Variablen haben dann ihre aktuellen Werte, nicht die Werte, wenn der Text endlich gesetzt wird. \newcommand (mit den optionalen Argumenten) erstellt "geschützte" Kommandos - man kann sie in bewegten Argumenten verwenden, aber sie werden erst dann ausgewertet, wenn sie gesetzt werden. Welche der beiden Möglichkeiten besser ist, hängt stark von der jeweiligen Situation ab. Begrenzungen -------------- Weil wir expandierbare Makros haben wollen, können wir nicht wirklich (wie es \@testopt macht) das nächste Token überprüfen, sondern nur das "nächste Makro-Argument". Der Vergleich ist auch nicht einfach, wenn man keine Zuweisungen/Definitionen machen darf. Als Workaround erstellen wir eine Kontrollsequenz aus dem nächsten Argument (und versuchen dabei, es etwas unschädlich zu machen), und vergleichen diese mit einer bekannten Kontrollsequenz. Das hat die folgenden Konsequenzen: - Wir können nicht zwischen [ und {[} unterscheiden - auch das zweite wird als "Anfang eines optionalen Argumentes" interpretiert. - Das nächste Argument (wenn es denn nicht ein einfaches '[' ist) muss sich friedlich verhalten - es sollte keine {} enthalten, und andere gefährliche Token (alles, was nicht Kategorie 10, 11, 12 hat) sollten nur an erster Stelle auftauchen (da können sie unschädlich gemacht werden). (Wenn das nächste Argument nur aus einem Token besteht, ist das kein Problem, auch nicht, wenn es einfach nur Text ist.) Andernfalls kann es zu merkwürdigen Fehlermeldungen kommen. - Falls vor dem '[' mehrere Leerzeichen-Token kommen, kann das unsere Makros verwirren. Also, bitte vor dem Verwenden dieser Makros aufpassen, dass die Beschränkungen nicht stören. Viel Spaß! ---------- [eo] LaTeX-pakaĵo 'exp-testopt' - plibonigoj por gmdoc. (Se vi ne scias, kio estas gmdoc, legu sube.) Aŭtoro: Paŭlo Ebermann (Paul-Ebermann@gmx.de). Licenzo: LPPL 1.3, 'maintained' (-> http://www.latex-project.org/lppl/). Versio 0.3 Kreitaj por propra uzo, sed eble iom de ĝi ankaŭ uzeblas por aliaj. La dokumentaro (escepte tiu ĉi dosiero) estas nur en la germana lingvo. La pakaĵo ne bezonas ajnajn aliajn pakaĵojn. Por la dokumentado bezonatas gmdoc, gmdoc-enhance kaj scrartcl (KOMA-script). La pakaĵo venas en .dtx + .ins. Voku "latex exp-testopt.ins" por krei la .sty-dosieron (kaj eble meti ĝin al texmf/tex/latex/paul/, se via docstrip estis konfigurita laŭe), voku "latex exp-testopt.dtx" por rekrei la dokumentaron. La dokumentaĵo estas en exp-testopt.pdf.