2418
|
1 |
% Mathpartir --- Math Paragraph for Typesetting Inference Rules
|
|
2 |
%
|
|
3 |
% Copyright (C) 2001, 2002, 2003 Didier Rémy
|
|
4 |
%
|
|
5 |
% Author : Didier Remy
|
|
6 |
% Version : 1.1.1
|
|
7 |
% Bug Reports : to author
|
|
8 |
% Web Site : http://pauillac.inria.fr/~remy/latex/
|
|
9 |
%
|
|
10 |
% WhizzyTeX is free software; you can redistribute it and/or modify
|
|
11 |
% it under the terms of the GNU General Public License as published by
|
|
12 |
% the Free Software Foundation; either version 2, or (at your option)
|
|
13 |
% any later version.
|
|
14 |
%
|
|
15 |
% Mathpartir is distributed in the hope that it will be useful,
|
|
16 |
% but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
17 |
% MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
18 |
% GNU General Public License for more details
|
|
19 |
% (http://pauillac.inria.fr/~remy/license/GPL).
|
|
20 |
%
|
|
21 |
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
|
|
22 |
% File mathpartir.sty (LaTeX macros)
|
|
23 |
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
|
|
24 |
|
|
25 |
\NeedsTeXFormat{LaTeX2e}
|
|
26 |
\ProvidesPackage{mathpartir}
|
|
27 |
[2003/07/10 version 1.1.1 Math Paragraph for Typesetting Inference Rules]
|
|
28 |
|
|
29 |
%%
|
|
30 |
|
|
31 |
%% Identification
|
|
32 |
%% Preliminary declarations
|
|
33 |
|
|
34 |
\RequirePackage {keyval}
|
|
35 |
|
|
36 |
%% Options
|
|
37 |
%% More declarations
|
|
38 |
|
|
39 |
%% PART I: Typesetting maths in paragraphe mode
|
|
40 |
|
|
41 |
\newdimen \mpr@tmpdim
|
|
42 |
|
|
43 |
% To ensure hevea \hva compatibility, \hva should expands to nothing
|
|
44 |
% in mathpar or in inferrule
|
|
45 |
\let \mpr@hva \empty
|
|
46 |
|
|
47 |
%% normal paragraph parametters, should rather be taken dynamically
|
|
48 |
\def \mpr@savepar {%
|
|
49 |
\edef \MathparNormalpar
|
|
50 |
{\noexpand \lineskiplimit \the\lineskiplimit
|
|
51 |
\noexpand \lineskip \the\lineskip}%
|
|
52 |
}
|
|
53 |
|
|
54 |
\def \mpr@rulelineskip {\lineskiplimit=0.3em\lineskip=0.2em plus 0.1em}
|
|
55 |
\def \mpr@lesslineskip {\lineskiplimit=0.6em\lineskip=0.5em plus 0.2em}
|
|
56 |
\def \mpr@lineskip {\lineskiplimit=1.2em\lineskip=1.2em plus 0.2em}
|
|
57 |
\let \MathparLineskip \mpr@lineskip
|
|
58 |
\def \mpr@paroptions {\MathparLineskip}
|
|
59 |
\let \mpr@prebindings \relax
|
|
60 |
|
|
61 |
\newskip \mpr@andskip \mpr@andskip 2em plus 0.5fil minus 0.5em
|
|
62 |
|
|
63 |
\def \mpr@goodbreakand
|
|
64 |
{\hskip -\mpr@andskip \penalty -1000\hskip \mpr@andskip}
|
|
65 |
\def \mpr@and {\hskip \mpr@andskip}
|
|
66 |
\def \mpr@andcr {\penalty 50\mpr@and}
|
|
67 |
\def \mpr@cr {\penalty -10000\mpr@and}
|
|
68 |
\def \mpr@eqno #1{\mpr@andcr #1\hskip 0em plus -1fil \penalty 10}
|
|
69 |
|
|
70 |
\def \mpr@bindings {%
|
|
71 |
\let \and \mpr@andcr
|
|
72 |
\let \par \mpr@andcr
|
|
73 |
\let \\\mpr@cr
|
|
74 |
\let \eqno \mpr@eqno
|
|
75 |
\let \hva \mpr@hva
|
|
76 |
}
|
|
77 |
\let \MathparBindings \mpr@bindings
|
|
78 |
|
|
79 |
% \@ifundefined {ignorespacesafterend}
|
|
80 |
% {\def \ignorespacesafterend {\aftergroup \ignorespaces}
|
|
81 |
|
|
82 |
\newenvironment{mathpar}[1][]
|
|
83 |
{$$\mpr@savepar \parskip 0em \hsize \linewidth \centering
|
|
84 |
\vbox \bgroup \mpr@prebindings \mpr@paroptions #1\ifmmode $\else
|
|
85 |
\noindent $\displaystyle\fi
|
|
86 |
\MathparBindings}
|
|
87 |
{\unskip \ifmmode $\fi\egroup $$\ignorespacesafterend}
|
|
88 |
|
|
89 |
% \def \math@mathpar #1{\setbox0 \hbox {$\displaystyle #1$}\ifnum
|
|
90 |
% \wd0 < \hsize $$\box0$$\else \bmathpar #1\emathpar \fi}
|
|
91 |
|
|
92 |
%%% HOV BOXES
|
|
93 |
|
|
94 |
\def \mathvbox@ #1{\hbox \bgroup \mpr@normallineskip
|
|
95 |
\vbox \bgroup \tabskip 0em \let \+ \cr
|
|
96 |
\halign \bgroup \hfil $##$\hfil\cr #1\crcr \egroup \egroup
|
|
97 |
\egroup}
|
|
98 |
|
|
99 |
\def \mathhvbox@ #1{\setbox0 \hbox {\let \+\qquad $#1$}\ifnum \wd0 < \hsize
|
|
100 |
\box0\else \mathvbox {#1}\fi}
|
|
101 |
|
|
102 |
|
|
103 |
%% Part II -- operations on lists
|
|
104 |
|
|
105 |
\newtoks \mpr@lista
|
|
106 |
\newtoks \mpr@listb
|
|
107 |
|
|
108 |
\long \def\mpr@cons #1\mpr@to#2{\mpr@lista {\+{#1}}\mpr@listb \expandafter
|
|
109 |
{#2}\edef #2{\the \mpr@lista \the \mpr@listb}}
|
|
110 |
|
|
111 |
\long \def\mpr@snoc #1\mpr@to#2{\mpr@lista {\+{#1}}\mpr@listb \expandafter
|
|
112 |
{#2}\edef #2{\the \mpr@listb\the\mpr@lista}}
|
|
113 |
|
|
114 |
\long \def \mpr@concat#1=#2\mpr@to#3{\mpr@lista \expandafter {#2}\mpr@listb
|
|
115 |
\expandafter {#3}\edef #1{\the \mpr@listb\the\mpr@lista}}
|
|
116 |
|
|
117 |
\def \mpr@head #1\mpr@to #2{\expandafter \mpr@head@ #1\mpr@head@ #1#2}
|
|
118 |
\long \def \mpr@head@ #1#2\mpr@head@ #3#4{\def #4{#1}\def#3{#2}}
|
|
119 |
|
|
120 |
\def \mpr@flatten #1\mpr@to #2{\expandafter \mpr@flatten@ #1\mpr@flatten@ #1#2}
|
|
121 |
\long \def \mpr@flatten@ \+#1\+#2\mpr@flatten@ #3#4{\def #4{#1}\def #3{\+#2}}
|
|
122 |
|
|
123 |
\def \mpr@makelist #1\mpr@to #2{\def \mpr@all {#1}%
|
|
124 |
\mpr@lista {\+}\mpr@listb \expandafter {\mpr@all}\edef \mpr@all {\the
|
|
125 |
\mpr@lista \the \mpr@listb \the \mpr@lista}\let #2\empty
|
|
126 |
\def \mpr@stripof ##1##2\mpr@stripend{\def \mpr@stripped{##2}}\loop
|
|
127 |
\mpr@flatten \mpr@all \mpr@to \mpr@one
|
|
128 |
\expandafter \mpr@snoc \mpr@one \mpr@to #2\expandafter \mpr@stripof
|
|
129 |
\mpr@all \mpr@stripend
|
|
130 |
\ifx \mpr@stripped \empty \let \mpr@isempty 0\else \let \mpr@isempty 1\fi
|
|
131 |
\ifx 1\mpr@isempty
|
|
132 |
\repeat
|
|
133 |
}
|
|
134 |
|
|
135 |
%% Part III -- Type inference rules
|
|
136 |
|
|
137 |
\def \mpr@rev #1\mpr@to #2{\let \mpr@tmp \empty
|
|
138 |
\def \+##1{\mpr@cons ##1\mpr@to \mpr@tmp}#1\let #2\mpr@tmp}
|
|
139 |
|
|
140 |
\newif \if@premisse
|
|
141 |
\newbox \mpr@hlist
|
|
142 |
\newbox \mpr@vlist
|
|
143 |
\newif \ifmpr@center \mpr@centertrue
|
|
144 |
\def \mpr@htovlist {%
|
|
145 |
\setbox \mpr@hlist
|
|
146 |
\hbox {\strut
|
|
147 |
\ifmpr@center \hskip -0.5\wd\mpr@hlist\fi
|
|
148 |
\unhbox \mpr@hlist}%
|
|
149 |
\setbox \mpr@vlist
|
|
150 |
\vbox {\if@premisse \box \mpr@hlist \unvbox \mpr@vlist
|
|
151 |
\else \unvbox \mpr@vlist \box \mpr@hlist
|
|
152 |
\fi}%
|
|
153 |
}
|
|
154 |
% OLD version
|
|
155 |
% \def \mpr@htovlist {%
|
|
156 |
% \setbox \mpr@hlist
|
|
157 |
% \hbox {\strut \hskip -0.5\wd\mpr@hlist \unhbox \mpr@hlist}%
|
|
158 |
% \setbox \mpr@vlist
|
|
159 |
% \vbox {\if@premisse \box \mpr@hlist \unvbox \mpr@vlist
|
|
160 |
% \else \unvbox \mpr@vlist \box \mpr@hlist
|
|
161 |
% \fi}%
|
|
162 |
% }
|
|
163 |
|
|
164 |
\def \mpr@sep{2em}
|
|
165 |
\def \mpr@blank { }
|
|
166 |
\def \mpr@hovbox #1#2{\hbox
|
|
167 |
\bgroup
|
|
168 |
\ifx #1T\@premissetrue
|
|
169 |
\else \ifx #1B\@premissefalse
|
|
170 |
\else
|
|
171 |
\PackageError{mathpartir}
|
|
172 |
{Premisse orientation should either be P or B}
|
|
173 |
{Fatal error in Package}%
|
|
174 |
\fi \fi
|
|
175 |
\def \@test {#2}\ifx \@test \mpr@blank\else
|
|
176 |
\setbox \mpr@hlist \hbox {}%
|
|
177 |
\setbox \mpr@vlist \vbox {}%
|
|
178 |
\if@premisse \let \snoc \mpr@cons \else \let \snoc \mpr@snoc \fi
|
|
179 |
\let \@hvlist \empty \let \@rev \empty
|
|
180 |
\mpr@tmpdim 0em
|
|
181 |
\expandafter \mpr@makelist #2\mpr@to \mpr@flat
|
|
182 |
\if@premisse \mpr@rev \mpr@flat \mpr@to \@rev \else \let \@rev \mpr@flat \fi
|
|
183 |
\def \+##1{%
|
|
184 |
\def \@test {##1}\ifx \@test \empty
|
|
185 |
\mpr@htovlist
|
|
186 |
\mpr@tmpdim 0em %%% last bug fix not extensively checked
|
|
187 |
\else
|
|
188 |
\setbox0 \hbox{$\displaystyle {##1}$}\relax
|
|
189 |
\advance \mpr@tmpdim by \wd0
|
|
190 |
%\mpr@tmpdim 1.02\mpr@tmpdim
|
|
191 |
\ifnum \mpr@tmpdim < \hsize
|
|
192 |
\ifnum \wd\mpr@hlist > 0
|
|
193 |
\if@premisse
|
|
194 |
\setbox \mpr@hlist
|
|
195 |
\hbox {\unhbox0 \hskip \mpr@sep \unhbox \mpr@hlist}%
|
|
196 |
\else
|
|
197 |
\setbox \mpr@hlist
|
|
198 |
\hbox {\unhbox \mpr@hlist \hskip \mpr@sep \unhbox0}%
|
|
199 |
\fi
|
|
200 |
\else
|
|
201 |
\setbox \mpr@hlist \hbox {\unhbox0}%
|
|
202 |
\fi
|
|
203 |
\else
|
|
204 |
\ifnum \wd \mpr@hlist > 0
|
|
205 |
\mpr@htovlist
|
|
206 |
\mpr@tmpdim \wd0
|
|
207 |
\fi
|
|
208 |
\setbox \mpr@hlist \hbox {\unhbox0}%
|
|
209 |
\fi
|
|
210 |
\advance \mpr@tmpdim by \mpr@sep
|
|
211 |
\fi
|
|
212 |
}%
|
|
213 |
\@rev
|
|
214 |
\mpr@htovlist
|
|
215 |
\ifmpr@center \hskip \wd\mpr@vlist\fi \box \mpr@vlist
|
|
216 |
\fi
|
|
217 |
\egroup
|
|
218 |
}
|
|
219 |
|
|
220 |
%%% INFERENCE RULES
|
|
221 |
|
|
222 |
\@ifundefined{@@over}{%
|
|
223 |
\let\@@over\over % fallback if amsmath is not loaded
|
|
224 |
\let\@@overwithdelims\overwithdelims
|
|
225 |
\let\@@atop\atop \let\@@atopwithdelims\atopwithdelims
|
|
226 |
\let\@@above\above \let\@@abovewithdelims\abovewithdelims
|
|
227 |
}{}
|
|
228 |
|
|
229 |
|
|
230 |
\def \mpr@@fraction #1#2{\hbox {\advance \hsize by -0.5em
|
|
231 |
$\displaystyle {#1\@@over #2}$}}
|
|
232 |
\let \mpr@fraction \mpr@@fraction
|
|
233 |
\def \mpr@@reduce #1#2{\hbox
|
|
234 |
{$\lower 0.01pt \mpr@@fraction {#1}{#2}\mkern -15mu\rightarrow$}}
|
|
235 |
\def \mpr@@rewrite #1#2#3{\hbox
|
|
236 |
{$\lower 0.01pt \mpr@@fraction {#2}{#3}\mkern -8mu#1$}}
|
|
237 |
\def \mpr@infercenter #1{\vcenter {\mpr@hovbox{T}{#1}}}
|
|
238 |
|
|
239 |
\def \mpr@empty {}
|
|
240 |
\def \mpr@inferrule
|
|
241 |
{\bgroup
|
|
242 |
\ifnum \linewidth<\hsize \hsize \linewidth\fi
|
|
243 |
\mpr@rulelineskip
|
|
244 |
\let \and \qquad
|
|
245 |
\let \hva \mpr@hva
|
|
246 |
\let \@rulename \mpr@empty
|
|
247 |
\let \@rule@options \mpr@empty
|
|
248 |
\mpr@inferrule@}
|
|
249 |
\newcommand {\mpr@inferrule@}[3][]
|
|
250 |
{\everymath={\displaystyle}%
|
|
251 |
\def \@test {#2}\ifx \empty \@test
|
|
252 |
\setbox0 \hbox {$\vcenter {\mpr@hovbox{B}{#3}}$}%
|
|
253 |
\else
|
|
254 |
\def \@test {#3}\ifx \empty \@test
|
|
255 |
\setbox0 \hbox {$\vcenter {\mpr@hovbox{T}{#2}}$}%
|
|
256 |
\else
|
|
257 |
\setbox0 \mpr@fraction {\mpr@hovbox{T}{#2}}{\mpr@hovbox{B}{#3}}%
|
|
258 |
\fi \fi
|
|
259 |
\def \@test {#1}\ifx \@test\empty \box0
|
|
260 |
\else \vbox
|
|
261 |
%%% Suggestion de Francois pour les etiquettes longues
|
|
262 |
%%% {\hbox to \wd0 {\RefTirName {#1}\hfil}\box0}\fi
|
|
263 |
{\hbox {\RefTirName {#1}}\box0}\fi
|
|
264 |
\egroup}
|
|
265 |
|
|
266 |
\def \mpr@vdotfil #1{\vbox to #1{\leaders \hbox{$\cdot$} \vfil}}
|
|
267 |
|
|
268 |
% They are two forms
|
|
269 |
% \inferrule [label]{[premisses}{conclusions}
|
|
270 |
% or
|
|
271 |
% \inferrule* [options]{[premisses}{conclusions}
|
|
272 |
%
|
|
273 |
% Premisses and conclusions are lists of elements separated by \+
|
|
274 |
% Each \+ produces a break, attempting horizontal breaks if possible,
|
|
275 |
% and vertical breaks if needed.
|
|
276 |
%
|
|
277 |
% An empty element obtained by \+\+ produces a vertical break in all cases.
|
|
278 |
%
|
|
279 |
% The former rule is aligned on the fraction bar.
|
|
280 |
% The optional label appears on top of the rule
|
|
281 |
% The second form to be used in a derivation tree is aligned on the last
|
|
282 |
% line of its conclusion
|
|
283 |
%
|
|
284 |
% The second form can be parameterized, using the key=val interface. The
|
|
285 |
% folloiwng keys are recognized:
|
|
286 |
%
|
|
287 |
% width set the width of the rule to val
|
|
288 |
% narrower set the width of the rule to val\hsize
|
|
289 |
% before execute val at the beginning/left
|
|
290 |
% lab put a label [Val] on top of the rule
|
|
291 |
% lskip add negative skip on the right
|
|
292 |
% left put a left label [Val]
|
|
293 |
% Left put a left label [Val], ignoring its width
|
|
294 |
% right put a right label [Val]
|
|
295 |
% Right put a right label [Val], ignoring its width
|
|
296 |
% leftskip skip negative space on the left-hand side
|
|
297 |
% rightskip skip negative space on the right-hand side
|
|
298 |
% vdots lift the rule by val and fill vertical space with dots
|
|
299 |
% after execute val at the end/right
|
|
300 |
%
|
|
301 |
% Note that most options must come in this order to avoid strange
|
|
302 |
% typesetting (in particular leftskip must preceed left and Left and
|
|
303 |
% rightskip must follow Right or right; vdots must come last
|
|
304 |
% or be only followed by rightskip.
|
|
305 |
%
|
|
306 |
|
|
307 |
\define@key {mprset}{flushleft}[]{\mpr@centerfalse}
|
|
308 |
\define@key {mprset}{center}[]{\mpr@centertrue}
|
|
309 |
\def \mprset #1{\setkeys{mprset}{#1}}
|
|
310 |
|
|
311 |
\newbox \mpr@right
|
|
312 |
\define@key {mpr}{flushleft}[]{\mpr@centerfalse}
|
|
313 |
\define@key {mpr}{center}[]{\mpr@centertrue}
|
|
314 |
\define@key {mpr}{left}{\setbox0 \hbox {$\TirName {#1}\;$}\relax
|
|
315 |
\advance \hsize by -\wd0\box0}
|
|
316 |
\define@key {mpr}{width}{\hsize #1}
|
|
317 |
\define@key {mpr}{sep}{\def\mpr@sep{#1}}
|
|
318 |
\define@key {mpr}{before}{#1}
|
|
319 |
\define@key {mpr}{lab}{\let \RefTirName \TirName \def \mpr@rulename {#1}}
|
|
320 |
\define@key {mpr}{Lab}{\let \RefTirName \TirName \def \mpr@rulename {#1}}
|
|
321 |
\define@key {mpr}{narrower}{\hsize #1\hsize}
|
|
322 |
\define@key {mpr}{leftskip}{\hskip -#1}
|
|
323 |
\define@key {mpr}{reduce}[]{\let \mpr@fraction \mpr@@reduce}
|
|
324 |
\define@key {mpr}{rightskip}
|
|
325 |
{\setbox \mpr@right \hbox {\unhbox \mpr@right \hskip -#1}}
|
|
326 |
\define@key {mpr}{LEFT}{\setbox0 \hbox {$#1$}\relax
|
|
327 |
\advance \hsize by -\wd0\box0}
|
|
328 |
\define@key {mpr}{left}{\setbox0 \hbox {$\TirName {#1}\;$}\relax
|
|
329 |
\advance \hsize by -\wd0\box0}
|
|
330 |
\define@key {mpr}{Left}{\llap{$\TirName {#1}\;$}}
|
|
331 |
\define@key {mpr}{right}
|
|
332 |
{\setbox0 \hbox {$\;\TirName {#1}$}\relax \advance \hsize by -\wd0
|
|
333 |
\setbox \mpr@right \hbox {\unhbox \mpr@right \unhbox0}}
|
|
334 |
\define@key {mpr}{RIGHT}
|
|
335 |
{\setbox0 \hbox {$#1$}\relax \advance \hsize by -\wd0
|
|
336 |
\setbox \mpr@right \hbox {\unhbox \mpr@right \unhbox0}}
|
|
337 |
\define@key {mpr}{Right}
|
|
338 |
{\setbox \mpr@right \hbox {\unhbox \mpr@right \rlap {$\;\TirName {#1}$}}}
|
|
339 |
\define@key {mpr}{vdots}{\def \mpr@vdots {\@@atop \mpr@vdotfil{#1}}}
|
|
340 |
\define@key {mpr}{after}{\edef \mpr@after {\mpr@after #1}}
|
|
341 |
|
|
342 |
\newdimen \rule@dimen
|
|
343 |
\newcommand \mpr@inferstar@ [3][]{\setbox0
|
|
344 |
\hbox {\let \mpr@rulename \mpr@empty \let \mpr@vdots \relax
|
|
345 |
\setbox \mpr@right \hbox{}%
|
|
346 |
$\setkeys{mpr}{#1}%
|
|
347 |
\ifx \mpr@rulename \mpr@empty \mpr@inferrule {#2}{#3}\else
|
|
348 |
\mpr@inferrule [{\mpr@rulename}]{#2}{#3}\fi
|
|
349 |
\box \mpr@right \mpr@vdots$}
|
|
350 |
\setbox1 \hbox {\strut}
|
|
351 |
\rule@dimen \dp0 \advance \rule@dimen by -\dp1
|
|
352 |
\raise \rule@dimen \box0}
|
|
353 |
|
|
354 |
\def \mpr@infer {\@ifnextchar *{\mpr@inferstar}{\mpr@inferrule}}
|
|
355 |
\newcommand \mpr@err@skipargs[3][]{}
|
|
356 |
\def \mpr@inferstar*{\ifmmode
|
|
357 |
\let \@do \mpr@inferstar@
|
|
358 |
\else
|
|
359 |
\let \@do \mpr@err@skipargs
|
|
360 |
\PackageError {mathpartir}
|
|
361 |
{\string\inferrule* can only be used in math mode}{}%
|
|
362 |
\fi \@do}
|
|
363 |
|
|
364 |
|
|
365 |
%%% Exports
|
|
366 |
|
|
367 |
% Envirnonment mathpar
|
|
368 |
|
|
369 |
\let \inferrule \mpr@infer
|
|
370 |
|
|
371 |
% make a short name \infer is not already defined
|
|
372 |
\@ifundefined {infer}{\let \infer \mpr@infer}{}
|
|
373 |
|
|
374 |
\def \tir@name #1{\hbox {\small \sc #1}}
|
|
375 |
\let \TirName \tir@name
|
|
376 |
\let \RefTirName \tir@name
|
|
377 |
|
|
378 |
%%% Other Exports
|
|
379 |
|
|
380 |
% \let \listcons \mpr@cons
|
|
381 |
% \let \listsnoc \mpr@snoc
|
|
382 |
% \let \listhead \mpr@head
|
|
383 |
% \let \listmake \mpr@makelist
|
|
384 |
|
|
385 |
|
|
386 |
|
|
387 |
|
|
388 |
\endinput
|