summaryrefslogtreecommitdiffstats
path: root/kghostview/kgv_view.cpp
diff options
context:
space:
mode:
authortoma <toma@283d02a7-25f6-0310-bc7c-ecb5cbfe19da>2009-11-25 17:56:58 +0000
committertoma <toma@283d02a7-25f6-0310-bc7c-ecb5cbfe19da>2009-11-25 17:56:58 +0000
commit47d455dd55be855e4cc691c32f687f723d9247ee (patch)
tree52e236aaa2576bdb3840ebede26619692fed6d7d /kghostview/kgv_view.cpp
downloadtdegraphics-47d455dd55be855e4cc691c32f687f723d9247ee.tar.gz
tdegraphics-47d455dd55be855e4cc691c32f687f723d9247ee.zip
Copy the KDE 3.5 branch to branches/trinity for new KDE 3.5 features.
BUG:215923 git-svn-id: svn://anonsvn.kde.org/home/kde/branches/trinity/kdegraphics@1054174 283d02a7-25f6-0310-bc7c-ecb5cbfe19da
Diffstat (limited to 'kghostview/kgv_view.cpp')
-rw-r--r--kghostview/kgv_view.cpp1037
1 files changed, 1037 insertions, 0 deletions
diff --git a/kghostview/kgv_view.cpp b/kghostview/kgv_view.cpp
new file mode 100644
index 00000000..3c25ef41
--- /dev/null
+++ b/kghostview/kgv_view.cpp
@@ -0,0 +1,1037 @@
+/**
+ * Copyright (C) 2000-2002 the KGhostView authors. See file AUTHORS.
+ *
+ * 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.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+
+#include <qfile.h>
+#include <qfileinfo.h>
+#include <qframe.h>
+#include <qtable.h>
+#include <qlayout.h>
+#include <qregexp.h>
+
+#include <kaction.h>
+#include <kapplication.h>
+#include <kconfig.h>
+#include <kdebug.h>
+#include <kdirwatch.h>
+#include <kglobalsettings.h>
+#include <kiconloader.h>
+#include <kinstance.h>
+#include <klocale.h>
+#include <kpopupmenu.h>
+#include <kstdaction.h>
+#include <kstdaccel.h>
+#include <ktempfile.h>
+#include <kio/scheduler.h>
+#include <kaboutdata.h>
+
+
+#include "kgv_view.h"
+#include "kgv_miniwidget.h"
+#include "kgvconfigdialog.h"
+#include "kgvdocument.h"
+#include "kgvpagedecorator.h"
+#include "kgvpageview.h"
+#include "kgvmainwidget.h"
+#include "kpswidget.h"
+#include "kgvfactory.h"
+#include "logwindow.h"
+#include "marklist.h"
+#include "scrollbox.h"
+#include "version.h"
+#include "configuration.h"
+
+#include <cmath>
+
+namespace KGV {
+ /*
+ * This is because Qt's iterators
+ * are not standard iterators bc of missing typedefs,
+ * so they are only *almost* STL compatible
+ */
+ template <typename T>
+ unsigned distance( T a, T b ) {
+ unsigned res = 0;
+ while ( a != b ) {
+ ++res;
+ ++a;
+ }
+ return res;
+ }
+}
+
+KGVPart::KGVPart( QWidget* parentWidget, const char*,
+ QObject* parent, const char* name,
+ const QStringList &args ) :
+ KParts::ReadOnlyPart( parent, name ),
+ _fitTimer( new QTimer( this ) ),
+ _job( 0 ),
+ _mimetypeScanner( 0 ),
+ _dirtyHandler( new QTimer( this ) ),
+ _isGuiInitialized( false ),
+ _isFileDirty( false ),
+ _stickyOptions( false ),
+ _embeddedInKGhostView( !args.contains( "KParts::ReadOnlyPart" ) ),
+ _customZoomIndex( -1 )
+{
+ setInstance( KGVFactory::instance() );
+
+ // Don't show the progress info dialog if we're embedded in Konqueror.
+ setProgressInfoEnabled( !args.contains( "Browser/View") );
+
+ _document = new KGVDocument( this );
+ connect( _document, SIGNAL( fileChangeFailed() ),
+ this, SLOT( slotCancelWatch() ) );
+ connect( _document, SIGNAL( completed() ),
+ this, SLOT( slotOpenFileCompleted() ) );
+ connect( _document, SIGNAL( canceled( const QString& ) ),
+ this, SIGNAL( canceled( const QString& ) ) );
+
+ _fileWatcher = new KDirWatch( this );
+ connect( _fileWatcher, SIGNAL( dirty( const QString& ) ),
+ this, SLOT( slotFileDirty( const QString& ) ) );
+ connect( _dirtyHandler, SIGNAL( timeout() ),
+ this, SLOT( slotDoFileDirty() ) );
+
+ // Setup main widget
+ _mainWidget = new KGVMainWidget( parentWidget );
+ _mainWidget->setFocusPolicy( QWidget::StrongFocus );
+ _mainWidget->installEventFilter( this );
+ _mainWidget->setAcceptDrops( true );
+ connect( _mainWidget, SIGNAL( spacePressed() ),
+ this, SLOT( slotReadDown() ) );
+ connect( _mainWidget, SIGNAL( urlDropped( const KURL& ) ),
+ this, SLOT( openURL( const KURL& ) ) );
+
+ QHBoxLayout* hlay = new QHBoxLayout( _mainWidget, 0, 0 );
+ QVBoxLayout* vlay = new QVBoxLayout( hlay );
+
+ const int PAGELIST_WIDTH = 75;
+
+ _scrollBox = new ScrollBox( _mainWidget , "scrollbox" );
+ _scrollBox->setFixedWidth( PAGELIST_WIDTH );
+ _scrollBox->setMinimumHeight( PAGELIST_WIDTH );
+ vlay->addWidget( _scrollBox );
+
+ _divider = new QFrame( _mainWidget, "divider" );
+ _divider->setFrameStyle( QFrame::Panel | QFrame::Raised );
+ _divider->setLineWidth( 1 );
+ _divider->setMinimumWidth( 3 );
+ hlay->addWidget( _divider );
+
+ _pageView = new KGVPageView( _mainWidget, "pageview" );
+ _pageView->viewport()->setBackgroundMode( QWidget::PaletteMid );
+ hlay->addWidget( _pageView, 1 );
+ _mainWidget->setFocusProxy( _pageView );
+ setWidget( _mainWidget );
+
+ _pageDecorator = new KGVPageDecorator( _pageView->viewport() );
+ _pageDecorator->hide();
+
+
+ _psWidget = new KPSWidget( _pageDecorator );
+ _psWidget->readSettings();
+ _pageView->setPage( _pageDecorator );
+ connect( _psWidget, SIGNAL( output( char*, int ) ),
+ this, SLOT( slotGhostscriptOutput( char*, int ) ) );
+
+ connect( _psWidget, SIGNAL( ghostscriptError( const QString& ) ),
+ this, SLOT( slotGhostscriptError( const QString& ) ) );
+
+
+ _logWindow = new LogWindow( i18n( "Ghostscript Messages" ), _mainWidget, "logwindow" );
+ _showLogWindow = false;
+
+ connect( _logWindow, SIGNAL( configureGS() ), SLOT( slotConfigure() ) );
+
+ _docManager = new KGVMiniWidget( this );
+ _docManager->setPSWidget( _psWidget );
+ _docManager->setDocument( document() );
+
+ _markList = new MarkList( _mainWidget, "marklist", _docManager );
+ _markList->setFixedWidth( PAGELIST_WIDTH );
+ vlay->addWidget( _markList, 1 );
+ connect( _markList, SIGNAL( contextMenuRequested ( int, int, const QPoint& ) ),
+ this, SLOT( showPopup( int, int, const QPoint& ) ) );
+
+
+ connect( _markList, SIGNAL( selected( int ) ),
+ _docManager, SLOT( goToPage( int ) ) );
+ connect( _docManager, SIGNAL( newPageShown( int ) ),
+ _markList, SLOT( select( int ) ) );
+ connect( _docManager, SIGNAL( setStatusBarText( const QString& ) ),
+ this, SIGNAL( setStatusBarText( const QString& ) ) );
+ connect( _scrollBox, SIGNAL( valueChangedRelative( int, int ) ),
+ _pageView, SLOT( scrollBy( int, int ) ) );
+ connect( _pageView, SIGNAL( pageSizeChanged( const QSize& ) ),
+ _scrollBox, SLOT( setPageSize( const QSize& ) ) );
+ connect( _pageView, SIGNAL( viewSizeChanged( const QSize& ) ),
+ _scrollBox, SLOT( setViewSize( const QSize& ) ) );
+ connect( _pageView, SIGNAL( contentsMoving( int, int ) ),
+ _scrollBox, SLOT( setViewPos( int, int ) ) );
+
+ //-- File Menu ----------------------------------------------------------
+ KStdAction::saveAs( document(), SLOT( saveAs() ),
+ actionCollection() );
+ new KAction( i18n( "Document &Info" ), 0,
+ miniWidget(), SLOT( info() ),
+ actionCollection(), "info" );
+
+ //-- Edit Menu -----------------------------------------------------
+ _popup = new KPopupMenu( _markList, "marklist_menu" );
+
+ KAction *act = new KAction( i18n( "Mark Current Page" ), "flag", CTRL+SHIFT+Key_M,
+ _markList, SLOT( markCurrent() ),
+ actionCollection(), "mark_current" );
+ act->plug( _popup );
+ act = new KAction( i18n( "Mark &All Pages" ), 0,
+ _markList, SLOT( markAll() ),
+ actionCollection(), "mark_all" );
+ act->plug( _popup );
+ act = new KAction( i18n( "Mark &Even Pages" ), 0,
+ _markList, SLOT( markEven() ),
+ actionCollection(), "mark_even" );
+ act->plug( _popup );
+ act = new KAction( i18n( "Mark &Odd Pages" ), 0,
+ _markList, SLOT( markOdd() ),
+ actionCollection(), "mark_odd" );
+ act->plug( _popup );
+ act = new KAction( i18n( "&Toggle Page Marks" ), 0,
+ _markList, SLOT( toggleMarks() ),
+ actionCollection(), "toggle" );
+ act->plug( _popup );
+ act = new KAction( i18n("&Remove Page Marks"), 0,
+ _markList, SLOT( removeMarks() ),
+ actionCollection(), "remove" );
+ act->plug( _popup );
+
+ // TODO -- disable entry if there aren't any page names
+
+ //-- View Menu ----------------------------------------------------------
+ _selectOrientation = new KSelectAction( i18n( "&Orientation" ), 0, 0, 0,
+ actionCollection(), "orientation_menu" );
+ _selectMedia = new KSelectAction( i18n( "Paper &Size" ), 0, 0, 0,
+ actionCollection(), "media_menu" );
+
+ _flick = new KToggleAction( i18n( "No &Flicker" ), 0,
+ this, SLOT( slotFlicker() ),
+ actionCollection(), "no_flicker" );
+
+ QStringList orientations;
+ orientations.append( i18n( "Auto" ) );
+ orientations.append( i18n( "Portrait" ) );
+ orientations.append( i18n( "Landscape" ) );
+ orientations.append( i18n( "Upside Down" ) );
+ orientations.append( i18n( "Seascape" ) );
+ _selectOrientation->setItems( orientations );
+
+ connect( _selectOrientation, SIGNAL( activated( int ) ),
+ this, SLOT( slotOrientation( int ) ) );
+ connect( _selectMedia, SIGNAL( activated( int ) ),
+ this, SLOT( slotMedia( int ) ) );
+
+ {
+ KShortcut zoomInShort = KStdAccel::zoomIn();
+ zoomInShort.append( KKey( CTRL+Key_Equal ) );
+ _zoomIn = KStdAction::zoomIn( this, SLOT( slotZoomIn() ),
+ actionCollection(), "zoomIn" );
+ _zoomIn->setShortcut( zoomInShort );
+ }
+ _zoomOut = KStdAction::zoomOut( this, SLOT( slotZoomOut() ),
+ actionCollection(), "zoomOut" );
+ _zoomTo = new KSelectAction( i18n( "Zoom" ), "viewmag", 0, actionCollection(), "zoomTo" );
+ connect( _zoomTo, SIGNAL( activated( const QString & ) ), this, SLOT( slotZoom( const QString& ) ) );
+ _zoomTo->setEditable( true );
+ _zoomTo->clear();
+ QValueList<double> mags = DisplayOptions::normalMagnificationValues();
+ QStringList zooms;
+ int idx = 0;
+ int cur = 0;
+ for ( QValueList<double>::iterator first = mags.begin(), last = mags.end();
+ first != last;
+ ++first ) {
+ QString str = QString( "%1%" ).arg( KGlobal::locale()->formatNumber( *first * 100.0, 2 ));
+ str.remove( KGlobal::locale()->decimalSymbol() + "00" );
+ zooms << str;
+ if ( *first == 1.0 ) idx = cur;
+ ++cur;
+ }
+ _zoomTo->setItems( zooms );
+ _zoomTo->setCurrentItem( idx );
+
+ _fitWidth = new KAction( i18n( "&Fit to Page Width" ), 0, this,
+ SLOT( slotFitToPage() ), actionCollection(),
+ "fit_to_page");
+ _fitScreen = new KAction( i18n( "&Fit to Screen" ), Key_S, this,
+ SLOT( slotFitToScreen() ), actionCollection(),
+ "fit_to_screen");
+
+ _prevPage = new KAction( i18n( "Previous Page" ), CTRL+Key_PageUp, this, SLOT( slotPrevPage() ),
+ actionCollection(), "prevPage" );
+ _prevPage->setWhatsThis( i18n( "Moves to the previous page of the document" ) );
+
+ _nextPage = new KAction( i18n( "Next Page" ), CTRL + Key_PageDown, this, SLOT( slotNextPage() ),
+ actionCollection(), "nextPage" );
+ _nextPage->setWhatsThis( i18n( "Moves to the next page of the document" ) );
+
+ _firstPage = KStdAction::firstPage( this, SLOT( slotGotoStart() ),
+ actionCollection(), "goToStart" );
+ _firstPage->setWhatsThis( i18n( "Moves to the first page of the document" ) );
+
+ _lastPage = KStdAction::lastPage( this, SLOT( slotGotoEnd() ),
+ actionCollection(), "goToEnd" );
+ _lastPage->setWhatsThis( i18n( "Moves to the last page of the document" ) );
+
+ KShortcut readUpShort = KStdAccel::shortcut( KStdAccel::Prior );
+ readUpShort.append( KKey( SHIFT+Key_Space ) );
+ _readUp = new KAction( i18n( "Read Up" ), "up",
+ readUpShort, this, SLOT( slotReadUp() ),
+ actionCollection(), "readUp" );
+
+ KShortcut readDownShort = KStdAccel::shortcut( KStdAccel::Next );
+ readDownShort.append( KKey( Key_Space ) );
+ _readDown = new KAction( i18n( "Read Down" ), "down",
+ readDownShort, this, SLOT( slotReadDown() ),
+ actionCollection(), "readDown" );
+
+ _gotoPage = KStdAction::gotoPage( _docManager, SLOT( goToPage() ),
+ actionCollection(), "goToPage" );
+
+ //-- Settings Menu ------------------------------------------------------
+ _showScrollBars = new KToggleAction( i18n( "Show &Scrollbars" ), 0,
+ actionCollection(), "show_scrollbars" );
+ _showScrollBars->setCheckedState(i18n("Hide &Scrollbars"));
+ _watchFile = new KToggleAction( i18n( "&Watch File" ), 0,
+ this, SLOT( slotWatchFile() ),
+ actionCollection(), "watch_file" );
+ _showPageList = new KToggleAction( i18n( "Show &Page List" ), 0,
+ actionCollection(), "show_page_list" );
+ _showPageList->setCheckedState(i18n("Hide &Page List"));
+ _showPageLabels = new KToggleAction( i18n("Show Page &Labels"), 0,
+ actionCollection(), "show_page_labels" );
+ _showPageLabels->setCheckedState(i18n("Hide Page &Labels"));
+ KStdAction::preferences( this, SLOT( slotConfigure() ), actionCollection() );
+ connect( _showScrollBars, SIGNAL( toggled( bool ) ),
+ SLOT( showScrollBars( bool ) ) );
+ connect( _showPageList, SIGNAL( toggled( bool ) ),
+ SLOT( showMarkList( bool ) ) );
+ connect( _showPageLabels, SIGNAL( toggled( bool ) ),
+ SLOT( showPageLabels( bool ) ) );
+
+ _extension = new KGVBrowserExtension( this );
+
+ setXMLFile( "kgv_part.rc" );
+
+ connect( miniWidget(), SIGNAL( newPageShown( int ) ),
+ this, SLOT( slotNewPage( int ) ) );
+ connect( _pageView, SIGNAL( contentsMoving( int, int ) ),
+ this, SLOT( slotPageMoved( int, int ) ) );
+
+ connect( _pageView, SIGNAL( nextPage() ), SLOT( slotNextPage() ));
+ connect( _pageView, SIGNAL( prevPage() ), SLOT( slotPrevPage() ));
+ connect( _pageView, SIGNAL( zoomIn() ), SLOT( slotZoomIn() ));
+ connect( _pageView, SIGNAL( zoomOut() ), SLOT( slotZoomOut() ));
+ connect( _pageView, SIGNAL( ReadUp() ), SLOT( slotReadUp() ));
+ connect( _pageView, SIGNAL( ReadDown() ), SLOT( slotReadDown() ));
+
+ QStringList items = document()->mediaNames();
+ items.prepend( i18n( "Auto ") );
+ _selectMedia->setItems( items );
+
+ readSettings();
+
+ updatePageDepActions();
+}
+
+KGVPart::~KGVPart()
+{
+ if ( _job ) _job -> kill();
+ delete _mimetypeScanner;
+ writeSettings();
+}
+
+KAboutData* KGVPart::createAboutData()
+{
+ KAboutData* about = new KAboutData( "kghostview", I18N_NOOP( "KGhostView"),
+ KGHOSTVIEW_VERSION,
+ I18N_NOOP( "Viewer for PostScript (.ps, .eps) and Portable Document Format (.pdf) files"),
+ KAboutData::License_GPL,
+ "(C) 1998 Mark Donohoe, (C) 1999-2000 David Sweet, "
+ "(C) 2000-2003 Wilco Greven",
+ I18N_NOOP( "KGhostView displays, prints, and saves "
+ "PostScript and PDF files.\n"
+ "Based on original work by Tim Theisen." ) );
+ about->addAuthor( "Luís Pedro Coelho",
+ I18N_NOOP( "Current maintainer" ),
+ "luis@luispedro.org",
+ "http://luispedro.org" );
+ about->addAuthor( "Wilco Greven",
+ I18N_NOOP( "Maintainer 2000-2003" ),
+ "greven@kde.org" );
+ about->addAuthor( "David Sweet",
+ I18N_NOOP( "Maintainer 1999-2000" ),
+ "dsweet@kde.org",
+ "http://www.andamooka.org/~dsweet" );
+ about->addAuthor( "Mark Donohoe",
+ I18N_NOOP( "Original author" ),
+ "donohoe@kde.org" );
+ about->addAuthor( "David Faure",
+ I18N_NOOP( "Basis for shell"),
+ "faure@kde.org" );
+ about->addAuthor( "Daniel Duley",
+ I18N_NOOP( "Port to KParts" ),
+ "mosfet@kde.org" );
+ about->addAuthor( "Espen Sand",
+ I18N_NOOP( "Dialog boxes" ),
+ "espen@kde.org" );
+ about->addCredit( "Russell Lang of Ghostgum Software Pty Ltd",
+ I18N_NOOP( "for contributing GSView's DSC parser." ),
+ 0,
+ "http://www.ghostgum.com.au/" );
+ about->addCredit( "The Ghostscript authors",
+ 0, 0,
+ "http://www.cs.wisc.edu/~ghost/" );
+ return about;
+}
+
+bool KGVPart::closeURL()
+{
+ document()->close();
+ _psWidget->stopInterpreter();
+ _docManager->getThumbnailService()->reset();
+ _markList->clear();
+ _pageDecorator->hide();
+ _scrollBox->clear();
+ _isFileDirty = false;
+ if ( _job )
+ {
+ _job -> kill();
+ _job = 0;
+ }
+ if( _mimetypeScanner != 0 )
+ _mimetypeScanner->abort();
+ if( !m_file.isEmpty() )
+ _fileWatcher->removeFile( m_file );
+ _mimetype = QString::null;
+ updatePageDepActions();
+ stateChanged( "initState" );
+ return KParts::ReadOnlyPart::closeURL();
+}
+
+void KGVPart::writeSettings()
+{
+ KConfigGroup general( KGVFactory::instance()->config(), "General" );
+ if ( !_embeddedInKGhostView )
+ general.writeEntry( "Display Options", DisplayOptions::toString( miniWidget()->displayOptions() ) );
+ general.sync();
+}
+
+void KGVPart::readSettings()
+{
+ KConfigGroup general( KGVFactory::instance()->config(), "General" );
+
+ _showScrollBars->setChecked( Configuration::showScrollBars() );
+ showScrollBars( _showScrollBars->isChecked() );
+
+ _watchFile->setChecked( Configuration::watchFile() );
+ slotWatchFile();
+
+ _showPageList->setChecked( Configuration::showPageList() );
+ showMarkList( _showPageList->isChecked() );
+
+ _showPageLabels->setChecked( Configuration::watchFile() );
+ showPageLabels( _showPageLabels->isChecked() );
+
+ _showLogWindow = Configuration::messages();
+ if ( !_embeddedInKGhostView ) {
+ DisplayOptions options;
+ if ( DisplayOptions::fromString( options, general.readEntry( "Display Options" ) ) )
+ setDisplayOptions( options );
+ }
+ _psWidget->readSettings();
+}
+
+void KGVPart::slotScrollLeft()
+{
+ _pageView->scrollLeft();
+}
+
+void KGVPart::slotFlicker()
+{
+ if ( _psWidget ) _psWidget->setDoubleBuffering( _flick->isChecked() );
+}
+
+void KGVPart::slotScrollRight()
+{
+ _pageView->scrollRight();
+}
+
+void KGVPart::slotScrollUp()
+{
+ _pageView->scrollUp();
+}
+
+void KGVPart::slotScrollDown()
+{
+ _pageView->scrollDown();
+}
+
+void KGVPart::slotReadUp()
+{
+ if( !( document() && document()->isOpen() ) )
+ return;
+
+ if( !_pageView->readUp() ) {
+ if (_docManager->prevPage())
+ _pageView->scrollBottom();
+ }
+}
+
+void KGVPart::slotReadDown()
+{
+ if( !( document() && document()->isOpen() ) )
+ return;
+
+ if( !_pageView->readDown() ) {
+ if( _docManager->nextPage() )
+ _pageView->scrollTop();
+ }
+}
+
+void KGVPart::slotPrevPage()
+{
+ if( !document() || !document()->isOpen() ) return;
+ _docManager->prevPage();
+}
+
+void KGVPart::slotNextPage()
+{
+ if( !document() || !document()->isOpen() ) return;
+ _docManager->nextPage();
+}
+
+void KGVPart::slotGotoStart()
+{
+ _docManager->firstPage();
+ _pageView->scrollTop();
+}
+
+void KGVPart::slotGotoEnd()
+{
+ _docManager->lastPage();
+ _pageView->scrollTop();
+}
+
+void KGVPart::slotWatchFile()
+{
+ if( _watchFile->isChecked() )
+ _fileWatcher->startScan();
+ else {
+ _dirtyHandler->stop();
+ _fileWatcher->stopScan();
+ }
+}
+
+void KGVPart::slotCancelWatch()
+{
+ _fileWatcher->stopScan();
+ _watchFile->setChecked( false );
+}
+
+/*
+void KGVPart::slotFitWidth()
+{
+ _docManager->fitWidth( pageView()->viewport()->width() -
+ 2*( pageDecorator()->margin() + pageDecorator()->borderWidth() ) );
+}
+*/
+
+void KGVPart::updateZoomActions()
+{
+ if( !( document() && document()->isOpen() ) )
+ return;
+
+ _zoomIn->setEnabled(!_docManager->atMaxZoom());
+ _zoomOut->setEnabled(!_docManager->atMinZoom());
+ _zoomTo->setEnabled( true );
+ QStringList items = _zoomTo->items();
+ bool updateItems = false;
+ if (_customZoomIndex != -1)
+ {
+ items.remove(items.at(_customZoomIndex));
+ _customZoomIndex = -1;
+ updateItems = true;
+ }
+ double zoom = floor(miniWidget()->displayOptions().magnification()*1000.0) / 10.0;
+ unsigned idx = 0;
+ for ( QStringList::iterator first = items.begin(), last = items.end();
+ first != last;
+ ++first ) {
+ QString cur = *first;
+ cur.remove( cur.find( '%' ), 1 );
+ cur = cur.simplifyWhiteSpace();
+ bool ok = false;
+ double z = cur.toDouble(&ok);
+ if ( ok ) {
+ if (std::abs( z - zoom ) < 0.1 ) {
+ if (updateItems)
+ _zoomTo->setItems( items );
+ _zoomTo->setCurrentItem( idx );
+ return;
+ }
+ if ( z > zoom )
+ break;
+ }
+ ++idx;
+ }
+
+ // Show percentage that isn't predefined
+ QString str = QString( "%1%" ).arg( KGlobal::locale()->formatNumber( zoom, 2 ));
+ str.remove( KGlobal::locale()->decimalSymbol() + "00" );
+ items.insert( items.at(idx), 1, str );
+ _zoomTo->setItems( items );
+ _zoomTo->setCurrentItem( idx );
+ _customZoomIndex = idx;
+}
+
+void KGVPart::updatePageDepActions()
+{
+ bool hasDoc = document() && document()->isOpen();
+
+ _fitWidth->setEnabled( hasDoc );
+ _fitScreen->setEnabled( hasDoc );
+
+ _prevPage->setEnabled( hasDoc && !_docManager->atFirstPage() );
+ _firstPage->setEnabled( hasDoc && !_docManager->atFirstPage() );
+ _nextPage->setEnabled( hasDoc && !_docManager->atLastPage() );
+ _lastPage->setEnabled( hasDoc && !_docManager->atLastPage() );
+ _gotoPage->setEnabled( hasDoc &&
+ !(_docManager->atFirstPage() && _docManager->atLastPage()) );
+
+ updateReadUpDownActions();
+}
+
+void KGVPart::updateReadUpDownActions()
+{
+ if( !( document() && document()->isOpen() ) )
+ {
+ _readUp->setEnabled( false );
+ _readDown->setEnabled( false );
+ return;
+ }
+
+ if( _docManager->atFirstPage() && _pageView->atTop() )
+ _readUp->setEnabled( false );
+ else
+ _readUp->setEnabled( true );
+
+ if( _docManager->atLastPage() && _pageView->atBottom() )
+ _readDown->setEnabled( false );
+ else
+ _readDown->setEnabled( true );
+}
+
+bool KGVPart::openURL( const KURL& url )
+{
+ if( !url.isValid() )
+ return false;
+ if( !closeURL() )
+ return false;
+
+ m_url = url;
+ if ( !_stickyOptions ) _options.reset();
+
+ emit setWindowCaption( m_url.prettyURL() );
+
+ _mimetypeScanner = new KGVRun( m_url, 0, m_url.isLocalFile(), false );
+ connect( _mimetypeScanner, SIGNAL( finished( const QString& ) ),
+ SLOT( slotMimetypeFinished( const QString& ) ) );
+ connect( _mimetypeScanner, SIGNAL( error() ),
+ SLOT( slotMimetypeError() ) );
+
+ return true;
+}
+
+void KGVPart::openURLContinue()
+{
+ kdDebug(4500) << "KGVPart::openURLContinue()" << endl;
+ if( m_url.isLocalFile() )
+ {
+ emit started( 0 );
+ m_file = m_url.path();
+ document()->openFile( m_file, _mimetype );
+ }
+ else
+ {
+ m_bTemp = true;
+ // Use same extension as remote file. This is important for
+ // mimetype-determination (e.g. koffice)
+ QString extension;
+ QString fileName = m_url.fileName();
+ int extensionPos = fileName.findRev( '.' );
+ if( extensionPos != -1 )
+ extension = fileName.mid( extensionPos ); // keep the '.'
+ KTempFile tempFile( QString::null, extension );
+ m_file = tempFile.name();
+ _tmpFile.setName( m_file );
+ _tmpFile.open( IO_ReadWrite );
+
+ /*
+ d->m_job = KIO::file_copy( m_url, m_file, 0600, true, false, d->m_showProgressInfo );
+ emit started( d->m_job );
+ connect( d->m_job, SIGNAL( result( KIO::Job * ) ), this, SLOT( slotJobFinished ( KIO::Job * ) ) );
+ */
+
+ _job = KIO::get( m_url, false, isProgressInfoEnabled() );
+
+ connect( _job, SIGNAL( data( KIO::Job*, const QByteArray& ) ),
+ SLOT( slotData( KIO::Job*, const QByteArray& ) ) );
+ connect( _job, SIGNAL( result( KIO::Job* ) ),
+ SLOT( slotJobFinished( KIO::Job* ) ) );
+
+ emit started( _job );
+ }
+}
+
+bool KGVPart::openFile()
+{
+ return false;
+}
+
+void KGVPart::slotOpenFileCompleted()
+{
+ _docManager->getThumbnailService()->setEnabled( true );
+ if( _isFileDirty )
+ {
+ _docManager->redisplay();
+ _isFileDirty = false;
+ }
+ else
+ {
+ if ( !_stickyOptions ) setDisplayOptions( DisplayOptions() );
+ _stickyOptions = false;
+
+ stateChanged( "documentState" );
+ if ( !_fileWatcher->contains( m_file ) )
+ _fileWatcher->addFile( m_file );
+ slotWatchFile();
+ updateZoomActions();
+ emit completed();
+ }
+}
+
+void KGVPart::slotGhostscriptOutput( char* data, int len )
+{
+ _logWindow->append( QString::fromLocal8Bit( data, len ) );
+ if( _showLogWindow )
+ _logWindow->show();
+}
+
+
+void KGVPart::slotGhostscriptError( const QString& error )
+{
+ _logWindow->setLabel( i18n( "<qt>An error occurred in rendering.<br>"
+ "<strong>%1</strong><br>"
+ "The display may contain errors.<br>"
+ "Below are any error messages which were received from Ghostscript "
+ "(<nobr><strong>%2</strong></nobr>) "
+ "which may help you.</qt>" )
+ .arg( error )
+ .arg( Configuration::interpreter() ),
+ true );
+ // The true above makes it show a "configure gs" option, but maybe we
+ // should trigger an auto-redetection?
+ // LPC (13 Apr 2003)
+ _logWindow->show();
+}
+
+
+void KGVPart::guiActivateEvent( KParts::GUIActivateEvent* event )
+{
+ if( event->activated() && !_isGuiInitialized )
+ {
+ stateChanged( "initState" );
+ _isGuiInitialized = true;
+ }
+ KParts::ReadOnlyPart::guiActivateEvent( event );
+}
+
+void KGVPart::slotData( KIO::Job* job, const QByteArray& data )
+{
+ Q_ASSERT( _job == job );
+
+ kdDebug(4500) << "KGVPart::slotData: received " << data.size() << " bytes." << endl;
+
+ _tmpFile.writeBlock( data );
+}
+
+void KGVPart::slotMimetypeFinished( const QString& type )
+{
+ kdDebug(4500) << "KGVPart::slotMimetypeFinished( " << type << " )" << endl;
+ _mimetype = type;
+ if( !_mimetypeScanner || _mimetypeScanner->hasError() )
+ emit canceled( QString::null );
+ else
+ openURLContinue();
+ _mimetypeScanner = 0;
+}
+
+void KGVPart::slotMimetypeError()
+{
+ kdDebug(4500) << "KGVPart::slotMimetypeError()" << endl;
+ _mimetypeScanner = 0;
+ emit started( 0 );
+ //kapp->processEvents();
+ emit canceled( QString::null );
+}
+
+void KGVPart::slotJobFinished( KIO::Job* job )
+{
+ Q_ASSERT( _job == job );
+
+ kdDebug(4500) << "KGVPart::slotJobFinished" << endl;
+
+ _job = 0;
+
+ _tmpFile.close();
+
+ if( job->error() )
+ emit canceled( job->errorString() );
+ else
+ document()->openFile( m_file, _mimetype );
+}
+
+void KGVPart::slotFileDirty( const QString& fileName )
+{
+ // The beauty of this is that each start cancels the previous one.
+ // This means that timeout() is only fired when there have
+ // no changes to the file for the last 750 milisecs.
+ // This is supposed to ensure that we don't update on every other byte
+ // that gets written to the file.
+ if ( fileName == m_file )
+ {
+ _dirtyHandler->start( 750, true );
+ }
+}
+
+void KGVPart::slotDoFileDirty()
+{
+ kdDebug(4500) << "KGVPart::File changed" << endl;
+ _isFileDirty = true;
+ reloadFile();
+}
+
+void KGVPart::slotNewPage( int )
+{
+ updatePageDepActions();
+ //media->setCurrentItem (miniWidget()->getSize()-1);
+ //orientation->setCurrentItem (miniWidget()->getOrientation()-1);
+ //TODO -- zoom
+}
+
+void KGVPart::slotPageMoved( int, int )
+{
+ updateReadUpDownActions();
+}
+
+void KGVPart::slotOrientation( int id )
+{
+ switch( id ) {
+ case 0: miniWidget()->restoreOverrideOrientation(); break;
+ case 1: miniWidget()->setOverrideOrientation( CDSC_PORTRAIT ); break;
+ case 2: miniWidget()->setOverrideOrientation( CDSC_LANDSCAPE ); break;
+ case 3: miniWidget()->setOverrideOrientation( CDSC_UPSIDEDOWN ); break;
+ case 4: miniWidget()->setOverrideOrientation( CDSC_SEASCAPE ); break;
+ default: ;
+ }
+}
+
+void KGVPart::slotMedia( int id )
+{
+ if( id == 0 )
+ miniWidget()->restoreOverridePageMedia();
+ else
+ miniWidget()->setOverridePageMedia( _document->mediaNames()[id-1] );
+}
+
+void KGVPart::showScrollBars( bool show )
+{
+ _pageView->enableScrollBars( show );
+}
+
+void KGVPart::showMarkList( bool show )
+{
+ _markList->setShown( show );
+ _scrollBox->setShown( show );
+ _divider->setShown( show );
+}
+
+void KGVPart::showPageLabels( bool show )
+{
+ _docManager->enablePageLabels( show );
+}
+
+void KGVPart::slotZoomIn()
+{
+ _docManager->zoomIn();
+ updateZoomActions();
+}
+
+void KGVPart::slotZoomOut()
+{
+ _docManager->zoomOut();
+ updateZoomActions();
+}
+
+void KGVPart::slotZoom( const QString& nz )
+{
+ QString z = nz;
+ double zoom;
+ z.remove( z.find( '%' ), 1 );
+ zoom = KGlobal::locale()->readNumber( z ) / 100;
+ kdDebug( 4500 ) << "ZOOM = " << nz << ", setting zoom = " << zoom << endl;
+
+ DisplayOptions options = miniWidget()->displayOptions();
+ options.setMagnification( zoom );
+ miniWidget()->setDisplayOptions( options );
+ miniWidget()->redisplay();
+ _mainWidget->setFocus();
+ updateZoomActions();
+}
+
+void KGVPart::slotFitToPage()
+{
+ kdDebug(4500) << "KGVPart::slotFitToPage()" << endl;
+ if( pageView()->page() )
+ miniWidget()->fitWidth( pageView()->viewport()->width() - 16 );
+ // We subtract 16 pixels because of the page decoration.
+ updateZoomActions();
+}
+
+void KGVPart::slotFitToScreen()
+{
+ kdDebug(4500) << "KGVPart::slotFitToScreen()" << endl;
+ if ( _fitTimer->isActive() ) {
+ disconnect( _fitTimer, SIGNAL( timeout() ), this, 0 );
+ connect( _fitTimer, SIGNAL( timeout() ), SLOT( slotDoFitToScreen() ) );
+ }
+ else slotDoFitToScreen();
+}
+
+void KGVPart::slotDoFitToScreen()
+{
+ kdDebug(4500) << "KGVPart::slotDoFitToScreen()" << endl;
+ if( pageView()->page() )
+ miniWidget()->fitWidthHeight( pageView()->viewport()->width() - 16,
+ pageView()->viewport()->height() - 16 );
+ updateZoomActions();
+}
+
+void KGVPart::reloadFile()
+{
+ _psWidget->stopInterpreter();
+ _docManager->getThumbnailService()->reset();
+ document()->openFile( m_file, _mimetype );
+}
+
+void KGVPart::slotConfigure()
+{
+ ConfigDialog::showSettings( this );
+}
+
+void KGVPart::slotConfigurationChanged()
+{
+ readSettings();
+ _psWidget->readSettings();
+ miniWidget()->redisplay();
+}
+
+void KGVPart::setDisplayOptions( const DisplayOptions& options )
+{
+ kdDebug(4500) << "KGVPart::setDisplayOptions()" << endl;
+ _stickyOptions = true;
+ _markList->select( options.page() );
+ _docManager->setDisplayOptions( options );
+ _selectOrientation->setCurrentItem( options.overrideOrientation() );
+ QStringList medias = document()->mediaNames();
+ QStringList::Iterator now = medias.find( options.overridePageMedia() );
+ if ( now != medias.end() ){
+ // The options are displayed in inverted order.
+ // Therefore, size() - index gets you the display index
+ _selectMedia->setCurrentItem( medias.size() - KGV::distance( medias.begin(), now ) );
+ } else {
+ _selectMedia->setCurrentItem( 0 );
+ }
+}
+
+
+KGVBrowserExtension::KGVBrowserExtension( KGVPart *parent ) :
+ KParts::BrowserExtension( parent, "KGVBrowserExtension" )
+{
+ emit enableAction( "print", true );
+ setURLDropHandlingEnabled( true );
+}
+
+void KGVBrowserExtension::print()
+{
+ ((KGVPart *)parent())->document()->print();
+}
+
+
+KGVRun::KGVRun( const KURL& url, mode_t mode, bool isLocalFile,
+ bool showProgressInfo ) :
+ KRun( url, mode, isLocalFile, showProgressInfo )
+{
+ connect( this, SIGNAL( finished() ), SLOT( emitFinishedWithMimetype() ) );
+}
+
+KGVRun::~KGVRun()
+{}
+
+void KGVRun::foundMimeType( const QString& mimetype )
+{
+ kdDebug(4500) << "KGVRun::foundMimeType( " << mimetype << " )" << endl;
+
+ if( m_job && m_job->inherits( "KIO::TransferJob" ) )
+ {
+ KIO::TransferJob *job = static_cast< KIO::TransferJob* >( m_job );
+ job->putOnHold();
+ m_job = 0;
+ }
+
+ _mimetype = mimetype;
+
+ m_bFinished = true;
+ m_timer.start( 0, true );
+}
+
+
+void KGVPart::updateFullScreen( bool fs )
+{
+ if ( fs ) showMarkList( false );
+ else showMarkList( _showPageList->isChecked() );
+}
+
+void KGVPart::showPopup( int, int, const QPoint& pos )
+{
+ _popup->exec( pos );
+}
+
+#include "kgv_view.moc"
+
+
+// vim:sw=4:sts=4:ts=8:sta:tw=78:noet