mirror of
https://invent.kde.org/graphics/okular
synced 2024-10-28 19:28:38 +00:00
4a71603ab6
[the compressed versions are not in the "generator" .desktop files because okular can deal natively with any gzip- or bzip2- compressed version of the supported document types] svn path=/trunk/KDE/kdegraphics/okular/; revision=755661
2926 lines
98 KiB
C++
2926 lines
98 KiB
C++
/***************************************************************************
|
|
* Copyright (C) 2004-2005 by Enrico Ros <eros.kde@email.it> *
|
|
* Copyright (C) 2004-2007 by Albert Astals Cid <aacid@kde.org> *
|
|
* *
|
|
* 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 "document.h"
|
|
#include "document_p.h"
|
|
|
|
#ifdef Q_OS_WIN
|
|
#define _WIN32_WINNT 0x0500
|
|
#include <windows.h>
|
|
#endif
|
|
|
|
// qt/kde/system includes
|
|
#include <QtCore/QtAlgorithms>
|
|
#include <QtCore/QDir>
|
|
#include <QtCore/QFile>
|
|
#include <QtCore/QFileInfo>
|
|
#include <QtCore/QMap>
|
|
#include <QtCore/QProcess>
|
|
#include <QtCore/QTextStream>
|
|
#include <QtCore/QTimer>
|
|
#include <QtGui/QApplication>
|
|
#include <QtGui/QLabel>
|
|
#include <QtGui/QPrinter>
|
|
#include <QtGui/QPrintDialog>
|
|
|
|
#include <kaboutdata.h>
|
|
#include <kauthorized.h>
|
|
#include <kcomponentdata.h>
|
|
#include <kconfigdialog.h>
|
|
#include <kdebug.h>
|
|
#include <klibloader.h>
|
|
#include <klocale.h>
|
|
#include <kmessagebox.h>
|
|
#include <kmimetypetrader.h>
|
|
#include <krun.h>
|
|
#include <kstandarddirs.h>
|
|
#include <ktemporaryfile.h>
|
|
#include <ktoolinvocation.h>
|
|
|
|
// local includes
|
|
#include "action.h"
|
|
#include "annotations.h"
|
|
#include "annotations_p.h"
|
|
#include "audioplayer.h"
|
|
#include "audioplayer_p.h"
|
|
#include "bookmarkmanager.h"
|
|
#include "chooseenginedialog_p.h"
|
|
#include "debug_p.h"
|
|
#include "generator_p.h"
|
|
#include "interfaces/configinterface.h"
|
|
#include "interfaces/guiinterface.h"
|
|
#include "interfaces/printinterface.h"
|
|
#include "observer.h"
|
|
#include "page.h"
|
|
#include "page_p.h"
|
|
#include "pagecontroller_p.h"
|
|
#include "settings.h"
|
|
#include "sourcereference.h"
|
|
|
|
#include <config-okular.h>
|
|
|
|
using namespace Okular;
|
|
|
|
struct AllocatedPixmap
|
|
{
|
|
// owner of the page
|
|
int id;
|
|
int page;
|
|
qulonglong memory;
|
|
// public constructor: initialize data
|
|
AllocatedPixmap( int i, int p, qulonglong m ) : id( i ), page( p ), memory( m ) {}
|
|
};
|
|
|
|
struct RunningSearch
|
|
{
|
|
// store search properties
|
|
int continueOnPage;
|
|
RegularAreaRect continueOnMatch;
|
|
QSet< int > highlightedPages;
|
|
|
|
// fields related to previous searches (used for 'continueSearch')
|
|
QString cachedString;
|
|
Document::SearchType cachedType;
|
|
Qt::CaseSensitivity cachedCaseSensitivity;
|
|
bool cachedViewportMove : 1;
|
|
bool cachedNoDialogs : 1;
|
|
QColor cachedColor;
|
|
};
|
|
|
|
#define foreachObserver( cmd ) {\
|
|
QMap< int, DocumentObserver * >::const_iterator it=d->m_observers.begin(), end=d->m_observers.end();\
|
|
for ( ; it != end ; ++ it ) { (*it)-> cmd ; } }
|
|
|
|
#define foreachObserverD( cmd ) {\
|
|
QMap< int, DocumentObserver * >::const_iterator it = m_observers.begin(), end = m_observers.end();\
|
|
for ( ; it != end ; ++ it ) { (*it)-> cmd ; } }
|
|
|
|
#define OKULAR_HISTORY_MAXSTEPS 100
|
|
#define OKULAR_HISTORY_SAVEDSTEPS 10
|
|
|
|
/***** Document ******/
|
|
|
|
QString DocumentPrivate::pagesSizeString() const
|
|
{
|
|
if (m_generator)
|
|
{
|
|
if (m_generator->pagesSizeMetric() != Generator::None)
|
|
{
|
|
QSizeF size = m_parent->allPagesSize();
|
|
if (size.isValid()) return localizedSize(size);
|
|
else return QString();
|
|
}
|
|
else return QString();
|
|
}
|
|
else return QString();
|
|
}
|
|
|
|
QString DocumentPrivate::localizedSize(const QSizeF &size) const
|
|
{
|
|
double inchesWidth = 0, inchesHeight = 0;
|
|
switch (m_generator->pagesSizeMetric())
|
|
{
|
|
case Generator::Points:
|
|
inchesWidth = size.width() / 72.0;
|
|
inchesHeight = size.height() / 72.0;
|
|
break;
|
|
|
|
case Generator::None:
|
|
break;
|
|
}
|
|
if (KGlobal::locale()->measureSystem() == KLocale::Imperial)
|
|
{
|
|
return i18n("%1 x %2 in", inchesWidth, inchesHeight);
|
|
}
|
|
else
|
|
{
|
|
return i18n("%1 x %2 mm", inchesWidth * 25.4, inchesHeight * 25.4);
|
|
}
|
|
}
|
|
|
|
void DocumentPrivate::cleanupPixmapMemory( qulonglong /*sure? bytesOffset*/ )
|
|
{
|
|
// [MEM] choose memory parameters based on configuration profile
|
|
qulonglong clipValue = ~0U;
|
|
qulonglong memoryToFree = ~0U;
|
|
switch ( Settings::memoryLevel() )
|
|
{
|
|
case Settings::EnumMemoryLevel::Low:
|
|
memoryToFree = m_allocatedPixmapsTotalMemory;
|
|
break;
|
|
|
|
case Settings::EnumMemoryLevel::Normal:
|
|
memoryToFree = m_allocatedPixmapsTotalMemory - getTotalMemory() / 3;
|
|
clipValue = (m_allocatedPixmapsTotalMemory - getFreeMemory()) / 2;
|
|
break;
|
|
|
|
case Settings::EnumMemoryLevel::Aggressive:
|
|
clipValue = (m_allocatedPixmapsTotalMemory - getFreeMemory()) / 2;
|
|
break;
|
|
}
|
|
|
|
if ( clipValue > memoryToFree )
|
|
memoryToFree = clipValue;
|
|
|
|
if ( memoryToFree > 0 )
|
|
{
|
|
// [MEM] free memory starting from older pixmaps
|
|
int pagesFreed = 0;
|
|
QLinkedList< AllocatedPixmap * >::iterator pIt = m_allocatedPixmapsFifo.begin();
|
|
QLinkedList< AllocatedPixmap * >::iterator pEnd = m_allocatedPixmapsFifo.end();
|
|
while ( (pIt != pEnd) && (memoryToFree > 0) )
|
|
{
|
|
AllocatedPixmap * p = *pIt;
|
|
if ( m_observers.value( p->id )->canUnloadPixmap( p->page ) )
|
|
{
|
|
// update internal variables
|
|
pIt = m_allocatedPixmapsFifo.erase( pIt );
|
|
m_allocatedPixmapsTotalMemory -= p->memory;
|
|
memoryToFree -= p->memory;
|
|
pagesFreed++;
|
|
// delete pixmap
|
|
m_pagesVector.at( p->page )->deletePixmap( p->id );
|
|
// delete allocation descriptor
|
|
delete p;
|
|
} else
|
|
++pIt;
|
|
}
|
|
//p--rintf("freeMemory A:[%d -%d = %d] \n", m_allocatedPixmapsFifo.count() + pagesFreed, pagesFreed, m_allocatedPixmapsFifo.count() );
|
|
}
|
|
}
|
|
|
|
qulonglong DocumentPrivate::getTotalMemory()
|
|
{
|
|
static qulonglong cachedValue = 0;
|
|
if ( cachedValue )
|
|
return cachedValue;
|
|
|
|
#if defined(Q_OS_LINUX)
|
|
// if /proc/meminfo doesn't exist, return 128MB
|
|
QFile memFile( "/proc/meminfo" );
|
|
if ( !memFile.open( QIODevice::ReadOnly ) )
|
|
return (cachedValue = 134217728);
|
|
|
|
// read /proc/meminfo and sum up the contents of 'MemFree', 'Buffers'
|
|
// and 'Cached' fields. consider swapped memory as used memory.
|
|
QTextStream readStream( &memFile );
|
|
while ( true )
|
|
{
|
|
QString entry = readStream.readLine();
|
|
if ( entry.isNull() ) break;
|
|
if ( entry.startsWith( "MemTotal:" ) )
|
|
return (cachedValue = (1024 * entry.section( ' ', -2, -2 ).toInt()));
|
|
}
|
|
#elif defined(Q_OS_WIN)
|
|
MEMORYSTATUSEX stat;
|
|
|
|
GlobalMemoryStatusEx (&stat);
|
|
|
|
return ( cachedValue = stat.ullTotalPhys );
|
|
#endif
|
|
return (cachedValue = 134217728);
|
|
}
|
|
|
|
qulonglong DocumentPrivate::getFreeMemory()
|
|
{
|
|
static QTime lastUpdate = QTime::currentTime();
|
|
static qulonglong cachedValue = 0;
|
|
|
|
if ( lastUpdate.secsTo( QTime::currentTime() ) <= 2 )
|
|
return cachedValue;
|
|
|
|
#if defined(Q_OS_LINUX)
|
|
// if /proc/meminfo doesn't exist, return MEMORY FULL
|
|
QFile memFile( "/proc/meminfo" );
|
|
if ( !memFile.open( QIODevice::ReadOnly ) )
|
|
return 0;
|
|
|
|
// read /proc/meminfo and sum up the contents of 'MemFree', 'Buffers'
|
|
// and 'Cached' fields. consider swapped memory as used memory.
|
|
qulonglong memoryFree = 0;
|
|
QString entry;
|
|
QTextStream readStream( &memFile );
|
|
while ( true )
|
|
{
|
|
entry = readStream.readLine();
|
|
if ( entry.isNull() ) break;
|
|
if ( entry.startsWith( "MemFree:" ) ||
|
|
entry.startsWith( "Buffers:" ) ||
|
|
entry.startsWith( "Cached:" ) ||
|
|
entry.startsWith( "SwapFree:" ) )
|
|
memoryFree += entry.section( ' ', -2, -2 ).toInt();
|
|
if ( entry.startsWith( "SwapTotal:" ) )
|
|
memoryFree -= entry.section( ' ', -2, -2 ).toInt();
|
|
}
|
|
memFile.close();
|
|
|
|
lastUpdate = QTime::currentTime();
|
|
|
|
return ( cachedValue = (1024 * memoryFree) );
|
|
#elif defined(Q_OS_WIN)
|
|
MEMORYSTATUSEX stat;
|
|
|
|
GlobalMemoryStatusEx (&stat);
|
|
|
|
lastUpdate = QTime::currentTime();
|
|
|
|
return ( cachedValue = stat.ullAvailPhys );
|
|
#else
|
|
// tell the memory is full.. will act as in LOW profile
|
|
return 0;
|
|
#endif
|
|
}
|
|
|
|
void DocumentPrivate::loadDocumentInfo()
|
|
// note: load data and stores it internally (document or pages). observers
|
|
// are still uninitialized at this point so don't access them
|
|
{
|
|
//kDebug(OkularDebug).nospace() << "Using '" << d->m_xmlFileName << "' as document info file.";
|
|
if ( m_xmlFileName.isEmpty() )
|
|
return;
|
|
|
|
QFile infoFile( m_xmlFileName );
|
|
if ( !infoFile.exists() || !infoFile.open( QIODevice::ReadOnly ) )
|
|
return;
|
|
|
|
// Load DOM from XML file
|
|
QDomDocument doc( "documentInfo" );
|
|
if ( !doc.setContent( &infoFile ) )
|
|
{
|
|
kDebug(OkularDebug) << "Can't load XML pair! Check for broken xml.";
|
|
infoFile.close();
|
|
return;
|
|
}
|
|
infoFile.close();
|
|
|
|
QDomElement root = doc.documentElement();
|
|
if ( root.tagName() != "documentInfo" )
|
|
return;
|
|
|
|
KUrl documentUrl( root.attribute( "url" ) );
|
|
|
|
// Parse the DOM tree
|
|
QDomNode topLevelNode = root.firstChild();
|
|
while ( topLevelNode.isElement() )
|
|
{
|
|
QString catName = topLevelNode.toElement().tagName();
|
|
|
|
// Restore page attributes (bookmark, annotations, ...) from the DOM
|
|
if ( catName == "pageList" )
|
|
{
|
|
QDomNode pageNode = topLevelNode.firstChild();
|
|
while ( pageNode.isElement() )
|
|
{
|
|
QDomElement pageElement = pageNode.toElement();
|
|
if ( pageElement.hasAttribute( "number" ) )
|
|
{
|
|
// get page number (node's attribute)
|
|
bool ok;
|
|
int pageNumber = pageElement.attribute( "number" ).toInt( &ok );
|
|
|
|
// pass the domElement to the right page, to read config data from
|
|
if ( ok && pageNumber >= 0 && pageNumber < (int)m_pagesVector.count() )
|
|
m_pagesVector[ pageNumber ]->d->restoreLocalContents( pageElement );
|
|
}
|
|
pageNode = pageNode.nextSibling();
|
|
}
|
|
}
|
|
|
|
// Restore 'general info' from the DOM
|
|
else if ( catName == "generalInfo" )
|
|
{
|
|
QDomNode infoNode = topLevelNode.firstChild();
|
|
while ( infoNode.isElement() )
|
|
{
|
|
QDomElement infoElement = infoNode.toElement();
|
|
|
|
// restore viewports history
|
|
if ( infoElement.tagName() == "history" )
|
|
{
|
|
// clear history
|
|
m_viewportHistory.clear();
|
|
// append old viewports
|
|
QDomNode historyNode = infoNode.firstChild();
|
|
while ( historyNode.isElement() )
|
|
{
|
|
QDomElement historyElement = historyNode.toElement();
|
|
if ( historyElement.hasAttribute( "viewport" ) )
|
|
{
|
|
QString vpString = historyElement.attribute( "viewport" );
|
|
m_viewportIterator = m_viewportHistory.insert( m_viewportHistory.end(),
|
|
DocumentViewport( vpString ) );
|
|
}
|
|
historyNode = historyNode.nextSibling();
|
|
}
|
|
// consistancy check
|
|
if ( m_viewportHistory.isEmpty() )
|
|
m_viewportIterator = m_viewportHistory.insert( m_viewportHistory.end(), DocumentViewport() );
|
|
}
|
|
else if ( infoElement.tagName() == "rotation" )
|
|
{
|
|
QString str = infoElement.text();
|
|
bool ok = true;
|
|
int newrotation = !str.isEmpty() ? ( str.toInt( &ok ) % 4 ) : 0;
|
|
if ( ok && newrotation != 0 )
|
|
{
|
|
setRotationInternal( newrotation, false );
|
|
}
|
|
}
|
|
infoNode = infoNode.nextSibling();
|
|
}
|
|
}
|
|
|
|
topLevelNode = topLevelNode.nextSibling();
|
|
} // </documentInfo>
|
|
}
|
|
|
|
QString DocumentPrivate::giveAbsolutePath( const QString & fileName ) const
|
|
{
|
|
if ( !m_url.isValid() )
|
|
return QString();
|
|
|
|
if ( !QDir::isRelativePath( fileName ) )
|
|
return fileName;
|
|
|
|
return m_url.upUrl().url() + fileName;
|
|
}
|
|
|
|
bool DocumentPrivate::openRelativeFile( const QString & fileName )
|
|
{
|
|
QString absFileName = giveAbsolutePath( fileName );
|
|
if ( absFileName.isEmpty() )
|
|
return false;
|
|
|
|
kDebug(OkularDebug).nospace() << "openDocument: '" << absFileName << "'";
|
|
|
|
emit m_parent->openUrl( absFileName );
|
|
return true;
|
|
}
|
|
|
|
Generator * DocumentPrivate::loadGeneratorLibrary( const KService::Ptr &service )
|
|
{
|
|
KPluginFactory *factory = KPluginLoader( service->library() ).factory();
|
|
if ( !factory )
|
|
{
|
|
kWarning(OkularDebug).nospace() << "Invalid plugin factory for " << service->library() << "!";
|
|
return 0;
|
|
}
|
|
Generator * generator = factory->create< Okular::Generator >( 0 );
|
|
GeneratorInfo info( factory->componentData() );
|
|
info.generator = generator;
|
|
if ( info.data.isValid() && info.data.aboutData() )
|
|
info.catalogName = info.data.aboutData()->catalogName();
|
|
m_loadedGenerators.insert( service->name(), info );
|
|
return generator;
|
|
}
|
|
|
|
void DocumentPrivate::loadAllGeneratorLibraries()
|
|
{
|
|
if ( m_generatorsLoaded )
|
|
return;
|
|
|
|
m_generatorsLoaded = true;
|
|
|
|
QString constraint("([X-KDE-Priority] > 0) and (exist Library)") ;
|
|
KService::List offers = KServiceTypeTrader::self()->query( "okular/Generator", constraint );
|
|
loadServiceList( offers );
|
|
}
|
|
|
|
void DocumentPrivate::loadServiceList( const KService::List& offers )
|
|
{
|
|
int count = offers.count();
|
|
if ( count <= 0 )
|
|
return;
|
|
|
|
for ( int i = 0; i < count; ++i )
|
|
{
|
|
QString propName = offers.at(i)->name();
|
|
// don't load already loaded generators
|
|
QHash< QString, GeneratorInfo >::const_iterator genIt = m_loadedGenerators.constFind( propName );
|
|
if ( !m_loadedGenerators.isEmpty() && genIt != m_loadedGenerators.end() )
|
|
continue;
|
|
|
|
Generator * g = loadGeneratorLibrary( offers.at(i) );
|
|
(void)g;
|
|
}
|
|
}
|
|
|
|
void DocumentPrivate::unloadGenerator( const GeneratorInfo& info )
|
|
{
|
|
delete info.generator;
|
|
}
|
|
|
|
void DocumentPrivate::cacheExportFormats()
|
|
{
|
|
if ( m_exportCached )
|
|
return;
|
|
|
|
const ExportFormat::List formats = m_generator->exportFormats();
|
|
for ( int i = 0; i < formats.count(); ++i )
|
|
{
|
|
if ( formats.at( i ).mimeType()->name() == QLatin1String( "text/plain" ) )
|
|
m_exportToText = formats.at( i );
|
|
else
|
|
m_exportFormats.append( formats.at( i ) );
|
|
}
|
|
|
|
m_exportCached = true;
|
|
}
|
|
|
|
ConfigInterface* DocumentPrivate::generatorConfig( GeneratorInfo& info )
|
|
{
|
|
if ( info.configChecked )
|
|
return info.config;
|
|
|
|
info.config = qobject_cast< Okular::ConfigInterface * >( info.generator );
|
|
info.configChecked = true;
|
|
return info.config;
|
|
}
|
|
|
|
void DocumentPrivate::saveDocumentInfo() const
|
|
{
|
|
if ( m_xmlFileName.isEmpty() )
|
|
return;
|
|
|
|
QFile infoFile( m_xmlFileName );
|
|
if (infoFile.open( QIODevice::WriteOnly | QIODevice::Truncate) )
|
|
{
|
|
// 1. Create DOM
|
|
QDomDocument doc( "documentInfo" );
|
|
QDomElement root = doc.createElement( "documentInfo" );
|
|
root.setAttribute( "url", m_url.pathOrUrl() );
|
|
doc.appendChild( root );
|
|
|
|
// 2.1. Save page attributes (bookmark state, annotations, ... ) to DOM
|
|
QDomElement pageList = doc.createElement( "pageList" );
|
|
root.appendChild( pageList );
|
|
// <page list><page number='x'>.... </page> save pages that hold data
|
|
QVector< Page * >::const_iterator pIt = m_pagesVector.begin(), pEnd = m_pagesVector.end();
|
|
for ( ; pIt != pEnd; ++pIt )
|
|
(*pIt)->d->saveLocalContents( pageList, doc );
|
|
|
|
// 2.2. Save document info (current viewport, history, ... ) to DOM
|
|
QDomElement generalInfo = doc.createElement( "generalInfo" );
|
|
root.appendChild( generalInfo );
|
|
// create rotation node
|
|
if ( m_rotation != Rotation0 )
|
|
{
|
|
QDomElement rotationNode = doc.createElement( "rotation" );
|
|
generalInfo.appendChild( rotationNode );
|
|
rotationNode.appendChild( doc.createTextNode( QString::number( (int)m_rotation ) ) );
|
|
}
|
|
// <general info><history> ... </history> save history up to OKULAR_HISTORY_SAVEDSTEPS viewports
|
|
QLinkedList< DocumentViewport >::const_iterator backIterator = m_viewportIterator;
|
|
if ( backIterator != m_viewportHistory.end() )
|
|
{
|
|
// go back up to OKULAR_HISTORY_SAVEDSTEPS steps from the current viewportIterator
|
|
int backSteps = OKULAR_HISTORY_SAVEDSTEPS;
|
|
while ( backSteps-- && backIterator != m_viewportHistory.begin() )
|
|
--backIterator;
|
|
|
|
// create history root node
|
|
QDomElement historyNode = doc.createElement( "history" );
|
|
generalInfo.appendChild( historyNode );
|
|
|
|
// add old[backIterator] and present[viewportIterator] items
|
|
QLinkedList< DocumentViewport >::const_iterator endIt = m_viewportIterator;
|
|
++endIt;
|
|
while ( backIterator != endIt )
|
|
{
|
|
QString name = (backIterator == m_viewportIterator) ? "current" : "oldPage";
|
|
QDomElement historyEntry = doc.createElement( name );
|
|
historyEntry.setAttribute( "viewport", (*backIterator).toString() );
|
|
historyNode.appendChild( historyEntry );
|
|
++backIterator;
|
|
}
|
|
}
|
|
|
|
// 3. Save DOM to XML file
|
|
QString xml = doc.toString();
|
|
QTextStream os( &infoFile );
|
|
os << xml;
|
|
}
|
|
infoFile.close();
|
|
}
|
|
|
|
void DocumentPrivate::slotTimedMemoryCheck()
|
|
{
|
|
// [MEM] clean memory (for 'free mem dependant' profiles only)
|
|
if ( Settings::memoryLevel() != Settings::EnumMemoryLevel::Low &&
|
|
m_allocatedPixmapsTotalMemory > 1024*1024 )
|
|
cleanupPixmapMemory();
|
|
}
|
|
|
|
void DocumentPrivate::sendGeneratorRequest()
|
|
{
|
|
// find a request
|
|
PixmapRequest * request = 0;
|
|
m_pixmapRequestsMutex.lock();
|
|
while ( !m_pixmapRequestsStack.isEmpty() && !request )
|
|
{
|
|
PixmapRequest * r = m_pixmapRequestsStack.last();
|
|
if (!r)
|
|
m_pixmapRequestsStack.pop_back();
|
|
|
|
// request only if page isn't already present or request has invalid id
|
|
else if ( r->page()->hasPixmap( r->id(), r->width(), r->height() ) || r->id() <= 0 || r->id() >= MAX_OBSERVER_ID)
|
|
{
|
|
m_pixmapRequestsStack.pop_back();
|
|
delete r;
|
|
}
|
|
else if ( (long)r->width() * (long)r->height() > 20000000L )
|
|
{
|
|
m_pixmapRequestsStack.pop_back();
|
|
if ( !m_warnedOutOfMemory )
|
|
{
|
|
kWarning(OkularDebug).nospace() << "Running out of memory on page " << r->pageNumber()
|
|
<< " (" << r->width() << "x" << r->height() << " px);";
|
|
kWarning(OkularDebug) << "this message will be reported only once.";
|
|
m_warnedOutOfMemory = true;
|
|
}
|
|
delete r;
|
|
}
|
|
else
|
|
request = r;
|
|
}
|
|
|
|
// if no request found (or already generated), return
|
|
if ( !request )
|
|
{
|
|
m_pixmapRequestsMutex.unlock();
|
|
return;
|
|
}
|
|
|
|
// [MEM] preventive memory freeing
|
|
qulonglong pixmapBytes = 4 * request->width() * request->height();
|
|
if ( pixmapBytes > (1024 * 1024) )
|
|
cleanupPixmapMemory( pixmapBytes );
|
|
|
|
// submit the request to the generator
|
|
if ( m_generator->canGeneratePixmap() )
|
|
{
|
|
kDebug(OkularDebug).nospace() << "sending request id=" << request->id() << " " <<request->width() << "x" << request->height() << "@" << request->pageNumber() << " async == " << request->asynchronous();
|
|
m_pixmapRequestsStack.removeAll ( request );
|
|
|
|
if ( (int)m_rotation % 2 )
|
|
request->d->swap();
|
|
|
|
// we always have to unlock _before_ the generatePixmap() because
|
|
// a sync generation would end with requestDone() -> deadlock, and
|
|
// we can not really know if the generator can do async requests
|
|
m_pixmapRequestsMutex.unlock();
|
|
m_generator->generatePixmap( request );
|
|
}
|
|
else
|
|
{
|
|
m_pixmapRequestsMutex.unlock();
|
|
// pino (7/4/2006): set the polling interval from 10 to 30
|
|
QTimer::singleShot( 30, m_parent, SLOT(sendGeneratorRequest()) );
|
|
}
|
|
}
|
|
|
|
void DocumentPrivate::rotationFinished( int page )
|
|
{
|
|
QMap< int, DocumentObserver * >::const_iterator it = m_observers.begin(), end = m_observers.end();
|
|
for ( ; it != end ; ++ it ) {
|
|
(*it)->notifyPageChanged( page, DocumentObserver::Pixmap | DocumentObserver::Annotations );
|
|
}
|
|
}
|
|
|
|
void DocumentPrivate::fontReadingProgress( int page )
|
|
{
|
|
emit m_parent->fontReadingProgress( page );
|
|
|
|
if ( page >= (int)m_parent->pages() - 1 )
|
|
{
|
|
emit m_parent->fontReadingEnded();
|
|
m_fontThread = 0;
|
|
m_fontsCached = true;
|
|
}
|
|
}
|
|
|
|
void DocumentPrivate::fontReadingGotFont( const Okular::FontInfo& font )
|
|
{
|
|
// TODO try to avoid duplicate fonts
|
|
m_fontsCache.append( font );
|
|
|
|
emit m_parent->gotFont( font );
|
|
}
|
|
|
|
void DocumentPrivate::slotGeneratorConfigChanged( const QString& )
|
|
{
|
|
if ( !m_generator )
|
|
return;
|
|
|
|
// reparse generator config and if something changed clear Pages
|
|
bool configchanged = false;
|
|
QHash< QString, GeneratorInfo >::iterator it = m_loadedGenerators.begin(), itEnd = m_loadedGenerators.end();
|
|
for ( ; it != itEnd; ++it )
|
|
{
|
|
Okular::ConfigInterface * iface = generatorConfig( it.value() );
|
|
if ( iface )
|
|
{
|
|
bool it_changed = iface->reparseConfig();
|
|
if ( it_changed && ( m_generator == it.value().generator ) )
|
|
configchanged = true;
|
|
}
|
|
}
|
|
if ( configchanged )
|
|
{
|
|
// invalidate pixmaps
|
|
QVector<Page*>::const_iterator it = m_pagesVector.begin(), end = m_pagesVector.end();
|
|
for ( ; it != end; ++it ) {
|
|
(*it)->deletePixmaps();
|
|
}
|
|
|
|
// [MEM] remove allocation descriptors
|
|
QLinkedList< AllocatedPixmap * >::const_iterator aIt = m_allocatedPixmapsFifo.begin();
|
|
QLinkedList< AllocatedPixmap * >::const_iterator aEnd = m_allocatedPixmapsFifo.end();
|
|
for ( ; aIt != aEnd; ++aIt )
|
|
delete *aIt;
|
|
m_allocatedPixmapsFifo.clear();
|
|
m_allocatedPixmapsTotalMemory = 0;
|
|
|
|
// send reload signals to observers
|
|
foreachObserverD( notifyContentsCleared( DocumentObserver::Pixmap ) );
|
|
}
|
|
|
|
// free memory if in 'low' profile
|
|
if ( Settings::memoryLevel() == Settings::EnumMemoryLevel::Low &&
|
|
!m_allocatedPixmapsFifo.isEmpty() && !m_pagesVector.isEmpty() )
|
|
cleanupPixmapMemory();
|
|
}
|
|
|
|
void DocumentPrivate::doContinueNextMatchSearch(void *pagesToNotifySet, void * theMatch, int currentPage, int searchID, const QString & text, int theCaseSensitivity, bool moveViewport, const QColor & color, bool noDialogs, int donePages)
|
|
{
|
|
RegularAreaRect * match = static_cast<RegularAreaRect *>(theMatch);
|
|
Qt::CaseSensitivity caseSensitivity = static_cast<Qt::CaseSensitivity>(theCaseSensitivity);
|
|
QSet< int > *pagesToNotify = static_cast< QSet< int > * >( pagesToNotifySet );
|
|
|
|
if (m_searchCancelled && !match)
|
|
{
|
|
// if the user cancelled but he just got a match, give him the match!
|
|
QApplication::restoreOverrideCursor();
|
|
emit m_parent->searchFinished( searchID, Document::SearchCancelled );
|
|
delete pagesToNotify;
|
|
return;
|
|
}
|
|
|
|
// if no match found, loop through the whole doc, starting from currentPage
|
|
if ( !match )
|
|
{
|
|
int pageCount = m_pagesVector.count();
|
|
if (donePages < pageCount)
|
|
{
|
|
bool doContinue = true;
|
|
if ( currentPage >= pageCount )
|
|
{
|
|
if ( noDialogs || KMessageBox::questionYesNo(m_parent->widget(), i18n("End of document reached.\nContinue from the beginning?"), QString(), KStandardGuiItem::cont(), KStandardGuiItem::cancel()) == KMessageBox::Yes )
|
|
currentPage = 0;
|
|
else
|
|
doContinue = false;
|
|
}
|
|
if (doContinue)
|
|
{
|
|
// get page
|
|
Page * page = m_pagesVector[ currentPage ];
|
|
// request search page if needed
|
|
if ( !page->hasTextPage() )
|
|
m_parent->requestTextPage( page->number() );
|
|
// if found a match on the current page, end the loop
|
|
match = page->findText( searchID, text, FromTop, caseSensitivity );
|
|
|
|
if ( !match ) currentPage++;
|
|
|
|
QMetaObject::invokeMethod(m_parent, "doContinueNextMatchSearch", Qt::QueuedConnection, Q_ARG(void *, pagesToNotifySet), Q_ARG(void *, match), Q_ARG(int, currentPage), Q_ARG(int, searchID), Q_ARG(QString, text), Q_ARG(int, caseSensitivity), Q_ARG(bool, moveViewport), Q_ARG(QColor, color), Q_ARG(bool, noDialogs), Q_ARG(int, donePages +1));
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
// reset cursor to previous shape
|
|
QApplication::restoreOverrideCursor();
|
|
|
|
bool foundAMatch = false;
|
|
|
|
// if a match has been found..
|
|
if ( match )
|
|
{
|
|
// update the RunningSearch structure adding this match..
|
|
RunningSearch * s = m_searches[searchID];
|
|
foundAMatch = true;
|
|
s->continueOnPage = currentPage;
|
|
s->continueOnMatch = *match;
|
|
s->highlightedPages.insert( currentPage );
|
|
// ..add highlight to the page..
|
|
m_pagesVector[ currentPage ]->d->setHighlight( searchID, match, color );
|
|
|
|
// ..queue page for notifying changes..
|
|
pagesToNotify->insert( currentPage );
|
|
|
|
// ..move the viewport to show the first of the searched word sequence centered
|
|
if ( moveViewport )
|
|
{
|
|
DocumentViewport searchViewport( currentPage );
|
|
searchViewport.rePos.enabled = true;
|
|
searchViewport.rePos.normalizedX = (match->first().left + match->first().right) / 2.0;
|
|
searchViewport.rePos.normalizedY = (match->first().top + match->first().bottom) / 2.0;
|
|
m_parent->setViewport( searchViewport, -1, true );
|
|
}
|
|
delete match;
|
|
}
|
|
else if ( !noDialogs )
|
|
KMessageBox::information( m_parent->widget(), i18n( "No matches found for '%1'.", text ) );
|
|
|
|
// notify observers about highlights changes
|
|
foreach(int pageNumber, *pagesToNotify)
|
|
foreach(DocumentObserver *observer, m_observers)
|
|
observer->notifyPageChanged( pageNumber, DocumentObserver::Highlights );
|
|
|
|
if (foundAMatch) emit m_parent->searchFinished( searchID, Document::MatchFound );
|
|
else emit m_parent->searchFinished( searchID, Document::NoMatchFound );
|
|
|
|
delete pagesToNotify;
|
|
}
|
|
|
|
void DocumentPrivate::doContinueAllDocumentSearch(void *pagesToNotifySet, void *pageMatchesMap, int currentPage, int searchID, const QString & text, int theCaseSensitivity, const QColor & color)
|
|
{
|
|
QMap< Page *, QVector<RegularAreaRect *> > *pageMatches = static_cast< QMap< Page *, QVector<RegularAreaRect *> > * >(pageMatchesMap);
|
|
Qt::CaseSensitivity caseSensitivity = static_cast<Qt::CaseSensitivity>(theCaseSensitivity);
|
|
QSet< int > *pagesToNotify = static_cast< QSet< int > * >( pagesToNotifySet );
|
|
|
|
if (m_searchCancelled)
|
|
{
|
|
typedef QVector<RegularAreaRect *> MatchesVector;
|
|
|
|
QApplication::restoreOverrideCursor();
|
|
emit m_parent->searchFinished( searchID, Document::SearchCancelled );
|
|
foreach(const MatchesVector &mv, *pageMatches) qDeleteAll(mv);
|
|
delete pageMatches;
|
|
delete pagesToNotify;
|
|
return;
|
|
}
|
|
|
|
if (currentPage < m_pagesVector.count())
|
|
{
|
|
// get page (from the first to the last)
|
|
Page *page = m_pagesVector.at(currentPage);
|
|
int pageNumber = page->number(); // redundant? is it == currentPage ?
|
|
|
|
// request search page if needed
|
|
if ( !page->hasTextPage() )
|
|
m_parent->requestTextPage( pageNumber );
|
|
|
|
// loop on a page adding highlights for all found items
|
|
RegularAreaRect * lastMatch = 0;
|
|
while ( 1 )
|
|
{
|
|
if ( lastMatch )
|
|
lastMatch = page->findText( searchID, text, NextResult, caseSensitivity, lastMatch );
|
|
else
|
|
lastMatch = page->findText( searchID, text, FromTop, caseSensitivity );
|
|
|
|
if ( !lastMatch )
|
|
break;
|
|
|
|
// add highligh rect to the matches map
|
|
(*pageMatches)[page].append(lastMatch);
|
|
}
|
|
delete lastMatch;
|
|
|
|
QMetaObject::invokeMethod(m_parent, "doContinueAllDocumentSearch", Qt::QueuedConnection, Q_ARG(void *, pagesToNotifySet), Q_ARG(void *, pageMatches), Q_ARG(int, currentPage + 1), Q_ARG(int, searchID), Q_ARG(QString, text), Q_ARG(int, caseSensitivity), Q_ARG(QColor, color));
|
|
}
|
|
else
|
|
{
|
|
// reset cursor to previous shape
|
|
QApplication::restoreOverrideCursor();
|
|
|
|
RunningSearch * s = m_searches[searchID];
|
|
bool foundAMatch = pageMatches->count() != 0;
|
|
QMap< Page *, QVector<RegularAreaRect *> >::const_iterator it, itEnd;
|
|
it = pageMatches->begin();
|
|
itEnd = pageMatches->end();
|
|
for ( ; it != itEnd; ++it)
|
|
{
|
|
foreach(RegularAreaRect *match, it.value())
|
|
{
|
|
it.key()->d->setHighlight( searchID, match, color );
|
|
delete match;
|
|
}
|
|
s->highlightedPages.insert( it.key()->number() );
|
|
pagesToNotify->insert( it.key()->number() );
|
|
}
|
|
|
|
foreach(DocumentObserver *observer, m_observers)
|
|
observer->notifySetup( m_pagesVector, 0 );
|
|
|
|
// notify observers about highlights changes
|
|
foreach(int pageNumber, *pagesToNotify)
|
|
foreach(DocumentObserver *observer, m_observers)
|
|
observer->notifyPageChanged( pageNumber, DocumentObserver::Highlights );
|
|
|
|
if (foundAMatch) emit m_parent->searchFinished(searchID, Document::MatchFound );
|
|
else emit m_parent->searchFinished( searchID, Document::NoMatchFound );
|
|
|
|
delete pageMatches;
|
|
delete pagesToNotify;
|
|
}
|
|
}
|
|
|
|
void DocumentPrivate::doContinueGooglesDocumentSearch(void *pagesToNotifySet, void *pageMatchesMap, int currentPage, int searchID, const QString & text, int theCaseSensitivity, const QColor & color, bool matchAll)
|
|
{
|
|
typedef QPair<RegularAreaRect *, QColor> MatchColor;
|
|
QMap< Page *, QVector<MatchColor> > *pageMatches = static_cast< QMap< Page *, QVector<MatchColor> > * >(pageMatchesMap);
|
|
Qt::CaseSensitivity caseSensitivity = static_cast<Qt::CaseSensitivity>(theCaseSensitivity);
|
|
QSet< int > *pagesToNotify = static_cast< QSet< int > * >( pagesToNotifySet );
|
|
|
|
if (m_searchCancelled)
|
|
{
|
|
typedef QVector<MatchColor> MatchesVector;
|
|
|
|
QApplication::restoreOverrideCursor();
|
|
emit m_parent->searchFinished( searchID, Document::SearchCancelled );
|
|
|
|
foreach(const MatchesVector &mv, *pageMatches)
|
|
{
|
|
foreach(const MatchColor &mc, mv) delete mc.first;
|
|
}
|
|
delete pageMatches;
|
|
delete pagesToNotify;
|
|
return;
|
|
}
|
|
|
|
QStringList words = text.split( " ", QString::SkipEmptyParts );
|
|
const int wordCount = words.count();
|
|
const int hueStep = (wordCount > 1) ? (60 / (wordCount - 1)) : 60;
|
|
int baseHue, baseSat, baseVal;
|
|
color.getHsv( &baseHue, &baseSat, &baseVal );
|
|
|
|
if (currentPage < m_pagesVector.count())
|
|
{
|
|
// get page (from the first to the last)
|
|
Page *page = m_pagesVector.at(currentPage);
|
|
int pageNumber = page->number(); // redundant? is it == currentPage ?
|
|
|
|
// request search page if needed
|
|
if ( !page->hasTextPage() )
|
|
m_parent->requestTextPage( pageNumber );
|
|
|
|
// loop on a page adding highlights for all found items
|
|
bool allMatched = wordCount > 0,
|
|
anyMatched = false;
|
|
for ( int w = 0; w < wordCount; w++ )
|
|
{
|
|
const QString &word = words[ w ];
|
|
int newHue = baseHue - w * hueStep;
|
|
if ( newHue < 0 )
|
|
newHue += 360;
|
|
QColor wordColor = QColor::fromHsv( newHue, baseSat, baseVal );
|
|
RegularAreaRect * lastMatch = 0;
|
|
// add all highlights for current word
|
|
bool wordMatched = false;
|
|
while ( 1 )
|
|
{
|
|
if ( lastMatch )
|
|
lastMatch = page->findText( searchID, word, NextResult, caseSensitivity, lastMatch );
|
|
else
|
|
lastMatch = page->findText( searchID, word, FromTop, caseSensitivity);
|
|
|
|
if ( !lastMatch )
|
|
break;
|
|
|
|
// add highligh rect to the matches map
|
|
(*pageMatches)[page].append(MatchColor(lastMatch, wordColor));
|
|
wordMatched = true;
|
|
}
|
|
allMatched = allMatched && wordMatched;
|
|
anyMatched = anyMatched || wordMatched;
|
|
}
|
|
|
|
// if not all words are present in page, remove partial highlights
|
|
if ( !allMatched && matchAll )
|
|
{
|
|
QVector<MatchColor> &matches = (*pageMatches)[page];
|
|
foreach(const MatchColor &mc, matches) delete mc.first;
|
|
pageMatches->remove(page);
|
|
}
|
|
|
|
QMetaObject::invokeMethod(m_parent, "doContinueGooglesDocumentSearch", Qt::QueuedConnection, Q_ARG(void *, pagesToNotifySet), Q_ARG(void *, pageMatches), Q_ARG(int, currentPage + 1), Q_ARG(int, searchID), Q_ARG(QString, text), Q_ARG(int, caseSensitivity), Q_ARG(QColor, color), Q_ARG(bool, matchAll));
|
|
}
|
|
else
|
|
{
|
|
// reset cursor to previous shape
|
|
QApplication::restoreOverrideCursor();
|
|
|
|
RunningSearch * s = m_searches[searchID];
|
|
bool foundAMatch = pageMatches->count() != 0;
|
|
QMap< Page *, QVector<MatchColor> >::const_iterator it, itEnd;
|
|
it = pageMatches->begin();
|
|
itEnd = pageMatches->end();
|
|
for ( ; it != itEnd; ++it)
|
|
{
|
|
foreach(const MatchColor &mc, it.value())
|
|
{
|
|
it.key()->d->setHighlight( searchID, mc.first, mc.second );
|
|
delete mc.first;
|
|
}
|
|
s->highlightedPages.insert( it.key()->number() );
|
|
pagesToNotify->insert( it.key()->number() );
|
|
}
|
|
|
|
// send page lists to update observers (since some filter on bookmarks)
|
|
foreach(DocumentObserver *observer, m_observers)
|
|
observer->notifySetup( m_pagesVector, 0 );
|
|
|
|
// notify observers about highlights changes
|
|
foreach(int pageNumber, *pagesToNotify)
|
|
foreach(DocumentObserver *observer, m_observers)
|
|
observer->notifyPageChanged( pageNumber, DocumentObserver::Highlights );
|
|
|
|
if (foundAMatch) emit m_parent->searchFinished( searchID, Document::MatchFound );
|
|
else emit m_parent->searchFinished( searchID, Document::NoMatchFound );
|
|
|
|
delete pageMatches;
|
|
delete pagesToNotify;
|
|
}
|
|
}
|
|
|
|
QVariant DocumentPrivate::documentMetaData( const QString &key, const QVariant &option ) const
|
|
{
|
|
if ( key == QLatin1String( "PaperColor" ) )
|
|
{
|
|
bool giveDefault = option.toBool();
|
|
// load paper color from Settings, or use the default color (white)
|
|
// if we were told to do so
|
|
QColor color;
|
|
if ( ( Settings::renderMode() == Settings::EnumRenderMode::Paper )
|
|
&& Settings::changeColors() )
|
|
{
|
|
color = Settings::paperColor();
|
|
}
|
|
else if ( giveDefault )
|
|
{
|
|
color = Qt::white;
|
|
}
|
|
return color;
|
|
}
|
|
else if ( key == QLatin1String( "ZoomFactor" ) )
|
|
{
|
|
return Settings::zoomFactor();
|
|
}
|
|
else if ( key == QLatin1String( "TextAntialias" ) )
|
|
{
|
|
// TODO: add a configuration
|
|
// TODO: eventually read the KDE configuration
|
|
return true;
|
|
}
|
|
else if ( key == QLatin1String( "GraphicsAntialias" ) )
|
|
{
|
|
// TODO: add a configuration
|
|
return true;
|
|
}
|
|
return QVariant();
|
|
}
|
|
|
|
|
|
Document::Document( QWidget *widget )
|
|
: QObject( widget ), d( new DocumentPrivate( this ) )
|
|
{
|
|
d->m_bookmarkManager = new BookmarkManager( d );
|
|
d->m_viewportIterator = d->m_viewportHistory.insert( d->m_viewportHistory.end(), DocumentViewport() );
|
|
|
|
connect( PageController::self(), SIGNAL( rotationFinished( int ) ),
|
|
this, SLOT( rotationFinished( int ) ) );
|
|
|
|
qRegisterMetaType<Okular::FontInfo>();
|
|
}
|
|
|
|
Document::~Document()
|
|
{
|
|
// stop any audio playback
|
|
AudioPlayer::instance()->stopPlaybacks();
|
|
|
|
// delete generator, pages, and related stuff
|
|
closeDocument();
|
|
|
|
// delete the bookmark manager
|
|
delete d->m_bookmarkManager;
|
|
|
|
// delete the loaded generators
|
|
QHash< QString, GeneratorInfo >::const_iterator it = d->m_loadedGenerators.constBegin(), itEnd = d->m_loadedGenerators.constEnd();
|
|
for ( ; it != itEnd; ++it )
|
|
d->unloadGenerator( it.value() );
|
|
d->m_loadedGenerators.clear();
|
|
|
|
// delete the private structure
|
|
delete d;
|
|
}
|
|
|
|
static bool kserviceMoreThan( const KService::Ptr &s1, const KService::Ptr &s2 )
|
|
{
|
|
return s1->property( "X-KDE-Priority" ).toInt() > s2->property( "X-KDE-Priority" ).toInt();
|
|
}
|
|
|
|
bool Document::openDocument( const QString & docFile, const KUrl& url, const KMimeType::Ptr &_mime )
|
|
{
|
|
KMimeType::Ptr mime = _mime;
|
|
QByteArray filedata;
|
|
qint64 document_size = -1;
|
|
bool isstdin = url.fileName( KUrl::ObeyTrailingSlash ) == QLatin1String( "-" );
|
|
if ( !isstdin )
|
|
{
|
|
if ( mime.count() <= 0 )
|
|
return false;
|
|
|
|
// docFile is always local so we can use QFile on it
|
|
QFile fileReadTest( docFile );
|
|
if ( !fileReadTest.open( QIODevice::ReadOnly ) )
|
|
{
|
|
d->m_docFileName.clear();
|
|
return false;
|
|
}
|
|
// determine the related "xml document-info" filename
|
|
d->m_url = url;
|
|
d->m_docFileName = docFile;
|
|
if ( url.isLocalFile() )
|
|
{
|
|
QString fn = docFile.contains('/') ? docFile.section('/', -1, -1) : docFile;
|
|
document_size = fileReadTest.size();
|
|
fn = QString::number( document_size ) + '.' + fn + ".xml";
|
|
fileReadTest.close();
|
|
QString newokular = "okular/docdata/" + fn;
|
|
QString newokularfile = KStandardDirs::locateLocal( "data", newokular );
|
|
if ( !QFile::exists( newokularfile ) )
|
|
{
|
|
QString oldkpdf = "kpdf/" + fn;
|
|
QString oldkpdffile = KStandardDirs::locateLocal( "data", oldkpdf );
|
|
if ( QFile::exists( oldkpdffile ) )
|
|
{
|
|
// ### copy or move?
|
|
if ( !QFile::copy( oldkpdffile, newokularfile ) )
|
|
return false;
|
|
}
|
|
}
|
|
d->m_xmlFileName = newokularfile;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
QFile qstdin;
|
|
qstdin.open( stdin, QIODevice::ReadOnly );
|
|
filedata = qstdin.readAll();
|
|
mime = KMimeType::findByContent( filedata );
|
|
if ( !mime || mime->name() == QLatin1String( "application/octet-stream" ) )
|
|
return false;
|
|
document_size = filedata.size();
|
|
}
|
|
|
|
// 0. load Generator
|
|
// request only valid non-disabled plugins suitable for the mimetype
|
|
QString constraint("([X-KDE-Priority] > 0) and (exist Library)") ;
|
|
KService::List offers = KMimeTypeTrader::self()->query(mime->name(),"okular/Generator",constraint);
|
|
if (offers.isEmpty())
|
|
{
|
|
emit error( i18n( "Can not find a plugin which is able to handle the passed document." ), -1 );
|
|
kWarning(OkularDebug).nospace() << "No plugin for mimetype '" << mime->name() << "'.";
|
|
return false;
|
|
}
|
|
int hRank=0;
|
|
// best ranked offer search
|
|
int offercount = offers.count();
|
|
if ( offercount > 1 )
|
|
{
|
|
// sort the offers: the offers with an higher priority come before
|
|
qStableSort( offers.begin(), offers.end(), kserviceMoreThan );
|
|
|
|
if ( Settings::chooseGenerators() )
|
|
{
|
|
QStringList list;
|
|
for ( int i = 0; i < offercount; ++i )
|
|
{
|
|
list << offers.at(i)->name();
|
|
}
|
|
|
|
ChooseEngineDialog choose( list, mime, widget() );
|
|
|
|
if ( choose.exec() == QDialog::Rejected )
|
|
return false;
|
|
|
|
hRank = choose.selectedGenerator();
|
|
}
|
|
}
|
|
|
|
QString propName = offers.at(hRank)->name();
|
|
QHash< QString, GeneratorInfo >::const_iterator genIt = d->m_loadedGenerators.constFind( propName );
|
|
QString catalogName;
|
|
if ( genIt != d->m_loadedGenerators.constEnd() )
|
|
{
|
|
d->m_generator = genIt.value().generator;
|
|
catalogName = genIt.value().catalogName;
|
|
}
|
|
else
|
|
{
|
|
d->m_generator = d->loadGeneratorLibrary( offers.at(hRank) );
|
|
if ( !d->m_generator )
|
|
return false;
|
|
genIt = d->m_loadedGenerators.constFind( propName );
|
|
Q_ASSERT( genIt != d->m_loadedGenerators.constEnd() );
|
|
catalogName = genIt.value().catalogName;
|
|
}
|
|
Q_ASSERT_X( d->m_generator, "Document::load()", "null generator?!" );
|
|
|
|
if ( !catalogName.isEmpty() )
|
|
KGlobal::locale()->insertCatalog( catalogName );
|
|
|
|
d->m_generator->d_func()->m_document = d;
|
|
|
|
// connect error reporting signals
|
|
connect( d->m_generator, SIGNAL( error( const QString&, int ) ), this, SIGNAL( error( const QString&, int ) ) );
|
|
connect( d->m_generator, SIGNAL( warning( const QString&, int ) ), this, SIGNAL( warning( const QString&, int ) ) );
|
|
connect( d->m_generator, SIGNAL( notice( const QString&, int ) ), this, SIGNAL( notice( const QString&, int ) ) );
|
|
|
|
// 1. load Document (and set busy cursor while loading)
|
|
QApplication::setOverrideCursor( Qt::WaitCursor );
|
|
bool openOk = false;
|
|
if ( !isstdin )
|
|
{
|
|
openOk = d->m_generator->loadDocument( docFile, d->m_pagesVector );
|
|
}
|
|
else if ( !filedata.isEmpty() )
|
|
{
|
|
if ( d->m_generator->hasFeature( Generator::ReadRawData ) )
|
|
{
|
|
openOk = d->m_generator->loadDocumentFromData( filedata, d->m_pagesVector );
|
|
}
|
|
else
|
|
{
|
|
d->m_tempFile = new KTemporaryFile();
|
|
if ( !d->m_tempFile->open() )
|
|
{
|
|
delete d->m_tempFile;
|
|
d->m_tempFile = 0;
|
|
}
|
|
else
|
|
{
|
|
d->m_tempFile->write( filedata );
|
|
QString tmpFileName = d->m_tempFile->fileName();
|
|
d->m_tempFile->close();
|
|
openOk = d->m_generator->loadDocument( tmpFileName, d->m_pagesVector );
|
|
}
|
|
}
|
|
}
|
|
|
|
QApplication::restoreOverrideCursor();
|
|
if ( !openOk || d->m_pagesVector.size() <= 0 )
|
|
{
|
|
if ( !catalogName.isEmpty() )
|
|
KGlobal::locale()->removeCatalog( catalogName );
|
|
|
|
d->m_generator = 0;
|
|
return openOk;
|
|
}
|
|
|
|
d->m_generatorName = propName;
|
|
|
|
// 2. load Additional Data (our bookmarks and metadata) about the document
|
|
d->loadDocumentInfo();
|
|
d->m_bookmarkManager->setUrl( d->m_url );
|
|
|
|
// 3. setup observers inernal lists and data
|
|
foreachObserver( notifySetup( d->m_pagesVector, DocumentObserver::DocumentChanged ) );
|
|
|
|
// 4. set initial page (restoring the page saved in xml if loaded)
|
|
DocumentViewport loadedViewport = (*d->m_viewportIterator);
|
|
if ( loadedViewport.isValid() )
|
|
{
|
|
(*d->m_viewportIterator) = DocumentViewport();
|
|
if ( loadedViewport.pageNumber >= (int)d->m_pagesVector.size() )
|
|
loadedViewport.pageNumber = d->m_pagesVector.size() - 1;
|
|
}
|
|
else
|
|
loadedViewport.pageNumber = 0;
|
|
setViewport( loadedViewport );
|
|
|
|
// start bookmark saver timer
|
|
if ( !d->m_saveBookmarksTimer )
|
|
{
|
|
d->m_saveBookmarksTimer = new QTimer( this );
|
|
connect( d->m_saveBookmarksTimer, SIGNAL( timeout() ), this, SLOT( saveDocumentInfo() ) );
|
|
}
|
|
d->m_saveBookmarksTimer->start( 5 * 60 * 1000 );
|
|
|
|
// start memory check timer
|
|
if ( !d->m_memCheckTimer )
|
|
{
|
|
d->m_memCheckTimer = new QTimer( this );
|
|
connect( d->m_memCheckTimer, SIGNAL( timeout() ), this, SLOT( slotTimedMemoryCheck() ) );
|
|
}
|
|
d->m_memCheckTimer->start( 2000 );
|
|
|
|
if (d->m_nextDocumentViewport.isValid())
|
|
{
|
|
setViewport(d->m_nextDocumentViewport);
|
|
d->m_nextDocumentViewport = DocumentViewport();
|
|
}
|
|
|
|
AudioPlayer::instance()->d->m_currentDocument = isstdin ? KUrl() : d->m_url;
|
|
d->m_docSize = document_size;
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
KXMLGUIClient* Document::guiClient()
|
|
{
|
|
if ( d->m_generator )
|
|
{
|
|
Okular::GuiInterface * iface = qobject_cast< Okular::GuiInterface * >( d->m_generator );
|
|
if ( iface )
|
|
return iface->guiClient();
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void Document::closeDocument()
|
|
{
|
|
// check if there's anything to close...
|
|
if ( !d->m_generator )
|
|
return;
|
|
|
|
if ( d->m_fontThread )
|
|
{
|
|
disconnect( d->m_fontThread, 0, this, 0 );
|
|
d->m_fontThread->stopExtraction();
|
|
d->m_fontThread->wait();
|
|
d->m_fontThread = 0;
|
|
}
|
|
|
|
// close the current document and save document info if a document is still opened
|
|
if ( d->m_generator && d->m_pagesVector.size() > 0 )
|
|
{
|
|
d->saveDocumentInfo();
|
|
d->m_generator->closeDocument();
|
|
}
|
|
|
|
// stop timers
|
|
if ( d->m_memCheckTimer )
|
|
d->m_memCheckTimer->stop();
|
|
if ( d->m_saveBookmarksTimer )
|
|
d->m_saveBookmarksTimer->stop();
|
|
|
|
if ( d->m_generator )
|
|
{
|
|
// disconnect the generator from this document ...
|
|
d->m_generator->d_func()->m_document = 0;
|
|
// .. and this document from the generator signals
|
|
disconnect( d->m_generator, 0, this, 0 );
|
|
|
|
QHash< QString, GeneratorInfo >::const_iterator genIt = d->m_loadedGenerators.constFind( d->m_generatorName );
|
|
Q_ASSERT( genIt != d->m_loadedGenerators.constEnd() );
|
|
if ( !genIt.value().catalogName.isEmpty() && !genIt.value().config )
|
|
KGlobal::locale()->removeCatalog( genIt.value().catalogName );
|
|
}
|
|
d->m_generator = 0;
|
|
d->m_generatorName = QString();
|
|
d->m_url = KUrl();
|
|
d->m_docFileName = QString();
|
|
d->m_xmlFileName = QString();
|
|
delete d->m_tempFile;
|
|
d->m_tempFile = 0;
|
|
d->m_docSize = -1;
|
|
d->m_exportCached = false;
|
|
d->m_exportFormats.clear();
|
|
d->m_exportToText = ExportFormat();
|
|
d->m_fontsCached = false;
|
|
d->m_fontsCache.clear();
|
|
d->m_rotation = Rotation0;
|
|
// remove requests left in queue
|
|
d->m_pixmapRequestsMutex.lock();
|
|
QLinkedList< PixmapRequest * >::const_iterator sIt = d->m_pixmapRequestsStack.begin();
|
|
QLinkedList< PixmapRequest * >::const_iterator sEnd = d->m_pixmapRequestsStack.end();
|
|
for ( ; sIt != sEnd; ++sIt )
|
|
delete *sIt;
|
|
d->m_pixmapRequestsStack.clear();
|
|
d->m_pixmapRequestsMutex.unlock();
|
|
|
|
// send an empty list to observers (to free their data)
|
|
foreachObserver( notifySetup( QVector< Page * >(), DocumentObserver::DocumentChanged ) );
|
|
|
|
// delete pages and clear 'd->m_pagesVector' container
|
|
QVector< Page * >::const_iterator pIt = d->m_pagesVector.begin();
|
|
QVector< Page * >::const_iterator pEnd = d->m_pagesVector.end();
|
|
for ( ; pIt != pEnd; ++pIt )
|
|
delete *pIt;
|
|
d->m_pagesVector.clear();
|
|
|
|
// clear 'memory allocation' descriptors
|
|
QLinkedList< AllocatedPixmap * >::const_iterator aIt = d->m_allocatedPixmapsFifo.begin();
|
|
QLinkedList< AllocatedPixmap * >::const_iterator aEnd = d->m_allocatedPixmapsFifo.end();
|
|
for ( ; aIt != aEnd; ++aIt )
|
|
delete *aIt;
|
|
d->m_allocatedPixmapsFifo.clear();
|
|
|
|
// clear 'running searches' descriptors
|
|
QMap< int, RunningSearch * >::const_iterator rIt = d->m_searches.begin();
|
|
QMap< int, RunningSearch * >::const_iterator rEnd = d->m_searches.end();
|
|
for ( ; rIt != rEnd; ++rIt )
|
|
delete *rIt;
|
|
d->m_searches.clear();
|
|
|
|
// clear the visible areas and notify the observers
|
|
QVector< VisiblePageRect * >::const_iterator vIt = d->m_pageRects.begin();
|
|
QVector< VisiblePageRect * >::const_iterator vEnd = d->m_pageRects.end();
|
|
for ( ; vIt != vEnd; ++vIt )
|
|
delete *vIt;
|
|
d->m_pageRects.clear();
|
|
foreachObserver( notifyVisibleRectsChanged() );
|
|
|
|
// reset internal variables
|
|
|
|
d->m_viewportHistory.clear();
|
|
d->m_viewportHistory.append( DocumentViewport() );
|
|
d->m_viewportIterator = d->m_viewportHistory.begin();
|
|
d->m_allocatedPixmapsTotalMemory = 0;
|
|
d->m_pageSize = PageSize();
|
|
d->m_pageSizes.clear();
|
|
AudioPlayer::instance()->d->m_currentDocument = KUrl();
|
|
}
|
|
|
|
void Document::addObserver( DocumentObserver * pObserver )
|
|
{
|
|
// keep the pointer to the observer in a map
|
|
d->m_observers.insert( pObserver->observerId(), pObserver );
|
|
|
|
// if the observer is added while a document is already opened, tell it
|
|
if ( !d->m_pagesVector.isEmpty() )
|
|
{
|
|
pObserver->notifySetup( d->m_pagesVector, DocumentObserver::DocumentChanged );
|
|
pObserver->notifyViewportChanged( false /*disables smoothMove*/ );
|
|
}
|
|
}
|
|
|
|
void Document::removeObserver( DocumentObserver * pObserver )
|
|
{
|
|
// remove observer from the map. it won't receive notifications anymore
|
|
if ( d->m_observers.contains( pObserver->observerId() ) )
|
|
{
|
|
// free observer's pixmap data
|
|
int observerId = pObserver->observerId();
|
|
QVector<Page*>::const_iterator it = d->m_pagesVector.begin(), end = d->m_pagesVector.end();
|
|
for ( ; it != end; ++it )
|
|
(*it)->deletePixmap( observerId );
|
|
|
|
// [MEM] free observer's allocation descriptors
|
|
QLinkedList< AllocatedPixmap * >::iterator aIt = d->m_allocatedPixmapsFifo.begin();
|
|
QLinkedList< AllocatedPixmap * >::iterator aEnd = d->m_allocatedPixmapsFifo.end();
|
|
while ( aIt != aEnd )
|
|
{
|
|
AllocatedPixmap * p = *aIt;
|
|
if ( p->id == observerId )
|
|
{
|
|
aIt = d->m_allocatedPixmapsFifo.erase( aIt );
|
|
delete p;
|
|
}
|
|
else
|
|
++aIt;
|
|
}
|
|
|
|
// delete observer entry from the map
|
|
d->m_observers.remove( observerId );
|
|
}
|
|
}
|
|
|
|
void Document::reparseConfig()
|
|
{
|
|
// reparse generator config and if something changed clear Pages
|
|
bool configchanged = false;
|
|
if ( d->m_generator )
|
|
{
|
|
Okular::ConfigInterface * iface = qobject_cast< Okular::ConfigInterface * >( d->m_generator );
|
|
if ( iface )
|
|
configchanged = iface->reparseConfig();
|
|
}
|
|
if ( configchanged )
|
|
{
|
|
// invalidate pixmaps
|
|
QVector<Page*>::const_iterator it = d->m_pagesVector.begin(), end = d->m_pagesVector.end();
|
|
for ( ; it != end; ++it ) {
|
|
(*it)->deletePixmaps();
|
|
}
|
|
|
|
// [MEM] remove allocation descriptors
|
|
QLinkedList< AllocatedPixmap * >::const_iterator aIt = d->m_allocatedPixmapsFifo.begin();
|
|
QLinkedList< AllocatedPixmap * >::const_iterator aEnd = d->m_allocatedPixmapsFifo.end();
|
|
for ( ; aIt != aEnd; ++aIt )
|
|
delete *aIt;
|
|
d->m_allocatedPixmapsFifo.clear();
|
|
d->m_allocatedPixmapsTotalMemory = 0;
|
|
|
|
// send reload signals to observers
|
|
foreachObserver( notifyContentsCleared( DocumentObserver::Pixmap ) );
|
|
}
|
|
|
|
// free memory if in 'low' profile
|
|
if ( Settings::memoryLevel() == Settings::EnumMemoryLevel::Low &&
|
|
!d->m_allocatedPixmapsFifo.isEmpty() && !d->m_pagesVector.isEmpty() )
|
|
d->cleanupPixmapMemory();
|
|
}
|
|
|
|
|
|
QWidget *Document::widget() const
|
|
{
|
|
return parent() ? static_cast< QWidget * >( parent() ) : 0;
|
|
}
|
|
|
|
bool Document::isOpened() const
|
|
{
|
|
return d->m_generator;
|
|
}
|
|
|
|
bool Document::canConfigurePrinter( ) const
|
|
{
|
|
if ( d->m_generator )
|
|
{
|
|
Okular::PrintInterface * iface = qobject_cast< Okular::PrintInterface * >( d->m_generator );
|
|
return iface ? true : false;
|
|
}
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
const DocumentInfo * Document::documentInfo() const
|
|
{
|
|
if ( d->m_generator )
|
|
{
|
|
const DocumentInfo *infoConst = d->m_generator->generateDocumentInfo();
|
|
if ( !infoConst )
|
|
return 0;
|
|
|
|
DocumentInfo *info = const_cast< DocumentInfo * >( infoConst );
|
|
QString pagesSize = d->pagesSizeString();
|
|
if ( d->m_docSize != -1 )
|
|
{
|
|
QString sizeString = KGlobal::locale()->formatByteSize( d->m_docSize );
|
|
info->set( "documentSize", sizeString, i18n( "File Size" ) );
|
|
}
|
|
if (!pagesSize.isEmpty())
|
|
{
|
|
info->set( "pagesSize", pagesSize, i18n("Pages Size") );
|
|
}
|
|
return info;
|
|
}
|
|
else return NULL;
|
|
}
|
|
|
|
const DocumentSynopsis * Document::documentSynopsis() const
|
|
{
|
|
return d->m_generator ? d->m_generator->generateDocumentSynopsis() : NULL;
|
|
}
|
|
|
|
void Document::startFontReading()
|
|
{
|
|
if ( !d->m_generator || !d->m_generator->hasFeature( Generator::FontInfo ) || d->m_fontThread )
|
|
return;
|
|
|
|
if ( d->m_fontsCached )
|
|
{
|
|
// in case we have cached fonts, simulate a reading
|
|
// this way the API is the same, and users no need to care about the
|
|
// internal caching
|
|
for ( int i = 0; i < d->m_fontsCache.count(); ++i )
|
|
{
|
|
emit gotFont( d->m_fontsCache.at( i ) );
|
|
emit fontReadingProgress( i / pages() );
|
|
}
|
|
emit fontReadingEnded();
|
|
return;
|
|
}
|
|
|
|
d->m_fontThread = new FontExtractionThread( d->m_generator, pages() );
|
|
connect( d->m_fontThread, SIGNAL( gotFont( const Okular::FontInfo& ) ), this, SLOT( fontReadingGotFont( const Okular::FontInfo& ) ) );
|
|
connect( d->m_fontThread, SIGNAL( progress( int ) ), this, SLOT( fontReadingProgress( int ) ) );
|
|
|
|
d->m_fontThread->startExtraction( /*d->m_generator->hasFeature( Generator::Threaded )*/true );
|
|
}
|
|
|
|
void Document::stopFontReading()
|
|
{
|
|
if ( !d->m_fontThread )
|
|
return;
|
|
|
|
disconnect( d->m_fontThread, 0, this, 0 );
|
|
d->m_fontThread->stopExtraction();
|
|
d->m_fontThread = 0;
|
|
d->m_fontsCache.clear();
|
|
}
|
|
|
|
bool Document::canProvideFontInformation() const
|
|
{
|
|
return d->m_generator ? d->m_generator->hasFeature( Generator::FontInfo ) : false;
|
|
}
|
|
|
|
const QList<EmbeddedFile*> *Document::embeddedFiles() const
|
|
{
|
|
return d->m_generator ? d->m_generator->embeddedFiles() : NULL;
|
|
}
|
|
|
|
const Page * Document::page( int n ) const
|
|
{
|
|
return ( n < d->m_pagesVector.count() ) ? d->m_pagesVector.at(n) : 0;
|
|
}
|
|
|
|
const DocumentViewport & Document::viewport() const
|
|
{
|
|
return (*d->m_viewportIterator);
|
|
}
|
|
|
|
const QVector< VisiblePageRect * > & Document::visiblePageRects() const
|
|
{
|
|
return d->m_pageRects;
|
|
}
|
|
|
|
void Document::setVisiblePageRects( const QVector< VisiblePageRect * > & visiblePageRects, int excludeId )
|
|
{
|
|
QVector< VisiblePageRect * >::const_iterator vIt = d->m_pageRects.begin();
|
|
QVector< VisiblePageRect * >::const_iterator vEnd = d->m_pageRects.end();
|
|
for ( ; vIt != vEnd; ++vIt )
|
|
delete *vIt;
|
|
d->m_pageRects = visiblePageRects;
|
|
// notify change to all other (different from id) observers
|
|
QMap< int, DocumentObserver * >::const_iterator it = d->m_observers.begin(), end = d->m_observers.end();
|
|
for ( ; it != end ; ++ it )
|
|
if ( it.key() != excludeId )
|
|
(*it)->notifyVisibleRectsChanged();
|
|
}
|
|
|
|
uint Document::currentPage() const
|
|
{
|
|
return (*d->m_viewportIterator).pageNumber;
|
|
}
|
|
|
|
uint Document::pages() const
|
|
{
|
|
return d->m_pagesVector.size();
|
|
}
|
|
|
|
KUrl Document::currentDocument() const
|
|
{
|
|
return d->m_url;
|
|
}
|
|
|
|
bool Document::isAllowed( Permission action ) const
|
|
{
|
|
#if !OKULAR_FORCE_DRM
|
|
if ( KAuthorized::authorize( "skip_drm" ) && !Okular::Settings::obeyDRM() )
|
|
return true;
|
|
#endif
|
|
|
|
return d->m_generator ? d->m_generator->isAllowed( action ) : false;
|
|
}
|
|
|
|
bool Document::supportsSearching() const
|
|
{
|
|
return d->m_generator ? d->m_generator->hasFeature( Generator::TextExtraction ) : false;
|
|
}
|
|
|
|
bool Document::supportsPageSizes() const
|
|
{
|
|
return d->m_generator ? d->m_generator->hasFeature( Generator::PageSizes ) : false;
|
|
}
|
|
|
|
PageSize::List Document::pageSizes() const
|
|
{
|
|
if ( d->m_generator )
|
|
{
|
|
if ( d->m_pageSizes.isEmpty() )
|
|
d->m_pageSizes = d->m_generator->pageSizes();
|
|
return d->m_pageSizes;
|
|
}
|
|
return PageSize::List();
|
|
}
|
|
|
|
bool Document::canExportToText() const
|
|
{
|
|
if ( !d->m_generator )
|
|
return false;
|
|
|
|
d->cacheExportFormats();
|
|
return !d->m_exportToText.isNull();
|
|
}
|
|
|
|
bool Document::exportToText( const QString& fileName ) const
|
|
{
|
|
if ( !d->m_generator )
|
|
return false;
|
|
|
|
d->cacheExportFormats();
|
|
if ( d->m_exportToText.isNull() )
|
|
return false;
|
|
|
|
return d->m_generator->exportTo( fileName, d->m_exportToText );
|
|
}
|
|
|
|
ExportFormat::List Document::exportFormats() const
|
|
{
|
|
if ( !d->m_generator )
|
|
return ExportFormat::List();
|
|
|
|
d->cacheExportFormats();
|
|
return d->m_exportFormats;
|
|
}
|
|
|
|
bool Document::exportTo( const QString& fileName, const ExportFormat& format ) const
|
|
{
|
|
return d->m_generator ? d->m_generator->exportTo( fileName, format ) : false;
|
|
}
|
|
|
|
bool Document::historyAtBegin() const
|
|
{
|
|
return d->m_viewportIterator == d->m_viewportHistory.begin();
|
|
}
|
|
|
|
bool Document::historyAtEnd() const
|
|
{
|
|
return d->m_viewportIterator == --(d->m_viewportHistory.end());
|
|
}
|
|
|
|
QVariant Document::metaData( const QString & key, const QVariant & option ) const
|
|
{
|
|
return d->m_generator ? d->m_generator->metaData( key, option ) : QVariant();
|
|
}
|
|
|
|
Rotation Document::rotation() const
|
|
{
|
|
return d->m_rotation;
|
|
}
|
|
|
|
QSizeF Document::allPagesSize() const
|
|
{
|
|
bool allPagesSameSize = true;
|
|
QSizeF size;
|
|
for (int i = 0; allPagesSameSize && i < d->m_pagesVector.count(); ++i)
|
|
{
|
|
const Page *p = d->m_pagesVector.at(i);
|
|
if (i == 0) size = QSizeF(p->width(), p->height());
|
|
else
|
|
{
|
|
allPagesSameSize = (size == QSizeF(p->width(), p->height()));
|
|
}
|
|
}
|
|
if (allPagesSameSize) return size;
|
|
else return QSizeF();
|
|
}
|
|
|
|
QString Document::pageSizeString(int page) const
|
|
{
|
|
if (d->m_generator)
|
|
{
|
|
if (d->m_generator->pagesSizeMetric() != Generator::None)
|
|
{
|
|
const Page *p = d->m_pagesVector.at( page );
|
|
return d->localizedSize(QSizeF(p->width(), p->height()));
|
|
}
|
|
}
|
|
return QString();
|
|
}
|
|
|
|
void Document::requestPixmaps( const QLinkedList< PixmapRequest * > & requests )
|
|
{
|
|
if ( requests.isEmpty() )
|
|
return;
|
|
|
|
if ( !d->m_generator )
|
|
{
|
|
// delete requests..
|
|
QLinkedList< PixmapRequest * >::const_iterator rIt = requests.begin(), rEnd = requests.end();
|
|
for ( ; rIt != rEnd; ++rIt )
|
|
delete *rIt;
|
|
// ..and return
|
|
return;
|
|
}
|
|
|
|
// 1. [CLEAN STACK] remove previous requests of requesterID
|
|
int requesterID = requests.first()->id();
|
|
d->m_pixmapRequestsMutex.lock();
|
|
QLinkedList< PixmapRequest * >::iterator sIt = d->m_pixmapRequestsStack.begin(), sEnd = d->m_pixmapRequestsStack.end();
|
|
while ( sIt != sEnd )
|
|
{
|
|
if ( (*sIt)->id() == requesterID )
|
|
{
|
|
// delete request and remove it from stack
|
|
delete *sIt;
|
|
sIt = d->m_pixmapRequestsStack.erase( sIt );
|
|
}
|
|
else
|
|
++sIt;
|
|
}
|
|
|
|
// 2. [ADD TO STACK] add requests to stack
|
|
bool threadingDisabled = !Settings::enableThreading();
|
|
QLinkedList< PixmapRequest * >::const_iterator rIt = requests.begin(), rEnd = requests.end();
|
|
for ( ; rIt != rEnd; ++rIt )
|
|
{
|
|
// set the 'page field' (see PixmapRequest) and check if it is valid
|
|
PixmapRequest * request = *rIt;
|
|
kDebug(OkularDebug).nospace() << "request id=" << request->id() << " " <<request->width() << "x" << request->height() << "@" << request->pageNumber();
|
|
if ( d->m_pagesVector.value( request->pageNumber() ) == 0 )
|
|
{
|
|
// skip requests referencing an invalid page (must not happen)
|
|
delete request;
|
|
continue;
|
|
}
|
|
|
|
request->d->mPage = d->m_pagesVector.value( request->pageNumber() );
|
|
|
|
if ( !request->asynchronous() )
|
|
request->d->mPriority = 0;
|
|
|
|
if ( request->asynchronous() && threadingDisabled )
|
|
request->d->mAsynchronous = false;
|
|
|
|
// add request to the 'stack' at the right place
|
|
if ( !request->priority() )
|
|
// add priority zero requests to the top of the stack
|
|
d->m_pixmapRequestsStack.append( request );
|
|
else
|
|
{
|
|
// insert in stack sorted by priority
|
|
sIt = d->m_pixmapRequestsStack.begin();
|
|
sEnd = d->m_pixmapRequestsStack.end();
|
|
while ( sIt != sEnd && (*sIt)->priority() > request->priority() )
|
|
++sIt;
|
|
d->m_pixmapRequestsStack.insert( sIt, request );
|
|
}
|
|
}
|
|
d->m_pixmapRequestsMutex.unlock();
|
|
|
|
// 3. [START FIRST GENERATION] if <NO>generator is ready, start a new generation,
|
|
// or else (if gen is running) it will be started when the new contents will
|
|
//come from generator (in requestDone())</NO>
|
|
// all handling of requests put into sendGeneratorRequest
|
|
// if ( generator->canRequestPixmap() )
|
|
d->sendGeneratorRequest();
|
|
}
|
|
|
|
void Document::requestTextPage( uint page )
|
|
{
|
|
Page * kp = d->m_pagesVector[ page ];
|
|
if ( !d->m_generator || !kp )
|
|
return;
|
|
|
|
// Memory management for TextPages
|
|
|
|
d->m_generator->generateTextPage( kp );
|
|
}
|
|
|
|
void Document::addPageAnnotation( int page, Annotation * annotation )
|
|
{
|
|
// find out the page to attach annotation
|
|
Page * kp = d->m_pagesVector[ page ];
|
|
if ( !d->m_generator || !kp )
|
|
return;
|
|
|
|
// the annotation belongs already to a page
|
|
if ( annotation->d_ptr->m_page )
|
|
return;
|
|
|
|
// add annotation to the page
|
|
kp->addAnnotation( annotation );
|
|
|
|
// notify observers about the change
|
|
foreachObserver( notifyPageChanged( page, DocumentObserver::Annotations ) );
|
|
}
|
|
|
|
void Document::modifyPageAnnotation( int page, Annotation * newannotation )
|
|
{
|
|
//TODO: modify annotations
|
|
|
|
// find out the page
|
|
Page * kp = d->m_pagesVector[ page ];
|
|
if ( !d->m_generator || !kp )
|
|
return;
|
|
|
|
kp->d->modifyAnnotation( newannotation );
|
|
// notify observers about the change
|
|
foreachObserver( notifyPageChanged( page, DocumentObserver::Annotations ) );
|
|
}
|
|
|
|
|
|
void Document::removePageAnnotation( int page, Annotation * annotation )
|
|
{
|
|
// find out the page
|
|
Page * kp = d->m_pagesVector[ page ];
|
|
if ( !d->m_generator || !kp )
|
|
return;
|
|
|
|
// try to remove the annotation
|
|
if ( kp->removeAnnotation( annotation ) )
|
|
{
|
|
// in case of success, notify observers about the change
|
|
foreachObserver( notifyPageChanged( page, DocumentObserver::Annotations ) );
|
|
}
|
|
}
|
|
|
|
void Document::removePageAnnotations( int page, const QList< Annotation * > &annotations )
|
|
{
|
|
// find out the page
|
|
Page * kp = d->m_pagesVector[ page ];
|
|
if ( !d->m_generator || !kp )
|
|
return;
|
|
|
|
bool changed = false;
|
|
foreach ( Annotation * annotation, annotations )
|
|
{
|
|
// try to remove the annotation
|
|
if ( kp->removeAnnotation( annotation ) )
|
|
{
|
|
changed = true;
|
|
}
|
|
}
|
|
if ( changed )
|
|
{
|
|
// in case we removed even only one annotation, notify observers about the change
|
|
foreachObserver( notifyPageChanged( page, DocumentObserver::Annotations ) );
|
|
}
|
|
}
|
|
|
|
void Document::setPageTextSelection( int page, RegularAreaRect * rect, const QColor & color )
|
|
{
|
|
Page * kp = d->m_pagesVector[ page ];
|
|
if ( !d->m_generator || !kp )
|
|
return;
|
|
|
|
// add or remove the selection basing whether rect is null or not
|
|
if ( rect )
|
|
kp->d->setTextSelections( rect, color );
|
|
else
|
|
kp->d->deleteTextSelections();
|
|
|
|
// notify observers about the change
|
|
foreachObserver( notifyPageChanged( page, DocumentObserver::TextSelection ) );
|
|
}
|
|
|
|
/* REFERENCE IMPLEMENTATION: better calling setViewport from other code
|
|
void Document::setNextPage()
|
|
{
|
|
// advance page and set viewport on observers
|
|
if ( (*d->m_viewportIterator).pageNumber < (int)d->m_pagesVector.count() - 1 )
|
|
setViewport( DocumentViewport( (*d->m_viewportIterator).pageNumber + 1 ) );
|
|
}
|
|
|
|
void Document::setPrevPage()
|
|
{
|
|
// go to previous page and set viewport on observers
|
|
if ( (*d->m_viewportIterator).pageNumber > 0 )
|
|
setViewport( DocumentViewport( (*d->m_viewportIterator).pageNumber - 1 ) );
|
|
}
|
|
*/
|
|
void Document::setViewportPage( int page, int excludeId, bool smoothMove )
|
|
{
|
|
// clamp page in range [0 ... numPages-1]
|
|
if ( page < 0 )
|
|
page = 0;
|
|
else if ( page > (int)d->m_pagesVector.count() )
|
|
page = d->m_pagesVector.count() - 1;
|
|
|
|
// make a viewport from the page and broadcast it
|
|
setViewport( DocumentViewport( page ), excludeId, smoothMove );
|
|
}
|
|
|
|
void Document::setViewport( const DocumentViewport & viewport, int excludeId, bool smoothMove )
|
|
{
|
|
// if already broadcasted, don't redo it
|
|
DocumentViewport & oldViewport = *d->m_viewportIterator;
|
|
// disabled by enrico on 2005-03-18 (less debug output)
|
|
//if ( viewport == oldViewport )
|
|
// kDebug(OkularDebug) << "setViewport with the same viewport.";
|
|
|
|
// set internal viewport taking care of history
|
|
if ( oldViewport.pageNumber == viewport.pageNumber || !oldViewport.isValid() )
|
|
{
|
|
// if page is unchanged save the viewport at current position in queue
|
|
oldViewport = viewport;
|
|
}
|
|
else
|
|
{
|
|
// remove elements after viewportIterator in queue
|
|
d->m_viewportHistory.erase( ++d->m_viewportIterator, d->m_viewportHistory.end() );
|
|
|
|
// keep the list to a reasonable size by removing head when needed
|
|
if ( d->m_viewportHistory.count() >= OKULAR_HISTORY_MAXSTEPS )
|
|
d->m_viewportHistory.pop_front();
|
|
|
|
// add the item at the end of the queue
|
|
d->m_viewportIterator = d->m_viewportHistory.insert( d->m_viewportHistory.end(), viewport );
|
|
}
|
|
|
|
// notify change to all other (different from id) observers
|
|
QMap< int, DocumentObserver * >::const_iterator it = d->m_observers.begin(), end = d->m_observers.end();
|
|
for ( ; it != end ; ++ it )
|
|
if ( it.key() != excludeId )
|
|
(*it)->notifyViewportChanged( smoothMove );
|
|
|
|
// [MEM] raise position of currently viewed page in allocation queue
|
|
if ( d->m_allocatedPixmapsFifo.count() > 1 )
|
|
{
|
|
const int page = viewport.pageNumber;
|
|
QLinkedList< AllocatedPixmap * > viewportPixmaps;
|
|
QLinkedList< AllocatedPixmap * >::iterator aIt = d->m_allocatedPixmapsFifo.begin();
|
|
QLinkedList< AllocatedPixmap * >::iterator aEnd = d->m_allocatedPixmapsFifo.end();
|
|
while ( aIt != aEnd )
|
|
{
|
|
if ( (*aIt)->page == page )
|
|
{
|
|
viewportPixmaps.append( *aIt );
|
|
aIt = d->m_allocatedPixmapsFifo.erase( aIt );
|
|
continue;
|
|
}
|
|
++aIt;
|
|
}
|
|
if ( !viewportPixmaps.isEmpty() )
|
|
d->m_allocatedPixmapsFifo += viewportPixmaps;
|
|
}
|
|
}
|
|
|
|
void Document::setZoom(int factor, int excludeId)
|
|
{
|
|
// notify change to all other (different from id) observers
|
|
QMap< int, DocumentObserver * >::const_iterator it = d->m_observers.begin(), end = d->m_observers.end();
|
|
for ( ; it != end ; ++ it )
|
|
if ( it.key() != excludeId )
|
|
(*it)->notifyZoom( factor );
|
|
}
|
|
|
|
void Document::setPrevViewport()
|
|
// restore viewport from the history
|
|
{
|
|
if ( d->m_viewportIterator != d->m_viewportHistory.begin() )
|
|
{
|
|
// restore previous viewport and notify it to observers
|
|
--d->m_viewportIterator;
|
|
foreachObserver( notifyViewportChanged( true ) );
|
|
}
|
|
}
|
|
|
|
void Document::setNextViewport()
|
|
// restore next viewport from the history
|
|
{
|
|
QLinkedList< DocumentViewport >::const_iterator nextIterator = d->m_viewportIterator;
|
|
++nextIterator;
|
|
if ( nextIterator != d->m_viewportHistory.end() )
|
|
{
|
|
// restore next viewport and notify it to observers
|
|
++d->m_viewportIterator;
|
|
foreachObserver( notifyViewportChanged( true ) );
|
|
}
|
|
}
|
|
|
|
void Document::setNextDocumentViewport( const DocumentViewport & viewport )
|
|
{
|
|
d->m_nextDocumentViewport = viewport;
|
|
}
|
|
|
|
void Document::searchText( int searchID, const QString & text, bool fromStart, Qt::CaseSensitivity caseSensitivity,
|
|
SearchType type, bool moveViewport, const QColor & color, bool noDialogs )
|
|
{
|
|
d->m_searchCancelled = false;
|
|
|
|
// safety checks: don't perform searches on empty or unsearchable docs
|
|
if ( !d->m_generator || !d->m_generator->hasFeature( Generator::TextExtraction ) || d->m_pagesVector.isEmpty() )
|
|
{
|
|
emit searchFinished( searchID, NoMatchFound );
|
|
return;
|
|
}
|
|
|
|
if ( !noDialogs )
|
|
{
|
|
KDialog *searchDialog = new KDialog(widget());
|
|
searchDialog->setCaption( i18n("Search in progress...") );
|
|
searchDialog->setButtons( KDialog::Cancel );
|
|
QLabel *searchLabel = new QLabel(i18n("Searching for %1", text), searchDialog);
|
|
searchDialog->setMainWidget( searchLabel );
|
|
|
|
QTimer::singleShot(500, searchDialog, SLOT(show()));
|
|
connect(this, SIGNAL( searchFinished(int, Okular::Document::SearchStatus) ), searchDialog, SLOT(deleteLater()));
|
|
connect(searchDialog, SIGNAL( finished() ), this, SLOT(cancelSearch()));
|
|
}
|
|
|
|
// if searchID search not recorded, create new descriptor and init params
|
|
QMap< int, RunningSearch * >::iterator searchIt = d->m_searches.find( searchID );
|
|
if ( searchIt == d->m_searches.end() )
|
|
{
|
|
RunningSearch * search = new RunningSearch();
|
|
search->continueOnPage = -1;
|
|
searchIt = d->m_searches.insert( searchID, search );
|
|
}
|
|
if (d->m_lastSearchID != searchID)
|
|
{
|
|
resetSearch(d->m_lastSearchID);
|
|
}
|
|
d->m_lastSearchID = searchID;
|
|
RunningSearch * s = *searchIt;
|
|
|
|
// update search stucture
|
|
bool newText = text != s->cachedString;
|
|
s->cachedString = text;
|
|
s->cachedType = type;
|
|
s->cachedCaseSensitivity = caseSensitivity;
|
|
s->cachedViewportMove = moveViewport;
|
|
s->cachedNoDialogs = noDialogs;
|
|
s->cachedColor = color;
|
|
|
|
// global data for search
|
|
QSet< int > *pagesToNotify = new QSet< int >;
|
|
|
|
// remove highlights from pages and queue them for notifying changes
|
|
*pagesToNotify += s->highlightedPages;
|
|
foreach(int pageNumber, s->highlightedPages)
|
|
d->m_pagesVector.at(pageNumber)->d->deleteHighlights( searchID );
|
|
s->highlightedPages.clear();
|
|
|
|
// set hourglass cursor
|
|
QApplication::setOverrideCursor( Qt::WaitCursor );
|
|
|
|
// 1. ALLDOC - proces all document marking pages
|
|
if ( type == AllDocument )
|
|
{
|
|
QMap< Page *, QVector<RegularAreaRect *> > *pageMatches = new QMap< Page *, QVector<RegularAreaRect *> >;
|
|
|
|
// search and highlight 'text' (as a solid phrase) on all pages
|
|
QMetaObject::invokeMethod(this, "doContinueAllDocumentSearch", Qt::QueuedConnection, Q_ARG(void *, pagesToNotify), Q_ARG(void *, pageMatches), Q_ARG(int, 0), Q_ARG(int, searchID), Q_ARG(QString, text), Q_ARG(int, caseSensitivity), Q_ARG(QColor, color));
|
|
}
|
|
// 2. NEXTMATCH - find next matching item (or start from top)
|
|
else if ( type == NextMatch )
|
|
{
|
|
// find out from where to start/resume search from
|
|
int viewportPage = (*d->m_viewportIterator).pageNumber;
|
|
int currentPage = fromStart ? 0 : ((s->continueOnPage != -1) ? s->continueOnPage : viewportPage);
|
|
Page * lastPage = fromStart ? 0 : d->m_pagesVector[ currentPage ];
|
|
|
|
// continue checking last TextPage first (if it is the current page)
|
|
RegularAreaRect * match = 0;
|
|
if ( lastPage && lastPage->number() == s->continueOnPage )
|
|
{
|
|
if ( newText )
|
|
match = lastPage->findText( searchID, text, FromTop, caseSensitivity );
|
|
else
|
|
match = lastPage->findText( searchID, text, NextResult, caseSensitivity, &s->continueOnMatch );
|
|
if ( !match )
|
|
currentPage++;
|
|
}
|
|
|
|
QMetaObject::invokeMethod(this, "doContinueNextMatchSearch", Qt::QueuedConnection, Q_ARG(void *, pagesToNotify), Q_ARG(void *, match), Q_ARG(int, currentPage), Q_ARG(int, searchID), Q_ARG(QString, text), Q_ARG(int, caseSensitivity), Q_ARG(bool, moveViewport), Q_ARG(QColor, color), Q_ARG(bool, noDialogs), Q_ARG(int, 1));
|
|
}
|
|
// 3. PREVMATCH //TODO
|
|
else if ( type == PreviousMatch )
|
|
{
|
|
}
|
|
// 4. GOOGLE* - process all document marking pages
|
|
else if ( type == GoogleAll || type == GoogleAny )
|
|
{
|
|
bool matchAll = type == GoogleAll;
|
|
|
|
QMap< Page *, QVector< QPair<RegularAreaRect *, QColor> > > *pageMatches = new QMap< Page *, QVector<QPair<RegularAreaRect *, QColor> > >;
|
|
|
|
// search and highlight every word in 'text' on all pages
|
|
QMetaObject::invokeMethod(this, "doContinueGooglesDocumentSearch", Qt::QueuedConnection, Q_ARG(void *, pagesToNotify), Q_ARG(void *, pageMatches), Q_ARG(int, 0), Q_ARG(int, searchID), Q_ARG(QString, text), Q_ARG(int, caseSensitivity), Q_ARG(QColor, color), Q_ARG(bool, matchAll));
|
|
}
|
|
}
|
|
|
|
void Document::continueSearch( int searchID )
|
|
{
|
|
// check if searchID is present in runningSearches
|
|
QMap< int, RunningSearch * >::const_iterator it = d->m_searches.constFind( searchID );
|
|
if ( it == d->m_searches.constEnd() )
|
|
{
|
|
emit searchFinished( searchID, NoMatchFound );
|
|
return;
|
|
}
|
|
|
|
// start search with cached parameters from last search by searchID
|
|
RunningSearch * p = *it;
|
|
searchText( searchID, p->cachedString, false, p->cachedCaseSensitivity,
|
|
p->cachedType, p->cachedViewportMove, p->cachedColor,
|
|
p->cachedNoDialogs );
|
|
}
|
|
|
|
void Document::resetSearch( int searchID )
|
|
{
|
|
// check if searchID is present in runningSearches
|
|
QMap< int, RunningSearch * >::iterator searchIt = d->m_searches.find( searchID );
|
|
if ( searchIt == d->m_searches.end() )
|
|
return;
|
|
|
|
// get previous parameters for search
|
|
RunningSearch * s = *searchIt;
|
|
|
|
// unhighlight pages and inform observers about that
|
|
foreach(int pageNumber, s->highlightedPages)
|
|
{
|
|
d->m_pagesVector.at(pageNumber)->d->deleteHighlights( searchID );
|
|
foreachObserver( notifyPageChanged( pageNumber, DocumentObserver::Highlights ) );
|
|
}
|
|
|
|
// send the setup signal too (to update views that filter on matches)
|
|
foreachObserver( notifySetup( d->m_pagesVector, 0 ) );
|
|
|
|
// remove serch from the runningSearches list and delete it
|
|
d->m_searches.erase( searchIt );
|
|
delete s;
|
|
}
|
|
|
|
void Document::cancelSearch()
|
|
{
|
|
d->m_searchCancelled = true;
|
|
}
|
|
|
|
BookmarkManager * Document::bookmarkManager() const
|
|
{
|
|
return d->m_bookmarkManager;
|
|
}
|
|
|
|
QList<int> Document::bookmarkedPageList() const
|
|
{
|
|
QList<int> list;
|
|
uint docPages = pages();
|
|
|
|
//pages are 0-indexed internally, but 1-indexed externally
|
|
for ( uint i = 0; i < docPages; i++ )
|
|
{
|
|
if ( bookmarkManager()->isBookmarked( i ) )
|
|
{
|
|
list << i + 1;
|
|
}
|
|
}
|
|
return list;
|
|
}
|
|
|
|
QString Document::bookmarkedPageRange() const
|
|
{
|
|
// Code formerly in Part::slotPrint()
|
|
// range detecting
|
|
QString range;
|
|
uint docPages = pages();
|
|
int startId = -1;
|
|
int endId = -1;
|
|
|
|
for ( uint i = 0; i < docPages; ++i )
|
|
{
|
|
if ( bookmarkManager()->isBookmarked( i ) )
|
|
{
|
|
if ( startId < 0 )
|
|
startId = i;
|
|
if ( endId < 0 )
|
|
endId = startId;
|
|
else
|
|
++endId;
|
|
}
|
|
else if ( startId >= 0 && endId >= 0 )
|
|
{
|
|
if ( !range.isEmpty() )
|
|
range += ',';
|
|
|
|
if ( endId - startId > 0 )
|
|
range += QString( "%1-%2" ).arg( startId + 1 ).arg( endId + 1 );
|
|
else
|
|
range += QString::number( startId + 1 );
|
|
startId = -1;
|
|
endId = -1;
|
|
}
|
|
}
|
|
if ( startId >= 0 && endId >= 0 )
|
|
{
|
|
if ( !range.isEmpty() )
|
|
range += ',';
|
|
|
|
if ( endId - startId > 0 )
|
|
range += QString( "%1-%2" ).arg( startId + 1 ).arg( endId + 1 );
|
|
else
|
|
range += QString::number( startId + 1 );
|
|
}
|
|
return range;
|
|
}
|
|
|
|
void Document::processAction( const Action * action )
|
|
{
|
|
if ( !action )
|
|
return;
|
|
|
|
switch( action->actionType() )
|
|
{
|
|
case Action::Goto: {
|
|
const GotoAction * go = static_cast< const GotoAction * >( action );
|
|
d->m_nextDocumentViewport = go->destViewport();
|
|
|
|
// Explanation of why d->m_nextDocumentViewport is needed:
|
|
// all openRelativeFile does is launch a signal telling we
|
|
// want to open another URL, the problem is that when the file is
|
|
// non local, the loading is done assynchronously so you can't
|
|
// do a setViewport after the if as it was because you are doing the setViewport
|
|
// on the old file and when the new arrives there is no setViewport for it and
|
|
// it does not show anything
|
|
|
|
// first open filename if link is pointing outside this document
|
|
if ( go->isExternal() && !d->openRelativeFile( go->fileName() ) )
|
|
{
|
|
kWarning(OkularDebug).nospace() << "Action: Error opening '" << go->fileName() << "'.";
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
// skip local links that point to nowhere (broken ones)
|
|
if (!d->m_nextDocumentViewport.isValid())
|
|
return;
|
|
|
|
setViewport( d->m_nextDocumentViewport, -1, true );
|
|
d->m_nextDocumentViewport = DocumentViewport();
|
|
}
|
|
|
|
} break;
|
|
|
|
case Action::Execute: {
|
|
const ExecuteAction * exe = static_cast< const ExecuteAction * >( action );
|
|
QString fileName = exe->fileName();
|
|
if ( fileName.endsWith( ".pdf" ) || fileName.endsWith( ".PDF" ) )
|
|
{
|
|
d->openRelativeFile( fileName );
|
|
return;
|
|
}
|
|
|
|
// Albert: the only pdf i have that has that kind of link don't define
|
|
// an application and use the fileName as the file to open
|
|
fileName = d->giveAbsolutePath( fileName );
|
|
KMimeType::Ptr mime = KMimeType::findByPath( fileName );
|
|
// Check executables
|
|
if ( KRun::isExecutableFile( fileName, mime->name() ) )
|
|
{
|
|
// Don't have any pdf that uses this code path, just a guess on how it should work
|
|
if ( !exe->parameters().isEmpty() )
|
|
{
|
|
fileName = d->giveAbsolutePath( exe->parameters() );
|
|
mime = KMimeType::findByPath( fileName );
|
|
if ( KRun::isExecutableFile( fileName, mime->name() ) )
|
|
{
|
|
// this case is a link pointing to an executable with a parameter
|
|
// that also is an executable, possibly a hand-crafted pdf
|
|
KMessageBox::information( widget(), i18n("The document is trying to execute an external application and for your safety okular does not allow that.") );
|
|
return;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// this case is a link pointing to an executable with no parameters
|
|
// core developers find unacceptable executing it even after asking the user
|
|
KMessageBox::information( widget(), i18n("The document is trying to execute an external application and for your safety okular does not allow that.") );
|
|
return;
|
|
}
|
|
}
|
|
|
|
KService::Ptr ptr = KMimeTypeTrader::self()->preferredService( mime->name(), "Application" );
|
|
if ( ptr )
|
|
{
|
|
KUrl::List lst;
|
|
lst.append( fileName );
|
|
KRun::run( *ptr, lst, 0 );
|
|
}
|
|
else
|
|
KMessageBox::information( widget(), i18n( "No application found for opening file of mimetype %1.", mime->name() ) );
|
|
} break;
|
|
|
|
case Action::DocAction: {
|
|
const DocumentAction * docaction = static_cast< const DocumentAction * >( action );
|
|
switch( docaction->documentActionType() )
|
|
{
|
|
case DocumentAction::PageFirst:
|
|
setViewportPage( 0 );
|
|
break;
|
|
case DocumentAction::PagePrev:
|
|
if ( (*d->m_viewportIterator).pageNumber > 0 )
|
|
setViewportPage( (*d->m_viewportIterator).pageNumber - 1 );
|
|
break;
|
|
case DocumentAction::PageNext:
|
|
if ( (*d->m_viewportIterator).pageNumber < (int)d->m_pagesVector.count() - 1 )
|
|
setViewportPage( (*d->m_viewportIterator).pageNumber + 1 );
|
|
break;
|
|
case DocumentAction::PageLast:
|
|
setViewportPage( d->m_pagesVector.count() - 1 );
|
|
break;
|
|
case DocumentAction::HistoryBack:
|
|
setPrevViewport();
|
|
break;
|
|
case DocumentAction::HistoryForward:
|
|
setNextViewport();
|
|
break;
|
|
case DocumentAction::Quit:
|
|
emit quit();
|
|
break;
|
|
case DocumentAction::Presentation:
|
|
emit linkPresentation();
|
|
break;
|
|
case DocumentAction::EndPresentation:
|
|
emit linkEndPresentation();
|
|
break;
|
|
case DocumentAction::Find:
|
|
emit linkFind();
|
|
break;
|
|
case DocumentAction::GoToPage:
|
|
emit linkGoToPage();
|
|
break;
|
|
case DocumentAction::Close:
|
|
emit close();
|
|
break;
|
|
}
|
|
} break;
|
|
|
|
case Action::Browse: {
|
|
const BrowseAction * browse = static_cast< const BrowseAction * >( action );
|
|
// if the url is a mailto one, invoke mailer
|
|
if ( browse->url().startsWith( "mailto:", Qt::CaseInsensitive ) )
|
|
KToolInvocation::invokeMailer( browse->url() );
|
|
else
|
|
{
|
|
QString url = browse->url();
|
|
|
|
// fix for #100366, documents with relative links that are the form of http:foo.pdf
|
|
if (url.indexOf("http:") == 0 && url.indexOf("http://") == -1 && url.right(4) == ".pdf")
|
|
{
|
|
d->openRelativeFile(url.mid(5));
|
|
return;
|
|
}
|
|
|
|
// Albert: this is not a leak!
|
|
new KRun( KUrl( url ), widget() );
|
|
}
|
|
} break;
|
|
|
|
case Action::Sound: {
|
|
const SoundAction * linksound = static_cast< const SoundAction * >( action );
|
|
AudioPlayer::instance()->playSound( linksound->sound(), linksound );
|
|
} break;
|
|
|
|
case Action::Movie:
|
|
//const MovieAction * movie = static_cast< const MovieAction * >( action );
|
|
// TODO this (Movie action)
|
|
break;
|
|
}
|
|
}
|
|
|
|
void Document::processSourceReference( const SourceReference * ref )
|
|
{
|
|
if ( !ref )
|
|
return;
|
|
|
|
if ( !QFile::exists( ref->fileName() ) )
|
|
{
|
|
kDebug(OkularDebug).nospace() << "No such file: '" << ref->fileName() << "'";
|
|
return;
|
|
}
|
|
|
|
static QHash< int, QString > editors;
|
|
// init the editors table if empty (on first run, usually)
|
|
if ( editors.isEmpty() )
|
|
{
|
|
editors[ Settings::EnumExternalEditor::Kate ] =
|
|
QLatin1String( "kate --use --line %l --column %c" );
|
|
editors[ Settings::EnumExternalEditor::Scite ] =
|
|
QLatin1String( "scite %f \"-goto:%l,%c\"" );
|
|
}
|
|
|
|
QHash< int, QString >::const_iterator it = editors.constFind( Settings::externalEditor() );
|
|
QString p;
|
|
if ( it != editors.end() )
|
|
p = *it;
|
|
else
|
|
p = Settings::externalEditorCommand();
|
|
// custom editor not yet configured
|
|
if ( p.isEmpty() )
|
|
return;
|
|
|
|
// replacing the placeholders
|
|
p.replace( QLatin1String( "%l" ), QString::number( ref->row() ) );
|
|
p.replace( QLatin1String( "%c" ), QString::number( ref->column() ) );
|
|
if ( p.indexOf( QLatin1String( "%f" ) ) > -1 )
|
|
p.replace( QLatin1String( "%f" ), ref->fileName() );
|
|
else
|
|
p.append( QLatin1String( " " ) + ref->fileName() );
|
|
|
|
// paranoic checks
|
|
if ( p.isEmpty() || p.trimmed() == ref->fileName() )
|
|
return;
|
|
|
|
QProcess::startDetached( p );
|
|
}
|
|
|
|
Document::PrintingType Document::printingSupport() const
|
|
{
|
|
if ( d->m_generator )
|
|
{
|
|
|
|
if ( d->m_generator->hasFeature( Generator::PrintNative ) )
|
|
{
|
|
return NativePrinting;
|
|
}
|
|
|
|
#ifndef Q_OS_WIN
|
|
if ( d->m_generator->hasFeature( Generator::PrintPostscript ) )
|
|
{
|
|
return PostscriptPrinting;
|
|
}
|
|
#endif
|
|
|
|
}
|
|
|
|
return NoPrinting;
|
|
}
|
|
|
|
bool Document::supportsPrintToFile() const
|
|
{
|
|
return d->m_generator ? d->m_generator->hasFeature( Generator::PrintToFile ) : false;
|
|
}
|
|
|
|
bool Document::print( QPrinter &printer )
|
|
{
|
|
return d->m_generator ? d->m_generator->print( printer ) : false;
|
|
}
|
|
|
|
QWidget* Document::printConfigurationWidget() const
|
|
{
|
|
if ( d->m_generator )
|
|
{
|
|
PrintInterface * iface = qobject_cast< Okular::PrintInterface * >( d->m_generator );
|
|
return iface ? iface->printConfigurationWidget() : 0;
|
|
}
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
void Document::fillConfigDialog( KConfigDialog * dialog )
|
|
{
|
|
if ( !dialog )
|
|
return;
|
|
|
|
// ensure that we have all the generators with settings loaded
|
|
QString constraint( "([X-KDE-Priority] > 0) and (exist Library) and ([X-KDE-okularHasInternalSettings])" );
|
|
KService::List offers = KServiceTypeTrader::self()->query( "okular/Generator", constraint );
|
|
d->loadServiceList( offers );
|
|
|
|
bool pagesAdded = false;
|
|
QHash< QString, GeneratorInfo >::iterator it = d->m_loadedGenerators.begin();
|
|
QHash< QString, GeneratorInfo >::iterator itEnd = d->m_loadedGenerators.end();
|
|
for ( ; it != itEnd; ++it )
|
|
{
|
|
Okular::ConfigInterface * iface = d->generatorConfig( it.value() );
|
|
if ( iface )
|
|
{
|
|
iface->addPages( dialog );
|
|
pagesAdded = true;
|
|
if ( !it.value().catalogName.isEmpty() )
|
|
KGlobal::locale()->insertCatalog( it.value().catalogName );
|
|
}
|
|
}
|
|
if ( pagesAdded )
|
|
{
|
|
connect( dialog, SIGNAL( settingsChanged( const QString& ) ),
|
|
this, SLOT( slotGeneratorConfigChanged( const QString& ) ) );
|
|
}
|
|
}
|
|
|
|
int Document::configurableGenerators() const
|
|
{
|
|
QString constraint( "([X-KDE-Priority] > 0) and (exist Library) and ([X-KDE-okularHasInternalSettings])" );
|
|
KService::List offers = KServiceTypeTrader::self()->query( "okular/Generator", constraint );
|
|
return offers.count();
|
|
}
|
|
|
|
QStringList Document::supportedMimeTypes() const
|
|
{
|
|
if ( !d->m_supportedMimeTypes.isEmpty() )
|
|
return d->m_supportedMimeTypes;
|
|
|
|
QString constraint( "(Library == 'okularpart')" );
|
|
QLatin1String basePartService( "KParts/ReadOnlyPart" );
|
|
KService::List offers = KServiceTypeTrader::self()->query( basePartService, constraint );
|
|
KService::List::ConstIterator it = offers.begin(), itEnd = offers.end();
|
|
for ( ; it != itEnd; ++it )
|
|
{
|
|
KService::Ptr service = *it;
|
|
QStringList mimeTypes = service->serviceTypes();
|
|
foreach ( const QString& mimeType, mimeTypes )
|
|
if ( mimeType != basePartService )
|
|
d->m_supportedMimeTypes.append( mimeType );
|
|
}
|
|
|
|
return d->m_supportedMimeTypes;
|
|
}
|
|
|
|
const KComponentData* Document::componentData() const
|
|
{
|
|
if ( !d->m_generator )
|
|
return 0;
|
|
|
|
QHash< QString, GeneratorInfo >::const_iterator genIt = d->m_loadedGenerators.constFind( d->m_generatorName );
|
|
Q_ASSERT( genIt != d->m_loadedGenerators.constEnd() );
|
|
const KComponentData* kcd = &genIt.value().data;
|
|
|
|
// empty about data
|
|
if ( kcd->isValid() && kcd->aboutData() && kcd->aboutData()->programName().isEmpty() )
|
|
return 0;
|
|
|
|
return kcd;
|
|
}
|
|
|
|
void DocumentPrivate::requestDone( PixmapRequest * req )
|
|
{
|
|
if ( !m_generator || !req )
|
|
return;
|
|
|
|
#ifndef NDEBUG
|
|
if ( !m_generator->canGeneratePixmap() )
|
|
kDebug(OkularDebug) << "requestDone with generator not in READY state.";
|
|
#endif
|
|
|
|
// [MEM] 1.1 find and remove a previous entry for the same page and id
|
|
QLinkedList< AllocatedPixmap * >::iterator aIt = m_allocatedPixmapsFifo.begin();
|
|
QLinkedList< AllocatedPixmap * >::iterator aEnd = m_allocatedPixmapsFifo.end();
|
|
for ( ; aIt != aEnd; ++aIt )
|
|
if ( (*aIt)->page == req->pageNumber() && (*aIt)->id == req->id() )
|
|
{
|
|
AllocatedPixmap * p = *aIt;
|
|
m_allocatedPixmapsFifo.erase( aIt );
|
|
m_allocatedPixmapsTotalMemory -= p->memory;
|
|
delete p;
|
|
break;
|
|
}
|
|
|
|
QMap< int, DocumentObserver * >::const_iterator itObserver = m_observers.constFind( req->id() );
|
|
if ( itObserver != m_observers.constEnd() )
|
|
{
|
|
// [MEM] 1.2 append memory allocation descriptor to the FIFO
|
|
qulonglong memoryBytes = 4 * req->width() * req->height();
|
|
AllocatedPixmap * memoryPage = new AllocatedPixmap( req->id(), req->pageNumber(), memoryBytes );
|
|
m_allocatedPixmapsFifo.append( memoryPage );
|
|
m_allocatedPixmapsTotalMemory += memoryBytes;
|
|
|
|
// 2. notify an observer that its pixmap changed
|
|
itObserver.value()->notifyPageChanged( req->pageNumber(), DocumentObserver::Pixmap );
|
|
}
|
|
#ifndef NDEBUG
|
|
else
|
|
kWarning(OkularDebug) << "Receiving a done request for the defunct observer" << req->id();
|
|
#endif
|
|
|
|
// 3. delete request
|
|
delete req;
|
|
|
|
// 4. start a new generation if some is pending
|
|
m_pixmapRequestsMutex.lock();
|
|
bool hasPixmaps = !m_pixmapRequestsStack.isEmpty();
|
|
m_pixmapRequestsMutex.unlock();
|
|
if ( hasPixmaps )
|
|
sendGeneratorRequest();
|
|
}
|
|
|
|
void Document::setRotation( int r )
|
|
{
|
|
d->setRotationInternal( r, true );
|
|
}
|
|
|
|
void DocumentPrivate::setRotationInternal( int r, bool notify )
|
|
{
|
|
Rotation rotation = (Rotation)r;
|
|
if ( !m_generator || ( m_rotation == rotation ) )
|
|
return;
|
|
|
|
// tell the pages to rotate
|
|
QVector< Okular::Page * >::const_iterator pIt = m_pagesVector.begin();
|
|
QVector< Okular::Page * >::const_iterator pEnd = m_pagesVector.end();
|
|
for ( ; pIt != pEnd; ++pIt )
|
|
(*pIt)->d->rotateAt( rotation );
|
|
if ( notify )
|
|
{
|
|
// notify the generator that the current rotation has changed
|
|
m_generator->rotationChanged( rotation, m_rotation );
|
|
}
|
|
// set the new rotation
|
|
m_rotation = rotation;
|
|
|
|
if ( notify )
|
|
{
|
|
foreachObserverD( notifySetup( m_pagesVector, DocumentObserver::NewLayoutForPages ) );
|
|
foreachObserverD( notifyContentsCleared( DocumentObserver::Pixmap | DocumentObserver::Highlights | DocumentObserver::Annotations ) );
|
|
}
|
|
kDebug(OkularDebug) << "Rotated:" << r;
|
|
}
|
|
|
|
void Document::setPageSize( const PageSize &size )
|
|
{
|
|
if ( !d->m_generator || !d->m_generator->hasFeature( Generator::PageSizes ) )
|
|
return;
|
|
|
|
if ( d->m_pageSizes.isEmpty() )
|
|
d->m_pageSizes = d->m_generator->pageSizes();
|
|
int sizeid = d->m_pageSizes.indexOf( size );
|
|
if ( sizeid == -1 )
|
|
return;
|
|
|
|
// tell the pages to change size
|
|
QVector< Okular::Page * >::const_iterator pIt = d->m_pagesVector.begin();
|
|
QVector< Okular::Page * >::const_iterator pEnd = d->m_pagesVector.end();
|
|
for ( ; pIt != pEnd; ++pIt )
|
|
(*pIt)->d->changeSize( size );
|
|
// clear 'memory allocation' descriptors
|
|
QLinkedList< AllocatedPixmap * >::const_iterator aIt = d->m_allocatedPixmapsFifo.begin();
|
|
QLinkedList< AllocatedPixmap * >::const_iterator aEnd = d->m_allocatedPixmapsFifo.end();
|
|
for ( ; aIt != aEnd; ++aIt )
|
|
delete *aIt;
|
|
d->m_allocatedPixmapsFifo.clear();
|
|
d->m_allocatedPixmapsTotalMemory = 0;
|
|
// notify the generator that the current page size has changed
|
|
d->m_generator->pageSizeChanged( size, d->m_pageSize );
|
|
// set the new page size
|
|
d->m_pageSize = size;
|
|
|
|
foreachObserver( notifySetup( d->m_pagesVector, DocumentObserver::NewLayoutForPages ) );
|
|
foreachObserver( notifyContentsCleared( DocumentObserver::Pixmap | DocumentObserver::Highlights ) );
|
|
kDebug(OkularDebug) << "New PageSize id:" << sizeid;
|
|
}
|
|
|
|
|
|
/** DocumentViewport **/
|
|
|
|
DocumentViewport::DocumentViewport( int n )
|
|
: pageNumber( n )
|
|
{
|
|
// default settings
|
|
rePos.enabled = false;
|
|
rePos.normalizedX = 0.5;
|
|
rePos.normalizedY = 0.0;
|
|
rePos.pos = Center;
|
|
autoFit.enabled = false;
|
|
autoFit.width = false;
|
|
autoFit.height = false;
|
|
}
|
|
|
|
DocumentViewport::DocumentViewport( const QString & xmlDesc )
|
|
: pageNumber( -1 )
|
|
{
|
|
// default settings (maybe overridden below)
|
|
rePos.enabled = false;
|
|
rePos.normalizedX = 0.5;
|
|
rePos.normalizedY = 0.0;
|
|
rePos.pos = Center;
|
|
autoFit.enabled = false;
|
|
autoFit.width = false;
|
|
autoFit.height = false;
|
|
|
|
// check for string presence
|
|
if ( xmlDesc.isEmpty() )
|
|
return;
|
|
|
|
// decode the string
|
|
bool ok;
|
|
int field = 0;
|
|
QString token = xmlDesc.section( ';', field, field );
|
|
while ( !token.isEmpty() )
|
|
{
|
|
// decode the current token
|
|
if ( field == 0 )
|
|
{
|
|
pageNumber = token.toInt( &ok );
|
|
if ( !ok )
|
|
return;
|
|
}
|
|
else if ( token.startsWith( "C1" ) )
|
|
{
|
|
rePos.enabled = true;
|
|
rePos.normalizedX = token.section( ':', 1, 1 ).toDouble();
|
|
rePos.normalizedY = token.section( ':', 2, 2 ).toDouble();
|
|
rePos.pos = Center;
|
|
}
|
|
else if ( token.startsWith( "C2" ) )
|
|
{
|
|
rePos.enabled = true;
|
|
rePos.normalizedX = token.section( ':', 1, 1 ).toDouble();
|
|
rePos.normalizedY = token.section( ':', 2, 2 ).toDouble();
|
|
if (token.section( ':', 3, 3 ).toInt() == 1) rePos.pos = Center;
|
|
else rePos.pos = TopLeft;
|
|
}
|
|
else if ( token.startsWith( "AF1" ) )
|
|
{
|
|
autoFit.enabled = true;
|
|
autoFit.width = token.section( ':', 1, 1 ) == "T";
|
|
autoFit.height = token.section( ':', 2, 2 ) == "T";
|
|
}
|
|
// proceed tokenizing string
|
|
field++;
|
|
token = xmlDesc.section( ';', field, field );
|
|
}
|
|
}
|
|
|
|
QString DocumentViewport::toString() const
|
|
{
|
|
// start string with page number
|
|
QString s = QString::number( pageNumber );
|
|
// if has center coordinates, save them on string
|
|
if ( rePos.enabled )
|
|
s += QString( ";C2:" ) + QString::number( rePos.normalizedX ) +
|
|
':' + QString::number( rePos.normalizedY ) +
|
|
':' + QString::number( rePos.pos );
|
|
// if has autofit enabled, save its state on string
|
|
if ( autoFit.enabled )
|
|
s += QString( ";AF1:" ) + (autoFit.width ? "T" : "F") +
|
|
':' + (autoFit.height ? "T" : "F");
|
|
return s;
|
|
}
|
|
|
|
bool DocumentViewport::isValid() const
|
|
{
|
|
return pageNumber >= 0;
|
|
}
|
|
|
|
bool DocumentViewport::operator==( const DocumentViewport & vp ) const
|
|
{
|
|
bool equal = ( pageNumber == vp.pageNumber ) &&
|
|
( rePos.enabled == vp.rePos.enabled ) &&
|
|
( autoFit.enabled == vp.autoFit.enabled );
|
|
if ( !equal )
|
|
return false;
|
|
if ( rePos.enabled &&
|
|
(( rePos.normalizedX != vp.rePos.normalizedX) ||
|
|
( rePos.normalizedY != vp.rePos.normalizedY ) || rePos.pos != vp.rePos.pos) )
|
|
return false;
|
|
if ( autoFit.enabled &&
|
|
(( autoFit.width != vp.autoFit.width ) ||
|
|
( autoFit.height != vp.autoFit.height )) )
|
|
return false;
|
|
return true;
|
|
}
|
|
|
|
|
|
/** DocumentInfo **/
|
|
|
|
DocumentInfo::DocumentInfo()
|
|
: QDomDocument( "DocumentInformation" )
|
|
{
|
|
QDomElement docElement = createElement( "DocumentInfo" );
|
|
appendChild( docElement );
|
|
}
|
|
|
|
void DocumentInfo::set( const QString &key, const QString &value,
|
|
const QString &title )
|
|
{
|
|
QDomElement docElement = documentElement();
|
|
QDomElement element;
|
|
|
|
// check whether key already exists
|
|
QDomNodeList list = docElement.elementsByTagName( key );
|
|
if ( list.count() > 0 )
|
|
element = list.item( 0 ).toElement();
|
|
else
|
|
element = createElement( key );
|
|
|
|
element.setAttribute( "value", value );
|
|
element.setAttribute( "title", title );
|
|
|
|
if ( list.count() == 0 )
|
|
docElement.appendChild( element );
|
|
}
|
|
|
|
void DocumentInfo::set( enum Key key, const QString &value )
|
|
{
|
|
switch ( key ) {
|
|
case Title:
|
|
set( "title", value, i18n( "Title" ) );
|
|
break;
|
|
case Subject:
|
|
set( "subject", value, i18n( "Subject" ) );
|
|
break;
|
|
case Description:
|
|
set( "description", value, i18n( "Description" ) );
|
|
break;
|
|
case Author:
|
|
set( "author", value, i18n( "Author" ) );
|
|
break;
|
|
case Creator:
|
|
set( "creator", value, i18n( "Creator" ) );
|
|
break;
|
|
case Producer:
|
|
set( "producer", value, i18n( "Producer" ) );
|
|
break;
|
|
case Copyright:
|
|
set( "copyright", value, i18n( "Copyright" ) );
|
|
break;
|
|
case Pages:
|
|
set( "pages", value, i18n( "Pages" ) );
|
|
break;
|
|
case CreationDate:
|
|
set( "creationDate", value, i18n( "Created" ) );
|
|
break;
|
|
case ModificationDate:
|
|
set( "modificationDate", value, i18n( "Modified" ) );
|
|
break;
|
|
case MimeType:
|
|
set( "mimeType", value, i18n( "Mime Type" ) );
|
|
break;
|
|
case Category:
|
|
set( "category", value, i18n( "Category" ) );
|
|
break;
|
|
case Keywords:
|
|
set( "keywords", value, i18n( "Keywords" ) );
|
|
break;
|
|
default:
|
|
kWarning(OkularDebug) << "Invalid key passed";
|
|
break;
|
|
}
|
|
}
|
|
|
|
QString DocumentInfo::get( const QString &key ) const
|
|
{
|
|
QDomElement docElement = documentElement();
|
|
QDomElement element;
|
|
|
|
// check whether key already exists
|
|
QDomNodeList list = docElement.elementsByTagName( key );
|
|
if ( list.count() > 0 )
|
|
return list.item( 0 ).toElement().attribute( "value" );
|
|
else
|
|
return QString();
|
|
}
|
|
|
|
|
|
/** DocumentSynopsis **/
|
|
|
|
DocumentSynopsis::DocumentSynopsis()
|
|
: QDomDocument( "DocumentSynopsis" )
|
|
{
|
|
// void implementation, only subclassed for naming
|
|
}
|
|
|
|
DocumentSynopsis::DocumentSynopsis( const QDomDocument &document )
|
|
: QDomDocument( document )
|
|
{
|
|
}
|
|
|
|
/** EmbeddedFile **/
|
|
|
|
EmbeddedFile::EmbeddedFile()
|
|
{
|
|
}
|
|
|
|
EmbeddedFile::~EmbeddedFile()
|
|
{
|
|
}
|
|
|
|
VisiblePageRect::VisiblePageRect( int page, const NormalizedRect &rectangle )
|
|
: pageNumber( page ), rect( rectangle )
|
|
{
|
|
}
|
|
|
|
#include "document.moc"
|