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.
437 lines
13 KiB
437 lines
13 KiB
/*************************************************************************** |
|
* Copyright (C) 2006 by Luigi Toscano <luigi.toscano@tiscali.it> * |
|
* * |
|
* 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. * |
|
***************************************************************************/ |
|
|
|
#include <okular/core/action.h> |
|
#include <okular/core/document.h> |
|
#include <okular/core/page.h> |
|
#include <okular/core/sourcereference.h> |
|
#include <okular/core/textpage.h> |
|
#include <okular/core/utils.h> |
|
|
|
#include "generator_dvi.h" |
|
#include "dviFile.h" |
|
#include "dviPageInfo.h" |
|
#include "dviRenderer.h" |
|
#include "pageSize.h" |
|
|
|
#include <qapplication.h> |
|
#include <qstring.h> |
|
#include <qurl.h> |
|
#include <qvector.h> |
|
#include <qstack.h> |
|
|
|
#include <kdebug.h> |
|
#include <kimageeffect.h> |
|
#include <klocale.h> |
|
|
|
OKULAR_EXPORT_PLUGIN(DviGenerator) |
|
|
|
DviGenerator::DviGenerator() : Okular::Generator(), |
|
m_docInfo( 0 ), m_docSynopsis( 0 ), ready( false ), m_dviRenderer( 0 ) |
|
{ |
|
setFeature( TextExtraction ); |
|
} |
|
|
|
bool DviGenerator::loadDocument( const QString & fileName, QVector< Okular::Page * > &pagesVector ) |
|
{ |
|
//kDebug() << "file: " << qPrintable( fileName ) << endl; |
|
KUrl base( fileName ); |
|
|
|
m_dviRenderer = new dviRenderer(); |
|
if ( ! m_dviRenderer->setFile( fileName, base ) ) |
|
return false; |
|
|
|
kDebug() << "# of pages: " << m_dviRenderer->dviFile->total_pages << endl; |
|
|
|
m_resolution = Okular::Utils::dpiY(); |
|
loadPages( pagesVector, 0 ); |
|
|
|
ready = true; |
|
return true; |
|
} |
|
|
|
bool DviGenerator::closeDocument() |
|
{ |
|
delete m_docInfo; |
|
m_docInfo = 0; |
|
delete m_docSynopsis; |
|
m_docSynopsis = 0; |
|
delete m_dviRenderer; |
|
m_dviRenderer = 0; |
|
|
|
ready = false; |
|
|
|
return true; |
|
} |
|
|
|
bool DviGenerator::canGeneratePixmap () const |
|
{ |
|
return ready; |
|
} |
|
|
|
static void rotateCoordinates( const int iWidth, const int iHeight, |
|
int &fWidth, int &fHeight, const int orientation) |
|
{ |
|
if ( orientation % 2 == 0 ) |
|
{ |
|
/* portrait */ |
|
fWidth = iWidth; |
|
fHeight = iHeight; |
|
} |
|
else |
|
{ |
|
/* landscape */ |
|
fWidth = iHeight; |
|
fHeight = iWidth; |
|
} |
|
|
|
} |
|
|
|
static void rotateCoordinates( const double iWidth, const double iHeight, |
|
double &fWidth, double &fHeight, const int orientation) |
|
{ |
|
if ( orientation % 2 == 0 ) |
|
{ |
|
/* portrait */ |
|
fWidth = iWidth; |
|
fHeight = iHeight; |
|
} |
|
else |
|
{ |
|
/* landscape */ |
|
fWidth = iHeight; |
|
fHeight = iWidth; |
|
} |
|
|
|
} |
|
|
|
void DviGenerator::fillViewportFromAnchor( Okular::DocumentViewport &vp, |
|
const Anchor &anch, int pW, int pH, |
|
int orientation ) |
|
{ |
|
vp.pageNumber = anch.page - 1; |
|
|
|
double vp_x = 0.0, vp_y = 0.0; |
|
|
|
SimplePageSize ps = m_dviRenderer->sizeOfPage( vp.pageNumber ); |
|
double resolution = 0; |
|
|
|
if ( orientation % 2 == 0 ) |
|
resolution = (double)(pW)/ps.width().getLength_in_inch(); |
|
else |
|
resolution = (double)(pH)/ps.height().getLength_in_inch(); |
|
|
|
double py = (double)anch.distance_from_top.getLength_in_inch()*resolution + 0.5; |
|
|
|
rotateCoordinates( 0.5, py / (double)pH, |
|
vp_x, vp_y, orientation ); |
|
vp.rePos.normalizedX = vp_x; |
|
vp.rePos.normalizedY = vp_y; |
|
vp.rePos.enabled = true; |
|
vp.rePos.pos = Okular::DocumentViewport::Center; |
|
} |
|
|
|
QLinkedList<Okular::ObjectRect*> DviGenerator::generateDviLinks( const dviPageInfo *pageInfo, |
|
int orientation ) |
|
{ |
|
QLinkedList<Okular::ObjectRect*> dviLinks; |
|
|
|
int pageWidth = 0, pageHeight = 0; |
|
|
|
rotateCoordinates( pageInfo->width, pageInfo->height, |
|
pageWidth, pageHeight, orientation ); |
|
|
|
foreach( const Hyperlink dviLink, pageInfo->hyperLinkList ) |
|
{ |
|
QRect boxArea = Okular::Utils::rotateRect( dviLink.box, pageWidth, pageHeight, |
|
orientation ); |
|
double nl = (double)boxArea.left() / pageWidth, |
|
nt = (double)boxArea.top() / pageHeight, |
|
nr = (double)boxArea.right() / pageWidth, |
|
nb = (double)boxArea.bottom() / pageHeight; |
|
|
|
Anchor anch = m_dviRenderer->findAnchor(dviLink.linkText); |
|
|
|
Okular::Link *okuLink = 0; |
|
|
|
/* distinguish between local (-> anchor) and remote links */ |
|
if (anch.isValid()) |
|
{ |
|
/* internal link */ |
|
Okular::DocumentViewport vp; |
|
fillViewportFromAnchor( vp, anch, pageWidth, pageHeight, |
|
orientation ); |
|
okuLink = new Okular::LinkGoto( "", vp ); |
|
} |
|
else |
|
{ |
|
okuLink = new Okular::LinkBrowse( dviLink.linkText ); |
|
} |
|
if ( okuLink ) |
|
{ |
|
Okular::ObjectRect *orlink = new Okular::ObjectRect( nl, nt, nr, nb, false, Okular::ObjectRect::Link, |
|
okuLink ); |
|
dviLinks.push_front( orlink ); |
|
} |
|
|
|
} |
|
return dviLinks; |
|
} |
|
|
|
void DviGenerator::generatePixmap( Okular::PixmapRequest *request ) |
|
{ |
|
|
|
dviPageInfo *pageInfo = new dviPageInfo(); |
|
pageSize ps; |
|
|
|
rotateCoordinates( request->width(), request->height(), |
|
pageInfo->width, pageInfo->height, request->page()->rotation() ); |
|
|
|
pageInfo->pageNumber = request->pageNumber() + 1; |
|
|
|
// pageInfo->resolution = m_resolution; |
|
|
|
SimplePageSize s = m_dviRenderer->sizeOfPage( pageInfo->pageNumber ); |
|
|
|
/* if ( s.width() != pageInfo->width) */ |
|
// if (!useDocumentSpecifiedSize) |
|
// s = userPreferredSize; |
|
|
|
if (s.isValid()) |
|
{ |
|
ps = s; /* it should be the user specified size, if any, instead */ |
|
} |
|
|
|
pageInfo->resolution = (double)(pageInfo->width)/ps.width().getLength_in_inch(); |
|
|
|
#if 0 |
|
kDebug() << request |
|
<< ", res:" << pageInfo->resolution << " - (" << pageInfo->width << "," |
|
<< ps.width().getLength_in_inch() << ")," << ps.width().getLength_in_mm() |
|
<< endl; |
|
#endif |
|
|
|
if ( m_dviRenderer ) |
|
{ |
|
m_dviRenderer->drawPage( pageInfo ); |
|
|
|
if ( ! pageInfo->img.isNull() ) |
|
{ |
|
kDebug() << "Image OK" << endl; |
|
|
|
request->page()->setPixmap( request->id(), new QPixmap( QPixmap::fromImage( pageInfo->img ) ) ); |
|
|
|
request->page()->setObjectRects( |
|
generateDviLinks( pageInfo, 0 ) ); |
|
} |
|
} |
|
|
|
ready = true; |
|
|
|
delete pageInfo; |
|
|
|
signalPixmapRequestDone( request ); |
|
} |
|
|
|
Okular::TextPage* DviGenerator::textPage( Okular::Page *page ) |
|
{ |
|
kDebug() << "DviGenerator::textPage( Okular::Page * page )" << endl; |
|
dviPageInfo *pageInfo = new dviPageInfo(); |
|
pageSize ps; |
|
|
|
rotateCoordinates( (int)(page->width()), (int)(page->height()), |
|
pageInfo->width, pageInfo->height, page->totalOrientation() ); |
|
|
|
pageInfo->pageNumber = page->number() + 1; |
|
|
|
pageInfo->resolution = m_resolution; |
|
SimplePageSize s = m_dviRenderer->sizeOfPage( pageInfo->pageNumber ); |
|
pageInfo->resolution = (double)(pageInfo->width)/ps.width().getLength_in_inch(); |
|
|
|
// get page text from m_dviRenderer |
|
Okular::TextPage *ktp = 0; |
|
if ( m_dviRenderer ) |
|
{ |
|
m_dviRenderer->getText( pageInfo ); |
|
|
|
ktp = extractTextFromPage( pageInfo, page->totalOrientation() ); |
|
} |
|
delete pageInfo; |
|
return ktp; |
|
} |
|
|
|
Okular::TextPage *DviGenerator::extractTextFromPage( dviPageInfo *pageInfo, int orientation ) |
|
{ |
|
QList<Okular::TextEntity*> textOfThePage; |
|
|
|
QVector<TextBox>::ConstIterator it = pageInfo->textBoxList.constBegin(); |
|
QVector<TextBox>::ConstIterator itEnd = pageInfo->textBoxList.constEnd(); |
|
QRect tmpRect; |
|
|
|
int pageWidth = 0, pageHeight = 0; |
|
|
|
rotateCoordinates( pageInfo->width, pageInfo->height, |
|
pageWidth, pageHeight, orientation ); |
|
|
|
for ( ; it != itEnd ; ++it ) |
|
{ |
|
TextBox curTB = *it; |
|
|
|
tmpRect = Okular::Utils::rotateRect( curTB.box, pageWidth, pageHeight, orientation ); |
|
|
|
#if 0 |
|
kDebug() << "orientation: " << orientation |
|
<< ", curTB.box: " << curTB.box |
|
<< ", tmpRect: " << tmpRect |
|
<< ", ( " << pageWidth << "," << pageHeight << " )" |
|
<<endl; |
|
#endif |
|
textOfThePage.push_back( new Okular::TextEntity( curTB.text, |
|
new Okular::NormalizedRect( tmpRect, pageWidth, pageHeight ) ) ); |
|
} |
|
|
|
Okular::TextPage* ktp = new Okular::TextPage( textOfThePage ); |
|
|
|
return ktp; |
|
} |
|
|
|
const Okular::DocumentInfo *DviGenerator::generateDocumentInfo() |
|
{ |
|
if ( m_docInfo ) |
|
return m_docInfo; |
|
|
|
m_docInfo = new Okular::DocumentInfo(); |
|
|
|
m_docInfo->set( "mimeType", "application/x-dvi" ); |
|
|
|
if ( m_dviRenderer && m_dviRenderer->dviFile ) |
|
{ |
|
dvifile *dvif = m_dviRenderer->dviFile; |
|
|
|
// read properties from dvif |
|
//m_docInfo->set( "filename", dvif->filename, i18n("Filename") ); |
|
m_docInfo->set( "generatorDate", dvif->generatorString, |
|
i18n("Generator/Date") ); |
|
m_docInfo->set( "pages", QString::number( dvif->total_pages ), |
|
i18n("Pages") ); |
|
} |
|
return m_docInfo; |
|
} |
|
|
|
const Okular::DocumentSynopsis *DviGenerator::generateDocumentSynopsis() |
|
{ |
|
if ( m_docSynopsis ) |
|
return m_docSynopsis; |
|
|
|
m_docSynopsis = new Okular::DocumentSynopsis(); |
|
|
|
QStack<QDomElement*> stack; |
|
|
|
QVector<PreBookmark> prebookmarks = m_dviRenderer->getPrebookmarks(); |
|
|
|
if ( prebookmarks.isEmpty() ) |
|
return m_docSynopsis; |
|
|
|
QVector<PreBookmark>::ConstIterator it = prebookmarks.begin(); |
|
QVector<PreBookmark>::ConstIterator itEnd = prebookmarks.end(); |
|
for( ; it != itEnd; ++it ) |
|
{ |
|
QDomElement *domel = new QDomElement; |
|
*domel= m_docSynopsis->createElement( (*it).title ); |
|
|
|
Anchor a = m_dviRenderer->findAnchor((*it).anchorName); |
|
if ( a.isValid() ) |
|
{ |
|
Okular::DocumentViewport vp; |
|
|
|
const Okular::Page *p = document()->page( a.page - 1 ); |
|
/* Don't care about rotations... */ |
|
fillViewportFromAnchor( vp, a, (int)p->width(), (int)p->height(), 0 ); |
|
domel->setAttribute( "Viewport", vp.toString() ); |
|
} |
|
if ( stack.isEmpty() ) |
|
m_docSynopsis->appendChild( *domel ); |
|
else |
|
{ |
|
stack.top()->appendChild( *domel ); |
|
stack.pop(); |
|
} |
|
for ( int i = 0; i < (*it).noOfChildren; ++i ) |
|
stack.push( domel ); |
|
if (!(*it).noOfChildren) |
|
delete domel; |
|
} |
|
|
|
return m_docSynopsis; |
|
} |
|
|
|
void DviGenerator::loadPages( QVector< Okular::Page * > &pagesVector, int orientation ) |
|
{ |
|
QSize pageRequiredSize; |
|
|
|
int numofpages = m_dviRenderer->dviFile->total_pages; |
|
pagesVector.resize( numofpages ); |
|
|
|
//kDebug() << "resolution: " << m_resolution << ", dviFile->preferred? " << endl; |
|
|
|
/* get the suggested size */ |
|
if ( m_dviRenderer->dviFile->suggestedPageSize ) |
|
{ |
|
pageRequiredSize = m_dviRenderer->dviFile->suggestedPageSize->sizeInPixel( |
|
m_resolution ); |
|
} |
|
else |
|
{ |
|
pageSize ps; |
|
pageRequiredSize = ps.sizeInPixel( m_resolution ); |
|
} |
|
|
|
|
|
if ( orientation % 2 != 0 ) |
|
{ |
|
/* landscape */ |
|
pageRequiredSize = QSize ( pageRequiredSize.height(), |
|
pageRequiredSize.width() ); |
|
} |
|
|
|
for ( int i = 0; i < numofpages; ++i ) |
|
{ |
|
//kDebug() << "getting status of page " << i << ":" << endl; |
|
|
|
if ( pagesVector[i] ) |
|
{ |
|
delete pagesVector[i]; |
|
} |
|
|
|
Okular::Page * page = new Okular::Page( i, |
|
pageRequiredSize.width(), |
|
pageRequiredSize.height(), |
|
(Okular::Rotation)orientation ); |
|
pagesVector[i] = page; |
|
} |
|
kDebug() << "pagesVector successfully inizialized ! " << endl; |
|
|
|
// filling the pages with the source references rects |
|
const QVector<DVI_SourceFileAnchor>& sourceAnchors = m_dviRenderer->sourceAnchors(); |
|
QVector< QLinkedList< Okular::SourceRefObjectRect * > > refRects( numofpages ); |
|
foreach ( const DVI_SourceFileAnchor& sfa, sourceAnchors ) |
|
{ |
|
if ( sfa.page < 1 || (int)sfa.page > numofpages ) |
|
continue; |
|
|
|
Okular::NormalizedPoint p( 0.5, (double)sfa.distance_from_top.getLength_in_pixel( Okular::Utils::dpiY() ) / (double)pageRequiredSize.height() ); |
|
Okular::SourceReference * sourceRef = new Okular::SourceReference( sfa.fileName, sfa.line ); |
|
refRects[ sfa.page - 1 ].append( new Okular::SourceRefObjectRect( p, sourceRef ) ); |
|
} |
|
for ( int i = 0; i < refRects.size(); ++i ) |
|
if ( !refRects.at(i).isEmpty() ) |
|
pagesVector[i]->setSourceReferences( refRects.at(i) ); |
|
} |
|
|
|
#include "generator_dvi.moc"
|
|
|