-
Notifications
You must be signed in to change notification settings - Fork 1
/
kbordermatrix.sty
151 lines (130 loc) · 5.39 KB
/
kbordermatrix.sty
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
\NeedsTeXFormat{LaTeX2e}
\ProvidesPackage{kbordermatrix}[2011/09/21 Bordered matrix with brackets]
% Author: Kim C Border <[email protected]>
% Date: SuperBowl XXXVII (Go Bucs)
% Revised 2003/09/20
% to allow flush right option.
% Revised 2011/09/21
% at urging of Bruno Calfa (CMU)
% to coexist with package arydshln
% by adding \def\@xarraycr ...
% Defines \kbordermatrix along the lines of plain tex's
% \bordermatrix (which is still available in LaTeX).
% In particular, as with \bordermatrix,
% 1. It takes the array as an argument. It does not use \begin{}..\end{}.
% Is this a feature or a bug?
% 2. The first row is spaced a bit further apart from the rest.
% 3. The lower (n-1) by (n-1) block is set off by delimiters.
% 4. There is an invisible bottom row of the same height as the segregated
% top row that adds to the height of the equation.
% Differences from \bordermatrix:
% 1. Square brackets are used in place of parentheses.
% 2. You may use \\ instead of \cr.
% 3. The line heights agree with LaTeX's line heights for the
% array environment, and \arraystretch is respected. This means
% the bottom (n-1) rows align with the rows of an (n-1)-rowed
% \begin{array}..\end{array} (with or without delimiters).
% 4. All columns are centered.
% ** Modified 2003-9-20 to allow flush right option.
% 5. The first column is spaced a bit further apart from the rest.
% Differences from \left\[\begin{array}...\end{array}\right\]
% 1. It takes the array as an argument. It does not use \begin{}..\end{}.
% Is this a feature or a bug?
% 2. Consequently, you cannot use a column specifier (e.g., {l|cr}).
% 3. Consequently the maximum number of columns is not specified.
% 4. Vertical rules must be put in each row in a separate column.
% 5. You can use \hline and \cline.
% At least it works in the cases I have tried, but I offer no guarantees.
% cf. \bordermatrix p. 361, and \vrulealign p. 392 of The TeXbook
% Style parameters, they may be redefined according to taste.
\newcommand{\kbldelim}{[} % Left delimiter
\newcommand{\kbrdelim}{]}% Right delimiter
\newcommand{\kbrowstyle}{\scriptstyle}% Style applied to first row
\newcommand{\kbcolstyle}{\scriptstyle}% Style applied to first column
\newlength{\kbcolsep} % Extra separation after first border column
\newlength{\kbrowsep} % Extra separation after first border row
\setlength{\kbcolsep}{.5\arraycolsep}
\setlength{\kbrowsep}{.2ex}
\newif\ifkbalignright
% Scratch lengths (to be computed)
\newlength{\br@kwd} % Width of delimiter
\newlength{\k@bordht} % Height of border column
% This is it
\newcommand{\kbordermatrix}[1]{%
\begingroup
% \br@kwd depends on font size, so compute it now.
\setbox0=\hbox{$\left\kbldelim\right.$}
\setlength{\br@kwd}{\wd0}
% Compute the array strut based on current value of \arraystretch.
\setbox\@arstrutbox\hbox{\vrule
\@height\arraystretch\ht\strutbox
\@depth\arraystretch\dp\strutbox
\@width\z@}
% Compute height of first row and extra space.
\setlength{\k@bordht}{\kbrowsep}
\addtolength{\k@bordht}{\ht\@arstrutbox}
\addtolength{\k@bordht}{\dp\@arstrutbox}
% turn off mathsurround
\m@th
% Set the first row style
\def\@kbrowstyle{\kbrowstyle}
% Swallow the alignment into box0:
\setbox0=\vbox{%
% Define \cr for first row to include the \kbrowsep
% and to reset the row style
\def\cr{\crcr\noalign{\kern\kbrowsep
\global\let\cr=\endline
\global\let\@kbrowstyle=\relax}}
% Redefine \\ a la LaTeX:
\let\\\@arraycr
% The following are needed to make a solid \vrule with no gaps
% between the lines.
\lineskip\z@skip
\baselineskip\z@skip
% Compute the length of the skip after the first column
\dimen0\kbcolsep \advance\dimen0\br@kwd
% Here begins the alignment:
\ialign{\tabskip\dimen0 % This space will show up after the first column
\kern\arraycolsep\hfil\@arstrut$\kbcolstyle ##$\hfil\kern\arraycolsep&
\tabskip\z@skip % Cancel extra space for other columns
\kern\arraycolsep\hfil$\@kbrowstyle ##$\ifkbalignright\relax\else\hfil\fi\kern\arraycolsep&&
\kern\arraycolsep\hfil$\@kbrowstyle ##$\ifkbalignright\relax\else\hfil\fi\kern\arraycolsep\crcr
% That ends the template.
% Here is the argument:
#1\crcr}% End \ialign
}% End \setbox0.
% \box0 now holds the array.
%
% This next line uses \box2 to hold a throwaway
% copy of \box0, leaving \box0 intact,
% while putting the last row in \box5.
\setbox2=\vbox{\unvcopy0 \global\setbox5=\lastbox}
% We want the width of the first column,
% so we lop off columns until there is only one left.
% It's not elegant or efficient, but at 1 gHz, who cares.
\loop
\setbox2=\hbox{\unhbox5 \unskip \global\setbox3=\lastbox}
\ifhbox3
\global\setbox5=\box2
\global\setbox1=\box3
\repeat
% \box1 now holds the first column of last row.
%
% This next line stores the alignment in \box2,
% while calculating the proper
% delimiter height and placement.
\setbox2=\hbox{$\kern\wd1\kern\kbcolsep\kern-\arraycolsep
\left\kbldelim
\kern-\wd1\kern-\kbcolsep\kern-\br@kwd
%
% Here is the output. The \vcenter aligns the array with the "math axis."
% The negative vertical \kern only shrinks the delimiter's height.
% BTW, I didn't find this in the TeXbook,
% I had to try various \kerns to see what they did in a
% \left[\vcenter{}\right].
\vcenter{\kern-\k@bordht\vbox{\unvbox0}}
\right\kbrdelim$}
\null\vbox{\kern\k@bordht\box2}
%
\endgroup
}