summaryrefslogtreecommitdiffstats
path: root/kdvi/dviRenderer_prescan.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'kdvi/dviRenderer_prescan.cpp')
-rw-r--r--kdvi/dviRenderer_prescan.cpp789
1 files changed, 789 insertions, 0 deletions
diff --git a/kdvi/dviRenderer_prescan.cpp b/kdvi/dviRenderer_prescan.cpp
new file mode 100644
index 00000000..59bdeb25
--- /dev/null
+++ b/kdvi/dviRenderer_prescan.cpp
@@ -0,0 +1,789 @@
+// dviRenderer_prescan.cpp
+//
+// Part of KDVI - A DVI previewer for the KDE desktop environemt
+//
+// (C) 2003--2004 Stefan Kebekus
+// Distributed under the GPL
+
+#include <config.h>
+
+#include "dviRenderer.h"
+#include "dvi.h"
+#include "dviFile.h"
+#include "fontpool.h"
+#include "kdvi_multipage.h"
+#include "performanceMeasurement.h"
+#include "prebookmark.h"
+#include "psgs.h"
+#include "TeXFont.h"
+#include "xdvi.h"
+
+#include <kdebug.h>
+#include <klocale.h>
+#include <kmimetype.h>
+#include <kprocess.h>
+#include <kprocio.h>
+#include <kprogress.h>
+#include <qapplication.h>
+#include <qbitmap.h>
+#include <qdir.h>
+#include <qfileinfo.h>
+#include <qimage.h>
+#include <qpainter.h>
+#include <qpaintdevice.h>
+
+
+extern QPainter foreGroundPaint;
+extern void parse_special_argument(const QString& strg, const char* argument_name, int* variable);
+
+
+//#define DEBUG_PRESCAN
+
+
+void dviRenderer::prescan_embedPS(char *cp, Q_UINT8 *beginningOfSpecialCommand)
+{
+#ifdef DEBUG_PRESCAN
+ kdDebug(4300) << "dviRenderer::prescan_embedPS( cp = " << cp << " ) " << endl;
+#endif
+
+ // Encapsulated Postscript File
+ if (strncasecmp(cp, "PSfile=", 7) != 0)
+ return;
+
+ QString command(cp+7);
+
+ QString include_command = command.simplifyWhiteSpace();
+
+ // The line is supposed to start with "..ile=", and then comes the
+ // filename. Figure out what the filename is and stow it away. Of
+ // course, this does not work if the filename contains spaces
+ // (already the simplifyWhiteSpace() above is wrong). If you have
+ // files like this, go away.
+ QString EPSfilename = include_command;
+ EPSfilename.truncate(EPSfilename.find(' '));
+
+ // Strip enclosing quotation marks which are included by some LaTeX
+ // macro packages (but not by others). This probably means that
+ // graphic files are no longer found if the filename really does
+ // contain quotes, but we don't really care that much.
+ if ((EPSfilename.at(0) == '\"') && (EPSfilename.at(EPSfilename.length()-1) == '\"'))
+ EPSfilename = EPSfilename.mid(1,EPSfilename.length()-2);
+
+ // If the file is neither a PostScript not a PDF file, we exit here.
+ // The graphic file is later read when the page is rendered.
+ KMimeType::Ptr const mime_type =
+ KMimeType::findByFileContent(EPSfilename);
+ QString const & mime_type_name = mime_type->name();
+
+ bool const is_ps_file = (mime_type_name == "application/postscript" ||
+ mime_type_name == "image/x-eps");
+ bool const is_pdf_file = (!is_ps_file &&
+ mime_type_name == "application/pdf");
+ if (!(is_ps_file || is_pdf_file))
+ return;
+
+ QString originalFName = EPSfilename;
+
+ embedPS_progress->setLabel(i18n("Embedding %1").arg(EPSfilename));
+ qApp->processEvents();
+
+
+ // Now locate the Gfx file on the hard disk...
+ EPSfilename = ghostscript_interface::locateEPSfile(EPSfilename, baseURL);
+
+ // If the EPSfilename really points to a PDF file, convert that file now.
+ if (is_pdf_file)
+ EPSfilename = dviFile->convertPDFtoPS(EPSfilename);
+
+ if (!QFile::exists(EPSfilename)) {
+ // Find the number of the page
+ Q_UINT32 currentOffset = beginningOfSpecialCommand - dviFile->dvi_Data();
+ Q_UINT16 page;
+ for(page=0; page < dviFile->total_pages; page++)
+ if ((dviFile->page_offset[page] <= currentOffset) && (currentOffset <= dviFile->page_offset[page+1]))
+ break;
+ errorMsg += i18n("Page %1: The PostScript file <strong>%2</strong> could not be found.<br>").arg(page+1).arg(originalFName);
+ embedPS_progress->progressBar()->advance(1);
+ qApp->processEvents();
+ return;
+ }
+
+ // Now parse the arguments.
+ int llx = 0;
+ int lly = 0;
+ int urx = 0;
+ int ury = 0;
+ int rwi = 0;
+ int rhi = 0;
+ int angle = 0;
+
+ // just to avoid ambiguities; the filename could contain keywords
+ include_command = include_command.mid(include_command.find(' '));
+
+ parse_special_argument(include_command, "llx=", &llx);
+ parse_special_argument(include_command, "lly=", &lly);
+ parse_special_argument(include_command, "urx=", &urx);
+ parse_special_argument(include_command, "ury=", &ury);
+ parse_special_argument(include_command, "rwi=", &rwi);
+ parse_special_argument(include_command, "rhi=", &rhi);
+ parse_special_argument(include_command, "angle=", &angle);
+
+ int clip=include_command.find(" clip"); // -1 if clip keyword is not present, >= 0 otherwise
+
+ // Generate the PostScript commands to be included
+ QString PS = QString("ps: @beginspecial %1 @llx %2 @lly %3 @urx %4 @ury").arg(llx).arg(lly).arg(urx).arg(ury);
+ if (rwi != 0)
+ PS.append( QString(" %1 @rwi").arg(rwi) );
+ if (rhi != 0)
+ PS.append( QString(" %1 @rhi").arg(rhi) );
+ if (angle != 0)
+ PS.append( QString(" %1 @angle").arg(angle) );
+ if (clip != -1)
+ PS.append(" @clip");
+ PS.append( " @setspecial\n" );
+
+ QFile file( EPSfilename );
+ if ( file.open( IO_ReadOnly ) ) {
+ QTextStream stream( &file );
+ while ( !stream.atEnd() ) {
+ PS += stream.readLine().section( '%', 0, 0);
+ PS += "\n";
+ }
+ file.close();
+ }
+ PS.append( "@endspecial" );
+ PS = PS.simplifyWhiteSpace();
+
+
+ _isModified = true;
+ Q_UINT32 lengthOfOldSpecial = command_pointer - beginningOfSpecialCommand;
+ Q_UINT32 lengthOfNewSpecial = PS.length()+5;
+
+ QMemArray<Q_UINT8> newDVI(dviFile->size_of_file + lengthOfNewSpecial-lengthOfOldSpecial);
+
+ Q_UINT8 *commandPtrSav = command_pointer;
+ Q_UINT8 *endPtrSav = end_pointer;
+ end_pointer = newDVI.data() + dviFile->size_of_file + lengthOfNewSpecial-lengthOfOldSpecial;
+ memcpy(newDVI.data(), dviFile->dvi_Data(), beginningOfSpecialCommand-dviFile->dvi_Data());
+ command_pointer = newDVI.data()+(beginningOfSpecialCommand-dviFile->dvi_Data());
+ command_pointer[0] = XXX4;
+ command_pointer++;
+ writeUINT32(PS.length());
+ memcpy(newDVI.data()+(beginningOfSpecialCommand-dviFile->dvi_Data())+5, PS.latin1(), PS.length() );
+ memcpy(newDVI.data()+(beginningOfSpecialCommand-dviFile->dvi_Data())+lengthOfNewSpecial, beginningOfSpecialCommand+lengthOfOldSpecial,
+ dviFile->size_of_file-(beginningOfSpecialCommand-dviFile->dvi_Data())-lengthOfOldSpecial );
+
+ // Adjust page pointers in the DVI file
+ dviFile->size_of_file = dviFile->size_of_file + lengthOfNewSpecial-lengthOfOldSpecial;
+ end_pointer = newDVI.data() + dviFile->size_of_file;
+ Q_UINT32 currentOffset = beginningOfSpecialCommand-dviFile->dvi_Data();
+ for(Q_UINT16 i=0; i < dviFile->total_pages; i++) {
+ if (dviFile->page_offset[i] > currentOffset) {
+ dviFile->page_offset[i] = dviFile->page_offset[i] + lengthOfNewSpecial-lengthOfOldSpecial;
+ command_pointer = dviFile->page_offset[i] + newDVI.data() + 4*10 + 1;
+ Q_UINT32 a = readUINT32();
+ if (a > currentOffset) {
+ a = a + lengthOfNewSpecial-lengthOfOldSpecial;
+ command_pointer = dviFile->page_offset[i] + newDVI.data() + 4*10 + 1;
+ writeUINT32(a);
+ }
+ }
+ }
+
+
+ dviFile->beginning_of_postamble = dviFile->beginning_of_postamble + lengthOfNewSpecial - lengthOfOldSpecial;
+ dviFile->page_offset[dviFile->total_pages] = dviFile->beginning_of_postamble;
+
+ command_pointer = newDVI.data() + dviFile->beginning_of_postamble + 1;
+ Q_UINT32 a = readUINT32();
+ if (a > currentOffset) {
+ a = a + lengthOfNewSpecial - lengthOfOldSpecial;
+ command_pointer = newDVI.data() + dviFile->beginning_of_postamble + 1;
+ writeUINT32(a);
+ }
+
+ command_pointer = newDVI.data() + dviFile->size_of_file - 1;
+ while((*command_pointer == TRAILER) && (command_pointer > newDVI.data()))
+ command_pointer--;
+ command_pointer -= 4;
+ writeUINT32(dviFile->beginning_of_postamble);
+ command_pointer -= 4;
+
+ command_pointer = commandPtrSav;
+ end_pointer = endPtrSav;
+
+ // Modify all pointers to point to the newly allocated memory
+ command_pointer = newDVI.data() + (command_pointer - dviFile->dvi_Data()) + lengthOfNewSpecial-lengthOfOldSpecial;
+ end_pointer = newDVI.data() + (end_pointer - dviFile->dvi_Data()) + lengthOfNewSpecial-lengthOfOldSpecial;
+
+ dviFile->setNewData(newDVI);
+
+ embedPS_progress->progressBar()->advance(1);
+ qApp->processEvents();
+ return;
+}
+
+
+void dviRenderer::prescan_removePageSizeInfo(char *cp, Q_UINT8 *beginningOfSpecialCommand)
+{
+#ifdef DEBUG_PRESCAN
+ kdDebug(4300) << "dviRenderer::prescan_embedPS( cp = " << cp << " ) " << endl;
+#endif
+
+ // Encapsulated Postscript File
+ if (strncasecmp(cp, "papersize=", 10) != 0)
+ return;
+
+ for (Q_UINT8 *ptr=beginningOfSpecialCommand; ptr<command_pointer; ptr++)
+ *ptr = NOP;
+}
+
+
+void dviRenderer::prescan_ParsePapersizeSpecial(const QString& _cp)
+{
+#ifdef DEBUG_PRESCAN
+ kdDebug(4300) << "Papersize-Special : papersize" << cp << endl;
+#endif
+
+ QString cp = _cp.simplifyWhiteSpace();
+
+ if (cp[0] == '=') {
+ cp = cp.mid(1);
+ dviFile->suggestedPageSize = new pageSize;
+ dviFile->suggestedPageSize->setPageSize(cp);
+ } else
+ printErrorMsgForSpecials(i18n("The papersize data '%1' could not be parsed.").arg(cp));
+
+ return;
+}
+
+
+void dviRenderer::prescan_ParseBackgroundSpecial(const QString& cp)
+{
+ QColor col = parseColorSpecification(cp.stripWhiteSpace());
+ if (col.isValid())
+ for(Q_UINT16 page=current_page; page < dviFile->total_pages; page++)
+ PS_interface->setBackgroundColor(page, col);
+ return;
+}
+
+
+void dviRenderer::prescan_ParseHTMLAnchorSpecial(const QString& _cp)
+{
+ QString cp = _cp;
+ cp.truncate(cp.find('"'));
+ Length l;
+ l.setLength_in_inch(currinf.data.dvi_v/(resolutionInDPI*shrinkfactor));
+ anchorList[cp] = Anchor(current_page+1, l);
+}
+
+
+void dviRenderer::prescan_ParsePSHeaderSpecial(const QString& cp)
+{
+#ifdef DEBUG_PRESCAN
+ kdDebug(4300) << "PostScript-special, header " << cp.latin1() << endl;
+#endif
+
+ QString _file = cp;
+
+ // If the file is not found in the current directory, use kpsewhich
+ // to find it.
+ if (!QFile::exists(_file)) {
+ // Otherwise, use kpsewhich to find the eps file.
+ KProcIO proc;
+ proc << "kpsewhich" << cp;
+ proc.start(KProcess::Block);
+ proc.readln(_file);
+ }
+
+ if (QFile::exists(_file))
+ PS_interface->PostScriptHeaderString->append( QString(" (%1) run\n").arg(_file) );
+}
+
+
+void dviRenderer::prescan_ParsePSBangSpecial(const QString& cp)
+{
+#ifdef DEBUG_PRESCAN
+ kdDebug(4300) << "PostScript-special, literal header " << cp.latin1() << endl;
+#endif
+
+ PS_interface->PostScriptHeaderString->append( " @defspecial \n" );
+ PS_interface->PostScriptHeaderString->append( cp );
+ PS_interface->PostScriptHeaderString->append( " @fedspecial \n" );
+}
+
+
+void dviRenderer::prescan_ParsePSQuoteSpecial(const QString& cp)
+{
+#ifdef DEBUG_PRESCAN
+ kdError(4300) << "PostScript-special, literal PostScript " << cp.latin1() << endl;
+#endif
+
+ double PS_H = (currinf.data.dvi_h*300.0)/(65536*1200)-300;
+ double PS_V = (currinf.data.dvi_v*300.0)/1200 - 300;
+ PostScriptOutPutString->append( QString(" %1 %2 moveto\n").arg(PS_H).arg(PS_V) );
+ PostScriptOutPutString->append( " @beginspecial @setspecial \n" );
+ PostScriptOutPutString->append( cp );
+ PostScriptOutPutString->append( " @endspecial \n" );
+}
+
+
+void dviRenderer::prescan_ParsePSSpecial(const QString& cp)
+{
+#ifdef DEBUG_PRESCAN
+ kdDebug(4300) << "PostScript-special, direct PostScript " << cp << endl;
+#endif
+
+ // Unfortunately, in some TeX distribution the hyperref package uses
+ // the dvips driver by default, rather than the hypertex driver. As
+ // a result, the DVI files produced are full of PostScript that
+ // specifies links and anchors, and KDVI would call the ghostscript
+ // interpreter for every page which makes it really slow. This is a
+ // major nuisance, so that we try to filter and interpret the
+ // hypertex generated PostScript here.
+ if (cp.startsWith("ps:SDict begin")) {
+ // We suspect this may be hyperref generated nonsense. Let's check
+ // for some known code that hyperref generates.
+ if (cp == "ps:SDict begin H.S end")
+ return; // start of hyperref rectangle
+ if (cp == "ps:SDict begin H.R end")
+ return; // end of hyperref rectangle
+ if (cp.endsWith("H.A end"))
+ return; // end of hyperref anchor
+ if (cp.endsWith("H.L end"))
+ return; // end of hyperref link
+ if (cp.startsWith("ps:SDict begin /product where{pop product(Distiller)"))
+ return; // hyperref tries to work around Distiller bug
+ if (cp.startsWith("ps:SDict begin [") && cp.endsWith(" pdfmark end")) { // hyperref definition of link/anchor/bookmark/etc
+ if (cp.contains("/DEST")) { // The PostScript code defines an anchor
+ QString anchorName = cp.section('(', 1, 1).section(')', 0, 0);
+ Length l;
+ l.setLength_in_inch(currinf.data.dvi_v/(resolutionInDPI*shrinkfactor));
+ anchorList[anchorName] = Anchor(current_page+1, l);
+ }
+ // The PostScript code defines a bookmark
+ if (cp.contains("/Dest") && cp.contains("/Title"))
+ prebookmarks.append(PreBookmark(PDFencodingToQString(cp.section('(', 2, 2).section(')', 0, 0)),
+ cp.section('(', 1, 1).section(')', 0, 0),
+ cp.section('-', 1, 1).section(' ', 0, 0).toUInt()
+ ));
+ return;
+ }
+ }
+
+ double PS_H = (currinf.data.dvi_h*300.0)/(65536*1200)-300;
+ double PS_V = (currinf.data.dvi_v*300.0)/1200 - 300;
+
+ if (cp.find("ps::[begin]", 0, false) == 0) {
+ PostScriptOutPutString->append( QString(" %1 %2 moveto\n").arg(PS_H).arg(PS_V) );
+ PostScriptOutPutString->append( QString(" %1\n").arg(cp.mid(11)) );
+ } else {
+ if (cp.find("ps::[end]", 0, false) == 0) {
+ PostScriptOutPutString->append( QString(" %1\n").arg(cp.mid(9)) );
+ } else {
+ if (cp.find("ps::", 0, false) == 0) {
+ PostScriptOutPutString->append( QString(" %1\n").arg(cp.mid(4)) );
+ } else {
+ PostScriptOutPutString->append( QString(" %1 %2 moveto\n").arg(PS_H).arg(PS_V) );
+ PostScriptOutPutString->append( QString(" %1\n").arg(cp.mid(3)) );
+ }
+ }
+ }
+}
+
+
+void dviRenderer::prescan_ParsePSFileSpecial(const QString& cp)
+{
+#ifdef DEBUG_PRESCAN
+ kdDebug(4300) << "epsf-special: psfile=" << cp <<endl;
+#endif
+
+ QString include_command = cp.simplifyWhiteSpace();
+
+ // The line is supposed to start with "..ile=", and then comes the
+ // filename. Figure out what the filename is and stow it away. Of
+ // course, this does not work if the filename contains spaces
+ // (already the simplifyWhiteSpace() above is wrong). If you have
+ // files like this, go away.
+ QString EPSfilename = include_command;
+ EPSfilename.truncate(EPSfilename.find(' '));
+
+ // Strip enclosing quotation marks which are included by some LaTeX
+ // macro packages (but not by others). This probably means that
+ // graphic files are no longer found if the filename really does
+ // contain quotes, but we don't really care that much.
+ if ((EPSfilename.at(0) == '\"') && (EPSfilename.at(EPSfilename.length()-1) == '\"')) {
+ EPSfilename = EPSfilename.mid(1,EPSfilename.length()-2);
+ }
+
+ // If the file name ends in 'png', 'gif', 'jpg' or 'jpeg', we assume
+ // that this is NOT a PostScript file, and we exit here.
+ QString ending = EPSfilename.section('.', -1).lower();
+ if ((ending == "png") || (ending == "gif") || (ending == "jpg") || (ending == "jpeg")) {
+ dviFile->numberOfExternalNONPSFiles++;
+ return;
+ }
+
+ // Now assume that the graphics file *is* a PostScript file
+ dviFile->numberOfExternalPSFiles++;
+
+ // Now locate the Gfx file on the hard disk...
+ EPSfilename = ghostscript_interface::locateEPSfile(EPSfilename, baseURL);
+
+ // If the EPSfilename really points to a PDF file, convert that file now.
+ if (ending == "pdf")
+ EPSfilename = dviFile->convertPDFtoPS(EPSfilename);
+
+ // Now parse the arguments.
+ int llx = 0;
+ int lly = 0;
+ int urx = 0;
+ int ury = 0;
+ int rwi = 0;
+ int rhi = 0;
+ int angle = 0;
+
+ // just to avoid ambiguities; the filename could contain keywords
+ include_command = include_command.mid(include_command.find(' '));
+
+ parse_special_argument(include_command, "llx=", &llx);
+ parse_special_argument(include_command, "lly=", &lly);
+ parse_special_argument(include_command, "urx=", &urx);
+ parse_special_argument(include_command, "ury=", &ury);
+ parse_special_argument(include_command, "rwi=", &rwi);
+ parse_special_argument(include_command, "rhi=", &rhi);
+ parse_special_argument(include_command, "angle=", &angle);
+
+ int clip=include_command.find(" clip"); // -1 if clip keyword is not present, >= 0 otherwise
+
+ if (QFile::exists(EPSfilename)) {
+ double PS_H = (currinf.data.dvi_h*300.0)/(65536*1200)-300;
+ double PS_V = (currinf.data.dvi_v*300.0)/1200 - 300;
+ PostScriptOutPutString->append( QString(" %1 %2 moveto\n").arg(PS_H).arg(PS_V) );
+ PostScriptOutPutString->append( "@beginspecial " );
+ PostScriptOutPutString->append( QString(" %1 @llx").arg(llx) );
+ PostScriptOutPutString->append( QString(" %1 @lly").arg(lly) );
+ PostScriptOutPutString->append( QString(" %1 @urx").arg(urx) );
+ PostScriptOutPutString->append( QString(" %1 @ury").arg(ury) );
+ if (rwi != 0)
+ PostScriptOutPutString->append( QString(" %1 @rwi").arg(rwi) );
+ if (rhi != 0)
+ PostScriptOutPutString->append( QString(" %1 @rhi").arg(rhi) );
+ if (angle != 0)
+ PostScriptOutPutString->append( QString(" %1 @angle").arg(angle) );
+ if (clip != -1)
+ PostScriptOutPutString->append(" @clip");
+ PostScriptOutPutString->append( " @setspecial \n" );
+ PostScriptOutPutString->append( QString(" (%1) run\n").arg(EPSfilename) );
+ PostScriptOutPutString->append( "@endspecial \n" );
+ }
+
+ return;
+}
+
+
+void dviRenderer::prescan_ParseSourceSpecial(const QString& cp)
+{
+ // if no rendering takes place, i.e. when the DVI file is first
+ // loaded, generate a DVI_SourceFileAnchor. These anchors are used
+ // in forward search, i.e. to relate references line
+ // "src:123file.tex" to positions in the DVI file
+
+ // extract the file name and the numeral part from the string
+ Q_UINT32 j;
+ for(j=0;j<cp.length();j++)
+ if (!cp.at(j).isNumber())
+ break;
+ Q_UINT32 sourceLineNumber = cp.left(j).toUInt();
+ QFileInfo fi1(dviFile->filename);
+ QString sourceFileName = QFileInfo(fi1.dir(), cp.mid(j).stripWhiteSpace()).absFilePath();
+ Length l;
+ l.setLength_in_inch(currinf.data.dvi_v/(resolutionInDPI*shrinkfactor));
+ DVI_SourceFileAnchor sfa(sourceFileName, sourceLineNumber, current_page+1, l);
+ sourceHyperLinkAnchors.push_back(sfa);
+}
+
+
+void dviRenderer::prescan_parseSpecials(char *cp, Q_UINT8 *)
+{
+ QString special_command(cp);
+
+ // Now to those specials which are only interpreted during the
+ // prescan phase, and NOT during rendering.
+
+ // PaperSize special
+ if (strncasecmp(cp, "papersize", 9) == 0) {
+ prescan_ParsePapersizeSpecial(special_command.mid(9));
+ return;
+ }
+
+ // color special for background color
+ if (strncasecmp(cp, "background", 10) == 0) {
+ prescan_ParseBackgroundSpecial(special_command.mid(10));
+ return;
+ }
+
+ // HTML anchor special
+ if (strncasecmp(cp, "html:<A name=", 13) == 0) {
+ prescan_ParseHTMLAnchorSpecial(special_command.mid(14));
+ return;
+ }
+
+ // Postscript Header File
+ if (strncasecmp(cp, "header=", 7) == 0) {
+ prescan_ParsePSHeaderSpecial(special_command.mid(7));
+ return;
+ }
+
+ // Literal Postscript Header
+ if (cp[0] == '!') {
+ prescan_ParsePSBangSpecial(special_command.mid(1));
+ return;
+ }
+
+ // Literal Postscript inclusion
+ if (cp[0] == '"') {
+ prescan_ParsePSQuoteSpecial(special_command.mid(1));
+ return;
+ }
+
+ // PS-Postscript inclusion
+ if (strncasecmp(cp, "ps:", 3) == 0) {
+ prescan_ParsePSSpecial(special_command);
+ return;
+ }
+
+ // Encapsulated Postscript File
+ if (strncasecmp(cp, "PSfile=", 7) == 0) {
+ prescan_ParsePSFileSpecial(special_command.mid(7));
+ return;
+ }
+
+ // source special
+ if (strncasecmp(cp, "src:", 4) == 0) {
+ prescan_ParseSourceSpecial(special_command.mid(4));
+ return;
+ }
+
+ // Finally there are those special commands which must be considered
+ // both during rendering and during the pre-scan phase
+
+ // HTML anchor end
+ if (strncasecmp(cp, "html:</A>", 9) == 0) {
+ html_anchor_end();
+ return;
+ }
+
+ return;
+}
+
+
+void dviRenderer::prescan_setChar(unsigned int ch)
+{
+ TeXFontDefinition *fontp = currinf.fontp;
+ if (fontp == NULL)
+ return;
+
+ if (currinf.set_char_p == &dviRenderer::set_char) {
+ glyph *g = ((TeXFont *)(currinf.fontp->font))->getGlyph(ch, true, globalColor);
+ if (g == NULL)
+ return;
+ currinf.data.dvi_h += (int)(currinf.fontp->scaled_size_in_DVI_units * dviFile->getCmPerDVIunit() *
+ (1200.0 / 2.54)/16.0 * g->dvi_advance_in_units_of_design_size_by_2e20 + 0.5);
+ return;
+ }
+
+ if (currinf.set_char_p == &dviRenderer::set_vf_char) {
+ macro *m = &currinf.fontp->macrotable[ch];
+ if (m->pos == NULL)
+ return;
+ currinf.data.dvi_h += (int)(currinf.fontp->scaled_size_in_DVI_units * dviFile->getCmPerDVIunit() *
+ (1200.0 / 2.54)/16.0 * m->dvi_advance_in_units_of_design_size_by_2e20 + 0.5);
+ return;
+ }
+}
+
+
+void dviRenderer::prescan(parseSpecials specialParser)
+{
+#ifdef DEBUG_PRESCAN
+ kdDebug(4300) << "dviRenderer::prescan( ... )" << endl;
+#endif
+
+ if (resolutionInDPI == 0.0)
+ setResolution(100);
+
+ Q_INT32 RRtmp=0, WWtmp=0, XXtmp=0, YYtmp=0, ZZtmp=0;
+ Q_UINT8 ch;
+ double fontPixelPerDVIunit = dviFile->getCmPerDVIunit() * 1200.0/2.54;
+
+ stack.clear();
+
+ currinf.fontp = NULL;
+ currinf.set_char_p = &dviRenderer::set_no_char;
+
+ for (;;) {
+ ch = readUINT8();
+
+ if (ch <= (unsigned char) (SETCHAR0 + 127)) {
+ prescan_setChar(ch);
+ continue;
+ }
+
+ if (FNTNUM0 <= ch && ch <= (unsigned char) (FNTNUM0 + 63)) {
+ currinf.fontp = currinf.fonttable->find(ch - FNTNUM0);
+ if (currinf.fontp == NULL) {
+ errorMsg = i18n("The DVI code referred to font #%1, which was not previously defined.").arg(ch - FNTNUM0);
+ return;
+ }
+ currinf.set_char_p = currinf.fontp->set_char_p;
+ continue;
+ }
+
+
+ Q_INT32 a, b;
+
+ switch (ch) {
+ case SET1:
+ prescan_setChar(readUINT8());
+ break;
+
+ case SETRULE:
+ /* Be careful, dvicopy outputs rules with height =
+ 0x80000000. We don't want any SIGFPE here. */
+ a = readUINT32();
+ b = readUINT32();
+ b = ((long) (b * 65536.0*fontPixelPerDVIunit));
+ currinf.data.dvi_h += b;
+ break;
+
+ case PUTRULE:
+ a = readUINT32();
+ b = readUINT32();
+ break;
+
+ case PUT1:
+ case NOP:
+ break;
+
+ case BOP:
+ command_pointer += 11 * 4;
+ currinf.data.dvi_h = 1200 << 16; // Reminder: DVI-coordinates start at (1",1") from top of page
+ currinf.data.dvi_v = 1200;
+ currinf.data.pxl_v = int(currinf.data.dvi_v/shrinkfactor);
+ currinf.data.w = currinf.data.x = currinf.data.y = currinf.data.z = 0;
+ break;
+
+ case PUSH:
+ stack.push(currinf.data);
+ break;
+
+ case POP:
+ if (stack.isEmpty())
+ return;
+ else
+ currinf.data = stack.pop();
+ break;
+
+ case RIGHT1:
+ case RIGHT2:
+ case RIGHT3:
+ case RIGHT4:
+ RRtmp = readINT(ch - RIGHT1 + 1);
+ currinf.data.dvi_h += ((long) (RRtmp * 65536.0*fontPixelPerDVIunit));
+ break;
+
+ case W1:
+ case W2:
+ case W3:
+ case W4:
+ WWtmp = readINT(ch - W0);
+ currinf.data.w = ((long) (WWtmp * 65536.0*fontPixelPerDVIunit));
+ case W0:
+ currinf.data.dvi_h += currinf.data.w;
+ break;
+
+ case X1:
+ case X2:
+ case X3:
+ case X4:
+ XXtmp = readINT(ch - X0);
+ currinf.data.x = ((long) (XXtmp * 65536.0*fontPixelPerDVIunit));
+ case X0:
+ currinf.data.dvi_h += currinf.data.x;
+ break;
+
+ case DOWN1:
+ case DOWN2:
+ case DOWN3:
+ case DOWN4:
+ {
+ Q_INT32 DDtmp = readINT(ch - DOWN1 + 1);
+ currinf.data.dvi_v += ((long) (DDtmp * 65536.0*fontPixelPerDVIunit))/65536;
+ currinf.data.pxl_v = int(currinf.data.dvi_v/shrinkfactor);
+ }
+ break;
+
+ case Y1:
+ case Y2:
+ case Y3:
+ case Y4:
+ YYtmp = readINT(ch - Y0);
+ currinf.data.y = ((long) (YYtmp * 65536.0*fontPixelPerDVIunit));
+ case Y0:
+ currinf.data.dvi_v += currinf.data.y/65536;
+ currinf.data.pxl_v = int(currinf.data.dvi_v/shrinkfactor);
+ break;
+
+ case Z1:
+ case Z2:
+ case Z3:
+ case Z4:
+ ZZtmp = readINT(ch - Z0);
+ currinf.data.z = ((long) (ZZtmp * 65536.0*fontPixelPerDVIunit));
+ case Z0:
+ currinf.data.dvi_v += currinf.data.z/65536;
+ currinf.data.pxl_v = int(currinf.data.dvi_v/shrinkfactor);
+ break;
+
+ case FNT1:
+ case FNT2:
+ case FNT3:
+ case FNT4:
+ currinf.fontp = currinf.fonttable->find(readUINT(ch - FNT1 + 1));
+ if (currinf.fontp == NULL)
+ return;
+ currinf.set_char_p = currinf.fontp->set_char_p;
+ break;
+
+ case XXX1:
+ case XXX2:
+ case XXX3:
+ case XXX4:
+ {
+ Q_UINT8 *beginningOfSpecialCommand = command_pointer-1;
+ a = readUINT(ch - XXX1 + 1);
+ if (a > 0) {
+ char *cmd = new char[a+1];
+ strncpy(cmd, (char *)command_pointer, a);
+ command_pointer += a;
+ cmd[a] = '\0';
+ (this->*specialParser)(cmd, beginningOfSpecialCommand);
+ delete [] cmd;
+ }
+ }
+ break;
+
+ case FNTDEF1:
+ case FNTDEF2:
+ case FNTDEF3:
+ case FNTDEF4:
+ command_pointer += 12 + ch - FNTDEF1 + 1;
+ command_pointer += readUINT8() + readUINT8();
+ break;
+
+ default:
+ return;
+ } /* end switch */
+ } /* end for */
+}