// // fontpool.cpp // // (C) 2001-2004 Stefan Kebekus // Distributed under the GPL #include #include #include #include #include #include #include #include #include #include #include #include #include #include "fontpool.h" #include "performanceMeasurement.h" #include "prefs.h" #include "TeXFont.h" //#define DEBUG_FONTPOOL // List of permissible MetaFontModes which are supported by kdvi. //const char *MFModes[] = { "cx", "ljfour", "lexmarks" }; //const char *MFModenames[] = { "Canon CX", "LaserJet 4", "Lexmark S" }; //const int MFResolutions[] = { 300, 600, 1200 }; #ifdef PERFORMANCE_MEASUREMENT QTime fontPoolTimer; bool fontPoolTimerFlag; #endif //#define DEBUG_FONTPOOL fontPool::fontPool() : progress( "fontgen", // Chapter in the documentation for help. i18n( "KDVI is currently generating bitmap fonts..." ), i18n( "Aborts the font generation. Don't do this." ), i18n( "KDVI is currently generating bitmap fonts which are needed to display your document. " "For this, KDVI uses a number of external programs, such as MetaFont. You can find " "the output of these programs later in the document info dialog." ), i18n( "KDVI is generating fonts. Please wait." ), 0 ) { #ifdef DEBUG_FONTPOOL kdDebug(4300) << "fontPool::fontPool() called" << endl; #endif setName("Font Pool"); displayResolution_in_dpi = 100.0; // A not-too-bad-default useFontHints = true; CMperDVIunit = 0; extraSearchPath = QString::null; fontList.setAutoDelete(true); #ifdef HAVE_FREETYPE // Initialize the Freetype Library if ( FT_Init_FreeType( &FreeType_library ) != 0 ) { kdError(4300) << "Cannot load the FreeType library. KDVI proceeds without FreeType support." << endl; FreeType_could_be_loaded = false; } else FreeType_could_be_loaded = true; #endif // Check if the QT library supports the alpha channel of // pixmaps. Experiments show that --depending of the configuration // of QT at compile and runtime or the availability of the XFt // extension, alpha channels are either supported, or silently // converted to 1-bit masks. QImage start(1, 1, 32); // Generate a 1x1 image, black with alpha=0x10 start.setAlphaBuffer(true); Q_UINT32 *destScanLine = (Q_UINT32 *)start.scanLine(0); *destScanLine = 0x80000000; QPixmap intermediate(start); QPixmap dest(1,1); dest.fill(Qt::white); QPainter paint( &dest ); paint.drawPixmap(0, 0, intermediate); paint.end(); start = dest.convertToImage().convertDepth(32); Q_UINT8 result = *(start.scanLine(0)) & 0xff; if ((result == 0xff) || (result == 0x00)) { #ifdef DEBUG_FONTPOOL kdDebug(4300) << "fontPool::fontPool(): QPixmap does not support the alpha channel" << endl; #endif QPixmapSupportsAlpha = false; } else { #ifdef DEBUG_FONTPOOL kdDebug(4300) << "fontPool::fontPool(): QPixmap supports the alpha channel" << endl; #endif QPixmapSupportsAlpha = true; } } fontPool::~fontPool() { #ifdef DEBUG_FONTPOOL kdDebug(4300) << "fontPool::~fontPool() called" << endl; #endif // need to manually clear the fonts _before_ freetype gets unloaded fontList.clear(); #ifdef HAVE_FREETYPE if (FreeType_could_be_loaded == true) FT_Done_FreeType( FreeType_library ); #endif } void fontPool::setParameters( bool _useFontHints ) { // Check if glyphs need to be cleared if (_useFontHints != useFontHints) { double displayResolution = displayResolution_in_dpi; TeXFontDefinition *fontp = fontList.first(); while(fontp != 0 ) { fontp->setDisplayResolution(displayResolution * fontp->enlargement); fontp=fontList.next(); } } useFontHints = _useFontHints; } TeXFontDefinition* fontPool::appendx(const QString& fontname, Q_UINT32 checksum, Q_UINT32 scale, double enlargement) { // Reuse font if possible: check if a font with that name and // natural resolution is already in the fontpool, and use that, if // possible. TeXFontDefinition *fontp = fontList.first(); while( fontp != 0 ) { if ((fontname == fontp->fontname) && ( (int)(enlargement*1000.0+0.5)) == (int)(fontp->enlargement*1000.0+0.5)) { // if font is already in the list fontp->mark_as_used(); return fontp; } fontp=fontList.next(); } // If font doesn't exist yet, we have to generate a new font. double displayResolution = displayResolution_in_dpi; fontp = new TeXFontDefinition(fontname, displayResolution*enlargement, checksum, scale, this, enlargement); if (fontp == 0) { kdError(4300) << i18n("Could not allocate memory for a font structure!") << endl; exit(0); } fontList.append(fontp); #ifdef PERFORMANCE_MEASUREMENT fontPoolTimer.start(); fontPoolTimerFlag = false; #endif // Now start kpsewhich/MetaFont, etc. if necessary return fontp; } QString fontPool::status() { #ifdef DEBUG_FONTPOOL kdDebug(4300) << "fontPool::status() called" << endl; #endif QString text; QStringList tmp; if (fontList.isEmpty()) return i18n("The fontlist is currently empty."); text.append(""); text.append( QString("") .arg(i18n("TeX Name")) .arg(i18n("Family")) .arg(i18n("Zoom")) .arg(i18n("Type")) .arg(i18n("Encoding")) .arg(i18n("Comment")) ); TeXFontDefinition *fontp = fontList.first(); while ( fontp != 0 ) { QString errMsg, encoding; if (!(fontp->flags & TeXFontDefinition::FONT_VIRTUAL)) { #ifdef HAVE_FREETYPE encoding = fontp->getFullEncodingName(); #endif if (fontp->font != 0) errMsg = fontp->font->errorMessage; else errMsg = i18n("Font file not found"); } #ifdef HAVE_FREETYPE tmp << QString ("") .arg(fontp->fontname) .arg(fontp->getFullFontName()) .arg((int)(fontp->enlargement*100 + 0.5)) .arg(fontp->getFontTypeName()) .arg(encoding) .arg(errMsg); #endif fontp=fontList.next(); } tmp.sort(); text.append(tmp.join("\n")); text.append("
%1 %2 %3 %4 %5 %6
%1 %2 %3% %4 %5 %6
"); return text; } bool fontPool::areFontsLocated() { #ifdef DEBUG_FONTPOOL kdDebug(4300) << "fontPool::areFontsLocated() called" << endl; #endif // Is there a font whose name we did not try to find out yet? TeXFontDefinition *fontp = fontList.first(); while( fontp != 0 ) { if ( !fontp->isLocated() ) return false; fontp=fontList.next(); } #ifdef DEBUG_FONTPOOL kdDebug(4300) << "... yes, all fonts are located (but not necessarily loaded)." << endl; #endif return true; // That says that all fonts are located. } void fontPool::locateFonts() { kpsewhichOutput = QString::null; // First, we try and find those fonts which exist on disk // already. If virtual fonts are found, they will add new fonts to // the list of fonts whose font files need to be located, so that we // repeat the lookup. bool vffound; do { vffound = false; locateFonts(false, false, &vffound); } while(vffound); // If still not all fonts are found, look again, this time with // on-demand generation of PK fonts enabled. if (!areFontsLocated()) locateFonts(true, false); // If still not all fonts are found, we look for TFM files as a last // resort, so that we can at least draw filled rectangles for // characters. if (!areFontsLocated()) locateFonts(false, true); // If still not all fonts are found, we give up. We mark all fonts // as 'located', so that wee won't look for them any more, and // present an error message to the user. if (!areFontsLocated()) { markFontsAsLocated(); QString details = QString("

PATH: %1

%2
").arg(getenv("PATH")).arg(kpsewhichOutput); KMessageBox::detailedError( 0, i18n("

KDVI was not able to locate all the font files " "which are necessary to display the current DVI file. " "Your document might be unreadable.

"), details, i18n("Not All Font Files Found") ); } } void fontPool::locateFonts(bool makePK, bool locateTFMonly, bool *virtualFontsFound) { // Set up the kpsewhich process. If pass == 0, look for vf-fonts and // disable automatic font generation as vf-fonts can't be // generated. If pass == 0, ennable font generation, if it was // enabled globally. emit setStatusBarText(i18n("Locating fonts...")); QStringList shellProcessCmdLine; KProcIO kpsewhichIO; // If PK fonts are generated, the kpsewhich command will re-route // the output of MetaFont into its stderr. Here we make sure this // output is intercepted and parsed. qApp->connect(&kpsewhichIO, SIGNAL(receivedStderr(KProcess *, char *, int)), this, SLOT(mf_output_receiver(KProcess *, char *, int))); kpsewhichIO.setUseShell(true); // Now generate the command line for the kpsewhich // program. Unfortunately, this can be rather long and involved... shellProcessCmdLine += "kpsewhich"; shellProcessCmdLine += QString("--dpi 1200"); shellProcessCmdLine += QString("--mode lexmarks"); // Disable automatic pk-font generation. if (makePK == true) shellProcessCmdLine += "--mktex pk"; else shellProcessCmdLine += "--no-mktex pk"; // Names of fonts that shall be located Q_UINT16 numFontsInJob = 0; TeXFontDefinition *fontp = fontList.first(); while ( fontp != 0 ) { if (!fontp->isLocated()) { numFontsInJob++; if (locateTFMonly == true) shellProcessCmdLine += KShellProcess::quote(QString("%1.tfm").arg(fontp->fontname)); else { #ifdef HAVE_FREETYPE if (FreeType_could_be_loaded == true) { const QString &filename = fontsByTeXName.findFileName(fontp->fontname); if (!filename.isEmpty()) shellProcessCmdLine += KShellProcess::quote(QString("%1").arg(filename)); } #endif shellProcessCmdLine += KShellProcess::quote(QString("%1.vf").arg(fontp->fontname)); shellProcessCmdLine += KShellProcess::quote(QString("%1.1200pk").arg(fontp->fontname)); } } fontp=fontList.next(); } if (numFontsInJob == 0) return; progress.setTotalSteps(numFontsInJob, &kpsewhichIO); // Now run... kpsewhich. In case of error, kick up a fuss. MetafontOutput = QString::null; kpsewhichOutput += "

"+shellProcessCmdLine.join(" ")+"

"; kpsewhichIO << shellProcessCmdLine; QString importanceOfKPSEWHICH = i18n("

KDVI relies on the kpsewhich program to locate font files " "on your hard disc and to generate PK fonts, if necessary.

"); if (kpsewhichIO.start(KProcess::NotifyOnExit, false) == false) { QString msg = i18n( "

The shell process for the kpsewhich program could not " "be started. Consequently, some font files could not be found, " "and your document might by unreadable. If this error is reproducable " "please report the issue to the KDVI developers using the 'Help' menu.

" ); QApplication::restoreOverrideCursor(); KMessageBox::error( 0, QString("%1%2").arg(importanceOfKPSEWHICH).arg(msg), i18n("Problem locating fonts - KDVI") ); markFontsAsLocated(); return; } // We wait here while the kpsewhich program is concurrently // running. Every second we call processEvents() to keep the GUI // updated. This is important, e.g. for the progress dialog that is // shown when PK fonts are generated by MetaFont. while(kpsewhichIO.wait(1) == false) qApp->processEvents(); progress.hide(); // Handle fatal errors. if (!kpsewhichIO.normalExit()) { KMessageBox::sorry( 0, "

The font generation was aborted. As a result, " "some font files could not be located, and your document might be unreadable.

", i18n("Font generation aborted - KDVI") ); // This makes sure the we don't try to run kpsewhich again if (makePK == false) markFontsAsLocated(); } else if (kpsewhichIO.exitStatus() == 127) { // An exit status of 127 means that the kpsewhich executable // could not be found. We give extra explanation then. QApplication::restoreOverrideCursor(); QString msg = i18n( "

There were problems running kpsewhich. As a result, " "some font files could not be located, and your document might be unreadable.

" "

Possible reason: The kpsewhich program is perhaps not installed on your system, or it " "cannot be found in the current search path.

" "

What you can do: The kpsewhich program is normally contained in distributions of the TeX " "typesetting system. If TeX is not installed on your system, you could install the TeTeX distribution (www.tetex.org). " "If you are sure that TeX is installed, please try to use the kpsewhich program from the command line to check if it " "really works.

"); QString details = QString("

PATH: %1

%2
").arg(getenv("PATH")).arg(kpsewhichOutput); KMessageBox::detailedError( 0, QString("%1%2").arg(importanceOfKPSEWHICH).arg(msg), details, i18n("Problem locating fonts - KDVI") ); // This makes sure the we don't try to run kpsewhich again markFontsAsLocated(); return; } // Create a list with all filenames found by the kpsewhich program. QStringList fileNameList; QString line; while(kpsewhichIO.readln(line) >= 0) fileNameList += line; // Now associate the file names found with the fonts fontp=fontList.first(); while ( fontp != 0 ) { if (fontp->filename.isEmpty() == true) { QStringList matchingFiles; #ifdef HAVE_FREETYPE const QString &fn = fontsByTeXName.findFileName(fontp->fontname); if (!fn.isEmpty()) matchingFiles = fileNameList.grep(fn); #endif if (matchingFiles.isEmpty() == true) matchingFiles += fileNameList.grep(fontp->fontname+"."); if (matchingFiles.isEmpty() != true) { #ifdef DEBUG_FONTPOOL kdDebug(4300) << "Associated " << fontp->fontname << " to " << matchingFiles.first() << endl; #endif QString fname = matchingFiles.first(); fontp->fontNameReceiver(fname); fontp->flags |= TeXFontDefinition::FONT_KPSE_NAME; if (fname.endsWith(".vf")) { if (virtualFontsFound != 0) *virtualFontsFound = true; // Constructing a virtual font will most likely insert other // fonts into the fontList. After that, fontList.next() will // no longer work. It is therefore safer to start over. fontp=fontList.first(); continue; } } } // of if (fontp->filename.isEmpty() == true) fontp = fontList.next(); } } void fontPool::setCMperDVIunit( double _CMperDVI ) { #ifdef DEBUG_FONTPOOL kdDebug(4300) << "fontPool::setCMperDVIunit( " << _CMperDVI << " )" << endl; #endif if (CMperDVIunit == _CMperDVI) return; CMperDVIunit = _CMperDVI; TeXFontDefinition *fontp = fontList.first(); while(fontp != 0 ) { fontp->setDisplayResolution(displayResolution_in_dpi * fontp->enlargement); fontp=fontList.next(); } } void fontPool::setDisplayResolution( double _displayResolution_in_dpi ) { #ifdef DEBUG_FONTPOOL kdDebug(4300) << "fontPool::setDisplayResolution( displayResolution_in_dpi=" << _displayResolution_in_dpi << " ) called" << endl; #endif // Ignore minute changes by less than 2 DPI. The difference would // hardly be visible anyway. That saves a lot of re-painting, // e.g. when the user resizes the window, and a flickery mouse // changes the window size by 1 pixel all the time. if ( fabs(displayResolution_in_dpi - _displayResolution_in_dpi) <= 2.0 ) { #ifdef DEBUG_FONTPOOL kdDebug(4300) << "fontPool::setDisplayResolution(...): resolution wasn't changed. Aborting." << endl; #endif return; } displayResolution_in_dpi = _displayResolution_in_dpi; double displayResolution = displayResolution_in_dpi; TeXFontDefinition *fontp = fontList.first(); while(fontp != 0 ) { fontp->setDisplayResolution(displayResolution * fontp->enlargement); fontp=fontList.next(); } // Do something that causes re-rendering of the dvi-window /*@@@@ emit fonts_have_been_loaded(this); */ } void fontPool::markFontsAsLocated() { TeXFontDefinition *fontp=fontList.first(); while ( fontp != 0 ) { fontp->markAsLocated(); fontp = fontList.next(); } } void fontPool::mark_fonts_as_unused() { #ifdef DEBUG_FONTPOOL kdDebug(4300) << "fontPool::mark_fonts_as_unused() called" << endl; #endif TeXFontDefinition *fontp = fontList.first(); while ( fontp != 0 ) { fontp->flags &= ~TeXFontDefinition::FONT_IN_USE; fontp=fontList.next(); } } void fontPool::release_fonts() { #ifdef DEBUG_FONTPOOL kdDebug(4300) << "Release_fonts" << endl; #endif TeXFontDefinition *fontp = fontList.first(); while(fontp != 0) { if ((fontp->flags & TeXFontDefinition::FONT_IN_USE) != TeXFontDefinition::FONT_IN_USE) { fontList.removeRef(fontp); fontp = fontList.first(); } else fontp = fontList.next(); } } void fontPool::mf_output_receiver(KProcess *, char *buffer, int buflen) { // Paranoia. if (buflen < 0) return; QString op = QString::fromLocal8Bit(buffer, buflen); kpsewhichOutput.append(op); MetafontOutput.append(op); // We'd like to print only full lines of text. int numleft; bool show_prog = false; while( (numleft = MetafontOutput.find('\n')) != -1) { QString line = MetafontOutput.left(numleft+1); #ifdef DEBUG_FONTPOOL kdDebug(4300) << "MF OUTPUT RECEIVED: " << line; #endif // Search for a line which marks the beginning of a MetaFont run // and show the progress dialog at the end of this method. if (line.find("kpathsea:") == 0) show_prog = true; // If the Output of the kpsewhich program contains a line starting // with "kpathsea:", this means that a new MetaFont-run has been // started. We filter these lines out and update the display // accordingly. int startlineindex = line.find("kpathsea:"); if (startlineindex != -1) { int endstartline = line.find("\n",startlineindex); QString startLine = line.mid(startlineindex,endstartline-startlineindex); // The last word in the startline is the name of the font which we // are generating. The second-to-last word is the resolution in // dots per inch. Display this info in the text label below the // progress bar. int lastblank = startLine.findRev(' '); QString fontName = startLine.mid(lastblank+1); int secondblank = startLine.findRev(' ',lastblank-1); QString dpi = startLine.mid(secondblank+1,lastblank-secondblank-1); progress.show(); progress.increaseNumSteps( i18n("Currently generating %1 at %2 dpi").arg(fontName).arg(dpi) ); } MetafontOutput = MetafontOutput.remove(0,numleft+1); } } #include "fontpool.moc"