dvirenderer_prescan.cpp

来自「okular」· C++ 代码 · 共 801 行 · 第 1/2 页

CPP
801
字号
  // 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 simplified() above is wrong). If you have  // files like this, go away.  QString EPSfilename = include_command;  EPSfilename.truncate(EPSfilename.indexOf(' '));  // 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).toLower();  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") {    QString convErrorMsg;    QString oEPSfilename = EPSfilename;    //emit setStatusBarText( i18n("Converting PDF-file %1...", EPSfilename) );    EPSfilename = dviFile->convertPDFtoPS(EPSfilename, &convErrorMsg);    //emit setStatusBarText( QString::null );    if (convErrorMsg.isEmpty() != true) {/*      KMessageBox::detailedError(parentWidget,                                 i18n("<qt><strong>File conversion error</strong> KDVI was not able to convert the external "                                      "PDF file <strong>%1</strong> into PostScript. Expect missing graphics or graphic errors.</qt>", oEPSfilename),                                 convErrorMsg, i18n("PDF/PS conversion error"));*/      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.indexOf(' '));  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.indexOf(" 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  qint32 j;  for(j=0;j<cp.length();j++)    if (!cp.at(j).isNumber())      break;  quint32 sourceLineNumber = cp.left(j).toUInt();  QFileInfo fi1(dviFile->filename);  QString  sourceFileName   = QFileInfo(fi1.dir(), cp.mid(j).trimmed()).absoluteFilePath();  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, quint8 *){  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  kDebug(kvs::dvi) << "dviRenderer::prescan( ... )" << endl;#endif  if (resolutionInDPI == 0.0)    setResolution(100);  qint32 RRtmp=0, WWtmp=0, XXtmp=0, YYtmp=0, ZZtmp=0;  quint8 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.", ch - FNTNUM0);        return;      }      currinf.set_char_p = currinf.fontp->set_char_p;      continue;    }    qint32 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:      {        qint32 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:      {        quint8 *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 */}

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?