/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed * with this work for additional information regarding copyright * ownership. The ASF licenses this file to you under the Apache * License, Version 2.0 (the "License"); you may not use this file * except in compliance with the License. You may obtain a copy of * the License at http://www.apache.org/licenses/LICENSE-2.0 . */ #include #include "DocumentRenderer.hxx" #include "DocumentRenderer.hrc" #include "drawdoc.hxx" #include "optsitem.hxx" #include "sdresid.hxx" #include "strings.hrc" #include "sdattr.hxx" #include "Window.hxx" #include "drawview.hxx" #include "DrawViewShell.hxx" #include "FrameView.hxx" #include "Outliner.hxx" #include "OutlineViewShell.hxx" #include "SlideSorterViewShell.hxx" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace ::com::sun::star; using namespace ::com::sun::star::uno; namespace sd { namespace { /** Convenience class to extract values from the sequence of properties given to one of the XRenderable methods. */ class PrintOptions { public: PrintOptions ( const vcl::PrinterOptionsHelper& rHelper, const std::vector& rSlidesPerPage) : mrProperties(rHelper), maSlidesPerPage(rSlidesPerPage) { } bool IsWarningOrientation() const { return GetBoolValue(nullptr, true); } bool IsPrintPageName() const { return GetBoolValue("IsPrintName", false); } bool IsDate() const { return GetBoolValue("IsPrintDateTime", false); } bool IsTime() const { return GetBoolValue("IsPrintDateTime", false); } bool IsHiddenPages() const { return GetBoolValue("IsPrintHidden", false); } bool IsHandoutHorizontal() const { return GetBoolValue("SlidesPerPageOrder", sal_Int32(0)); } sal_Int32 GetHandoutPageCount() const { sal_uInt32 nIndex = static_cast(mrProperties.getIntValue("SlidesPerPage", sal_Int32(0))); if (nIndex(mrProperties.getIntValue( "Quality", sal_Int32(0) )); return nQuality; } bool IsPageSize() const { return GetBoolValue("PageOptions", sal_Int32(1)); } bool IsTilePage() const { return GetBoolValue("PageOptions", sal_Int32(2)) || GetBoolValue("PageOptions", sal_Int32(3)); } bool IsCutPage() const { return GetBoolValue("PageOptions", sal_Int32(0)); } bool IsBooklet() const { return GetBoolValue("PrintProspect", false); } bool IsPrinterPreferred(DocumentType eDocType) const { bool bIsDraw = eDocType == DOCUMENT_TYPE_DRAW; return IsTilePage() || IsPageSize() || IsBooklet() || (!bIsDraw && !IsNotes()); } bool IsPrintExcluded() const { return (IsNotes() || IsDraw() || IsHandout()) && IsHiddenPages(); } bool IsPrintFrontPage() const { sal_Int32 nInclude = static_cast(mrProperties.getIntValue( "PrintProspectInclude", 0 )); return nInclude == 0 || nInclude == 1; } bool IsPrintBackPage() const { sal_Int32 nInclude = static_cast(mrProperties.getIntValue( "PrintProspectInclude", 0 )); return nInclude == 0 || nInclude == 2; } bool IsPaperBin() const { return GetBoolValue("PrintPaperFromSetup", false); } bool IsPrintMarkedOnly() const { return GetBoolValue("PrintContent", sal_Int32(2)); } OUString GetPrinterSelection (sal_Int32 nPageCount, sal_Int32 nCurrentPageIndex) const { sal_Int32 nContent = static_cast(mrProperties.getIntValue( "PrintContent", 0 )); OUString sFullRange = "1-" + OUString::number(nPageCount); if (nContent == 0) // all pages/slides { return sFullRange; } if (nContent == 1) // range { OUString sValue = mrProperties.getStringValue("PageRange"); return sValue.isEmpty() ? sFullRange : sValue; } if (nContent == 2 && // selection nCurrentPageIndex >= 0) { return OUString::number(nCurrentPageIndex + 1); } return OUString(); } private: const vcl::PrinterOptionsHelper& mrProperties; const std::vector maSlidesPerPage; /** When the value of the property with name pName is a boolean then return its value. When the property is unknown then bDefaultValue is returned. Otherwise is returned. */ bool GetBoolValue ( const sal_Char* pName, const bool bDefaultValue) const { bool bValue = mrProperties.getBoolValue( pName, bDefaultValue ); return bValue; } /** Return when the value of the property with name pName is an integer and its value is nTriggerValue. Otherwise is returned. */ bool GetBoolValue ( const sal_Char* pName, const sal_Int32 nTriggerValue) const { sal_Int32 nValue = static_cast(mrProperties.getIntValue( pName, 0 )); return nValue == nTriggerValue; } }; /** A collection of values that helps to reduce the number of arguments given to some functions. Note that not all values are set at the same time. */ class PrintInfo { public: PrintInfo ( Printer* pPrinter, const bool bPrintMarkedOnly) : mpPrinter(pPrinter), mnDrawMode(DrawModeFlags::Default), msTimeDate(), msPageString(), maPrintSize(0,0), maPageSize(0,0), meOrientation(ORIENTATION_PORTRAIT), maMap(), mbPrintMarkedOnly(bPrintMarkedOnly) {} const VclPtr mpPrinter; DrawModeFlags mnDrawMode; OUString msTimeDate; OUString msPageString; Size maPrintSize; Size maPageSize; Orientation meOrientation; MapMode maMap; const bool mbPrintMarkedOnly; }; /** Output one page of the document to the given printer. Note that more than one document page may be output to one printer page. */ void PrintPage ( Printer& rPrinter, ::sd::View& rPrintView, SdPage& rPage, View* pView, const bool bPrintMarkedOnly, const SetOfByte& rVisibleLayers, const SetOfByte& rPrintableLayers) { rPrintView.ShowSdrPage(&rPage); const MapMode aOriginalMapMode (rPrinter.GetMapMode()); // Set the visible layers SdrPageView* pPageView = rPrintView.GetSdrPageView(); OSL_ASSERT(pPageView!=nullptr); pPageView->SetVisibleLayers(rVisibleLayers); pPageView->SetPrintableLayers(rPrintableLayers); if (pView!=nullptr && bPrintMarkedOnly) pView->DrawMarkedObj(rPrinter); else rPrintView.CompleteRedraw(&rPrinter, vcl::Region(Rectangle(Point(0,0), rPage.GetSize()))); rPrinter.SetMapMode(aOriginalMapMode); rPrintView.HideSdrPage(); } /** Output a string (that typically is not part of a document page) to the given printer. */ void PrintMessage ( Printer& rPrinter, const OUString& rsPageString, const Point& rPageStringOffset) { const vcl::Font aOriginalFont (rPrinter.OutputDevice::GetFont()); rPrinter.SetFont(vcl::Font(FAMILY_SWISS, Size(0, 423))); rPrinter.DrawText(rPageStringOffset, rsPageString); rPrinter.SetFont(aOriginalFont); } /** Read the resource file and process it into a sequence of properties that can be passed to the printing dialog. */ class DialogCreator : public Resource { public: DialogCreator (ViewShellBase &rBase, bool bImpress, sal_Int32 nCurPage) : Resource(SdResId(STR_IMPRESS_PRINT_UI_OPTIONS)) , mrBase(rBase) , mbImpress(bImpress) , mnCurPage(nCurPage) { ProcessResource(); } const std::vector< beans::PropertyValue >& GetDialogControls() const { return maProperties; } const std::vector& GetSlidesPerPage() const { return maSlidesPerPage; } private: ViewShellBase &mrBase; std::vector maProperties; std::vector maSlidesPerPage; bool mbImpress; sal_Int32 mnCurPage; void ProcessResource() { // load the writer PrinterOptions into the custom tab beans::PropertyValue aOptionsUIFile; aOptionsUIFile.Name = "OptionsUIFile"; if( mbImpress ) aOptionsUIFile.Value <<= OUString("modules/simpress/ui/printeroptions.ui"); else aOptionsUIFile.Value <<= OUString("modules/sdraw/ui/printeroptions.ui"); maProperties.push_back(aOptionsUIFile); SvtModuleOptions aOpt; OUString aAppGroupname(SD_RESSTR(STR_IMPRESS_PRINT_UI_GROUP_NAME)); aAppGroupname = aAppGroupname.replaceFirst("%s", aOpt.GetModuleName( mbImpress ? SvtModuleOptions::EModule::IMPRESS : SvtModuleOptions::EModule::DRAW)); AddDialogControl(vcl::PrinterOptionsHelper::setGroupControlOpt("tabcontrol-page2", aAppGroupname, ".HelpID:vcl:PrintDialog:TabPage:AppPage")); uno::Sequence< OUString > aHelpIds, aWidgetIds; if( mbImpress ) { vcl::PrinterOptionsHelper::UIControlOptions aPrintOpt; aPrintOpt.maGroupHint = "JobPage" ; AddDialogControl( vcl::PrinterOptionsHelper::setSubgroupControlOpt("extraimpressprintoptions", SD_RESSTR(STR_IMPRESS_PRINT_UI_PRINT_GROUP), "", aPrintOpt )); aHelpIds.realloc( 1 ); aHelpIds[0] = ".HelpID:vcl:PrintDialog:PageContentType:ListBox" ; AddDialogControl( vcl::PrinterOptionsHelper::setChoiceListControlOpt( "impressdocument", SD_RESSTR(STR_IMPRESS_PRINT_UI_CONTENT), aHelpIds, "PageContentType" , CreateChoice(STR_IMPRESS_PRINT_UI_CONTENT_CHOICES), 0) ); aHelpIds[0] = ".HelpID:vcl:PrintDialog:SlidesPerPage:ListBox" ; vcl::PrinterOptionsHelper::UIControlOptions aContentOpt( "PageContentType" , 1 ); AddDialogControl( vcl::PrinterOptionsHelper::setChoiceListControlOpt( "slidesperpage", SD_RESSTR(STR_IMPRESS_PRINT_UI_SLIDESPERPAGE), aHelpIds, "SlidesPerPage" , GetSlidesPerPageSequence(), 0, Sequence< sal_Bool >(), aContentOpt ) ); aHelpIds[0] = ".HelpID:vcl:PrintDialog:SlidesPerPageOrder:ListBox" ; vcl::PrinterOptionsHelper::UIControlOptions aSlidesPerPageOpt( "SlidesPerPage" , -1, true ); AddDialogControl( vcl::PrinterOptionsHelper::setChoiceListControlOpt( "slidesperpageorder", SD_RESSTR(STR_IMPRESS_PRINT_UI_ORDER), aHelpIds, "SlidesPerPageOrder" , CreateChoice(STR_IMPRESS_PRINT_UI_ORDER_CHOICES), 0, Sequence< sal_Bool >(), aSlidesPerPageOpt ) ); } AddDialogControl( vcl::PrinterOptionsHelper::setSubgroupControlOpt("contents", SD_RESSTR(STR_IMPRESS_PRINT_UI_INCLUDE_CONTENT), "" ) ); if( mbImpress ) { AddDialogControl( vcl::PrinterOptionsHelper::setBoolControlOpt("printname", SD_RESSTR(STR_IMPRESS_PRINT_UI_IS_PRINT_NAME), ".HelpID:vcl:PrintDialog:IsPrintName:CheckBox" , "IsPrintName" , false ) ); } else { AddDialogControl( vcl::PrinterOptionsHelper::setBoolControlOpt("printname", SD_RESSTR(STR_DRAW_PRINT_UI_IS_PRINT_NAME), ".HelpID:vcl:PrintDialog:IsPrintName:CheckBox" , "IsPrintName" , false ) ); } AddDialogControl( vcl::PrinterOptionsHelper::setBoolControlOpt("printdatetime", SD_RESSTR(STR_IMPRESS_PRINT_UI_IS_PRINT_DATE), ".HelpID:vcl:PrintDialog:IsPrintDateTime:CheckBox" , "IsPrintDateTime" , false ) ); if( mbImpress ) { AddDialogControl( vcl::PrinterOptionsHelper::setBoolControlOpt("printhidden", SD_RESSTR(STR_IMPRESS_PRINT_UI_IS_PRINT_HIDDEN), ".HelpID:vcl:PrintDialog:IsPrintHidden:CheckBox" , "IsPrintHidden" , false ) ); } AddDialogControl( vcl::PrinterOptionsHelper::setSubgroupControlOpt("color", SD_RESSTR(STR_IMPRESS_PRINT_UI_QUALITY), "" ) ); aHelpIds.realloc( 3 ); aHelpIds[0] = ".HelpID:vcl:PrintDialog:Quality:RadioButton:0" ; aHelpIds[1] = ".HelpID:vcl:PrintDialog:Quality:RadioButton:1" ; aHelpIds[2] = ".HelpID:vcl:PrintDialog:Quality:RadioButton:2" ; aWidgetIds.realloc( 3 ); aWidgetIds[0] = "originalcolors"; aWidgetIds[1] = "grayscale"; aWidgetIds[2] = "blackandwhite"; AddDialogControl( vcl::PrinterOptionsHelper::setChoiceRadiosControlOpt( aWidgetIds, "", aHelpIds, "Quality" , CreateChoice(STR_IMPRESS_PRINT_UI_QUALITY_CHOICES), 0) ); AddDialogControl( vcl::PrinterOptionsHelper::setSubgroupControlOpt("pagesizes", SD_RESSTR(STR_IMPRESS_PRINT_UI_PAGE_OPTIONS), "" ) ); aHelpIds.realloc( 4 ); aHelpIds[0] = ".HelpID:vcl:PrintDialog:PageOptions:RadioButton:0" ; aHelpIds[1] = ".HelpID:vcl:PrintDialog:PageOptions:RadioButton:1" ; aHelpIds[2] = ".HelpID:vcl:PrintDialog:PageOptions:RadioButton:2" ; aHelpIds[3] = ".HelpID:vcl:PrintDialog:PageOptions:RadioButton:3" ; aWidgetIds.realloc( 4 ); aWidgetIds[0] = "originalsize"; aWidgetIds[1] = "fittoprintable"; aWidgetIds[2] = "distributeonmultiple"; aWidgetIds[3] = "tilesheet"; vcl::PrinterOptionsHelper::UIControlOptions aPageOptionsOpt("PrintProspect", 0); AddDialogControl( vcl::PrinterOptionsHelper::setChoiceRadiosControlOpt( aWidgetIds, "", aHelpIds, "PageOptions" , CreateChoice(mbImpress ? STR_IMPRESS_PRINT_UI_PAGE_OPTIONS_CHOICES : STR_IMPRESS_PRINT_UI_PAGE_OPTIONS_CHOICES_DRAW), 0, Sequence< sal_Bool >(), aPageOptionsOpt ) ); vcl::PrinterOptionsHelper::UIControlOptions aBrochureOpt; aBrochureOpt.maGroupHint = "LayoutPage" ; AddDialogControl( vcl::PrinterOptionsHelper::setSubgroupControlOpt("pagesides", SD_RESSTR(STR_IMPRESS_PRINT_UI_PAGE_SIDES), "", aBrochureOpt ) ); // brochure printing AddDialogControl( vcl::PrinterOptionsHelper::setBoolControlOpt("brochure", SD_RESSTR(STR_IMPRESS_PRINT_UI_BROCHURE), ".HelpID:vcl:PrintDialog:PrintProspect:CheckBox" , "PrintProspect" , false, aBrochureOpt ) ); vcl::PrinterOptionsHelper::UIControlOptions aIncludeOpt( "PrintProspect" , -1, false ); aIncludeOpt.maGroupHint = "LayoutPage" ; aHelpIds.realloc( 1 ); aHelpIds[0] = ".HelpID:vcl:PrintDialog:PrintProspectInclude:ListBox" ; AddDialogControl( vcl::PrinterOptionsHelper::setChoiceListControlOpt( "brochureinclude", SD_RESSTR(STR_IMPRESS_PRINT_UI_BROCHURE_INCLUDE), aHelpIds, "PrintProspectInclude" , CreateChoice(STR_IMPRESS_PRINT_UI_BROCHURE_INCLUDE_LIST), 0, Sequence< sal_Bool >(), aIncludeOpt ) ); // paper tray (on options page) vcl::PrinterOptionsHelper::UIControlOptions aPaperTrayOpt; aPaperTrayOpt.maGroupHint = "OptionsPageOptGroup" ; AddDialogControl( vcl::PrinterOptionsHelper::setBoolControlOpt("printpaperfromsetup", SD_RESSTR(STR_IMPRESS_PRINT_UI_PAPER_TRAY), ".HelpID:vcl:PrintDialog:PrintPaperFromSetup:CheckBox" , "PrintPaperFromSetup" , false, aPaperTrayOpt ) ); // print range selection vcl::PrinterOptionsHelper::UIControlOptions aPrintRangeOpt; aPrintRangeOpt.mbInternalOnly = true; aPrintRangeOpt.maGroupHint = "PrintRange" ; AddDialogControl( vcl::PrinterOptionsHelper::setSubgroupControlOpt("printrange", SD_RESSTR(STR_IMPRESS_PRINT_UI_PAGE_RANGE), "", aPrintRangeOpt ) ); // create a choice for the content to create OUString aPrintRangeName( "PrintContent" ); aHelpIds.realloc( 3 ); aHelpIds[0] = ".HelpID:vcl:PrintDialog:PrintContent:RadioButton:0" ; aHelpIds[1] = ".HelpID:vcl:PrintDialog:PrintContent:RadioButton:1" ; aHelpIds[2] = ".HelpID:vcl:PrintDialog:PrintContent:RadioButton:2" ; aWidgetIds.realloc( 3 ); aWidgetIds[0] = "printallpages"; aWidgetIds[1] = "printpages"; aWidgetIds[2] = "printselection"; // check if there is a selection of slides OUString aPageRange(OUString::number(mnCurPage + 1)); int nPrintRange(0); using sd::slidesorter::SlideSorterViewShell; SlideSorterViewShell* const pSSViewSh(SlideSorterViewShell::GetSlideSorter(mrBase)); if (pSSViewSh) { const std::shared_ptr pPageSelection(pSSViewSh->GetPageSelection()); if (bool(pPageSelection) && pPageSelection->size() > 1) { OUStringBuffer aBuf; // TODO: this could be improved by writing ranges instead of consecutive page // numbers if appropriate. Do we have a helper function for that somewhere? bool bFirst(true); for (auto pPage: *pPageSelection) { if (bFirst) bFirst = false; else aBuf.append(','); aBuf.append(OUString::number(pPage->GetPageNum() / 2 + 1)); } aPageRange = aBuf.getStr(); nPrintRange = 1; } } AddDialogControl( vcl::PrinterOptionsHelper::setChoiceRadiosControlOpt(aWidgetIds, "", aHelpIds, aPrintRangeName, CreateChoice(mbImpress ? STR_IMPRESS_PRINT_UI_PAGE_RANGE_CHOICE : STR_DRAW_PRINT_UI_PAGE_RANGE_CHOICE), nPrintRange ) ); // create a an Edit dependent on "Pages" selected vcl::PrinterOptionsHelper::UIControlOptions aPageRangeOpt( aPrintRangeName, 1, true ); AddDialogControl(vcl::PrinterOptionsHelper::setEditControlOpt("pagerange", "", ".HelpID:vcl:PrintDialog:PageRange:Edit", "PageRange", aPageRange, aPageRangeOpt)); FreeResource(); } void AddDialogControl( const Any& i_rCtrl ) { beans::PropertyValue aVal; aVal.Value = i_rCtrl; maProperties.push_back( aVal ); } static Sequence CreateChoice (const sal_uInt16 nResourceId) { SdResId aResourceId (nResourceId); ResStringArray aChoiceStrings (aResourceId); const sal_uInt32 nCount (aChoiceStrings.Count()); Sequence aChoices (nCount); for (sal_uInt32 nIndex=0; nIndex GetSlidesPerPageSequence() { const Sequence aChoice ( CreateChoice(STR_IMPRESS_PRINT_UI_SLIDESPERPAGE_CHOICES)); maSlidesPerPage.clear(); maSlidesPerPage.push_back(0); // first is using the default for (sal_Int32 nIndex=1,nCount=aChoice.getLength(); nIndexGetSize()); const Size aPrintSize (rPrinter.GetOutputSize()); const sal_Int32 nPageWidth (aPageSize.Width() + mnGap - pPageToPrint->GetLftBorder() - pPageToPrint->GetRgtBorder()); const sal_Int32 nPageHeight (aPageSize.Height() + mnGap - pPageToPrint->GetUppBorder() - pPageToPrint->GetLwrBorder()); if (nPageWidth<=0 || nPageHeight<=0) return; // Print at least two rows and columns. More if the document // page fits completely onto the printer page. const sal_Int32 nColumnCount (std::max(sal_Int32(2), sal_Int32(aPrintSize.Width() / nPageWidth))); const sal_Int32 nRowCount (std::max(sal_Int32(2), sal_Int32(aPrintSize.Height() / nPageHeight))); for (sal_Int32 nRow=0; nRow& rPageIndices, const MapMode& rMapMode, const OUString& rsPageString, const Point& rPageStringOffset, const DrawModeFlags nDrawMode, const Orientation eOrientation, const sal_uInt16 nPaperTray) : PrinterPage(PK_HANDOUT, rMapMode, false, rsPageString, rPageStringOffset, nDrawMode, eOrientation, nPaperTray), mnHandoutPageIndex(nHandoutPageIndex), maPageIndices(rPageIndices) { } virtual void Print ( Printer& rPrinter, SdDrawDocument& rDocument, ViewShell& rViewShell, View* pView, DrawView& rPrintView, const SetOfByte& rVisibleLayers, const SetOfByte& rPrintableLayers) const override { SdPage& rHandoutPage (*rDocument.GetSdPage(0, PK_HANDOUT)); Reference< css::beans::XPropertySet > xHandoutPage( rHandoutPage.getUnoPage(), UNO_QUERY ); const OUString sPageNumber( "Number" ); // Collect the page objects of the handout master. std::vector aHandoutPageObjects; SdrObjListIter aShapeIter (rHandoutPage); while (aShapeIter.IsMore()) { SdrPageObj* pPageObj = dynamic_cast(aShapeIter.Next()); if (pPageObj) aHandoutPageObjects.push_back(pPageObj); } if (aHandoutPageObjects.empty()) return; // Connect page objects with pages. std::vector::iterator aPageObjIter (aHandoutPageObjects.begin()); for (std::vector::const_iterator iPageIndex(maPageIndices.begin()), iEnd(maPageIndices.end()); iPageIndex!=iEnd && aPageObjIter!=aHandoutPageObjects.end(); ++iPageIndex) { // Check if the page still exists. if (*iPageIndex >= rDocument.GetSdPageCount(PK_STANDARD)) continue; SdrPageObj* pPageObj = (*aPageObjIter++); pPageObj->SetReferencedPage(rDocument.GetSdPage(*iPageIndex, PK_STANDARD)); } // if there are more page objects than pages left, set the rest to invisible int nHangoverCount = 0; while (aPageObjIter != aHandoutPageObjects.end()) { (*aPageObjIter++)->SetReferencedPage(nullptr); nHangoverCount++; } // Hide outlines for objects that have pages attached. if (nHangoverCount > 0) { int nSkip = aHandoutPageObjects.size() - nHangoverCount; aShapeIter.Reset(); while (aShapeIter.IsMore()) { SdrPathObj* pPathObj = dynamic_cast(aShapeIter.Next()); if (pPathObj) { if (nSkip > 0) --nSkip; else pPathObj->SetMergedItem(XLineStyleItem(drawing::LineStyle_NONE)); } } } if( xHandoutPage.is() ) try { xHandoutPage->setPropertyValue( sPageNumber, Any( static_cast(mnHandoutPageIndex) ) ); } catch( Exception& ) { } rViewShell.SetPrintedHandoutPageNum( mnHandoutPageIndex + 1 ); MapMode aMap (rPrinter.GetMapMode()); rPrinter.SetMapMode(maMap); PrintPage( rPrinter, rPrintView, rHandoutPage, pView, false, rVisibleLayers, rPrintableLayers); PrintMessage( rPrinter, msPageString, maPageStringOffset); if( xHandoutPage.is() ) try { xHandoutPage->setPropertyValue( sPageNumber, Any( static_cast(0) ) ); } catch( Exception& ) { } rViewShell.SetPrintedHandoutPageNum(1); // Restore outlines. if (nHangoverCount > 0) { aShapeIter.Reset(); while (aShapeIter.IsMore()) { SdrPathObj* pPathObj = dynamic_cast(aShapeIter.Next()); if (pPathObj != nullptr) pPathObj->SetMergedItem(XLineStyleItem(drawing::LineStyle_SOLID)); } } } private: const sal_uInt16 mnHandoutPageIndex; const std::vector maPageIndices; }; /** The outline information (title, subtitle, outline objects) of the document. There is no fixed mapping of slides to printer pages. */ class OutlinerPrinterPage : public PrinterPage { public: OutlinerPrinterPage ( OutlinerParaObject* pParaObject, const MapMode& rMapMode, const OUString& rsPageString, const Point& rPageStringOffset, const DrawModeFlags nDrawMode, const Orientation eOrientation, const sal_uInt16 nPaperTray) : PrinterPage(PK_HANDOUT, rMapMode, false, rsPageString, rPageStringOffset, nDrawMode, eOrientation, nPaperTray), mpParaObject(pParaObject) { } virtual ~OutlinerPrinterPage() { mpParaObject.reset(); } virtual void Print ( Printer& rPrinter, SdDrawDocument& rDocument, ViewShell& rViewShell, View* pView, DrawView& rPrintView, const SetOfByte& rVisibleLayers, const SetOfByte& rPrintableLayers) const override { (void)rViewShell; (void)pView; (void)rPrintView; (void)rVisibleLayers; (void)rPrintableLayers; // Set up the printer. rPrinter.SetMapMode(maMap); // Get and set up the outliner. const Rectangle aOutRect (rPrinter.GetPageOffset(), rPrinter.GetOutputSize()); Outliner* pOutliner = rDocument.GetInternalOutliner(); const sal_uInt16 nSavedOutlMode (pOutliner->GetMode()); const bool bSavedUpdateMode (pOutliner->GetUpdateMode()); const Size aSavedPaperSize (pOutliner->GetPaperSize()); pOutliner->Init(OUTLINERMODE_OUTLINEVIEW); pOutliner->SetPaperSize(aOutRect.GetSize()); pOutliner->SetUpdateMode(true); pOutliner->Clear(); pOutliner->SetText(*mpParaObject); pOutliner->Draw(&rPrinter, aOutRect); PrintMessage( rPrinter, msPageString, maPageStringOffset); // Restore outliner and printer. pOutliner->Clear(); pOutliner->SetUpdateMode(bSavedUpdateMode); pOutliner->SetPaperSize(aSavedPaperSize); pOutliner->Init(nSavedOutlMode); } private: std::unique_ptr mpParaObject; }; } //===== DocumentRenderer::Implementation ====================================== class DocumentRenderer::Implementation : public SfxListener, public vcl::PrinterOptionsHelper { public: explicit Implementation (ViewShellBase& rBase) : mxObjectShell(rBase.GetDocShell()) , mrBase(rBase) , mbIsDisposed(false) , mpPrinter(nullptr) , mpOptions() , maPrinterPages() , mpPrintView() , mbHasOrientationWarningBeenShown(false) { DialogCreator aCreator( mrBase, mrBase.GetDocShell()->GetDocumentType() == DOCUMENT_TYPE_IMPRESS, GetCurrentPageIndex() ); m_aUIProperties = aCreator.GetDialogControls(); maSlidesPerPage = aCreator.GetSlidesPerPage(); StartListening(mrBase); } virtual ~Implementation() { EndListening(mrBase); } virtual void Notify (SfxBroadcaster& rBroadcaster, const SfxHint& rHint) override { if (&rBroadcaster != &static_cast(mrBase)) return; const SfxSimpleHint* pSimpleHint = dynamic_cast(&rHint); if (pSimpleHint != nullptr && pSimpleHint->GetId() == SFX_HINT_DYING) { Dispose(); } } /** Process the sequence of properties given to one of the XRenderable methods. */ void ProcessProperties (const css::uno::Sequence& rOptions) { OSL_ASSERT(!mbIsDisposed); if (mbIsDisposed) return; bool bIsValueChanged = processProperties( rOptions ); bool bIsPaperChanged = false; // The RenderDevice property is handled specially: its value is // stored in mpPrinter instead of being retrieved on demand. Any aDev( getValue( "RenderDevice" ) ); Reference xRenderDevice; if (aDev >>= xRenderDevice) { VCLXDevice* pDevice = VCLXDevice::GetImplementation(xRenderDevice); VclPtr< OutputDevice > pOut = pDevice ? pDevice->GetOutputDevice() : VclPtr< OutputDevice >(); mpPrinter = dynamic_cast(pOut.get()); Size aPageSizePixel = mpPrinter ? mpPrinter->GetPaperSizePixel() : Size(); if( aPageSizePixel != maPrinterPageSizePixel ) { bIsPaperChanged = true; maPrinterPageSizePixel = aPageSizePixel; } } if (bIsValueChanged) { if ( ! mpOptions ) mpOptions.reset(new PrintOptions(*this, maSlidesPerPage)); } if( bIsValueChanged || bIsPaperChanged ) PreparePages(); } /** Return the number of pages that are to be printed. */ sal_Int32 GetPrintPageCount() { OSL_ASSERT(!mbIsDisposed); if (mbIsDisposed) return 0; else return maPrinterPages.size(); } /** Return a sequence of properties that can be returned by the XRenderable::getRenderer() method. */ css::uno::Sequence GetProperties ( const css::uno::Sequence& rOptions) { (void)rOptions; css::uno::Sequence aProperties (3); aProperties[0].Name = "ExtraPrintUIOptions"; aProperties[0].Value <<= comphelper::containerToSequence(m_aUIProperties); aProperties[1].Name = "PageSize"; aProperties[1].Value <<= maPrintSize; // FIXME: is this always true ? aProperties[2].Name = "PageIncludesNonprintableArea"; aProperties[2].Value = makeAny( true ); return aProperties; } /** Print one of the prepared pages. */ void PrintPage (const sal_Int32 nIndex) { OSL_ASSERT(!mbIsDisposed); if (mbIsDisposed) return; Printer& rPrinter (*mpPrinter); std::shared_ptr pViewShell (mrBase.GetMainViewShell()); if ( ! pViewShell) return; SdDrawDocument* pDocument = pViewShell->GetDoc(); OSL_ASSERT(pDocument!=nullptr); std::shared_ptr pDrawViewShell( std::dynamic_pointer_cast(mrBase.GetMainViewShell())); if (!mpPrintView) mpPrintView.reset(new DrawView(mrBase.GetDocShell(), &rPrinter, nullptr)); if (nIndex<0 || sal::static_int_cast(nIndex)>=maPrinterPages.size()) return; const std::shared_ptr pPage (maPrinterPages[nIndex]); OSL_ASSERT(pPage); if ( ! pPage) return; const Orientation eSavedOrientation (rPrinter.GetOrientation()); const DrawModeFlags nSavedDrawMode (rPrinter.GetDrawMode()); const MapMode aSavedMapMode (rPrinter.GetMapMode()); const sal_uInt16 nSavedPaperBin (rPrinter.GetPaperBin()); // Set page orientation. if ( ! rPrinter.SetOrientation(pPage->GetOrientation())) { if ( ! mbHasOrientationWarningBeenShown && mpOptions->IsWarningOrientation()) { mbHasOrientationWarningBeenShown = true; // Show warning that the orientation could not be set. if (pViewShell) { ScopedVclPtrInstance aWarnBox( pViewShell->GetActiveWindow(), (WinBits)(WB_OK_CANCEL | WB_DEF_CANCEL), SD_RESSTR(STR_WARN_PRINTFORMAT_FAILURE)); if (aWarnBox->Execute() != RET_OK) return; } } } // Set the draw mode. rPrinter.SetDrawMode(pPage->GetDrawMode()); // Set paper tray. rPrinter.SetPaperBin(pPage->GetPaperTray()); // Print the actual page. pPage->Print( rPrinter, *pDocument, *pViewShell, pDrawViewShell ? pDrawViewShell->GetView() : nullptr, *mpPrintView, pViewShell->GetFrameView()->GetVisibleLayers(), pViewShell->GetFrameView()->GetPrintableLayers()); rPrinter.SetOrientation(eSavedOrientation); rPrinter.SetDrawMode(nSavedDrawMode); rPrinter.SetMapMode(aSavedMapMode); rPrinter.SetPaperBin(nSavedPaperBin); } private: // rhbz#657394: keep the document alive: prevents crash when SfxObjectShellRef mxObjectShell; // destroying mpPrintView ViewShellBase& mrBase; bool mbIsDisposed; VclPtr mpPrinter; Size maPrinterPageSizePixel; std::unique_ptr mpOptions; std::vector< std::shared_ptr< ::sd::PrinterPage> > maPrinterPages; std::unique_ptr mpPrintView; bool mbHasOrientationWarningBeenShown; std::vector maSlidesPerPage; awt::Size maPrintSize; void Dispose() { mbIsDisposed = true; } sal_Int32 GetCurrentPageIndex() const { const ViewShell *pShell = mrBase.GetMainViewShell().get(); const SdPage *pCurrentPage = pShell ? pShell->getCurrentPage() : nullptr; return pCurrentPage ? (pCurrentPage->GetPageNum()-1)/2 : -1; } /** Determine and set the paper orientation. */ bool SetupPaperOrientation ( const PageKind ePageKind, PrintInfo& rInfo) { SdDrawDocument* pDocument = mrBase.GetMainViewShell()->GetDoc(); rInfo.meOrientation = ORIENTATION_PORTRAIT; if( ! mpOptions->IsBooklet()) { rInfo.meOrientation = pDocument->GetSdPage(0, ePageKind)->GetOrientation(); } else if (rInfo.maPageSize.Width() < rInfo.maPageSize.Height()) rInfo.meOrientation = ORIENTATION_LANDSCAPE; // Draw and Notes should usually abide by their specified paper size Size aPaperSize; if (!mpOptions->IsPrinterPreferred(pDocument->GetDocumentType())) { aPaperSize.setWidth(rInfo.maPageSize.Width()); aPaperSize.setHeight(rInfo.maPageSize.Height()); } else { aPaperSize.setWidth(rInfo.mpPrinter->GetPaperSize().Width()); aPaperSize.setHeight(rInfo.mpPrinter->GetPaperSize().Height()); } maPrintSize = awt::Size(aPaperSize.Width(), aPaperSize.Height()); if (mpOptions->IsPrinterPreferred(pDocument->GetDocumentType())) { if( (rInfo.meOrientation == ORIENTATION_LANDSCAPE && (aPaperSize.Width() < aPaperSize.Height())) || (rInfo.meOrientation == ORIENTATION_PORTRAIT && (aPaperSize.Width() > aPaperSize.Height())) ) { maPrintSize = awt::Size(aPaperSize.Height(), aPaperSize.Width()); } } return true; } /** Top most method for preparing printer pages. In this and the other Prepare... methods the various special cases are detected and handled. For every page that is to be printed (that may contain several slides) one PrinterPage object is created and inserted into maPrinterPages. */ void PreparePages() { mpPrintView.reset(); maPrinterPages.clear(); mbHasOrientationWarningBeenShown = false; ViewShell* pShell = mrBase.GetMainViewShell().get(); PrintInfo aInfo (mpPrinter, mpOptions->IsPrintMarkedOnly()); if (aInfo.mpPrinter!=nullptr && pShell!=nullptr) { MapMode aMap (aInfo.mpPrinter->GetMapMode()); aMap.SetMapUnit(MAP_100TH_MM); aInfo.maMap = aMap; mpPrinter->SetMapMode(aMap); ::Outliner& rOutliner = mrBase.GetDocument()->GetDrawOutliner(); const EEControlBits nSavedControlWord (rOutliner.GetControlWord()); EEControlBits nCntrl = nSavedControlWord; nCntrl &= ~EEControlBits::MARKFIELDS; nCntrl &= ~EEControlBits::ONLINESPELLING; rOutliner.SetControlWord( nCntrl ); // When in outline view then apply all pending changes to the model. if( dynamic_cast< OutlineViewShell *>( pShell ) != nullptr) static_cast(pShell)->PrepareClose (false); // Collect some frequently used data. if (mpOptions->IsDate()) { aInfo.msTimeDate += GetSdrGlobalData().GetLocaleData()->getDate( Date( Date::SYSTEM ) ); aInfo.msTimeDate += " "; } if (mpOptions->IsTime()) aInfo.msTimeDate += GetSdrGlobalData().GetLocaleData()->getTime( ::tools::Time( ::tools::Time::SYSTEM ), false ); // Draw and Notes should usually use specified paper size when printing if (!mpOptions->IsPrinterPreferred(mrBase.GetDocShell()->GetDocumentType())) { aInfo.maPrintSize = mrBase.GetDocument()->GetSdPage(0, PK_STANDARD)->GetSize(); maPrintSize = awt::Size(aInfo.maPrintSize.Width(), aInfo.maPrintSize.Height()); } else { aInfo.maPrintSize = aInfo.mpPrinter->GetOutputSize(); maPrintSize = awt::Size( aInfo.mpPrinter->GetPaperSize().Width(), aInfo.mpPrinter->GetPaperSize().Height()); } switch (mpOptions->GetOutputQuality()) { case 1: // Grayscale aInfo.mnDrawMode = DrawModeFlags::GrayLine | DrawModeFlags::GrayFill | DrawModeFlags::GrayText | DrawModeFlags::GrayBitmap | DrawModeFlags::GrayGradient; break; case 2: // Black & White aInfo.mnDrawMode = DrawModeFlags::BlackLine | DrawModeFlags::WhiteFill | DrawModeFlags::BlackText | DrawModeFlags::GrayBitmap | DrawModeFlags::WhiteGradient; break; default: aInfo.mnDrawMode = DrawModeFlags::Default; } if (mpOptions->IsDraw()) PrepareStdOrNotes(PK_STANDARD, aInfo); if (mpOptions->IsNotes()) PrepareStdOrNotes(PK_NOTES, aInfo); if (mpOptions->IsHandout()) { InitHandoutTemplate(); PrepareHandout(aInfo); } if (mpOptions->IsOutline()) PrepareOutline(aInfo); rOutliner.SetControlWord(nSavedControlWord); } } /** Create the page objects of the handout template. When the actual printing takes place then the page objects are assigned different sets of slides for each printed page (see HandoutPrinterPage::Print). */ void InitHandoutTemplate() { const sal_Int32 nSlidesPerHandout (mpOptions->GetHandoutPageCount()); const bool bHandoutHorizontal (mpOptions->IsHandoutHorizontal()); AutoLayout eLayout = AUTOLAYOUT_HANDOUT6; switch (nSlidesPerHandout) { case 0: eLayout = AUTOLAYOUT_NONE; break; // AUTOLAYOUT_HANDOUT1; break; case 1: eLayout = AUTOLAYOUT_HANDOUT1; break; case 2: eLayout = AUTOLAYOUT_HANDOUT2; break; case 3: eLayout = AUTOLAYOUT_HANDOUT3; break; case 4: eLayout = AUTOLAYOUT_HANDOUT4; break; default: case 6: eLayout = AUTOLAYOUT_HANDOUT6; break; case 9: eLayout = AUTOLAYOUT_HANDOUT9; break; } if( !mrBase.GetDocument() ) return; SdDrawDocument& rModel = *mrBase.GetDocument(); // first, prepare handout page (not handout master) SdPage* pHandout = rModel.GetSdPage(0, PK_HANDOUT); if( !pHandout ) return; // delete all previous shapes from handout page while( pHandout->GetObjCount() ) { SdrObject* pObj = pHandout->NbcRemoveObject(0); if( pObj ) SdrObject::Free( pObj ); } const bool bDrawLines (eLayout == AUTOLAYOUT_HANDOUT3); std::vector< Rectangle > aAreas; SdPage::CalculateHandoutAreas( rModel, eLayout, bHandoutHorizontal, aAreas ); std::vector< Rectangle >::iterator iter( aAreas.begin() ); while( iter != aAreas.end() ) { pHandout->NbcInsertObject( new SdrPageObj((*iter++)) ); if( bDrawLines && (iter != aAreas.end()) ) { Rectangle aRect( (*iter++) ); basegfx::B2DPolygon aPoly; aPoly.insert(0, basegfx::B2DPoint( aRect.Left(), aRect.Top() ) ); aPoly.insert(1, basegfx::B2DPoint( aRect.Right(), aRect.Top() ) ); basegfx::B2DHomMatrix aMatrix; aMatrix.translate( 0.0, static_cast< double >( aRect.GetHeight() / 7 ) ); basegfx::B2DPolyPolygon aPathPoly; for( sal_uInt16 nLine = 0; nLine < 7; nLine++ ) { aPoly.transform( aMatrix ); aPathPoly.append( aPoly ); } SdrPathObj* pPathObj = new SdrPathObj(OBJ_PATHLINE, aPathPoly ); pPathObj->SetMergedItem(XLineStyleItem(drawing::LineStyle_SOLID)); pPathObj->SetMergedItem(XLineColorItem(OUString(), Color(COL_BLACK))); pHandout->NbcInsertObject( pPathObj ); } } } /** Detect whether the specified slide is to be printed. @return When the slide is not to be printed then is returned. Otherwise a pointer to the slide is returned. */ SdPage* GetFilteredPage ( const sal_Int32 nPageIndex, const PageKind ePageKind) const { OSL_ASSERT(mrBase.GetDocument() != nullptr); OSL_ASSERT(nPageIndex>=0); SdPage* pPage = mrBase.GetDocument()->GetSdPage( sal::static_int_cast(nPageIndex), ePageKind); if (pPage == nullptr) return nullptr; if ( ! pPage->IsExcluded() || mpOptions->IsPrintExcluded()) return pPage; else return nullptr; } /** Prepare the outline of the document for printing. There is no fixed number of slides whose outline data is put onto one printer page. If the current printer page has enough room for the outline of the current slide then that is added. Otherwise a new printer page is started. */ void PrepareOutline (PrintInfo& rInfo) { MapMode aMap (rInfo.maMap); Point aPageOfs (rInfo.mpPrinter->GetPageOffset() ); aMap.SetScaleX(Fraction(1,2)); aMap.SetScaleY(Fraction(1,2)); mpPrinter->SetMapMode(aMap); Rectangle aOutRect(aPageOfs, rInfo.mpPrinter->GetOutputSize()); if( aOutRect.GetWidth() > aOutRect.GetHeight() ) { Size aPaperSize( rInfo.mpPrinter->PixelToLogic( rInfo.mpPrinter->GetPaperSizePixel(), MapMode( MAP_100TH_MM ) ) ); maPrintSize.Width = aPaperSize.Height(); maPrintSize.Height = aPaperSize.Width(); const long nRotatedWidth = aOutRect.GetHeight(); const long nRotatedHeight = aOutRect.GetWidth(); aOutRect = Rectangle( Point( aPageOfs.Y(), aPageOfs.X() ), Size( nRotatedWidth, nRotatedHeight ) ); } Outliner* pOutliner = mrBase.GetDocument()->GetInternalOutliner(); pOutliner->Init(OUTLINERMODE_OUTLINEVIEW); const sal_uInt16 nSavedOutlMode (pOutliner->GetMode()); const bool bSavedUpdateMode (pOutliner->GetUpdateMode()); const Size aSavedPaperSize (pOutliner->GetPaperSize()); const MapMode aSavedMapMode (pOutliner->GetRefMapMode()); pOutliner->SetPaperSize(aOutRect.GetSize()); pOutliner->SetUpdateMode(true); long nPageH = aOutRect.GetHeight(); std::vector< sal_Int32 > aPages; sal_Int32 nPageCount = mrBase.GetDocument()->GetSdPageCount(PK_STANDARD); StringRangeEnumerator::getRangesFromString( mpOptions->GetPrinterSelection(nPageCount, GetCurrentPageIndex()), aPages, 0, nPageCount-1); for (size_t nIndex = 0, nCount = aPages.size(); nIndex < nCount;) { pOutliner->Clear(); pOutliner->SetFirstPageNumber(aPages[nIndex]+1); Paragraph* pPara = nullptr; sal_Int32 nH (0); while (nH < nPageH && nIndexGetObjCount()) { SdrObject* pObj = pPage->GetObj(nObj++); if (pObj->GetObjInventor() == SdrInventor && pObj->GetObjIdentifier() == OBJ_TITLETEXT) { pTextObj = dynamic_cast(pObj); } } pPara = pOutliner->GetParagraph(pOutliner->GetParagraphCount() - 1); if (pTextObj!=nullptr && !pTextObj->IsEmptyPresObj() && pTextObj->GetOutlinerParaObject()) { pOutliner->AddText(*(pTextObj->GetOutlinerParaObject())); } else pOutliner->Insert(OUString()); pTextObj = nullptr; nObj = 0; while (pTextObj==nullptr && nObjGetObjCount()) { SdrObject* pObj = pPage->GetObj(nObj++); if (pObj->GetObjInventor() == SdrInventor && pObj->GetObjIdentifier() == OBJ_OUTLINETEXT) { pTextObj = dynamic_cast(pObj); } } bool bSubTitle (false); if (!pTextObj) { bSubTitle = true; pTextObj = dynamic_cast(pPage->GetPresObj(PRESOBJ_TEXT)); // Untertitel vorhanden? } sal_Int32 nParaCount1 = pOutliner->GetParagraphCount(); if (pTextObj!=nullptr && !pTextObj->IsEmptyPresObj() && pTextObj->GetOutlinerParaObject()) { pOutliner->AddText(*(pTextObj->GetOutlinerParaObject())); } if (bSubTitle ) { const sal_Int32 nParaCount2 (pOutliner->GetParagraphCount()); for (sal_Int32 nPara=nParaCount1; nParaGetParagraph(nPara); if (pP!=nullptr && pOutliner->GetDepth(nPara) > 0) pOutliner->SetDepth(pP, 0); } } nH = pOutliner->GetTextHeight(); } // Remove the last paragraph when that does not fit completely on // the current page. if (nH > nPageH && pPara!=nullptr) { sal_Int32 nCnt = pOutliner->GetAbsPos( pOutliner->GetParagraph( pOutliner->GetParagraphCount() - 1 ) ); sal_Int32 nParaPos = pOutliner->GetAbsPos( pPara ); nCnt -= nParaPos; pPara = pOutliner->GetParagraph( ++nParaPos ); if ( nCnt && pPara ) { pOutliner->Remove(pPara, nCnt); --nIndex; } } maPrinterPages.push_back( std::shared_ptr( new OutlinerPrinterPage( pOutliner->CreateParaObject(), aMap, rInfo.msTimeDate, aPageOfs, rInfo.mnDrawMode, rInfo.meOrientation, rInfo.mpPrinter->GetPaperBin()))); } pOutliner->SetRefMapMode(aSavedMapMode); pOutliner->SetUpdateMode(bSavedUpdateMode); pOutliner->SetPaperSize(aSavedPaperSize); pOutliner->Init(nSavedOutlMode); } /** Prepare handout pages for slides that are to be printed. */ void PrepareHandout (PrintInfo& rInfo) { SdDrawDocument* pDocument = mrBase.GetDocument(); OSL_ASSERT(pDocument != nullptr); SdPage& rHandoutPage (*pDocument->GetSdPage(0, PK_HANDOUT)); const bool bScalePage (mpOptions->IsPageSize()); sal_uInt16 nPaperBin; if ( ! mpOptions->IsPaperBin()) nPaperBin = rHandoutPage.GetPaperBin(); else nPaperBin = rInfo.mpPrinter->GetPaperBin(); // Change orientation? SdPage& rMaster (dynamic_cast(rHandoutPage.TRG_GetMasterPage())); rInfo.meOrientation = rMaster.GetOrientation(); const Size aPaperSize (rInfo.mpPrinter->GetPaperSize()); if( (rInfo.meOrientation == ORIENTATION_LANDSCAPE && (aPaperSize.Width() < aPaperSize.Height())) || (rInfo.meOrientation == ORIENTATION_PORTRAIT && (aPaperSize.Width() > aPaperSize.Height())) ) { maPrintSize = awt::Size(aPaperSize.Height(), aPaperSize.Width()); } else { maPrintSize = awt::Size(aPaperSize.Width(), aPaperSize.Height()); } MapMode aMap (rInfo.maMap); const Point aPageOfs (rInfo.mpPrinter->GetPageOffset()); if ( bScalePage ) { const Size aPageSize (rHandoutPage.GetSize()); const Size aPrintSize (rInfo.mpPrinter->GetOutputSize()); const double fHorz = (double) aPrintSize.Width() / aPageSize.Width(); const double fVert = (double) aPrintSize.Height() / aPageSize.Height(); Fraction aFract; if ( fHorz < fVert ) aFract = Fraction(aPrintSize.Width(), aPageSize.Width()); else aFract = Fraction(aPrintSize.Height(), aPageSize.Height()); aMap.SetScaleX(aFract); aMap.SetScaleY(aFract); aMap.SetOrigin(Point()); } std::shared_ptr pViewShell (mrBase.GetMainViewShell()); pViewShell->WriteFrameViewData(); // Count page shapes. sal_uInt32 nShapeCount (0); SdrObjListIter aShapeIter (rHandoutPage); while (aShapeIter.IsMore()) { SdrPageObj* pPageObj = dynamic_cast(aShapeIter.Next()); if (pPageObj) ++nShapeCount; } const sal_uInt16 nPageCount = mrBase.GetDocument()->GetSdPageCount(PK_STANDARD); const sal_uInt16 nHandoutPageCount = nShapeCount ? (nPageCount + nShapeCount - 1) / nShapeCount : 0; pViewShell->SetPrintedHandoutPageCount( nHandoutPageCount ); mrBase.GetDocument()->setHandoutPageCount( nHandoutPageCount ); // Distribute pages to handout pages. StringRangeEnumerator aRangeEnum( mpOptions->GetPrinterSelection(nPageCount, GetCurrentPageIndex()), 0, nPageCount-1); std::vector aPageIndices; sal_uInt16 nPrinterPageIndex = 0; StringRangeEnumerator::Iterator it = aRangeEnum.begin(), itEnd = aRangeEnum.end(); bool bLastLoop = (it == itEnd); while (!bLastLoop) { sal_Int32 nPageIndex = *it; ++it; bLastLoop = (it == itEnd); if (GetFilteredPage(nPageIndex, PK_STANDARD)) aPageIndices.push_back(nPageIndex); else if (!bLastLoop) continue; // Create a printer page when we have found one page for each // placeholder or when this is the last (and special) loop. if (!aPageIndices.empty() && (aPageIndices.size() == nShapeCount || bLastLoop)) { maPrinterPages.push_back( std::shared_ptr( new HandoutPrinterPage( nPrinterPageIndex++, aPageIndices, aMap, rInfo.msTimeDate, aPageOfs, rInfo.mnDrawMode, rInfo.meOrientation, nPaperBin))); aPageIndices.clear(); } } } /** Prepare the notes pages or regular slides. */ void PrepareStdOrNotes ( const PageKind ePageKind, PrintInfo& rInfo) { OSL_ASSERT(rInfo.mpPrinter != nullptr); // Fill in page kind specific data. SdDrawDocument* pDocument = mrBase.GetMainViewShell()->GetDoc(); if (pDocument->GetSdPageCount(ePageKind) == 0) return; SdPage* pRefPage = pDocument->GetSdPage(0, ePageKind); rInfo.maPageSize = pRefPage->GetSize(); if ( ! SetupPaperOrientation(ePageKind, rInfo)) return; MapMode aMap (rInfo.maMap); rInfo.maMap = aMap; if (mpOptions->IsBooklet()) PrepareBooklet(ePageKind, rInfo); else PrepareRegularPages(ePageKind, rInfo); } /** Prepare slides in a non-booklet way: one slide per one to many printer pages. */ void PrepareRegularPages ( const PageKind ePageKind, PrintInfo& rInfo) { std::shared_ptr pViewShell (mrBase.GetMainViewShell()); pViewShell->WriteFrameViewData(); sal_Int32 nPageCount = mrBase.GetDocument()->GetSdPageCount(PK_STANDARD); StringRangeEnumerator aRangeEnum( mpOptions->GetPrinterSelection(nPageCount, GetCurrentPageIndex()), 0, nPageCount-1); for (StringRangeEnumerator::Iterator it = aRangeEnum.begin(), itEnd = aRangeEnum.end(); it != itEnd; ++it) { SdPage* pPage = GetFilteredPage(*it, ePageKind); if (pPage == nullptr) continue; MapMode aMap (rInfo.maMap); // is it possible that the page size changed? const Size aPageSize = pPage->GetSize(); if (mpOptions->IsPageSize()) { const double fHorz ((double) rInfo.maPrintSize.Width() / aPageSize.Width()); const double fVert ((double) rInfo.maPrintSize.Height() / aPageSize.Height()); Fraction aFract; if (fHorz < fVert) aFract = Fraction(rInfo.maPrintSize.Width(), aPageSize.Width()); else aFract = Fraction(rInfo.maPrintSize.Height(), aPageSize.Height()); aMap.SetScaleX(aFract); aMap.SetScaleY(aFract); aMap.SetOrigin(Point()); } if (mpOptions->IsPrintPageName()) { rInfo.msPageString = pPage->GetName(); rInfo.msPageString += " "; } else rInfo.msPageString.clear(); rInfo.msPageString += rInfo.msTimeDate; long aPageWidth = aPageSize.Width() - pPage->GetLftBorder() - pPage->GetRgtBorder(); long aPageHeight = aPageSize.Height() - pPage->GetUppBorder() - pPage->GetLwrBorder(); // Bugfix for 44530: // if it was implicitly changed (Landscape/Portrait), // this is considered for tiling, respectively for the splitting up // (Poster) if( ( rInfo.maPrintSize.Width() > rInfo.maPrintSize.Height() && aPageWidth < aPageHeight ) || ( rInfo.maPrintSize.Width() < rInfo.maPrintSize.Height() && aPageWidth > aPageHeight ) ) { const sal_Int32 nTmp (rInfo.maPrintSize.Width()); rInfo.maPrintSize.Width() = rInfo.maPrintSize.Height(); rInfo.maPrintSize.Height() = nTmp; } if (mpOptions->IsTilePage() && aPageWidth < rInfo.maPrintSize.Width() && aPageHeight < rInfo.maPrintSize.Height()) { // Put multiple slides on one printer page. PrepareTiledPage(*it, *pPage, ePageKind, rInfo); } else { rInfo.maMap = aMap; PrepareScaledPage(*it, *pPage, ePageKind, rInfo); } } } /** Put two slides on one printer page. */ void PrepareBooklet ( const PageKind ePageKind, const PrintInfo& rInfo) { MapMode aStdMap (rInfo.maMap); Point aOffset; Size aPrintSize_2 (rInfo.maPrintSize); Size aPageSize_2 (rInfo.maPageSize); if (rInfo.meOrientation == ORIENTATION_LANDSCAPE) aPrintSize_2.Width() >>= 1; else aPrintSize_2.Height() >>= 1; const double fPageWH = (double) aPageSize_2.Width() / aPageSize_2.Height(); const double fPrintWH = (double) aPrintSize_2.Width() / aPrintSize_2.Height(); if( fPageWH < fPrintWH ) { aPageSize_2.Width() = (long) ( aPrintSize_2.Height() * fPageWH ); aPageSize_2.Height()= aPrintSize_2.Height(); } else { aPageSize_2.Width() = aPrintSize_2.Width(); aPageSize_2.Height() = (long) ( aPrintSize_2.Width() / fPageWH ); } MapMode aMap (rInfo.maMap); aMap.SetScaleX( Fraction( aPageSize_2.Width(), rInfo.maPageSize.Width() ) ); aMap.SetScaleY( Fraction( aPageSize_2.Height(), rInfo.maPageSize.Height() ) ); // calculate adjusted print size const Size aAdjustedPrintSize (OutputDevice::LogicToLogic( rInfo.maPrintSize, aStdMap, aMap)); if (rInfo.meOrientation == ORIENTATION_LANDSCAPE) { aOffset.X() = ( ( aAdjustedPrintSize.Width() >> 1 ) - rInfo.maPageSize.Width() ) >> 1; aOffset.Y() = ( aAdjustedPrintSize.Height() - rInfo.maPageSize.Height() ) >> 1; } else { aOffset.X() = ( aAdjustedPrintSize.Width() - rInfo.maPageSize.Width() ) >> 1; aOffset.Y() = ( ( aAdjustedPrintSize.Height() >> 1 ) - rInfo.maPageSize.Height() ) >> 1; } // create vector of pages to print sal_Int32 nPageCount = mrBase.GetDocument()->GetSdPageCount(ePageKind); StringRangeEnumerator aRangeEnum( mpOptions->GetPrinterSelection(nPageCount, GetCurrentPageIndex()), 0, nPageCount-1); std::vector< sal_uInt16 > aPageVector; for (StringRangeEnumerator::Iterator it = aRangeEnum.begin(), itEnd = aRangeEnum.end(); it != itEnd; ++it) { SdPage* pPage = GetFilteredPage(*it, ePageKind); if (pPage != nullptr) aPageVector.push_back(*it); } // create pairs of pages to print on each page typedef std::vector< std::pair< sal_uInt16, sal_uInt16 > > PairVector; PairVector aPairVector; if ( ! aPageVector.empty()) { sal_uInt32 nFirstIndex = 0, nLastIndex = aPageVector.size() - 1; if( aPageVector.size() & 1 ) aPairVector.push_back( std::make_pair( (sal_uInt16) 65535, aPageVector[ nFirstIndex++ ] ) ); else aPairVector.push_back( std::make_pair( aPageVector[ nLastIndex-- ], aPageVector[ nFirstIndex++ ] ) ); while( nFirstIndex < nLastIndex ) { if( nFirstIndex & 1 ) aPairVector.push_back( std::make_pair( aPageVector[ nFirstIndex++ ], aPageVector[ nLastIndex-- ] ) ); else aPairVector.push_back( std::make_pair( aPageVector[ nLastIndex-- ], aPageVector[ nFirstIndex++ ] ) ); } } for (sal_uInt32 nIndex=0, nCount=aPairVector.size(); nIndex < nCount; ++nIndex) { const bool bIsIndexOdd (nIndex & 1); if ((!bIsIndexOdd && mpOptions->IsPrintFrontPage()) || (bIsIndexOdd && mpOptions->IsPrintBackPage())) { const std::pair aPair (aPairVector[nIndex]); Point aSecondOffset (aOffset); if (rInfo.meOrientation == ORIENTATION_LANDSCAPE) aSecondOffset.X() += aAdjustedPrintSize.Width() / 2; else aSecondOffset.Y() += aAdjustedPrintSize.Height() / 2; maPrinterPages.push_back( std::shared_ptr( new BookletPrinterPage( aPair.first, aPair.second, aOffset, aSecondOffset, ePageKind, aMap, rInfo.mbPrintMarkedOnly, rInfo.mnDrawMode, rInfo.meOrientation, rInfo.mpPrinter->GetPaperBin()))); } } } /** Print one slide multiple times on one printer page so that the whole printer page is covered. */ void PrepareTiledPage ( const sal_Int32 nPageIndex, const SdPage& rPage, const PageKind ePageKind, const PrintInfo& rInfo) { sal_uInt16 nPaperBin; if ( ! mpOptions->IsPaperBin()) nPaperBin = rPage.GetPaperBin(); else nPaperBin = rInfo.mpPrinter->GetPaperBin(); maPrinterPages.push_back( std::shared_ptr( new TiledPrinterPage( sal::static_int_cast(nPageIndex), ePageKind, 500, rInfo.mbPrintMarkedOnly, rInfo.msPageString, rInfo.mpPrinter->GetPageOffset(), rInfo.mnDrawMode, rInfo.meOrientation, nPaperBin))); } /** Print one standard slide or notes page on one to many printer pages. More than on printer page is used when the slide is larger than the printable area. */ void PrepareScaledPage ( const sal_Int32 nPageIndex, const SdPage& rPage, const PageKind ePageKind, const PrintInfo& rInfo) { const Point aPageOffset (rInfo.mpPrinter->GetPageOffset()); sal_uInt16 nPaperBin; if ( ! mpOptions->IsPaperBin()) nPaperBin = rPage.GetPaperBin(); else nPaperBin = rInfo.mpPrinter->GetPaperBin(); // For pages larger then the printable area there // are three options: // 1. Scale down to the page to the printable area. // 2. Print only the upper left part of the page // (without the unprintable borders). // 3. Split the page into parts of the size of the // printable area. const bool bScalePage (mpOptions->IsPageSize()); const bool bCutPage (mpOptions->IsCutPage()); MapMode aMap (rInfo.maMap); if (bScalePage || bCutPage) { // Handle 1 and 2. // if CutPage is set then do not move it, otherwise move the // scaled page to printable area maPrinterPages.push_back( std::shared_ptr( new RegularPrinterPage( sal::static_int_cast(nPageIndex), ePageKind, aMap, rInfo.mbPrintMarkedOnly, rInfo.msPageString, aPageOffset, rInfo.mnDrawMode, rInfo.meOrientation, nPaperBin))); } else { // Handle 3. Print parts of the page in the size of the // printable area until the whole page is covered. // keep the page content at its position if it fits, otherwise // move it to the printable area const long nPageWidth ( rInfo.maPageSize.Width() - rPage.GetLftBorder() - rPage.GetRgtBorder()); const long nPageHeight ( rInfo.maPageSize.Height() - rPage.GetUppBorder() - rPage.GetLwrBorder()); Point aOrigin ( 0, 0 ); for (Point aPageOrigin = aOrigin; -aPageOrigin.Y()( new RegularPrinterPage( sal::static_int_cast(nPageIndex), ePageKind, aMap, rInfo.mbPrintMarkedOnly, rInfo.msPageString, aPageOffset, rInfo.mnDrawMode, rInfo.meOrientation, nPaperBin))); } } } } }; //===== DocumentRenderer ====================================================== DocumentRenderer::DocumentRenderer (ViewShellBase& rBase) : DocumentRendererInterfaceBase(m_aMutex), mpImpl(new Implementation(rBase)) { } DocumentRenderer::~DocumentRenderer() { } //----- XRenderable ----------------------------------------------------------- sal_Int32 SAL_CALL DocumentRenderer::getRendererCount ( const css::uno::Any& aSelection, const css::uno::Sequence& rOptions) throw (css::lang::IllegalArgumentException, css::uno::RuntimeException, std::exception) { (void)aSelection; mpImpl->ProcessProperties(rOptions); return mpImpl->GetPrintPageCount(); } Sequence SAL_CALL DocumentRenderer::getRenderer ( sal_Int32 nRenderer, const css::uno::Any& rSelection, const css::uno::Sequence& rOptions) throw (css::lang::IllegalArgumentException, css::uno::RuntimeException, std::exception) { (void)nRenderer; (void)rSelection; mpImpl->ProcessProperties(rOptions); return mpImpl->GetProperties(rOptions); } void SAL_CALL DocumentRenderer::render ( sal_Int32 nRenderer, const css::uno::Any& rSelection, const css::uno::Sequence& rOptions) throw (css::lang::IllegalArgumentException, css::uno::RuntimeException, std::exception) { (void)rSelection; mpImpl->ProcessProperties(rOptions); mpImpl->PrintPage(nRenderer); } } // end of namespace sd /* vim:set shiftwidth=4 softtabstop=4 expandtab: */