/*************************************************************************** * Copyright (C) 2005 by Stefan Kebekus * * kebekus@kde.org * * * * Copyright (C) 2005 by Wilfried Huss * * Wilfried.Huss@gmx.at * * * * This program is free software; you can redistribute it and/or modify * * it under the terms of the GNU General Public License as published by * * the Free Software Foundation; either version 2 of the License, or * * (at your option) any later version. * * * * This program is distributed in the hope that it will be useful, * * but WITHOUT ANY WARRANTY; without even the implied warranty of * * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * * GNU General Public License for more details. * * * * You should have received a copy of the GNU General Public License * * along with this program; if not, write to the * * Free Software Foundation, Inc., * * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * ***************************************************************************/ #include <kmessagebox.h> #include <kdebug.h> #include <klocale.h> #include <tqfileinfo.h> #include <tqimage.h> #include <tqpainter.h> #include <kapp.h> #include "GBitmap.h" #include "BSByteStream.h" #include "IFFByteStream.h" #include "prefs.h" #include "documentWidget.h" #include "djvurenderer.h" #include "djvumultipage.h" #include "hyperlink.h" #include "renderedDocumentPagePixmap.h" #include "textBox.h" //#define KF_DEBUG inline GUTF8String GStringFromTQString(const TQString& x) { GUTF8String retval=(const char*)x.utf8(); return retval; } inline TQString TQStringFromGString(const GUTF8String& x) { TQString retval=TQString::fromUtf8((const char*)x); return retval; } DjVuRenderer::DjVuRenderer(TQWidget* par) : DocumentRenderer(par) { #ifdef KF_DEBUG kdError() << "DjVuRenderer( parent=" << par << " )" << endl; #endif PPMstream = ByteStream::create(); } DjVuRenderer::~DjVuRenderer() { #ifdef KF_DEBUG kdDebug() << "~DjVuRenderer" << endl; #endif // Wait for all access to this documentRenderer to finish TQMutexLocker locker( &mutex ); } void DjVuRenderer::drawPage(double resolution, RenderedDocumentPage* page) { #ifdef KF_DEBUG kdDebug() << "DjVuRenderer::drawPage(documentPage*) called, page number " << page->getPageNumber() << endl; #endif // Paranoid safety checks if (page == 0) { kdError() << "DjVuRenderer::drawPage(documentPage*) called with argument == 0" << endl; return; } if (page->getPageNumber() == 0) { kdError() << "DjVuRenderer::drawPage(documentPage*) called for a documentPage with page number 0" << endl; return; } // Wait for all access to this documentRenderer to finish TQMutexLocker locker( &mutex ); // more paranoid safety checks if (page->getPageNumber() > numPages) { kdError() << "DjVuRenderer::drawPage(documentPage*) called for a documentPage with page number " << page->getPageNumber() << " but the current fax file has only " << numPages << " pages." << endl; return; } int pageNumber = page->getPageNumber() - 1; GP<DjVuImage> djvuPage = document->get_page(pageNumber, true); if (!djvuPage->wait_for_complete_decode()) { kdDebug() << "decoding failed." << endl; return; } if (!pageSizes[pageNumber].isValid()) { int djvuResolution = djvuPage->get_dpi(); int djvuPageWidth = djvuPage->get_width(); int djvuPageHeight = djvuPage->get_height(); Length w, h; w.setLength_in_inch(djvuPageWidth / (double)djvuResolution); h.setLength_in_inch(djvuPageHeight / (double)djvuResolution); pageSizes[pageNumber].setPageSize(w, h); SimplePageSize ps = sizeOfPage(page->getPageNumber()); // If we are not printing we need to resize the pixmap. RenderedDocumentPagePixmap* pagePixmap = dynamic_cast<RenderedDocumentPagePixmap*>(page); if (pagePixmap) pagePixmap->resize(ps.sizeInPixel(resolution)); } //kdDebug() << "render page " << pageNumber + 1 << " at size (" << pageWidth << ", " << pageHeight << ")" << endl; int pageHeight = page->height(); int pageWidth = page->width(); GRect pageRect(0, 0, pageWidth, pageHeight); GP<GPixmap> djvuPixmap; if (Prefs::renderMode() == Prefs::EnumRenderMode::Color) djvuPixmap = djvuPage->get_pixmap(pageRect, pageRect); else if (Prefs::renderMode() == Prefs::EnumRenderMode::Foreground) djvuPixmap = djvuPage->get_fg_pixmap(pageRect, pageRect); else if (Prefs::renderMode() == Prefs::EnumRenderMode::Background) djvuPixmap = djvuPage->get_bg_pixmap(pageRect, pageRect); TQPainter* foreGroundPaint = page->getPainter(); if (foreGroundPaint != 0) { if(djvuPixmap && Prefs::renderMode() != Prefs::EnumRenderMode::BlackAndWhite) { PPMstream->seek(0); djvuPixmap->save_ppm(*PPMstream); long pixmapsize = PPMstream->tell(); PPMstream->seek(0); uchar* buf = new uchar[pixmapsize]; long bytesRead = PPMstream->readall(buf, pixmapsize); bool ok = pixmap.loadFromData(buf, bytesRead, "PPM"); if (!ok) { kdError() << "loading failed" << endl; //draw an empty page foreGroundPaint->fillRect(0, 0, pageWidth, pageHeight, TQt::white); } foreGroundPaint->drawPixmap(0, 0, pixmap); delete[] buf; /* for (int i = 0; i < pageHeight; i++) { GPixel* pixmapRow = (*djvuPixmap)[i]; for (int j = 0; j < pageWidth; j++) { GPixel pixel = pixmapRow[j]; foreGroundPaint->setPen(TQColor(pixel.r, pixel.g, pixel.b)); foreGroundPaint->drawPoint(j, pageHeight - i - 1); } }*/ } else { GP<GBitmap> djvuBitmap = djvuPage->get_bitmap(pageRect, pageRect); if(djvuBitmap) { PPMstream->seek(0); if(djvuBitmap->get_grays() == 2) djvuBitmap->save_pbm(*PPMstream); else djvuBitmap->save_pgm(*PPMstream); long pixmapsize = PPMstream->tell(); PPMstream->seek(0); uchar* buf = new uchar[pixmapsize]; long bytesRead = PPMstream->readall(buf, pixmapsize); bool ok = pixmap.loadFromData(buf, bytesRead, "PPM"); if (!ok) { kdError() << "loading failed" << endl; //draw an empty page foreGroundPaint->fillRect(0, 0, pageWidth, pageHeight, TQt::white); } foreGroundPaint->drawPixmap(0, 0, pixmap); delete[] buf; /* for (int i = 0; i < pageHeight; i++) { unsigned char* bitmapRow = (*djvuBitmap)[i]; for (int j = 0; j < pageWidth; j++) { unsigned char pixel = 255-bitmapRow[j]; foreGroundPaint->setPen(TQColor(pixel, pixel, pixel)); foreGroundPaint->drawPoint(j, pageHeight - i - 1); } }*/ } else { //draw an empty page foreGroundPaint->fillRect(0, 0, pageWidth, pageHeight, TQt::white); } } //kdDebug() << "rendering page " << pageNumber + 1 << " at size (" << pageWidth << ", " << pageHeight << ") finished." << endl; page->returnPainter(foreGroundPaint); } GP<DjVuTXT> pageText = getText(pageNumber); if (pageText) { TQSize djvuPageSize(djvuPage->get_width(), djvuPage->get_real_height()); fillInText(page, pageText, pageText->page_zone, djvuPageSize); //kdDebug() << "Text of page " << pageNumber << endl; //kdDebug() << (const char*)pageText->textUTF8 << endl; } getAnnotations(page, djvuPage); page->isEmpty = false; } bool DjVuRenderer::setFile(const TQString &fname, const KURL &) { #ifdef KF_DEBUG kdDebug() << "DjVuRenderer::setFile(" << fname << ") called" << endl; #endif // Wait for all access to this documentRenderer to finish TQMutexLocker locker( &mutex ); // If fname is the empty string, then this means: "close". if (fname.isEmpty()) { kdDebug() << "DjVuRenderer::setFile( ... ) called with empty filename. Closing the file." << endl; return true; } // Paranoid saftey checks: make sure the file actually exists, and // that it is a file, not a directory. Otherwise, show an error // message and exit.. TQFileInfo fi(fname); TQString filename = fi.absFilePath(); if (!fi.exists() || fi.isDir()) { KMessageBox::error( parentWidget, i18n("<qt><strong>File error.</strong> The specified file '%1' does not exist.</qt>").tqarg(filename), i18n("File Error")); // the return value 'false' indicates that this operation was not successful. return false; } // Clear previously loaded document clear(); // Now we assume that the file is fine and load the file. G_TRY { document = DjVuDocEditor::create_wait(GURL::Filename::UTF8(GStringFromTQString(filename))); } G_CATCH(ex) { ; } G_ENDCATCH; // If the above assumption was false. if (!document) { KMessageBox::error( parentWidget, i18n("<qt><strong>File error.</strong> The specified file '%1' could not be loaded.</qt>").tqarg(filename), i18n("File Error")); clear(); kdDebug(1223) << "Loading of document failed." << endl; return false; } bool r = initializeDocument(); // the return value 'true' indicates that this operation was successful. return r; } void DjVuRenderer::getAnnotations(RenderedDocumentPage* page, GP<DjVuImage> djvuPage) { GP<ByteStream> annotations = djvuPage->get_anno(); if (!(annotations && annotations->size())) return; GP<DjVuANT> ant = DjVuANT::create(); GP<IFFByteStream> iff = IFFByteStream::create(annotations); GUTF8String chkid; while (iff->get_chunk(chkid)) { if (chkid == "ANTa") { ant->merge(*iff->get_bytestream()); } else if (chkid == "ANTz") { GP<ByteStream> bsiff = BSByteStream::create(iff->get_bytestream()); ant->merge(*bsiff); } iff->close_chunk(); } if (!ant->is_empty()) { // Scaling factors for the coordinate conversion. // TODO: Refractor this into a function shared with fillInText. int pageWidth = page->width(); int pageHeight = page->height(); double scaleX = pageWidth / (double)djvuPage->get_width(); double scaleY = pageHeight / (double)djvuPage->get_height(); GPList<GMapArea> map = ant->map_areas; for (GPosition pos = map; pos; ++pos) { // Currently we only support rectangular links if (!map[pos]->get_tqshape_type() == GMapArea::RECT) continue; GRect rect = map[pos]->get_bound_rect(); TQRect hyperlinkRect((int)(rect.xmin*scaleX+0.5), (int)((djvuPage->get_height()-rect.ymax)*scaleY+0.5), (int)(rect.width()*scaleX +0.5), (int)(rect.height()*scaleY+0.5)); TQString url((const char*)map[pos]->url); TQString target((const char*)map[pos]->target); TQString comment((const char*)map[pos]->comment); // Create an anchor for this link. if (!anchorList.contains(url)) { // For now we only accept links to pages in the same document. if(url[0] == '#' && target == "_self") { bool conversionOk; PageNumber targetPage = url.remove('#').toInt(&conversionOk); if (conversionOk) anchorList[url] = Anchor(targetPage, Length()); } } Hyperlink hyperlink(hyperlinkRect.bottom(), hyperlinkRect, url); page->hyperLinkList.push_back(hyperlink); } } } bool DjVuRenderer::initializeDocument() { if (document == 0) return false; if (!document->wait_for_complete_init()) { kdDebug() << "Document Initialization failed." << endl; return false; } // Set the number of pages page sizes numPages = document->get_pages_num(); pageSizes.resize(numPages); Length w,h; // Set the page sizes in the pageSizes array. Give feedback for // very long documents if (numPages > 100) emit setStatusBarText(i18n("Loading file. Computing page sizes...")); for(TQ_UINT16 i=0; i<numPages; i++) { // Keep the GUI updated if (i%100 == 0) kapp->processEvents(); GP<DjVuFile> djvuFile = document->get_djvu_file(i); int resolution; int pageWidth; int pageHeight; bool ok = getPageInfo(djvuFile, pageWidth, pageHeight, resolution); if (!ok) kdError() << "Decoding info of page " << i << " failed." << endl; else { w.setLength_in_inch(pageWidth / (double)resolution); h.setLength_in_inch(pageHeight / (double)resolution); pageSizes[i].setPageSize(w, h); } } emit setStatusBarText(TQString()); // We will also generate a list of hyperlink-anchors in the document. // So declare the existing lists empty. anchorList.clear(); return true; } GP<DjVuTXT> DjVuRenderer::getText(PageNumber pageNumber) { GUTF8String chkid; const GP<DjVuFile> file = document->get_djvu_file(pageNumber); const GP<ByteStream> bs(file->get_text()); if (bs) { long int i=0; const GP<IFFByteStream> iff(IFFByteStream::create(bs)); while (iff->get_chunk(chkid)) { i++; if (chkid == GUTF8String("TXTa")) { GP<DjVuTXT> txt = DjVuTXT::create(); txt->decode(iff->get_bytestream()); return txt; } else if (chkid == GUTF8String("TXTz")) { GP<DjVuTXT> txt = DjVuTXT::create(); GP<ByteStream> bsiff=BSByteStream::create(iff->get_bytestream()); txt->decode(bsiff); return txt; } iff->close_chunk(); } } return 0; } void DjVuRenderer::fillInText(RenderedDocumentPage* page, const GP<DjVuTXT>& text, DjVuTXT::Zone& zone, TQSize& djvuPageSize) { if (zone.tqchildren.isempty()) { int pageWidth = page->width(); int pageHeight = page->height(); double scaleX = pageWidth / (double)djvuPageSize.width(); double scaleY = pageHeight / (double)djvuPageSize.height(); TQString zoneString = TQStringFromGString(text->textUTF8.substr(zone.text_start, zone.text_length)); //kdDebug() << "zone text: " << zoneString << endl; TQRect textRect((int)(zone.rect.xmin*scaleX+0.5), (int)((djvuPageSize.height()-zone.rect.ymax)*scaleY+0.5), (int)(zone.rect.width()*scaleX+0.5), (int)(zone.rect.height()*scaleY+0.5)); //kdDebug() << "zone rect: " << textRect.x() << ", " << textRect.y() << ", " << textRect.width() << ", " << textRect.height() << endl; TextBox textBox(textRect, zoneString); page->textBoxList.push_back(textBox); } else { for (GPosition pos=zone.tqchildren; pos; ++pos) { fillInText(page, text, zone.tqchildren[pos], djvuPageSize); } } } bool DjVuRenderer::getPageInfo(GP<DjVuFile> file, int& width, int& height, int& dpi) { if (!file || !file->is_all_data_present()) return false; const GP<ByteStream> pbs(file->get_djvu_bytestream(false, false)); const GP<IFFByteStream> iff(IFFByteStream::create(pbs)); GUTF8String chkid; if (iff->get_chunk(chkid)) { if (chkid == "FORM:DJVU") { while (iff->get_chunk(chkid) && chkid!="INFO") iff->close_chunk(); if (chkid == "INFO") { GP<ByteStream> gbs = iff->get_bytestream(); GP<DjVuInfo> info=DjVuInfo::create(); info->decode(*gbs); int rot = ((360-GRect::findangle(info->orientation))/90)%4; width = (rot&1) ? info->height : info->width; height = (rot&1) ? info->width : info->height; dpi = info->dpi; return true; } } else if (chkid == "FORM:BM44" || chkid == "FORM:PM44") { while (iff->get_chunk(chkid) && chkid!="BM44" && chkid!="PM44") iff->close_chunk(); if (chkid=="BM44" || chkid=="PM44") { GP<ByteStream> gbs = iff->get_bytestream(); if (gbs->read8() == 0) { gbs->read8(); gbs->read8(); unsigned char xhi = gbs->read8(); unsigned char xlo = gbs->read8(); unsigned char yhi = gbs->read8(); unsigned char ylo = gbs->read8(); width = (xhi<<8)+xlo; height = (yhi<<8)+ylo; dpi = 100; return true; } } } } return false; } void DjVuRenderer::getText(RenderedDocumentPage* page) { TQMutexLocker locker( &mutex ); int pageNumber = page->getPageNumber() - 1; GP<DjVuTXT> pageText = getText(pageNumber); if (pageText) { GP<DjVuFile> djvuFile = document->get_djvu_file(pageNumber); int resolution; int pageWidth; int pageHeight; bool ok = getPageInfo(djvuFile, pageWidth, pageHeight, resolution); if (ok) { TQSize djvuPageSize(pageWidth, pageHeight); fillInText(page, pageText, pageText->page_zone, djvuPageSize); } } } bool DjVuRenderer::convertToPSFile( DjVuToPS &converter, TQString filename, TQValueList<int> &pageList ) { if (document == 0) { kdError(1223) << "DjVuRenderer::convertToPSFile(..) called when document was 0" << endl; return false; } TQMutexLocker locker( &mutex ); // Set up progress dialog KProgressDialog *pdialog = new KProgressDialog(parentWidget, "Printing-ProgressDialog", i18n("Printing..."), i18n("Preparing pages for printing..."), true); pdialog->setButtonText(i18n("Abort")); pdialog->showCancelButton(true); pdialog->progressBar()->setTotalSteps(pageList.size()); pdialog->progressBar()->setFormat(TQString()); // Open output file GURL outname = GURL::Filename::UTF8(GStringFromTQString(filename)); GP<ByteStream> obs = ByteStream::create(outname, "w"); TQString pagename; TQValueList<int>::ConstIterator it = pageList.begin(); while (true) { pagename += TQString::number(*it); ++it; if (it == pageList.end()) break; pagename += ","; } GUTF8String pages = GStringFromTQString(pagename); converter.set_info_cb(printerInfoCallBack, (void*)pdialog); bool iscancelled = false; G_TRY { converter.print(*obs, (DjVuDocument *)document, pages ); } G_CATCH(ex) { iscancelled = true; } G_ENDCATCH; delete pdialog; // This is to keep the GUI updated. kapp->processEvents(); obs->flush(); return !iscancelled; } void DjVuRenderer::deletePages(TQ_UINT16 from, TQ_UINT16 to) { // Paranoia security checks if (document == 0) { kdError(1223) << "DjVuRenderer::deletePages(...) called when no document was loaded" << endl; return; } if ((from > to) || (from == 0) || (from > totalPages()) || (to > totalPages())) { kdError(1223) << "DjVuRenderer::deletePages(...) called with invalid arguments" << endl; return; } TQMutexLocker locker( &mutex ); KProgressDialog *pdialog = 0; if (to-from > 9) { pdialog = new KProgressDialog(parentWidget, "Printing-ProgressDialog", i18n("Deleting pages..."), i18n("Please wait while pages are removed..."), true); pdialog->showCancelButton(false); pdialog->progressBar()->setTotalSteps(to-from+1); pdialog->progressBar()->setFormat(TQString()); pdialog->show(); kapp->processEvents(); } // set the document pointer temporarily to 0, so that no-one tries // to render a page while we are deleting pages GP<DjVuDocEditor> document_new = document; document = 0; // Delete pages if (pdialog == 0) { GList<int> pageList; for(TQ_UINT16 i=from; i<= to; i++) pageList.append(i-1); document_new->remove_pages(pageList); } else { for(TQ_UINT16 i=from; i<=to; i++) { document_new->remove_page(from-1); pdialog->progressBar()->setProgress(i-from); pdialog->progressBar()->setFormat(i18n("deleting page %1").tqarg(i)); kapp->processEvents(); } delete pdialog; } _isModified = true; document = document_new; initializeDocument(); } bool DjVuRenderer::save(const TQString &filename) { if (document == 0) { kdError() << "DjVuRenderer::save(..) called when document==0" << endl; return false; } TQMutexLocker locker( &mutex ); G_TRY { document->save_as(GURL::Filename::UTF8(GStringFromTQString(filename)), true); } G_CATCH(ex) { return false; } G_ENDCATCH; document->save_as(GURL::Filename::UTF8(filename.ascii()), true); if (TQFile::exists(filename) == false) return false; _isModified = false; return true; } void DjVuRenderer::printerInfoCallBack(int page_num, int page_count, int, DjVuToPS::Stage, void *pd) { if (pd == 0) return; // Update the progress dialog. KProgressDialog *pdialog = (KProgressDialog *)pd; pdialog->progressBar()->setProgress(page_count); pdialog->progressBar()->setFormat(i18n("processing page %1").tqarg(page_num+1)); pdialog->show(); if (pdialog->wasCancelled()) G_THROW("STOP"); // This is to keep the GUI updated. kapp->processEvents(); } #include "djvurenderer.moc"