You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 

469 lines
14 KiB

/***************************************************************************
* Copyright (C) 2003-2004 by Christophe Devriese *
* <Christophe.Devriese@student.kuleuven.ac.be> *
* Copyright (C) 2003 by Andy Goossens <andygoossens@telenet.be> *
* Copyright (C) 2003 by Scott Wheeler <wheeler@kde.org> *
* Copyright (C) 2003 by Ingo Klöcker <kloecker@kde.org> *
* Copyright (C) 2003 by Will Andrews <will@csociety.org> *
* Copyright (C) 2004 by Dominique Devriese <devriese@kde.org> *
* Copyright (C) 2004 by Waldo Bastian <bastian@kde.org> *
* Copyright (C) 2004 by Albert Astals Cid <tsdgeos@terra.es> *
* *
* 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 <kdebug.h>
#include <qpixmap.h>
#include <qimage.h>
#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 );
}