/*************************************************************************** * Copyright (C) 2003-2004 by Christophe Devriese * * * * Copyright (C) 2003 by Andy Goossens * * Copyright (C) 2003 by Scott Wheeler * * Copyright (C) 2003 by Ingo Klöcker * * Copyright (C) 2003 by Will Andrews * * Copyright (C) 2004 by Dominique Devriese * * Copyright (C) 2004 by Waldo Bastian * * Copyright (C) 2004 by Albert Astals Cid * * * * 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. * ***************************************************************************/ #ifdef __GNUC__ #pragma implementation #endif #include #include #include #include "gp_outputdev.h" #include "generator_pdf.h" #include "core/document.h" // for DocumentViewport #include "core/page.h" #include "core/link.h" #include "xpdf/Link.h" #include "xpdf/GfxState.h" #include "xpdf/TextOutputDev.h" #include "splash/SplashBitmap.h" //NOTE: XPDF/Splash *implementation dependant* code is marked with '###' /** KPDFOutputDev implementation **/ KPDFOutputDev::KPDFOutputDev( SplashColor paperColor ) : SplashOutputDev( splashModeRGB8, false, paperColor ), m_doc( 0 ), m_pixmap( 0 ), m_image( 0 ), m_text( 0 ) { } KPDFOutputDev::~KPDFOutputDev() { clear(); } void KPDFOutputDev::initDevice( PDFDoc * pdfDoc ) { m_doc = pdfDoc; startDoc( pdfDoc->getXRef() ); } void KPDFOutputDev::setParams( int width, int height, bool genT, bool genL, bool genI, bool safe ) { clear(); m_pixmapWidth = width; m_pixmapHeight = height; m_qtThreadSafety = safe; m_generateText = genT; m_generateLinks = genL; m_generateImages = genI; if ( m_generateText ) m_text = new TextPage( gFalse ); } QPixmap * KPDFOutputDev::takePixmap() { QPixmap * pix = m_pixmap; m_pixmap = 0; return pix; } QImage * KPDFOutputDev::takeImage() { QImage * img = m_image; m_image = 0; return img; } TextPage * KPDFOutputDev::takeTextPage() { TextPage * text = m_text; m_text = 0; return text; } QValueList< ObjectRect * > KPDFOutputDev::takeObjectRects() { if ( m_rects.isEmpty() ) return m_rects; QValueList< ObjectRect * > rectsCopy( m_rects ); m_rects.clear(); return rectsCopy; } //BEGIN - OutputDev hooked calls void KPDFOutputDev::startPage( int pageNum, GfxState *state ) { if ( m_generateText ) m_text->startPage( state ); SplashOutputDev::startPage( pageNum, state ); } void KPDFOutputDev::endPage() { SplashOutputDev::endPage(); if ( m_generateText ) { m_text->endPage(); m_text->coalesce( gTrue ); } int bh = getBitmap()->getHeight(), bw = getBitmap()->getWidth(); SplashColorPtr dataPtr = getBitmap()->getDataPtr(); // construct a qimage SHARING the raw bitmap data in memory QImage * img = new QImage( (uchar*)dataPtr.rgb8, bw, bh, 32, 0, 0, QImage::IgnoreEndian ); // use the QImage or convert it immediately to QPixmap for better // handling and memory unloading if ( m_qtThreadSafety ) { delete m_image; // it may happen (in fact it doesn't) that we need a rescaling if ( bw != m_pixmapWidth && bh != m_pixmapHeight ) m_image = new QImage( img->smoothScale( m_pixmapWidth, m_pixmapHeight ) ); else // dereference image from the xpdf memory m_image = new QImage( img->copy() ); } else { delete m_pixmap; // it may happen (in fact it doesn't) that we need a rescaling if ( bw != m_pixmapWidth || bh != m_pixmapHeight ) m_pixmap = new QPixmap( img->smoothScale( m_pixmapWidth, m_pixmapHeight ) ); else m_pixmap = new QPixmap( *img ); } // destroy the shared descriptor and (###) unload underlying xpdf bitmap delete img; SplashOutputDev::startPage( 0, NULL ); } void KPDFOutputDev::drawLink( Link * link, Catalog * catalog ) { if ( !link->isOk() ) return; if ( m_generateLinks ) { // create the link descriptor KPDFLink * l = generateLink( link->getAction() ); if ( l ) { // create the page rect representing the link double x1, y1, x2, y2; link->getRect( &x1, &y1, &x2, &y2 ); int left, top, right, bottom; cvtUserToDev( x1, y1, &left, &top ); cvtUserToDev( x2, y2, &right, &bottom ); double nl = (double)left / (double)m_pixmapWidth, nt = (double)top / (double)m_pixmapHeight, nr = (double)right / (double)m_pixmapWidth, nb = (double)bottom / (double)m_pixmapHeight; // create the rect using normalized coords and attach the KPDFLink to it ObjectRect * rect = new ObjectRect( nl, nt, nr, nb, ObjectRect::Link, l ); // add the ObjectRect to the vector container m_rects.push_back( rect ); } } SplashOutputDev::drawLink( link, catalog ); } void KPDFOutputDev::updateFont( GfxState *state ) { if ( m_generateText ) m_text->updateFont( state ); SplashOutputDev::updateFont( state ); } void KPDFOutputDev::drawChar( GfxState *state, double x, double y, double dx, double dy, double originX, double originY, CharCode code, Unicode *u, int uLen ) { if ( m_generateText ) m_text->addChar( state, x, y, dx, dy, code, u, uLen ); SplashOutputDev::drawChar( state, x, y, dx, dy, originX, originY, code, u, uLen ); } GBool KPDFOutputDev::beginType3Char( GfxState *state, double x, double y, double dx, double dy, CharCode code, Unicode *u, int uLen ) { if ( m_generateText ) m_text->addChar( state, x, y, dx, dy, code, u, uLen ); return SplashOutputDev::beginType3Char( state, x, y, dx, dy, code, u, uLen ); } void KPDFOutputDev::drawImage( GfxState *state, Object *ref, Stream *str, int _width, int _height, GfxImageColorMap *colorMap, int *maskColors, GBool inlineImg ) { if ( m_generateImages ) { // find out image rect from the Coord Transform Matrix double * ctm = state->getCTM(); int left = (int)ctm[4], top = (int)ctm[5], width = (int)ctm[0], height = (int)ctm[3]; // normalize width if ( width < 0 ) { width = -width; left -= width; } // normalize height if ( height < 0 ) { height = -height; top -= height; } if ( width > 10 && height > 10 ) { // build a descriptor for the image rect double nl = (double)left / (double)m_pixmapWidth, nt = (double)top / (double)m_pixmapHeight, nr = (double)(left + width) / (double)m_pixmapWidth, nb = (double)(top + height) / (double)m_pixmapHeight; // create the rect using normalized coords and set it of KPDFImage type ObjectRect * rect = new ObjectRect( nl, nt, nr, nb, ObjectRect::Image, 0 ); // add the ObjectRect to the vector container m_rects.push_back( rect ); } } SplashOutputDev::drawImage( state, ref, str, _width, _height, colorMap, maskColors, inlineImg ); } //END - OutputDev hooked calls //BEGIN - private helpers void KPDFOutputDev::clear() { // delete rects if ( m_rects.count() ) { QValueList< ObjectRect * >::iterator it = m_rects.begin(), end = m_rects.end(); for ( ; it != end; ++it ) delete *it; m_rects.clear(); } // delete pixmap if ( m_pixmap ) { delete m_pixmap; m_pixmap = 0; } // delete image if ( m_image ) { delete m_image; m_image = 0; } // delete text if ( m_text ) { delete m_text; m_text = 0; } } KPDFLink * KPDFOutputDev::generateLink( LinkAction * a ) // note: this function is called when processing a page, when the MUTEX is already LOCKED { KPDFLink * link = NULL; if ( a ) switch ( a->getKind() ) { case actionGoTo: { LinkGoTo * g = (LinkGoTo *) a; // create link: no ext file, namedDest, object pointer link = new KPDFLinkGoto( QString::null, decodeViewport( g->getNamedDest(), g->getDest() ) ); } break; case actionGoToR: { LinkGoToR * g = (LinkGoToR *) a; // copy link file const char * fileName = g->getFileName()->getCString(); // create link: fileName, namedDest, object pointer link = new KPDFLinkGoto( (QString)fileName, decodeViewport( g->getNamedDest(), g->getDest() ) ); } break; case actionLaunch: { LinkLaunch * e = (LinkLaunch *)a; GString * p = e->getParams(); link = new KPDFLinkExecute( e->getFileName()->getCString(), p ? p->getCString() : 0 ); } break; case actionNamed: { const char * name = ((LinkNamed *)a)->getName()->getCString(); if ( !strcmp( name, "NextPage" ) ) link = new KPDFLinkAction( KPDFLinkAction::PageNext ); else if ( !strcmp( name, "PrevPage" ) ) link = new KPDFLinkAction( KPDFLinkAction::PagePrev ); else if ( !strcmp( name, "FirstPage" ) ) link = new KPDFLinkAction( KPDFLinkAction::PageFirst ); else if ( !strcmp( name, "LastPage" ) ) link = new KPDFLinkAction( KPDFLinkAction::PageLast ); else if ( !strcmp( name, "GoBack" ) ) link = new KPDFLinkAction( KPDFLinkAction::HistoryBack ); else if ( !strcmp( name, "GoForward" ) ) link = new KPDFLinkAction( KPDFLinkAction::HistoryForward ); else if ( !strcmp( name, "Quit" ) ) link = new KPDFLinkAction( KPDFLinkAction::Quit ); else if ( !strcmp( name, "GoToPage" ) ) link = new KPDFLinkAction( KPDFLinkAction::GoToPage ); else if ( !strcmp( name, "Find" ) ) link = new KPDFLinkAction( KPDFLinkAction::Find ); else if ( !strcmp( name, "FullScreen" ) ) link = new KPDFLinkAction( KPDFLinkAction::Presentation ); else if ( !strcmp( name, "Close" ) ) link = new KPDFLinkAction( KPDFLinkAction::EndPresentation ); else kdDebug() << "Unknown named action: '" << name << "'" << endl; } break; case actionURI: link = new KPDFLinkBrowse( ((LinkURI *)a)->getURI()->getCString() ); break; case actionMovie: /* { TODO this (Movie link) m_type = Movie; LinkMovie * m = (LinkMovie *) a; // copy Movie parameters (2 IDs and a const char *) Ref * r = m->getAnnotRef(); m_refNum = r->num; m_refGen = r->gen; copyString( m_uri, m->getTitle()->getCString() ); } */ break; case actionUnknown: kdDebug() << "Unknown link." << endl; break; } // link may be zero at that point return link; } DocumentViewport KPDFOutputDev::decodeViewport( GString * namedDest, LinkDest * dest ) // note: this function is called when processing a page, when the MUTEX is already LOCKED { DocumentViewport vp( -1 ); if ( namedDest && !dest ) dest = m_doc->findDest( namedDest ); if ( !dest || !dest->isOk() ) return vp; // get destination page number if ( !dest->isPageRef() ) vp.pageNumber = dest->getPageNum() - 1; else { Ref ref = dest->getPageRef(); vp.pageNumber = m_doc->findPage( ref.num, ref.gen ) - 1; } // get destination position // TODO add other attributes to the viewport (taken from link) switch ( dest->getKind() ) { case destXYZ: if (dest->getChangeLeft() || dest->getChangeTop()) { int left, top; cvtUserToDev( dest->getLeft(), dest->getTop(), &left, &top ); vp.rePos.normalizedX = (double)left / (double)m_pixmapWidth; vp.rePos.normalizedY = (double)top / (double)m_pixmapHeight; vp.rePos.enabled = true; vp.rePos.pos = DocumentViewport::TopLeft; } // TODO //if ( dest->getChangeZoom() ) // make zoom change break; case destFit: case destFitB: //vp.fitWidth = true; //vp.fitHeight = true; break; case destFitH: case destFitBH: //read top, fit Width //vp.fitWidth = true; break; case destFitV: case destFitBV: //read left, fit Height //vp.fitHeight = true; break; case destFitR: //read and fit left,bottom,right,top break; } return vp; } //END - private helpers /** KPDFTextDev implementation **/ KPDFTextDev::KPDFTextDev() { m_text = new TextPage( gFalse ); } KPDFTextDev::~KPDFTextDev() { delete m_text; } TextPage * KPDFTextDev::takeTextPage() { TextPage * t = m_text; m_text = 0; return t; } void KPDFTextDev::startPage( int, GfxState *state ) { if ( !m_text ) m_text = new TextPage( gFalse ); m_text->startPage( state ); } void KPDFTextDev::endPage() { m_text->endPage(); m_text->coalesce( gTrue ); } void KPDFTextDev::updateFont( GfxState *state ) { m_text->updateFont( state ); } void KPDFTextDev::drawChar( GfxState *state, double x, double y, double dx, double dy, double /*originX*/, double /*originY*/, CharCode code, Unicode *u, int uLen ) { m_text->addChar( state, x, y, dx, dy, code, u, uLen ); }