xref: /illumos-gate/usr/src/cmd/mandoc/mdoc_html.c (revision 1702cce751c5cb7ead878d0205a6c90b027e3de8)
1 /*	$Id: mdoc_html.c,v 1.294 2017/07/15 17:57:51 schwarze Exp $ */
2 /*
3  * Copyright (c) 2008-2011, 2014 Kristaps Dzonsons <kristaps@bsd.lv>
4  * Copyright (c) 2014, 2015, 2016, 2017 Ingo Schwarze <schwarze@openbsd.org>
5  *
6  * Permission to use, copy, modify, and distribute this software for any
7  * purpose with or without fee is hereby granted, provided that the above
8  * copyright notice and this permission notice appear in all copies.
9  *
10  * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHORS DISCLAIM ALL WARRANTIES
11  * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
12  * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR
13  * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
14  * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
15  * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
16  * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
17  */
18 #include "config.h"
19 
20 #include <sys/types.h>
21 
22 #include <assert.h>
23 #include <ctype.h>
24 #include <stdio.h>
25 #include <stdlib.h>
26 #include <string.h>
27 #include <unistd.h>
28 
29 #include "mandoc_aux.h"
30 #include "mandoc.h"
31 #include "roff.h"
32 #include "mdoc.h"
33 #include "out.h"
34 #include "html.h"
35 #include "main.h"
36 
37 #define	INDENT		 5
38 
39 #define	MDOC_ARGS	  const struct roff_meta *meta, \
40 			  struct roff_node *n, \
41 			  struct html *h
42 
43 #ifndef MIN
44 #define	MIN(a,b)	((/*CONSTCOND*/(a)<(b))?(a):(b))
45 #endif
46 
47 struct	htmlmdoc {
48 	int		(*pre)(MDOC_ARGS);
49 	void		(*post)(MDOC_ARGS);
50 };
51 
52 static	char		 *cond_id(const struct roff_node *);
53 static	void		  print_mdoc_head(MDOC_ARGS);
54 static	void		  print_mdoc_node(MDOC_ARGS);
55 static	void		  print_mdoc_nodelist(MDOC_ARGS);
56 static	void		  synopsis_pre(struct html *,
57 				const struct roff_node *);
58 
59 static	void		  mdoc_root_post(MDOC_ARGS);
60 static	int		  mdoc_root_pre(MDOC_ARGS);
61 
62 static	void		  mdoc__x_post(MDOC_ARGS);
63 static	int		  mdoc__x_pre(MDOC_ARGS);
64 static	int		  mdoc_ad_pre(MDOC_ARGS);
65 static	int		  mdoc_an_pre(MDOC_ARGS);
66 static	int		  mdoc_ap_pre(MDOC_ARGS);
67 static	int		  mdoc_ar_pre(MDOC_ARGS);
68 static	int		  mdoc_bd_pre(MDOC_ARGS);
69 static	int		  mdoc_bf_pre(MDOC_ARGS);
70 static	void		  mdoc_bk_post(MDOC_ARGS);
71 static	int		  mdoc_bk_pre(MDOC_ARGS);
72 static	int		  mdoc_bl_pre(MDOC_ARGS);
73 static	int		  mdoc_cd_pre(MDOC_ARGS);
74 static	int		  mdoc_cm_pre(MDOC_ARGS);
75 static	int		  mdoc_d1_pre(MDOC_ARGS);
76 static	int		  mdoc_dv_pre(MDOC_ARGS);
77 static	int		  mdoc_fa_pre(MDOC_ARGS);
78 static	int		  mdoc_fd_pre(MDOC_ARGS);
79 static	int		  mdoc_fl_pre(MDOC_ARGS);
80 static	int		  mdoc_fn_pre(MDOC_ARGS);
81 static	int		  mdoc_ft_pre(MDOC_ARGS);
82 static	int		  mdoc_em_pre(MDOC_ARGS);
83 static	void		  mdoc_eo_post(MDOC_ARGS);
84 static	int		  mdoc_eo_pre(MDOC_ARGS);
85 static	int		  mdoc_er_pre(MDOC_ARGS);
86 static	int		  mdoc_ev_pre(MDOC_ARGS);
87 static	int		  mdoc_ex_pre(MDOC_ARGS);
88 static	void		  mdoc_fo_post(MDOC_ARGS);
89 static	int		  mdoc_fo_pre(MDOC_ARGS);
90 static	int		  mdoc_ic_pre(MDOC_ARGS);
91 static	int		  mdoc_igndelim_pre(MDOC_ARGS);
92 static	int		  mdoc_in_pre(MDOC_ARGS);
93 static	int		  mdoc_it_pre(MDOC_ARGS);
94 static	int		  mdoc_lb_pre(MDOC_ARGS);
95 static	int		  mdoc_li_pre(MDOC_ARGS);
96 static	int		  mdoc_lk_pre(MDOC_ARGS);
97 static	int		  mdoc_mt_pre(MDOC_ARGS);
98 static	int		  mdoc_ms_pre(MDOC_ARGS);
99 static	int		  mdoc_nd_pre(MDOC_ARGS);
100 static	int		  mdoc_nm_pre(MDOC_ARGS);
101 static	int		  mdoc_no_pre(MDOC_ARGS);
102 static	int		  mdoc_ns_pre(MDOC_ARGS);
103 static	int		  mdoc_pa_pre(MDOC_ARGS);
104 static	void		  mdoc_pf_post(MDOC_ARGS);
105 static	int		  mdoc_pp_pre(MDOC_ARGS);
106 static	void		  mdoc_quote_post(MDOC_ARGS);
107 static	int		  mdoc_quote_pre(MDOC_ARGS);
108 static	int		  mdoc_rs_pre(MDOC_ARGS);
109 static	int		  mdoc_sh_pre(MDOC_ARGS);
110 static	int		  mdoc_skip_pre(MDOC_ARGS);
111 static	int		  mdoc_sm_pre(MDOC_ARGS);
112 static	int		  mdoc_ss_pre(MDOC_ARGS);
113 static	int		  mdoc_st_pre(MDOC_ARGS);
114 static	int		  mdoc_sx_pre(MDOC_ARGS);
115 static	int		  mdoc_sy_pre(MDOC_ARGS);
116 static	int		  mdoc_va_pre(MDOC_ARGS);
117 static	int		  mdoc_vt_pre(MDOC_ARGS);
118 static	int		  mdoc_xr_pre(MDOC_ARGS);
119 static	int		  mdoc_xx_pre(MDOC_ARGS);
120 
121 static	const struct htmlmdoc __mdocs[MDOC_MAX - MDOC_Dd] = {
122 	{NULL, NULL}, /* Dd */
123 	{NULL, NULL}, /* Dt */
124 	{NULL, NULL}, /* Os */
125 	{mdoc_sh_pre, NULL }, /* Sh */
126 	{mdoc_ss_pre, NULL }, /* Ss */
127 	{mdoc_pp_pre, NULL}, /* Pp */
128 	{mdoc_d1_pre, NULL}, /* D1 */
129 	{mdoc_d1_pre, NULL}, /* Dl */
130 	{mdoc_bd_pre, NULL}, /* Bd */
131 	{NULL, NULL}, /* Ed */
132 	{mdoc_bl_pre, NULL}, /* Bl */
133 	{NULL, NULL}, /* El */
134 	{mdoc_it_pre, NULL}, /* It */
135 	{mdoc_ad_pre, NULL}, /* Ad */
136 	{mdoc_an_pre, NULL}, /* An */
137 	{mdoc_ap_pre, NULL}, /* Ap */
138 	{mdoc_ar_pre, NULL}, /* Ar */
139 	{mdoc_cd_pre, NULL}, /* Cd */
140 	{mdoc_cm_pre, NULL}, /* Cm */
141 	{mdoc_dv_pre, NULL}, /* Dv */
142 	{mdoc_er_pre, NULL}, /* Er */
143 	{mdoc_ev_pre, NULL}, /* Ev */
144 	{mdoc_ex_pre, NULL}, /* Ex */
145 	{mdoc_fa_pre, NULL}, /* Fa */
146 	{mdoc_fd_pre, NULL}, /* Fd */
147 	{mdoc_fl_pre, NULL}, /* Fl */
148 	{mdoc_fn_pre, NULL}, /* Fn */
149 	{mdoc_ft_pre, NULL}, /* Ft */
150 	{mdoc_ic_pre, NULL}, /* Ic */
151 	{mdoc_in_pre, NULL}, /* In */
152 	{mdoc_li_pre, NULL}, /* Li */
153 	{mdoc_nd_pre, NULL}, /* Nd */
154 	{mdoc_nm_pre, NULL}, /* Nm */
155 	{mdoc_quote_pre, mdoc_quote_post}, /* Op */
156 	{mdoc_ft_pre, NULL}, /* Ot */
157 	{mdoc_pa_pre, NULL}, /* Pa */
158 	{mdoc_ex_pre, NULL}, /* Rv */
159 	{mdoc_st_pre, NULL}, /* St */
160 	{mdoc_va_pre, NULL}, /* Va */
161 	{mdoc_vt_pre, NULL}, /* Vt */
162 	{mdoc_xr_pre, NULL}, /* Xr */
163 	{mdoc__x_pre, mdoc__x_post}, /* %A */
164 	{mdoc__x_pre, mdoc__x_post}, /* %B */
165 	{mdoc__x_pre, mdoc__x_post}, /* %D */
166 	{mdoc__x_pre, mdoc__x_post}, /* %I */
167 	{mdoc__x_pre, mdoc__x_post}, /* %J */
168 	{mdoc__x_pre, mdoc__x_post}, /* %N */
169 	{mdoc__x_pre, mdoc__x_post}, /* %O */
170 	{mdoc__x_pre, mdoc__x_post}, /* %P */
171 	{mdoc__x_pre, mdoc__x_post}, /* %R */
172 	{mdoc__x_pre, mdoc__x_post}, /* %T */
173 	{mdoc__x_pre, mdoc__x_post}, /* %V */
174 	{NULL, NULL}, /* Ac */
175 	{mdoc_quote_pre, mdoc_quote_post}, /* Ao */
176 	{mdoc_quote_pre, mdoc_quote_post}, /* Aq */
177 	{mdoc_xx_pre, NULL}, /* At */
178 	{NULL, NULL}, /* Bc */
179 	{mdoc_bf_pre, NULL}, /* Bf */
180 	{mdoc_quote_pre, mdoc_quote_post}, /* Bo */
181 	{mdoc_quote_pre, mdoc_quote_post}, /* Bq */
182 	{mdoc_xx_pre, NULL}, /* Bsx */
183 	{mdoc_xx_pre, NULL}, /* Bx */
184 	{mdoc_skip_pre, NULL}, /* Db */
185 	{NULL, NULL}, /* Dc */
186 	{mdoc_quote_pre, mdoc_quote_post}, /* Do */
187 	{mdoc_quote_pre, mdoc_quote_post}, /* Dq */
188 	{NULL, NULL}, /* Ec */ /* FIXME: no space */
189 	{NULL, NULL}, /* Ef */
190 	{mdoc_em_pre, NULL}, /* Em */
191 	{mdoc_eo_pre, mdoc_eo_post}, /* Eo */
192 	{mdoc_xx_pre, NULL}, /* Fx */
193 	{mdoc_ms_pre, NULL}, /* Ms */
194 	{mdoc_no_pre, NULL}, /* No */
195 	{mdoc_ns_pre, NULL}, /* Ns */
196 	{mdoc_xx_pre, NULL}, /* Nx */
197 	{mdoc_xx_pre, NULL}, /* Ox */
198 	{NULL, NULL}, /* Pc */
199 	{mdoc_igndelim_pre, mdoc_pf_post}, /* Pf */
200 	{mdoc_quote_pre, mdoc_quote_post}, /* Po */
201 	{mdoc_quote_pre, mdoc_quote_post}, /* Pq */
202 	{NULL, NULL}, /* Qc */
203 	{mdoc_quote_pre, mdoc_quote_post}, /* Ql */
204 	{mdoc_quote_pre, mdoc_quote_post}, /* Qo */
205 	{mdoc_quote_pre, mdoc_quote_post}, /* Qq */
206 	{NULL, NULL}, /* Re */
207 	{mdoc_rs_pre, NULL}, /* Rs */
208 	{NULL, NULL}, /* Sc */
209 	{mdoc_quote_pre, mdoc_quote_post}, /* So */
210 	{mdoc_quote_pre, mdoc_quote_post}, /* Sq */
211 	{mdoc_sm_pre, NULL}, /* Sm */
212 	{mdoc_sx_pre, NULL}, /* Sx */
213 	{mdoc_sy_pre, NULL}, /* Sy */
214 	{NULL, NULL}, /* Tn */
215 	{mdoc_xx_pre, NULL}, /* Ux */
216 	{NULL, NULL}, /* Xc */
217 	{NULL, NULL}, /* Xo */
218 	{mdoc_fo_pre, mdoc_fo_post}, /* Fo */
219 	{NULL, NULL}, /* Fc */
220 	{mdoc_quote_pre, mdoc_quote_post}, /* Oo */
221 	{NULL, NULL}, /* Oc */
222 	{mdoc_bk_pre, mdoc_bk_post}, /* Bk */
223 	{NULL, NULL}, /* Ek */
224 	{NULL, NULL}, /* Bt */
225 	{NULL, NULL}, /* Hf */
226 	{mdoc_em_pre, NULL}, /* Fr */
227 	{NULL, NULL}, /* Ud */
228 	{mdoc_lb_pre, NULL}, /* Lb */
229 	{mdoc_pp_pre, NULL}, /* Lp */
230 	{mdoc_lk_pre, NULL}, /* Lk */
231 	{mdoc_mt_pre, NULL}, /* Mt */
232 	{mdoc_quote_pre, mdoc_quote_post}, /* Brq */
233 	{mdoc_quote_pre, mdoc_quote_post}, /* Bro */
234 	{NULL, NULL}, /* Brc */
235 	{mdoc__x_pre, mdoc__x_post}, /* %C */
236 	{mdoc_skip_pre, NULL}, /* Es */
237 	{mdoc_quote_pre, mdoc_quote_post}, /* En */
238 	{mdoc_xx_pre, NULL}, /* Dx */
239 	{mdoc__x_pre, mdoc__x_post}, /* %Q */
240 	{mdoc__x_pre, mdoc__x_post}, /* %U */
241 	{NULL, NULL}, /* Ta */
242 };
243 static	const struct htmlmdoc *const mdocs = __mdocs - MDOC_Dd;
244 
245 
246 /*
247  * See the same function in mdoc_term.c for documentation.
248  */
249 static void
250 synopsis_pre(struct html *h, const struct roff_node *n)
251 {
252 
253 	if (NULL == n->prev || ! (NODE_SYNPRETTY & n->flags))
254 		return;
255 
256 	if (n->prev->tok == n->tok &&
257 	    MDOC_Fo != n->tok &&
258 	    MDOC_Ft != n->tok &&
259 	    MDOC_Fn != n->tok) {
260 		print_otag(h, TAG_BR, "");
261 		return;
262 	}
263 
264 	switch (n->prev->tok) {
265 	case MDOC_Fd:
266 	case MDOC_Fn:
267 	case MDOC_Fo:
268 	case MDOC_In:
269 	case MDOC_Vt:
270 		print_paragraph(h);
271 		break;
272 	case MDOC_Ft:
273 		if (MDOC_Fn != n->tok && MDOC_Fo != n->tok) {
274 			print_paragraph(h);
275 			break;
276 		}
277 		/* FALLTHROUGH */
278 	default:
279 		print_otag(h, TAG_BR, "");
280 		break;
281 	}
282 }
283 
284 void
285 html_mdoc(void *arg, const struct roff_man *mdoc)
286 {
287 	struct html	*h;
288 	struct tag	*t;
289 
290 	h = (struct html *)arg;
291 
292 	if ((h->oflags & HTML_FRAGMENT) == 0) {
293 		print_gen_decls(h);
294 		print_otag(h, TAG_HTML, "");
295 		t = print_otag(h, TAG_HEAD, "");
296 		print_mdoc_head(&mdoc->meta, mdoc->first->child, h);
297 		print_tagq(h, t);
298 		print_otag(h, TAG_BODY, "");
299 	}
300 
301 	mdoc_root_pre(&mdoc->meta, mdoc->first->child, h);
302 	t = print_otag(h, TAG_DIV, "c", "manual-text");
303 	print_mdoc_nodelist(&mdoc->meta, mdoc->first->child, h);
304 	print_tagq(h, t);
305 	mdoc_root_post(&mdoc->meta, mdoc->first->child, h);
306 	print_tagq(h, NULL);
307 }
308 
309 static void
310 print_mdoc_head(MDOC_ARGS)
311 {
312 	char	*cp;
313 
314 	print_gen_head(h);
315 
316 	if (meta->arch != NULL && meta->msec != NULL)
317 		mandoc_asprintf(&cp, "%s(%s) (%s)", meta->title,
318 		    meta->msec, meta->arch);
319 	else if (meta->msec != NULL)
320 		mandoc_asprintf(&cp, "%s(%s)", meta->title, meta->msec);
321 	else if (meta->arch != NULL)
322 		mandoc_asprintf(&cp, "%s (%s)", meta->title, meta->arch);
323 	else
324 		cp = mandoc_strdup(meta->title);
325 
326 	print_otag(h, TAG_TITLE, "");
327 	print_text(h, cp);
328 	free(cp);
329 }
330 
331 static void
332 print_mdoc_nodelist(MDOC_ARGS)
333 {
334 
335 	while (n != NULL) {
336 		print_mdoc_node(meta, n, h);
337 		n = n->next;
338 	}
339 }
340 
341 static void
342 print_mdoc_node(MDOC_ARGS)
343 {
344 	int		 child;
345 	struct tag	*t;
346 
347 	if (n->flags & NODE_NOPRT)
348 		return;
349 
350 	child = 1;
351 	t = h->tag;
352 	n->flags &= ~NODE_ENDED;
353 
354 	switch (n->type) {
355 	case ROFFT_TEXT:
356 		/* No tables in this mode... */
357 		assert(NULL == h->tblt);
358 
359 		/*
360 		 * Make sure that if we're in a literal mode already
361 		 * (i.e., within a <PRE>) don't print the newline.
362 		 */
363 		if (*n->string == ' ' && n->flags & NODE_LINE &&
364 		    (h->flags & (HTML_LITERAL | HTML_NONEWLINE)) == 0)
365 			print_otag(h, TAG_BR, "");
366 		if (NODE_DELIMC & n->flags)
367 			h->flags |= HTML_NOSPACE;
368 		print_text(h, n->string);
369 		if (NODE_DELIMO & n->flags)
370 			h->flags |= HTML_NOSPACE;
371 		return;
372 	case ROFFT_EQN:
373 		print_eqn(h, n->eqn);
374 		break;
375 	case ROFFT_TBL:
376 		/*
377 		 * This will take care of initialising all of the table
378 		 * state data for the first table, then tearing it down
379 		 * for the last one.
380 		 */
381 		print_tbl(h, n->span);
382 		return;
383 	default:
384 		/*
385 		 * Close out the current table, if it's open, and unset
386 		 * the "meta" table state.  This will be reopened on the
387 		 * next table element.
388 		 */
389 		if (h->tblt != NULL) {
390 			print_tblclose(h);
391 			t = h->tag;
392 		}
393 		assert(h->tblt == NULL);
394 		if (n->tok < ROFF_MAX) {
395 			roff_html_pre(h, n);
396 			child = 0;
397 			break;
398 		}
399 		assert(n->tok >= MDOC_Dd && n->tok < MDOC_MAX);
400 		if (mdocs[n->tok].pre != NULL &&
401 		    (n->end == ENDBODY_NOT || n->child != NULL))
402 			child = (*mdocs[n->tok].pre)(meta, n, h);
403 		break;
404 	}
405 
406 	if (h->flags & HTML_KEEP && n->flags & NODE_LINE) {
407 		h->flags &= ~HTML_KEEP;
408 		h->flags |= HTML_PREKEEP;
409 	}
410 
411 	if (child && n->child)
412 		print_mdoc_nodelist(meta, n->child, h);
413 
414 	print_stagq(h, t);
415 
416 	switch (n->type) {
417 	case ROFFT_EQN:
418 		break;
419 	default:
420 		if (n->tok < ROFF_MAX ||
421 		    mdocs[n->tok].post == NULL ||
422 		    n->flags & NODE_ENDED)
423 			break;
424 		(*mdocs[n->tok].post)(meta, n, h);
425 		if (n->end != ENDBODY_NOT)
426 			n->body->flags |= NODE_ENDED;
427 		break;
428 	}
429 }
430 
431 static void
432 mdoc_root_post(MDOC_ARGS)
433 {
434 	struct tag	*t, *tt;
435 
436 	t = print_otag(h, TAG_TABLE, "c", "foot");
437 	tt = print_otag(h, TAG_TR, "");
438 
439 	print_otag(h, TAG_TD, "c", "foot-date");
440 	print_text(h, meta->date);
441 	print_stagq(h, tt);
442 
443 	print_otag(h, TAG_TD, "c", "foot-os");
444 	print_text(h, meta->os);
445 	print_tagq(h, t);
446 }
447 
448 static int
449 mdoc_root_pre(MDOC_ARGS)
450 {
451 	struct tag	*t, *tt;
452 	char		*volume, *title;
453 
454 	if (NULL == meta->arch)
455 		volume = mandoc_strdup(meta->vol);
456 	else
457 		mandoc_asprintf(&volume, "%s (%s)",
458 		    meta->vol, meta->arch);
459 
460 	if (NULL == meta->msec)
461 		title = mandoc_strdup(meta->title);
462 	else
463 		mandoc_asprintf(&title, "%s(%s)",
464 		    meta->title, meta->msec);
465 
466 	t = print_otag(h, TAG_TABLE, "c", "head");
467 	tt = print_otag(h, TAG_TR, "");
468 
469 	print_otag(h, TAG_TD, "c", "head-ltitle");
470 	print_text(h, title);
471 	print_stagq(h, tt);
472 
473 	print_otag(h, TAG_TD, "c", "head-vol");
474 	print_text(h, volume);
475 	print_stagq(h, tt);
476 
477 	print_otag(h, TAG_TD, "c", "head-rtitle");
478 	print_text(h, title);
479 	print_tagq(h, t);
480 
481 	free(title);
482 	free(volume);
483 	return 1;
484 }
485 
486 static char *
487 cond_id(const struct roff_node *n)
488 {
489 	if (n->child != NULL &&
490 	    n->child->type == ROFFT_TEXT &&
491 	    (n->prev == NULL ||
492 	     (n->prev->type == ROFFT_TEXT &&
493 	      strcmp(n->prev->string, "|") == 0)) &&
494 	    (n->parent->tok == MDOC_It ||
495 	     (n->parent->tok == MDOC_Xo &&
496 	      n->parent->parent->prev == NULL &&
497 	      n->parent->parent->parent->tok == MDOC_It)))
498 		return html_make_id(n);
499 	return NULL;
500 }
501 
502 static int
503 mdoc_sh_pre(MDOC_ARGS)
504 {
505 	char	*id;
506 
507 	switch (n->type) {
508 	case ROFFT_HEAD:
509 		id = html_make_id(n);
510 		print_otag(h, TAG_H1, "cTi", "Sh", id);
511 		if (id != NULL)
512 			print_otag(h, TAG_A, "chR", "selflink", id);
513 		free(id);
514 		break;
515 	case ROFFT_BODY:
516 		if (n->sec == SEC_AUTHORS)
517 			h->flags &= ~(HTML_SPLIT|HTML_NOSPLIT);
518 		break;
519 	default:
520 		break;
521 	}
522 	return 1;
523 }
524 
525 static int
526 mdoc_ss_pre(MDOC_ARGS)
527 {
528 	char	*id;
529 
530 	if (n->type != ROFFT_HEAD)
531 		return 1;
532 
533 	id = html_make_id(n);
534 	print_otag(h, TAG_H2, "cTi", "Ss", id);
535 	if (id != NULL)
536 		print_otag(h, TAG_A, "chR", "selflink", id);
537 	free(id);
538 	return 1;
539 }
540 
541 static int
542 mdoc_fl_pre(MDOC_ARGS)
543 {
544 	char	*id;
545 
546 	if ((id = cond_id(n)) != NULL)
547 		print_otag(h, TAG_A, "chR", "selflink", id);
548 	print_otag(h, TAG_B, "cTi", "Fl", id);
549 	free(id);
550 
551 	print_text(h, "\\-");
552 	if (!(n->child == NULL &&
553 	    (n->next == NULL ||
554 	     n->next->type == ROFFT_TEXT ||
555 	     n->next->flags & NODE_LINE)))
556 		h->flags |= HTML_NOSPACE;
557 
558 	return 1;
559 }
560 
561 static int
562 mdoc_cm_pre(MDOC_ARGS)
563 {
564 	char	*id;
565 
566 	if ((id = cond_id(n)) != NULL)
567 		print_otag(h, TAG_A, "chR", "selflink", id);
568 	print_otag(h, TAG_B, "cTi", "Cm", id);
569 	free(id);
570 	return 1;
571 }
572 
573 static int
574 mdoc_nd_pre(MDOC_ARGS)
575 {
576 	if (n->type != ROFFT_BODY)
577 		return 1;
578 
579 	/* XXX: this tag in theory can contain block elements. */
580 
581 	print_text(h, "\\(em");
582 	print_otag(h, TAG_SPAN, "cT", "Nd");
583 	return 1;
584 }
585 
586 static int
587 mdoc_nm_pre(MDOC_ARGS)
588 {
589 	switch (n->type) {
590 	case ROFFT_HEAD:
591 		print_otag(h, TAG_TD, "");
592 		/* FALLTHROUGH */
593 	case ROFFT_ELEM:
594 		print_otag(h, TAG_B, "cT", "Nm");
595 		return 1;
596 	case ROFFT_BODY:
597 		print_otag(h, TAG_TD, "");
598 		return 1;
599 	default:
600 		break;
601 	}
602 	synopsis_pre(h, n);
603 	print_otag(h, TAG_TABLE, "c", "Nm");
604 	print_otag(h, TAG_TR, "");
605 	return 1;
606 }
607 
608 static int
609 mdoc_xr_pre(MDOC_ARGS)
610 {
611 	if (NULL == n->child)
612 		return 0;
613 
614 	if (h->base_man)
615 		print_otag(h, TAG_A, "cThM", "Xr",
616 		    n->child->string, n->child->next == NULL ?
617 		    NULL : n->child->next->string);
618 	else
619 		print_otag(h, TAG_A, "cT", "Xr");
620 
621 	n = n->child;
622 	print_text(h, n->string);
623 
624 	if (NULL == (n = n->next))
625 		return 0;
626 
627 	h->flags |= HTML_NOSPACE;
628 	print_text(h, "(");
629 	h->flags |= HTML_NOSPACE;
630 	print_text(h, n->string);
631 	h->flags |= HTML_NOSPACE;
632 	print_text(h, ")");
633 	return 0;
634 }
635 
636 static int
637 mdoc_ns_pre(MDOC_ARGS)
638 {
639 
640 	if ( ! (NODE_LINE & n->flags))
641 		h->flags |= HTML_NOSPACE;
642 	return 1;
643 }
644 
645 static int
646 mdoc_ar_pre(MDOC_ARGS)
647 {
648 	print_otag(h, TAG_VAR, "cT", "Ar");
649 	return 1;
650 }
651 
652 static int
653 mdoc_xx_pre(MDOC_ARGS)
654 {
655 	print_otag(h, TAG_SPAN, "c", "Ux");
656 	return 1;
657 }
658 
659 static int
660 mdoc_it_pre(MDOC_ARGS)
661 {
662 	const struct roff_node	*bl;
663 	struct tag		*t;
664 	const char		*cattr;
665 	enum mdoc_list		 type;
666 
667 	bl = n->parent;
668 	while (bl->tok != MDOC_Bl)
669 		bl = bl->parent;
670 	type = bl->norm->Bl.type;
671 
672 	switch (type) {
673 	case LIST_bullet:
674 		cattr = "It-bullet";
675 		break;
676 	case LIST_dash:
677 	case LIST_hyphen:
678 		cattr = "It-dash";
679 		break;
680 	case LIST_item:
681 		cattr = "It-item";
682 		break;
683 	case LIST_enum:
684 		cattr = "It-enum";
685 		break;
686 	case LIST_diag:
687 		cattr = "It-diag";
688 		break;
689 	case LIST_hang:
690 		cattr = "It-hang";
691 		break;
692 	case LIST_inset:
693 		cattr = "It-inset";
694 		break;
695 	case LIST_ohang:
696 		cattr = "It-ohang";
697 		break;
698 	case LIST_tag:
699 		cattr = "It-tag";
700 		break;
701 	case LIST_column:
702 		cattr = "It-column";
703 		break;
704 	default:
705 		break;
706 	}
707 
708 	switch (type) {
709 	case LIST_bullet:
710 	case LIST_dash:
711 	case LIST_hyphen:
712 	case LIST_item:
713 	case LIST_enum:
714 		switch (n->type) {
715 		case ROFFT_HEAD:
716 			return 0;
717 		case ROFFT_BODY:
718 			print_otag(h, TAG_LI, "c", cattr);
719 			break;
720 		default:
721 			break;
722 		}
723 		break;
724 	case LIST_diag:
725 	case LIST_hang:
726 	case LIST_inset:
727 	case LIST_ohang:
728 		switch (n->type) {
729 		case ROFFT_HEAD:
730 			print_otag(h, TAG_DT, "c", cattr);
731 			if (type == LIST_diag)
732 				print_otag(h, TAG_B, "c", cattr);
733 			break;
734 		case ROFFT_BODY:
735 			print_otag(h, TAG_DD, "csw*+l", cattr,
736 			    bl->norm->Bl.width);
737 			break;
738 		default:
739 			break;
740 		}
741 		break;
742 	case LIST_tag:
743 		switch (n->type) {
744 		case ROFFT_HEAD:
745 			if (h->style != NULL && !bl->norm->Bl.comp &&
746 			    (n->parent->prev == NULL ||
747 			     n->parent->prev->body == NULL ||
748 			     n->parent->prev->body->child != NULL)) {
749 				t = print_otag(h, TAG_DT, "csw*+-l",
750 				    cattr, bl->norm->Bl.width);
751 				print_text(h, "\\ ");
752 				print_tagq(h, t);
753 				t = print_otag(h, TAG_DD, "c", cattr);
754 				print_text(h, "\\ ");
755 				print_tagq(h, t);
756 			}
757 			print_otag(h, TAG_DT, "csw*+-l", cattr,
758 			    bl->norm->Bl.width);
759 			break;
760 		case ROFFT_BODY:
761 			if (n->child == NULL) {
762 				print_otag(h, TAG_DD, "css?", cattr,
763 				    "width", "auto");
764 				print_text(h, "\\ ");
765 			} else
766 				print_otag(h, TAG_DD, "c", cattr);
767 			break;
768 		default:
769 			break;
770 		}
771 		break;
772 	case LIST_column:
773 		switch (n->type) {
774 		case ROFFT_HEAD:
775 			break;
776 		case ROFFT_BODY:
777 			print_otag(h, TAG_TD, "c", cattr);
778 			break;
779 		default:
780 			print_otag(h, TAG_TR, "c", cattr);
781 		}
782 	default:
783 		break;
784 	}
785 
786 	return 1;
787 }
788 
789 static int
790 mdoc_bl_pre(MDOC_ARGS)
791 {
792 	char		 cattr[21];
793 	struct tag	*t;
794 	struct mdoc_bl	*bl;
795 	size_t		 i;
796 	enum htmltag	 elemtype;
797 
798 	bl = &n->norm->Bl;
799 
800 	switch (n->type) {
801 	case ROFFT_BODY:
802 		return 1;
803 
804 	case ROFFT_HEAD:
805 		if (bl->type != LIST_column || bl->ncols == 0)
806 			return 0;
807 
808 		/*
809 		 * For each column, print out the <COL> tag with our
810 		 * suggested width.  The last column gets min-width, as
811 		 * in terminal mode it auto-sizes to the width of the
812 		 * screen and we want to preserve that behaviour.
813 		 */
814 
815 		t = print_otag(h, TAG_COLGROUP, "");
816 		for (i = 0; i < bl->ncols - 1; i++)
817 			print_otag(h, TAG_COL, "sw+w", bl->cols[i]);
818 		print_otag(h, TAG_COL, "swW", bl->cols[i]);
819 		print_tagq(h, t);
820 		return 0;
821 
822 	default:
823 		break;
824 	}
825 
826 	switch (bl->type) {
827 	case LIST_bullet:
828 		elemtype = TAG_UL;
829 		(void)strlcpy(cattr, "Bl-bullet", sizeof(cattr));
830 		break;
831 	case LIST_dash:
832 	case LIST_hyphen:
833 		elemtype = TAG_UL;
834 		(void)strlcpy(cattr, "Bl-dash", sizeof(cattr));
835 		break;
836 	case LIST_item:
837 		elemtype = TAG_UL;
838 		(void)strlcpy(cattr, "Bl-item", sizeof(cattr));
839 		break;
840 	case LIST_enum:
841 		elemtype = TAG_OL;
842 		(void)strlcpy(cattr, "Bl-enum", sizeof(cattr));
843 		break;
844 	case LIST_diag:
845 		elemtype = TAG_DL;
846 		(void)strlcpy(cattr, "Bl-diag", sizeof(cattr));
847 		break;
848 	case LIST_hang:
849 		elemtype = TAG_DL;
850 		(void)strlcpy(cattr, "Bl-hang", sizeof(cattr));
851 		break;
852 	case LIST_inset:
853 		elemtype = TAG_DL;
854 		(void)strlcpy(cattr, "Bl-inset", sizeof(cattr));
855 		break;
856 	case LIST_ohang:
857 		elemtype = TAG_DL;
858 		(void)strlcpy(cattr, "Bl-ohang", sizeof(cattr));
859 		break;
860 	case LIST_tag:
861 		if (bl->offs)
862 			print_otag(h, TAG_DIV, "cswl", "Bl-tag", bl->offs);
863 		print_otag(h, TAG_DL, "csw*+l", bl->comp ?
864 		    "Bl-tag Bl-compact" : "Bl-tag", bl->width);
865 		return 1;
866 	case LIST_column:
867 		elemtype = TAG_TABLE;
868 		(void)strlcpy(cattr, "Bl-column", sizeof(cattr));
869 		break;
870 	default:
871 		abort();
872 	}
873 	if (bl->comp)
874 		(void)strlcat(cattr, " Bl-compact", sizeof(cattr));
875 	print_otag(h, elemtype, "cswl", cattr, bl->offs);
876 	return 1;
877 }
878 
879 static int
880 mdoc_ex_pre(MDOC_ARGS)
881 {
882 	if (n->prev)
883 		print_otag(h, TAG_BR, "");
884 	return 1;
885 }
886 
887 static int
888 mdoc_st_pre(MDOC_ARGS)
889 {
890 	print_otag(h, TAG_SPAN, "cT", "St");
891 	return 1;
892 }
893 
894 static int
895 mdoc_em_pre(MDOC_ARGS)
896 {
897 	print_otag(h, TAG_I, "cT", "Em");
898 	return 1;
899 }
900 
901 static int
902 mdoc_d1_pre(MDOC_ARGS)
903 {
904 	if (n->type != ROFFT_BLOCK)
905 		return 1;
906 
907 	print_otag(h, TAG_DIV, "c", "D1");
908 
909 	if (n->tok == MDOC_Dl)
910 		print_otag(h, TAG_CODE, "c", "Li");
911 
912 	return 1;
913 }
914 
915 static int
916 mdoc_sx_pre(MDOC_ARGS)
917 {
918 	char	*id;
919 
920 	id = html_make_id(n);
921 	print_otag(h, TAG_A, "cThR", "Sx", id);
922 	free(id);
923 	return 1;
924 }
925 
926 static int
927 mdoc_bd_pre(MDOC_ARGS)
928 {
929 	int			 comp, offs, sv;
930 	struct roff_node	*nn;
931 
932 	if (n->type == ROFFT_HEAD)
933 		return 0;
934 
935 	if (n->type == ROFFT_BLOCK) {
936 		comp = n->norm->Bd.comp;
937 		for (nn = n; nn && ! comp; nn = nn->parent) {
938 			if (nn->type != ROFFT_BLOCK)
939 				continue;
940 			if (MDOC_Ss == nn->tok || MDOC_Sh == nn->tok)
941 				comp = 1;
942 			if (nn->prev)
943 				break;
944 		}
945 		if ( ! comp)
946 			print_paragraph(h);
947 		return 1;
948 	}
949 
950 	/* Handle the -offset argument. */
951 
952 	if (n->norm->Bd.offs == NULL ||
953 	    ! strcmp(n->norm->Bd.offs, "left"))
954 		offs = 0;
955 	else if ( ! strcmp(n->norm->Bd.offs, "indent"))
956 		offs = INDENT;
957 	else if ( ! strcmp(n->norm->Bd.offs, "indent-two"))
958 		offs = INDENT * 2;
959 	else
960 		offs = -1;
961 
962 	if (offs == -1)
963 		print_otag(h, TAG_DIV, "cswl", "Bd", n->norm->Bd.offs);
964 	else
965 		print_otag(h, TAG_DIV, "cshl", "Bd", offs);
966 
967 	if (n->norm->Bd.type != DISP_unfilled &&
968 	    n->norm->Bd.type != DISP_literal)
969 		return 1;
970 
971 	print_otag(h, TAG_PRE, "c", "Li");
972 
973 	/* This can be recursive: save & set our literal state. */
974 
975 	sv = h->flags & HTML_LITERAL;
976 	h->flags |= HTML_LITERAL;
977 
978 	for (nn = n->child; nn; nn = nn->next) {
979 		print_mdoc_node(meta, nn, h);
980 		/*
981 		 * If the printed node flushes its own line, then we
982 		 * needn't do it here as well.  This is hacky, but the
983 		 * notion of selective eoln whitespace is pretty dumb
984 		 * anyway, so don't sweat it.
985 		 */
986 		switch (nn->tok) {
987 		case ROFF_br:
988 		case ROFF_sp:
989 		case MDOC_Sm:
990 		case MDOC_Bl:
991 		case MDOC_D1:
992 		case MDOC_Dl:
993 		case MDOC_Lp:
994 		case MDOC_Pp:
995 			continue;
996 		default:
997 			break;
998 		}
999 		if (h->flags & HTML_NONEWLINE ||
1000 		    (nn->next && ! (nn->next->flags & NODE_LINE)))
1001 			continue;
1002 		else if (nn->next)
1003 			print_text(h, "\n");
1004 
1005 		h->flags |= HTML_NOSPACE;
1006 	}
1007 
1008 	if (0 == sv)
1009 		h->flags &= ~HTML_LITERAL;
1010 
1011 	return 0;
1012 }
1013 
1014 static int
1015 mdoc_pa_pre(MDOC_ARGS)
1016 {
1017 	print_otag(h, TAG_I, "cT", "Pa");
1018 	return 1;
1019 }
1020 
1021 static int
1022 mdoc_ad_pre(MDOC_ARGS)
1023 {
1024 	print_otag(h, TAG_I, "c", "Ad");
1025 	return 1;
1026 }
1027 
1028 static int
1029 mdoc_an_pre(MDOC_ARGS)
1030 {
1031 	if (n->norm->An.auth == AUTH_split) {
1032 		h->flags &= ~HTML_NOSPLIT;
1033 		h->flags |= HTML_SPLIT;
1034 		return 0;
1035 	}
1036 	if (n->norm->An.auth == AUTH_nosplit) {
1037 		h->flags &= ~HTML_SPLIT;
1038 		h->flags |= HTML_NOSPLIT;
1039 		return 0;
1040 	}
1041 
1042 	if (h->flags & HTML_SPLIT)
1043 		print_otag(h, TAG_BR, "");
1044 
1045 	if (n->sec == SEC_AUTHORS && ! (h->flags & HTML_NOSPLIT))
1046 		h->flags |= HTML_SPLIT;
1047 
1048 	print_otag(h, TAG_SPAN, "cT", "An");
1049 	return 1;
1050 }
1051 
1052 static int
1053 mdoc_cd_pre(MDOC_ARGS)
1054 {
1055 	synopsis_pre(h, n);
1056 	print_otag(h, TAG_B, "cT", "Cd");
1057 	return 1;
1058 }
1059 
1060 static int
1061 mdoc_dv_pre(MDOC_ARGS)
1062 {
1063 	char	*id;
1064 
1065 	if ((id = cond_id(n)) != NULL)
1066 		print_otag(h, TAG_A, "chR", "selflink", id);
1067 	print_otag(h, TAG_CODE, "cTi", "Dv", id);
1068 	free(id);
1069 	return 1;
1070 }
1071 
1072 static int
1073 mdoc_ev_pre(MDOC_ARGS)
1074 {
1075 	char	*id;
1076 
1077 	if ((id = cond_id(n)) != NULL)
1078 		print_otag(h, TAG_A, "chR", "selflink", id);
1079 	print_otag(h, TAG_CODE, "cTi", "Ev", id);
1080 	free(id);
1081 	return 1;
1082 }
1083 
1084 static int
1085 mdoc_er_pre(MDOC_ARGS)
1086 {
1087 	char	*id;
1088 
1089 	id = n->sec == SEC_ERRORS &&
1090 	    (n->parent->tok == MDOC_It ||
1091 	     (n->parent->tok == MDOC_Bq &&
1092 	      n->parent->parent->parent->tok == MDOC_It)) ?
1093 	    html_make_id(n) : NULL;
1094 
1095 	if (id != NULL)
1096 		print_otag(h, TAG_A, "chR", "selflink", id);
1097 	print_otag(h, TAG_CODE, "cTi", "Er", id);
1098 	free(id);
1099 	return 1;
1100 }
1101 
1102 static int
1103 mdoc_fa_pre(MDOC_ARGS)
1104 {
1105 	const struct roff_node	*nn;
1106 	struct tag		*t;
1107 
1108 	if (n->parent->tok != MDOC_Fo) {
1109 		print_otag(h, TAG_VAR, "cT", "Fa");
1110 		return 1;
1111 	}
1112 
1113 	for (nn = n->child; nn; nn = nn->next) {
1114 		t = print_otag(h, TAG_VAR, "cT", "Fa");
1115 		print_text(h, nn->string);
1116 		print_tagq(h, t);
1117 		if (nn->next) {
1118 			h->flags |= HTML_NOSPACE;
1119 			print_text(h, ",");
1120 		}
1121 	}
1122 
1123 	if (n->child && n->next && n->next->tok == MDOC_Fa) {
1124 		h->flags |= HTML_NOSPACE;
1125 		print_text(h, ",");
1126 	}
1127 
1128 	return 0;
1129 }
1130 
1131 static int
1132 mdoc_fd_pre(MDOC_ARGS)
1133 {
1134 	struct tag	*t;
1135 	char		*buf, *cp;
1136 
1137 	synopsis_pre(h, n);
1138 
1139 	if (NULL == (n = n->child))
1140 		return 0;
1141 
1142 	assert(n->type == ROFFT_TEXT);
1143 
1144 	if (strcmp(n->string, "#include")) {
1145 		print_otag(h, TAG_B, "cT", "Fd");
1146 		return 1;
1147 	}
1148 
1149 	print_otag(h, TAG_B, "cT", "In");
1150 	print_text(h, n->string);
1151 
1152 	if (NULL != (n = n->next)) {
1153 		assert(n->type == ROFFT_TEXT);
1154 
1155 		if (h->base_includes) {
1156 			cp = n->string;
1157 			if (*cp == '<' || *cp == '"')
1158 				cp++;
1159 			buf = mandoc_strdup(cp);
1160 			cp = strchr(buf, '\0') - 1;
1161 			if (cp >= buf && (*cp == '>' || *cp == '"'))
1162 				*cp = '\0';
1163 			t = print_otag(h, TAG_A, "cThI", "In", buf);
1164 			free(buf);
1165 		} else
1166 			t = print_otag(h, TAG_A, "cT", "In");
1167 
1168 		print_text(h, n->string);
1169 		print_tagq(h, t);
1170 
1171 		n = n->next;
1172 	}
1173 
1174 	for ( ; n; n = n->next) {
1175 		assert(n->type == ROFFT_TEXT);
1176 		print_text(h, n->string);
1177 	}
1178 
1179 	return 0;
1180 }
1181 
1182 static int
1183 mdoc_vt_pre(MDOC_ARGS)
1184 {
1185 	if (n->type == ROFFT_BLOCK) {
1186 		synopsis_pre(h, n);
1187 		return 1;
1188 	} else if (n->type == ROFFT_ELEM) {
1189 		synopsis_pre(h, n);
1190 	} else if (n->type == ROFFT_HEAD)
1191 		return 0;
1192 
1193 	print_otag(h, TAG_VAR, "cT", "Vt");
1194 	return 1;
1195 }
1196 
1197 static int
1198 mdoc_ft_pre(MDOC_ARGS)
1199 {
1200 	synopsis_pre(h, n);
1201 	print_otag(h, TAG_VAR, "cT", "Ft");
1202 	return 1;
1203 }
1204 
1205 static int
1206 mdoc_fn_pre(MDOC_ARGS)
1207 {
1208 	struct tag	*t;
1209 	char		 nbuf[BUFSIZ];
1210 	const char	*sp, *ep;
1211 	int		 sz, pretty;
1212 
1213 	pretty = NODE_SYNPRETTY & n->flags;
1214 	synopsis_pre(h, n);
1215 
1216 	/* Split apart into type and name. */
1217 	assert(n->child->string);
1218 	sp = n->child->string;
1219 
1220 	ep = strchr(sp, ' ');
1221 	if (NULL != ep) {
1222 		t = print_otag(h, TAG_VAR, "cT", "Ft");
1223 
1224 		while (ep) {
1225 			sz = MIN((int)(ep - sp), BUFSIZ - 1);
1226 			(void)memcpy(nbuf, sp, (size_t)sz);
1227 			nbuf[sz] = '\0';
1228 			print_text(h, nbuf);
1229 			sp = ++ep;
1230 			ep = strchr(sp, ' ');
1231 		}
1232 		print_tagq(h, t);
1233 	}
1234 
1235 	t = print_otag(h, TAG_B, "cT", "Fn");
1236 
1237 	if (sp)
1238 		print_text(h, sp);
1239 
1240 	print_tagq(h, t);
1241 
1242 	h->flags |= HTML_NOSPACE;
1243 	print_text(h, "(");
1244 	h->flags |= HTML_NOSPACE;
1245 
1246 	for (n = n->child->next; n; n = n->next) {
1247 		if (NODE_SYNPRETTY & n->flags)
1248 			t = print_otag(h, TAG_VAR, "cTss?", "Fa",
1249 			    "white-space", "nowrap");
1250 		else
1251 			t = print_otag(h, TAG_VAR, "cT", "Fa");
1252 		print_text(h, n->string);
1253 		print_tagq(h, t);
1254 		if (n->next) {
1255 			h->flags |= HTML_NOSPACE;
1256 			print_text(h, ",");
1257 		}
1258 	}
1259 
1260 	h->flags |= HTML_NOSPACE;
1261 	print_text(h, ")");
1262 
1263 	if (pretty) {
1264 		h->flags |= HTML_NOSPACE;
1265 		print_text(h, ";");
1266 	}
1267 
1268 	return 0;
1269 }
1270 
1271 static int
1272 mdoc_sm_pre(MDOC_ARGS)
1273 {
1274 
1275 	if (NULL == n->child)
1276 		h->flags ^= HTML_NONOSPACE;
1277 	else if (0 == strcmp("on", n->child->string))
1278 		h->flags &= ~HTML_NONOSPACE;
1279 	else
1280 		h->flags |= HTML_NONOSPACE;
1281 
1282 	if ( ! (HTML_NONOSPACE & h->flags))
1283 		h->flags &= ~HTML_NOSPACE;
1284 
1285 	return 0;
1286 }
1287 
1288 static int
1289 mdoc_skip_pre(MDOC_ARGS)
1290 {
1291 
1292 	return 0;
1293 }
1294 
1295 static int
1296 mdoc_pp_pre(MDOC_ARGS)
1297 {
1298 
1299 	print_paragraph(h);
1300 	return 0;
1301 }
1302 
1303 static int
1304 mdoc_lk_pre(MDOC_ARGS)
1305 {
1306 	const struct roff_node *link, *descr, *punct;
1307 	struct tag	*t;
1308 
1309 	if ((link = n->child) == NULL)
1310 		return 0;
1311 
1312 	/* Find beginning of trailing punctuation. */
1313 	punct = n->last;
1314 	while (punct != link && punct->flags & NODE_DELIMC)
1315 		punct = punct->prev;
1316 	punct = punct->next;
1317 
1318 	/* Link target and link text. */
1319 	descr = link->next;
1320 	if (descr == punct)
1321 		descr = link;  /* no text */
1322 	t = print_otag(h, TAG_A, "cTh", "Lk", link->string);
1323 	do {
1324 		if (descr->flags & (NODE_DELIMC | NODE_DELIMO))
1325 			h->flags |= HTML_NOSPACE;
1326 		print_text(h, descr->string);
1327 		descr = descr->next;
1328 	} while (descr != punct);
1329 	print_tagq(h, t);
1330 
1331 	/* Trailing punctuation. */
1332 	while (punct != NULL) {
1333 		h->flags |= HTML_NOSPACE;
1334 		print_text(h, punct->string);
1335 		punct = punct->next;
1336 	}
1337 	return 0;
1338 }
1339 
1340 static int
1341 mdoc_mt_pre(MDOC_ARGS)
1342 {
1343 	struct tag	*t;
1344 	char		*cp;
1345 
1346 	for (n = n->child; n; n = n->next) {
1347 		assert(n->type == ROFFT_TEXT);
1348 
1349 		mandoc_asprintf(&cp, "mailto:%s", n->string);
1350 		t = print_otag(h, TAG_A, "cTh", "Mt", cp);
1351 		print_text(h, n->string);
1352 		print_tagq(h, t);
1353 		free(cp);
1354 	}
1355 
1356 	return 0;
1357 }
1358 
1359 static int
1360 mdoc_fo_pre(MDOC_ARGS)
1361 {
1362 	struct tag	*t;
1363 
1364 	if (n->type == ROFFT_BODY) {
1365 		h->flags |= HTML_NOSPACE;
1366 		print_text(h, "(");
1367 		h->flags |= HTML_NOSPACE;
1368 		return 1;
1369 	} else if (n->type == ROFFT_BLOCK) {
1370 		synopsis_pre(h, n);
1371 		return 1;
1372 	}
1373 
1374 	if (n->child == NULL)
1375 		return 0;
1376 
1377 	assert(n->child->string);
1378 	t = print_otag(h, TAG_B, "cT", "Fn");
1379 	print_text(h, n->child->string);
1380 	print_tagq(h, t);
1381 	return 0;
1382 }
1383 
1384 static void
1385 mdoc_fo_post(MDOC_ARGS)
1386 {
1387 
1388 	if (n->type != ROFFT_BODY)
1389 		return;
1390 	h->flags |= HTML_NOSPACE;
1391 	print_text(h, ")");
1392 	h->flags |= HTML_NOSPACE;
1393 	print_text(h, ";");
1394 }
1395 
1396 static int
1397 mdoc_in_pre(MDOC_ARGS)
1398 {
1399 	struct tag	*t;
1400 
1401 	synopsis_pre(h, n);
1402 	print_otag(h, TAG_B, "cT", "In");
1403 
1404 	/*
1405 	 * The first argument of the `In' gets special treatment as
1406 	 * being a linked value.  Subsequent values are printed
1407 	 * afterward.  groff does similarly.  This also handles the case
1408 	 * of no children.
1409 	 */
1410 
1411 	if (NODE_SYNPRETTY & n->flags && NODE_LINE & n->flags)
1412 		print_text(h, "#include");
1413 
1414 	print_text(h, "<");
1415 	h->flags |= HTML_NOSPACE;
1416 
1417 	if (NULL != (n = n->child)) {
1418 		assert(n->type == ROFFT_TEXT);
1419 
1420 		if (h->base_includes)
1421 			t = print_otag(h, TAG_A, "cThI", "In", n->string);
1422 		else
1423 			t = print_otag(h, TAG_A, "cT", "In");
1424 		print_text(h, n->string);
1425 		print_tagq(h, t);
1426 
1427 		n = n->next;
1428 	}
1429 
1430 	h->flags |= HTML_NOSPACE;
1431 	print_text(h, ">");
1432 
1433 	for ( ; n; n = n->next) {
1434 		assert(n->type == ROFFT_TEXT);
1435 		print_text(h, n->string);
1436 	}
1437 
1438 	return 0;
1439 }
1440 
1441 static int
1442 mdoc_ic_pre(MDOC_ARGS)
1443 {
1444 	char	*id;
1445 
1446 	if ((id = cond_id(n)) != NULL)
1447 		print_otag(h, TAG_A, "chR", "selflink", id);
1448 	print_otag(h, TAG_B, "cTi", "Ic", id);
1449 	free(id);
1450 	return 1;
1451 }
1452 
1453 static int
1454 mdoc_va_pre(MDOC_ARGS)
1455 {
1456 	print_otag(h, TAG_VAR, "cT", "Va");
1457 	return 1;
1458 }
1459 
1460 static int
1461 mdoc_ap_pre(MDOC_ARGS)
1462 {
1463 
1464 	h->flags |= HTML_NOSPACE;
1465 	print_text(h, "\\(aq");
1466 	h->flags |= HTML_NOSPACE;
1467 	return 1;
1468 }
1469 
1470 static int
1471 mdoc_bf_pre(MDOC_ARGS)
1472 {
1473 	const char	*cattr;
1474 
1475 	if (n->type == ROFFT_HEAD)
1476 		return 0;
1477 	else if (n->type != ROFFT_BODY)
1478 		return 1;
1479 
1480 	if (FONT_Em == n->norm->Bf.font)
1481 		cattr = "Em";
1482 	else if (FONT_Sy == n->norm->Bf.font)
1483 		cattr = "Sy";
1484 	else if (FONT_Li == n->norm->Bf.font)
1485 		cattr = "Li";
1486 	else
1487 		cattr = "No";
1488 
1489 	/*
1490 	 * We want this to be inline-formatted, but needs to be div to
1491 	 * accept block children.
1492 	 */
1493 
1494 	print_otag(h, TAG_DIV, "css?hl", cattr, "display", "inline", 1);
1495 	return 1;
1496 }
1497 
1498 static int
1499 mdoc_ms_pre(MDOC_ARGS)
1500 {
1501 	char *id;
1502 
1503 	if ((id = cond_id(n)) != NULL)
1504 		print_otag(h, TAG_A, "chR", "selflink", id);
1505 	print_otag(h, TAG_B, "cTi", "Ms", id);
1506 	free(id);
1507 	return 1;
1508 }
1509 
1510 static int
1511 mdoc_igndelim_pre(MDOC_ARGS)
1512 {
1513 
1514 	h->flags |= HTML_IGNDELIM;
1515 	return 1;
1516 }
1517 
1518 static void
1519 mdoc_pf_post(MDOC_ARGS)
1520 {
1521 
1522 	if ( ! (n->next == NULL || n->next->flags & NODE_LINE))
1523 		h->flags |= HTML_NOSPACE;
1524 }
1525 
1526 static int
1527 mdoc_rs_pre(MDOC_ARGS)
1528 {
1529 	if (n->type != ROFFT_BLOCK)
1530 		return 1;
1531 
1532 	if (n->prev && SEC_SEE_ALSO == n->sec)
1533 		print_paragraph(h);
1534 
1535 	print_otag(h, TAG_CITE, "cT", "Rs");
1536 	return 1;
1537 }
1538 
1539 static int
1540 mdoc_no_pre(MDOC_ARGS)
1541 {
1542 	char *id;
1543 
1544 	if ((id = cond_id(n)) != NULL)
1545 		print_otag(h, TAG_A, "chR", "selflink", id);
1546 	print_otag(h, TAG_SPAN, "ci", "No", id);
1547 	free(id);
1548 	return 1;
1549 }
1550 
1551 static int
1552 mdoc_li_pre(MDOC_ARGS)
1553 {
1554 	char	*id;
1555 
1556 	if ((id = cond_id(n)) != NULL)
1557 		print_otag(h, TAG_A, "chR", "selflink", id);
1558 	print_otag(h, TAG_CODE, "ci", "Li", id);
1559 	free(id);
1560 	return 1;
1561 }
1562 
1563 static int
1564 mdoc_sy_pre(MDOC_ARGS)
1565 {
1566 	print_otag(h, TAG_B, "cT", "Sy");
1567 	return 1;
1568 }
1569 
1570 static int
1571 mdoc_lb_pre(MDOC_ARGS)
1572 {
1573 	if (SEC_LIBRARY == n->sec && NODE_LINE & n->flags && n->prev)
1574 		print_otag(h, TAG_BR, "");
1575 
1576 	print_otag(h, TAG_SPAN, "cT", "Lb");
1577 	return 1;
1578 }
1579 
1580 static int
1581 mdoc__x_pre(MDOC_ARGS)
1582 {
1583 	const char	*cattr;
1584 	enum htmltag	 t;
1585 
1586 	t = TAG_SPAN;
1587 
1588 	switch (n->tok) {
1589 	case MDOC__A:
1590 		cattr = "RsA";
1591 		if (n->prev && MDOC__A == n->prev->tok)
1592 			if (NULL == n->next || MDOC__A != n->next->tok)
1593 				print_text(h, "and");
1594 		break;
1595 	case MDOC__B:
1596 		t = TAG_I;
1597 		cattr = "RsB";
1598 		break;
1599 	case MDOC__C:
1600 		cattr = "RsC";
1601 		break;
1602 	case MDOC__D:
1603 		cattr = "RsD";
1604 		break;
1605 	case MDOC__I:
1606 		t = TAG_I;
1607 		cattr = "RsI";
1608 		break;
1609 	case MDOC__J:
1610 		t = TAG_I;
1611 		cattr = "RsJ";
1612 		break;
1613 	case MDOC__N:
1614 		cattr = "RsN";
1615 		break;
1616 	case MDOC__O:
1617 		cattr = "RsO";
1618 		break;
1619 	case MDOC__P:
1620 		cattr = "RsP";
1621 		break;
1622 	case MDOC__Q:
1623 		cattr = "RsQ";
1624 		break;
1625 	case MDOC__R:
1626 		cattr = "RsR";
1627 		break;
1628 	case MDOC__T:
1629 		cattr = "RsT";
1630 		break;
1631 	case MDOC__U:
1632 		print_otag(h, TAG_A, "ch", "RsU", n->child->string);
1633 		return 1;
1634 	case MDOC__V:
1635 		cattr = "RsV";
1636 		break;
1637 	default:
1638 		abort();
1639 	}
1640 
1641 	print_otag(h, t, "c", cattr);
1642 	return 1;
1643 }
1644 
1645 static void
1646 mdoc__x_post(MDOC_ARGS)
1647 {
1648 
1649 	if (MDOC__A == n->tok && n->next && MDOC__A == n->next->tok)
1650 		if (NULL == n->next->next || MDOC__A != n->next->next->tok)
1651 			if (NULL == n->prev || MDOC__A != n->prev->tok)
1652 				return;
1653 
1654 	/* TODO: %U */
1655 
1656 	if (NULL == n->parent || MDOC_Rs != n->parent->tok)
1657 		return;
1658 
1659 	h->flags |= HTML_NOSPACE;
1660 	print_text(h, n->next ? "," : ".");
1661 }
1662 
1663 static int
1664 mdoc_bk_pre(MDOC_ARGS)
1665 {
1666 
1667 	switch (n->type) {
1668 	case ROFFT_BLOCK:
1669 		break;
1670 	case ROFFT_HEAD:
1671 		return 0;
1672 	case ROFFT_BODY:
1673 		if (n->parent->args != NULL || n->prev->child == NULL)
1674 			h->flags |= HTML_PREKEEP;
1675 		break;
1676 	default:
1677 		abort();
1678 	}
1679 
1680 	return 1;
1681 }
1682 
1683 static void
1684 mdoc_bk_post(MDOC_ARGS)
1685 {
1686 
1687 	if (n->type == ROFFT_BODY)
1688 		h->flags &= ~(HTML_KEEP | HTML_PREKEEP);
1689 }
1690 
1691 static int
1692 mdoc_quote_pre(MDOC_ARGS)
1693 {
1694 	if (n->type != ROFFT_BODY)
1695 		return 1;
1696 
1697 	switch (n->tok) {
1698 	case MDOC_Ao:
1699 	case MDOC_Aq:
1700 		print_text(h, n->child != NULL && n->child->next == NULL &&
1701 		    n->child->tok == MDOC_Mt ?  "<" : "\\(la");
1702 		break;
1703 	case MDOC_Bro:
1704 	case MDOC_Brq:
1705 		print_text(h, "\\(lC");
1706 		break;
1707 	case MDOC_Bo:
1708 	case MDOC_Bq:
1709 		print_text(h, "\\(lB");
1710 		break;
1711 	case MDOC_Oo:
1712 	case MDOC_Op:
1713 		print_text(h, "\\(lB");
1714 		h->flags |= HTML_NOSPACE;
1715 		print_otag(h, TAG_SPAN, "c", "Op");
1716 		break;
1717 	case MDOC_En:
1718 		if (NULL == n->norm->Es ||
1719 		    NULL == n->norm->Es->child)
1720 			return 1;
1721 		print_text(h, n->norm->Es->child->string);
1722 		break;
1723 	case MDOC_Do:
1724 	case MDOC_Dq:
1725 	case MDOC_Qo:
1726 	case MDOC_Qq:
1727 		print_text(h, "\\(lq");
1728 		break;
1729 	case MDOC_Po:
1730 	case MDOC_Pq:
1731 		print_text(h, "(");
1732 		break;
1733 	case MDOC_Ql:
1734 		print_text(h, "\\(oq");
1735 		h->flags |= HTML_NOSPACE;
1736 		print_otag(h, TAG_CODE, "c", "Li");
1737 		break;
1738 	case MDOC_So:
1739 	case MDOC_Sq:
1740 		print_text(h, "\\(oq");
1741 		break;
1742 	default:
1743 		abort();
1744 	}
1745 
1746 	h->flags |= HTML_NOSPACE;
1747 	return 1;
1748 }
1749 
1750 static void
1751 mdoc_quote_post(MDOC_ARGS)
1752 {
1753 
1754 	if (n->type != ROFFT_BODY && n->type != ROFFT_ELEM)
1755 		return;
1756 
1757 	h->flags |= HTML_NOSPACE;
1758 
1759 	switch (n->tok) {
1760 	case MDOC_Ao:
1761 	case MDOC_Aq:
1762 		print_text(h, n->child != NULL && n->child->next == NULL &&
1763 		    n->child->tok == MDOC_Mt ?  ">" : "\\(ra");
1764 		break;
1765 	case MDOC_Bro:
1766 	case MDOC_Brq:
1767 		print_text(h, "\\(rC");
1768 		break;
1769 	case MDOC_Oo:
1770 	case MDOC_Op:
1771 	case MDOC_Bo:
1772 	case MDOC_Bq:
1773 		print_text(h, "\\(rB");
1774 		break;
1775 	case MDOC_En:
1776 		if (n->norm->Es == NULL ||
1777 		    n->norm->Es->child == NULL ||
1778 		    n->norm->Es->child->next == NULL)
1779 			h->flags &= ~HTML_NOSPACE;
1780 		else
1781 			print_text(h, n->norm->Es->child->next->string);
1782 		break;
1783 	case MDOC_Qo:
1784 	case MDOC_Qq:
1785 	case MDOC_Do:
1786 	case MDOC_Dq:
1787 		print_text(h, "\\(rq");
1788 		break;
1789 	case MDOC_Po:
1790 	case MDOC_Pq:
1791 		print_text(h, ")");
1792 		break;
1793 	case MDOC_Ql:
1794 	case MDOC_So:
1795 	case MDOC_Sq:
1796 		print_text(h, "\\(cq");
1797 		break;
1798 	default:
1799 		abort();
1800 	}
1801 }
1802 
1803 static int
1804 mdoc_eo_pre(MDOC_ARGS)
1805 {
1806 
1807 	if (n->type != ROFFT_BODY)
1808 		return 1;
1809 
1810 	if (n->end == ENDBODY_NOT &&
1811 	    n->parent->head->child == NULL &&
1812 	    n->child != NULL &&
1813 	    n->child->end != ENDBODY_NOT)
1814 		print_text(h, "\\&");
1815 	else if (n->end != ENDBODY_NOT ? n->child != NULL :
1816 	    n->parent->head->child != NULL && (n->child != NULL ||
1817 	    (n->parent->tail != NULL && n->parent->tail->child != NULL)))
1818 		h->flags |= HTML_NOSPACE;
1819 	return 1;
1820 }
1821 
1822 static void
1823 mdoc_eo_post(MDOC_ARGS)
1824 {
1825 	int	 body, tail;
1826 
1827 	if (n->type != ROFFT_BODY)
1828 		return;
1829 
1830 	if (n->end != ENDBODY_NOT) {
1831 		h->flags &= ~HTML_NOSPACE;
1832 		return;
1833 	}
1834 
1835 	body = n->child != NULL || n->parent->head->child != NULL;
1836 	tail = n->parent->tail != NULL && n->parent->tail->child != NULL;
1837 
1838 	if (body && tail)
1839 		h->flags |= HTML_NOSPACE;
1840 	else if ( ! tail)
1841 		h->flags &= ~HTML_NOSPACE;
1842 }
1843