2000-12-05 18:49:44 +00:00

475 lines
11 KiB
C++

// -*- C++ -*-
/* Copyright (C) 1989-2000 Free Software Foundation, Inc.
Written by James Clark (jjc@jclark.com)
This file is part of groff.
groff is free software; you can redistribute it and/or modify it under
the terms of the GNU General Public License as published by the Free
Software Foundation; either version 2, or (at your option) any later
version.
groff is distributed in the hope that it will be useful, but WITHOUT ANY
WARRANTY; without even the implied warranty of MERCHANTABILITY or
FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
for more details.
You should have received a copy of the GNU General Public License along
with groff; see the file COPYING. If not, write to the Free Software
Foundation, 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
#include "driver.h"
#include "device.h"
#ifndef SHRT_MIN
#define SHRT_MIN (-32768)
#endif
#ifndef SHRT_MAX
#define SHRT_MAX 32767
#endif
#define TAB_WIDTH 8
static int horizontal_tab_flag = 0;
static int form_feed_flag = 0;
static int bold_flag = 1;
static int underline_flag = 1;
static int overstrike_flag = 1;
static int draw_flag = 1;
enum {
UNDERLINE_MODE = 01,
BOLD_MODE = 02,
VDRAW_MODE = 04,
HDRAW_MODE = 010
};
// Mode to use for bold-underlining.
static unsigned char bold_underline_mode = BOLD_MODE|UNDERLINE_MODE;
class tty_font : public font {
tty_font(const char *);
unsigned char mode;
public:
~tty_font();
unsigned char get_mode() { return mode; }
#if 0
void handle_x_command(int argc, const char **argv);
#endif
static tty_font *load_tty_font(const char *);
};
tty_font *tty_font::load_tty_font(const char *s)
{
tty_font *f = new tty_font(s);
if (!f->load()) {
delete f;
return 0;
}
const char *num = f->get_internal_name();
long n;
if (num != 0 && (n = strtol(num, 0, 0)) != 0)
f->mode = int(n & (BOLD_MODE|UNDERLINE_MODE));
if (!underline_flag)
f->mode &= ~UNDERLINE_MODE;
if (!bold_flag)
f->mode &= ~BOLD_MODE;
if ((f->mode & (BOLD_MODE|UNDERLINE_MODE)) == (BOLD_MODE|UNDERLINE_MODE))
f->mode = (f->mode & ~(BOLD_MODE|UNDERLINE_MODE)) | bold_underline_mode;
return f;
}
tty_font::tty_font(const char *nm)
: font(nm), mode(0)
{
}
tty_font::~tty_font()
{
}
#if 0
void tty_font::handle_x_command(int argc, const char **argv)
{
if (argc >= 1 && strcmp(argv[0], "bold") == 0)
mode |= BOLD_MODE;
else if (argc >= 1 && strcmp(argv[0], "underline") == 0)
mode |= UNDERLINE_MODE;
}
#endif
class glyph {
static glyph *free_list;
public:
glyph *next;
short hpos;
unsigned int code;
unsigned char mode;
void *operator new(size_t);
void operator delete(void *);
inline int draw_mode() { return mode & (VDRAW_MODE|HDRAW_MODE); }
};
glyph *glyph::free_list = 0;
void *glyph::operator new(size_t)
{
if (!free_list) {
const int BLOCK = 1024;
free_list = (glyph *)new char[sizeof(glyph)*BLOCK];
for (int i = 0; i < BLOCK - 1; i++)
free_list[i].next = free_list + i + 1;
free_list[BLOCK - 1].next = 0;
}
glyph *p = free_list;
free_list = free_list->next;
p->next = 0;
return p;
}
void glyph::operator delete(void *p)
{
if (p) {
((glyph *)p)->next = free_list;
free_list = (glyph *)p;
}
}
class tty_printer : public printer {
int is_utf8;
glyph **lines;
int nlines;
int cached_v;
int cached_vpos;
void add_char(unsigned int, int, int, unsigned char);
public:
tty_printer(const char *device);
~tty_printer();
void set_char(int, font *, const environment *, int, const char *name);
void draw(int code, int *p, int np, const environment *env);
void put_char(unsigned int);
void begin_page(int) { }
void end_page(int page_length);
font *make_font(const char *);
};
tty_printer::tty_printer(const char *device) : cached_v(0)
{
is_utf8 = !strcmp(device,"utf8");
nlines = 66;
lines = new glyph *[nlines];
for (int i = 0; i < nlines; i++)
lines[i] = 0;
}
tty_printer::~tty_printer()
{
a_delete lines;
}
void tty_printer::set_char(int i, font *f, const environment *env, int w, const char *name)
{
if (w != font::hor)
fatal("width of character not equal to horizontal resolution");
add_char(f->get_code(i), env->hpos, env->vpos, ((tty_font *)f)->get_mode());
}
void tty_printer::add_char(unsigned int c, int h, int v, unsigned char mode)
{
#if 0
// This is too expensive.
if (h % font::hor != 0)
fatal("horizontal position not a multiple of horizontal resolution");
#endif
int hpos = h / font::hor;
if (hpos < SHRT_MIN || hpos > SHRT_MAX) {
error("character with ridiculous horizontal position discarded");
return;
}
int vpos;
if (v == cached_v && cached_v != 0)
vpos = cached_vpos;
else {
if (v % font::vert != 0)
fatal("vertical position not a multiple of vertical resolution");
vpos = v / font::vert;
if (vpos > nlines) {
glyph **old_lines = lines;
lines = new glyph *[vpos + 1];
memcpy(lines, old_lines, nlines*sizeof(glyph *));
for (int i = nlines; i <= vpos; i++)
lines[i] = 0;
a_delete old_lines;
nlines = vpos + 1;
}
// Note that the first output line corresponds to groff
// position font::vert.
if (vpos <= 0) {
error("character above first line discarded");
return;
}
cached_v = v;
cached_vpos = vpos;
}
glyph *g = new glyph;
g->hpos = hpos;
g->code = c;
g->mode = mode;
// The list will be reversed later. After reversal, it must be in
// increasing order of hpos, with HDRAW characters before VDRAW
// characters before normal characters at each hpos, and otherwise
// in order of occurrence.
glyph **pp;
for (pp = lines + (vpos - 1); *pp; pp = &(*pp)->next)
if ((*pp)->hpos < hpos
|| ((*pp)->hpos == hpos && (*pp)->draw_mode() >= g->draw_mode()))
break;
g->next = *pp;
*pp = g;
}
void tty_printer::draw(int code, int *p, int np, const environment *env)
{
if (code != 'l' || !draw_flag)
return;
if (np != 2) {
error("2 arguments required for line");
return;
}
if (p[0] == 0) {
// vertical line
int v = env->vpos;
int len = p[1];
if (len < 0) {
v += len;
len = -len;
}
while (len >= 0) {
add_char('|', env->hpos, v, VDRAW_MODE);
len -= font::vert;
v += font::vert;
}
}
if (p[1] == 0) {
// horizontal line
int h = env->hpos;
int len = p[0];
if (len < 0) {
h += len;
len = -len;
}
while (len >= 0) {
add_char('-', h, env->vpos, HDRAW_MODE);
len -= font::hor;
h += font::hor;
}
}
}
void tty_printer::put_char (unsigned int wc)
{
if (is_utf8 && wc >= 0x80) {
char buf[6+1];
int count;
char *p = buf;
if (wc < 0x800)
count = 1, *p = (unsigned char) ((wc >> 6) | 0xC0);
else if (wc < 0x10000)
count = 2, *p = (unsigned char) ((wc >> 12) | 0xE0);
else if (wc < 0x200000)
count = 3, *p = (unsigned char) ((wc >> 18) | 0xF0);
else if (wc < 0x4000000)
count = 4, *p = (unsigned char) ((wc >> 24) | 0xF8);
else if (wc <= 0x7FFFFFFF)
count = 5, *p = (unsigned char) ((wc >> 30) | 0xFC);
else
return;
do *++p = (unsigned char)(((wc >> (6 * --count)) & 0x3F) | 0x80);
while (count > 0);
*++p = '\0';
fputs(buf,stdout);
} else {
putchar(wc);
}
}
void tty_printer::end_page(int page_length)
{
if (page_length % font::vert != 0)
error("vertical position at end of page not multiple of vertical resolution");
int lines_per_page = page_length / font::vert;
int last_line;
for (last_line = nlines; last_line > 0; last_line--)
if (lines[last_line - 1])
break;
#if 0
if (last_line > lines_per_page) {
error("characters past last line discarded");
do {
--last_line;
while (lines[last_line]) {
glyph *tem = lines[last_line];
lines[last_line] = tem->next;
delete tem;
}
} while (last_line > lines_per_page);
}
#endif
for (int i = 0; i < last_line; i++) {
glyph *p = lines[i];
lines[i] = 0;
glyph *g = 0;
while (p) {
glyph *tem = p->next;
p->next = g;
g = p;
p = tem;
}
int hpos = 0;
glyph *nextp;
for (p = g; p; delete p, p = nextp) {
nextp = p->next;
if (nextp && p->hpos == nextp->hpos) {
if (p->draw_mode() == HDRAW_MODE && nextp->draw_mode() == VDRAW_MODE) {
nextp->code = '+';
continue;
}
if (p->draw_mode() != 0 && p->draw_mode() == nextp->draw_mode()) {
nextp->code = p->code;
continue;
}
if (!overstrike_flag)
continue;
}
if (hpos > p->hpos) {
do {
putchar('\b');
hpos--;
} while (hpos > p->hpos);
}
else {
if (horizontal_tab_flag) {
for (;;) {
int next_tab_pos = ((hpos + TAB_WIDTH) / TAB_WIDTH) * TAB_WIDTH;
if (next_tab_pos > p->hpos)
break;
putchar('\t');
hpos = next_tab_pos;
}
}
for (; hpos < p->hpos; hpos++)
putchar(' ');
}
assert(hpos == p->hpos);
if (p->mode & UNDERLINE_MODE) {
putchar('_');
putchar('\b');
}
if (p->mode & BOLD_MODE) {
put_char(p->code);
putchar('\b');
}
put_char(p->code);
hpos++;
}
putchar('\n');
}
if (form_feed_flag) {
if (last_line < lines_per_page)
putchar('\f');
}
else {
for (; last_line < lines_per_page; last_line++)
putchar('\n');
}
}
font *tty_printer::make_font(const char *nm)
{
return tty_font::load_tty_font(nm);
}
printer *make_printer()
{
return new tty_printer(device);
}
static void usage();
int main(int argc, char **argv)
{
program_name = argv[0];
static char stderr_buf[BUFSIZ];
setbuf(stderr, stderr_buf);
int c;
while ((c = getopt(argc, argv, "F:vhfbuoBUd")) != EOF)
switch(c) {
case 'v':
{
extern const char *Version_string;
fprintf(stderr, "grotty version %s\n", Version_string);
fflush(stderr);
break;
}
case 'b':
// Do not embolden by overstriking.
bold_flag = 0;
break;
case 'u':
// Do not underline.
underline_flag = 0;
break;
case 'o':
// Do not overstrike (other than emboldening and underlining).
overstrike_flag = 0;
break;
case 'B':
// Do bold-underlining as bold.
bold_underline_mode = BOLD_MODE;
break;
case 'U':
// Do bold-underlining as underlining.
bold_underline_mode = UNDERLINE_MODE;
break;
case 'h':
// Use horizontal tabs.
horizontal_tab_flag = 1;
break;
case 'f':
form_feed_flag = 1;
break;
case 'F':
font::command_line_font_dir(optarg);
break;
case 'd':
// Ignore \D commands.
draw_flag = 0;
break;
case '?':
usage();
break;
default:
assert(0);
}
if (optind >= argc)
do_file("-");
else {
for (int i = optind; i < argc; i++)
do_file(argv[i]);
}
delete pr;
return 0;
}
static void usage()
{
fprintf(stderr, "usage: %s [-hfvbuodBU] [-F dir] [files ...]\n",
program_name);
exit(1);
}