* src/include/nonposix.h: Remove first line -- this file is used in C also.
[s-roff.git] / src / preproc / eqn / delim.cc
blob29deded383431461e58e1afadac18a0f429a5c06
1 // -*- C++ -*-
2 /* Copyright (C) 1989, 1990, 1991, 1992 Free Software Foundation, Inc.
3 Written by James Clark (jjc@jclark.com)
5 This file is part of groff.
7 groff is free software; you can redistribute it and/or modify it under
8 the terms of the GNU General Public License as published by the Free
9 Software Foundation; either version 2, or (at your option) any later
10 version.
12 groff is distributed in the hope that it will be useful, but WITHOUT ANY
13 WARRANTY; without even the implied warranty of MERCHANTABILITY or
14 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
15 for more details.
17 You should have received a copy of the GNU General Public License along
18 with groff; see the file COPYING. If not, write to the Free Software
19 Foundation, 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
21 #include "eqn.h"
22 #include "pbox.h"
24 enum left_or_right_t { LEFT_DELIM = 01, RIGHT_DELIM = 02 };
26 // Small must be none-zero and must exist in each device.
27 // Small will be put in the roman font, others are assumed to be
28 // on the special font (so no font change will be necessary.)
30 struct delimiter {
31 const char *name;
32 int flags;
33 const char *small;
34 const char *chain_format;
35 const char *ext;
36 const char *top;
37 const char *mid;
38 const char *bot;
39 } delim_table[] = {
41 "(", LEFT_DELIM|RIGHT_DELIM, "(", "\\[parenleft%s]",
42 "\\[parenleftex]",
43 "\\[parenlefttp]",
45 "\\[parenleftbt]",
48 ")", LEFT_DELIM|RIGHT_DELIM, ")", "\\[parenright%s]",
49 "\\[parenrightex]",
50 "\\[parenrighttp]",
52 "\\[parenrightbt]",
55 "[", LEFT_DELIM|RIGHT_DELIM, "[", "\\[bracketleft%s]",
56 "\\[bracketleftex]",
57 "\\[bracketlefttp]",
59 "\\[bracketleftbt]",
62 "]", LEFT_DELIM|RIGHT_DELIM, "]", "\\[bracketright%s]",
63 "\\[bracketrightex]",
64 "\\[bracketrighttp]",
66 "\\[bracketrightbt]",
69 "{", LEFT_DELIM|RIGHT_DELIM, "{", "\\[braceleft%s]",
70 "\\[braceleftex]",
71 "\\[bracelefttp]",
72 "\\[braceleftmid]",
73 "\\[braceleftbt]",
76 "}", LEFT_DELIM|RIGHT_DELIM, "}", "\\[braceright%s]",
77 "\\[bracerightex]",
78 "\\[bracerighttp]",
79 "\\[bracerightmid]",
80 "\\[bracerightbt]",
83 "|", LEFT_DELIM|RIGHT_DELIM, "|", "\\[bar%s]",
84 "\\[barex]",
87 "floor", LEFT_DELIM, "\\(lf", "\\[floorleft%s]",
88 "\\[bracketleftex]",
91 "\\[bracketleftbt]",
94 "floor", RIGHT_DELIM, "\\(rf", "\\[floorright%s]",
95 "\\[bracketrightex]",
98 "\\[bracketrightbt]",
101 "ceiling", LEFT_DELIM, "\\(lc", "\\[ceilingleft%s]",
102 "\\[bracketleftex]",
103 "\\[bracketlefttp]",
106 "ceiling", RIGHT_DELIM, "\\(rc", "\\[ceilingright%s]",
107 "\\[bracketrightex]",
108 "\\[bracketrighttp]",
111 "||", LEFT_DELIM|RIGHT_DELIM, "|", "\\[bar%s]",
112 "\\[bardblex]",
115 "<", LEFT_DELIM|RIGHT_DELIM, "\\(la", "\\[angleleft%s]",
118 ">", LEFT_DELIM|RIGHT_DELIM, "\\(ra", "\\[angleright%s]",
121 "uparrow", LEFT_DELIM|RIGHT_DELIM, "\\(ua", "\\[arrowup%s]",
122 "\\[arrowvertex]",
123 "\\[arrowverttp]",
126 "downarrow", LEFT_DELIM|RIGHT_DELIM, "\\(da", "\\[arrowdown%s]",
127 "\\[arrowvertex]",
130 "\\[arrowvertbt]",
133 "updownarrow", LEFT_DELIM|RIGHT_DELIM, "\\(va", "\\[arrowupdown%s]",
134 "\\[arrowvertex]",
135 "\\[arrowverttp]",
137 "\\[arrowvertbt]",
141 const int DELIM_TABLE_SIZE = int(sizeof(delim_table)/sizeof(delim_table[0]));
143 class delim_box : public box {
144 private:
145 char *left;
146 char *right;
147 box *p;
148 public:
149 delim_box(char *, box *, char *);
150 ~delim_box();
151 int compute_metrics(int);
152 void output();
153 void check_tabs(int);
154 void debug_print();
157 box *make_delim_box(char *l, box *pp, char *r)
159 if (l != 0 && *l == '\0') {
160 a_delete l;
161 l = 0;
163 if (r != 0 && *r == '\0') {
164 a_delete r;
165 r = 0;
167 return new delim_box(l, pp, r);
170 delim_box::delim_box(char *l, box *pp, char *r)
171 : left(l), right(r), p(pp)
175 delim_box::~delim_box()
177 a_delete left;
178 a_delete right;
179 delete p;
182 static void build_extensible(const char *ext, const char *top, const char *mid,
183 const char *bot)
185 assert(ext != 0);
186 printf(".nr " DELIM_WIDTH_REG " 0\\w" DELIMITER_CHAR "%s" DELIMITER_CHAR "\n",
187 ext);
188 printf(".nr " EXT_HEIGHT_REG " 0\\n[rst]\n");
189 printf(".nr " EXT_DEPTH_REG " 0-\\n[rsb]\n");
190 if (top) {
191 printf(".nr " DELIM_WIDTH_REG " 0\\n[" DELIM_WIDTH_REG "]"
192 ">?\\w" DELIMITER_CHAR "%s" DELIMITER_CHAR "\n",
193 top);
194 printf(".nr " TOP_HEIGHT_REG " 0\\n[rst]\n");
195 printf(".nr " TOP_DEPTH_REG " 0-\\n[rsb]\n");
197 if (mid) {
198 printf(".nr " DELIM_WIDTH_REG " 0\\n[" DELIM_WIDTH_REG "]"
199 ">?\\w" DELIMITER_CHAR "%s" DELIMITER_CHAR "\n",
200 mid);
201 printf(".nr " MID_HEIGHT_REG " 0\\n[rst]\n");
202 printf(".nr " MID_DEPTH_REG " 0-\\n[rsb]\n");
204 if (bot) {
205 printf(".nr " DELIM_WIDTH_REG " 0\\n[" DELIM_WIDTH_REG "]"
206 ">?\\w" DELIMITER_CHAR "%s" DELIMITER_CHAR "\n",
207 bot);
208 printf(".nr " BOT_HEIGHT_REG " 0\\n[rst]\n");
209 printf(".nr " BOT_DEPTH_REG " 0-\\n[rsb]\n");
211 printf(".nr " TOTAL_HEIGHT_REG " 0");
212 if (top)
213 printf("+\\n[" TOP_HEIGHT_REG "]+\\n[" TOP_DEPTH_REG "]");
214 if (bot)
215 printf("+\\n[" BOT_HEIGHT_REG "]+\\n[" BOT_DEPTH_REG "]");
216 if (mid)
217 printf("+\\n[" MID_HEIGHT_REG "]+\\n[" MID_DEPTH_REG "]");
218 printf("\n");
219 // determine how many extensible characters we need
220 printf(".nr " TEMP_REG " \\n[" DELTA_REG "]-\\n[" TOTAL_HEIGHT_REG "]");
221 if (mid)
222 printf("/2");
223 printf(">?0+\\n[" EXT_HEIGHT_REG "]+\\n[" EXT_DEPTH_REG "]-1/(\\n["
224 EXT_HEIGHT_REG "]+\\n[" EXT_DEPTH_REG "])\n");
226 printf(".nr " TOTAL_HEIGHT_REG " +(\\n[" EXT_HEIGHT_REG "]+\\n["
227 EXT_DEPTH_REG "]*\\n[" TEMP_REG "]");
228 if (mid)
229 printf("*2");
230 printf(")\n");
231 printf(".ds " DELIM_STRING " \\Z" DELIMITER_CHAR
232 "\\v'-%dM-(\\n[" TOTAL_HEIGHT_REG "]u/2u)'\n",
233 axis_height);
234 if (top)
235 printf(".as " DELIM_STRING " \\v'\\n[" TOP_HEIGHT_REG "]u'"
236 "\\Z" DELIMITER_CHAR "%s" DELIMITER_CHAR
237 "\\v'\\n[" TOP_DEPTH_REG "]u'\n",
238 top);
240 // this macro appends $2 copies of $3 to string $1
241 printf(".de " REPEAT_APPEND_STRING_MACRO "\n"
242 ".if \\\\$2 \\{.as \\\\$1 \"\\\\$3\n"
243 "." REPEAT_APPEND_STRING_MACRO " \\\\$1 \\\\$2-1 \"\\\\$3\"\n"
244 ".\\}\n"
245 "..\n");
247 printf("." REPEAT_APPEND_STRING_MACRO " " DELIM_STRING " \\n[" TEMP_REG "] "
248 "\\v'\\n[" EXT_HEIGHT_REG "]u'"
249 "\\Z" DELIMITER_CHAR "%s" DELIMITER_CHAR
250 "\\v'\\n[" EXT_DEPTH_REG "]u'\n",
251 ext);
253 if (mid) {
254 printf(".as " DELIM_STRING " \\v'\\n[" MID_HEIGHT_REG "]u'"
255 "\\Z" DELIMITER_CHAR "%s" DELIMITER_CHAR
256 "\\v'\\n[" MID_DEPTH_REG "]u'\n",
257 mid);
258 printf("." REPEAT_APPEND_STRING_MACRO " " DELIM_STRING
259 " \\n[" TEMP_REG "] "
260 "\\v'\\n[" EXT_HEIGHT_REG "]u'"
261 "\\Z" DELIMITER_CHAR "%s" DELIMITER_CHAR
262 "\\v'\\n[" EXT_DEPTH_REG "]u'\n",
263 ext);
265 if (bot)
266 printf(".as " DELIM_STRING " \\v'\\n[" BOT_HEIGHT_REG "]u'"
267 "\\Z" DELIMITER_CHAR "%s" DELIMITER_CHAR
268 "\\v'\\n[" BOT_DEPTH_REG "]u'\n",
269 bot);
270 printf(".as " DELIM_STRING " " DELIMITER_CHAR "\n");
273 static void define_extensible_string(char *delim, int uid,
274 left_or_right_t left_or_right)
276 printf(".ds " DELIM_STRING "\n");
277 delimiter *d = delim_table;
278 int delim_len = strlen(delim);
279 int i;
280 for (i = 0; i < DELIM_TABLE_SIZE; i++, d++)
281 if (strncmp(delim, d->name, delim_len) == 0
282 && (left_or_right & d->flags) != 0)
283 break;
284 if (i >= DELIM_TABLE_SIZE) {
285 error("there is no `%1' delimiter", delim);
286 printf(".nr " DELIM_WIDTH_REG " 0\n");
287 return;
290 printf(".nr " DELIM_WIDTH_REG " 0\\w" DELIMITER_CHAR "\\f[%s]%s\\fP" DELIMITER_CHAR "\n"
291 ".ds " DELIM_STRING " \\Z" DELIMITER_CHAR
292 "\\v'\\n[rsb]u+\\n[rst]u/2u-%dM'\\f[%s]%s\\fP" DELIMITER_CHAR "\n"
293 ".nr " TOTAL_HEIGHT_REG " \\n[rst]-\\n[rsb]\n"
294 ".if \\n[" TOTAL_HEIGHT_REG "]<\\n[" DELTA_REG "] "
295 "\\{",
296 current_roman_font, d->small, axis_height,
297 current_roman_font, d->small);
299 char buf[256];
300 sprintf(buf, d->chain_format, "\\\\n[" INDEX_REG "]");
301 printf(".nr " INDEX_REG " 0\n"
302 ".de " TEMP_MACRO "\n"
303 ".ie c%s \\{\\\n"
304 ".nr " DELIM_WIDTH_REG " 0\\w" DELIMITER_CHAR "%s" DELIMITER_CHAR "\n"
305 ".ds " DELIM_STRING " \\Z" DELIMITER_CHAR
306 "\\v'\\\\n[rsb]u+\\\\n[rst]u/2u-%dM'%s" DELIMITER_CHAR "\n"
307 ".nr " TOTAL_HEIGHT_REG " \\\\n[rst]-\\\\n[rsb]\n"
308 ".if \\\\n[" TOTAL_HEIGHT_REG "]<\\n[" DELTA_REG "] "
309 "\\{.nr " INDEX_REG " +1\n"
310 "." TEMP_MACRO "\n"
311 ".\\}\\}\n"
312 ".el .nr " INDEX_REG " 0-1\n"
313 "..\n"
314 "." TEMP_MACRO "\n",
315 buf, buf, axis_height, buf);
316 if (d->ext) {
317 printf(".if \\n[" INDEX_REG "]<0 \\{.if c%s \\{\\\n", d->ext);
318 build_extensible(d->ext, d->top, d->mid, d->bot);
319 printf(".\\}\\}\n");
321 printf(".\\}\n");
322 printf(".as " DELIM_STRING " \\h'\\n[" DELIM_WIDTH_REG "]u'\n");
323 printf(".nr " WIDTH_FORMAT " +\\n[" DELIM_WIDTH_REG "]\n", uid);
324 printf(".nr " HEIGHT_FORMAT " \\n[" HEIGHT_FORMAT "]"
325 ">?(\\n[" TOTAL_HEIGHT_REG "]/2+%dM)\n",
326 uid, uid, axis_height);
327 printf(".nr " DEPTH_FORMAT " \\n[" DEPTH_FORMAT "]"
328 ">?(\\n[" TOTAL_HEIGHT_REG "]/2-%dM)\n",
329 uid, uid, axis_height);
332 int delim_box::compute_metrics(int style)
334 int r = p->compute_metrics(style);
335 printf(".nr " WIDTH_FORMAT " 0\\n[" WIDTH_FORMAT "]\n", uid, p->uid);
336 printf(".nr " HEIGHT_FORMAT " \\n[" HEIGHT_FORMAT "]\n", uid, p->uid);
337 printf(".nr " DEPTH_FORMAT " \\n[" DEPTH_FORMAT "]\n", uid, p->uid);
338 printf(".nr " DELTA_REG " \\n[" HEIGHT_FORMAT "]-%dM"
339 ">?(\\n[" DEPTH_FORMAT "]+%dM)\n",
340 p->uid, axis_height, p->uid, axis_height);
341 printf(".nr " DELTA_REG " 0\\n[" DELTA_REG "]*%d/500"
342 ">?(\\n[" DELTA_REG "]*2-%dM)\n",
343 delimiter_factor, delimiter_shortfall);
344 if (left) {
345 define_extensible_string(left, uid, LEFT_DELIM);
346 printf(".rn " DELIM_STRING " " LEFT_DELIM_STRING_FORMAT "\n",
347 uid);
348 if (r)
349 printf(".nr " MARK_REG " +\\n[" DELIM_WIDTH_REG "]\n");
351 if (right) {
352 define_extensible_string(right, uid, RIGHT_DELIM);
353 printf(".rn " DELIM_STRING " " RIGHT_DELIM_STRING_FORMAT "\n",
354 uid);
356 return r;
359 void delim_box::output()
361 if (left)
362 printf("\\*[" LEFT_DELIM_STRING_FORMAT "]", uid);
363 p->output();
364 if (right)
365 printf("\\*[" RIGHT_DELIM_STRING_FORMAT "]", uid);
368 void delim_box::check_tabs(int level)
370 p->check_tabs(level);
373 void delim_box::debug_print()
375 fprintf(stderr, "left \"%s\" { ", left ? left : "");
376 p->debug_print();
377 fprintf(stderr, " }");
378 if (right)
379 fprintf(stderr, " right \"%s\"", right);