/* HTML-PSformat.c - Module for NCSA's Mosaic software * * Purpose: to parse Hypertext widget contents into appropriate PostScript * * Author: Ameet A. Raval & Frans van Hoesel & Andrew Ford * (aar@gfdl.gov & hoesel@chem.rug.nl). * send bugreports to hoesel@chem.rug.nl * * Institution: for Ameet A. Raval: * Geophysical Fluid Dynamics Laboratory, * National Oceanic and Atmospheric Administration, * U.S. Department of Commerce * P.O. Box 308 * Princeton, NJ 08542 * for Frans van Hoesel: * Xtreme graphics software * Herepoortenmolendrift 36 * 9711 DH Groningen * The Netherlands * also: * Andrew Ford * Independent Software Consultant * 30 Upper Cheltenham Place, * Montpelier, Bristol, BS6 5HR, GB * E-mail: andrew@icarus.demon.co.uk * * Date: 1 aug 1993 * Modification: 8 nov 1993 * o added support for bold/italics courier * o removed unused or no longer needed stuff * o fixed the font alignment problem * 23 nov 1993 * o added support for horizontal ruler * o on request of Ameet, added a specific * line about whome to send bugreports to * 15 jun 1994 * o add headers, footers and footnotes to convey * title, page number, url of document, date (of * printing) and urls of links. * o use A4 or US Letter size paper (currently hard coded) * * 9 may 1995 (Andrew Ford) * o general overhaul * * Copyright: This work is the product of the United States Government, * and is precluded from copyright protection. It is hereby * released into the public domain. * * WE MAKE NO REPRESENTATIONS ABOUT THE SUITABILITY OF THIS SOFTWARE FOR * ANY PURPOSE. IT IS PROVIDED "AS IS" WITHOUT EXPRESS OR IMPLIED * WARRANTY. WE SHALL NOT BE LIABLE FOR ANY DAMAGES SUFFERED BY THE * USERS OF THIS SOFTWARE. * * pieces of code are taken from xvps by kind * permission of John Bradley. * * To-do: * * printing of tables and forms fields (and any new features that appear) * */ #include "../config.h" #include #include #include #include #include #ifdef __bsdi__ #include #else #ifndef VAXC #include #endif #endif #include #include #include "HTMLP.h" /* Fix thanks to robm. */ #ifdef __alpha #include #endif #define CR '\015' #define LF '\012' #ifndef DISABLE_TRACE extern int htmlwTrace; #endif extern int SwapElements(); #define USLETTER 0 #define A4 1 #if !defined(DEFAULT_PAGE_SIZE) #define DEFAULT_PAGE_SIZE USLETTER #endif #define F_FULLCOLOR 0 #define F_GREYSCALE 1 #define F_BWDITHER 2 #define F_REDUCED 3 #define L_PAREN '(' #define R_PAREN ')' #define B_SLASH '\\' #define MAX_ASCII '\177' /* MONO returns total intensity of r,g,b components .33R+ .5G+ .17B */ #define MONO(rd,gn,bl) (((rd)*11 + (gn)*16 + (bl)*5) >> 13) /* PSconst_out outputs to the postscript buffer an array of constant * strings */ #define PSconst_out(txt) { \ int n=(sizeof txt)/(sizeof txt[0]); \ int i; \ for (i=0; i10000.0) dpi = 72.0; return dpi; } /* * PSprintf - dynamic string concatenation function. * * In successive calls, the formatted string will be appended to the global * output string Sp. * It assumes that on each call, the length of the text appended to Sp * is less than 1024. * The format string is used as in printf, so you can use additional * arguments. * * When successful, PSprintf returns the number of characters printed * in this call, otherwise it returns EOF (just as printf does) * */ static int PSprintf(char *format, ...) { int len; char *s; va_list args; if (PS_size - PS_len < 1024) { PS_size += 1024; if ((s = (char *) realloc(PS_string, PS_size)) == NULL) { #ifndef DISABLE_TRACE if (htmlwTrace) { fprintf(stderr, "PSprintf malloc failed\n"); } #endif return(EOF); } PS_string = s; } va_start(args, format); len = vsprintf(PS_string+PS_len, format, args); /* this is a hack to make it work on systems were vsprintf(s,...) * returns s, instead of the len. */ if (len != EOF && len != 0) PS_len += strlen(PS_string+PS_len); va_end(args); return(len); } /* * PShex - output hex byte * * Append the byte "val" to an internal string buffer in hexadecimal * format. If the argument "flush" is True, or if the buffer has filled * up, flush the buffer to the larger postscript output buffer (using * PSprintf). * */ static int PShex(unsigned char val, int flush) { static unsigned char hexline[80]; static char digit[] = "0123456789abcdef"; if (!flush) { hexline[PS_hexi++] = (char) digit[((unsigned) val >> (unsigned) 4) & (unsigned) 0x0f]; hexline[PS_hexi++] = (char) digit[(unsigned) val & (unsigned) 0x0f]; } /* Changed from ">78" to ">77" on advice of debra@info.win.tue.nl (Paul De Bra). */ if ((flush && PS_hexi) || (PS_hexi>77)) { hexline[PS_hexi] = '\0'; PS_hexi=0; return (PSprintf("%s\n", hexline)); } return (0); } /* * PSfont - change font * * change local font in buf to "font" * fontfamily indicates if the overall style is times, helvetica, century * schoolbook or lucida. * */ static void PSfont(HTMLWidget hw, XFontStruct *font, int fontfamily) { PS_fontstyle fn; int style, size; int fs; static PS_fontstyle fontstyle[17] = { RF, IF, BF, FF, BF, BF, BF, BF, BF, BF, IF, FF, FF, FB, FI, FB, FI }; static char fnchar[6][3] = {"RF", "BF", "IF", "FF", "FB", "FI"}; /* fontsizes as set in gui.c and in HTML.c (listing font is only * defined in HTML.c) */ static int fontsizes[4][3][17] = { /* times font sizes */ { {14, 14, 14, 14, 18, 17, 14, 12, 10, 8, 14, 12, 12, 14, 14, 12, 12}, {17, 17, 17, 17, 24, 18, 17, 14, 12, 10, 17, 14, 12, 17, 17, 14, 14}, {20, 20, 20, 20, 25, 24, 20, 18, 17, 14, 20, 18, 12, 20, 20, 18, 18} }, /* helvetica sizes */ { {14, 14, 14, 14, 18, 17, 14, 12, 10, 8, 14, 12, 12, 14, 14, 12, 12}, {17, 17, 17, 17, 24, 18, 17, 14, 12, 10, 17, 14, 12, 17, 17, 14, 14}, {20, 20, 20, 20, 25, 24, 20, 18, 17, 14, 20, 18, 12, 20, 20, 18, 18} }, /* new century schoolbook sizes */ { {14, 14, 14, 14, 18, 17, 14, 12, 10, 8, 14, 12, 12, 14, 14, 12, 12}, {18, 18, 18, 18, 24, 18, 17, 14, 12, 10, 18, 14, 12, 18, 18, 14, 14}, {20, 20, 20, 20, 25, 24, 20, 18, 17, 14, 20, 18, 12, 20, 20, 18, 18} }, /* lucida sizes */ { {14, 14, 14, 14, 18, 17, 14, 12, 11, 10, 14, 12, 12, 14, 14, 12, 12}, {17, 17, 17, 17, 24, 18, 17, 14, 12, 10, 17, 14, 12, 17, 17, 14, 14}, {20, 20, 20, 20, 25, 24, 20, 18, 17, 14, 20, 18, 12, 20, 20, 18, 18} } }; /* next is for each fontfamily the ascent value as given by the * medium sized bold x-font (the regular font has the same * ascent value for both the medium and the large size Century * font). * it is use in the check for the fontsize (small, medium, large) */ static int medium_fontascent[4] = { 14, 14, 16, 15 }; /* for each fontfamily, for each fontsize, and for each font style * give the ascent value, so the output from Postscript is correct. * If the value is given between parenthesis, then it is different * from the value as stored in the x-font. * Note that this is a fix, and need to be changed, if the browser * is fixed (in the current version 1.2 the baseline of various fonts * is not aligned very well). */ static int fontascent[4][3][17] = { /*rg, itl, bld, fix, h1, h2, h3, h4, h5, h6, add, pla, lis, fixbold, fixital, plabold, plaital, */ /* times */ { {12, 11, 12, 10, 15, 14, 12, 10, 8, 6, 11, 9, 10, 10, 10, 9, 9}, {13, 13, 14, 12, 20, 15, 14, 12, 10, 8, 13, 10, 10, 12, 12, 10, 10}, {16, 15, 15, 13, 21, 20, 15, 15, 14, 12, 15, 13, 10, 13, 13, 13, 13} }, /* helvetica */ { {12, 12, 12, 10, 15, 14, 12, 10, 9, 7, 12, 9, 10, 10, 10, 9, 9}, {14, 14, 14, 12, 22, 15, 14, 12, 10, 9, 14, 10, 10, 12, 12, 10, 10}, {16, 16, 16, 13, 22, 22, 16, 15, 14, 12, 16, 13, 10, 13, 13, 13, 13} }, /* new century schoolbook */ { {12, 12, 13, 10, 16, 14, 13, 10, 9, 7, 12, 9, 10, 10, 10, 9, 9}, {16, 14, 16, 13, 22, 16, 14, 13, 10, 9, 14, 10, 10, 13, 13, 10, 10}, {17, 16, 17, 13, 22, 22, 17, 16, 14, 13, 16, 13, 10, 13, 13, 13, 13} }, /* lucida bright */ { {11, 11, 11, 11, 15, 14, 11, 10, 9, 7, 11, 9, 10, 11, 10, 9, 9}, {14, 15, 14, 13, 20, 15, 14, 11, 10, 7, 15, 11, 10, 13, 13, 11, 10}, {17, 17, 17, 16, 21, 20, 17, 15, 14, 11, 17, 14, 10, 16, 13, 14, 13} } }; /* NULL case - reflush old font or the builtin default: */ if ((hw == NULL) || (font == NULL)) { if (PS_oldfs != 0) PSprintf( "%2s %d SF\n", fnchar[PS_oldfn], PS_oldfs); return; } /* added the next line in case xmosaic version 199.4 has more fonts */ style = 3; if (font == hw->html.font) { style = 0; } else if (font == hw->html.italic_font) { style = 1; } else if (font == hw->html.bold_font) { style = 2; } else if (font == hw->html.fixed_font) { style = 3; } else if (font == hw->html.header1_font) { style = 4; } else if (font == hw->html.header2_font) { style = 5; } else if (font == hw->html.header3_font) { style = 6; } else if (font == hw->html.header4_font) { style = 7; } else if (font == hw->html.header5_font) { style = 8; } else if (font == hw->html.header6_font) { style = 9; } else if (font == hw->html.address_font) { style = 10; } else if (font == hw->html.plain_font) { style = 11; } else if (font == hw->html.listing_font) { style = 12; } else if (font == hw->html.fixedbold_font) { style = 13; } else if (font == hw->html.fixeditalic_font) { style = 14; } else if (font == hw->html.plainbold_font) { style = 15; } else if (font == hw->html.plainitalic_font) { style = 16; } /* check size, by looking at the size of the regular font */ size = 1; if (hw->html.bold_font->ascent > medium_fontascent[fontfamily]) { /* large font */ size = 2; } else if (hw->html.bold_font->ascent < medium_fontascent[fontfamily]) { /* small font */ size = 0; } fn = fontstyle[style]; fs = fontsizes[fontfamily][size][style]; PS_fontascent = fontascent[fontfamily][size][style]; if (fn != PS_oldfn || fs != PS_oldfs) { PSprintf( "%2s %d SF\n", fnchar[fn], fs); PS_oldfn=fn, PS_oldfs=fs; } } /* * PSshowpage - end of page function * * show the current page and restore any changes to the printer state. * Any accumulated footnotes are output and the outstanding footnote count * reset to zero. Footnotes are preceded by a footnote rule and each footnote * is consists of a raised mark and the footnote text (i.e. the url). The mark * is in a smaller font than the text. The ideas are filched from LaTeX. */ static void PSshowpage(void) { PSprintf("restore\n"); if (n_saved_ftns > 0) { int i; PSprintf("gsave 0.2 setlinewidth newpath %.2f %.2f M %.2f 0 RL stroke\n", page_dimens.left_margin, (page_dimens.bot_margin + (footnote_ptsize * n_saved_ftns) + 4), (page_dimens.text_width * 0.4)); for (i = 0; n_saved_ftns; n_saved_ftns--, i++) { PSprintf("newpath %.2f %.2f M RF %.2f SF (%d) S 3 -2 R RF %d SF (%s) S\n", page_dimens.left_margin, page_dimens.bot_margin + 5 + (n_saved_ftns - 1) * footnote_ptsize, (0.7 * footnote_ptsize), cur_ftn_no - n_saved_ftns, footnote_ptsize, footnotes[i]); } PSprintf("grestore\n"); } PSprintf("showpage\n"); } /* * PSnewpage - begin a fresh page * * increment the page count and handle the structured comment * conventions * */ static void PSnewpage(void) { PS_curr_page++; /* the PostScript reference Manual states that the Page: Tag should have a label and a ordinal; otherwise programs like psutils fail -gustaf */ PSprintf("%%%%Page: %d %d\n", PS_curr_page, PS_curr_page); PSprintf("save\n"); if (HTML_Print_Headers) PSprintf("%d ", PS_curr_page); PSprintf("NP\n"); PSfont( NULL, NULL, 0); /* force re-flush of last font used */ } /* * PSinit_latin1 - handle ISO encoding * * print out initializing PostScript text for ISO Latin1 font encoding * This code is copied from the Idraw program (from Stanford's InterViews * package), courtesy of Steinar Kjaernsr|d, steinar@ifi.uio.no * */ static void PSinit_latin1(void) { static char *txt[] = { "/reencodeISO {", "dup dup findfont dup length dict begin", "{ 1 index /FID ne { def }{ pop pop } ifelse } forall", "/Encoding ISOLatin1Encoding D", "currentdict end definefont", "} D", "/ISOLatin1Encoding [", "/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef", "/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef", "/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef", "/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef", "/space/exclam/quotedbl/numbersign/dollar/percent/ampersand/quoteright", "/parenleft/parenright/asterisk/plus/comma/minus/period/slash", "/zero/one/two/three/four/five/six/seven/eight/nine/colon/semicolon", "/less/equal/greater/question/at/A/B/C/D/E/F/G/H/I/J/K/L/M/N", "/O/P/Q/R/S/T/U/V/W/X/Y/Z/bracketleft/backslash/bracketright", "/asciicircum/underscore/quoteleft/a/b/c/d/e/f/g/h/i/j/k/l/m", "/n/o/p/q/r/s/t/u/v/w/x/y/z/braceleft/bar/braceright/asciitilde", "/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef", "/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef", "/.notdef/dotlessi/grave/acute/circumflex/tilde/macron/breve", "/dotaccent/dieresis/.notdef/ring/cedilla/.notdef/hungarumlaut", "/ogonek/caron/space/exclamdown/cent/sterling/currency/yen/brokenbar", "/section/dieresis/copyright/ordfeminine/guillemotleft/logicalnot", "/hyphen/registered/macron/degree/plusminus/twosuperior/threesuperior", "/acute/mu/paragraph/periodcentered/cedilla/onesuperior/ordmasculine", "/guillemotright/onequarter/onehalf/threequarters/questiondown", "/Agrave/Aacute/Acircumflex/Atilde/Adieresis/Aring/AE/Ccedilla", "/Egrave/Eacute/Ecircumflex/Edieresis/Igrave/Iacute/Icircumflex", "/Idieresis/Eth/Ntilde/Ograve/Oacute/Ocircumflex/Otilde/Odieresis", "/multiply/Oslash/Ugrave/Uacute/Ucircumflex/Udieresis/Yacute", "/Thorn/germandbls/agrave/aacute/acircumflex/atilde/adieresis", "/aring/ae/ccedilla/egrave/eacute/ecircumflex/edieresis/igrave", "/iacute/icircumflex/idieresis/eth/ntilde/ograve/oacute/ocircumflex", "/otilde/odieresis/divide/oslash/ugrave/uacute/ucircumflex/udieresis", "/yacute/thorn/ydieresis", "] D", "[RF BF IF FF FB FI] {reencodeISO D} forall" }; PSconst_out(txt); } /* * PSinit - initialize Postscript output * * does the initialization per html document * */ static void PSinit(void) { PS_size = PS_len = PS_offset = PS_hexi = PS_page_offset = 0; PS_start_y = 0; PS_string = (char *) malloc(1); PS_oldfs = 0; PS_oldfn = RF; PS_curr_page = 0 ; n_saved_ftns = 0; cur_ftn_no = 1; } /* * PSheader - initialize Postscript output * * Prints out the prolog. The following PostScript macros are defined * D def - define a macro * E exch - exhange parameters * M moveto * R rmoveto * L lineto * RL rlineto * SQ draw a unit square * U underline a string * B draw a bullet * OB draw an open bullet * HR draw a horizontal rule * SF set font * RF roman font (dependent on font family) * BF bold font (dependent on font family) * IF italic font (dependent on font family) * FF fixed font (courier) * FB fixed bold font (courier bold) * FI fixed italic font (courier oblique) * nstr buffer for creating page number string * pageno literal "Page " * url URL of document * title title of document * date date modified/printed */ static void PSheader(char *title, int font, char *url, char *time_str) { static char *notitle="Untitled"; int set_to_null=0; static char *fontname[] = { /* in order: regular, bold, italic */ "Times-Roman", "Times-Bold", "Times-Italic", "Helvetica", "Helvetica-Bold", "Helvetica-Oblique", "NewCenturySchlbk-Roman", "NewCenturySchlbk-Bold", "NewCenturySchlbk-Italic", /* this is a nasty trick, I have put Times in place of * Lucida, because most printers don't have Lucida font */ "Times-Roman", "Times-Bold", "Times-Italic" /* "Lucida", "Lucida-Bold", "Lucida-Italic" */ }; static char *txt[] = { "/M {moveto} D", "/S {show} D", "/R {rmoveto} D", "/L {lineto} D", "/RL {rlineto} D", "/SQ {newpath 0 0 M 0 1 L 1 1 L 1 0 L closepath} D", "/U {gsave currentpoint currentfont /FontInfo get /UnderlinePosition get", " 0 E currentfont /FontMatrix get dtransform E pop add newpath moveto", " dup stringwidth rlineto stroke grestore S } D", "/B {/r E D gsave -13 0 R currentpoint ", " newpath r 0 360 arc closepath fill grestore } D", "/OB {/r E D gsave -13 0 R currentpoint ", " newpath r 0 360 arc closepath stroke grestore } D", "/HR {/l E D gsave 0 1 RL l 0 RL 0 -1 RL l neg 0 RL stroke grestore } D", "/SF {E findfont E scalefont setfont } D", "/FF {/Courier } D", "/FB {/Courier-Bold } D", "/FI {/Courier-Oblique } D" }; char time_buf[40]; time_t clock = time(NULL); #if !defined(VMS) || defined (__DECC) strftime(time_buf, sizeof(time_buf), "Printed %a %b %e %T %Y", localtime(&clock)); #else sprintf(time_buf,"Printed %s",asctime(localtime(&clock))); #endif /* Always show the print date -- SWP */ time_str = time_buf; /* if (!time_str || (time_str[0] == '\0')) { time_str = time_buf; } */ PSprintf("%%!PS-Adobe-1.0\n"); PSprintf("%%%%Creator: NCSA Mosaic, Postscript by Ameet Raval, Frans van Hoesel\n"); PSprintf("%%%% and Andrew Ford\n"); if (!title) { title=notitle; set_to_null=1; } { char *tmp; for (tmp = title; *tmp; tmp++) if (*tmp == CR || *tmp == LF) *tmp = ' '; PSprintf("%%%%Title: %s\n", title); } PSprintf("%%%%CreationDate: %s\n", time_buf + 8); PSprintf("%%%%Pages: (atend)\n"); PSprintf("%%%%PageOrder: Ascend\n"); PSprintf("%%%%BoundingBox: %d %d %d %d\n", (int)page_dimens.left_margin, (int)(page_dimens.bot_margin - 12), (int)(page_dimens.left_margin + page_dimens.text_width + 0.5), (int)(page_dimens.bot_margin + page_dimens.text_height + 12.5)); PSprintf("%%%%DocumentFonts: %s %s %s Courier Courier-Bold Courier-Oblique\n", fontname[font*3], fontname[font*3+1], fontname[font*3+2]); PSprintf("%%%%EndComments\n"); PSprintf("save /D {def} def /E {exch} D\n"); PSprintf("/RF {/%s} D\n", fontname[font*3]); PSprintf("/BF {/%s} D\n", fontname[font*3+1]); PSprintf("/IF {/%s} D\n", fontname[font*3+2]); PSprintf("/nstr 6 string D /pgno (Page ) D\n"); PSprintf("/url (%s) D\n", url); PSprintf("/title (%s) D\n", title); PSprintf("/date (%s) D\n", time_str); PSconst_out(txt); /* Output the newpage definition. */ PSprintf("/NP {"); if (HTML_Print_Headers) { PSprintf("gsave 0.4 setlinewidth\n"); PSprintf(" newpath %.2f %.2f M %.2f 0 RL stroke", page_dimens.left_margin, (page_dimens.bot_margin + page_dimens.text_height), page_dimens.text_width); PSprintf(" newpath %.2f %.2f M %.2f 0 RL stroke\n", page_dimens.left_margin, page_dimens.bot_margin, page_dimens.text_width); PSprintf(" BF 12 SF %.2f %.2f M (%s) S\n", page_dimens.left_margin, (page_dimens.bot_margin + page_dimens.text_height + 6), title); PSprintf(" nstr cvs dup stringwidth pop pgno stringwidth pop add\n"); PSprintf(" %.2f E sub %.2f M pgno S S\n", (page_dimens.left_margin + page_dimens.text_width), (page_dimens.bot_margin + page_dimens.text_height + 6)); PSprintf(" BF 10 SF %.2f %.2f M (%s) S\n", page_dimens.left_margin, page_dimens.bot_margin - 12, url); PSprintf(" (%s) dup stringwidth pop %.2f E sub %.2f M S grestore\n", time_str, page_dimens.left_margin + page_dimens.text_width, page_dimens.bot_margin - 12); } PSprintf(" %.2f %.2f translate %.5f %.5f scale } D\n", page_dimens.left_margin, page_dimens.bot_margin + page_dimens.text_height, Points_Pixel, Points_Pixel); PSinit_latin1(); PSprintf("%%%%EndProlog\n"); if (set_to_null) { title=NULL; } } /* * PStrailer - write postscript trailer * */ static void PStrailer(void) { PSprintf("%%%%Trailer\n"); PSprintf("restore\n"); PSprintf("%%%%Pages: %d\n", PS_curr_page); if (footnotes) { free(footnotes); footnotes = NULL; ftn_array_size = 0; } } /* * PSmoveto - move to new x,y location * * if the Y value does not fit on the current page, begin a new page * (I think in the current implementation, this never happens) * */ static void PSmoveto(int x, int y) { if (y > PS_start_y + Pixels_Page) { PS_start_y = y; PSshowpage(); PSnewpage(); } PS_offset = 0; PSprintf( "%d %d M\n", x, -(y - PS_start_y)); } /* * PSmove_offset - set Y-offset * * do a relative vertical move, whenever the offset changes * */ static void PSmove_offset(int offset) { if (offset != PS_offset) { PSprintf("0 %d R\n", PS_offset - offset ); PS_offset = offset; } } /* * Return an indication of whether or not the current element has a footnote. * * an element has a footnote if it is text or an image and its anchorHRef is not null. * If the element is a textual element with an anchorHRef, that has been split across * lines then it should be followed by a linefeed element and a text element with the * same anchorHRef. In this case say that the element doesn't have a footnote so as * to avoid duplicate footnotes. */ static int has_footnote(struct ele_rec *el) { int rc = 0; struct ele_rec *next; char *anchorHRef; if (el == NULL) { return 0; } anchorHRef = el->anchorHRef; if (anchorHRef != NULL) { switch (el->type) { case E_TEXT: case E_IMAGE: for (next = el->next; el; el = next, next = el->next) { if (next == NULL) { rc = 1; break; } else if (next->anchorHRef == NULL) { rc = 1; break; } else if (!STREQ(next->anchorHRef, anchorHRef)) { rc = 1; break; } else if ( (next->type == E_TEXT) || (next->type == E_IMAGE)) { rc = 0; break; } } break; default: break; } } return rc; } /* * PSfootnote - output a footnote mark and store the footnote * * the footnote mark is placed at the current point, enclosed in a gsave/grestore * pair so that the position of the following output is not affected. * The reference is stored in a malloced array (which may need to be expanded), to * be output at the end of the page. */ static void PSfootnote(char *href, double height) { PSprintf("gsave 0 %.2f R RF %d SF (%d) S grestore\n", height, footnote_ptsize, cur_ftn_no++); if (n_saved_ftns == ftn_array_size) { ftn_array_size += 16; if (!footnotes) { footnotes = (char **)calloc(ftn_array_size,sizeof(char *)); } else { footnotes = (char **)realloc((void *)footnotes, (ftn_array_size * sizeof(char *))); } if (footnotes == NULL) { #ifndef DISABLE_TRACE if (htmlwTrace) { fprintf(stderr, "PSfootnote realloc failed\n"); } #endif return; } } footnotes[n_saved_ftns++] = href; } /* * PStext - output text * * show text "t", and protect special characters if needed * if Underline is non-zero, the text is underlined. * */ static void PStext(HTMLWidget hw, struct ele_rec *eptr, int fontfamily, String s) { String s2; String stmp; unsigned char ch; int underline = eptr->underline_number; int ascent; PSfont(hw, eptr->font, fontfamily); /* set font */ if (PS_fontascent == 0) ascent = eptr->font->ascent; else ascent = PS_fontascent; PSmove_offset(eptr->y_offset + ascent); /* Allocate a string long enough to hold the original string with every character stored as an octal escape (worst case scenario). */ s2 = (String) malloc(strlen(s) * 4 + 1); if (s2 == NULL) { #ifndef DISABLE_TRACE if (htmlwTrace) { fprintf(stderr, "PStext malloc failed\n"); } #endif return; } /* For each char in s, if it is a special char, insert "\" * into the new string s2, then insert the actual char */ for (stmp = s2; (ch = *s++) != '\0';) { if ((ch == L_PAREN) || (ch == R_PAREN) || (ch == B_SLASH)) { *stmp++ = B_SLASH; *stmp++ = ch; } else if (ch > (unsigned char) MAX_ASCII) { /* convert to octal */ *stmp++ = B_SLASH; *stmp++ = ((ch >> 6) & 007) + '0'; *stmp++ = ((ch >> 3) & 007) + '0'; *stmp++ = (ch & 007) + '0'; } else { *stmp++ = ch; } } *(stmp) = '\0'; PSprintf("(%s)%c\n", s2, (underline)?'U':'S'); if (HTML_Print_Footers && has_footnote(eptr)) { PSfootnote(eptr->anchorHRef, 0.7 * ascent); } free(s2); } /* * PSbullet - output a bullet * * The bullet is normally filled, except for a bullet with an indent level * of two. The size of the higher level bullets is just somewhat smaller * */ static void PSbullet(HTMLWidget hw, struct ele_rec *eptr, int fontfamily) { int width = ( eptr->font->max_bounds.lbearing + eptr->font->max_bounds.rbearing); int offset = eptr->y_offset + eptr->font->ascent; int level = eptr->indent_level; double size = eptr->line_height / 5.5; if (size < 1.1) size = 1.1; if (level > 2) size /= 1.33333; /* the next line is a hack to get a good position of the * bullet in most practical cases, otherwise the * bullet may appear just a bit too low (for large fonts) * What is does is to compare the lineheight with * the lineheight of the next element, to correct * for the possibly too large y_offset */ if ( (eptr->next != NULL) && ( (eptr->next->type == E_TEXT) || (eptr->next->type == E_IMAGE))) { offset += (eptr->line_height - eptr->next->line_height); } PSfont(hw, eptr->font, fontfamily); PSmove_offset(offset - width/4); PSprintf(" %f %s\n", size, (level == 2) ? "OB" : "B"); } /* * PShrule - draw a horizontal line with the given width * * nothing special, just draw a line, from the current position to * the right side of the paper. * */ static void PShrule(HTMLWidget hw, struct ele_rec *eptr, int fontfamily) { int length = hw->html.doc_width; PSmove_offset(eptr->y_offset); PSprintf("%d HR\n", length); } /* * PStable - draw a table * * Currently just draw a box of the dimensions of the table. * */ static void PStable(HTMLWidget hw, struct ele_rec *eptr, int fontfamily) { struct table_rec *tptr = eptr->table_data; int width = tptr->width; int height = tptr->height; #if 0 PSmove_offset(eptr->y_offset); PSprintf("gsave currentpoint %d sub translate ", height); PSprintf("%d %d scale\n", width, height); PSprintf("SQ stroke\n"); PSprintf("grestore\n"); #endif } /* * PSwidget - draw a widget (form field) * * Currently just draw a grey box of the dimensions of the field. * This is nowhere near complete but is a first step. * The widget record type field gives the type of field: * 0 textfield * 1 checkbox * 2 radiobox * 3 pushbutton * 4 password * 5 option menu */ static void PSwidget(HTMLWidget hw, struct ele_rec *eptr, int fontfamily) { struct wid_rec *wptr = eptr->widget_data; int w = wptr->width; int h = wptr->height; #if 1 /* Comment out? */ PSmove_offset(eptr->y_offset); PSprintf("gsave currentpoint %d sub translate ", h); PSprintf("%d %d scale\n", w, h); PSprintf("SQ 0.9 setgray fill\n"); PSprintf("grestore\n"); #endif } /* * PSrle_encode - perform run length encoding * * does the run-length encoding. This is done to reduce the file size and * therefore the time to send the file to the printer. You get longer * processing time instead. * * rle is encoded as such: * # 'run' of count+1 equal pixels * # count+1 non-equal pixels * count can range between 0 and 127 * * returns length of the rleline vector * */ static int PSrle_encode(unsigned char *scanline, unsigned char *rleline, int wide) { int i, j, blocklen, isrun, rlen; unsigned char block[256], pix; blocklen = isrun = rlen = 0; for (i = 0; i < wide; i++) { /* there are 5 possible states: * 0: block empty. * 1: block is a run, current pix == previous pix * 2: block is a run, current pix != previous pix * 3: block not a run, current pix == previous pix * 4: block not a run, current pix != previous pix */ pix = scanline[i]; if (!blocklen) { /* case 0: empty */ block[blocklen++] = pix; isrun = 1; } else if (isrun) { if (pix == block[blocklen-1]) { /* case 1: isrun, prev==cur */ block[blocklen++] = pix; } else { /* case 2: isrun, prev!=cur */ if (blocklen>1) { /* we have a run block to flush */ rleline[rlen++] = blocklen-1; rleline[rlen++] = block[0]; /* start new run block with pix */ block[0] = pix; blocklen = 1; } else { /* blocklen<=1, turn into non-run */ isrun = 0; block[blocklen++] = pix; } } } else { /* not a run */ if (pix == block[blocklen-1]) { /* case 3: non-run, prev==cur */ if (blocklen>1) { /* have a non-run block to flush */ rleline[rlen++] = (blocklen-1) | 0x80; for (j=0; jI function", " /rgbdata exch store % call input 'rgbdata'", " rgbdata length 3 idiv", " /npixls exch store", " /rgbindx 0 store", " /grays npixls string store % str to hold the result", " 0 1 npixls 1 sub {", " grays exch", " rgbdata rgbindx get 20 mul % Red", " rgbdata rgbindx 1 add get 32 mul % Green", " rgbdata rgbindx 2 add get 12 mul % Blue", " add add 64 idiv % I = .5G + .31R + .18B", " put", " /rgbindx rgbindx 3 add store", " } for", " grays", " } bind def\n", /* Utility procedure for colorimage operator. * This procedure takes two procedures off the * stack and merges them into a single procedure */ " /mergeprocs { % def", " dup length", " 3 -1 roll dup length dup 5 1 roll", " 3 -1 roll add array cvx dup", " 3 -1 roll 0 exch putinterval", " dup 4 2 roll putinterval", " } bind def\n", " /colorimage { % def", /* remove 'false 3' operands */ " pop pop", " {colortogray} mergeprocs", " image", " } bind def", /* end of 'false' case */ " } ifelse" }; PSconst_out(txt); } /* * PScolormap - write colormap * * spits out code for the colormap of the following image * if !color, it spits out a mono-ized graymap * */ static void PScolormap(int color, int nc, int *rmap, int *gmap, int *bmap) { int i; /* define the colormap */ PSprintf("/cmap %d string def\n\n\n", nc * ((color) ? 3 : 1)); /* load up the colormap */ PSprintf("currentfile cmap readhexstring\n"); for (i=0; i>8, gmap[i]>>8, bmap[i]>>8); else PSprintf("%02x ", MONO(rmap[i], gmap[i], bmap[i])); if ((i%10) == 9) PSprintf("\n"); } PSprintf("\n"); PSprintf("pop pop\n"); /* lose return values from readhexstring */ } /* * PSrle_cmapimage - define rlecmapimage operator * */ static void PSrle_cmapimage(int color) { static char *txt[] = { /* rlecmapimage expects to have 'w h bits matrix' on stack */ "/rlecmapimage {", " /buffer 1 string def", " /rgbval 3 string def", " /block 384 string def", " { currentfile buffer readhexstring pop", " /bcount exch 0 get store", " bcount 128 ge", " { ", " 0 1 bcount 128 sub", " { currentfile buffer readhexstring pop pop" }; static char *txt_color[] = { " /rgbval cmap buffer 0 get 3 mul 3 getinterval store", " block exch 3 mul rgbval putinterval", " } for", " block 0 bcount 127 sub 3 mul getinterval", " }", " { ", " currentfile buffer readhexstring pop pop", " /rgbval cmap buffer 0 get 3 mul 3 getinterval store", " 0 1 bcount { block exch 3 mul rgbval putinterval } for", " block 0 bcount 1 add 3 mul getinterval", " } ifelse", " }", " false 3 colorimage", "} bind def" }; static char *txt_gray[] = { " /rgbval cmap buffer 0 get 1 getinterval store", " block exch rgbval putinterval", " } for", " block 0 bcount 127 sub getinterval", " }", " { ", " currentfile buffer readhexstring pop pop", " /rgbval cmap buffer 0 get 1 getinterval store", " 0 1 bcount { block exch rgbval putinterval } for", " block 0 bcount 1 add getinterval", " } ifelse", " }", " image", "} bind def" }; PSconst_out(txt); if (color) { PSconst_out(txt_color); } else { PSconst_out(txt_gray); } } /* * PSwrite_bw - write B&W image * * Write the given image array 'pic' (B/W stippled, 1 byte per pixel, * 0=blk,1=wht) out as hexadecimal, max of 72 hex chars per line. If * 'flipbw', then 0=white, 1=black. Returns '0' if everythings fine, * 'EOF' if writing failed. * */ static int PSwrite_bw(unsigned char *pic, int w, int h, int flipbw) { int i, j; int err=0; unsigned char outbyte, bitnum, bit; outbyte = bitnum = 0; for (i=0; ipic_data; unsigned char *imgp = img->image_data; int anchor = (eptr->anchorHRef != NULL); int ncolors = img->num_colors; int i, j; int w = img->width; int h = img->height; int slen, colorps, colortype, bits; int err=0; int extra = 0; /* Isgray returns true if the nth color index is a gray value */ # define Isgray(i,n) (i->reds[n]==i->greens[n] && i->reds[n]==i->blues[n]) /* Is_bg returns true if the nth color index is the screen background */ # define Is_bg(i,n) (i->reds[n]==bg_color.red && \ i->greens[n]==bg_color.green && i->blues[n]==bg_color.blue) /* Is_fg returns true if the nth color index is the screen foreground */ # define Is_fg(i,n) (i->reds[n]==fg_color.red && \ i->greens[n]==fg_color.green && i->blues[n]==fg_color.blue) PSmove_offset(eptr->y_offset); if (anchor) { /* draw an outline by drawing a slightly larger black square * below the actual image */ PSprintf("gsave currentpoint %d sub translate ", h); PSprintf("0 -2 translate %d %d scale\n", w+4, h+4); PSprintf("SQ fill\n"); PSprintf("grestore\n"); extra = 4; } if (imgp == NULL) { /* image was not available... do something instead * draw an empty square for example */ PSprintf("gsave currentpoint %d sub translate", h); if (anchor) PSprintf(" 2 0 translate"); else PSprintf(" 0 2 translate"); PSprintf(" %d %d scale\n", w, h); PSprintf("0.9 setgray SQ fill\n"); PSprintf("grestore\n"); /* move currentpoint just right of image */ PSprintf("%d 0 R\n", w+extra); return; } /* this is a hack to see if the image is Black & White, * Greyscale or 8 bit color * assume it's bw if it has only one or two colors, both some grey's * assume it's greyscale if all the colors (>2) are grey * Images with only one color do occur too. */ if ( ( (ncolors == 2) && ( (Isgray(img,0) && Isgray(img,1)) || (Is_bg(img,0) && Is_fg(img,1)) || (Is_fg(img,0) && Is_bg(img,1)) )) || ( (ncolors == 1) && (Isgray(img,0) || Is_bg(img,0) || Is_fg(img,0)))) { colortype = F_BWDITHER; slen = (w+7)/8; bits = 1; colorps = 0; } else { colortype = F_GREYSCALE; slen = w; bits = 8; colorps = 0; for (i=0; ireds[0], img->greens[0],img->blues[0]) > MONO(img->reds[1], img->greens[1], img->blues[1])) || (ncolors == 1 && MONO(img->reds[0], img->greens[0],img->blues[0]) > MONO(127, 127, 127) )) { flipbw=1; } /* dimensions of data */ PSprintf("%d %d %d\n", w, h, bits); /* mapping matrix */ PSprintf("[%d 0 0 %d 0 %d]\n\n", w, -h, h); PSprintf("{currentfile pix readhexstring pop}\n"); PSprintf("image\n"); /* write the actual image data */ err = PSwrite_bw(imgp, w, h, flipbw); } else { /* all other formats */ unsigned char *rleline = (unsigned char *) NULL; int rlen; /* if we're using color, make sure 'colorimage' is defined */ if (colorps) PScolor_image(); PScolormap(colorps, ncolors, img->reds, img->greens, img->blues); PSrle_cmapimage(colorps); /* dimensions of data */ PSprintf("%d %d %d\n", w, h, bits); /* mapping matrix */ PSprintf("[%d 0 0 %d 0 %d]\n", w, -h, h); PSprintf("rlecmapimage\n"); rleline = (unsigned char *) malloc(w * 2); if (!rleline) { #ifndef DISABLE_TRACE if (htmlwTrace) { fprintf(stderr,"failed to malloc space for rleline\n"); } #endif return; } for (i=0; ianchorHRef, 2.0); } /* forget about the macro's */ # undef Isgray # undef Is_fg # undef Is_bg } /* * ParseTextToPSString - entry point for postscript output * * Parse all the formatted text elements from start to end * into an ascii text string, and return it. * Very like ParseTextToString() except the text is prettied up * into Postscript to show headers and the like. * space_width and lmargin tell us how many spaces * to indent lines. * Because this routine is only used to print whole documents, * some parameters are not needed at all! * Also it assumes that you are indeed printing the whole document, and * not just a selected portion of it. It therefore can assume that * only for the first page the initialization is needed, and only * the last page has the trailers. You cannot use ParseTextToPSString() * as you can use ParseTextToString() because of this initialization code. * * the fontfamily parameter is new * The font is encoded as: * 0: times (default for now) * 1: helvetica * 2: new century schoolbook * 3: lucida */ String ParseTextToPSString(HTMLWidget hw, struct ele_rec *elist, struct ele_rec *startp, struct ele_rec *endp, int start_pos, int end_pos, int space_width, int lmargin, int fontfamily, char *url, char *time_str) { int xpos, ypos, epos; int height; double pagewidth; int line = -1; struct ele_rec *eptr; struct ele_rec *start; struct ele_rec *end; struct ele_rec *last; struct ele_rec *tmpptr; unsigned long fg_pixel, bg_pixel; int footnotes_this_page = 0; int footnotes_this_line; int reserved_space; if (startp == NULL) return(NULL); /* * Get the foreground and background colors so we can check later * for black&white documents */ XtVaGetValues (hw->html.view, #ifdef MOTIF XtNforeground, &fg_pixel, #endif XtNbackground, &bg_pixel, NULL); #ifndef MOTIF XtVaGetValues ((Widget)hw, XtNforeground, &fg_pixel, NULL); #endif fg_color.pixel = fg_pixel; bg_color.pixel = bg_pixel; XQueryColor(XtDisplay(hw->html.view), (installed_colormap ? installed_cmap : DefaultColormap(XtDisplay(hw->html.view), DefaultScreen(XtDisplay(hw->html.view)))), &fg_color); XQueryColor(XtDisplay(hw->html.view), (installed_colormap ? installed_cmap : DefaultColormap(XtDisplay(hw->html.view), DefaultScreen(XtDisplay(hw->html.view)))), &bg_color); /* this piece of code is needed if the user selects a portion * of the document with the mouse. * I think it will never be used, but I left it in anyway. F. */ if (SwapElements(startp, endp, start_pos, end_pos)) { start = endp; end = startp; epos = start_pos; start_pos = end_pos; end_pos = epos; } else { start = startp; end = endp; } /* Setup page size according to user preference. */ if (HTML_Print_Paper_Size_A4) page_dimens = a4_page_dimens; else page_dimens = us_letter_page_dimens; page_dimens.text_height = ( page_dimens.page_height - page_dimens.top_margin - page_dimens.bot_margin); page_dimens.text_width = ( page_dimens.page_width - page_dimens.left_margin - page_dimens.right_margin); /* Calculate the number of Postscript points per pixel of current * screen, and the height of the page in pixels (used in figuring * when we've hit the bottom of the page). */ Points_Pixel = 72.0 / GetDpi(hw); #ifdef OLD pagewidth = hw->html.doc_width; #else /* gustaf fix */ pagewidth = hw->html.view_width; #endif /* gustaf fix */ /* reduce the scaling if the width used for formatting is greater * than 8 * 72 pixels (8 inch) * In theory, this is not what you want for A4 paper (only 8.27 inch * wide), but I guess that the hw->html.doc_width includes some * left and right margins, so it seems to work in practice. */ if (pagewidth > page_dimens.text_width) Points_Pixel = Points_Pixel * page_dimens.text_width / pagewidth; Pixels_Page = (int) (page_dimens.text_height / Points_Pixel); PSinit(); PSheader(hw->html.title, fontfamily, url, time_str); PSnewpage(); last = start; eptr = start; while ((eptr != NULL) && (eptr != end)) { /* Skip the special internal text added for multi-page * documents. */ if (eptr->internal == True) { if (eptr->type == E_LINEFEED) { PS_page_offset += eptr->line_height; } eptr = eptr->next; continue; } /* check if this is a newline */ if (line != eptr->line_number) { /* calculate max height */ height = 0; footnotes_this_line = 0; line = eptr->line_number; tmpptr = eptr; while (tmpptr != NULL && tmpptr->line_number == line) { if (tmpptr->line_height > height) height = tmpptr->line_height; tmpptr = tmpptr->next; if (HTML_Print_Footers && has_footnote(tmpptr)) footnotes_this_line++; } ypos = eptr->y - PS_page_offset ; xpos = eptr->x - lmargin; if (xpos < 0) xpos = 0; /* check if line fits completly on page */ reserved_space = 0; if (footnotes_this_page || footnotes_this_line) { reserved_space = ( ( footnote_space + ( footnote_ptsize * ( footnotes_this_page + footnotes_this_line))) / Points_Pixel); } if (ypos + height + reserved_space > PS_start_y + Pixels_Page) { PS_start_y = ypos; PSshowpage(); PSnewpage(); footnotes_this_page = 0; } footnotes_this_page += footnotes_this_line; PSmoveto( xpos, ypos); } switch (eptr->type) { case E_TEXT: PStext(hw, eptr, fontfamily, (String)((eptr == start) ? (eptr->edata + start_pos) : eptr->edata)); break; case E_BULLET: PSbullet(hw, eptr, fontfamily); break; case E_IMAGE: PSimage(hw, eptr, fontfamily); break; case E_LINEFEED: break; case E_HRULE: PShrule(hw, eptr, fontfamily); break; case E_TABLE: PStable(hw, eptr, fontfamily); break; case E_WIDGET: PSwidget(hw, eptr, fontfamily); break; } last = eptr; eptr = eptr->next; } PSshowpage(); PStrailer(); return (PS_string); }