home *** CD-ROM | disk | FTP | other *** search
/ Frozen Fish 1: Amiga / FrozenFish-Apr94.iso / bbs / gnu / groff-1.09-src.lha / src / amiga / groff-1.09 / troff / node.cc < prev    next >
C/C++ Source or Header  |  1994-02-11  |  95KB  |  4,852 lines

  1. // -*- C++ -*-
  2. /* Copyright (C) 1989, 1990, 1991, 1992 Free Software Foundation, Inc.
  3.      Written by James Clark (jjc@jclark.com)
  4.  
  5. This file is part of groff.
  6.  
  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.
  11.  
  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.
  16.  
  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, 675 Mass Ave, Cambridge, MA 02139, USA. */
  20.  
  21. #include "troff.h"
  22. #include "symbol.h"
  23. #include "dictionary.h"
  24. #include "hvunits.h"
  25. #include "env.h"
  26. #include "request.h"
  27. #include "node.h"
  28. #include "token.h"
  29. #include "charinfo.h"
  30. #include "font.h"
  31. #include "reg.h"
  32.  
  33. #define STORE_WIDTH 1
  34.  
  35. symbol HYPHEN_SYMBOL("hy");
  36.  
  37. // Character used when a hyphen is inserted at a line break.
  38. static charinfo *soft_hyphen_char;
  39.  
  40. enum constant_space_type { 
  41.   CONSTANT_SPACE_NONE,
  42.   CONSTANT_SPACE_RELATIVE,
  43.   CONSTANT_SPACE_ABSOLUTE
  44.   };
  45.  
  46. struct special_font_list {
  47.   int n;
  48.   special_font_list *next;
  49. };
  50.  
  51. special_font_list *global_special_fonts;
  52. static int global_ligature_mode = 1;
  53. static int global_kern_mode = 1;
  54.  
  55. class track_kerning_function {
  56.   int non_zero;
  57.   units min_size;
  58.   hunits min_amount;
  59.   units max_size;
  60.   hunits max_amount;
  61. public:
  62.   track_kerning_function();
  63.   track_kerning_function(units, hunits, units, hunits);
  64.   int operator==(const track_kerning_function &);
  65.   int operator!=(const track_kerning_function &);
  66.   hunits compute(int point_size);
  67. };
  68.  
  69. // embolden fontno when this is the current font
  70.  
  71. struct conditional_bold {
  72.   conditional_bold *next;
  73.   int fontno;
  74.   hunits offset;
  75.   conditional_bold(int, hunits, conditional_bold * = 0);
  76. };
  77.  
  78. struct tfont;
  79.  
  80. class font_info {
  81.   tfont *last_tfont;
  82.   int number;
  83.   font_size last_size;
  84.   int last_height;
  85.   int last_slant;
  86.   symbol internal_name;
  87.   symbol external_name;
  88.   font *fm;
  89.   char is_bold;
  90.   hunits bold_offset;
  91.   track_kerning_function track_kern;
  92.   constant_space_type is_constant_spaced;
  93.   units constant_space;
  94.   int last_ligature_mode;
  95.   int last_kern_mode;
  96.   conditional_bold *cond_bold_list;
  97.   void flush();
  98. public:
  99.   special_font_list *sf;
  100.   
  101.   font_info(symbol nm, int n, symbol enm, font *f);
  102.   int contains(charinfo *);
  103.   void set_bold(hunits);
  104.   void unbold();
  105.   void set_conditional_bold(int, hunits);
  106.   void conditional_unbold(int);
  107.   void set_track_kern(track_kerning_function &);
  108.   void set_constant_space(constant_space_type, units = 0);
  109.   int is_named(symbol);
  110.   symbol get_name();
  111.   tfont *get_tfont(font_size, int, int, int);
  112.   hunits get_space_width(font_size, int);
  113.   hunits get_narrow_space_width(font_size);
  114.   hunits get_half_narrow_space_width(font_size);
  115.   int get_bold(hunits *);
  116.   int is_special();
  117.   int is_style();
  118. };
  119.  
  120. class tfont_spec {
  121. protected:
  122.   symbol name;
  123.   int input_position;
  124.   font *fm;
  125.   font_size size;
  126.   char is_bold;
  127.   char is_constant_spaced;
  128.   int ligature_mode;
  129.   int kern_mode;
  130.   hunits bold_offset;
  131.   hunits track_kern;            // add this to the width
  132.   hunits constant_space_width;
  133.   int height;
  134.   int slant;
  135. public:
  136.   tfont_spec(symbol nm, int pos, font *, font_size, int, int);
  137.   tfont_spec(const tfont_spec &spec) { *this = spec; }
  138.   tfont_spec plain();
  139.   int operator==(const tfont_spec &);
  140.   friend tfont *font_info::get_tfont(font_size fs, int, int, int);
  141. };
  142.  
  143. class tfont : public tfont_spec {
  144.   static tfont *tfont_list;
  145.   tfont *next;
  146.   tfont *plain_version;
  147. public:
  148.   tfont(tfont_spec &);
  149.   int contains(charinfo *);
  150.   hunits get_width(charinfo *c);
  151.   int get_bold(hunits *);
  152.   int get_constant_space(hunits *);
  153.   hunits get_track_kern();
  154.   tfont *get_plain();
  155.   font_size get_size();
  156.   symbol get_name();
  157.   charinfo *get_lig(charinfo *c1, charinfo *c2);
  158.   int get_kern(charinfo *c1, charinfo *c2, hunits *res);
  159.   int get_input_position();
  160.   int get_character_type(charinfo *);
  161.   int get_height();
  162.   int get_slant();
  163.   vunits get_char_height(charinfo *);
  164.   vunits get_char_depth(charinfo *);
  165.   hunits get_char_skew(charinfo *);
  166.   hunits get_italic_correction(charinfo *);
  167.   hunits get_left_italic_correction(charinfo *);
  168.   hunits get_subscript_correction(charinfo *);
  169.   friend tfont *make_tfont(tfont_spec &);
  170. };
  171.  
  172. inline int env_definite_font(environment *env)
  173. {
  174.   return env->get_family()->make_definite(env->get_font());
  175. }
  176.  
  177. /* font_info functions */
  178.  
  179. static font_info **font_table = 0;
  180. static int font_table_size = 0;
  181.  
  182. font_info::font_info(symbol nm, int n, symbol enm, font *f)
  183. : internal_name(nm), external_name(enm), fm(f), number(n),
  184.   is_constant_spaced(CONSTANT_SPACE_NONE),
  185.   sf(0), is_bold(0), cond_bold_list(0),
  186.   last_ligature_mode(1), last_kern_mode(1),
  187.   last_tfont(0), last_size(0)
  188. {
  189. }
  190.  
  191. inline int font_info::contains(charinfo *ci)
  192. {
  193.   return fm != 0 && fm->contains(ci->get_index());
  194. }
  195.  
  196. inline int font_info::is_special()
  197. {
  198.   return fm != 0 && fm->is_special();
  199. }
  200.  
  201. inline int font_info::is_style()
  202. {
  203.   return fm == 0;
  204. }
  205.  
  206. // this is the current_font, fontno is where we found the character,
  207. // presumably a special font
  208.  
  209. tfont *font_info::get_tfont(font_size fs, int height, int slant, int fontno)
  210. {
  211.   if (last_tfont == 0 || fs != last_size
  212.       || height != last_height || slant != last_slant
  213.       || global_ligature_mode != last_ligature_mode
  214.       || global_kern_mode != last_kern_mode
  215.       || fontno != number) {
  216.     font_info *f = font_table[fontno];
  217.     tfont_spec spec(f->external_name, f->number, f->fm, fs, height, slant);
  218.     for (conditional_bold *p = cond_bold_list; p; p = p->next)
  219.       if (p->fontno == fontno) {
  220.         spec.is_bold = 1;
  221.         spec.bold_offset = p->offset;
  222.         break;
  223.       }
  224.     if (!spec.is_bold && is_bold) {
  225.       spec.is_bold = 1;
  226.       spec.bold_offset = bold_offset;
  227.     }
  228.     spec.track_kern = track_kern.compute(fs.to_scaled_points());
  229.     spec.ligature_mode = global_ligature_mode;
  230.     spec.kern_mode = global_kern_mode;
  231.     switch (is_constant_spaced) {
  232.     case CONSTANT_SPACE_NONE:
  233.       break;
  234.     case CONSTANT_SPACE_ABSOLUTE:
  235.       spec.is_constant_spaced = 1;
  236.       spec.constant_space_width = constant_space;
  237.       break;
  238.     case CONSTANT_SPACE_RELATIVE:
  239.       spec.is_constant_spaced = 1;
  240.       spec.constant_space_width
  241.         = scale(constant_space*fs.to_scaled_points(),
  242.             units_per_inch,
  243.             36*72*sizescale);
  244.       break;
  245.     default:
  246.       assert(0);
  247.     }
  248.     if (fontno != number)
  249.       return make_tfont(spec);
  250.     last_tfont = make_tfont(spec);
  251.     last_size = fs;
  252.     last_height = height;
  253.     last_slant = slant;
  254.     last_ligature_mode = global_ligature_mode;
  255.     last_kern_mode = global_kern_mode;
  256.       }
  257.   return last_tfont;
  258. }
  259.  
  260. int font_info::get_bold(hunits *res)
  261. {
  262.   if (is_bold) {
  263.     *res = bold_offset;
  264.     return 1;
  265.   }
  266.   else
  267.     return 0;
  268. }
  269.  
  270. void font_info::unbold()
  271. {
  272.   if (is_bold) {
  273.     is_bold = 0;
  274.     flush();
  275.   }
  276. }
  277.  
  278. void font_info::set_bold(hunits offset)
  279. {
  280.   if (!is_bold || offset != bold_offset) {
  281.     is_bold = 1;
  282.     bold_offset = offset;
  283.     flush();
  284.   }
  285. }
  286.  
  287. void font_info::set_conditional_bold(int fontno, hunits offset)
  288. {
  289.   for (conditional_bold *p = cond_bold_list; p; p = p->next)
  290.     if (p->fontno == fontno) {
  291.       if (offset != p->offset) {
  292.     p->offset = offset;
  293.     flush();
  294.       }
  295.       return;
  296.     }
  297.   cond_bold_list = new conditional_bold(fontno, offset, cond_bold_list);
  298. }
  299.  
  300. conditional_bold::conditional_bold(int f, hunits h, conditional_bold *x)
  301.      : fontno(f), offset(h), next(x)
  302. {
  303. }
  304.  
  305. void font_info::conditional_unbold(int fontno)
  306. {
  307.   for (conditional_bold **p = &cond_bold_list; *p; p = &(*p)->next)
  308.     if ((*p)->fontno == fontno) {
  309.       conditional_bold *tem = *p;
  310.       *p = (*p)->next;
  311.       delete tem;
  312.       flush();
  313.       return;
  314.     }
  315. }
  316.                  
  317. void font_info::set_constant_space(constant_space_type type, units x)
  318. {
  319.   if (type != is_constant_spaced
  320.       || (type != CONSTANT_SPACE_NONE && x != constant_space)) {
  321.     flush();
  322.     is_constant_spaced = type;
  323.     constant_space = x;
  324.       }
  325. }
  326.  
  327. void font_info::set_track_kern(track_kerning_function  &tk)
  328. {
  329.   if (track_kern != tk) {
  330.     track_kern = tk;
  331.     flush();
  332.   }
  333. }
  334.  
  335. void font_info::flush()
  336. {
  337.   last_tfont = 0;
  338. }
  339.  
  340. int font_info::is_named(symbol s)
  341. {
  342.   return internal_name == s;
  343. }
  344.  
  345. symbol font_info::get_name()
  346. {
  347.   return internal_name;
  348. }
  349.  
  350. hunits font_info::get_space_width(font_size fs, int space_size)
  351. {
  352.   if (is_constant_spaced == CONSTANT_SPACE_NONE)
  353.     return scale(hunits(fm->get_space_width(fs.to_scaled_points())),
  354.             space_size, 12);
  355.   else if (is_constant_spaced == CONSTANT_SPACE_ABSOLUTE)
  356.     return constant_space;
  357.   else
  358.     return scale(constant_space*fs.to_scaled_points(),
  359.          units_per_inch, 36*72*sizescale);
  360. }
  361.  
  362. hunits font_info::get_narrow_space_width(font_size fs)
  363. {
  364.   charinfo *ci = get_charinfo(symbol("|"));
  365.   if (fm->contains(ci->get_index()))
  366.     return hunits(fm->get_width(ci->get_index(), fs.to_scaled_points()));
  367.   else
  368.     return hunits(fs.to_units()/6);
  369. }
  370.  
  371. hunits font_info::get_half_narrow_space_width(font_size fs)
  372. {
  373.   charinfo *ci = get_charinfo(symbol("^"));
  374.   if (fm->contains(ci->get_index()))
  375.     return hunits(fm->get_width(ci->get_index(), fs.to_scaled_points()));
  376.   else
  377.     return hunits(fs.to_units()/12);
  378. }
  379.  
  380. /* tfont */
  381.  
  382. tfont_spec::tfont_spec(symbol nm, int n, font *f, 
  383.                font_size s, int h, int sl)
  384.      : name(nm), input_position(n), fm(f), size(s),
  385.      is_bold(0), is_constant_spaced(0), ligature_mode(1), kern_mode(1),
  386.      height(h), slant(sl)
  387. {
  388.   if (height == size.to_scaled_points())
  389.     height = 0;
  390. }
  391.  
  392. int tfont_spec::operator==(const tfont_spec &spec)
  393. {
  394.   if (fm == spec.fm 
  395.       && size == spec.size
  396.       && input_position == spec.input_position
  397.       && name == spec.name
  398.       && height == spec.height
  399.       && slant == spec.slant
  400.       && (is_bold 
  401.       ? (spec.is_bold && bold_offset == spec.bold_offset)
  402.       : !spec.is_bold)
  403.       && track_kern == spec.track_kern
  404.       && (is_constant_spaced
  405.       ? (spec.is_constant_spaced 
  406.          && constant_space_width == spec.constant_space_width)
  407.       : !spec.is_constant_spaced)
  408.       && ligature_mode == spec.ligature_mode
  409.       && kern_mode == spec.kern_mode)
  410.     return 1;
  411.   else
  412.     return 0;
  413. }
  414.  
  415. tfont_spec tfont_spec::plain()
  416. {
  417.   return tfont_spec(name, input_position, fm, size, height, slant);
  418. }
  419.  
  420. hunits tfont::get_width(charinfo *c)
  421. {
  422.   if (is_constant_spaced)
  423.     return constant_space_width;
  424.   else if (is_bold)
  425.     return (hunits(fm->get_width(c->get_index(), size.to_scaled_points()))
  426.         + track_kern + bold_offset);
  427.   else
  428.     return (hunits(fm->get_width(c->get_index(), size.to_scaled_points())) + track_kern);
  429. }
  430.  
  431. vunits tfont::get_char_height(charinfo *c)
  432. {
  433.   vunits v = fm->get_height(c->get_index(), size.to_scaled_points());
  434.   if (height != 0 && height != size.to_scaled_points())
  435.     return scale(v, height, size.to_scaled_points());
  436.   else
  437.     return v;
  438. }
  439.  
  440. vunits tfont::get_char_depth(charinfo *c)
  441. {
  442.   vunits v = fm->get_depth(c->get_index(), size.to_scaled_points());
  443.   if (height != 0 && height != size.to_scaled_points())
  444.     return scale(v, height, size.to_scaled_points());
  445.   else
  446.     return v;
  447. }
  448.  
  449. hunits tfont::get_char_skew(charinfo *c)
  450. {
  451.   return hunits(fm->get_skew(c->get_index(), size.to_scaled_points(), slant));
  452. }
  453.  
  454. hunits tfont::get_italic_correction(charinfo *c)
  455. {
  456.   return hunits(fm->get_italic_correction(c->get_index(), size.to_scaled_points()));
  457. }
  458.  
  459. hunits tfont::get_left_italic_correction(charinfo *c)
  460. {
  461.   return hunits(fm->get_left_italic_correction(c->get_index(),
  462.                            size.to_scaled_points()));
  463. }
  464.  
  465. hunits tfont::get_subscript_correction(charinfo *c)
  466. {
  467.   return hunits(fm->get_subscript_correction(c->get_index(),
  468.                          size.to_scaled_points()));
  469. }
  470.  
  471. inline int tfont::get_input_position()
  472. {
  473.   return input_position;
  474. }
  475.  
  476. inline int tfont::contains(charinfo *ci)
  477. {
  478.   return fm->contains(ci->get_index());
  479. }
  480.  
  481. inline int tfont::get_character_type(charinfo *ci)
  482. {
  483.   return fm->get_character_type(ci->get_index());
  484. }
  485.  
  486. inline int tfont::get_bold(hunits *res)
  487. {
  488.   if (is_bold) {
  489.     *res = bold_offset;
  490.     return 1;
  491.   }
  492.   else
  493.     return 0;
  494. }
  495.  
  496. inline int tfont::get_constant_space(hunits *res)
  497. {
  498.   if (is_constant_spaced) {
  499.     *res = constant_space_width;
  500.     return 1;
  501.   }
  502.   else
  503.     return 0;
  504. }
  505.  
  506. inline hunits tfont::get_track_kern()
  507. {
  508.   return track_kern;
  509. }
  510.  
  511. inline tfont *tfont::get_plain()
  512. {
  513.   return plain_version;
  514. }
  515.  
  516. inline font_size tfont::get_size()
  517. {
  518.   return size;
  519. }
  520.  
  521. inline symbol tfont::get_name()
  522. {
  523.   return name;
  524. }
  525.  
  526. inline int tfont::get_height()
  527. {
  528.   return height;
  529. }
  530.  
  531. inline int tfont::get_slant()
  532. {
  533.   return slant;
  534. }
  535.  
  536. symbol SYMBOL_ff("ff");
  537. symbol SYMBOL_fi("fi");
  538. symbol SYMBOL_fl("fl");
  539. symbol SYMBOL_Fi("Fi");
  540. symbol SYMBOL_Fl("Fl");
  541.  
  542. charinfo *tfont::get_lig(charinfo *c1, charinfo *c2)
  543. {
  544.   if (ligature_mode == 0)
  545.     return 0;
  546.   charinfo *ci = 0;
  547.   if (c1->get_ascii_code() == 'f') {
  548.     switch (c2->get_ascii_code()) {
  549.     case 'f':
  550.       if (fm->has_ligature(font::LIG_ff))
  551.     ci = get_charinfo(SYMBOL_ff);
  552.       break;
  553.     case 'i':
  554.       if (fm->has_ligature(font::LIG_fi))
  555.     ci = get_charinfo(SYMBOL_fi);
  556.       break;
  557.     case 'l':
  558.       if (fm->has_ligature(font::LIG_fl))
  559.     ci = get_charinfo(SYMBOL_fl);
  560.       break;
  561.     }
  562.   }
  563.   else if (ligature_mode != 2 && c1->nm == SYMBOL_ff) {
  564.     switch (c2->get_ascii_code()) {
  565.     case 'i':
  566.       if (fm->has_ligature(font::LIG_ffi))
  567.     ci = get_charinfo(SYMBOL_Fi);
  568.       break;
  569.     case 'l':
  570.       if (fm->has_ligature(font::LIG_ffl))
  571.     ci = get_charinfo(SYMBOL_Fl);
  572.       break;
  573.     }
  574.   }
  575.   if (ci != 0 && fm->contains(ci->get_index()))
  576.     return ci;
  577.   return 0;
  578. }
  579.  
  580. inline int tfont::get_kern(charinfo *c1, charinfo *c2, hunits *res)
  581. {
  582.   if (kern_mode == 0)
  583.     return 0;
  584.   else {
  585.     int n = fm->get_kern(c1->get_index(),
  586.              c2->get_index(),
  587.              size.to_scaled_points());
  588.     if (n) {
  589.       *res = hunits(n);
  590.       return 1;
  591.     }
  592.     else
  593.       return 0;
  594.   }
  595. }
  596.  
  597. tfont *make_tfont(tfont_spec &spec)
  598. {
  599.   for (tfont *p = tfont::tfont_list; p; p = p->next)
  600.     if (*p == spec)
  601.       return p;
  602.   return new tfont(spec);
  603. }
  604.  
  605. tfont *tfont::tfont_list = 0;
  606.  
  607. tfont::tfont(tfont_spec &spec) : tfont_spec(spec)
  608. {
  609.   next = tfont_list;
  610.   tfont_list = this;
  611.   tfont_spec plain_spec = plain();
  612.   for (tfont *p = tfont_list; p; p = p->next)
  613.     if (*p == plain_spec) {
  614.       plain_version = p;
  615.       break;
  616.     }
  617.   if (!p)
  618.     plain_version = new tfont(plain_spec);
  619. }
  620.  
  621. /* output_file */
  622.  
  623. class real_output_file : public output_file {
  624. #ifndef POPEN_MISSING
  625.   int piped;
  626. #endif
  627.   int printing;
  628.   virtual void really_transparent_char(unsigned char) = 0;
  629.   virtual void really_print_line(hunits x, vunits y, node *n,
  630.                  vunits before, vunits after) = 0;
  631.   virtual void really_begin_page(int pageno, vunits page_length) = 0;
  632.   virtual void really_copy_file(hunits x, vunits y, const char *filename);
  633. protected:
  634.   FILE *fp;
  635. public:
  636.   real_output_file();
  637.   ~real_output_file();
  638.   void flush();
  639.   void transparent_char(unsigned char);
  640.   void print_line(hunits x, vunits y, node *n, vunits before, vunits after);
  641.   void begin_page(int pageno, vunits page_length);
  642.   int is_printing();
  643.   void copy_file(hunits x, vunits y, const char *filename);
  644. };
  645.  
  646. class suppress_output_file : public real_output_file {
  647. public:
  648.   suppress_output_file();
  649.   void really_transparent_char(unsigned char);
  650.   void really_print_line(hunits x, vunits y, node *n, vunits, vunits);
  651.   void really_begin_page(int pageno, vunits page_length);
  652. };
  653.  
  654. class ascii_output_file : public real_output_file {
  655. public:
  656.   ascii_output_file();
  657.   void really_transparent_char(unsigned char);
  658.   void really_print_line(hunits x, vunits y, node *n, vunits, vunits);
  659.   void really_begin_page(int pageno, vunits page_length);
  660.   void outc(unsigned char c);
  661.   void outs(const char *s);
  662. };
  663.  
  664. void ascii_output_file::outc(unsigned char c)
  665. {
  666.   fputc(c, fp);
  667. }
  668.  
  669. void ascii_output_file::outs(const char *s)
  670. {
  671.   fputc('<', fp);
  672.   if (s)
  673.     fputs(s, fp);
  674.   fputc('>', fp);
  675. }
  676.  
  677. struct hvpair;
  678.       
  679. class troff_output_file : public real_output_file {
  680.   units hpos;
  681.   units vpos;
  682.   units output_vpos;
  683.   units output_hpos;
  684.   int force_motion;
  685.   int current_size;
  686.   int current_slant;
  687.   int current_height;
  688.   tfont *current_tfont;
  689.   int current_font_number;
  690.   symbol *font_position;
  691.   int nfont_positions;
  692.   enum { TBUF_SIZE = 256 };
  693.   char tbuf[TBUF_SIZE];
  694.   int tbuf_len;
  695.   int tbuf_kern;
  696.   int begun_page;
  697.   void do_motion();
  698.   void put(char c);
  699.   void put(unsigned char c);
  700.   void put(int i);
  701.   void put(const char *s);
  702.   void set_font(tfont *tf);
  703.   void flush_tbuf();
  704. public:
  705.   troff_output_file();
  706.   ~troff_output_file();
  707.   void trailer(vunits page_length);
  708.   void put_char(charinfo *ci, tfont *tf);
  709.   void put_char_width(charinfo *ci, tfont *tf, hunits w, hunits k);
  710.   void right(hunits);
  711.   void down(vunits);
  712.   void moveto(hunits, vunits);
  713.   void start_special();
  714.   void special_char(unsigned char c);
  715.   void end_special();
  716.   void word_marker();
  717.   void really_transparent_char(unsigned char c);
  718.   void really_print_line(hunits x, vunits y, node *n, vunits before, vunits after);
  719.   void really_begin_page(int pageno, vunits page_length);
  720.   void really_copy_file(hunits x, vunits y, const char *filename);
  721.   void draw(char, hvpair *, int, font_size);
  722.   int get_hpos() { return hpos; }
  723.   int get_vpos() { return vpos; }
  724. };
  725.  
  726. static void put_string(const char *s, FILE *fp)
  727. {
  728.   for (; *s != '\0'; ++s)
  729.     putc(*s, fp);
  730. }
  731.  
  732. inline void troff_output_file::put(char c)
  733. {
  734.   putc(c, fp);
  735. }
  736.  
  737. inline void troff_output_file::put(unsigned char c)
  738. {
  739.   putc(c, fp);
  740. }
  741.  
  742. inline void troff_output_file::put(const char *s)
  743. {
  744.   put_string(s, fp);
  745. }
  746.  
  747. inline void troff_output_file::put(int i)
  748. {
  749.   put_string(itoa(i), fp);
  750. }
  751.  
  752. void troff_output_file::start_special()
  753. {
  754.   flush_tbuf();
  755.   do_motion();
  756.   put("x X ");
  757. }
  758.  
  759. void troff_output_file::special_char(unsigned char c)
  760. {
  761.   put(c);
  762.   if (c == '\n')
  763.     put('+');
  764. }
  765.  
  766. void troff_output_file::end_special()
  767. {
  768.   put('\n');
  769. }
  770.  
  771. inline void troff_output_file::moveto(hunits h, vunits v)
  772. {
  773.   hpos = h.to_units();
  774.   vpos = v.to_units();
  775. }
  776.  
  777. void troff_output_file::really_print_line(hunits x, vunits y, node *n,
  778.                       vunits before, vunits after)
  779. {
  780.   moveto(x, y);
  781.   while (n != 0) {
  782.     n->tprint(this);
  783.     n = n->next;
  784.   }
  785.   flush_tbuf();
  786.   // This ensures that transparent throughput will have a more predictable
  787.   // position.
  788.   do_motion();
  789.   force_motion = 1;
  790.   hpos = 0;
  791.   put('n');
  792.   put(before.to_units());
  793.   put(' ');
  794.   put(after.to_units());
  795.   put('\n');
  796. }
  797.  
  798. inline void troff_output_file::word_marker()
  799. {
  800.   flush_tbuf();
  801.   put('w');
  802. }
  803.  
  804. inline void troff_output_file::right(hunits n)
  805. {
  806.   hpos += n.to_units();
  807. }
  808.  
  809. inline void troff_output_file::down(vunits n)
  810. {
  811.   vpos += n.to_units();
  812. }
  813.  
  814. void troff_output_file::do_motion()
  815. {
  816.   if (force_motion) {
  817.     put('V');
  818.     put(vpos);
  819.     put('\n');
  820.     put('H');
  821.     put(hpos);
  822.     put('\n');
  823.   }
  824.   else {
  825.     if (hpos != output_hpos) {
  826.       units n = hpos - output_hpos;
  827.       if (n > 0 && n < hpos) {
  828.     put('h');
  829.     put(n);
  830.       }
  831.       else {
  832.     put('H');
  833.     put(hpos);
  834.       }
  835.       put('\n');
  836.     }
  837.     if (vpos != output_vpos) {
  838.       units n = vpos - output_vpos;
  839.       if (n > 0 && n < vpos) {
  840.     put('v');
  841.     put(n);
  842.       }
  843.       else {
  844.     put('V');
  845.     put(vpos);
  846.       }
  847.       put('\n');
  848.     }
  849.   }
  850.   output_vpos = vpos;
  851.   output_hpos = hpos;
  852.   force_motion = 0;
  853. }
  854.  
  855. void troff_output_file::flush_tbuf()
  856. {
  857.   if (tbuf_len == 0)
  858.     return;
  859.   if (tbuf_kern == 0)
  860.     put('t');
  861.   else {
  862.     put('u');
  863.     put(tbuf_kern);
  864.     put(' ');
  865.   }
  866.   for (int i = 0; i < tbuf_len; i++)
  867.     put(tbuf[i]);
  868.   put('\n');
  869.   tbuf_len = 0;
  870. }
  871.  
  872. void troff_output_file::put_char_width(charinfo *ci, tfont *tf, hunits w,
  873.                        hunits k)
  874. {
  875.   if (tf != current_tfont) {
  876.     flush_tbuf();
  877.     set_font(tf);
  878.   }
  879.   char c = ci->get_ascii_code();
  880.   int kk = k.to_units();
  881.   if (c == '\0') {
  882.     flush_tbuf();
  883.     do_motion();
  884.     if (ci->numbered()) {
  885.       put('N');
  886.       put(ci->get_number());
  887.     }
  888.     else {
  889.       put('C');
  890.       const char *s = ci->nm.contents();
  891.       if (s[1] == 0) {
  892.     put('\\');
  893.     put(s[0]);
  894.       }
  895.       else
  896.     put(s);
  897.     }
  898.     put('\n');
  899.     hpos += w.to_units() + kk;
  900.   }
  901.   else if (tcommand_flag) {
  902.     if (tbuf_len > 0 && hpos == output_hpos && vpos == output_vpos
  903.     && kk == tbuf_kern
  904.     && tbuf_len < TBUF_SIZE) {
  905.       tbuf[tbuf_len++] = c;
  906.       output_hpos += w.to_units() + kk;
  907.       hpos = output_hpos;
  908.       return;
  909.     }
  910.     flush_tbuf();
  911.     do_motion();
  912.     tbuf[tbuf_len++] = c;
  913.     output_hpos += w.to_units() + kk;
  914.     tbuf_kern = kk;
  915.     hpos = output_hpos;
  916.   }
  917.   else {
  918.     // flush_tbuf();
  919.     int n = hpos - output_hpos;
  920.     if (vpos == output_vpos && n > 0 && n < 100 && !force_motion) {
  921.       put(char(n/10 + '0'));
  922.       put(char(n%10 + '0'));
  923.       put(c);
  924.       output_hpos = hpos;
  925.     }
  926.     else {
  927.       do_motion();
  928.       put('c');
  929.       put(c);
  930.     }
  931.     hpos += w.to_units() + kk;
  932.   }
  933. }
  934.  
  935. void troff_output_file::put_char(charinfo *ci, tfont *tf)
  936. {
  937.   flush_tbuf();
  938.   if (tf != current_tfont)
  939.     set_font(tf);
  940.   char c = ci->get_ascii_code();
  941.   if (c == '\0') {
  942.     do_motion();
  943.     if (ci->numbered()) {
  944.       put('N');
  945.       put(ci->get_number());
  946.     }
  947.     else {
  948.       put('C');
  949.       const char *s = ci->nm.contents();
  950.       if (s[1] == 0) {
  951.     put('\\');
  952.     put(s[0]);
  953.       }
  954.       else
  955.     put(s);
  956.     }
  957.     put('\n');
  958.   }
  959.   else {
  960.     int n = hpos - output_hpos;
  961.     if (vpos == output_vpos && n > 0 && n < 100) {
  962.       put(char(n/10 + '0'));
  963.       put(char(n%10 + '0'));
  964.       put(c);
  965.       output_hpos = hpos;
  966.     }
  967.     else {
  968.       do_motion();
  969.       put('c');
  970.       put(c);
  971.     }
  972.   }
  973. }
  974.  
  975. void troff_output_file::set_font(tfont *tf)
  976. {
  977.   if (current_tfont == tf)
  978.     return;
  979.   int n = tf->get_input_position();
  980.   symbol nm = tf->get_name();
  981.   if (n >= nfont_positions || font_position[n] != nm) {
  982.     put("x font ");
  983.     put(n);
  984.     put(' ');
  985.     put(nm.contents());
  986.     put('\n');
  987.     if (n >= nfont_positions) {
  988.       int old_nfont_positions = nfont_positions;
  989.       symbol *old_font_position = font_position;
  990.       nfont_positions *= 3;
  991.       nfont_positions /= 2;
  992.       if (nfont_positions <= n)
  993.     nfont_positions = n + 10;
  994.       font_position = new symbol[nfont_positions];
  995.       memcpy(font_position, old_font_position,
  996.          old_nfont_positions*sizeof(symbol));
  997.       a_delete old_font_position;
  998.     }
  999.     font_position[n] = nm;
  1000.   }
  1001.   if (current_font_number != n) {
  1002.     put('f');
  1003.     put(n);
  1004.     put('\n');
  1005.     current_font_number = n;
  1006.   }
  1007.   int size = tf->get_size().to_scaled_points();
  1008.   if (current_size != size) {
  1009.     put('s');
  1010.     put(size);
  1011.     put('\n');
  1012.     current_size = size;
  1013.   }
  1014.   int slant = tf->get_slant();
  1015.   if (current_slant != slant) {
  1016.     put("x Slant ");
  1017.     put(slant);
  1018.     put('\n');
  1019.     current_slant = slant;
  1020.   }
  1021.   int height = tf->get_height();
  1022.   if (current_height != height) {
  1023.     put("x Height ");
  1024.     put(height == 0 ? current_size : height);
  1025.     put('\n');
  1026.     current_height = height;
  1027.   }
  1028.   current_tfont = tf;
  1029. }
  1030.  
  1031. void troff_output_file::draw(char code, hvpair *point, int npoints,
  1032.                  font_size fsize)
  1033. {
  1034.   flush_tbuf();
  1035.   do_motion();
  1036.   int size = fsize.to_scaled_points();
  1037.   if (current_size != size) {
  1038.     put('s');
  1039.     put(size);
  1040.     put('\n');
  1041.     current_size = size;
  1042.     current_tfont = 0;
  1043.   }
  1044.   put('D');
  1045.   put(code);
  1046.   int i;
  1047.   if (code == 'c') {
  1048.     put(' ');
  1049.     put(point[0].h.to_units());
  1050.   }
  1051.   else
  1052.     for (i = 0; i < npoints; i++) {
  1053.       put(' ');
  1054.       put(point[i].h.to_units());
  1055.       put(' ');
  1056.       put(point[i].v.to_units());
  1057.     }
  1058.   for (i = 0; i < npoints; i++)
  1059.     output_hpos += point[i].h.to_units();
  1060.   hpos = output_hpos;
  1061.   if (code != 'e') {
  1062.     for (i = 0; i < npoints; i++)
  1063.       output_vpos += point[i].v.to_units();
  1064.     vpos = output_vpos;
  1065.   }
  1066.   put('\n');
  1067. }
  1068.  
  1069. void troff_output_file::really_begin_page(int pageno, vunits page_length)
  1070. {
  1071.   flush_tbuf();
  1072.   if (begun_page) {
  1073.     if (page_length > V0) {
  1074.       put('V');
  1075.       put(page_length.to_units());
  1076.       put('\n');
  1077.     }
  1078.   }
  1079.   else
  1080.     begun_page = 1;
  1081.   current_tfont = 0;
  1082.   current_font_number = -1;
  1083.   current_size = 0;
  1084.   // current_height = 0;
  1085.   // current_slant = 0;
  1086.   hpos = 0;
  1087.   vpos = 0;
  1088.   output_hpos = 0;
  1089.   output_vpos = 0;
  1090.   force_motion = 1;
  1091.   for (int i = 0; i < nfont_positions; i++)
  1092.     font_position[i] = NULL_SYMBOL;
  1093.   put('p');
  1094.   put(pageno);
  1095.   put('\n');
  1096. }
  1097.  
  1098. void troff_output_file::really_copy_file(hunits x, vunits y, const char *filename)
  1099. {
  1100.   moveto(x, y);
  1101.   flush_tbuf();
  1102.   do_motion();
  1103.   errno = 0;
  1104.   FILE *ifp = fopen(filename, "r");
  1105.   if (ifp == 0)
  1106.     error("can't open `%1': %2", filename, strerror(errno));
  1107.   else {
  1108.     int c;
  1109.     while ((c = getc(ifp)) != EOF)
  1110.       put(char(c));
  1111.     fclose(ifp);
  1112.   }
  1113.   force_motion = 1;
  1114.   current_size = 0;
  1115.   current_tfont = 0;
  1116.   current_font_number = -1;
  1117.   for (int i = 0; i < nfont_positions; i++)
  1118.     font_position[i] = NULL_SYMBOL;
  1119. }
  1120.   
  1121. void troff_output_file::really_transparent_char(unsigned char c)
  1122. {
  1123.   put(c);
  1124. }
  1125.  
  1126. troff_output_file::~troff_output_file()
  1127. {
  1128.   a_delete font_position;
  1129. }
  1130.  
  1131. void troff_output_file::trailer(vunits page_length)
  1132. {
  1133.   flush_tbuf();
  1134.   if (page_length > V0) {
  1135.     put("x trailer\n");
  1136.     put('V');
  1137.     put(page_length.to_units());
  1138.     put('\n');
  1139.   }
  1140.   put("x stop\n");
  1141. }
  1142.  
  1143. troff_output_file::troff_output_file()
  1144. : current_height(0), current_slant(0), tbuf_len(0), nfont_positions(10),
  1145.   begun_page(0)
  1146. {
  1147.   font_position = new symbol[nfont_positions];
  1148.   put("x T ");
  1149.   put(device);
  1150.   put('\n');
  1151.   put("x res ");
  1152.   put(units_per_inch);
  1153.   put(' ');
  1154.   put(hresolution);
  1155.   put(' ');
  1156.   put(vresolution);
  1157.   put('\n');
  1158.   put("x init\n");
  1159. }
  1160.  
  1161. /* output_file */
  1162.  
  1163. output_file *the_output = 0;
  1164.  
  1165. output_file::output_file()
  1166. {
  1167. }
  1168.  
  1169. output_file::~output_file()
  1170. {
  1171. }
  1172.  
  1173. void output_file::trailer(vunits)
  1174. {
  1175. }
  1176.  
  1177. real_output_file::real_output_file()
  1178. : printing(0)
  1179. {
  1180. #ifndef POPEN_MISSING
  1181.   if (pipe_command) {
  1182.     if ((fp = popen(pipe_command, "w")) != 0) {
  1183.       piped = 1;
  1184.       return;
  1185.     }
  1186.     error("pipe open failed: %1", strerror(errno));
  1187.   }
  1188.   piped = 0;
  1189. #endif /* not POPEN_MISSING */
  1190.   fp = stdout;
  1191. }
  1192.  
  1193. real_output_file::~real_output_file()
  1194. {
  1195.   if (!fp)
  1196.     return;
  1197.   // To avoid looping, set fp to 0 before calling fatal().
  1198.   if (ferror(fp) || fflush(fp) < 0) {
  1199.     fp = 0;
  1200.     fatal("error writing output file");
  1201.   }
  1202. #ifndef POPEN_MISSING
  1203.   if (piped) {
  1204.     int result = pclose(fp);
  1205.     fp = 0;
  1206.     if (result < 0)
  1207.       fatal("pclose failed");
  1208.     if ((result & 0x7f) != 0)
  1209.       error("output process `%1' got fatal signal %2",
  1210.         pipe_command, result & 0x7f);
  1211.     else {
  1212.       int exit_status = (result >> 8) & 0xff;
  1213.       if (exit_status != 0)
  1214.     error("output process `%1' exited with status %2",
  1215.           pipe_command, exit_status);
  1216.     }
  1217.   }
  1218.   else
  1219. #endif /* not POPEN MISSING */
  1220.   if (fclose(fp) < 0) {
  1221.     fp = 0;
  1222.     fatal("error closing output file");
  1223.   }
  1224. }
  1225.  
  1226. void real_output_file::flush()
  1227. {
  1228.   if (fflush(fp) < 0)
  1229.     fatal("error writing output file");
  1230. }
  1231.  
  1232. int real_output_file::is_printing()
  1233. {
  1234.   return printing;
  1235. }
  1236.  
  1237. void real_output_file::begin_page(int pageno, vunits page_length)
  1238. {
  1239.   printing = in_output_page_list(pageno);
  1240.   if (printing)
  1241.     really_begin_page(pageno, page_length);
  1242. }
  1243.  
  1244. void real_output_file::copy_file(hunits x, vunits y, const char *filename)
  1245. {
  1246.   if (printing)
  1247.     really_copy_file(x, y, filename);
  1248. }
  1249.  
  1250. void real_output_file::transparent_char(unsigned char c)
  1251. {
  1252.   if (printing)
  1253.     really_transparent_char(c);
  1254. }
  1255.  
  1256. void real_output_file::print_line(hunits x, vunits y, node *n,
  1257.                  vunits before, vunits after)
  1258. {
  1259.   if (printing)
  1260.     really_print_line(x, y, n, before, after);
  1261.   delete_node_list(n);
  1262. }
  1263.  
  1264. void real_output_file::really_copy_file(hunits, vunits, const char *)
  1265. {
  1266.   // do nothing
  1267. }
  1268.  
  1269.  
  1270. /* ascii_output_file */
  1271.  
  1272. void ascii_output_file::really_transparent_char(unsigned char c)
  1273. {
  1274.   putc(c, fp);
  1275. }
  1276.  
  1277. void ascii_output_file::really_print_line(hunits, vunits, node *n, vunits, vunits)
  1278. {
  1279.   while (n != 0) {
  1280.     n->ascii_print(this);
  1281.     n = n->next;
  1282.   }
  1283.   fputc('\n', fp);
  1284. }
  1285.  
  1286. void ascii_output_file::really_begin_page(int /*pageno*/, vunits /*page_length*/)
  1287. {
  1288.   fputs("<beginning of page>\n", fp);
  1289. }
  1290.  
  1291. ascii_output_file::ascii_output_file()
  1292. {
  1293. }
  1294.  
  1295. /* suppress_output_file */
  1296.  
  1297. suppress_output_file::suppress_output_file()
  1298. {
  1299. }
  1300.  
  1301. void suppress_output_file::really_print_line(hunits, vunits, node *, vunits, vunits)
  1302. {
  1303. }
  1304.  
  1305. void suppress_output_file::really_begin_page(int, vunits)
  1306. {
  1307. }
  1308.  
  1309. void suppress_output_file::really_transparent_char(unsigned char)
  1310. {
  1311. }
  1312.  
  1313. /* glyphs, ligatures, kerns, discretionary breaks */
  1314.  
  1315. class glyph_node : public node {
  1316.   static glyph_node *free_list;
  1317. protected:
  1318.   charinfo *ci;
  1319.   tfont *tf;
  1320. #ifdef STORE_WIDTH
  1321.   hunits wid;
  1322.   glyph_node(charinfo *, tfont *, hunits, node * = 0);
  1323. #endif
  1324. public:
  1325.   void *operator new(size_t);
  1326.   void operator delete(void *);
  1327.   glyph_node(charinfo *, tfont *, node * = 0);
  1328.   ~glyph_node() {}
  1329.   node *copy();
  1330.   node *merge_glyph_node(glyph_node *);
  1331.   node *merge_self(node *);
  1332.   hunits width();
  1333.   node *last_char_node();
  1334.   units size();
  1335.   void vertical_extent(vunits *, vunits *);
  1336.   hunits subscript_correction();
  1337.   hunits italic_correction();
  1338.   hunits left_italic_correction();
  1339.   hunits skew();
  1340.   hyphenation_type get_hyphenation_type();
  1341.   tfont *get_tfont();
  1342.   void tprint(troff_output_file *);
  1343.   void zero_width_tprint(troff_output_file *);
  1344.   hyphen_list *get_hyphen_list(hyphen_list *ss = 0);
  1345.   node *add_self(node *, hyphen_list **);
  1346.   int ends_sentence();
  1347.   int overlaps_vertically();
  1348.   int overlaps_horizontally();
  1349.   void ascii_print(ascii_output_file *);
  1350.   void asciify(macro *);
  1351.   int character_type();
  1352.   int same(node *);
  1353.   const char *type();
  1354. };
  1355.  
  1356. glyph_node *glyph_node::free_list = 0;
  1357.  
  1358. class ligature_node : public glyph_node {
  1359.   node *n1;
  1360.   node *n2;
  1361. #ifdef STORE_WIDTH
  1362.   ligature_node(charinfo *, tfont *, hunits, node *gn1, node *gn2, node *x = 0);
  1363. #endif
  1364. public:
  1365.   void *operator new(size_t);
  1366.   void operator delete(void *);
  1367.   ligature_node(charinfo *, tfont *, node *gn1, node *gn2, node *x = 0);
  1368.   ~ligature_node();
  1369.   node *copy();
  1370.   node *add_self(node *, hyphen_list **);
  1371.   hyphen_list *get_hyphen_list(hyphen_list *ss = 0);
  1372.   void ascii_print(ascii_output_file *);
  1373.   void asciify(macro *);
  1374.   int same(node *);
  1375.   const char *type();
  1376. };
  1377.  
  1378. class kern_pair_node : public node {
  1379.   hunits amount;
  1380.   node *n1;
  1381.   node *n2;
  1382. public:
  1383.   kern_pair_node(hunits n, node *first, node *second, node *x = 0);
  1384.   ~kern_pair_node();
  1385.   node *copy();
  1386.   node *merge_glyph_node(glyph_node *);
  1387.   node *add_self(node *, hyphen_list **);
  1388.   hyphen_list *get_hyphen_list(hyphen_list *ss = 0);
  1389.   node *add_discretionary_hyphen();
  1390.   hunits width();
  1391.   node *last_char_node();
  1392.   hunits italic_correction();
  1393.   hunits subscript_correction();
  1394.   void tprint(troff_output_file *);
  1395.   hyphenation_type get_hyphenation_type();
  1396.   int ends_sentence();
  1397.   void ascii_print(ascii_output_file *);
  1398.   void asciify(macro *);
  1399.   int same(node *);
  1400.   const char *type();
  1401. };
  1402.  
  1403. class dbreak_node : public node {
  1404.   node *none;
  1405.   node *pre;
  1406.   node *post;
  1407. public:
  1408.   dbreak_node(node *n, node *p, node *x = 0);
  1409.   ~dbreak_node();
  1410.   node *copy();
  1411.   node *merge_glyph_node(glyph_node *);
  1412.   node *add_discretionary_hyphen();
  1413.   hunits width();
  1414.   node *last_char_node();
  1415.   hunits italic_correction();
  1416.   hunits subscript_correction();
  1417.   void tprint(troff_output_file *);
  1418.   breakpoint *get_breakpoints(hunits width, int ns, breakpoint *rest = 0,
  1419.                   int is_inner = 0);
  1420.   int nbreaks();
  1421.   int ends_sentence();
  1422.   void split(int, node **, node **);
  1423.   hyphenation_type get_hyphenation_type();
  1424.   void ascii_print(ascii_output_file *);
  1425.   void asciify(macro *);
  1426.   int same(node *);
  1427.   const char *type();
  1428. };
  1429.  
  1430. void *glyph_node::operator new(size_t n)
  1431. {
  1432.   assert(n == sizeof(glyph_node));
  1433.   if (!free_list) {
  1434.     const int BLOCK = 1024;
  1435.     free_list = (glyph_node *)new char[sizeof(glyph_node)*BLOCK];
  1436.     for (int i = 0; i < BLOCK - 1; i++)
  1437.       free_list[i].next = free_list + i + 1;
  1438.     free_list[BLOCK-1].next = 0;
  1439.   }
  1440.   glyph_node *p = free_list;
  1441.   free_list = (glyph_node *)(free_list->next);
  1442.   p->next = 0;
  1443.   return p;
  1444. }
  1445.  
  1446. void *ligature_node::operator new(size_t n)
  1447. {
  1448.   return new char[n];
  1449. }
  1450.  
  1451. void glyph_node::operator delete(void *p)
  1452. {
  1453.   if (p) {
  1454.     ((glyph_node *)p)->next = free_list;
  1455.     free_list = (glyph_node *)p;
  1456.   }
  1457. }
  1458.  
  1459. void ligature_node::operator delete(void *p)
  1460. {
  1461.   delete p;
  1462. }
  1463.  
  1464. glyph_node::glyph_node(charinfo *c, tfont *t, node *x)
  1465.      : ci(c), tf(t), node(x)
  1466. {
  1467. #ifdef STORE_WIDTH
  1468.   wid = tf->get_width(ci);
  1469. #endif
  1470. }
  1471.  
  1472. #ifdef STORE_WIDTH
  1473. glyph_node::glyph_node(charinfo *c, tfont *t, hunits w, node *x)
  1474.      : ci(c), tf(t), wid(w), node(x)
  1475. {
  1476. }
  1477. #endif
  1478.  
  1479. node *glyph_node::copy()
  1480. {
  1481. #ifdef STORE_WIDTH
  1482.   return new glyph_node(ci, tf, wid);
  1483. #else
  1484.   return new glyph_node(ci, tf);
  1485. #endif
  1486. }
  1487.  
  1488. node *glyph_node::merge_self(node *nd)
  1489. {
  1490.   return nd->merge_glyph_node(this);
  1491. }
  1492.  
  1493. int glyph_node::character_type()
  1494. {
  1495.   return tf->get_character_type(ci);
  1496. }
  1497.  
  1498. node *glyph_node::add_self(node *n, hyphen_list **p)
  1499. {
  1500.   assert(ci->get_hyphenation_code() == (*p)->hyphenation_code);
  1501.   next = 0;
  1502.   node *nn;
  1503.   if (n == 0 || (nn = n->merge_glyph_node(this)) == 0) {
  1504.     next = n;
  1505.     nn = this;
  1506.   }
  1507.   if ((*p)->hyphen)
  1508.     nn = nn->add_discretionary_hyphen();
  1509.   hyphen_list *pp = *p;
  1510.   *p = (*p)->next;
  1511.   delete pp;
  1512.   return nn;
  1513. }
  1514.  
  1515. int glyph_node::overlaps_horizontally()
  1516. {
  1517.   return ci->overlaps_horizontally();
  1518. }
  1519.  
  1520. int glyph_node::overlaps_vertically()
  1521. {
  1522.   return ci->overlaps_vertically();
  1523. }
  1524.  
  1525. units glyph_node::size()
  1526. {
  1527.   return tf->get_size().to_units();
  1528. }
  1529.  
  1530. hyphen_list *glyph_node::get_hyphen_list(hyphen_list *tail)
  1531. {
  1532.   return new hyphen_list(ci->get_hyphenation_code(), tail);
  1533. }
  1534.  
  1535.  
  1536. tfont *node::get_tfont()
  1537. {
  1538.   return 0;
  1539. }
  1540.  
  1541. tfont *glyph_node::get_tfont()
  1542. {
  1543.   return tf;
  1544. }
  1545.  
  1546. node *node::merge_glyph_node(glyph_node * /*gn*/)
  1547. {
  1548.   return 0;
  1549. }
  1550.  
  1551. node *glyph_node::merge_glyph_node(glyph_node *gn)
  1552. {
  1553.   if (tf == gn->tf) {
  1554.     charinfo *lig;
  1555.     if ((lig = tf->get_lig(ci, gn->ci)) != 0) {
  1556.       node *next1 = next;
  1557.       next = 0;
  1558.       return new ligature_node(lig, tf, this, gn, next1);
  1559.     }
  1560.     hunits kern;
  1561.     if (tf->get_kern(ci, gn->ci, &kern)) {
  1562.       node *next1 = next;
  1563.       next = 0;
  1564.       return new kern_pair_node(kern, this, gn, next1);
  1565.     }
  1566.   }
  1567.   return 0;
  1568. }
  1569.  
  1570. #ifdef STORE_WIDTH
  1571. inline
  1572. #endif
  1573. hunits glyph_node::width()
  1574. {
  1575. #ifdef STORE_WIDTH
  1576.   return wid;
  1577. #else
  1578.   return tf->get_width(ci);
  1579. #endif
  1580. }
  1581.  
  1582. node *glyph_node::last_char_node()
  1583. {
  1584.   return this;
  1585. }
  1586.  
  1587. void glyph_node::vertical_extent(vunits *min, vunits *max)
  1588. {
  1589.   *min = -tf->get_char_height(ci);
  1590.   *max = tf->get_char_depth(ci);
  1591. }
  1592.  
  1593. hunits glyph_node::skew()
  1594. {
  1595.   return tf->get_char_skew(ci);
  1596. }
  1597.  
  1598. hunits glyph_node::subscript_correction()
  1599. {
  1600.   return tf->get_subscript_correction(ci);
  1601. }
  1602.  
  1603. hunits glyph_node::italic_correction()
  1604. {
  1605.   return tf->get_italic_correction(ci);
  1606. }
  1607.  
  1608. hunits glyph_node::left_italic_correction()
  1609. {
  1610.   return tf->get_left_italic_correction(ci);
  1611. }
  1612.  
  1613. hyphenation_type glyph_node::get_hyphenation_type()
  1614. {
  1615.   return HYPHEN_MIDDLE;
  1616. }
  1617.  
  1618. int glyph_node::ends_sentence()
  1619. {
  1620.   if (ci->ends_sentence())
  1621.     return 1;
  1622.   else if (ci->transparent())
  1623.     return 2;
  1624.   else
  1625.     return 0;
  1626. }
  1627.  
  1628. void glyph_node::ascii_print(ascii_output_file *ascii)
  1629. {
  1630.   unsigned char c = ci->get_ascii_code();
  1631.   if (c != 0)
  1632.     ascii->outc(c);
  1633.   else
  1634.     ascii->outs(ci->nm.contents());
  1635. }
  1636.  
  1637. ligature_node::ligature_node(charinfo *c, tfont *t, 
  1638.                  node *gn1, node *gn2, node *x)
  1639.      : glyph_node(c, t, x), n1(gn1), n2(gn2)
  1640. {
  1641. }
  1642.  
  1643. #ifdef STORE_WIDTH
  1644. ligature_node::ligature_node(charinfo *c, tfont *t, hunits w,
  1645.                    node *gn1, node *gn2, node *x)
  1646.      : glyph_node(c, t, w, x), n1(gn1), n2(gn2)
  1647. {
  1648. }
  1649. #endif
  1650.  
  1651. ligature_node::~ligature_node()
  1652. {
  1653.   delete n1;
  1654.   delete n2;
  1655. }
  1656.  
  1657. node *ligature_node::copy()
  1658. {
  1659. #ifdef STORE_WIDTH
  1660.   return new ligature_node(ci, tf, wid, n1->copy(), n2->copy());
  1661. #else
  1662.   return new ligature_node(ci, tf, n1->copy(), n2->copy());
  1663. #endif
  1664. }
  1665.  
  1666. void ligature_node::ascii_print(ascii_output_file *ascii)
  1667. {
  1668.   n1->ascii_print(ascii);
  1669.   n2->ascii_print(ascii);
  1670. }
  1671.  
  1672. hyphen_list *ligature_node::get_hyphen_list(hyphen_list *tail)
  1673. {
  1674.   return n1->get_hyphen_list(n2->get_hyphen_list(tail));
  1675. }
  1676.  
  1677. node *ligature_node::add_self(node *n, hyphen_list **p)
  1678. {
  1679.   n = n1->add_self(n, p);
  1680.   n = n2->add_self(n, p);
  1681.   n1 = n2 = 0;
  1682.   delete this;
  1683.   return n;
  1684. }
  1685.  
  1686. kern_pair_node::kern_pair_node(hunits n, node *first, node *second, node *x)
  1687.      : node(x), n1(first), n2(second), amount(n)
  1688. {
  1689. }
  1690.  
  1691. dbreak_node::dbreak_node(node *n, node *p, node *x) 
  1692.      : node(x), none(n), pre(p), post(0)
  1693. {
  1694. }
  1695.  
  1696. node *dbreak_node::merge_glyph_node(glyph_node *gn)
  1697. {
  1698.   glyph_node *gn2 = (glyph_node *)gn->copy();
  1699.   node *new_none = none ? none->merge_glyph_node(gn) : 0;
  1700.   node *new_post = post ? post->merge_glyph_node(gn2) : 0;
  1701.   if (new_none == 0 && new_post == 0) {
  1702.     delete gn2;
  1703.     return 0;
  1704.   }
  1705.   if (new_none != 0)
  1706.     none = new_none;
  1707.   else {
  1708.     gn->next = none;
  1709.     none = gn;
  1710.   }
  1711.   if (new_post != 0)
  1712.     post = new_post;
  1713.   else {
  1714.     gn2->next = post;
  1715.     post = gn2;
  1716.   }
  1717.   return this;
  1718. }
  1719.  
  1720. node *kern_pair_node::merge_glyph_node(glyph_node *gn)
  1721. {
  1722.   node *nd = n2->merge_glyph_node(gn);
  1723.   if (nd == 0)
  1724.     return 0;
  1725.   n2 = nd;
  1726.   nd = n2->merge_self(n1);
  1727.   if (nd) {
  1728.     nd->next = next;
  1729.     n1 = 0;
  1730.     n2 = 0;
  1731.     delete this;
  1732.     return nd;
  1733.   }
  1734.   return this;
  1735. }
  1736.  
  1737.  
  1738. hunits kern_pair_node::italic_correction()
  1739. {
  1740.   return n2->italic_correction();
  1741. }
  1742.  
  1743. hunits kern_pair_node::subscript_correction()
  1744. {
  1745.   return n2->subscript_correction();
  1746. }
  1747.  
  1748. node *kern_pair_node::add_discretionary_hyphen()
  1749. {
  1750.   tfont *tf = n2->get_tfont();
  1751.   if (tf) {
  1752.     if (tf->contains(soft_hyphen_char)) {
  1753.       node *next1 = next;
  1754.       next = 0;
  1755.       node *n = copy();
  1756.       glyph_node *gn = new glyph_node(soft_hyphen_char, tf);
  1757.       node *nn = n->merge_glyph_node(gn);
  1758.       if (nn == 0) {
  1759.     gn->next = n;
  1760.     nn = gn;
  1761.       }
  1762.       return new dbreak_node(this, nn, next1);
  1763.     }
  1764.   }
  1765.   return this;
  1766. }
  1767.  
  1768.  
  1769. kern_pair_node::~kern_pair_node()
  1770. {
  1771.   if (n1 != 0)
  1772.     delete n1;
  1773.   if (n2 != 0)
  1774.     delete n2;
  1775. }
  1776.  
  1777. dbreak_node::~dbreak_node()
  1778. {
  1779.   delete_node_list(pre);
  1780.   delete_node_list(post);
  1781.   delete_node_list(none);
  1782. }
  1783.  
  1784. node *kern_pair_node::copy()
  1785. {
  1786.   return new kern_pair_node(amount, n1->copy(), n2->copy());
  1787. }
  1788.  
  1789. node *copy_node_list(node *n)
  1790. {
  1791.   node *p = 0;
  1792.   while (n != 0) {
  1793.     node *nn = n->copy();
  1794.     nn->next = p;
  1795.     p = nn;
  1796.     n = n->next;
  1797.   }
  1798.   while (p != 0) {
  1799.     node *pp = p->next;
  1800.     p->next = n;
  1801.     n = p;
  1802.     p = pp;
  1803.   }
  1804.   return n;
  1805. }
  1806.  
  1807. void delete_node_list(node *n)
  1808. {
  1809.   while (n != 0) {
  1810.     node *tem = n;
  1811.     n = n->next;
  1812.     delete tem;
  1813.   }
  1814. }
  1815.  
  1816. node *dbreak_node::copy()
  1817. {
  1818.   dbreak_node *p = new dbreak_node(copy_node_list(none), copy_node_list(pre));
  1819.   p->post = copy_node_list(post);
  1820.   return p;
  1821. }
  1822.  
  1823. hyphen_list *node::get_hyphen_list(hyphen_list *tail)
  1824. {
  1825.   return tail;
  1826. }
  1827.  
  1828.  
  1829. hyphen_list *kern_pair_node::get_hyphen_list(hyphen_list *tail)
  1830. {
  1831.   return n1->get_hyphen_list(n2->get_hyphen_list(tail));
  1832. }
  1833.  
  1834. class hyphen_inhibitor_node : public node {
  1835. public:
  1836.   hyphen_inhibitor_node(node *nd = 0);
  1837.   node *copy();
  1838.   int same(node *);
  1839.   const char *type();
  1840.   hyphenation_type get_hyphenation_type();
  1841. };
  1842.  
  1843. hyphen_inhibitor_node::hyphen_inhibitor_node(node *nd) : node(nd)
  1844. {
  1845. }
  1846.  
  1847. node *hyphen_inhibitor_node::copy()
  1848. {
  1849.   return new hyphen_inhibitor_node;
  1850. }
  1851.  
  1852. int hyphen_inhibitor_node::same(node *)
  1853. {
  1854.   return 1;
  1855. }
  1856.  
  1857. const char *hyphen_inhibitor_node::type()
  1858. {
  1859.   return "hyphen_inhibitor_node";
  1860. }
  1861.  
  1862. hyphenation_type hyphen_inhibitor_node::get_hyphenation_type()
  1863. {
  1864.   return HYPHEN_INHIBIT;
  1865. }
  1866.  
  1867. /* add_discretionary_hyphen methods */
  1868.  
  1869. node *dbreak_node::add_discretionary_hyphen()
  1870. {
  1871.   if (post)
  1872.     post = post->add_discretionary_hyphen();
  1873.   if (none)
  1874.     none = none->add_discretionary_hyphen();
  1875.   return this;
  1876. }
  1877.  
  1878.  
  1879. node *node::add_discretionary_hyphen()
  1880. {
  1881.   tfont *tf = get_tfont();
  1882.   if (!tf)
  1883.     return new hyphen_inhibitor_node(this);
  1884.   if (tf->contains(soft_hyphen_char)) {
  1885.     node *next1 = next;
  1886.     next = 0;
  1887.     node *n = copy();
  1888.     glyph_node *gn = new glyph_node(soft_hyphen_char, tf);
  1889.     node *n1 = n->merge_glyph_node(gn);
  1890.     if (n1 == 0) {
  1891.       gn->next = n;
  1892.       n1 = gn;
  1893.     }
  1894.     return new dbreak_node(this, n1, next1);
  1895.   }
  1896.   return this;
  1897. }
  1898.  
  1899.  
  1900. node *node::merge_self(node *)
  1901. {
  1902.   return 0;
  1903. }
  1904.  
  1905. node *node::add_self(node *n, hyphen_list ** /*p*/)
  1906. {
  1907.   next = n;
  1908.   return this;
  1909. }
  1910.  
  1911. node *kern_pair_node::add_self(node *n, hyphen_list **p)
  1912. {
  1913.   n = n1->add_self(n, p);
  1914.   n = n2->add_self(n, p);
  1915.   n1 = n2 = 0;
  1916.   delete this;
  1917.   return n;
  1918. }
  1919.  
  1920.  
  1921. hunits node::width()
  1922. {
  1923.   return H0;
  1924. }
  1925.  
  1926. node *node::last_char_node()
  1927. {
  1928.   return 0;
  1929. }
  1930.  
  1931. hunits hmotion_node::width()
  1932. {
  1933.   return n;
  1934. }
  1935.  
  1936. units node::size()
  1937. {
  1938.   return points_to_units(10);
  1939. }
  1940.  
  1941. hunits kern_pair_node::width()
  1942. {
  1943.   return n1->width() + n2->width() + amount;
  1944. }
  1945.  
  1946. node *kern_pair_node::last_char_node()
  1947. {
  1948.   node *nd = n2->last_char_node();
  1949.   if (nd)
  1950.     return nd;
  1951.   return n1->last_char_node();
  1952. }
  1953.  
  1954. hunits dbreak_node::width()
  1955. {
  1956.   hunits x = H0;
  1957.   for (node *n = none; n != 0; n = n->next)
  1958.     x += n->width();
  1959.   return x;
  1960. }
  1961.  
  1962. node *dbreak_node::last_char_node()
  1963. {
  1964.   for (node *n = none; n; n = n->next) {
  1965.     node *last = n->last_char_node();
  1966.     if (last)
  1967.       return last;
  1968.   }
  1969.   return 0;
  1970. }
  1971.  
  1972. hunits dbreak_node::italic_correction()
  1973. {
  1974.   return none ? none->italic_correction() : H0;
  1975. }
  1976.  
  1977. hunits dbreak_node::subscript_correction()
  1978. {
  1979.   return none ? none->subscript_correction() : H0;
  1980. }
  1981.  
  1982. class italic_corrected_node : public node {
  1983.   node *n;
  1984.   hunits x;
  1985. public:
  1986.   italic_corrected_node(node *, hunits, node * = 0);
  1987.   ~italic_corrected_node();
  1988.   node *copy();
  1989.   void ascii_print(ascii_output_file *);
  1990.   void asciify(macro *m);
  1991.   hunits width();
  1992.   node *last_char_node();
  1993.   void vertical_extent(vunits *, vunits *);
  1994.   int ends_sentence();
  1995.   int overlaps_horizontally();
  1996.   int overlaps_vertically();
  1997.   int same(node *);
  1998.   hyphenation_type get_hyphenation_type();
  1999.   tfont *get_tfont();
  2000.   hyphen_list *get_hyphen_list(hyphen_list *ss = 0);
  2001.   int character_type();
  2002.   void tprint(troff_output_file *);
  2003.   hunits subscript_correction();
  2004.   hunits skew();
  2005.   node *add_self(node *, hyphen_list **);
  2006.   const char *type();
  2007. };
  2008.  
  2009. node *node::add_italic_correction(hunits *width)
  2010. {
  2011.   hunits ic = italic_correction();
  2012.   if (ic.is_zero())
  2013.     return this;
  2014.   else {
  2015.     node *next1 = next;
  2016.     next = 0;
  2017.     *width += ic;
  2018.     return new italic_corrected_node(this, ic, next1);
  2019.   }
  2020. }
  2021.  
  2022. italic_corrected_node::italic_corrected_node(node *nn, hunits xx, node *p)
  2023. : n(nn), x(xx), node(p)
  2024. {
  2025.   assert(n != 0);
  2026. }
  2027.  
  2028. italic_corrected_node::~italic_corrected_node()
  2029. {
  2030.   delete n;
  2031. }
  2032.  
  2033. node *italic_corrected_node::copy()
  2034. {
  2035.   return new italic_corrected_node(n->copy(), x);
  2036. }
  2037.  
  2038. hunits italic_corrected_node::width()
  2039. {
  2040.   return n->width() + x;
  2041. }
  2042.  
  2043. void italic_corrected_node::vertical_extent(vunits *min, vunits *max)
  2044. {
  2045.   n->vertical_extent(min, max);
  2046. }
  2047.  
  2048. void italic_corrected_node::tprint(troff_output_file *out)
  2049. {
  2050.   n->tprint(out);
  2051.   out->right(x);
  2052. }
  2053.  
  2054. hunits italic_corrected_node::skew()
  2055. {
  2056.   return n->skew() - x/2;
  2057. }
  2058.  
  2059. hunits italic_corrected_node::subscript_correction()
  2060. {
  2061.   return n->subscript_correction() - x;
  2062. }
  2063.  
  2064. void italic_corrected_node::ascii_print(ascii_output_file *out)
  2065. {
  2066.   n->ascii_print(out);
  2067. }
  2068.  
  2069. int italic_corrected_node::ends_sentence()
  2070. {
  2071.   return n->ends_sentence();
  2072. }
  2073.  
  2074. int italic_corrected_node::overlaps_horizontally()
  2075. {
  2076.   return n->overlaps_horizontally();
  2077. }
  2078.  
  2079. int italic_corrected_node::overlaps_vertically()
  2080. {
  2081.   return n->overlaps_vertically();
  2082. }
  2083.  
  2084. node *italic_corrected_node::last_char_node()
  2085. {
  2086.   return n->last_char_node();
  2087. }
  2088.  
  2089. tfont *italic_corrected_node::get_tfont()
  2090. {
  2091.   return n->get_tfont();
  2092. }
  2093.  
  2094. hyphenation_type italic_corrected_node::get_hyphenation_type()
  2095. {
  2096.   return n->get_hyphenation_type();
  2097. }
  2098.  
  2099. node *italic_corrected_node::add_self(node *nd, hyphen_list **p)
  2100. {
  2101.   nd = n->add_self(nd, p);
  2102.   hunits not_interested;
  2103.   nd = nd->add_italic_correction(¬_interested);
  2104.   n = 0;
  2105.   delete this;
  2106.   return nd;
  2107. }
  2108.  
  2109. hyphen_list *italic_corrected_node::get_hyphen_list(hyphen_list *tail)
  2110. {
  2111.   return n->get_hyphen_list(tail);
  2112. }
  2113.  
  2114. int italic_corrected_node::character_type()
  2115. {
  2116.   return n->character_type();
  2117. }
  2118.  
  2119. class break_char_node : public node {
  2120.   node *ch;
  2121.   char break_code;
  2122. public:
  2123.   break_char_node(node *, int, node * = 0);
  2124.   ~break_char_node();
  2125.   node *copy();
  2126.   hunits width();
  2127.   vunits vertical_width();
  2128.   node *last_char_node();
  2129.   int character_type();
  2130.   int ends_sentence();
  2131.   node *add_self(node *, hyphen_list **);
  2132.   hyphen_list *get_hyphen_list(hyphen_list *s = 0);
  2133.   void tprint(troff_output_file *);
  2134.   void zero_width_tprint(troff_output_file *);
  2135.   void ascii_print(ascii_output_file *);
  2136.   void asciify(macro *m);
  2137.   hyphenation_type get_hyphenation_type();
  2138.   int overlaps_vertically();
  2139.   int overlaps_horizontally();
  2140.   units size();
  2141.   tfont *get_tfont();
  2142.   int same(node *);
  2143.   const char *type();
  2144. };
  2145.  
  2146. break_char_node::break_char_node(node *n, int c, node *x)
  2147. : node(x), ch(n), break_code(c)
  2148. {
  2149. }
  2150.  
  2151. break_char_node::~break_char_node()
  2152. {
  2153.   delete ch;
  2154. }
  2155.  
  2156. node *break_char_node::copy()
  2157. {
  2158.   return new break_char_node(ch->copy(), break_code);
  2159. }
  2160.  
  2161. hunits break_char_node::width()
  2162. {
  2163.   return ch->width();
  2164. }
  2165.  
  2166. vunits break_char_node::vertical_width()
  2167. {
  2168.   return ch->vertical_width();
  2169. }
  2170.  
  2171. node *break_char_node::last_char_node()
  2172. {
  2173.   return ch->last_char_node();
  2174. }
  2175.  
  2176. int break_char_node::character_type()
  2177. {
  2178.   return ch->character_type();
  2179. }
  2180.  
  2181. int break_char_node::ends_sentence()
  2182. {
  2183.   return ch->ends_sentence();
  2184. }
  2185.  
  2186. node *break_char_node::add_self(node *n, hyphen_list **p)
  2187. {
  2188.   assert((*p)->hyphenation_code == 0);
  2189.   if ((*p)->breakable && (break_code & 1)) {
  2190.     n = new space_node(H0, n);
  2191.     n->freeze_space();
  2192.   }
  2193.   next = n;
  2194.   n = this;
  2195.   if ((*p)->breakable && (break_code & 2)) {
  2196.     n = new space_node(H0, n);
  2197.     n->freeze_space();
  2198.   }
  2199.   hyphen_list *pp = *p;
  2200.   *p = (*p)->next;
  2201.   delete pp;
  2202.   return n;
  2203. }
  2204.  
  2205. hyphen_list *break_char_node::get_hyphen_list(hyphen_list *tail)
  2206. {
  2207.   return new hyphen_list(0, tail);
  2208. }
  2209.  
  2210. hyphenation_type break_char_node::get_hyphenation_type()
  2211. {
  2212.   return HYPHEN_MIDDLE;
  2213. }
  2214.  
  2215. void break_char_node::ascii_print(ascii_output_file *ascii)
  2216. {
  2217.   ch->ascii_print(ascii);
  2218. }
  2219.  
  2220. int break_char_node::overlaps_vertically()
  2221. {
  2222.   return ch->overlaps_vertically();
  2223. }
  2224.  
  2225. int break_char_node::overlaps_horizontally()
  2226. {
  2227.   return ch->overlaps_horizontally();
  2228. }
  2229.  
  2230. units break_char_node::size()
  2231. {
  2232.   return ch->size();
  2233. }
  2234.  
  2235. tfont *break_char_node::get_tfont()
  2236. {
  2237.   return ch->get_tfont();
  2238. }
  2239.  
  2240. node *extra_size_node::copy()
  2241.   return new extra_size_node(n); 
  2242. }
  2243.  
  2244. node *vertical_size_node::copy()
  2245.   return new vertical_size_node(n); 
  2246. }
  2247.  
  2248. node *hmotion_node::copy()
  2249.   return new hmotion_node(n);
  2250. }
  2251.  
  2252. node *space_char_hmotion_node::copy()
  2253. {
  2254.   return new space_char_hmotion_node(n);
  2255. }
  2256.  
  2257. node *vmotion_node::copy()
  2258.   return new vmotion_node(n);
  2259. }
  2260.   
  2261. node *dummy_node::copy()
  2262.   return new dummy_node;
  2263. }
  2264.  
  2265. node *transparent_dummy_node::copy()
  2266.   return new transparent_dummy_node;
  2267. }
  2268.  
  2269. hline_node::~hline_node()
  2270. {
  2271.   if (n)
  2272.     delete n;
  2273. }
  2274.  
  2275. node *hline_node::copy()
  2276. {
  2277.   return new hline_node(x, n ? n->copy() : 0);
  2278. }
  2279.  
  2280. hunits hline_node::width()
  2281. {
  2282.   return x < H0 ? H0 : x;
  2283. }
  2284.  
  2285.  
  2286. vline_node::~vline_node()
  2287. {
  2288.   if (n)
  2289.     delete n;
  2290. }
  2291.  
  2292. node *vline_node::copy()
  2293. {
  2294.   return new vline_node(x, n ? n->copy() : 0);
  2295. }
  2296.  
  2297. hunits vline_node::width()
  2298. {
  2299.   return n == 0 ? H0 : n->width();
  2300. }
  2301.  
  2302.  
  2303. zero_width_node::zero_width_node(node *nd) : n(nd)
  2304. {
  2305. }
  2306.  
  2307. zero_width_node::~zero_width_node()
  2308. {
  2309.   delete_node_list(n);
  2310. }
  2311.  
  2312. node *zero_width_node::copy()
  2313. {
  2314.   return new zero_width_node(copy_node_list(n));
  2315. }
  2316.  
  2317. int node_list_character_type(node *p)
  2318. {
  2319.   int t = 0;
  2320.   for (; p; p = p->next)
  2321.     t |= p->character_type();
  2322.   return t;
  2323. }
  2324.  
  2325. int zero_width_node::character_type()
  2326. {
  2327.   return node_list_character_type(n);
  2328. }
  2329.  
  2330. void node_list_vertical_extent(node *p, vunits *min, vunits *max)
  2331. {
  2332.   *min = V0;
  2333.   *max = V0;
  2334.   vunits cur_vpos = V0;
  2335.   vunits v1, v2;
  2336.   for (; p; p = p->next) {
  2337.     p->vertical_extent(&v1, &v2);
  2338.     v1 += cur_vpos;
  2339.     if (v1 < *min)
  2340.       *min = v1;
  2341.     v2 += cur_vpos;
  2342.     if (v2 > *max)
  2343.       *max = v2;
  2344.     cur_vpos += p->vertical_width();
  2345.   }
  2346. }
  2347.  
  2348. void zero_width_node::vertical_extent(vunits *min, vunits *max)
  2349. {
  2350.   node_list_vertical_extent(n, min, max);
  2351. }
  2352.  
  2353. overstrike_node::overstrike_node() : max_width(H0), list(0)
  2354. {
  2355. }
  2356.  
  2357. overstrike_node::~overstrike_node()
  2358. {
  2359.   delete_node_list(list);
  2360. }
  2361.  
  2362. node *overstrike_node::copy()
  2363. {
  2364.   overstrike_node *on = new overstrike_node;
  2365.   for (node *tem = list; tem; tem = tem->next)
  2366.     on->overstrike(tem->copy());
  2367.   return on;
  2368. }
  2369.  
  2370. void overstrike_node::overstrike(node *n)
  2371. {
  2372.   if (n == 0)
  2373.     return;
  2374.   hunits w = n->width();
  2375.   if (w > max_width)
  2376.     max_width = w;
  2377.   for (node **p = &list; *p; p = &(*p)->next)
  2378.     ;
  2379.   n->next = 0;
  2380.   *p = n;
  2381. }
  2382.  
  2383. hunits overstrike_node::width()
  2384. {
  2385.   return max_width;
  2386. }
  2387.  
  2388. bracket_node::bracket_node() : max_width(H0), list(0)
  2389. {
  2390. }
  2391.  
  2392. bracket_node::~bracket_node()
  2393. {
  2394.   delete_node_list(list);
  2395. }
  2396.  
  2397. node *bracket_node::copy()
  2398. {
  2399.   bracket_node *on = new bracket_node;
  2400.   for (node *tem = list; tem; tem = tem->next)
  2401.     on->bracket(tem->copy());
  2402.   return on;
  2403. }
  2404.  
  2405.  
  2406. void bracket_node::bracket(node *n)
  2407. {
  2408.   if (n == 0)
  2409.     return;
  2410.   hunits w = n->width();
  2411.   if (w > max_width)
  2412.     max_width = w;
  2413.   n->next = list;
  2414.   list = n;
  2415. }
  2416.  
  2417. hunits bracket_node::width()
  2418. {
  2419.   return max_width;
  2420. }
  2421.  
  2422. int node::nspaces()
  2423. {
  2424.   return 0;
  2425. }
  2426.  
  2427. int node::merge_space(hunits)
  2428. {
  2429.   return 0;
  2430. }
  2431.  
  2432. #if 0
  2433. space_node *space_node::free_list = 0;
  2434.  
  2435. void *space_node::operator new(size_t n)
  2436. {
  2437.   assert(n == sizeof(space_node));
  2438.   if (!free_list) {
  2439.     free_list = (space_node *)new char[sizeof(space_node)*BLOCK];
  2440.     for (int i = 0; i < BLOCK - 1; i++)
  2441.       free_list[i].next = free_list + i + 1;
  2442.     free_list[BLOCK-1].next = 0;
  2443.   }
  2444.   space_node *p = free_list;
  2445.   free_list = (space_node *)(free_list->next);
  2446.   p->next = 0;
  2447.   return p;
  2448. }
  2449.  
  2450. inline void space_node::operator delete(void *p)
  2451. {
  2452.   if (p) {
  2453.     ((space_node *)p)->next = free_list;
  2454.     free_list = (space_node *)p;
  2455.   }
  2456. }
  2457. #endif
  2458.  
  2459. space_node::space_node(hunits nn, node *p) : node(p), n(nn), set(0)
  2460. {
  2461. }
  2462.  
  2463. space_node::space_node(hunits nn, int s, node *p) : node(p), n(nn), set(s)
  2464. {
  2465. }
  2466.  
  2467. #if 0
  2468. space_node::~space_node()
  2469. {
  2470. }
  2471. #endif
  2472.  
  2473. node *space_node::copy()
  2474. {
  2475.   return new space_node(n, set);
  2476. }
  2477.  
  2478. int space_node::nspaces()
  2479. {
  2480.   return set ? 0 : 1;
  2481. }
  2482.  
  2483. int space_node::merge_space(hunits h)
  2484. {
  2485.   n += h;
  2486.   return 1;
  2487. }
  2488.  
  2489. hunits space_node::width()
  2490. {
  2491.   return n;
  2492. }
  2493.  
  2494. void node::spread_space(int*, hunits*)
  2495. {
  2496. }
  2497.  
  2498. void space_node::spread_space(int *nspaces, hunits *desired_space)
  2499. {
  2500.   if (!set) {
  2501.     assert(*nspaces > 0);
  2502.     if (*nspaces == 1) {
  2503.       n += *desired_space;
  2504.       *desired_space = H0;
  2505.     }
  2506.     else {
  2507.       hunits extra = *desired_space / *nspaces;
  2508.       *desired_space -= extra;
  2509.       n += extra;
  2510.     }
  2511.     *nspaces -= 1;
  2512.     set = 1;
  2513.   }
  2514. }
  2515.  
  2516. void node::freeze_space()
  2517. {
  2518. }
  2519.  
  2520. void space_node::freeze_space()
  2521. {
  2522.   set = 1;
  2523. }
  2524.  
  2525. diverted_space_node::diverted_space_node(vunits d, node *p)
  2526. : node(p), n(d)
  2527. {
  2528. }
  2529.  
  2530. node *diverted_space_node::copy()
  2531. {
  2532.   return new diverted_space_node(n);
  2533. }
  2534.  
  2535. diverted_copy_file_node::diverted_copy_file_node(symbol s, node *p)
  2536. : node(p), filename(s)
  2537. {
  2538. }
  2539.  
  2540. node *diverted_copy_file_node::copy()
  2541. {
  2542.   return new diverted_copy_file_node(filename);
  2543. }
  2544.  
  2545. int node::ends_sentence()
  2546. {
  2547.   return 0;
  2548. }
  2549.  
  2550. int kern_pair_node::ends_sentence()
  2551. {
  2552.   switch (n2->ends_sentence()) {
  2553.   case 0:
  2554.     return 0;
  2555.   case 1:
  2556.     return 1;
  2557.   case 2:
  2558.     break;
  2559.   default:
  2560.     assert(0);
  2561.   }
  2562.   return n1->ends_sentence();
  2563. }
  2564.  
  2565. int node_list_ends_sentence(node *n)
  2566. {
  2567.   for (; n != 0; n = n->next)
  2568.     switch (n->ends_sentence()) {
  2569.     case 0:
  2570.       return 0;
  2571.     case 1:
  2572.       return 1;
  2573.     case 2:
  2574.       break;
  2575.     default:
  2576.       assert(0);
  2577.     }
  2578.   return 2;
  2579. }
  2580.  
  2581.     
  2582. int dbreak_node::ends_sentence()
  2583. {
  2584.   return node_list_ends_sentence(none);
  2585. }
  2586.  
  2587.  
  2588. int node::overlaps_horizontally()
  2589. {
  2590.   return 0;
  2591. }
  2592.  
  2593. int node::overlaps_vertically()
  2594. {
  2595.   return 0;
  2596. }
  2597.  
  2598. int node::discardable()
  2599. {
  2600.   return 0;
  2601. }
  2602.  
  2603. int space_node::discardable()
  2604. {
  2605.   return set ? 0 : 1;
  2606. }
  2607.  
  2608.  
  2609. vunits node::vertical_width()
  2610. {
  2611.   return V0;
  2612. }
  2613.  
  2614. vunits vline_node::vertical_width()
  2615. {
  2616.   return x;
  2617. }
  2618.  
  2619. vunits vmotion_node::vertical_width()
  2620. {
  2621.   return n;
  2622. }
  2623.  
  2624. int node::character_type()
  2625. {
  2626.   return 0;
  2627. }
  2628.  
  2629. hunits node::subscript_correction()
  2630. {
  2631.   return H0;
  2632. }
  2633.  
  2634. hunits node::italic_correction()
  2635. {
  2636.   return H0;
  2637. }
  2638.  
  2639. hunits node::left_italic_correction()
  2640. {
  2641.   return H0;
  2642. }
  2643.  
  2644. hunits node::skew()
  2645. {
  2646.   return H0;
  2647. }
  2648.  
  2649.  
  2650. /* vertical_extent methods */
  2651.  
  2652. void node::vertical_extent(vunits *min, vunits *max)
  2653. {
  2654.   vunits v = vertical_width();
  2655.   if (v < V0) {
  2656.     *min = v;
  2657.     *max = V0;
  2658.   }
  2659.   else {
  2660.     *max = v;
  2661.     *min = V0;
  2662.   }
  2663. }
  2664.  
  2665. void vline_node::vertical_extent(vunits *min, vunits *max)
  2666. {
  2667.   if (n == 0)
  2668.     node::vertical_extent(min, max);
  2669.   else {
  2670.     vunits cmin, cmax;
  2671.     n->vertical_extent(&cmin, &cmax);
  2672.     vunits h = n->size();
  2673.     if (x < V0) {
  2674.       if (-x < h) {
  2675.     *min = x;
  2676.     *max = V0;
  2677.       }
  2678.       else {
  2679.     // we print the first character and then move up, so
  2680.     *max = cmax;
  2681.     // we print the last character and then move up h
  2682.     *min = cmin + h;
  2683.     if (*min > V0)
  2684.       *min = V0;
  2685.     *min += x;
  2686.       }
  2687.     }
  2688.     else {
  2689.       if (x < h) {
  2690.     *max = x;
  2691.     *min = V0;
  2692.       }
  2693.       else {
  2694.     // we move down by h and then print the first character, so
  2695.     *min = cmin + h;
  2696.     if (*min > V0)
  2697.       *min = V0;
  2698.     *max = x + cmax;
  2699.       }
  2700.     }
  2701.   }
  2702. }
  2703.  
  2704. /* ascii_print methods */
  2705.  
  2706.  
  2707. static void ascii_print_reverse_node_list(ascii_output_file *ascii, node *n)
  2708. {
  2709.   if (n == 0)
  2710.     return;
  2711.   ascii_print_reverse_node_list(ascii, n->next);
  2712.   n->ascii_print(ascii);
  2713. }
  2714.  
  2715. void dbreak_node::ascii_print(ascii_output_file *ascii)
  2716. {
  2717.   ascii_print_reverse_node_list(ascii, none);
  2718. }
  2719.  
  2720. void kern_pair_node::ascii_print(ascii_output_file *ascii)
  2721. {
  2722.   n1->ascii_print(ascii);
  2723.   n2->ascii_print(ascii);
  2724. }
  2725.  
  2726.  
  2727. void node::ascii_print(ascii_output_file *)
  2728. {
  2729. }
  2730.  
  2731. void space_node::ascii_print(ascii_output_file *ascii)
  2732. {
  2733.   if (!n.is_zero())
  2734.     ascii->outc(' ');
  2735. }
  2736.  
  2737. void hmotion_node::ascii_print(ascii_output_file *ascii)
  2738. {
  2739.   // this is pretty arbitrary
  2740.   if (n >= points_to_units(2))
  2741.     ascii->outc(' ');
  2742. }
  2743.  
  2744. void space_char_hmotion_node::ascii_print(ascii_output_file *ascii)
  2745. {
  2746.   ascii->outc(' ');
  2747. }
  2748.  
  2749. /* asciify methods */
  2750.  
  2751. void node::asciify(macro *m)
  2752. {
  2753.   m->append(this);
  2754. }
  2755.       
  2756. void glyph_node::asciify(macro *m)
  2757. {
  2758.   unsigned char c = ci->get_ascii_code();
  2759.   if (c != 0) {
  2760.     m->append(c);
  2761.     delete this;
  2762.   }
  2763.   else
  2764.     m->append(this);
  2765. }
  2766.  
  2767. void kern_pair_node::asciify(macro *m)
  2768. {
  2769.   n1->asciify(m);
  2770.   n2->asciify(m);
  2771.   n1 = n2 = 0;
  2772.   delete this;
  2773. }
  2774.  
  2775. static void asciify_reverse_node_list(macro *m, node *n)
  2776. {
  2777.   if (n == 0)
  2778.     return;
  2779.   asciify_reverse_node_list(m, n->next);
  2780.   n->asciify(m);
  2781. }
  2782.  
  2783. void dbreak_node::asciify(macro *m)
  2784. {
  2785.   asciify_reverse_node_list(m, none);
  2786.   none = 0;
  2787.   delete this;
  2788. }
  2789.  
  2790. void ligature_node::asciify(macro *m)
  2791. {
  2792.   n1->asciify(m);
  2793.   n2->asciify(m);
  2794.   n1 = n2 = 0;
  2795.   delete this;
  2796. }
  2797.  
  2798. void break_char_node::asciify(macro *m)
  2799. {
  2800.   ch->asciify(m);
  2801.   ch = 0;
  2802.   delete this;
  2803. }
  2804.  
  2805. void italic_corrected_node::asciify(macro *m)
  2806. {
  2807.   n->asciify(m);
  2808.   n = 0;
  2809.   delete this;
  2810. }
  2811.  
  2812. void left_italic_corrected_node::asciify(macro *m)
  2813. {
  2814.   if (n) {
  2815.     n->asciify(m);
  2816.     n = 0;
  2817.   }
  2818.   delete this;
  2819. }
  2820.  
  2821. space_char_hmotion_node::space_char_hmotion_node(hunits i, node *next)
  2822. : hmotion_node(i, next)
  2823. {
  2824. }
  2825.  
  2826. void space_char_hmotion_node::asciify(macro *m)
  2827. {
  2828.   m->append(' ');
  2829.   delete this;
  2830. }
  2831.  
  2832. void line_start_node::asciify(macro *)
  2833. {
  2834.   delete this;
  2835. }
  2836.  
  2837. void vertical_size_node::asciify(macro *)
  2838. {
  2839.   delete this;
  2840. }
  2841.  
  2842. breakpoint *node::get_breakpoints(hunits /*width*/, int /*nspaces*/,
  2843.                   breakpoint *rest, int /*is_inner*/)
  2844. {
  2845.   return rest;
  2846. }
  2847.  
  2848. int node::nbreaks()
  2849. {
  2850.   return 0;
  2851. }
  2852.  
  2853. breakpoint *space_node::get_breakpoints(hunits width, int ns, breakpoint *rest,
  2854.                 int is_inner)
  2855. {
  2856.   if (next->discardable())
  2857.     return rest;
  2858.   breakpoint *bp = new breakpoint;
  2859.   bp->next = rest;
  2860.   bp->width = width;
  2861.   bp->nspaces = ns;
  2862.   bp->hyphenated = 0;
  2863.   if (is_inner) {
  2864.     assert(rest != 0);
  2865.     bp->index = rest->index + 1;
  2866.     bp->nd = rest->nd;
  2867.   }
  2868.   else {
  2869.     bp->nd = this;
  2870.     bp->index = 0;
  2871.   }
  2872.   return bp;
  2873. }
  2874.  
  2875. int space_node::nbreaks()
  2876. {
  2877.   if (next->discardable())
  2878.     return 0;
  2879.   else
  2880.     return 1;
  2881. }
  2882.  
  2883. static breakpoint *node_list_get_breakpoints(node *p, hunits *widthp,
  2884.                          int ns, breakpoint *rest)
  2885. {
  2886.   if (p != 0) {
  2887.     rest = p->get_breakpoints(*widthp, 
  2888.                   ns, 
  2889.                   node_list_get_breakpoints(p->next, widthp, ns,
  2890.                             rest),
  2891.                   1);
  2892.     *widthp += p->width();
  2893.   }
  2894.   return rest;
  2895. }
  2896.  
  2897.  
  2898. breakpoint *dbreak_node::get_breakpoints(hunits width, int ns,
  2899.                      breakpoint *rest, int is_inner)
  2900. {
  2901.   breakpoint *bp = new breakpoint;
  2902.   bp->next = rest;
  2903.   bp->width = width;
  2904.   for (node *tem = pre; tem != 0; tem = tem->next)
  2905.     bp->width += tem->width();
  2906.   bp->nspaces = ns;
  2907.   bp->hyphenated = 1;
  2908.   if (is_inner) {
  2909.     assert(rest != 0);
  2910.     bp->index = rest->index + 1;
  2911.     bp->nd = rest->nd;
  2912.   }
  2913.   else {
  2914.     bp->nd = this;
  2915.     bp->index = 0;
  2916.   }
  2917.   return node_list_get_breakpoints(none, &width, ns, bp);
  2918. }
  2919.  
  2920. int dbreak_node::nbreaks()
  2921. {
  2922.   int i = 1;
  2923.   for (node *tem = none; tem != 0; tem = tem->next)
  2924.     i += tem->nbreaks();
  2925.   return i;
  2926. }
  2927.  
  2928. void node::split(int /*where*/, node ** /*prep*/, node ** /*postp*/)
  2929. {
  2930.   assert(0);
  2931. }
  2932.  
  2933. void space_node::split(int where, node **pre, node **post)
  2934. {
  2935.   assert(where == 0);
  2936.   *pre = next;
  2937.   *post = 0;
  2938.   delete this;
  2939. }
  2940.  
  2941. static void node_list_split(node *p, int *wherep, node **prep, node **postp)
  2942. {
  2943.   if (p == 0)
  2944.     return;
  2945.   int nb = p->nbreaks();
  2946.   node_list_split(p->next, wherep, prep, postp);
  2947.   if (*wherep < 0) {
  2948.     p->next = *postp;
  2949.     *postp = p;
  2950.   }
  2951.   else if (*wherep < nb) {
  2952.     p->next = *prep;
  2953.     p->split(*wherep, prep, postp);
  2954.   }
  2955.   else {
  2956.     p->next = *prep;
  2957.     *prep = p;
  2958.   }
  2959.   *wherep -= nb;
  2960. }
  2961.  
  2962. void dbreak_node::split(int where, node **prep, node **postp)
  2963. {
  2964.   assert(where >= 0);
  2965.   if (where == 0) {
  2966.     *postp = post;
  2967.     post = 0;
  2968.     if (pre == 0)
  2969.       *prep = next;
  2970.     else {
  2971.       for (node *tem = pre; tem->next != 0; tem = tem->next)
  2972.     ;
  2973.       tem->next = next;
  2974.       *prep = pre;
  2975.     }
  2976.     pre = 0;
  2977.     delete this;
  2978.   }
  2979.   else {
  2980.     *prep = next;
  2981.     where -= 1;
  2982.     node_list_split(none, &where, prep, postp);
  2983.     none = 0;
  2984.     delete this;
  2985.   }
  2986. }
  2987.   
  2988.  
  2989. hyphenation_type node::get_hyphenation_type()
  2990. {
  2991.   return HYPHEN_BOUNDARY;
  2992. }
  2993.  
  2994.  
  2995. hyphenation_type dbreak_node::get_hyphenation_type()
  2996. {
  2997.   return HYPHEN_INHIBIT;
  2998. }
  2999.  
  3000. hyphenation_type kern_pair_node::get_hyphenation_type()
  3001. {
  3002.   return HYPHEN_MIDDLE;
  3003. }
  3004.  
  3005. hyphenation_type dummy_node::get_hyphenation_type()
  3006. {
  3007.   return HYPHEN_MIDDLE;
  3008. }
  3009.  
  3010. hyphenation_type transparent_dummy_node::get_hyphenation_type()
  3011. {
  3012.   return HYPHEN_MIDDLE;
  3013. }
  3014.  
  3015. int node::interpret(macro *)
  3016. {
  3017.   return 0;
  3018. }
  3019.  
  3020. special_node::special_node(const macro &m)
  3021. : mac(m)
  3022. {
  3023. }
  3024.  
  3025. int special_node::same(node *n)
  3026. {
  3027.   return mac == ((special_node *)n)->mac;
  3028. }
  3029.  
  3030. const char *special_node::type()
  3031. {
  3032.   return "special_node";
  3033. }
  3034.  
  3035. node *special_node::copy()
  3036. {
  3037.   return new special_node(mac);
  3038. }
  3039.  
  3040. void special_node::tprint_start(troff_output_file *out)
  3041. {
  3042.   out->start_special();
  3043. }
  3044.  
  3045. void special_node::tprint_char(troff_output_file *out, unsigned char c)
  3046. {
  3047.   out->special_char(c);
  3048. }
  3049.  
  3050. void special_node::tprint_end(troff_output_file *out)
  3051. {
  3052.   out->end_special();
  3053. }
  3054.  
  3055. /* composite_node */
  3056.  
  3057. class composite_node : public node {
  3058.   charinfo *ci;
  3059.   node *n;
  3060.   tfont *tf;
  3061. public:
  3062.   composite_node(node *, charinfo *, tfont *, node * = 0);
  3063.   ~composite_node();
  3064.   node *copy();
  3065.   hunits width();
  3066.   node *last_char_node();
  3067.   units size();
  3068.   void tprint(troff_output_file *);
  3069.   hyphenation_type get_hyphenation_type();
  3070.   int overlaps_horizontally();
  3071.   int overlaps_vertically();
  3072.   void ascii_print(ascii_output_file *);
  3073.   void asciify(macro *);
  3074.   hyphen_list *get_hyphen_list(hyphen_list *tail);
  3075.   node *add_self(node *, hyphen_list **);
  3076.   tfont *get_tfont();
  3077.   int same(node *);
  3078.   const char *type();
  3079.   void vertical_extent(vunits *, vunits *);
  3080.   vunits vertical_width();
  3081. };
  3082.  
  3083. composite_node::composite_node(node *p, charinfo *c, tfont *t, node *x)
  3084. : node(x), n(p), ci(c), tf(t)
  3085. {
  3086. }
  3087.  
  3088. composite_node::~composite_node()
  3089. {
  3090.   delete_node_list(n);
  3091. }
  3092.  
  3093. node *composite_node::copy()
  3094. {
  3095.   return new composite_node(copy_node_list(n), ci, tf);
  3096. }
  3097.  
  3098. hunits composite_node::width()
  3099. {
  3100.   hunits x;
  3101.   if (tf->get_constant_space(&x))
  3102.     return x;
  3103.   x = H0;
  3104.   for (node *tem = n; tem; tem = tem->next)
  3105.     x += tem->width();
  3106.   hunits offset;
  3107.   if (tf->get_bold(&offset))
  3108.     x += offset;
  3109.   x += tf->get_track_kern();
  3110.   return x;
  3111. }
  3112.  
  3113. node *composite_node::last_char_node()
  3114. {
  3115.   return this;
  3116. }
  3117.  
  3118. vunits composite_node::vertical_width()
  3119. {
  3120.   vunits v = V0;
  3121.   for (node *tem = n; tem; tem = tem->next)
  3122.     v += tem->vertical_width();
  3123.   return v;
  3124. }
  3125.  
  3126. units composite_node::size()
  3127. {
  3128.   return tf->get_size().to_units();
  3129. }
  3130.  
  3131. hyphenation_type composite_node::get_hyphenation_type()
  3132. {
  3133.   return HYPHEN_MIDDLE;
  3134. }
  3135.  
  3136. int composite_node::overlaps_horizontally()
  3137. {
  3138.   return ci->overlaps_horizontally();
  3139. }
  3140.  
  3141. int composite_node::overlaps_vertically()
  3142. {
  3143.   return ci->overlaps_vertically();
  3144. }
  3145.  
  3146. void composite_node::asciify(macro *m)
  3147. {
  3148.   unsigned char c = ci->get_ascii_code();
  3149.   if (c != 0) {
  3150.     m->append(c);
  3151.     delete this;
  3152.   }
  3153.   else
  3154.     m->append(this);
  3155. }
  3156.  
  3157. void composite_node::ascii_print(ascii_output_file *ascii)
  3158. {
  3159.   unsigned char c = ci->get_ascii_code();
  3160.   if (c != 0)
  3161.     ascii->outc(c);
  3162.   else
  3163.     ascii->outs(ci->nm.contents());
  3164.  
  3165. }
  3166.  
  3167. hyphen_list *composite_node::get_hyphen_list(hyphen_list *tail)
  3168. {
  3169.   return new hyphen_list(ci->get_hyphenation_code(), tail);
  3170.  
  3171. }
  3172.  
  3173. node *composite_node::add_self(node *nn, hyphen_list **p)
  3174. {
  3175.   assert(ci->get_hyphenation_code() == (*p)->hyphenation_code);
  3176.   next = nn;
  3177.   nn = this;
  3178.   if ((*p)->hyphen)
  3179.     nn = nn->add_discretionary_hyphen();
  3180.   hyphen_list *pp = *p;
  3181.   *p = (*p)->next;
  3182.   delete pp;
  3183.   return nn;
  3184. }
  3185.  
  3186. tfont *composite_node::get_tfont()
  3187. {
  3188.   return tf;
  3189. }
  3190.  
  3191. node *reverse_node_list(node *n)
  3192. {
  3193.   node *r = 0;
  3194.   while (n) {
  3195.     node *tem = n;
  3196.     n = n->next;
  3197.     tem->next = r;
  3198.     r = tem;
  3199.   }
  3200.   return r;
  3201. }
  3202.  
  3203. void composite_node::vertical_extent(vunits *min, vunits *max)
  3204. {
  3205.   n = reverse_node_list(n);
  3206.   node_list_vertical_extent(n, min, max);
  3207.   n = reverse_node_list(n);
  3208. }
  3209.  
  3210. word_space_node::word_space_node(hunits d, node *x) : space_node(d, x)
  3211. {
  3212. }
  3213.  
  3214. word_space_node::word_space_node(hunits d, int s, node *x)
  3215. : space_node(d, s, x)
  3216. {
  3217. }
  3218.  
  3219. node *word_space_node::copy()
  3220. {
  3221.   return new word_space_node(n, set);
  3222. }
  3223.  
  3224. void word_space_node::tprint(troff_output_file *out)
  3225. {
  3226.   out->word_marker();
  3227.   space_node::tprint(out);
  3228. }
  3229.  
  3230. unbreakable_space_node::unbreakable_space_node(hunits d, node *x)
  3231. : word_space_node(d, x)
  3232. {
  3233. }
  3234.  
  3235. unbreakable_space_node::unbreakable_space_node(hunits d, int s, node *x)
  3236. : word_space_node(d, s, x)
  3237. {
  3238. }
  3239.  
  3240. node *unbreakable_space_node::copy()
  3241. {
  3242.   return new unbreakable_space_node(n, set);
  3243. }
  3244.  
  3245. breakpoint *unbreakable_space_node::get_breakpoints(hunits, int,
  3246.                             breakpoint *rest, int)
  3247. {
  3248.   return rest;
  3249. }
  3250.  
  3251. int unbreakable_space_node::nbreaks()
  3252. {
  3253.   return 0;
  3254. }
  3255.  
  3256. void unbreakable_space_node::split(int, node **, node **)
  3257. {
  3258.   assert(0);
  3259. }
  3260.  
  3261. int unbreakable_space_node::merge_space(hunits)
  3262. {
  3263.   return 0;
  3264. }
  3265.  
  3266. hvpair::hvpair()
  3267. {
  3268. }
  3269.  
  3270. draw_node::draw_node(char c, hvpair *p, int np, font_size s)
  3271.      : code(c), npoints(np), sz(s)
  3272. {
  3273.   point = new hvpair[npoints];
  3274.   for (int i = 0; i < npoints; i++)
  3275.     point[i] = p[i];
  3276. }
  3277.  
  3278. int draw_node::same(node *n)
  3279. {
  3280.   draw_node *nd = (draw_node *)n;
  3281.   if (code != nd->code || npoints != nd->npoints || sz != nd->sz)
  3282.     return 0;
  3283.   for (int i = 0; i < npoints; i++)
  3284.     if (point[i].h != nd->point[i].h || point[i].v != nd->point[i].v)
  3285.       return 0;
  3286.   return 1;
  3287. }
  3288.  
  3289. const char *draw_node::type()
  3290. {
  3291.   return "draw_node";
  3292. }
  3293.  
  3294. draw_node::~draw_node()
  3295. {
  3296.   if (point)
  3297.     a_delete point;
  3298. }
  3299.  
  3300. hunits draw_node::width()
  3301. {
  3302.   hunits x = H0;
  3303.   for (int i = 0; i < npoints; i++)
  3304.     x += point[i].h;
  3305.   return x;
  3306. }
  3307.  
  3308. vunits draw_node::vertical_width()
  3309. {
  3310.   if (code == 'e')
  3311.     return V0;
  3312.   vunits x = V0;
  3313.   for (int i = 0; i < npoints; i++)
  3314.     x += point[i].v;
  3315.   return x;
  3316. }
  3317.  
  3318. node *draw_node::copy()
  3319. {
  3320.   return new draw_node(code, point, npoints, sz);
  3321. }
  3322.  
  3323. void draw_node::tprint(troff_output_file *out)
  3324. {
  3325.   out->draw(code, point, npoints, sz);
  3326. }
  3327.   
  3328. /* tprint methods */
  3329.  
  3330. void glyph_node::tprint(troff_output_file *out)
  3331. {
  3332.   tfont *ptf = tf->get_plain();
  3333.   if (ptf == tf)
  3334.     out->put_char_width(ci, ptf, width(), H0);
  3335.   else {
  3336.     hunits offset;
  3337.     int bold = tf->get_bold(&offset);
  3338.     hunits w = ptf->get_width(ci);
  3339.     hunits k = H0;
  3340.     hunits x;
  3341.     int cs = tf->get_constant_space(&x);
  3342.     if (cs) {
  3343.       x -= w;
  3344.       if (bold)
  3345.     x -= offset;
  3346.       hunits x2 = x/2;
  3347.       out->right(x2);
  3348.       k = x - x2;
  3349.     }
  3350.     else
  3351.       k = tf->get_track_kern();
  3352.     if (bold) {
  3353.       out->put_char(ci, ptf);
  3354.       out->right(offset);
  3355.     }
  3356.     out->put_char_width(ci, ptf, w, k);
  3357.   }
  3358. }
  3359.  
  3360. void glyph_node::zero_width_tprint(troff_output_file *out)
  3361. {
  3362.   tfont *ptf = tf->get_plain();
  3363.   hunits offset;
  3364.   int bold = tf->get_bold(&offset);
  3365.   hunits x;
  3366.   int cs = tf->get_constant_space(&x);
  3367.   if (cs) {
  3368.     x -= ptf->get_width(ci);
  3369.     if (bold)
  3370.       x -= offset;
  3371.     x = x/2;
  3372.     out->right(x);
  3373.   }
  3374.   out->put_char(ci, ptf);
  3375.   if (bold) {
  3376.     out->right(offset);
  3377.     out->put_char(ci, ptf);
  3378.     out->right(-offset);
  3379.   }
  3380.   if (cs)
  3381.     out->right(-x);
  3382. }
  3383.  
  3384. void break_char_node::tprint(troff_output_file *t)
  3385. {
  3386.   ch->tprint(t);
  3387. }
  3388.  
  3389. void break_char_node::zero_width_tprint(troff_output_file *t)
  3390. {
  3391.   ch->zero_width_tprint(t);
  3392. }
  3393.  
  3394. void hline_node::tprint(troff_output_file *out)
  3395. {
  3396.   if (x < H0) {
  3397.     out->right(x);
  3398.     x = -x;
  3399.   }
  3400.   if (n == 0) {
  3401.     out->right(x);
  3402.     return;
  3403.   }
  3404.   hunits w = n->width();
  3405.   if (w <= H0) {
  3406.     error("horizontal line drawing character must have positive width");
  3407.     out->right(x);
  3408.     return;
  3409.   }
  3410.   int i = int(x/w);
  3411.   if (i == 0) {
  3412.     hunits xx = x - w;
  3413.     hunits xx2 = xx/2;
  3414.     out->right(xx2);
  3415.     n->tprint(out);
  3416.     out->right(xx - xx2);
  3417.   }
  3418.   else {
  3419.     hunits rem = x - w*i;
  3420.     if (rem > H0)
  3421.       if (n->overlaps_horizontally()) {
  3422.     n->tprint(out);
  3423.     out->right(rem - w);
  3424.       }
  3425.       else
  3426.     out->right(rem);
  3427.     while (--i >= 0)
  3428.       n->tprint(out);
  3429.   }
  3430. }
  3431.  
  3432. void vline_node::tprint(troff_output_file *out)
  3433. {
  3434.   if (n == 0) {
  3435.     out->down(x);
  3436.     return;
  3437.   }
  3438.   vunits h = n->size();
  3439.   int overlaps = n->overlaps_vertically();
  3440.   vunits y = x;
  3441.   if (y < V0) {
  3442.     y = -y;
  3443.     int i = y / h;
  3444.     vunits rem = y - i*h;
  3445.     if (i == 0) {
  3446.       out->right(n->width());
  3447.       out->down(-rem);
  3448.     }
  3449.     else {
  3450.       while (--i > 0) {
  3451.     n->zero_width_tprint(out);
  3452.     out->down(-h);
  3453.       }
  3454.       if (overlaps) {
  3455.     n->zero_width_tprint(out);
  3456.     out->down(-rem);
  3457.     n->tprint(out);
  3458.     out->down(-h);
  3459.       }
  3460.       else {
  3461.     n->tprint(out);
  3462.     out->down(-h - rem);
  3463.       }
  3464.     }
  3465.   }
  3466.   else {
  3467.     int i = y / h;
  3468.     vunits rem = y - i*h;
  3469.     if (i == 0) {
  3470.       out->down(rem);
  3471.       out->right(n->width());
  3472.     }
  3473.     else {
  3474.       out->down(h);
  3475.       if (overlaps)
  3476.     n->zero_width_tprint(out);
  3477.       out->down(rem);
  3478.       while (--i > 0) {
  3479.     n->zero_width_tprint(out);
  3480.     out->down(h);
  3481.       }
  3482.       n->tprint(out);
  3483.     }
  3484.   }
  3485. }
  3486.  
  3487. void zero_width_node::tprint(troff_output_file *out)
  3488. {
  3489.   if (!n)
  3490.     return;
  3491.   if (!n->next) {
  3492.     n->zero_width_tprint(out);
  3493.     return;
  3494.   }
  3495.   int hpos = out->get_hpos();
  3496.   int vpos = out->get_vpos();
  3497.   node *tem = n;
  3498.   while (tem) {
  3499.     tem->tprint(out);
  3500.     tem = tem->next;
  3501.   }
  3502.   out->moveto(hpos, vpos);
  3503. }
  3504.  
  3505. void overstrike_node::tprint(troff_output_file *out)
  3506. {
  3507.   hunits pos = H0;
  3508.   for (node *tem = list; tem; tem = tem->next) {
  3509.     hunits x = (max_width - tem->width())/2;
  3510.     out->right(x - pos);
  3511.     pos = x;
  3512.     tem->zero_width_tprint(out);
  3513.   }
  3514.   out->right(max_width - pos);
  3515. }
  3516.  
  3517. void bracket_node::tprint(troff_output_file *out)
  3518. {
  3519.   if (list == 0)
  3520.     return;
  3521.   int npieces = 0;
  3522.   for (node *tem = list; tem; tem = tem->next)
  3523.     ++npieces;
  3524.   vunits h = list->size();
  3525.   vunits totalh = h*npieces;
  3526.   vunits y = (totalh - h)/2;
  3527.   out->down(y);
  3528.   for (tem = list; tem; tem = tem->next) {
  3529.     tem->zero_width_tprint(out);
  3530.     out->down(-h);
  3531.   }
  3532.   out->right(max_width);
  3533.   out->down(totalh - y);
  3534. }
  3535.  
  3536. void node::tprint(troff_output_file *)
  3537. {
  3538. }
  3539.  
  3540. void node::zero_width_tprint(troff_output_file *out)
  3541. {
  3542.   int hpos = out->get_hpos();
  3543.   int vpos = out->get_vpos();
  3544.   tprint(out);
  3545.   out->moveto(hpos, vpos);
  3546. }
  3547.  
  3548. void space_node::tprint(troff_output_file *out)
  3549. {
  3550.   out->right(n);
  3551. }
  3552.  
  3553. void hmotion_node::tprint(troff_output_file *out)
  3554. {
  3555.   out->right(n);
  3556. }
  3557.  
  3558. void vmotion_node::tprint(troff_output_file *out)
  3559. {
  3560.   out->down(n);
  3561. }
  3562.  
  3563. void kern_pair_node::tprint(troff_output_file *out)
  3564. {
  3565.   n1->tprint(out);
  3566.   out->right(amount);
  3567.   n2->tprint(out);
  3568. }
  3569.  
  3570. static void tprint_reverse_node_list(troff_output_file *out, node *n)
  3571. {
  3572.   if (n == 0)
  3573.     return;
  3574.   tprint_reverse_node_list(out, n->next);
  3575.   n->tprint(out);
  3576. }
  3577.  
  3578. void dbreak_node::tprint(troff_output_file *out)
  3579. {
  3580.   tprint_reverse_node_list(out, none);
  3581. }
  3582.  
  3583. void composite_node::tprint(troff_output_file *out)
  3584. {
  3585.   hunits bold_offset;
  3586.   int is_bold = tf->get_bold(&bold_offset);
  3587.   hunits track_kern = tf->get_track_kern();
  3588.   hunits constant_space;
  3589.   int is_constant_spaced = tf->get_constant_space(&constant_space);
  3590.   hunits x = H0;
  3591.   if (is_constant_spaced) {
  3592.     x = constant_space;
  3593.     for (node *tem = n; tem; tem = tem->next)
  3594.       x -= tem->width();
  3595.     if (is_bold)
  3596.       x -= bold_offset;
  3597.     hunits x2 = x/2;
  3598.     out->right(x2);
  3599.     x -= x2;
  3600.   }
  3601.   if (is_bold) {
  3602.     int hpos = out->get_hpos();
  3603.     int vpos = out->get_vpos();
  3604.     tprint_reverse_node_list(out, n);
  3605.     out->moveto(hpos, vpos);
  3606.     out->right(bold_offset);
  3607.   }
  3608.   tprint_reverse_node_list(out, n);
  3609.   if (is_constant_spaced)
  3610.     out->right(x);
  3611.   else
  3612.     out->right(track_kern);
  3613. }
  3614.  
  3615. node *make_composite_node(charinfo *s, environment *env)
  3616. {
  3617.   int fontno = env_definite_font(env);
  3618.   if (fontno < 0) {
  3619.     error("no current font");
  3620.     return 0;
  3621.   }
  3622.   assert(fontno < font_table_size && font_table[fontno] != 0);
  3623.   node *n = charinfo_to_node_list(s, env);
  3624.   font_size fs = env->get_font_size();
  3625.   int char_height = env->get_char_height();
  3626.   int char_slant = env->get_char_slant();
  3627.   tfont *tf = font_table[fontno]->get_tfont(fs, char_height, char_slant,
  3628.                         fontno);
  3629.   if (env->is_composite())
  3630.     tf = tf->get_plain();
  3631.   return new composite_node(n, s, tf);
  3632. }
  3633.  
  3634. node *make_glyph_node(charinfo *s, environment *env, int no_error_message = 0)
  3635. {
  3636.   int fontno = env_definite_font(env);
  3637.   if (fontno < 0) {
  3638.     error("no current font");
  3639.     return 0;
  3640.   }
  3641.   assert(fontno < font_table_size && font_table[fontno] != 0);
  3642.   int fn = fontno;
  3643.   int found = font_table[fontno]->contains(s);
  3644.   if (!found) {
  3645.     if (s->numbered()) {
  3646.       if (!no_error_message)
  3647.     warning(WARN_CHAR, "can't find numbered character %1",
  3648.         s->get_number());
  3649.       return 0;
  3650.     }
  3651.     special_font_list *sf = font_table[fontno]->sf;
  3652.     while (sf != 0 && !found) {
  3653.       fn = sf->n;
  3654.       if (font_table[fn])
  3655.     found = font_table[fn]->contains(s);
  3656.       sf = sf->next;
  3657.     }
  3658.     if (!found) {
  3659.       sf = global_special_fonts;
  3660.       while (sf != 0 && !found) {
  3661.     fn = sf->n;
  3662.     if (font_table[fn])
  3663.       found = font_table[fn]->contains(s);
  3664.     sf = sf->next;
  3665.       }
  3666.     }
  3667.     if (!found
  3668. #if 0    
  3669.     && global_special_fonts == 0 && font_table[fontno]->sf == 0
  3670. #endif
  3671.     ) {
  3672.       for (fn = 0; fn < font_table_size; fn++)
  3673.     if (font_table[fn] 
  3674.         && font_table[fn]->is_special()
  3675.         && font_table[fn]->contains(s)) {
  3676.           found = 1;
  3677.           break;
  3678.         }
  3679.     }
  3680.     if (!found) {
  3681.       if (!no_error_message && s->first_time_not_found()) {
  3682.     unsigned char input_code = s->get_ascii_code();
  3683.     if (input_code != 0) {
  3684.       if (csgraph(input_code))
  3685.         warning(WARN_CHAR, "can't find character `%1'", input_code);
  3686.       else
  3687.         warning(WARN_CHAR, "can't find character with input code %1",
  3688.             int(input_code));
  3689.     }
  3690.     else
  3691.       warning(WARN_CHAR, "can't find special character `%1'",
  3692.           s->nm.contents());
  3693.       }
  3694.       return 0;
  3695.     }
  3696.   }
  3697.   font_size fs = env->get_font_size();
  3698.   int char_height = env->get_char_height();
  3699.   int char_slant = env->get_char_slant();
  3700.   tfont *tf = font_table[fontno]->get_tfont(fs, char_height, char_slant, fn);
  3701.   if (env->is_composite())
  3702.     tf = tf->get_plain();
  3703.   return new glyph_node(s, tf);
  3704. }
  3705.  
  3706. node *make_node(charinfo *ci, environment *env)
  3707. {
  3708.   switch (ci->get_special_translation()) {
  3709.   case charinfo::TRANSLATE_SPACE:
  3710.     return new space_char_hmotion_node(env->get_space_width());
  3711.   case charinfo::TRANSLATE_DUMMY:
  3712.     return new dummy_node;
  3713.   case charinfo::TRANSLATE_HYPHEN_INDICATOR:
  3714.     error("translation to \\% ignored in this context");
  3715.     break;
  3716.   }
  3717.   charinfo *tem = ci->get_translation();
  3718.   if (tem)
  3719.     ci = tem;
  3720.   macro *mac = ci->get_macro();
  3721.   if (mac)
  3722.     return make_composite_node(ci, env);
  3723.   else
  3724.     return make_glyph_node(ci, env);
  3725. }
  3726.  
  3727. int character_exists(charinfo *ci, environment *env)
  3728. {
  3729.   if (ci->get_special_translation() != charinfo::TRANSLATE_NONE)
  3730.     return 1;
  3731.   charinfo *tem = ci->get_translation();
  3732.   if (tem)
  3733.     ci = tem;
  3734.   if (ci->get_macro())
  3735.     return 1;
  3736.   node *nd = make_glyph_node(ci, env, 1);
  3737.   if (nd) {
  3738.     delete nd;
  3739.     return 1;
  3740.   }
  3741.   return 0;
  3742. }
  3743.  
  3744. node *node::add_char(charinfo *ci, environment *env, hunits *widthp)
  3745. {
  3746.   node *res;
  3747.   switch (ci->get_special_translation()) {
  3748.   case charinfo::TRANSLATE_SPACE:
  3749.     res = new space_char_hmotion_node(env->get_space_width(), this);
  3750.     *widthp += res->width();
  3751.     return res;
  3752.   case charinfo::TRANSLATE_DUMMY:
  3753.     return new dummy_node(this);
  3754.   case charinfo::TRANSLATE_HYPHEN_INDICATOR:
  3755.     return add_discretionary_hyphen();
  3756.   }
  3757.   charinfo *tem = ci->get_translation();
  3758.   if (tem)
  3759.     ci = tem;
  3760.   macro *mac = ci->get_macro();
  3761.   if (mac) {
  3762.     res = make_composite_node(ci, env);
  3763.     if (res) {
  3764.       res->next = this;
  3765.       *widthp += res->width();
  3766.     }
  3767.     else
  3768.       return this;
  3769.   }
  3770.   else {
  3771.     node *gn = make_glyph_node(ci, env);
  3772.     if (gn == 0)
  3773.       return this;
  3774.     else {
  3775.       hunits old_width = width();
  3776.       node *p = gn->merge_self(this);
  3777.       if (p == 0) {
  3778.     *widthp += gn->width();
  3779.     gn->next = this;
  3780.     res = gn;
  3781.       }
  3782.       else {
  3783.     *widthp += p->width() - old_width;
  3784.     res = p;
  3785.       }
  3786.     }
  3787.   }
  3788.   int break_code = 0;
  3789.   if (ci->can_break_before())
  3790.     break_code = 1;
  3791.   if (ci->can_break_after())
  3792.     break_code |= 2;
  3793.   if (break_code) {
  3794.     node *next1 = res->next;
  3795.     res->next = 0;
  3796.     res = new break_char_node(res, break_code, next1);
  3797.   }
  3798.   return res;
  3799. }
  3800.  
  3801.  
  3802. #ifdef __GNUG__
  3803. inline
  3804. #endif
  3805. int same_node(node *n1, node *n2)
  3806. {
  3807.   if (n1 != 0) {
  3808.     if (n2 != 0)
  3809.       return n1->type() == n2->type() && n1->same(n2);
  3810.     else
  3811.       return 0;
  3812.   }
  3813.   else
  3814.     return n2 == 0;
  3815. }
  3816.  
  3817. int same_node_list(node *n1, node *n2)
  3818. {
  3819.   while (n1 && n2) {
  3820.     if (n1->type() != n2->type() || !n1->same(n2))
  3821.       return 0;
  3822.     n1 = n1->next;
  3823.     n2 = n2->next;
  3824.   }
  3825.   return !n1 && !n2;
  3826. }
  3827.  
  3828. int extra_size_node::same(node *nd)
  3829. {
  3830.   return n == ((extra_size_node *)nd)->n;
  3831. }
  3832.  
  3833. const char *extra_size_node::type()
  3834. {
  3835.   return "extra_size_node";
  3836. }
  3837.  
  3838. int vertical_size_node::same(node *nd)
  3839. {
  3840.   return n == ((vertical_size_node *)nd)->n;
  3841. }
  3842.  
  3843. const char *vertical_size_node::type()
  3844. {
  3845.   return "vertical_size_node";
  3846. }
  3847.  
  3848. int hmotion_node::same(node *nd)
  3849. {
  3850.   return n == ((hmotion_node *)nd)->n;
  3851. }
  3852.  
  3853. const char *hmotion_node::type()
  3854. {
  3855.   return "hmotion_node";
  3856. }
  3857.  
  3858. int space_char_hmotion_node::same(node *nd)
  3859. {
  3860.   return n == ((space_char_hmotion_node *)nd)->n;
  3861. }
  3862.  
  3863. const char *space_char_hmotion_node::type()
  3864. {
  3865.   return "space_char_hmotion_node";
  3866. }
  3867.  
  3868. int vmotion_node::same(node *nd)
  3869. {
  3870.   return n == ((vmotion_node *)nd)->n;
  3871. }
  3872.  
  3873. const char *vmotion_node::type()
  3874. {
  3875.   return "vmotion_node";
  3876. }
  3877.  
  3878. int hline_node::same(node *nd)
  3879. {
  3880.   return x == ((hline_node *)nd)->x && same_node(n, ((hline_node *)nd)->n);
  3881. }
  3882.  
  3883. const char *hline_node::type()
  3884. {
  3885.   return "hline_node";
  3886. }
  3887.  
  3888. int vline_node::same(node *nd)
  3889. {
  3890.   return x == ((vline_node *)nd)->x && same_node(n, ((vline_node *)nd)->n);
  3891. }
  3892.  
  3893. const char *vline_node::type()
  3894. {
  3895.   return "vline_node";
  3896. }
  3897.  
  3898. int dummy_node::same(node * /*nd*/)
  3899. {
  3900.   return 1;
  3901. }
  3902.  
  3903. const char *dummy_node::type()
  3904. {
  3905.   return "dummy_node";
  3906. }
  3907.  
  3908. int transparent_dummy_node::same(node * /*nd*/)
  3909. {
  3910.   return 1;
  3911. }
  3912.  
  3913. const char *transparent_dummy_node::type()
  3914. {
  3915.   return "transparent_dummy_node";
  3916. }
  3917.  
  3918. int transparent_dummy_node::ends_sentence()
  3919. {
  3920.   return 2;
  3921. }
  3922.  
  3923. int zero_width_node::same(node *nd)
  3924. {
  3925.   return same_node_list(n, ((zero_width_node *)nd)->n);
  3926. }
  3927.  
  3928. const char *zero_width_node::type()
  3929. {
  3930.   return "zero_width_node";
  3931. }
  3932.  
  3933. int italic_corrected_node::same(node *nd)
  3934. {
  3935.   return (x == ((italic_corrected_node *)nd)->x
  3936.       && same_node(n, ((italic_corrected_node *)nd)->n));
  3937. }
  3938.  
  3939. const char *italic_corrected_node::type()
  3940. {
  3941.   return "italic_corrected_node";
  3942. }
  3943.  
  3944.  
  3945. left_italic_corrected_node::left_italic_corrected_node(node *x)
  3946. : n(0), node(x)
  3947. {
  3948. }
  3949.  
  3950. left_italic_corrected_node::~left_italic_corrected_node()
  3951. {
  3952.   delete n;
  3953. }
  3954.  
  3955. node *left_italic_corrected_node::merge_glyph_node(glyph_node *gn)
  3956. {
  3957.   if (n == 0) {
  3958.     hunits lic = gn->left_italic_correction();
  3959.     if (!lic.is_zero()) {
  3960.       x = lic;
  3961.       n = gn;
  3962.       return this;
  3963.     }
  3964.   }
  3965.   else {
  3966.     node *nd = n->merge_glyph_node(gn);
  3967.     if (nd) {
  3968.       n = nd;
  3969.       x = n->left_italic_correction();
  3970.       return this;
  3971.     }
  3972.   }
  3973.   return 0;
  3974. }
  3975.  
  3976. node *left_italic_corrected_node::copy()
  3977. {
  3978.   left_italic_corrected_node *nd = new left_italic_corrected_node;
  3979.   if (n) {
  3980.     nd->n = n->copy();
  3981.     nd->x = x;
  3982.   }
  3983.   return nd;
  3984. }
  3985.  
  3986. void left_italic_corrected_node::tprint(troff_output_file *out)
  3987. {
  3988.   if (n) {
  3989.     out->right(x);
  3990.     n->tprint(out);
  3991.   }
  3992. }
  3993.  
  3994. const char *left_italic_corrected_node::type()
  3995. {
  3996.   return "left_italic_corrected_node";
  3997. }
  3998.  
  3999. int left_italic_corrected_node::same(node *nd)
  4000. {
  4001.   return (x == ((left_italic_corrected_node *)nd)->x
  4002.       && same_node(n, ((left_italic_corrected_node *)nd)->n));
  4003. }
  4004.  
  4005. void left_italic_corrected_node::ascii_print(ascii_output_file *out)
  4006. {
  4007.   if (n)
  4008.     n->ascii_print(out);
  4009. }
  4010.  
  4011. hunits left_italic_corrected_node::width()
  4012. {
  4013.   return n ? n->width() + x : H0;
  4014. }
  4015.  
  4016. void left_italic_corrected_node::vertical_extent(vunits *min, vunits *max)
  4017. {
  4018.   if (n)
  4019.     n->vertical_extent(min, max);
  4020.   else
  4021.     node::vertical_extent(min, max);
  4022. }
  4023.  
  4024. hunits left_italic_corrected_node::skew()
  4025. {
  4026.   return n ? n->skew() + x/2 : H0;
  4027. }
  4028.  
  4029. hunits left_italic_corrected_node::subscript_correction()
  4030. {
  4031.   return n ? n->subscript_correction() : H0;
  4032. }
  4033.  
  4034. hunits left_italic_corrected_node::italic_correction()
  4035. {
  4036.   return n ? n->italic_correction() : H0;
  4037. }
  4038.  
  4039. int left_italic_corrected_node::ends_sentence()
  4040. {
  4041.   return n ? n->ends_sentence() : 0;
  4042. }
  4043.  
  4044. int left_italic_corrected_node::overlaps_horizontally()
  4045. {
  4046.   return n ? n->overlaps_horizontally() : 0;
  4047. }
  4048.  
  4049. int left_italic_corrected_node::overlaps_vertically()
  4050. {
  4051.   return n ? n->overlaps_vertically() : 0;
  4052. }
  4053.  
  4054. node *left_italic_corrected_node::last_char_node()
  4055. {
  4056.   return n ? n->last_char_node() : 0;
  4057. }
  4058.  
  4059. tfont *left_italic_corrected_node::get_tfont()
  4060. {
  4061.   return n ? n->get_tfont() : 0;
  4062. }
  4063.  
  4064. hyphenation_type left_italic_corrected_node::get_hyphenation_type()
  4065. {
  4066.   if (n)
  4067.     return n->get_hyphenation_type();
  4068.   else
  4069.     return HYPHEN_MIDDLE;
  4070. }
  4071.  
  4072. hyphen_list *left_italic_corrected_node::get_hyphen_list(hyphen_list *tail)
  4073. {
  4074.   return n ? n->get_hyphen_list(tail) : tail;
  4075. }
  4076.  
  4077. node *left_italic_corrected_node::add_self(node *nd, hyphen_list **p)
  4078. {
  4079.   if (n) {
  4080.     nd = new left_italic_corrected_node(nd);
  4081.     nd = n->add_self(nd, p);
  4082.     n = 0;
  4083.     delete this;
  4084.   }
  4085.   return nd;
  4086. }
  4087.  
  4088. int left_italic_corrected_node::character_type()
  4089. {
  4090.   return n ? n->character_type() : 0;
  4091. }
  4092.  
  4093. int overstrike_node::same(node *nd)
  4094. {
  4095.   return same_node_list(list, ((overstrike_node *)nd)->list);
  4096. }
  4097.  
  4098. const char *overstrike_node::type()
  4099. {
  4100.   return "overstrike_node";
  4101. }
  4102.  
  4103. int bracket_node::same(node *nd)
  4104. {
  4105.   return same_node_list(list, ((bracket_node *)nd)->list);
  4106. }
  4107.  
  4108. const char *bracket_node::type()
  4109. {
  4110.   return "bracket_node";
  4111. }
  4112.  
  4113. int composite_node::same(node *nd)
  4114. {
  4115.   return ci == ((composite_node *)nd)->ci
  4116.     && same_node_list(n, ((composite_node *)nd)->n);
  4117. }
  4118.  
  4119. const char *composite_node::type()
  4120. {
  4121.   return "composite_node";
  4122. }
  4123.  
  4124. int glyph_node::same(node *nd)
  4125. {
  4126.   return ci == ((glyph_node *)nd)->ci && tf == ((glyph_node *)nd)->tf;
  4127. }
  4128.  
  4129. const char *glyph_node::type()
  4130. {
  4131.   return "glyph_node";
  4132. }
  4133.  
  4134. int ligature_node::same(node *nd)
  4135. {
  4136.   return (same_node(n1, ((ligature_node *)nd)->n1) 
  4137.       && same_node(n2, ((ligature_node *)nd)->n2)
  4138.       && glyph_node::same(nd));
  4139. }
  4140.  
  4141. const char *ligature_node::type()
  4142. {
  4143.   return "ligature_node";
  4144. }
  4145.  
  4146. int kern_pair_node::same(node *nd)
  4147. {
  4148.   return (amount == ((kern_pair_node *)nd)->amount
  4149.       && same_node(n1, ((kern_pair_node *)nd)->n1)
  4150.       && same_node(n2, ((kern_pair_node *)nd)->n2));
  4151. }
  4152.  
  4153. const char *kern_pair_node::type()
  4154. {
  4155.   return "kern_pair_node";
  4156. }
  4157.  
  4158. int dbreak_node::same(node *nd)
  4159. {
  4160.   return (same_node_list(none, ((dbreak_node *)nd)->none)
  4161.       && same_node_list(pre, ((dbreak_node *)nd)->pre)
  4162.       && same_node_list(post, ((dbreak_node *)nd)->post));
  4163. }
  4164.  
  4165. const char *dbreak_node::type()
  4166. {
  4167.   return "dbreak_node";
  4168. }
  4169.  
  4170. int break_char_node::same(node *nd)
  4171. {
  4172.   return (break_code == ((break_char_node *)nd)->break_code
  4173.       && same_node(ch, ((break_char_node *)nd)->ch));
  4174. }
  4175.  
  4176. const char *break_char_node::type()
  4177. {
  4178.   return "break_char_node";
  4179. }
  4180.  
  4181. int line_start_node::same(node * /*nd*/)
  4182. {
  4183.   return 1;
  4184. }
  4185.  
  4186. const char *line_start_node::type()
  4187. {
  4188.   return "line_start_node";
  4189. }
  4190.  
  4191. int space_node::same(node *nd)
  4192. {
  4193.   return n == ((space_node *)nd)->n && set == ((space_node *)nd)->set;
  4194. }
  4195.  
  4196. const char *space_node::type()
  4197. {
  4198.   return "space_node";
  4199. }
  4200.  
  4201. int word_space_node::same(node *nd)
  4202. {
  4203.   return (n == ((word_space_node *)nd)->n
  4204.       && set == ((word_space_node *)nd)->set);
  4205. }
  4206.  
  4207. const char *word_space_node::type()
  4208. {
  4209.   return "word_space_node";
  4210. }
  4211.  
  4212. int unbreakable_space_node::same(node *nd)
  4213. {
  4214.   return (n == ((unbreakable_space_node *)nd)->n
  4215.       && set == ((unbreakable_space_node *)nd)->set);
  4216. }
  4217.  
  4218. const char *unbreakable_space_node::type()
  4219. {
  4220.   return "unbreakable_space_node";
  4221. }
  4222.  
  4223. int diverted_space_node::same(node *nd)
  4224. {
  4225.   return n == ((diverted_space_node *)nd)->n;
  4226. }
  4227.  
  4228. const char *diverted_space_node::type()
  4229. {
  4230.   return "diverted_space_node";
  4231. }
  4232.  
  4233. int diverted_copy_file_node::same(node *nd)
  4234. {
  4235.   return filename == ((diverted_copy_file_node *)nd)->filename;
  4236. }
  4237.  
  4238. const char *diverted_copy_file_node::type()
  4239. {
  4240.   return "diverted_copy_file_node";
  4241. }
  4242.  
  4243. // Grow the font_table so that its size is > n.
  4244.  
  4245. static void grow_font_table(int n)
  4246. {
  4247.   assert(n >= font_table_size);
  4248.   font_info **old_font_table = font_table;
  4249.   int old_font_table_size = font_table_size;
  4250.   font_table_size = font_table_size ? (font_table_size*3)/2 : 10;
  4251.   if (font_table_size <= n)
  4252.     font_table_size = n + 10;
  4253.   font_table = new font_info *[font_table_size];
  4254.   if (old_font_table_size)
  4255.     memcpy(font_table, old_font_table,
  4256.        old_font_table_size*sizeof(font_info *));
  4257.   a_delete old_font_table;
  4258.   for (int i = old_font_table_size; i < font_table_size; i++)
  4259.     font_table[i] = 0;
  4260. }
  4261.  
  4262. dictionary font_translation_dictionary(17);
  4263.  
  4264. static symbol get_font_translation(symbol nm)
  4265. {
  4266.   void *p = font_translation_dictionary.lookup(nm);
  4267.   return p ? symbol((char *)p) : nm;
  4268. }
  4269.  
  4270. dictionary font_dictionary(50);
  4271.  
  4272. static int mount_font_no_translate(int n, symbol name, symbol external_name)
  4273. {
  4274.   assert(n >= 0);
  4275.   // We store the address of this char in font_dictionary to indicate
  4276.   // that we've previously tried to mount the font and failed.
  4277.   static char a_char;
  4278.   font *fm = 0;
  4279.   void *p = font_dictionary.lookup(external_name);
  4280.   if (p == 0) {
  4281.     int not_found;
  4282.     fm = font::load_font(external_name.contents(), ¬_found);
  4283.     if (!fm) {
  4284.       if (not_found)
  4285.     warning(WARN_FONT, "can't find font `%1'", external_name.contents());
  4286.       font_dictionary.lookup(external_name, &a_char);
  4287.       return 0;
  4288.     }
  4289.     font_dictionary.lookup(name, fm);
  4290.   }
  4291.   else if (p == &a_char) {
  4292. #if 0
  4293.     error("invalid font `%1'", external_name.contents());
  4294. #endif
  4295.     return 0;
  4296.   }
  4297.   else
  4298.     fm = (font*)p;
  4299.   if (n >= font_table_size) {
  4300.     if (n - font_table_size > 1000) {
  4301.       error("font position too much larger than first unused position");
  4302.       return 0;
  4303.     }
  4304.     grow_font_table(n);
  4305.   }
  4306.   else if (font_table[n] != 0)
  4307.     delete font_table[n];
  4308.   font_table[n] = new font_info(name, n, external_name, fm);
  4309.   font_family::invalidate_fontno(n);
  4310.   return 1;
  4311. }
  4312.  
  4313. int mount_font(int n, symbol name, symbol external_name)
  4314. {
  4315.   assert(n >= 0);
  4316.   name = get_font_translation(name);
  4317.   if (external_name.is_null())
  4318.     external_name = name;
  4319.   else
  4320.     external_name = get_font_translation(external_name);
  4321.   return mount_font_no_translate(n, name, external_name);
  4322. }
  4323.  
  4324. void mount_style(int n, symbol name)
  4325. {
  4326.   assert(n >= 0);
  4327.   if (n >= font_table_size) {
  4328.     if (n - font_table_size > 1000) {
  4329.       error("font position too much larger than first unused position");
  4330.       return;
  4331.     }
  4332.     grow_font_table(n);
  4333.   }
  4334.   else if (font_table[n] != 0)
  4335.     delete font_table[n];
  4336.   font_table[n] = new font_info(get_font_translation(name), n, NULL_SYMBOL, 0);
  4337.   font_family::invalidate_fontno(n);
  4338. }
  4339.  
  4340. /* global functions */
  4341.  
  4342. void font_translate()
  4343. {
  4344.   symbol from = get_name(1);
  4345.   if (!from.is_null()) {
  4346.     symbol to = get_name();
  4347.     if (to.is_null() || from == to)
  4348.       font_translation_dictionary.remove(from);
  4349.     else
  4350.       font_translation_dictionary.lookup(from, (void *)to.contents());
  4351.   }
  4352.   skip_line();
  4353. }
  4354.  
  4355. void font_position()
  4356. {
  4357.   int n;
  4358.   if (get_integer(&n)) {
  4359.     if (n < 0)
  4360.       error("negative font position");
  4361.     else {
  4362.       symbol internal_name = get_name(1);
  4363.       if (!internal_name.is_null()) {
  4364.     symbol external_name = get_long_name(0);
  4365.     mount_font(n, internal_name, external_name); // ignore error
  4366.       }
  4367.     }
  4368.   }
  4369.   skip_line();
  4370. }
  4371.  
  4372. font_family::font_family(symbol s)
  4373. : nm(s), map_size(10)
  4374. {
  4375.   map = new int[map_size];
  4376.   for (int i = 0; i < map_size; i++)
  4377.     map[i] = -1;
  4378. }
  4379.  
  4380. font_family::~font_family()
  4381. {
  4382.   a_delete map;
  4383. }
  4384.  
  4385. int font_family::make_definite(int i)
  4386. {
  4387.   if (i >= 0) {
  4388.     if (i < map_size && map[i] >= 0)
  4389.       return map[i];
  4390.     else {
  4391.       if (i < font_table_size && font_table[i] != 0) {
  4392.     if (i >= map_size) {
  4393.       int old_map_size = map_size;
  4394.       int *old_map = map;
  4395.       map_size *= 3;
  4396.       map_size /= 2;
  4397.       if (i >= map_size)
  4398.         map_size = i + 10;
  4399.       map = new int[map_size];
  4400.       memcpy(map, old_map, old_map_size*sizeof(int));
  4401.       a_delete old_map;
  4402.       for (int j = old_map_size; j < map_size; j++)
  4403.         map[j] = -1;
  4404.     }
  4405.     if (font_table[i]->is_style()) {
  4406.       symbol sty = font_table[i]->get_name();
  4407.       symbol f = concat(nm, sty);
  4408.       int n;
  4409.       // don't use symbol_fontno, because that might return a style
  4410.       // and because we don't want to translate the name
  4411.       for (n = 0; n < font_table_size; n++)
  4412.         if (font_table[n] != 0 && font_table[n]->is_named(f)
  4413.         && !font_table[n]->is_style())
  4414.           break;
  4415.       if (n >= font_table_size) {
  4416.         n = next_available_font_position();
  4417.         if (!mount_font_no_translate(n, f, f))
  4418.           return -1;
  4419.       }
  4420.       return map[i] = n;
  4421.     }
  4422.     else
  4423.       return map[i] = i;
  4424.       }
  4425.       else
  4426.     return -1;
  4427.     }
  4428.   }
  4429.   else
  4430.     return -1;
  4431. }
  4432.  
  4433. dictionary family_dictionary(5);
  4434.  
  4435. font_family *lookup_family(symbol nm)
  4436. {
  4437.   font_family *f = (font_family *)family_dictionary.lookup(nm);
  4438.   if (!f) {
  4439.     f = new font_family(nm);
  4440.     (void)family_dictionary.lookup(nm, f);
  4441.   }
  4442.   return f;
  4443.  
  4444. void font_family::invalidate_fontno(int n)
  4445. {
  4446.   assert(n >= 0 && n < font_table_size);
  4447.   dictionary_iterator iter(family_dictionary);
  4448.   symbol nm;
  4449.   font_family *fam;
  4450.   while (iter.get(&nm, (void **)&fam)) {
  4451.     int map_size = fam->map_size;
  4452.     if (n < map_size)
  4453.       fam->map[n] = -1;
  4454.     for (int i = 0; i < map_size; i++)
  4455.       if (fam->map[i] == n)
  4456.     fam->map[i] = -1;
  4457.   }
  4458. }
  4459.  
  4460. void style()
  4461. {
  4462.   int n;
  4463.   if (get_integer(&n)) {
  4464.     if (n < 0)
  4465.       error("negative font position");
  4466.     else {
  4467.       symbol internal_name = get_name(1);
  4468.       if (!internal_name.is_null())
  4469.     mount_style(n, internal_name);
  4470.     }
  4471.   }
  4472.   skip_line();
  4473. }
  4474.  
  4475. static int get_fontno()
  4476. {
  4477.   int n;
  4478.   tok.skip();
  4479.   if (tok.delimiter()) {
  4480.     symbol s = get_name(1);
  4481.     if (!s.is_null()) {
  4482.       n = symbol_fontno(s);
  4483.       if (n < 0) {
  4484.     n = next_available_font_position();
  4485.     if (!mount_font(n, s))
  4486.       return -1;
  4487.       }
  4488.       return curenv->get_family()->make_definite(n);
  4489.     }
  4490.   }
  4491.   else if (get_integer(&n)) {
  4492.     if (n < 0 || n >= font_table_size || font_table[n] == 0)
  4493.       error("bad font number");
  4494.     else
  4495.       return curenv->get_family()->make_definite(n);
  4496.   }
  4497.   return -1;
  4498. }
  4499.  
  4500. static int underline_fontno = 2;
  4501.  
  4502. void underline_font()
  4503. {
  4504.   int n = get_fontno();
  4505.   if (n >= 0)
  4506.     underline_fontno = n;
  4507.   skip_line();
  4508. }
  4509.  
  4510. int get_underline_fontno()
  4511. {
  4512.   return underline_fontno;
  4513. }
  4514.     
  4515. static void read_special_fonts(special_font_list **sp)
  4516. {
  4517.   special_font_list *s = *sp;
  4518.   *sp = 0;
  4519.   while (s != 0) {
  4520.     special_font_list *tem = s;
  4521.     s = s->next;
  4522.     delete tem;
  4523.   }
  4524.   special_font_list **p = sp;
  4525.   while (has_arg()) {
  4526.     int i = get_fontno();
  4527.     if (i >= 0) {
  4528.       special_font_list *tem = new special_font_list;
  4529.       tem->n = i;
  4530.       tem->next = 0;
  4531.       *p = tem;
  4532.       p = &(tem->next);
  4533.     }
  4534.   }
  4535. }
  4536.  
  4537. void font_special_request()
  4538. {
  4539.   int n = get_fontno();
  4540.   if (n >= 0)
  4541.     read_special_fonts(&font_table[n]->sf); 
  4542.   skip_line();
  4543. }
  4544.  
  4545.   
  4546. void special_request()
  4547. {
  4548.   read_special_fonts(&global_special_fonts);
  4549.   skip_line();
  4550. }
  4551.  
  4552. int next_available_font_position()
  4553. {
  4554.   for (int i = 1; i < font_table_size && font_table[i] != 0; i++)
  4555.     ;
  4556.   return i;
  4557. }
  4558.  
  4559. int symbol_fontno(symbol s)
  4560. {
  4561.   s = get_font_translation(s);
  4562.   for (int i = 0; i < font_table_size; i++)
  4563.     if (font_table[i] != 0 && font_table[i]->is_named(s))
  4564.       return i;
  4565.   return -1;
  4566. }
  4567.  
  4568. int is_good_fontno(int n)
  4569. {
  4570.   return n >= 0 && n < font_table_size && font_table[n] != NULL;
  4571. }
  4572.  
  4573. int get_bold_fontno(int n)
  4574. {
  4575.   if (n >= 0 && n < font_table_size && font_table[n] != 0) {
  4576.     hunits offset;
  4577.     if (font_table[n]->get_bold(&offset))
  4578.       return offset.to_units() + 1;
  4579.     else
  4580.       return 0;
  4581.   }
  4582.   else
  4583.     return 0;
  4584. }
  4585.  
  4586. hunits env_digit_width(environment *env)
  4587. {
  4588.   node *n = make_glyph_node(charset_table['0'], env);
  4589.   if (n) {
  4590.     hunits x = n->width();
  4591.     delete n;
  4592.     return x;
  4593.   }
  4594.   else
  4595.     return H0;
  4596. }
  4597.  
  4598. hunits env_space_width(environment *env)
  4599. {
  4600.   int fn = env_definite_font(env);
  4601.   font_size fs = env->get_font_size();
  4602.   if (fn < 0 || fn >= font_table_size || font_table[fn] == 0)
  4603.     return scale(fs.to_units()/3, env->get_space_size(), 12);
  4604.   else
  4605.     return font_table[fn]->get_space_width(fs, env->get_space_size());
  4606. }
  4607.  
  4608. hunits env_sentence_space_width(environment *env)
  4609. {
  4610.   int fn = env_definite_font(env);
  4611.   font_size fs = env->get_font_size();
  4612.   if (fn < 0 || fn >= font_table_size || font_table[fn] == 0)
  4613.     return scale(fs.to_units()/3, env->get_sentence_space_size(), 12);
  4614.   else
  4615.     return font_table[fn]->get_space_width(fs, env->get_sentence_space_size());
  4616. }
  4617.  
  4618. hunits env_half_narrow_space_width(environment *env)
  4619. {
  4620.   int fn = env_definite_font(env);
  4621.   font_size fs = env->get_font_size();
  4622.   if (fn < 0 || fn >= font_table_size || font_table[fn] == 0)
  4623.     return 0;
  4624.   else
  4625.     return font_table[fn]->get_half_narrow_space_width(fs);
  4626. }
  4627.  
  4628. hunits env_narrow_space_width(environment *env)
  4629. {
  4630.   int fn = env_definite_font(env);
  4631.   font_size fs = env->get_font_size();
  4632.   if (fn < 0 || fn >= font_table_size || font_table[fn] == 0)
  4633.     return 0;
  4634.   else
  4635.     return font_table[fn]->get_narrow_space_width(fs);
  4636. }
  4637.  
  4638. void bold_font()
  4639. {
  4640.   int n = get_fontno();
  4641.   if (n >= 0) {
  4642.     if (has_arg()) {
  4643.       if (tok.delimiter()) {
  4644.     int f = get_fontno();
  4645.     if (f >= 0) {
  4646.       units offset;
  4647.       if (has_arg() && get_number(&offset, 'u') && offset >= 1)
  4648.         font_table[f]->set_conditional_bold(n, hunits(offset - 1));
  4649.       else
  4650.         font_table[f]->conditional_unbold(n);
  4651.     }
  4652.       }
  4653.       else {
  4654.     units offset;
  4655.     if (get_number(&offset, 'u') && offset >= 1)
  4656.       font_table[n]->set_bold(hunits(offset - 1));
  4657.     else
  4658.       font_table[n]->unbold();
  4659.       }
  4660.     }
  4661.     else
  4662.       font_table[n]->unbold();
  4663.   }
  4664.   skip_line();
  4665. }
  4666.  
  4667. track_kerning_function::track_kerning_function() : non_zero(0)
  4668. {
  4669. }
  4670.  
  4671. track_kerning_function::track_kerning_function(int min_s, hunits min_a, 
  4672.                            int max_s, hunits max_a)
  4673.      : non_zero(1), 
  4674.      min_size(min_s), min_amount(min_a),
  4675.      max_size(max_s), max_amount(max_a)
  4676. {
  4677. }
  4678.  
  4679. int track_kerning_function::operator==(const track_kerning_function &tk)
  4680. {
  4681.   if (non_zero)
  4682.     return (tk.non_zero
  4683.         && min_size == tk.min_size
  4684.         && min_amount == tk.min_amount
  4685.         && max_size == tk.max_size
  4686.         && max_amount == tk.max_amount);
  4687.   else
  4688.     return !tk.non_zero;
  4689. }
  4690.  
  4691. int track_kerning_function::operator!=(const track_kerning_function &tk)
  4692. {
  4693.   if (non_zero)
  4694.     return (!tk.non_zero
  4695.         || min_size != tk.min_size
  4696.         || min_amount != tk.min_amount
  4697.         || max_size != tk.max_size
  4698.         || max_amount != tk.max_amount);
  4699.   else
  4700.     return tk.non_zero;
  4701. }
  4702.  
  4703. hunits track_kerning_function::compute(int size)
  4704. {
  4705.   if (non_zero) {
  4706.     if (max_size <= min_size)
  4707.       return min_amount;
  4708.     else if (size <= min_size)
  4709.       return min_amount;
  4710.     else if (size >= max_size)
  4711.       return max_amount;
  4712.     else
  4713.       return (scale(max_amount, size - min_size, max_size - min_size)
  4714.           + scale(min_amount, max_size - size, max_size - min_size));
  4715.   }
  4716.   else
  4717.     return H0;
  4718. }
  4719.  
  4720. void track_kern()
  4721. {
  4722.   int n = get_fontno();
  4723.   if (n >= 0) {
  4724.     int min_s, max_s;
  4725.     hunits min_a, max_a;
  4726.     if (has_arg()
  4727.     && get_number(&min_s, 'z')
  4728.     && get_hunits(&min_a, 'p')
  4729.     && get_number(&max_s, 'z')
  4730.     && get_hunits(&max_a, 'p')) {
  4731.       track_kerning_function tk(min_s, min_a, max_s, max_a);
  4732.       font_table[n]->set_track_kern(tk);
  4733.     }
  4734.     else {
  4735.       track_kerning_function tk;
  4736.       font_table[n]->set_track_kern(tk);
  4737.     }
  4738.   }
  4739.   skip_line();
  4740. }
  4741.  
  4742. void constant_space()
  4743. {
  4744.   int n = get_fontno();
  4745.   if (n >= 0) {
  4746.     int x, y;
  4747.     if (!has_arg() || !get_integer(&x))
  4748.       font_table[n]->set_constant_space(CONSTANT_SPACE_NONE);
  4749.     else {
  4750.       if (!has_arg() || !get_number(&y, 'z'))
  4751.     font_table[n]->set_constant_space(CONSTANT_SPACE_RELATIVE, x);
  4752.       else
  4753.     font_table[n]->set_constant_space(CONSTANT_SPACE_ABSOLUTE, 
  4754.                       scale(y*x,
  4755.                         units_per_inch,
  4756.                         36*72*sizescale));
  4757.     }
  4758.   }
  4759.   skip_line();
  4760. }
  4761.  
  4762. void ligature()
  4763. {
  4764.   int lig;
  4765.   if (has_arg() && get_integer(&lig) && lig >= 0 && lig <= 2)
  4766.     global_ligature_mode = lig;
  4767.   else
  4768.     global_ligature_mode = 1;
  4769.   skip_line();
  4770. }
  4771.  
  4772. void kern_request()
  4773. {
  4774.   int k;
  4775.   if (has_arg() && get_integer(&k))
  4776.     global_kern_mode = k != 0;
  4777.   else
  4778.     global_kern_mode = 1;
  4779.   skip_line();
  4780. }
  4781.  
  4782. void set_soft_hyphen_char()
  4783. {
  4784.   soft_hyphen_char = get_optional_char();
  4785.   if (!soft_hyphen_char)
  4786.     soft_hyphen_char = get_charinfo(HYPHEN_SYMBOL);
  4787.   skip_line();
  4788. }
  4789.  
  4790. void init_output()
  4791. {
  4792.   if (suppress_output_flag)
  4793.     the_output = new suppress_output_file;
  4794.   else if (ascii_output_flag)
  4795.     the_output = new ascii_output_file;
  4796.   else
  4797.     the_output = new troff_output_file;
  4798. }
  4799.  
  4800. class next_available_font_position_reg : public reg {
  4801. public:
  4802.   const char *get_string();
  4803. };
  4804.  
  4805. const char *next_available_font_position_reg::get_string()
  4806. {
  4807.   return itoa(next_available_font_position());
  4808. }
  4809.  
  4810. class printing_reg : public reg {
  4811. public:
  4812.   const char *get_string();
  4813. };
  4814.  
  4815. const char *printing_reg::get_string()
  4816. {
  4817.   if (the_output)
  4818.     return the_output->is_printing() ? "1" : "0";
  4819.   else
  4820.     return "0";
  4821. }
  4822.  
  4823. void init_node_requests()
  4824. {
  4825.   init_request("fp", font_position);
  4826.   init_request("sty", style);
  4827.   init_request("cs", constant_space);
  4828.   init_request("bd", bold_font);
  4829.   init_request("uf", underline_font);
  4830.   init_request("lg", ligature);
  4831.   init_request("kern", kern_request);
  4832.   init_request("tkf", track_kern);
  4833.   init_request("special", special_request);
  4834.   init_request("fspecial", font_special_request);
  4835.   init_request("ftr", font_translate);
  4836.   init_request("shc", set_soft_hyphen_char);
  4837.   number_reg_dictionary.define(".fp", new next_available_font_position_reg);
  4838.   number_reg_dictionary.define(".kern",
  4839.                    new constant_int_reg(&global_kern_mode));
  4840.   number_reg_dictionary.define(".lg",
  4841.                    new constant_int_reg(&global_ligature_mode));
  4842.   number_reg_dictionary.define(".P", new printing_reg);
  4843.   soft_hyphen_char = get_charinfo(HYPHEN_SYMBOL);
  4844. }
  4845.