%% BEGIN multido.tex %% \def\fileversion{1.4} \def\filedate{93/01/14} %% %% COPYRIGHT 1993, 1994, 1999 by Timothy Van Zandt, tvz@nwu.edu. %% %% This program can be redistributed and/or modified under the terms %% of the LaTeX Project Public License Distributed from CTAN %% archives in directory macros/latex/base/lppl.txt. %% %% DESCRIPTION: %% multido.tex/multido.sty contains a loop macro, \multido, that supports %% fixed-point addition and has a nice interface. Among other things, it %% is useful for pictures and putting numbers on axes. multido.tex %% is compatible with most TeX macro packages, including Plain TeX, LaTeX, %% AmSTeX, and Ams-LaTeX. %% %% INSTALLATION: %% Put this file where your TeX looks for inputs, under the name multido.tex. %% Use the file multido.sty with LaTeX. %% %% DOCUMENTATION: %% See multido.doc. %% %% COPYING: %% Copying of part or all of this file is allowed under the following %% conditions only: %% (1) You may freely distribute unchanged copies of the file. Please %% include the documentation when you do so. %% (2) You may modify a renamed copy of the file, but only for personal %% use or use within an organization. %% (3) You may copy fragments from the file, for personal use or for %% distribution, as long as credit is given where credit is due. %% %% You are NOT ALLOWED to take money for the distribution or use of %% this file or modified versions or fragments thereof, except for %% a nominal charge for copying etc. %% %% CODE: % These macros use TeX primitives, plus the Plain TeX commands: % \dimen@, \dimen@i, \count@, \newcount, \newtoks, \@ne, \tw@, \@z@. % % Check if file is loaded, announce file on terminal, and take care of @: % \message{ v\fileversion, \filedate\space } \csname MultidoLoaded\endcsname \let\MultidoLoaded\endinput \edef\TheAtCode{\the\catcode`\@} \catcode`\@=11 % These are some useful hacks. "\multido@dimtonum" strips the value of "#1", % a dimension register, of the "pt", and assigns the result to "#2", % a command sequence. \def\@empty{} \def\@nnil{\@nil} \def\multido@dimtonum#1#2{\edef#2{\multido@@dimtonum#1}} \def\multido@@dimtonum#1{\expandafter\multido@@@dimtonum\the#1} \edef\next{% \noexpand\def\noexpand\multido@@@dimtonum##1\string p\string t{##1}} \next % "\multido@count" stores the number of repetitions. "\multidocount" keeps % track of the iteration. These are also used locally as scratch counters % by "\FPadd@". "\multido@stuff" is used to store the that is % to be repeated. \newcount\multido@count \newcount\multidocount \newtoks\multido@stuff \def\multido{\multido@{}{\begingroup}{\endgroup}} \def\mmultido{\multido@{\multido@stepvar}{\begingroup}{\endgroup}} \def\Multido{\multido@{}{}{}} \def\MMultido{\multido@{\multido@stepvar}{}{}} % "\multido@initvar" processes the variable declarations, initializing % the value of the variables and defining "\multido@stepvar" to increment % the variables with each repetition. "\multido@count" is set to the number % of repetitions. \long\def\multido@#1#2#3#4#5#6{% #2% \multido@count=#5\relax \ifnum\multido@count=\z@\else\multido@@{#1}{#4}{#6}\fi #3% \ignorespaces} \def\multido@@#1#2#3{% \def\multido@stepvar{}% \def\do{\noexpand\do\noexpand}% \multido@initvar#2,\@nil,% \let\do\noexpand \edef\multido@stepvar{\multido@stepvar}% \ifnum\multido@count<\z@\multido@count=-\multido@count\fi \multidocount=1\relax #1% \multido@stuff{#3}% \multido@loop} \def\multido@loop{% \the\multido@stuff \ifnum\multidocount<\multido@count \advance\multidocount\@ne \multido@stepvar \expandafter\multido@loop \fi} \def\multidostop{\multidocount=\multido@count} \def\multido@badvar{% \multido@count=0 \errhelp{\multido command will be skipped.}% \errmessage{Bad \string\multido\space variable declaration}} \def\multido@initvar#1,{% \def\multido@temp{#1}% \ifx\multido@temp\@nnil\else \ifx\multido@temp\@empty\else \multido@@initvar#1\@nil=+\@nil\relax \fi \expandafter\multido@initvar \fi} \def\multido@@initvar#1=#2+#3\@nil#4\relax{% \ifx\@empty#4\@empty \multido@badvar \else \multido@vartype#1\@empty \ifx\multido@temp\relax \multido@badvar \else \multido@temp{#2}{#3}#1% \fi \fi} \def\multido@vartype#1{% \ifcat\noexpand\@nil\noexpand#1% \expandafter\multido@@vartype\string#1\@nil \else \let\multido@temp\relax \fi} \def\multido@@vartype#1#2#3\@nil{% \expandafter\let\expandafter\multido@temp \csname multido@init@#2\endcsname} \def\MultidoCheckNames{% \let\multido@@@vartype\multido@vartype \def\multido@vartype##1{% \ifx\undefined##1% \multido@@@vartype{##1}% \else \errhelp{\multido command will be skipped.}% \errmessage{Multido variable \string##1 already defined}% \let\multido@temp\relax \fi}} % For each variable type, we must now define \multido@init. % The syntax of these macros is: % % \multido@init@{initial value}{increment}{variable} % % and the outcome should be: % % 1. Set to the initial value. % 2. Invoke \multido@addtostep{step stuff}, where is % whatever should be done to increment the variable. is % expanded first, with expansion suppressed by "\do". \def\multido@addtostep#1{\edef\multido@stepvar{\multido@stepvar#1}} % Variable type for dimensions ("d", "D"): \def\multido@init@d#1#2#3{% \expandafter\ifx\csname pssetlength\endcsname\relax \dimen@=#1\relax\dimen@i=#2\relax \else \pssetlength\dimen@{#1}\pssetlength\dimen@i{#2}% \fi \edef#3{\number\dimen@ sp}% \ifnum\multido@count<\z@\dimen@i=-\dimen@i\fi \multido@addtostep{\do\multido@step@d{\do#3}{\number\dimen@i sp}}} \def\multido@step@d#1#2{% \dimen@=#1\advance\dimen@#2 \edef#1{\number\dimen@ sp}}% \def\multido@init@D{\multido@init@d} % Variable type for integers ("i", "I"): \def\multido@init@i#1#2#3{% \count@=#1\relax \edef#3{\the\count@}% \count@=#2\relax \ifnum\multido@count<\z@\count@=-\count@\fi \multido@addtostep{\do\multido@step@i{\do#3}{\the\count@}}} \def\multido@step@i#1#2{% \count@=#1\advance\count@ by #2 \edef#1{\the\count@}} \def\multido@init@I{\multido@init@i} % Variable type for reals ("r", "R"). \def\multido@init@r#1#2#3{% \dimen@=#1pt \multido@dimtonum\dimen@#3% \dimen@=#2pt \ifnum\multido@count<\z@\dimen@=-\dimen@\fi \multido@addtostep{\do\multido@step@r{\do#3}{\number\dimen@ sp}}} \def\multido@step@r#1#2{% \dimen@=#1pt\advance\dimen@#2 \multido@dimtonum\dimen@#1} \def\multido@init@R{\multido@init@r} % Variable type for numbers ("n", "N"). \def\multido@init@n#1#2#3{% \edef#3{#1}% \ifnum\multido@count<\z@\expandafter\FPsub\else\expandafter\FPadd\fi {0}{#2}\multido@temp \multido@addtostep{\do\FPadd{\do#3}{\multido@temp}{\do#3}}} \def\multido@init@N{\multido@init@n} % Simple fixed-point addition couldn't be harder. % "\dimen@" and "\dimen@i" are used as scratch {\em counters} (to avoid % creating a new counter). "\count@", "\multido@count" and "\multidocount" % are also used as scratch counters. % This is probably pretty close to optimal, given the requirements that % leading and trailing spaces be OK in the arguments, and that there need % not be any numbers to the left of the decimal. \def\FPadd#1#2#3{% \edef\multido@temp{#1..\noexpand\@nil#2}% \expandafter\FPadd@\multido@temp..\@nil \let#3\multido@temp} \def\FPsub#1#2{% \edef\multido@temp{\noexpand\FPsub@#2\noexpand\@empty}% \FPadd{#1}{\multido@temp}} \def\FPsub@#1{\ifx-#1\else-#1\fi} \def\FPadd@#1.#2.#3\@nil#4.#5.#6\@nil{% \begingroup \def\multido@temp{\let\next\relax}% \let\next\z@ \afterassignment\multido@temp\count@=#1\next \dimen@i=0#2sp\relax \let\next\z@ \afterassignment\multido@temp\multido@count=#4\next \multidocount=0#5\relax \dimen@=\number\count@ sp \count@=\@ne \FPadd@@@#500000000\@nil \ifnum#11<\z@ \count@=-\tw@ \dimen@=-\dimen@ \fi \ifnum#41<\z@ \count@=-\count@ \multido@count=-\multido@count \fi \ifnum\count@>\z@ \advance\multido@count\dimen@ \advance\multidocount\dimen@i \ifnum\multidocount<\multido@temp\relax \advance\multidocount\multido@temp\relax \else \advance\multido@count\@ne \fi \advance\count@-3 \else \advance\multido@count-\dimen@ \advance\multidocount-\dimen@i \ifnum\multido@count<\z@ \multido@count=-\multido@count \multidocount=-\multidocount \advance\count@\@ne \else \ifnum\multido@count=\z@ \ifnum\multidocount<\z@ \multidocount=-\multidocount \advance\count@\@ne \fi \fi \fi \ifnum\multidocount<\z@ \advance\multidocount\multido@temp \advance\multido@count-\@ne \fi \advance\multidocount\multido@temp\relax \fi \expandafter\FPadd@@\the\multidocount\@empty\@nil \endgroup} \def\FPadd@@#1#2#3\@nil{\xdef\multido@temp{% \ifnum\count@=-1-\fi \the\multido@count\ifx#2\@empty\else.#2#3\fi}} \def\FPadd@@@#1#2#3#4#5#6#7#8#9\@nil{\def\multido@temp{1#9}} \catcode`\@=\TheAtCode\relax \endinput %% %% END multido.tex