]> mj.ucw.cz Git - ucwmac.git/blob - ucwmac.tex
Added facilities for defining macros with optional arguments
[ucwmac.git] / ucwmac.tex
1 % The UCW Macro Collection (a successor of mjmac.tex)
2 % Written by Martin Mares <mj@ucw.cz> in 2010--2018 and placed into public domain
3 % -------------------------------------------------------------------------------
4
5 %%% Prolog %%%
6
7 % We'll use internal macros of plain TeX
8 \catcode`@=11
9
10 \ifx\eTeXversion\undefined
11 \errmessage{ucwmac requires the e-TeX engine or its successor}
12 \fi
13
14 %%% PDF output detection %%%
15
16 \newif\ifpdf
17 \pdffalse
18
19 \ifx\pdfoutput\undefined
20 \else\ifnum\pdfoutput>0
21         \pdftrue
22         \pdfpkresolution=600    % Provide a reasonable default
23 \fi\fi
24
25 %%% Page size and margins %%%
26
27 % If you modify these registers, call \setuppage afterwards
28 \ifx\luatexversion\undefined
29         % In LuaTeX, \pagewidth and \pageheight are primitive
30         \newdimen\pagewidth
31         \newdimen\pageheight
32 \fi
33 \newdimen\leftmargin
34 \newdimen\rightmargin
35 \newdimen\topmargin
36 \newdimen\bottommargin
37 \newdimen\evenpageshift
38
39 \def\setuppage{%
40         \hsize=\pagewidth
41         \advance\hsize by -\leftmargin
42         \advance\hsize by -\rightmargin
43         \vsize=\pageheight
44         \advance\vsize by -\topmargin
45         \advance\vsize by -\bottommargin
46         \hoffset=\leftmargin
47         \advance\hoffset by -1truein
48         \voffset=\topmargin
49         \advance\voffset by -1truein
50         \ifpdf
51                 \pdfhorigin=1truein
52                 \pdfvorigin=1truein
53                 \ifx\luatexversion\undefined
54                         \pdfpagewidth=\pagewidth
55                         \pdfpageheight=\pageheight
56                 \fi
57         \fi
58 }
59
60 % Set multiple margins to the same value
61 \def\sethmargins#1{\leftmargin=#1\relax\rightmargin=#1\relax\evenpageshift=0pt\relax}
62 \def\setvmargins#1{\topmargin=#1\relax\bottommargin=#1\relax}
63 \def\setmargins#1{\sethmargins{#1}\setvmargins{#1}}
64
65 % Define inner/outer margin instead of left/right
66 \def\setinneroutermargin#1#2{\leftmargin#1\relax\rightmargin#2\relax\evenpageshift=\rightmargin\advance\evenpageshift by -\leftmargin}
67
68 % Use a predefined paper format, calls \setuppage automagically
69 \def\setpaper#1{%
70         \expandafter\let\expandafter\currentpaper\csname paper-#1\endcsname
71         \ifx\currentpaper\relax
72                 \errmessage{Undefined paper format #1}
73         \fi
74         \currentpaper
75 }
76
77 % Switch to landscape orientation, calls \setuppage automagically
78 \def\landscape{%
79         \dimen0=\pageheight
80         \pageheight=\pagewidth
81         \pagewidth=\dimen0
82         \setuppage
83 }
84
85 % Common paper sizes
86 \def\defpaper#1#2#3{\expandafter\def\csname paper-#1\endcsname{\pagewidth=#2\pageheight=#3\setuppage}}
87 \defpaper{a3}{297truemm}{420truemm}
88 \defpaper{a4}{210truemm}{297truemm}
89 \defpaper{a5}{148truemm}{210truemm}
90 \defpaper{letter}{8.5truein}{11truein}
91 \defpaper{legal}{8.5truein}{14truein}
92
93 % Default page parameters
94 \setmargins{1truein}
95 \setpaper{a4}
96
97 %%% Macros with optional arguments %%%
98
99 % After \def\a{\withoptarg\b}, the macro \a behaves in this way:
100 %       \a[arg]         does \def\optarg{arg} and then it expands \b
101 %       \a              does \let\optarg=\relax and then it expands \b
102 \def\withoptarg#1{\let\xoptcall=#1\futurelet\next\xopt}
103 \def\xopt{\ifx\next[\expandafter\xoptwith\else\let\optarg=\relax\expandafter\xoptcall\fi}
104 \def\xoptwith[#1]{\def\optarg{#1}\xoptcall}
105
106 % A shortcut for defining macros with optional arguments:
107 % \optdef\macro behaves as \def\domacro, while \macro itself is defined
108 % as a wrapper calling \domacro using \withoptarg.
109 \def\optdef#1{%
110         \edef\xoptname{\expandafter\eatbackslash\string#1}%
111         \edef#1{\noexpand\withoptarg\csname do\xoptname\endcsname}%
112         \expandafter\def\csname do\xoptname\endcsname
113 }
114
115 % Trick: \eatbackslash eats the next backslash of category 12
116 \begingroup\lccode`\+=`\\
117 \lowercase{\endgroup\def\eatbackslash+{}}
118
119 % Expand to the optional argument if it exists
120 \def\optargorempty{\ifx\optarg\relax\else\optarg\fi}
121
122 %%% Placing material at specified coordinates %%%
123
124 % Set all dimensions of a given box register to zero
125 \def\smashbox#1{\ht#1=0pt \dp#1=0pt \wd#1=0pt}
126 \long\def\smashedhbox#1{{\setbox0=\hbox{#1}\smashbox0\box0}}
127 \long\def\smashedvbox#1{{\setbox0=\vbox{#1}\smashbox0\box0}}
128
129 % Variants of \llap and \rlap working equally on both sides and/or vertically
130 \def\hlap#1{\hbox to 0pt{\hss #1\hss}}
131 \def\vlap#1{\vbox to 0pt{\vss #1\vss}}
132 \def\clap#1{\vlap{\hlap{#1}}}
133
134 % \placeat{right}{down}{hmaterial} places <hmaterial>, so that its
135 % reference point lies at the given position wrt. the current ref point
136 \long\def\placeat#1#2#3{\smashedhbox{\hskip #1\lower #2\hbox{#3}}}
137
138 % Like \vbox, but with reference point in the upper left corner
139 \long\def\vhang#1{\vtop{\hrule height 0pt\relax #1}}
140
141 % Like \vhang, but respecting interline skips
142 \long\def\vhanglines#1{\vtop{\hbox to 0pt{}#1}}
143
144 % Crosshair with reference point in its center
145 \def\crosshair#1{\clap{\vrule height 0.2pt width #1}\clap{\vrule height #1 width 0.2pt}}
146
147 %%% Output routine %%%
148
149 \newbox\pageunderlays
150 \newbox\pageoverlays
151 \newbox\commonunderlays
152 \newbox\commonoverlays
153
154 % In addition to the normal page contents, you can define page overlays
155 % and underlays, which are zero-size vboxes positioned absolutely in the
156 % front / in the back of the normal material. Also, there are global
157 % versions of both which are not reset after every page.
158 \def\addlay#1#2{\setbox#1=\vbox{\ifvbox#1\box#1\fi\nointerlineskip\smashedvbox{#2}}}
159 \def\pageunderlay{\addlay\pageunderlays}
160 \def\pageoverlay{\addlay\pageoverlays}
161 \def\commonunderlay{\addlay\commonoverlays}
162 \def\commonoverlay{\addlay\commonoverlays}
163
164 % Our variation on \plainoutput, which manages inner/outer margins and overlays
165 \output{\ucwoutput}
166 \newdimen\pagebodydepth
167 \def\ucwoutput{\wigglepage\shipout\vbox{%
168         \makeheadline
169         \ifvbox\commonunderlays\copy\commonunderlays\nointerlineskip\fi
170         \ifvbox\pageunderlays\box\pageunderlays\nointerlineskip\fi
171         \pagebody
172         \pagebodydepth=\prevdepth
173         \nointerlineskip
174         \ifvbox\commonoverlays\vbox to 0pt{\vskip -\vsize\copy\commonoverlays\vss}\nointerlineskip\fi
175         \ifvbox\pageoverlays\vbox to 0pt{\vskip -\vsize\box\pageoverlays\vss}\nointerlineskip\fi
176         \prevdepth=\pagebodydepth
177         \makefootline
178 }\advancepageno
179 \ifnum\outputpenalty>-\@MM \else\dosupereject\fi}
180
181 \def\wigglepage{\ifodd\pageno\else\advance\hoffset by \evenpageshift\fi}
182
183 % Make it easier to redefine footline font (also, fix it so that OFS won't change it unless asked)
184 \let\footfont=\tenrm
185 \footline={\hss\footfont\folio\hss}
186
187 %%% Itemization %%%
188
189 % Default dimensions of itemized lists
190 \newdimen\itemindent            \itemindent=0.5in
191 \newdimen\itemnarrow            \itemnarrow=0.5in                       % make lines narrower by this amount
192 \newskip\itemmarkerskip         \itemmarkerskip=0.4em                   % between marker and the item
193 \newskip\preitemizeskip         \preitemizeskip=3pt plus 2pt minus 1pt  % before the list
194 \newskip\postitemizeskip        \postitemizeskip=3pt plus 2pt minus 1pt % after the list
195 \newskip\interitemskip          \interitemskip=2pt plus 1pt minus 0.5pt % between two items
196
197 % Analogues for nested lists
198 \newdimen\nesteditemindent      \nesteditemindent=0.25in
199 \newdimen\nesteditemnarrow      \nesteditemnarrow=0.25in
200 \newskip\prenesteditemizeskip   \prenesteditemizeskip=0pt
201 \newskip\postnesteditemizeskip  \postnesteditemizeskip=0pt
202
203 \newif\ifitems\itemsfalse
204 \newbox\itembox
205 \newcount\itemcount
206
207 \def\preitemize{
208         \ifitems
209                 \vskip\prenesteditemizeskip
210                 \advance\leftskip by \nesteditemindent
211                 \advance\rightskip by \nesteditemnarrow
212         \else
213                 \vskip\preitemizeskip
214                 \advance\leftskip by \itemindent
215                 \advance\rightskip by \itemnarrow
216         \fi
217         \parskip=\interitemskip
218 }
219
220 \def\postitemize{
221         \ifitems
222                 \vskip\postnesteditemizeskip
223         \else
224                 \vskip\postitemizeskip
225         \fi
226 }
227
228 \def\inititemize{\begingroup\preitemize\itemstrue\parindent=0pt}
229
230 \def\itemize#1{\inititemize\setbox\itembox\llap{#1\hskip\itemmarkerskip}%
231 \let\:=\singleitem}
232
233 \def\singleitem{\par\leavevmode\copy\itembox\ignorespaces}
234
235 \def\endlist{\par\endgroup\postitemize}
236
237 \def\ibull{\raise0.2ex\hbox{$\bullet$}} % Signs frequently used for \itemize
238 \def\idot{\raise0.2ex\hbox{$\cdot$}}
239 \def\istar{\raise0.2ex\hbox{$\ast$}}
240
241 \def\numlist#1{\inititemize\itemcount=0\let\:=\numbereditem
242 \let\itemnumbering=#1}
243
244 \def\numbereditem{\par\leavevmode\advance\itemcount by 1
245 \llap{\itemnumbering\hskip\itemmarkerskip}\ignorespaces}
246
247 \def\nnorm{\the\itemcount}
248 \def\ndotted{\nnorm.}
249 \def\nparen{\nnorm)}
250 \def\nparenp{(\nnorm)}
251 \def\nroman{\romannumeral\itemcount}
252 \def\nromanp{\nroman)}
253 \def\nalpha{\count@=96\advance\count@ by\itemcount\char\count@)}
254 \def\nAlpha{\count@=64\advance\count@ by\itemcount\char\count@)}
255 \def\ngreek{$\ifcase\itemcount\or\alpha\or\beta\or\gamma\or\delta\or\epsilon\or
256 \zeta\or\eta\or\theta\or\iota\or\kappa\or\lambda\or\mu\or\nu\or\xi\or\pi\or\rho
257 \or\sigma\or\tau\or\upsilon\or\phi\or\chi\or\psi\or\omega\fi$)}
258
259 %%% Miscellanea %%%
260
261 % {\I italic} with automatic italic correction
262 \def\I{\it\aftergroup\/}
263
264 % A breakable dash, to be repeated on the next line
265 \def\={\discretionary{-}{-}{-}}
266
267 % Non-breakable identifiers
268 \def\<#1>{\leavevmode\hbox{\I #1}}
269
270 % Handy shortcuts
271 \let\>=\noindent
272 \def\\{\hfil\break}
273
274 % Variants of \centerline, \leftline and \rightline, which are compatible with
275 % verbatim environments and other catcode hacks
276 \def\cline{\bgroup\def\linet@mp{\aftergroup\box\aftergroup0\aftergroup\egroup\hss\bgroup\aftergroup\hss\aftergroup\egroup}\afterassignment\linet@mp\setbox0\hbox to \hsize}
277 \def\lline{\bgroup\def\linet@mp{\aftergroup\box\aftergroup0\aftergroup\egroup\bgroup\aftergroup\hss\aftergroup\egroup}\afterassignment\linet@mp\setbox0\hbox to \hsize}
278 \def\rline{\bgroup\def\linet@mp{\aftergroup\box\aftergroup0\aftergroup\egroup\hss\bgroup\aftergroup\egroup}\afterassignment\linet@mp\setbox0\hbox to \hsize}
279
280 % Insert a PDF picture
281 % \putimage{width specification}{file}
282 \def\putimage#1#2{\hbox{\pdfximage #1{#2}\pdfrefximage\pdflastximage}}
283
284 % Let ~ be protected
285 \let\plaintilde=~
286 \protected\def~{\plaintilde}
287
288 %%% Epilog %%%
289
290 % Let's hide all internal macros
291 \catcode`@=12