/*************************************************************************** * Copyright (C) 2006 by Luigi Toscano * * * * 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 #include #include #include #include #include #include "generator_dvi.h" #include "dviFile.h" #include "dviPageInfo.h" #include "dviRenderer.h" #include "pageSize.h" #include #include #include #include #include #include #include #include OKULAR_EXPORT_PLUGIN(DviGenerator) DviGenerator::DviGenerator() : Okular::Generator(), m_docInfo( 0 ), m_docSynopsis( 0 ), ready( false ), m_dviRenderer( 0 ) { } 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 ( bool /*async*/ ) 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 DviGenerator::generateDviLinks( const dviPageInfo *pageInfo, int orientation ) { QLinkedList 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; signalRequestDone( request ); } void DviGenerator::generateSyncTextPage( Okular::Page* page ) { kDebug() << "DviGenerator::generateSyncTextPage( 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 if ( m_dviRenderer ) { m_dviRenderer->getText( pageInfo ); Okular::TextPage *ktp = extractTextFromPage( pageInfo, page->totalOrientation() ); page->setTextPage( ktp ); } delete pageInfo; } Okular::TextPage *DviGenerator::extractTextFromPage( dviPageInfo *pageInfo, int orientation ) { QList textOfThePage; QVector::ConstIterator it = pageInfo->textBoxList.constBegin(); QVector::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 << " )" <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 stack; QVector prebookmarks = m_dviRenderer->getPrebookmarks(); if ( prebookmarks.isEmpty() ) return m_docSynopsis; QVector::ConstIterator it = prebookmarks.begin(); QVector::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; } bool DviGenerator::hasFeature( GeneratorFeature feature ) const { switch ( feature ) { case TextExtraction: return true; default: ; } return false; } 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& 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"