#include <config.h> #include <tdeaction.h> #include <tdeapplication.h> #include <kdebug.h> #include <tdefiledialog.h> #include <kiconloader.h> #include <tdeio/job.h> #include <tdelocale.h> #include <tdemessagebox.h> #include <kprinter.h> #include <kstdaction.h> #include <tqobject.h> #include <tqlayout.h> #include <tqpaintdevicemetrics.h> #include <tqprogressdialog.h> #include <tqsplitter.h> #include <tqurl.h> #include <tqtoolbox.h> #include <tqvbox.h> #include "documentWidget.h" #include "marklist.h" #include "tableOfContents.h" #include "kprintDialogPage_pageoptions.h" #include "kvsprefs.h" #include "tdemultipage.h" #include "pageNumber.h" #include "renderedDocumentPagePrinter.h" #include "searchWidget.h" #include "textBox.h" #include "zoomlimits.h" //#define DEBUG_KMULTIPAGE KMultiPage::KMultiPage(TQWidget *parentWidget, const char *widgetName, TQObject *parent, const char *name) : DCOPObject("tdemultipage"), KParts::ReadOnlyPart(parent, name) { // For reasons which I don't understand, the initialization of the // DCOPObject above does not work properly, the name is ignored. It // works fine if we repeat the name here. -- Stefan Kebekus // This is because of the virtual inheritance. Get rid of it (but it's BC, and this is a lib...) -- DF setObjId("tdemultipage"); parentWdg = parentWidget; lastCurrentPage = 0; timer_id = -1; searchInProgress = false; TQVBox* verticalBox = new TQVBox(parentWidget); verticalBox->setFocusPolicy(TQ_StrongFocus); setWidget(verticalBox); splitterWidget = new TQSplitter(verticalBox, widgetName); splitterWidget->setOpaqueResize(false); splitterWidget->setSizePolicy(TQSizePolicy::MinimumExpanding, TQSizePolicy::MinimumExpanding); // Create SideBar sideBar = new TQToolBox(splitterWidget, "sidebar"); // Create ContentsList tableOfContents = new TableOfContents(sideBar); sideBar->addItem(tableOfContents, TQIconSet(SmallIcon("contents")), i18n("Contents")); connect(tableOfContents, TQT_SIGNAL(gotoPage(const Anchor&)), this, TQT_SLOT(gotoPage(const Anchor&))); // Create MarkList _markList = new MarkList(sideBar, "marklist"); sideBar->addItem(_markList, TQIconSet(SmallIcon("thumbnail")), i18n("Thumbnails")); // Restore state of the sidebar sideBar->setCurrentItem(sideBar->item(KVSPrefs::sideBarItem())); splitterWidget->setResizeMode(sideBar, TQSplitter::KeepSize); connect(_markList, TQT_SIGNAL(selected(const PageNumber&)), this, TQT_SLOT(gotoPage(const PageNumber&))); _scrollView = new PageView(splitterWidget, widgetName); // Create Search Panel searchWidget = new SearchWidget(verticalBox); searchWidget->hide(); connect(searchWidget, TQT_SIGNAL(findNextText()), this, TQT_SLOT(findNextText())); connect(searchWidget, TQT_SIGNAL(findPrevText()), this, TQT_SLOT(findPrevText())); sideBar->setMinimumWidth(80); sideBar->setMaximumWidth(300); connect(_scrollView, TQT_SIGNAL(currentPageChanged(const PageNumber&)), this, TQT_SLOT(setCurrentPageNumber(const PageNumber&))); connect(_scrollView, TQT_SIGNAL(viewSizeChanged(const TQSize&)), scrollView(), TQT_SLOT(calculateCurrentPageNumber())); connect(_scrollView, TQT_SIGNAL(wheelEventReceived(TQWheelEvent *)), this, TQT_SLOT(wheelEvent(TQWheelEvent*))); connect(this, TQT_SIGNAL(enableMoveTool(bool)), _scrollView, TQT_SLOT(slotEnableMoveTool(bool))); splitterWidget->setCollapsible(sideBar, false); splitterWidget->setSizes(KVSPrefs::guiLayout()); connect(searchWidget, TQT_SIGNAL(searchEnabled(bool)), this, TQT_SIGNAL(searchEnabled(bool))); connect(searchWidget, TQT_SIGNAL(stopSearch()), this, TQT_SLOT(stopSearch())); } KMultiPage::~KMultiPage() { writeSettings(); if (timer_id != -1) killTimer(timer_id); delete pageCache; } void KMultiPage::readSettings() { } void KMultiPage::writeSettings() { // Save TOC layout tableOfContents->writeSettings(); KVSPrefs::setGuiLayout(splitterWidget->sizes()); // Save state of the sidebar KVSPrefs::setSideBarItem(sideBar->indexOf(sideBar->currentItem())); KVSPrefs::writeConfig(); } TQString KMultiPage::name_of_current_file() { return m_file; } bool KMultiPage::is_file_loaded(const TQString& filename) { return (filename == m_file); } void KMultiPage::slotSave_defaultFilename() { slotSave(); } void KMultiPage::slotSave() { // Try to guess the proper ending... TQString formats; TQString ending; int rindex = m_file.findRev("."); if (rindex == -1) { ending = TQString(); formats = TQString(); } else { ending = m_file.mid(rindex); // e.g. ".dvi" formats = fileFormats().grep(ending).join("\n"); } TQString fileName = KFileDialog::getSaveFileName(TQString(), formats, 0, i18n("Save File As")); if (fileName.isEmpty()) return; // Add the ending to the filename. I hope the user likes it that // way. if (!ending.isEmpty() && fileName.find(ending) == -1) fileName = fileName+ending; if (TQFile(fileName).exists()) { int r = KMessageBox::warningContinueCancel (0, i18n("The file %1\nexists. Shall I overwrite that file?").arg(fileName), i18n("Overwrite File"), i18n("Overwrite")); if (r == KMessageBox::Cancel) return; } TDEIO::Job *job = TDEIO::file_copy( KURL( m_file ), KURL( fileName ), 0600, true, false, true ); connect( job, TQT_SIGNAL( result( TDEIO::Job * ) ), this, TQT_SLOT( slotIOJobFinished ( TDEIO::Job * ) ) ); return; } void KMultiPage::setFile(bool) { return; } bool KMultiPage::closeURL() { #ifdef DEBUG_KMULTIPAGE kdDebug(1233) << "KMultiPage::closeURL()" << endl; #endif if (renderer.isNull()) return false; // Clear navigation history. document_history.clear(); // Close the file. renderer->setFile(TQString(), KURL()); renderer->clear(); // Delete Page Widgets. widgetList.setAutoDelete(true); widgetList.resize(0); widgetList.setAutoDelete(false); // Update ScrollView. scrollView()->layoutPages(); enableActions(false); // Clear Thumbnail List. markList()->clear(); // Clear Table of Contents tableOfContents->clear(); // Clear Status Bar emit setStatusBarText(TQString()); return true; } void KMultiPage::slotIOJobFinished ( TDEIO::Job *job ) { if ( job->error() ) job->showErrorDialog( 0L ); } void KMultiPage::slotShowScrollbars(bool status) { _scrollView->slotShowScrollbars(status); } void KMultiPage::slotShowSidebar(bool show) { if (show) sideBar->show(); else sideBar->hide(); } void KMultiPage::slotShowThumbnails(bool show) { markList()->slotShowThumbnails(show); } void KMultiPage::slotSetFullPage(bool fullpage) { _scrollView->setFullScreenMode(fullpage); if (fullpage) slotShowSidebar(false); } void KMultiPage::preferencesChanged() { // We need to read the config options otherwise the KVSPrefs-object would // not be syncronized between the kviewpart and the tdemultipage. KVSPrefs::self()->readConfig(); slotShowThumbnails(KVSPrefs::showThumbnails()); // if we are in overviewmode and the number of columns or rows has changed if (scrollView()->overviewMode() && (scrollView()->getNrColumns() != KVSPrefs::overviewModeColumns() || scrollView()->getNrRows() != KVSPrefs::overviewModeRows())) { setViewMode(KVSPrefs::EnumViewMode::Overview); } if (KVSPrefs::changeColors() && KVSPrefs::renderMode() == KVSPrefs::EnumRenderMode::Paper) renderer->setAccessibleBackground(true, KVSPrefs::paperColor()); else renderer->setAccessibleBackground(false); renderModeChanged(); } void KMultiPage::setViewMode(int mode) { #ifdef DEBUG_KMULTIPAGE kdDebug(1233) << "KMultiPage::setViewMode(" << mode << ")" << endl; #endif // Save the current page number because when we are changing the columns // and rows in the scrollview the currently shown Page probably out of view. PageNumber currentPage = currentPageNumber(); // Save viewMode for future uses of KViewShell switch (mode) { case KVSPrefs::EnumViewMode::SinglePage: KVSPrefs::setViewMode(KVSPrefs::EnumViewMode::SinglePage); // Don't do anything if the view mode is already set if ((scrollView()->getNrColumns() == 1) && (scrollView()->getNrRows() == 1) && (scrollView()->isContinuous() == false)) return; scrollView()->setNrColumns(1); scrollView()->setNrRows(1); scrollView()->setContinuousViewMode(false); // We scroll the view to the top, so that top and not the bottom // of the visible page is shown. scrollView()->scrollTop(); break; case KVSPrefs::EnumViewMode::ContinuousFacing: KVSPrefs::setViewMode(KVSPrefs::EnumViewMode::ContinuousFacing); // Don't do anything if the view mode is already set if ((scrollView()->getNrColumns() == 2) && (scrollView()->getNrRows() == 1) && (scrollView()->isContinuous() == true)) return; scrollView()->setNrColumns(2); scrollView()->setNrRows(1); scrollView()->setContinuousViewMode(true); break; case KVSPrefs::EnumViewMode::Overview: KVSPrefs::setViewMode(KVSPrefs::EnumViewMode::Overview); // Don't do anything if the view mode is already set if ((scrollView()->getNrColumns() == KVSPrefs::overviewModeColumns()) && (scrollView()->getNrRows() == KVSPrefs::overviewModeRows()) && (scrollView()->isContinuous() == false)) return; scrollView()->setNrColumns(KVSPrefs::overviewModeColumns()); scrollView()->setNrRows(KVSPrefs::overviewModeRows()); scrollView()->setContinuousViewMode(false); // We scroll the view to the top, so that top and not the bottom // of the visible tableau is shown. scrollView()->scrollTop(); break; default: //KVSPrefs::EnumViewMode::Continuous KVSPrefs::setViewMode(KVSPrefs::EnumViewMode::Continuous); // Don't do anything if the view mode is already set if ((scrollView()->getNrColumns() == 1) && (scrollView()->getNrRows() == 1) && (scrollView()->isContinuous() == true)) return; scrollView()->setNrColumns(1); scrollView()->setNrRows(1); scrollView()->setContinuousViewMode(true); } generateDocumentWidgets(currentPage); KVSPrefs::writeConfig(); emit viewModeChanged(); } void KMultiPage::initializePageCache() { pageCache = new DocumentPageCache(); } DocumentWidget* KMultiPage::createDocumentWidget() { DocumentWidget* documentWidget = new DocumentWidget(scrollView()->viewport(), scrollView(), pageCache, "singlePageWidget"); connect(documentWidget, TQT_SIGNAL(clearSelection()), this, TQT_SLOT(clearSelection())); connect(this, TQT_SIGNAL(enableMoveTool(bool)), documentWidget, TQT_SLOT(slotEnableMoveTool(bool))); return documentWidget; } void KMultiPage::generateDocumentWidgets(const PageNumber& _startPage) { PageNumber startPage = _startPage; #ifdef DEBUG_KMULTIPAGE kdDebug(1233) << "KMultiPage::generateDocumentWidgets(" << startPage << ")" << endl; #endif // Do nothing if no document is loaded. if (getRenderer().isNull() || getRenderer()->isEmpty()) return; // This function is only called with an invalid pagenumber, when // the file has been loaded or reloaded. bool reload = !startPage.isValid(); if (reload) { // Find the number of the current page, for later use. startPage = currentPageNumber(); } // Make sure that startPage is in the permissible range. if (startPage < 1) startPage = 1; if (startPage > numberOfPages()) startPage = numberOfPages(); unsigned int tableauStartPage = startPage; // Find out how many widgets are needed, and resize the widgetList accordingly. widgetList.setAutoDelete(true); TQ_UINT16 oldwidgetListSize = widgetList.size(); if (numberOfPages() == 0) widgetList.resize(0); else { switch (KVSPrefs::viewMode()) { case KVSPrefs::EnumViewMode::SinglePage: widgetList.resize(1); break; case KVSPrefs::EnumViewMode::Overview: { // Calculate the number of pages shown in overview mode. unsigned int visiblePages = KVSPrefs::overviewModeColumns() * KVSPrefs::overviewModeRows(); // Calculate the number of the first page in the tableau. tableauStartPage = startPage - ((startPage - 1) % visiblePages); // We cannot have more widgets then pages in the document. visiblePages = TQMIN(visiblePages, numberOfPages() - tableauStartPage + 1); if (widgetList.size() != visiblePages) widgetList.resize(visiblePages); break; } default: // In KVS_Continuous and KVS_ContinuousFacing all pages in the document are shown. widgetList.resize(numberOfPages()); } } bool isWidgetListResized = (widgetList.size() != oldwidgetListSize); widgetList.setAutoDelete(false); // If the widgetList is empty, there is nothing left to do. if (widgetList.size() == 0) { scrollView()->addChild(&widgetList); return; } // Allocate DocumentWidget structures so that all entries of // widgetList point to a valid DocumentWidget. DocumentWidget *documentWidget; for(TQ_UINT16 i=0; i<widgetList.size(); i++) { documentWidget = widgetList[i]; if (documentWidget == 0) { documentWidget = createDocumentWidget(); widgetList.insert(i, documentWidget); documentWidget->show(); connect(documentWidget, TQT_SIGNAL(localLink(const TQString &)), this, TQT_SLOT(handleLocalLink(const TQString &))); connect(documentWidget, TQT_SIGNAL(setStatusBarText(const TQString&)), this, TQT_SIGNAL(setStatusBarText(const TQString&)) ); } } // Set the page numbers for the newly allocated widgets. How this is // done depends on the viewMode. if (KVSPrefs::viewMode() == KVSPrefs::EnumViewMode::SinglePage) { // In KVS_SinglePage mode, any number between 1 and the maximum // number of pages is acceptable. If an acceptable value is found, // nothing is done, and otherwise '1' is set as a default. documentWidget = widgetList[0]; if (documentWidget != 0) { // Paranoia safety check documentWidget->setPageNumber(startPage); documentWidget->update(); } else kdError(4300) << "Zero-Pointer in widgetList in KMultiPage::generateDocumentWidgets()" << endl; } else { // In all other modes, the widgets will be numbered continuously, // starting from firstShownPage. for(TQ_UINT16 i=0; i<widgetList.size(); i++) { documentWidget = widgetList[i]; if (documentWidget != 0) // Paranoia safety check { if (KVSPrefs::viewMode() == KVSPrefs::EnumViewMode::Overview) documentWidget->setPageNumber(i+tableauStartPage); else documentWidget->setPageNumber(i+1); } else kdError(4300) << "Zero-Pointer in widgetList in KMultiPage::generateDocumentWidgets()" << endl; } } // Make the changes in the widgetList known to the scrollview. so // that the scrollview may update its contents. scrollView()->addChild(&widgetList); // If the number of widgets has changed, or the viewmode has been changed the widget // that displays the current page may not be visible anymore. Bring it back into focus. if (isWidgetListResized || !reload) gotoPage(startPage); } bool KMultiPage::gotoPage(const PageNumber& page) { return gotoPage(page, 0, true); } bool KMultiPage::gotoPage(const PageNumber& page, int y, bool isLink) { #ifdef DEBUG_KMULTIPAGE kdDebug(1233) << "KMultiPage::gotoPage()" << endl; #endif if (widgetList.size() == 0) { kdError(4300) << "KMultiPage::gotoPage(" << page << ", y) called, but widgetList is empty" << endl; return false; } if (!page.isValid()) { kdDebug(1223) << "KMultiPage::gotoPage(" << page << ") invalid pageNumber." << endl; return false; } if (isLink) document_history.add(page, y); DocumentWidget* pageWidget; // If we are in overview viewmode if (KVSPrefs::viewMode() == KVSPrefs::EnumViewMode::Overview) { unsigned int visiblePages = KVSPrefs::overviewModeColumns() * KVSPrefs::overviewModeRows(); // Pagenumber of the first visibile Page in the current tableau unsigned int firstPage = ((DocumentWidget*)widgetList[0])->getPageNumber(); // Pagenumber of the first page in the new tableau. unsigned int tableauStartPage = page + 1 - (page % visiblePages); // If these numbers arn't equal "page" is not in the current tableu. if (firstPage != tableauStartPage) // widgets need to be updated { if ((numberOfPages() - tableauStartPage + 1 < visiblePages) || (widgetList.size() < visiblePages)) { // resize widgetList // the pages are also set correctly by "generateDocumentWidgets" generateDocumentWidgets(tableauStartPage); } else { // "page" is not shown in the scrollview, so we have to switch widgets. // Here we don't need to resize the widgetList. for (unsigned int i = 0; i < widgetList.size(); i++) { pageWidget = (DocumentWidget*)(widgetList[i]); if (pageWidget != 0) pageWidget->setPageNumber(tableauStartPage + i); } scrollView()->layoutPages(); } } // move scrollview to "page". // Make the widget pageWidget visible in the scrollview. Somehow this // doesn't seem to trigger the signal contentsMoved in the // TQScrollview, so that we better call setCurrentPage() ourselves. pageWidget = (DocumentWidget*)(widgetList[page % visiblePages]); scrollView()->moveViewportToWidget(pageWidget, y); // Set current page number. setCurrentPageNumber(page); return true; } else if (widgetList.size() == 1) { // If the widget list contains only a single element, then either // the document contains only one page, or we are in "single page" // view mode. In either case, we set the page number of the single // widget to 'page' pageWidget = (DocumentWidget*)(widgetList[0]); // Paranoia security check if (pageWidget == 0) { kdError(4300) << "KMultiPage::goto_Page() called with widgetList.size() == 1, but widgetList[0] == 0" << endl; return false; } if (pageCache->sizeOfPageInPixel(currentPageNumber()) == pageCache->sizeOfPageInPixel(page)) { // We are rendering the page before we switch the widget to the new page. // To make a smooth transition. We only do this if the size of the current and new page are equal, // otherwise we would have to render the page twice, if autozoom is enabled. pageCache->getPage(page); } pageWidget->setPageNumber(page); scrollView()->layoutPages(); scrollView()->moveViewportToWidget(pageWidget, y); } else { // There are multiple widgets, then we are either in the // "Continuous" or in the "Continouous-Facing" view mode. In that // case, we find the widget which is supposed to display page // 'page' and move the scrollview to make it visible // Paranoia security checks if (widgetList.size() < page) { kdError(4300) << "KMultiPage::goto_Page(page,y ) called with widgetList.size()=" << widgetList.size() << ", and page=" << page << endl; return false; } pageWidget = (DocumentWidget*)(widgetList[page-1]); if (pageWidget == 0) { kdError(4300) << "KMultiPage::goto_Page() called with widgetList.size() > 1, but widgetList[page] == 0" << endl; return false; } scrollView()->moveViewportToWidget(pageWidget, y); } if (isLink && y != 0) pageWidget->flash(y); // Set current page number. setCurrentPageNumber(page); return true; } void KMultiPage::handleLocalLink(const TQString &linkText) { #ifdef DEBUG_SPECIAL kdDebug(4300) << "hit: local link to " << linkText << endl; #endif if (renderer.isNull()) { kdError(4300) << "KMultiPage::handleLocalLink( " << linkText << " ) called, but renderer==0" << endl; return; } TQString locallink; if (linkText[0] == '#' ) locallink = linkText.mid(1); // Drop the '#' at the beginning else locallink = linkText; Anchor anch = renderer->findAnchor(locallink); if (anch.isValid()) gotoPage(anch); else { if (linkText[0] != '#' ) { // We could in principle use TDEIO::Netaccess::run() here, but // it is perhaps not a very good idea to allow a DVI-file to // specify arbitrary commands, such as "rm -rvf /". Using // the kfmclient seems to be MUCH safer. TQUrl DVI_Url(m_file); TQUrl Link_Url(DVI_Url, linkText, true); TQStringList args; args << "openURL"; args << Link_Url.toString(); kapp->tdeinitExec("kfmclient", args); } } } void KMultiPage::setCurrentPageNumber(const PageNumber& page) { #ifdef DEBUG_KMULTIPAGE kdDebug(1233) << "KMultiPage::setCurrentPageNumber()" << endl; #endif if (page != currentPageNumber()) { markList()->setCurrentPageNumber(page); emit pageInfo(numberOfPages(), currentPageNumber()); } } PageNumber KMultiPage::currentPageNumber() { return markList()->currentPageNumber(); } void KMultiPage::doGoBack() { HistoryItem *it = document_history.back(); if (it != 0) gotoPage(it->page, it->ypos, false); // Do not add a history item. else kdDebug(4300) << "Faulty return -- bad history buffer" << endl; return; } void KMultiPage::doGoForward() { HistoryItem *it = document_history.forward(); if (it != 0) gotoPage(it->page, it->ypos, false); // Do not add a history item. else kdDebug(4300) << "Faulty return -- bad history buffer" << endl; return; } void KMultiPage::renderModeChanged() { pageCache->clear(); generateDocumentWidgets(); scrollView()->layoutPages(); for (TQ_UINT16 i=0; i < widgetList.size(); i++) { DocumentWidget* documentWidget = widgetList[i]; if (documentWidget == 0) continue; documentWidget->update(); } markList()->repaintThumbnails(); } void KMultiPage::repaintAllVisibleWidgets() { #ifdef DEBUG_KMULTIPAGE kdDebug(1233) << "KMultiPage::repaintAllVisibleWidgets()" << endl; #endif bool everResized = false; // Go through the list of widgets and resize them, if necessary for(TQ_UINT16 i=0; i<widgetList.size(); i++) { DocumentWidget* documentWidget = widgetList[i]; if (documentWidget == 0) continue; // Resize, if necessary TQSize pageSize = pageCache->sizeOfPageInPixel(documentWidget->getPageNumber()); if (pageSize != documentWidget->pageSize()) { documentWidget->setPageSize(pageSize); everResized = true; } } // If at least one widget was resized, all widgets should be // re-aligned. This will automatically update all necessary // widgets. if (everResized == true) scrollView()->layoutPages(true); } double KMultiPage::setZoom(double zoom) { #ifdef DEBUG_KMULTIPAGE kdDebug(1233) << "KMultiPage::setZoom(" << zoom << ")" << endl; #endif if (zoom < ZoomLimits::MinZoom/1000.0) zoom = ZoomLimits::MinZoom/1000.0; if (zoom > ZoomLimits::MaxZoom/1000.0) zoom = ZoomLimits::MaxZoom/1000.0; pageCache->setResolution(TQPaintDevice::x11AppDpiX()*zoom); emit zoomChanged(); return zoom; } void KMultiPage::print() { // Paranoid safety checks if (renderer.isNull()) return; if (renderer->isEmpty()) return; // Allocate the printer structure KPrinter *printer = getPrinter(); if (printer == 0) return; // initialize the printer using the print dialog if ( printer->setup(parentWdg, i18n("Print %1").arg(m_file.section('/', -1))) ) { // Now do the printing. TQValueList<int> pageList = printer->pageList(); if (pageList.isEmpty()) printer->abort(); else { printer->setCreator("kviewshell"); printer->setDocName(m_file); RenderedDocumentPagePrinter rdpp(printer); // Obtain papersize information that is required to perform // the resizing and centering, if this is wanted by the user. Length paperWidth, paperHeight; TQPaintDeviceMetrics pdm(printer); paperWidth.setLength_in_mm(pdm.widthMM()); paperHeight.setLength_in_mm(pdm.heightMM()); TQValueList<int>::ConstIterator it = pageList.begin(); while (true) { SimplePageSize paper_s(paperWidth, paperHeight); // Printing usually takes a while. This is to keep the GUI // updated. tqApp->processEvents(); TQPainter *paint = rdpp.getPainter(); if (paint != 0) { // Before drawing the page, we figure out the zoom-value, // taking the "page sizes and placement" options from the // printer dialog into account double factual_zoom = 1.0; // Obtain pagesize information that is required to perform the // resizing and centering, if this is wanted by the user. SimplePageSize page_s = sizeOfPage(*it); paint->save(); // Rotate the page, if appropriate. By default, page // rotation is enabled. This is also hardcoded into // KPrintDialogPage_PageOptions.cpp if ((page_s.isPortrait() != paper_s.isPortrait()) && (printer->option( "kde-kviewshell-rotatepage" ) != "false")) { paint->rotate(-90); paint->translate(-printer->resolution()*paperHeight.getLength_in_inch(), 0.0); paper_s = paper_s.rotate90(); } double suggested_zoom = page_s.zoomToFitInto(paper_s); // By default, "shrink page" and "expand page" are off. This // is also hardcoded into KPrintDialogPage_PageOptions.cpp if ((suggested_zoom < 1.0) && (printer->option( "kde-kviewshell-shrinkpage" ) == "true")) factual_zoom = suggested_zoom; if ((suggested_zoom > 1.0) && (printer->option( "kde-kviewshell-expandpage" ) == "true")) factual_zoom = suggested_zoom; Length delX, delY; // By default, "center page" is on. This is also hardcoded // into KPrintDialogPage_PageOptions.cpp if (printer->option( "kde-kviewshell-centerpage" ) != "false") { delX = (paper_s.width() - page_s.width()*factual_zoom)/2.0; delY = (paper_s.height() - page_s.height()*factual_zoom)/2.0; } // Now draw the page. rdpp.setPageNumber(*it); double resolution = factual_zoom*printer->resolution(); paint->translate(resolution*delX.getLength_in_inch(), resolution*delY.getLength_in_inch()); renderer->drawPage(resolution, &rdpp); paint->restore(); } ++it; if ((it == pageList.end()) || (printer->aborted() == true)) break; printer->newPage(); } // At this point the rdpp is destructed. The last page is then // printed. } } delete printer; } void KMultiPage::setRenderer(DocumentRenderer* _renderer) { renderer = _renderer; // Initialize documentPageCache. initializePageCache(); pageCache->setRenderer(renderer); _markList->setPageCache(pageCache); // Clear widget list. widgetList.resize(0); // Relay signals. connect(renderer, TQT_SIGNAL(setStatusBarText(const TQString&)), this, TQT_SIGNAL(setStatusBarText(const TQString&))); connect(pageCache, TQT_SIGNAL(paperSizeChanged()), this, TQT_SLOT(renderModeChanged())); connect(pageCache, TQT_SIGNAL(textSelected(bool)), this, TQT_SIGNAL(textSelected(bool))); connect(renderer, TQT_SIGNAL(documentIsChanged()), this, TQT_SLOT(renderModeChanged())); connect(this, TQT_SIGNAL(zoomChanged()), this, TQT_SLOT(repaintAllVisibleWidgets())); } void KMultiPage::updateWidgetSize(const PageNumber& pageNumber) { for(TQ_UINT16 i=0; i<widgetList.size(); i++) { DocumentWidget* documentWidget = widgetList[i]; if (documentWidget == 0) continue; if (documentWidget->getPageNumber() == pageNumber) { // Resize, if necessary TQSize pageSize = pageCache->sizeOfPageInPixel(documentWidget->getPageNumber()); if (pageSize != documentWidget->pageSize()) { documentWidget->setPageSize(pageSize); scrollView()->layoutPages(); } // We have just one widget per page. break; } } // Update marklist markList()->updateWidgetSize(pageNumber); } PageNumber KMultiPage::widestPage() const { Length maxWidth; PageNumber pageNumber = 1; for (int i = 1; i <= numberOfPages(); i++) { Length width = pageCache->sizeOfPage(i).width(); if (width > maxWidth) { maxWidth = width; pageNumber = i; } } return pageNumber; } double KMultiPage::zoomForWidthColumns(unsigned int viewportWidth) const { Length maxLeftColumnWidth; Length maxRightColumnWidth; Length maxWidth; PageNumber widestPageLeft; PageNumber widestPageRight; for (int i = 1; i <= numberOfPages(); i++) { Length width = pageCache->sizeOfPage(i).width(); if ( i % 2 == 0) // page is in left column { if (width > maxLeftColumnWidth) { maxLeftColumnWidth = width; widestPageLeft = i; } } if ( i % 2 == 1) // page is in right column { if (width > maxRightColumnWidth) maxRightColumnWidth = width; widestPageRight = i; } } double ratio = maxLeftColumnWidth / (maxLeftColumnWidth + maxRightColumnWidth); // This number is the amount of space the left column should occupy in the viewport. unsigned int leftTargetWidth = (unsigned int)(ratio * viewportWidth); return pageCache->sizeOfPage(widestPageLeft).zoomForWidth(leftTargetWidth); } double KMultiPage::calculateFitToHeightZoomValue() { PageNumber pageNumber = 1; // See below, in the documentation of the method "calculatefitToWidthZoomLevel" // for an explanation of the complicated calculation we are doing here. int columns = scrollView()->getNrColumns(); int rows = scrollView()->getNrRows(); int continuousViewmode = scrollView()->isContinuous(); bool fullScreenMode = scrollView()->fullScreenMode(); if (columns == 1 && rows == 1 && !continuousViewmode) // single page mode { pageNumber = currentPageNumber(); if (!pageNumber.isValid()) pageNumber = 1; } int pageDistance = scrollView()->distanceBetweenPages(); if (columns == 1 && rows == 1 && !continuousViewmode && fullScreenMode) { // In Single Page Fullscreen Mode we want to fit the page to the // window without a margin around it. pageDistance = 0; } int targetViewportHeight = scrollView()->viewportSize(0,0).height(); int targetPageHeight = (targetViewportHeight - rows*pageDistance)/rows; int targetPageWidth = (int)(targetPageHeight * pageCache->sizeOfPage(pageNumber).aspectRatio() ); int targetViewportWidth = targetPageWidth * columns + (columns+1)*pageDistance; targetViewportHeight = scrollView()->viewportSize(targetViewportWidth, targetViewportHeight).height(); targetPageHeight = (targetViewportHeight - rows*pageDistance)/rows; return pageCache->sizeOfPage(pageNumber).zoomForHeight(targetPageHeight); } double KMultiPage::calculateFitToWidthZoomValue() { PageNumber pageNumber = 1; int columns = scrollView()->getNrColumns(); int rows = scrollView()->getNrRows(); int continuousViewmode = scrollView()->isContinuous(); bool fullScreenMode = scrollView()->fullScreenMode(); if (columns == 1 && rows == 1 && !continuousViewmode) // single page mode { // To calculate the zoom level in single page mode we need the size // of the current page. When a new document is opened this function // is called while the currentPageNumber is invalid. We use the size // of the first page of the document in this case. pageNumber = currentPageNumber(); if (!pageNumber.isValid()) pageNumber = 1; } if (columns == 1 && rows == 1 && continuousViewmode) // continuous viewmode { pageNumber = widestPage(); if (!pageNumber.isValid()) pageNumber = 1; } // rows should be 1 for Single Page Viewmode, // the number of Pages in Continuous Viewmode // and number of Pages/2 in Continuous-Facing Viewmode if (continuousViewmode) rows = (int)(ceil(numberOfPages() / (double)columns)); int pageDistance = scrollView()->distanceBetweenPages(); if (columns == 1 && rows == 1 && !continuousViewmode && fullScreenMode) { // In Single Page Fullscreen Mode we want to fit the page to the // window without a margin around it. pageDistance = 0; } // There is a slight complication here... if we just take the width // of the viewport and scale the contents by a factor x so that it // fits the viewport exactly, then, depending on chosen papersize // (landscape, etc.), the contents may be higher than the viewport // and the TQScrollview may or may not insert a scrollbar at the // right. If the scrollbar appears, then the usable width of the // viewport becomes smaller, and scaling by x does not really fit // the (now smaller page) anymore. // Calculate the width and height of the view, disregarding the // possible complications with scrollbars, e.g. assuming the maximal // space is available. // width of the widget excluding possible scrollbars int targetViewportWidth = scrollView()->viewportSize(0,0).width(); // maximal width of a single page int targetPageWidth = (targetViewportWidth - (columns+1) * pageDistance) / columns; // maximal height of a single page int targetPageHeight = (int)(targetPageWidth/pageCache->sizeOfPage(pageNumber).aspectRatio()); // FIXME: this is only correct if all pages in the document have the same height int targetViewportHeight = rows * targetPageHeight + (rows+1) * pageDistance; // Think again, this time use only the area which is really // acessible (which, in case that targetWidth targetHeight don't fit // the viewport, is really smaller because of the scrollbars). targetViewportWidth = scrollView()->viewportSize(targetViewportWidth, targetViewportHeight).width(); if (columns == 2 && continuousViewmode) // continuous facing { // TODO Generalize this for more than 2 columns return zoomForWidthColumns(targetViewportWidth - (columns+1) * pageDistance); } // maximal width of a single page (now the scrollbars are taken into account) targetPageWidth = (targetViewportWidth - (columns+1) * pageDistance) / columns; return pageCache->sizeOfPage(pageNumber).zoomForWidth(targetPageWidth); } void KMultiPage::prevPage() { TQ_UINT8 cols = scrollView()->getNrColumns(); TQ_UINT8 rows = scrollView()->getNrRows(); PageNumber np = 1; if (cols*rows < currentPageNumber()) { np = currentPageNumber() - cols*rows; } gotoPage(np); } void KMultiPage::nextPage() { TQ_UINT8 cols = scrollView()->getNrColumns(); TQ_UINT8 rows = scrollView()->getNrRows(); PageNumber np = TQMIN(currentPageNumber() + cols*rows, (TQ_UINT16)numberOfPages()); gotoPage(np); } void KMultiPage::firstPage() { gotoPage(1); } void KMultiPage::lastPage() { gotoPage(numberOfPages()); } void KMultiPage::scroll(TQ_INT32 deltaInPixel) { TQScrollBar* scrollBar = scrollView()->verticalScrollBar(); if (scrollBar == 0) { kdError(4300) << "KMultiPage::scroll called without scrollBar" << endl; return; } if (deltaInPixel < 0) { if (scrollBar->value() == scrollBar->minValue()) { if ( (currentPageNumber() == 1) || (changePageDelayTimer.isActive()) ) return; if (scrollView()->isContinuous()) return; changePageDelayTimer.stop(); prevPage(); scrollView()->setContentsPos(scrollView()->contentsX(), scrollBar->maxValue()); return; } } if (deltaInPixel > 0) { if (scrollBar->value() == scrollBar->maxValue()) { if ( (currentPageNumber() == numberOfPages()) || (changePageDelayTimer.isActive()) ) return; if (scrollView()->isContinuous()) return; changePageDelayTimer.stop(); nextPage(); scrollView()->setContentsPos(scrollView()->contentsX(), 0); return; } } scrollBar->setValue(scrollBar->value() + deltaInPixel); if ( (scrollBar->value() == scrollBar->maxValue()) || (scrollBar->value() == scrollBar->minValue()) ) changePageDelayTimer.start(200,true); else changePageDelayTimer.stop(); } void KMultiPage::scrollUp() { TQScrollBar* scrollBar = scrollView()->verticalScrollBar(); if (scrollBar == 0) return; scroll(-scrollBar->lineStep()); } void KMultiPage::scrollDown() { TQScrollBar* scrollBar = scrollView()->verticalScrollBar(); if (scrollBar == 0) return; scroll(scrollBar->lineStep()); } void KMultiPage::scrollLeft() { TQScrollBar* scrollBar = scrollView()->horizontalScrollBar(); if (scrollBar) scrollBar->subtractLine(); } void KMultiPage::scrollRight() { TQScrollBar* scrollBar = scrollView()->horizontalScrollBar(); if (scrollBar) scrollBar->addLine(); } void KMultiPage::scrollUpPage() { TQScrollBar* scrollBar = scrollView()->verticalScrollBar(); if (scrollBar) scrollBar->subtractPage(); } void KMultiPage::scrollDownPage() { TQScrollBar* scrollBar = scrollView()->verticalScrollBar(); if (scrollBar) scrollBar->addPage(); } void KMultiPage::scrollLeftPage() { TQScrollBar* scrollBar = scrollView()->horizontalScrollBar(); if (scrollBar) scrollBar->subtractPage(); } void KMultiPage::scrollRightPage() { TQScrollBar* scrollBar = scrollView()->horizontalScrollBar(); if (scrollBar) scrollBar->addPage(); } void KMultiPage::readDown() { PageView* sv = scrollView(); if (sv->atBottom()) { if (sv->isContinuous()) return; if (currentPageNumber() == numberOfPages()) return; nextPage(); sv->setContentsPos(sv->contentsX(), 0); } else sv->readDown(); } void KMultiPage::readUp() { PageView* sv = scrollView(); if (sv->atTop()) { if (sv->isContinuous()) return; if (currentPageNumber() == 1) return; prevPage(); sv->setContentsPos(sv->contentsX(), sv->contentsHeight()); } else sv->readUp(); } void KMultiPage::jumpToReference(const TQString& reference) { if (renderer.isNull()) return; gotoPage(renderer->parseReference(reference)); } void KMultiPage::gotoPage(const Anchor &a) { if (!a.page.isValid() || (renderer.isNull())) return; gotoPage(a.page, (int)(a.distance_from_top.getLength_in_inch()*pageCache->getResolution() + 0.5), true); } void KMultiPage::gotoPage(const TextSelection& selection) { if (selection.isEmpty()) { kdError(4300) << "KMultiPage::gotoPage(...) called with empty TextSelection." << endl; return; } RenderedDocumentPage* pageData = pageCache->getPage(selection.getPageNumber()); if (pageData == 0) { #ifdef DEBUG_DOCUMENTWIDGET kdDebug(4300) << "DocumentWidget::paintEvent: no documentPage generated" << endl; #endif return; } switch (widgetList.size()) { case 0: kdError(4300) << "KMultiPage::select() while widgetList is empty" << endl; break; case 1: ((DocumentWidget*)widgetList[0])->select(selection); break; default: if (widgetList.size() < currentPageNumber()) kdError(4300) << "KMultiPage::select() while widgetList.size()=" << widgetList.size() << "and currentPageNumber()=" << currentPageNumber() << endl; else ((DocumentWidget*)widgetList[selection.getPageNumber() - 1])->select(selection); } unsigned int y = pageData->textBoxList[selection.getSelectedTextStart()].box.top(); gotoPage(selection.getPageNumber(), y, false); } void KMultiPage::doSelectAll() { switch( widgetList.size() ) { case 0: kdError(4300) << "KMultiPage::doSelectAll() while widgetList is empty" << endl; break; case 1: ((DocumentWidget *)widgetList[0])->selectAll(); break; default: if (widgetList.size() < currentPageNumber()) kdError(4300) << "KMultiPage::doSelectAll() while widgetList.size()=" << widgetList.size() << "and currentPageNumber()=" << currentPageNumber() << endl; else ((DocumentWidget *)widgetList[currentPageNumber()-1])->selectAll(); } } void KMultiPage::showFindTextDialog() { if ((renderer.isNull()) || (renderer->supportsTextSearch() == false)) return; searchWidget->show(); searchWidget->setFocus(); } void KMultiPage::stopSearch() { if (searchInProgress) { // stop the search searchInProgress = false; } else searchWidget->hide(); } void KMultiPage::findNextText() { #ifdef KDVI_MULTIPAGE_DEBUG kdDebug(4300) << "KMultiPage::findNextText() called" << endl; #endif searchInProgress = true; // Used to remember if the documentPage we use is from the cache. // If not we need to delete it manually to avoid a memory leak. bool cachedPage = false; TQString searchText = searchWidget->getText(); if (searchText.isEmpty()) { kdError(4300) << "KMultiPage::findNextText() called when search text was empty" << endl; return; } bool case_sensitive = searchWidget->caseSensitive(); // Find the page and text position on the page where the search will // start. If nothing is selected, we start at the beginning of the // current page. Otherwise, start after the selected text. TODO: // Optimize this to get a better 'user feeling' TQ_UINT16 startingPage; TQ_UINT16 startingTextItem; TextSelection userSelection = pageCache->selectedText(); if (userSelection.isEmpty()) { startingPage = currentPageNumber(); startingTextItem = 0; } else { startingPage = userSelection.getPageNumber(); startingTextItem = userSelection.getSelectedTextEnd()+1; } TextSelection foundSelection; RenderedDocumentPagePixmap* searchPage = 0; for(unsigned int i = 0; i < numberOfPages(); i++) { unsigned int pageNumber = (i + startingPage - 1) % numberOfPages() + 1; if (!searchInProgress) { // Interrupt the search setStatusBarText(i18n("Search interrupted")); if (!cachedPage) delete searchPage; return; } if (i != 0) { setStatusBarText(i18n("Search page %1 of %2").arg(pageNumber).arg(numberOfPages())); kapp->processEvents(); } // Check if we already have a rendered version of the page in the cache. As we are only interested in the // text we don't care about the page size. if (pageCache->isPageCached(pageNumber)) { // If the last search page used was created locally, we need to destroy it if (!cachedPage) delete searchPage; searchPage = pageCache->getPage(pageNumber); cachedPage = true; } else { // If the page is not in the cache we draw a small version of it, since this is faster. // We only create a new searchPage if we need to, otherwise reuse the existing one. if (!searchPage || cachedPage) searchPage = new RenderedDocumentPagePixmap(); cachedPage = false; searchPage->resize(1,1); searchPage->setPageNumber(pageNumber); renderer->getText(searchPage); } // If there is no text in the current page, try the next one. if (searchPage->textBoxList.size() == 0) continue; foundSelection = searchPage->find(searchText, startingTextItem, case_sensitive); if (foundSelection.isEmpty()) { // In the next page, start search again at the beginning. startingTextItem = 0; clearSelection(); if (pageNumber == numberOfPages()) { int answ = KMessageBox::questionYesNo(scrollView(), i18n("<qt>The search string <strong>%1</strong> could not be found by the " "end of the document. Should the search be restarted from the beginning " "of the document?</qt>").arg(searchText), i18n("Text Not Found"), KStdGuiItem::cont(), KStdGuiItem::cancel()); if (answ != KMessageBox::Yes) { setStatusBarText(TQString()); searchInProgress = false; if (!cachedPage) delete searchPage; return; } } } else { pageCache->selectText(foundSelection); gotoPage(pageCache->selectedText()); setStatusBarText(TQString()); searchInProgress = false; if (!cachedPage) delete searchPage; return; } } KMessageBox::sorry(scrollView(), i18n("<qt>The search string <strong>%1</strong> could not be found.</qt>").arg(searchText)); setStatusBarText(TQString()); searchInProgress = false; if (!cachedPage) delete searchPage; } void KMultiPage::findPrevText() { #ifdef KDVI_MULTIPAGE_DEBUG kdDebug(4300) << "KMultiPage::findPrevText() called" << endl; #endif searchInProgress = true; // Used to remember if the documentPage we use is from the cache. // If not we need to delete it manually to avoid a memory leak. bool cachedPage = false; TQString searchText = searchWidget->getText(); if (searchText.isEmpty()) { kdError(4300) << "KMultiPage::findPrevText() called when search text was empty" << endl; return; } bool case_sensitive = searchWidget->caseSensitive(); // Find the page and text position on the page where the search will // start. If nothing is selected, we start at the beginning of the // current page. Otherwise, start after the selected text. TODO: // Optimize this to get a better 'user feeling' unsigned int startingPage; int startingTextItem; TextSelection userSelection = pageCache->selectedText(); if (userSelection.isEmpty()) { startingPage = currentPageNumber(); startingTextItem = -1; } else { startingPage = userSelection.getPageNumber(); startingTextItem = userSelection.getSelectedTextStart()-1; } TextSelection foundSelection; RenderedDocumentPagePixmap* searchPage = 0; for(unsigned int i = 0; i < numberOfPages(); i++) { int pageNumber = startingPage - i; if (pageNumber <= 0) pageNumber += numberOfPages(); if (!searchInProgress) { // Interrupt the search setStatusBarText(i18n("Search interrupted")); if (!cachedPage) delete searchPage; return; } if (i != 0) { setStatusBarText(i18n("Search page %1 of %2").arg(pageNumber).arg(numberOfPages())); kapp->processEvents(); } // Check if we already have a rendered version of the page in the cache. As we are only interested in the // text we don't care about the page size. if (pageCache->isPageCached(pageNumber)) { // If the last search page used was created locally, we need to destroy it if (!cachedPage) delete searchPage; searchPage = pageCache->getPage(pageNumber); cachedPage = true; } else { // If the page is not in the cache we draw a small version of it, since this is faster. // We only create a new searchPage if we need to, otherwise reuse the existing one. if (!searchPage || cachedPage) searchPage = new RenderedDocumentPagePixmap(); cachedPage = false; searchPage->resize(1,1); searchPage->setPageNumber(pageNumber); renderer->getText(searchPage); } // If there is no text in the current page, try the next one. if (searchPage->textBoxList.size() == 0) continue; foundSelection = searchPage->findRev(searchText, startingTextItem, case_sensitive); if (foundSelection.isEmpty()) { // In the next page, start search again at the beginning. startingTextItem = -1; clearSelection(); if (pageNumber == 1) { int answ = KMessageBox::questionYesNo(scrollView(), i18n("<qt>The search string <strong>%1</strong> could not be found by the " "beginning of the document. Should the search be restarted from the end " "of the document?</qt>").arg(searchText), i18n("Text Not Found"), KStdGuiItem::cont(), KStdGuiItem::cancel()); if (answ != KMessageBox::Yes) { setStatusBarText(TQString()); searchInProgress = false; if (!cachedPage) delete searchPage; return; } } } else { pageCache->selectText(foundSelection); gotoPage(pageCache->selectedText()); setStatusBarText(TQString()); searchInProgress = false; if (!cachedPage) delete searchPage; return; } } KMessageBox::sorry(scrollView(), i18n("<qt>The search string <strong>%1</strong> could not be found.</qt>").arg(searchText)); setStatusBarText(TQString()); searchInProgress = false; if (!cachedPage) delete searchPage; } void KMultiPage::clearSelection() { PageNumber page = pageCache->selectedText().getPageNumber(); if (!page.isValid()) return; // Clear selection pageCache->deselectText(); // Now we need to update the widget which contained the selection switch(widgetList.size()) { case 0: kdError(4300) << "KMultiPage::clearSelection() while widgetList is empty" << endl; break; case 1: widgetList[0]->update(); break; default: for (unsigned int i = 0; i < widgetList.size(); i++) { DocumentWidget* pageWidget = (DocumentWidget*)widgetList[i]; if (pageWidget->getPageNumber() == page) { pageWidget->update(); break; } } } } void KMultiPage::copyText() { pageCache->selectedText().copyText(); } void KMultiPage::timerEvent( TQTimerEvent * ) { #ifdef KMULTIPAGE_DEBUG kdDebug(4300) << "Timer Event " << endl; #endif reload(); } void KMultiPage::reload() { #ifdef KMULTIPAGE_DEBUG kdDebug(4300) << "Reload file " << m_file << endl; #endif if (renderer.isNull()) { kdError() << "KMultiPage::reload() called, but no renderer was set" << endl; return; } if (renderer->isValidFile(m_file)) { pageCache->clear(); pageCache->deselectText(); document_history.clear(); emit setStatusBarText(i18n("Reloading file %1").arg(m_file)); TQ_INT32 pg = currentPageNumber(); killTimer(timer_id); timer_id = -1; bool r = renderer->setFile(m_file, m_url); generateDocumentWidgets(); // Set Table of Contents tableOfContents->setContents(renderer->getBookmarks()); // Adjust number of widgets in the thumbnail sidebar markList()->clear(); markList()->setNumberOfPages(numberOfPages(), KVSPrefs::showThumbnails()); setCurrentPageNumber(pg); setFile(r); emit setStatusBarText(TQString()); } else { if (timer_id == -1) timer_id = startTimer(1000); } } bool KMultiPage::openFile() { if (renderer.isNull()) { kdError(4300) << "KMultiPage::openFile() called when no renderer was set" << endl; return false; } pageCache->deselectText(); document_history.clear(); pageCache->clear(); emit setStatusBarText(i18n("Loading file %1").arg(m_file)); bool r = renderer->setFile(m_file, m_url); if (r) { setCurrentPageNumber(1); generateDocumentWidgets(); // Set number of widgets in the thumbnail sidebar markList()->clear(); markList()->setNumberOfPages(numberOfPages(), KVSPrefs::showThumbnails()); TQString reference = url().ref(); if (!reference.isEmpty()) gotoPage(renderer->parseReference(reference)); // Set Table of Contents tableOfContents->setContents(renderer->getBookmarks()); } else m_file = TQString(); setFile(r); // Clear Statusbar emit setStatusBarText(TQString()); return r; } bool KMultiPage::openURL(const TQString &filename, const KURL &base_url) { m_file = filename; m_url = base_url; bool success = openFile(); if (success) setCurrentPageNumber(1); return success; } void KMultiPage::enableActions(bool fileLoaded) { Q_UNUSED(fileLoaded); } void KMultiPage::wheelEvent(TQWheelEvent *e) { TQScrollBar *sb = scrollView()->verticalScrollBar(); if (sb == 0) return; // Zoom in/out if (e->state() & ControlButton) { if (e->delta() < 0) emit zoomOut(); else emit zoomIn(); return; } TQ_INT32 pxl = -(e->delta()*sb->lineStep())/60; if (pxl == 0) { if (e->delta() > 0) pxl = -1; else pxl = 1; } // Faster scrolling if (e->state() & ShiftButton) pxl *= 10; scroll(pxl); } KPrinter *KMultiPage::getPrinter(bool enablePageSizeFeatures) { // Allocate a new KPrinter structure, if necessary KPrinter *printer = new KPrinter(true); if (printer == 0) { kdError(1223) << "KMultiPage::getPrinter(..): Cannot allocate new KPrinter structure" << endl; return 0; } // Allocate a new KPrintDialogPage structure and add it to the // printer, if the tdemultipage implementation requests that if (enablePageSizeFeatures == true) { KPrintDialogPage_PageOptions *pageOptions = new KPrintDialogPage_PageOptions(); if (pageOptions == 0) { kdError(1223) << "KMultiPage::getPrinter(..): Cannot allocate new KPrintDialogPage_PageOptions structure" << endl; delete printer; return 0; } printer->addDialogPage( pageOptions ); } // Feed the printer with useful defaults and information. printer->setPageSelection( KPrinter::ApplicationSide ); printer->setCurrentPage( currentPageNumber() ); printer->setMinMax( 1, numberOfPages() ); printer->setFullPage( true ); // If pages are marked, give a list of marked pages to the // printer. We try to be smart and optimize the list by using ranges // ("5-11") wherever possible. The user will be tankful for // that. Complicated? Yeah, but that's life. TQValueList<int> selectedPageNo = selectedPages(); if (selectedPageNo.isEmpty() == true) printer->setOption( "kde-range", "" ); else { int commaflag = 0; TQString range; TQValueList<int>::ConstIterator it = selectedPageNo.begin(); do{ int val = *it; if (commaflag == 1) range += TQString(", "); else commaflag = 1; int endval = val; if (it != selectedPageNo.end()) { TQValueList<int>::ConstIterator jt = it; jt++; do{ int val2 = *jt; if (val2 == endval+1) endval++; else break; jt++; } while( jt != selectedPageNo.end() ); it = jt; } else it++; if (endval == val) range += TQString("%1").arg(val); else range += TQString("%1-%2").arg(val).arg(endval); } while (it != selectedPageNo.end() ); printer->setOption( "kde-range", range ); } return printer; } void KMultiPage::doExportText() { // Generate a suggestion for a reasonable file name TQString suggestedName = url().filename(); suggestedName = suggestedName.left(suggestedName.find(".")) + ".txt"; TQString fileName = KFileDialog::getSaveFileName(suggestedName, i18n("*.txt|Plain Text (Latin 1) (*.txt)"), scrollView(), i18n("Export File As")); if (fileName.isEmpty()) return; TQFileInfo finfo(fileName); if (finfo.exists()) { int r = KMessageBox::warningContinueCancel (scrollView(), i18n("The file %1\nexists. Do you want to overwrite that file?").arg(fileName), i18n("Overwrite File"), i18n("Overwrite")); if (r == KMessageBox::Cancel) return; } TQFile textFile(fileName); textFile.open(IO_WriteOnly); TQTextStream stream(&textFile); TQProgressDialog progress(i18n("Exporting to text..."), i18n("Abort"), renderer->totalPages(), scrollView(), "export_text_progress", true); progress.setMinimumDuration(300); RenderedDocumentPagePixmap dummyPage; dummyPage.resize(1, 1); for(unsigned int page = 1; page <= renderer->totalPages(); page++) { progress.setProgress(page); tqApp->processEvents(); if (progress.wasCancelled()) break; dummyPage.setPageNumber(page); // We gracefully ignore any errors (bad file, etc.) renderer->getText(&dummyPage); for(unsigned int i = 0; i < dummyPage.textBoxList.size(); i++) { // We try to detect newlines if (i > 0) { // Like all our textalgorithmns this currently assumes left to right text. // TODO: make this more generic. But we first would need to guess the corrent // orientation. if (dummyPage.textBoxList[i].box.top() > dummyPage.textBoxList[i-1].box.bottom() && dummyPage.textBoxList[i].box.x() < dummyPage.textBoxList[i-1].box.x()) { stream << "\n"; } } stream << dummyPage.textBoxList[i].text; } // Send newline after each page. stream << "\n"; } // Switch off the progress dialog, etc. progress.setProgress(renderer->totalPages()); return; } void KMultiPage::slotEnableMoveTool(bool enable) { emit enableMoveTool(enable); } #include "tdemultipage.moc"