1 /**************************************************************
2  *
3  * Licensed to the Apache Software Foundation (ASF) under one
4  * or more contributor license agreements.  See the NOTICE file
5  * distributed with this work for additional information
6  * regarding copyright ownership.  The ASF licenses this file
7  * to you under the Apache License, Version 2.0 (the
8  * "License"); you may not use this file except in compliance
9  * with the License.  You may obtain a copy of the License at
10  *
11  *   http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing,
14  * software distributed under the License is distributed on an
15  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16  * KIND, either express or implied.  See the License for the
17  * specific language governing permissions and limitations
18  * under the License.
19  *
20  *************************************************************/
21 
22 
23 
24 #include "precompiled_sd.hxx"
25 
26 #include <com/sun/star/beans/XPropertySet.hpp>
27 
28 #include "DocumentRenderer.hxx"
29 #include "DocumentRenderer.hrc"
30 
31 #include "drawdoc.hxx"
32 #include "optsitem.hxx"
33 #include "sdresid.hxx"
34 #include "strings.hrc"
35 #include "sdattr.hxx"
36 #include "Window.hxx"
37 #include "drawview.hxx"
38 #include "DrawViewShell.hxx"
39 #include "FrameView.hxx"
40 #include "Outliner.hxx"
41 #include "OutlineViewShell.hxx"
42 
43 #include <basegfx/polygon/b2dpolygon.hxx>
44 #include <basegfx/polygon/b2dpolypolygon.hxx>
45 #include <basegfx/matrix/b2dhommatrix.hxx>
46 #include <sfx2/printer.hxx>
47 #include <editeng/editstat.hxx>
48 #include <editeng/outlobj.hxx>
49 #include <svx/svdetc.hxx>
50 #include <svx/svditer.hxx>
51 #include <svx/svdopage.hxx>
52 #include <svx/svdopath.hxx>
53 #include <svx/xlnclit.hxx>
54 #include <toolkit/awt/vclxdevice.hxx>
55 #include <tools/resary.hxx>
56 #include <unotools/localedatawrapper.hxx>
57 #include <vcl/msgbox.hxx>
58 #include <unotools/moduleoptions.hxx>
59 
60 #include <vector>
61 
62 using namespace ::com::sun::star;
63 using namespace ::com::sun::star::uno;
64 using ::rtl::OUString;
65 
66 
67 namespace sd {
68 
69 namespace {
70     OUString A2S (const char* pString)
71     {
72         return OUString::createFromAscii(pString);
73     }
74 
75 
76 
77     /** Convenience class to extract values from the sequence of properties
78         given to one of the XRenderable methods.
79     */
80     class PrintOptions
81     {
82     public:
83         PrintOptions (
84             const vcl::PrinterOptionsHelper& rHelper,
85             const ::std::vector<sal_Int32>& rSlidesPerPage)
86             : mrProperties(rHelper),
87               maSlidesPerPage(rSlidesPerPage)
88         {
89         }
90 
91         bool IsWarningOrientation (void) const
92         {
93             return GetBoolValue(NULL, true);
94         }
95 
96         bool IsPrintPageName (void) const
97         {
98             return GetBoolValue("IsPrintName");
99         }
100 
101         bool IsDate (void) const
102         {
103             return GetBoolValue("IsPrintDateTime");
104         }
105 
106         bool IsTime (void) const
107         {
108             return GetBoolValue("IsPrintDateTime");
109         }
110 
111         bool IsHiddenPages (void) const
112         {
113             return GetBoolValue("IsPrintHidden");
114         }
115 
116         bool IsHandoutHorizontal (void) const
117         {
118             return GetBoolValue("SlidesPerPageOrder", NULL, true);
119         }
120 
121         sal_Int32 GetHandoutPageCount (void) const
122         {
123             sal_uInt32 nIndex = static_cast<sal_Int32>(mrProperties.getIntValue("SlidesPerPage", sal_Int32(0)));
124             if (nIndex<maSlidesPerPage.size())
125                 return maSlidesPerPage[nIndex];
126             else if ( ! maSlidesPerPage.empty())
127                 return maSlidesPerPage[0];
128             else
129                 return 0;
130         }
131 
132         bool IsDraw (void) const
133         {
134             return GetBoolValue("PageContentType", sal_Int32(0));
135         }
136 
137         bool IsHandout (void) const
138         {
139             return GetBoolValue("PageContentType", sal_Int32(1));
140         }
141 
142         bool IsNotes (void) const
143         {
144             return GetBoolValue("PageContentType", sal_Int32(2));
145         }
146 
147         bool IsOutline (void) const
148         {
149             return GetBoolValue("PageContentType", sal_Int32(3));
150         }
151 
152         sal_uLong GetOutputQuality (void) const
153         {
154             sal_Int32 nQuality = static_cast<sal_Int32>(mrProperties.getIntValue( "Quality", sal_Int32(0) ));
155             return nQuality;
156         }
157 
158         bool IsPageSize (void) const
159         {
160             return GetBoolValue("PageOptions", sal_Int32(1));
161         }
162 
163         bool IsTilePage (void) const
164         {
165             return GetBoolValue("PageOptions", sal_Int32(2)) || GetBoolValue("PageOptions", sal_Int32(3));
166         }
167 
168         bool IsCutPage (void) const
169         {
170             return GetBoolValue("PageOptions", sal_Int32(0));
171         }
172 
173         bool IsBooklet (void) const
174         {
175             return GetBoolValue("PrintProspect", false);
176         }
177 
178         bool IsPrintExcluded (void) const
179         {
180             return (IsNotes() || IsDraw() || IsHandout()) &&  IsHiddenPages();
181         }
182 
183         bool IsPrintFrontPage (void) const
184         {
185             sal_Int32 nInclude = static_cast<sal_Int32>(mrProperties.getIntValue( "PrintProspectInclude", 0 ));
186             return nInclude == 0 || nInclude == 1;
187         }
188 
189         bool IsPrintBackPage (void) const
190         {
191             sal_Int32 nInclude = static_cast<sal_Int32>(mrProperties.getIntValue( "PrintProspectInclude", 0 ));
192             return nInclude == 0 || nInclude == 2;
193         }
194 
195         bool IsPaperBin (void) const
196         {
197             return GetBoolValue("PrintPaperFromSetup", false);
198         }
199 
200         OUString GetPrinterSelection (void) const
201         {
202             sal_Int32 nContent = static_cast<sal_Int32>(mrProperties.getIntValue( "PrintContent", 0 ));
203             OUString sValue( A2S("all") );
204             if( nContent == 1 )
205                 sValue = mrProperties.getStringValue( "PageRange", A2S( "all" ) );
206             else if( nContent == 2 )
207                 sValue = A2S( "selection" );
208             return sValue;
209         }
210 
211     private:
212         const vcl::PrinterOptionsHelper& mrProperties;
213         const ::std::vector<sal_Int32> maSlidesPerPage;
214 
215         /** When the value of the property with name pName is a boolean then
216             return its value. When the property is unknown then
217             bDefaultValue is returned.  Otherwise <FALSE/> is returned.
218         */
219         bool GetBoolValue (
220             const sal_Char* pName,
221             const bool bDefaultValue = false) const
222         {
223             sal_Bool bValue = mrProperties.getBoolValue( pName, bDefaultValue );
224             return bValue;
225         }
226 
227         /** Return <TRUE/> when the value of the property with name pName is
228             a string and its value equals pValue. When the property is
229             unknown then bDefaultValue is returned.  Otherwise <FALSE/> is
230             returned.
231         */
232         bool GetBoolValue (
233             const sal_Char* pName,
234             const sal_Char* pValue,
235             const bool bDefaultValue = false) const
236         {
237             OUString sValue( mrProperties.getStringValue( pName ) );
238             if (sValue.getLength())
239                 return sValue.equalsAscii(pValue);
240             else
241                 return bDefaultValue;
242         }
243 
244         /** Return <TRUE/> when the value of the property with name pName is
245             an integer and its value is nTriggerValue. Otherwise <FALSE/> is
246             returned.
247         */
248         bool GetBoolValue (
249             const sal_Char* pName,
250             const sal_Int32 nTriggerValue) const
251         {
252             sal_Int32 nValue = static_cast<sal_Int32>(mrProperties.getIntValue( pName ));
253             return nValue == nTriggerValue;
254         }
255     };
256 
257 
258 
259     /** This class is like MultiSelection but understands two special values.
260         "all" indicates that all pages are selected.  "selection" indicates that no
261         pages but a set of shapes is selected.
262     */
263     class Selection
264     {
265     public:
266         Selection (const OUString& rsSelection, const SdPage* pCurrentPage)
267             : mbAreAllPagesSelected(rsSelection.equalsAscii("all")),
268               mbIsShapeSelection(rsSelection.equalsAscii("selection")),
269               mnCurrentPageIndex(pCurrentPage!=NULL ? (pCurrentPage->GetPageNum()-1)/2 : -1),
270               mpSelectedPages()
271         {
272             if ( ! (mbAreAllPagesSelected || mbIsShapeSelection))
273                 mpSelectedPages.reset(new MultiSelection(rsSelection));
274         }
275 
276         bool IsMarkedOnly (void) const
277         {
278             return mbIsShapeSelection;
279         }
280 
281         /** Call with a 0 based page index.
282         */
283         bool IsSelected (const sal_Int32 nIndex) const
284         {
285             if (mbAreAllPagesSelected)
286                 return true;
287             else if (mpSelectedPages)
288                 return mpSelectedPages->IsSelected(nIndex+1);
289             else if (mbIsShapeSelection && nIndex==mnCurrentPageIndex)
290                 return true;
291             else
292                 return false;
293         }
294 
295     private:
296         const bool mbAreAllPagesSelected;
297         const bool mbIsShapeSelection;
298         const sal_Int32 mnCurrentPageIndex;
299         ::boost::scoped_ptr<MultiSelection> mpSelectedPages;
300     };
301 
302     /** A collection of values that helps to reduce the number of arguments
303         given to some functions.  Note that not all values are set at the
304         same time.
305     */
306     class PrintInfo
307     {
308     public:
309         PrintInfo (
310             const Printer* pPrinter,
311             const OUString& rsPrinterSelection,
312             const ::boost::shared_ptr<ViewShell> pView)
313             : mpPrinter(pPrinter),
314               mnDrawMode(DRAWMODE_DEFAULT),
315               msTimeDate(),
316               msPageString(),
317               maPrintSize(0,0),
318               maPageSize(0,0),
319               meOrientation(ORIENTATION_PORTRAIT),
320               maMap(),
321               maSelection(rsPrinterSelection, pView ? pView->getCurrentPage() : NULL),
322               mbPrintMarkedOnly(maSelection.IsMarkedOnly())
323         {}
324 
325         const Printer* mpPrinter;
326         sal_uLong mnDrawMode;
327         ::rtl::OUString msTimeDate;
328         ::rtl::OUString msPageString;
329         Size maPrintSize;
330         Size maPageSize;
331         Orientation meOrientation;
332         MapMode maMap;
333         const Selection maSelection;
334         bool mbPrintMarkedOnly;
335     };
336 
337 
338 
339     /** Output one page of the document to the given printer.  Note that
340         more than one document page may be output to one printer page.
341     */
342     void PrintPage (
343         Printer& rPrinter,
344         ::sd::View& rPrintView,
345         SdPage& rPage,
346         View* pView,
347         const bool bPrintMarkedOnly,
348         const SetOfByte& rVisibleLayers,
349         const SetOfByte& rPrintableLayers)
350     {
351         rPrintView.ShowSdrPage(&rPage);
352 
353         const MapMode aOriginalMapMode (rPrinter.GetMapMode());
354 
355         // Set the visible layers
356         SdrPageView* pPageView = rPrintView.GetSdrPageView();
357         OSL_ASSERT(pPageView!=NULL);
358         pPageView->SetVisibleLayers(rVisibleLayers);
359         pPageView->SetPrintableLayers(rPrintableLayers);
360 
361         if (pView!=NULL && bPrintMarkedOnly)
362             pView->DrawMarkedObj(rPrinter);
363         else
364             rPrintView.CompleteRedraw(&rPrinter, Rectangle(Point(0,0), rPage.GetSize()));
365 
366         rPrinter.SetMapMode(aOriginalMapMode);
367 
368         rPrintView.HideSdrPage();
369     }
370 
371 
372 
373 
374     /** Output a string (that typically is not part of a document page) to
375         the given printer.
376     */
377     void PrintMessage (
378         Printer& rPrinter,
379         const ::rtl::OUString& rsPageString,
380         const Point& rPageStringOffset)
381     {
382         const Font aOriginalFont (rPrinter.OutputDevice::GetFont());
383         rPrinter.SetFont(Font(FAMILY_SWISS, Size(0, 423)));
384         rPrinter.DrawText(rPageStringOffset, rsPageString);
385         rPrinter.SetFont(aOriginalFont);
386     }
387 
388 
389 
390 
391     /** Read the resource file and process it into a sequence of properties
392         that can be passed to the printing dialog.
393     */
394     class DialogCreator : Resource
395     {
396     public:
397         DialogCreator (bool bImpress)
398             : Resource(SdResId(_STR_IMPRESS_PRINT_UI_OPTIONS))
399             , mbImpress(bImpress)
400         {
401             ProcessResource();
402         }
403 
404         Sequence< beans::PropertyValue > GetDialogControls(void) const
405         {
406             if (maProperties.empty())
407                 return Sequence< beans::PropertyValue >();
408             else
409             {
410                 return Sequence<beans::PropertyValue>(
411                         &maProperties.front(),
412                         maProperties.size());
413             }
414         }
415 
416         ::std::vector<sal_Int32> GetSlidesPerPage (void) const
417         {
418             return maSlidesPerPage;
419         }
420 
421     private:
422         Any maDialog;
423         ::std::vector<beans::PropertyValue> maProperties;
424         ::std::vector<sal_Int32> maSlidesPerPage;
425         bool mbImpress;
426 
427         void ProcessResource (void)
428         {
429             SvtModuleOptions aOpt;
430             String aAppGroupname( String( SdResId( _STR_IMPRESS_PRINT_UI_GROUP_NAME ) ) );
431             aAppGroupname.SearchAndReplace( String( RTL_CONSTASCII_USTRINGPARAM( "%s" ) ),
432                                            aOpt.GetModuleName( mbImpress ? SvtModuleOptions::E_SIMPRESS : SvtModuleOptions::E_SDRAW ) );
433             AddDialogControl( vcl::PrinterOptionsHelper::getGroupControlOpt(
434                                 aAppGroupname,
435                                 rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( ".HelpID:vcl:PrintDialog:TabPage:AppPage" ) )
436                                 ) );
437 
438             uno::Sequence< rtl::OUString > aHelpIds;
439             if( mbImpress )
440             {
441                 vcl::PrinterOptionsHelper::UIControlOptions aPrintOpt;
442                 aPrintOpt.maGroupHint = rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "JobPage" ) );
443                 AddDialogControl( vcl::PrinterOptionsHelper::getSubgroupControlOpt(
444                                     String( SdResId(_STR_IMPRESS_PRINT_UI_PRINT_GROUP) ),
445                                     rtl::OUString(),
446                                     aPrintOpt )
447                                     );
448 
449                 aHelpIds.realloc( 1 );
450                 aHelpIds[0] = rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( ".HelpID:vcl:PrintDialog:PageContentType:ListBox" ) );
451                 AddDialogControl( vcl::PrinterOptionsHelper::getChoiceControlOpt(
452                                     String( SdResId( _STR_IMPRESS_PRINT_UI_CONTENT ) ),
453                                     aHelpIds,
454                                     OUString( RTL_CONSTASCII_USTRINGPARAM( "PageContentType" ) ),
455                                     CreateChoice(_STR_IMPRESS_PRINT_UI_CONTENT_CHOICES),
456                                     0,
457                                     OUString( RTL_CONSTASCII_USTRINGPARAM( "List" ) )
458                                     )
459                                 );
460 
461                 aHelpIds[0] = rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( ".HelpID:vcl:PrintDialog:SlidesPerPage:ListBox" ) );
462                 vcl::PrinterOptionsHelper::UIControlOptions
463                     aContentOpt( OUString( RTL_CONSTASCII_USTRINGPARAM( "PageContentType" ) ), 1 );
464                 AddDialogControl( vcl::PrinterOptionsHelper::getChoiceControlOpt(
465                                     String( SdResId( _STR_IMPRESS_PRINT_UI_SLIDESPERPAGE ) ),
466                                     aHelpIds,
467                                     OUString( RTL_CONSTASCII_USTRINGPARAM( "SlidesPerPage" ) ),
468                                     GetSlidesPerPageSequence(),
469                                     0,
470                                     OUString( RTL_CONSTASCII_USTRINGPARAM( "List" ) ),
471                                     Sequence< sal_Bool >(),
472                                     aContentOpt
473                                     )
474                                 );
475 
476                 aHelpIds[0] = rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( ".HelpID:vcl:PrintDialog:SlidesPerPageOrder:ListBox" ) );
477                 vcl::PrinterOptionsHelper::UIControlOptions
478                     aSlidesPerPageOpt( OUString( RTL_CONSTASCII_USTRINGPARAM( "SlidesPerPage" ) ), -1, sal_True );
479                 AddDialogControl( vcl::PrinterOptionsHelper::getChoiceControlOpt(
480                                     String( SdResId( _STR_IMPRESS_PRINT_UI_ORDER ) ),
481                                     aHelpIds,
482                                     OUString( RTL_CONSTASCII_USTRINGPARAM( "SlidesPerPageOrder" ) ),
483                                     CreateChoice(_STR_IMPRESS_PRINT_UI_ORDER_CHOICES),
484                                     0,
485                                     OUString( RTL_CONSTASCII_USTRINGPARAM( "List" ) ),
486                                     Sequence< sal_Bool >(),
487                                     aSlidesPerPageOpt )
488                                 );
489             }
490 
491             AddDialogControl( vcl::PrinterOptionsHelper::getSubgroupControlOpt(
492                                String( SdResId(_STR_IMPRESS_PRINT_UI_INCLUDE_CONTENT) ), rtl::OUString() ) );
493 
494 
495             if( mbImpress )
496             {
497                 AddDialogControl( vcl::PrinterOptionsHelper::getBoolControlOpt(
498                                     String( SdResId(_STR_IMPRESS_PRINT_UI_IS_PRINT_NAME) ),
499                                     OUString( RTL_CONSTASCII_USTRINGPARAM( ".HelpID:vcl:PrintDialog:IsPrintName:CheckBox" ) ),
500                                     OUString( RTL_CONSTASCII_USTRINGPARAM( "IsPrintName" ) ),
501                                     sal_False
502                                     )
503                                 );
504             }
505             else
506             {
507                 AddDialogControl( vcl::PrinterOptionsHelper::getBoolControlOpt(
508                                     String( SdResId(_STR_DRAW_PRINT_UI_IS_PRINT_NAME) ),
509                                     OUString( RTL_CONSTASCII_USTRINGPARAM( ".HelpID:vcl:PrintDialog:IsPrintName:CheckBox" ) ),
510                                     OUString( RTL_CONSTASCII_USTRINGPARAM( "IsPrintName" ) ),
511                                     sal_False
512                                     )
513                                 );
514             }
515 
516             AddDialogControl( vcl::PrinterOptionsHelper::getBoolControlOpt(
517                                 String( SdResId(_STR_IMPRESS_PRINT_UI_IS_PRINT_DATE) ),
518                                 OUString( RTL_CONSTASCII_USTRINGPARAM( ".HelpID:vcl:PrintDialog:IsPrintDateTime:CheckBox" ) ),
519                                 OUString( RTL_CONSTASCII_USTRINGPARAM( "IsPrintDateTime" ) ),
520                                 sal_False
521                                 )
522                             );
523 
524             if( mbImpress )
525             {
526                 AddDialogControl( vcl::PrinterOptionsHelper::getBoolControlOpt(
527                                     String( SdResId(_STR_IMPRESS_PRINT_UI_IS_PRINT_HIDDEN) ),
528                                     OUString( RTL_CONSTASCII_USTRINGPARAM( ".HelpID:vcl:PrintDialog:IsPrintHidden:CheckBox" ) ),
529                                     OUString( RTL_CONSTASCII_USTRINGPARAM( "IsPrintHidden" ) ),
530                                     sal_False
531                                     )
532                                 );
533             }
534 
535             AddDialogControl( vcl::PrinterOptionsHelper::getSubgroupControlOpt(
536                                String( SdResId(_STR_IMPRESS_PRINT_UI_QUALITY) ), rtl::OUString() ) );
537 
538             aHelpIds.realloc( 3 );
539             aHelpIds[0] = rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( ".HelpID:vcl:PrintDialog:Quality:RadioButton:0" ) );
540             aHelpIds[1] = rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( ".HelpID:vcl:PrintDialog:Quality:RadioButton:1" ) );
541             aHelpIds[2] = rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( ".HelpID:vcl:PrintDialog:Quality:RadioButton:2" ) );
542             AddDialogControl( vcl::PrinterOptionsHelper::getChoiceControlOpt(
543                                 rtl::OUString(),
544                                 aHelpIds,
545                                 OUString( RTL_CONSTASCII_USTRINGPARAM( "Quality" ) ),
546                                 CreateChoice(_STR_IMPRESS_PRINT_UI_QUALITY_CHOICES),
547                                 0
548                                 )
549                             );
550 
551             AddDialogControl( vcl::PrinterOptionsHelper::getSubgroupControlOpt(
552                                String( SdResId(_STR_IMPRESS_PRINT_UI_PAGE_OPTIONS) ), rtl::OUString() ) );
553 
554             aHelpIds.realloc( 4 );
555             aHelpIds[0] = rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( ".HelpID:vcl:PrintDialog:PageOptions:RadioButton:0" ) );
556             aHelpIds[1] = rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( ".HelpID:vcl:PrintDialog:PageOptions:RadioButton:1" ) );
557             aHelpIds[2] = rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( ".HelpID:vcl:PrintDialog:PageOptions:RadioButton:2" ) );
558             aHelpIds[3] = rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( ".HelpID:vcl:PrintDialog:PageOptions:RadioButton:3" ) );
559             if( mbImpress )
560             {
561                 // FIXME: additional dependency on PrintProspect = false
562                 vcl::PrinterOptionsHelper::UIControlOptions
563                     aPageOptionsOpt( OUString( RTL_CONSTASCII_USTRINGPARAM( "PageContentType" ) ), 0 );
564                 AddDialogControl( vcl::PrinterOptionsHelper::getChoiceControlOpt(
565                                     rtl::OUString(),
566                                     aHelpIds,
567                                     OUString( RTL_CONSTASCII_USTRINGPARAM( "PageOptions" ) ),
568                                     CreateChoice(_STR_IMPRESS_PRINT_UI_PAGE_OPTIONS_CHOICES),
569                                     0,
570                                     OUString( RTL_CONSTASCII_USTRINGPARAM( "Radio" ) ),
571                                     Sequence< sal_Bool >(),
572                                     aPageOptionsOpt
573                                     )
574                                 );
575             }
576             else
577             {
578                 vcl::PrinterOptionsHelper::UIControlOptions
579                     aPageOptionsOpt( OUString( RTL_CONSTASCII_USTRINGPARAM( "PrintProspect" ) ), sal_False );
580                 AddDialogControl( vcl::PrinterOptionsHelper::getChoiceControlOpt(
581                                     rtl::OUString(),
582                                     aHelpIds,
583                                     OUString( RTL_CONSTASCII_USTRINGPARAM( "PageOptions" ) ),
584                                     CreateChoice(_STR_IMPRESS_PRINT_UI_PAGE_OPTIONS_CHOICES_DRAW),
585                                     0,
586                                     OUString( RTL_CONSTASCII_USTRINGPARAM( "Radio" ) ),
587                                     Sequence< sal_Bool >(),
588                                     aPageOptionsOpt
589                                     )
590                                 );
591             }
592 
593             vcl::PrinterOptionsHelper::UIControlOptions aBrochureOpt;
594             aBrochureOpt.maGroupHint = OUString( RTL_CONSTASCII_USTRINGPARAM( "LayoutPage" ) );
595             AddDialogControl( vcl::PrinterOptionsHelper::getSubgroupControlOpt(
596                                String( SdResId(_STR_IMPRESS_PRINT_UI_PAGE_SIDES) ), rtl::OUString(),
597                                aBrochureOpt ) );
598 
599             // brochure printing
600             AddDialogControl( vcl::PrinterOptionsHelper::getBoolControlOpt(
601                                 String( SdResId(_STR_IMPRESS_PRINT_UI_BROCHURE) ),
602                                 rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( ".HelpID:vcl:PrintDialog:PrintProspect:CheckBox" ) ),
603                                 OUString( RTL_CONSTASCII_USTRINGPARAM( "PrintProspect" ) ),
604                                 sal_False,
605                                 aBrochureOpt
606                                 )
607                             );
608 
609             vcl::PrinterOptionsHelper::UIControlOptions
610                 aIncludeOpt( OUString( RTL_CONSTASCII_USTRINGPARAM( "PrintProspect" ) ), -1, sal_False );
611             aIncludeOpt.maGroupHint = OUString( RTL_CONSTASCII_USTRINGPARAM( "LayoutPage" ) );
612             aHelpIds.realloc( 1 );
613             aHelpIds[0] = rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( ".HelpID:vcl:PrintDialog:PrintProspectInclude:ListBox" ) );
614             AddDialogControl( vcl::PrinterOptionsHelper::getChoiceControlOpt(
615                                 String( SdResId(_STR_IMPRESS_PRINT_UI_BROCHURE_INCLUDE) ),
616                                 aHelpIds,
617                                 OUString( RTL_CONSTASCII_USTRINGPARAM( "PrintProspectInclude" ) ),
618                                 CreateChoice(_STR_IMPRESS_PRINT_UI_BROCHURE_INCLUDE_LIST),
619                                 0,
620                                 OUString( RTL_CONSTASCII_USTRINGPARAM( "List" ) ),
621                                 Sequence< sal_Bool >(),
622                                 aIncludeOpt
623                                 )
624                             );
625 
626             // paper tray (on options page)
627             vcl::PrinterOptionsHelper::UIControlOptions aPaperTrayOpt;
628             aPaperTrayOpt.maGroupHint = rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "OptionsPageOptGroup" ) );
629             AddDialogControl( vcl::PrinterOptionsHelper::getBoolControlOpt(
630                                 String( SdResId(_STR_IMPRESS_PRINT_UI_PAPER_TRAY) ),
631                                 rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( ".HelpID:vcl:PrintDialog:PrintPaperFromSetup:CheckBox" ) ),
632                                 OUString( RTL_CONSTASCII_USTRINGPARAM( "PrintPaperFromSetup" ) ),
633                                 sal_False,
634                                 aPaperTrayOpt
635                                 )
636                             );
637             // print range selection
638             vcl::PrinterOptionsHelper::UIControlOptions aPrintRangeOpt;
639             aPrintRangeOpt.mbInternalOnly = sal_True;
640             aPrintRangeOpt.maGroupHint = rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "PrintRange" ) );
641             AddDialogControl( vcl::PrinterOptionsHelper::getSubgroupControlOpt(
642                                 String( SdResId( _STR_IMPRESS_PRINT_UI_PAGE_RANGE ) ),
643                                 rtl::OUString(),
644                                 aPrintRangeOpt )
645                              );
646 
647             // create a choice for the content to create
648             rtl::OUString aPrintRangeName( RTL_CONSTASCII_USTRINGPARAM( "PrintContent" ) );
649             aHelpIds.realloc( 3 );
650             aHelpIds[0] = rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( ".HelpID:vcl:PrintDialog:PrintContent:RadioButton:0" ) );
651             aHelpIds[1] = rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( ".HelpID:vcl:PrintDialog:PrintContent:RadioButton:1" ) );
652             aHelpIds[2] = rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( ".HelpID:vcl:PrintDialog:PrintContent:RadioButton:2" ) );
653             AddDialogControl( vcl::PrinterOptionsHelper::getChoiceControlOpt( rtl::OUString(),
654                                 aHelpIds,
655                                 aPrintRangeName,
656                                 CreateChoice(mbImpress
657                                              ? _STR_IMPRESS_PRINT_UI_PAGE_RANGE_CHOICE
658                                              : _STR_DRAW_PRINT_UI_PAGE_RANGE_CHOICE),
659                                 0 )
660                             );
661             // create a an Edit dependent on "Pages" selected
662             vcl::PrinterOptionsHelper::UIControlOptions aPageRangeOpt( aPrintRangeName, 1, sal_True );
663             AddDialogControl( vcl::PrinterOptionsHelper::getEditControlOpt( rtl::OUString(),
664                                 rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( ".HelpID:vcl:PrintDialog:PageRange:Edit" ) ),
665                                 rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "PageRange" ) ),
666                                 rtl::OUString(),
667                                 aPageRangeOpt )
668                             );
669 
670             FreeResource();
671         }
672 
673         void AddDialogControl( const Any& i_rCtrl )
674         {
675             beans::PropertyValue aVal;
676             aVal.Value = i_rCtrl;
677             maProperties.push_back( aVal );
678         }
679 
680         Sequence<rtl::OUString> CreateChoice (const sal_uInt16 nResourceId) const
681         {
682             SdResId aResourceId (nResourceId);
683             ResStringArray aChoiceStrings (aResourceId);
684 
685             const sal_uInt32 nCount (aChoiceStrings.Count());
686             Sequence<rtl::OUString> aChoices (nCount);
687             for (sal_uInt32 nIndex=0; nIndex<nCount; ++nIndex)
688                 aChoices[nIndex] = aChoiceStrings.GetString(nIndex);
689 
690             return aChoices;
691         }
692 
693         Sequence<rtl::OUString> GetSlidesPerPageSequence (void)
694         {
695             const Sequence<rtl::OUString> aChoice (
696                 CreateChoice(_STR_IMPRESS_PRINT_UI_SLIDESPERPAGE_CHOICES));
697             maSlidesPerPage.clear();
698 			maSlidesPerPage.push_back(0); // first is using the default
699             for (sal_Int32 nIndex=1,nCount=aChoice.getLength(); nIndex<nCount; ++nIndex)
700                 maSlidesPerPage.push_back(aChoice[nIndex].toInt32());
701             return aChoice;
702         }
703     };
704 
705 
706 
707 
708     /** The Prepare... methods of the DocumentRenderer::Implementation class
709         create a set of PrinterPage objects that contain all necessary
710         information to do the actual printing.  There is one PrinterPage
711         object per printed page.  Derived classes implement the actual, mode
712         specific printing.
713 
714         This and all derived classes support the asynchronous printing
715         process by not storing pointers to any data with lifetime shorter
716         than the PrinterPage objects, i.e. slides, shapes, (one of) the
717         outliner (of the document).
718     */
719     class PrinterPage
720     {
721     public:
722         PrinterPage (
723             const PageKind ePageKind,
724             const MapMode& rMapMode,
725             const bool bPrintMarkedOnly,
726             const ::rtl::OUString& rsPageString,
727             const Point& rPageStringOffset,
728             const sal_uLong nDrawMode,
729             const Orientation eOrientation,
730             const sal_uInt16 nPaperTray)
731             : mePageKind(ePageKind),
732               maMap(rMapMode),
733               mbPrintMarkedOnly(bPrintMarkedOnly),
734               msPageString(rsPageString),
735               maPageStringOffset(rPageStringOffset),
736               mnDrawMode(nDrawMode),
737               meOrientation(eOrientation),
738               mnPaperTray(nPaperTray)
739         {
740         }
741 
742         virtual ~PrinterPage (void) {}
743 
744         virtual void Print (
745             Printer& rPrinter,
746             SdDrawDocument& rDocument,
747             ViewShell& rViewShell,
748             View* pView,
749             DrawView& rPrintView,
750             const SetOfByte& rVisibleLayers,
751             const SetOfByte& rPrintableLayers) const = 0;
752 
753         sal_uLong GetDrawMode (void) const { return mnDrawMode; }
754         Orientation GetOrientation (void) const { return meOrientation; }
755         sal_uInt16 GetPaperTray (void) const { return mnPaperTray; }
756 
757     protected:
758         const PageKind mePageKind;
759         const MapMode maMap;
760         const bool mbPrintMarkedOnly;
761         const ::rtl::OUString msPageString;
762         const Point maPageStringOffset;
763         const sal_uLong mnDrawMode;
764         const Orientation meOrientation;
765         const sal_uInt16 mnPaperTray;
766     };
767 
768 
769 
770 
771     /** The RegularPrinterPage is used for printing one regular slide (no
772         notes, handout, or outline) to one printer page.
773     */
774     class RegularPrinterPage : public PrinterPage
775     {
776     public:
777         RegularPrinterPage (
778             const sal_uInt16 nPageIndex,
779             const PageKind ePageKind,
780             const MapMode& rMapMode,
781             const bool bPrintMarkedOnly,
782             const ::rtl::OUString& rsPageString,
783             const Point& rPageStringOffset,
784             const sal_uLong nDrawMode,
785             const Orientation eOrientation,
786             const sal_uInt16 nPaperTray)
787             : PrinterPage(ePageKind, rMapMode, bPrintMarkedOnly, rsPageString,
788                 rPageStringOffset, nDrawMode, eOrientation, nPaperTray),
789               mnPageIndex(nPageIndex)
790         {
791         }
792 
793         virtual ~RegularPrinterPage (void) {}
794 
795         virtual void Print (
796             Printer& rPrinter,
797             SdDrawDocument& rDocument,
798             ViewShell& rViewShell,
799             View* pView,
800             DrawView& rPrintView,
801             const SetOfByte& rVisibleLayers,
802             const SetOfByte& rPrintableLayers) const
803         {
804             (void)rViewShell;
805             SdPage* pPageToPrint = rDocument.GetSdPage(mnPageIndex, mePageKind);
806             rPrinter.SetMapMode(maMap);
807             PrintPage(
808                 rPrinter,
809                 rPrintView,
810                 *pPageToPrint,
811                 pView,
812                 mbPrintMarkedOnly,
813                 rVisibleLayers,
814                 rPrintableLayers);
815             PrintMessage(
816                 rPrinter,
817                 msPageString,
818                 maPageStringOffset);
819         }
820 
821     private:
822         const sal_uInt16 mnPageIndex;
823     };
824 
825 
826 
827 
828     /** Print one slide multiple times on a printer page so that the whole
829         printer page is covered.
830     */
831     class TiledPrinterPage : public PrinterPage
832     {
833     public:
834         TiledPrinterPage (
835             const sal_uInt16 nPageIndex,
836             const PageKind ePageKind,
837             const sal_Int32 nGap,
838             const bool bPrintMarkedOnly,
839             const ::rtl::OUString& rsPageString,
840             const Point& rPageStringOffset,
841             const sal_uLong nDrawMode,
842             const Orientation eOrientation,
843             const sal_uInt16 nPaperTray)
844             : PrinterPage(ePageKind, MapMode(), bPrintMarkedOnly, rsPageString,
845                 rPageStringOffset, nDrawMode, eOrientation, nPaperTray),
846               mnPageIndex(nPageIndex),
847               mnGap(nGap)
848         {
849         }
850 
851         virtual ~TiledPrinterPage (void) {}
852 
853         virtual void Print (
854             Printer& rPrinter,
855             SdDrawDocument& rDocument,
856             ViewShell& rViewShell,
857             View* pView,
858             DrawView& rPrintView,
859             const SetOfByte& rVisibleLayers,
860             const SetOfByte& rPrintableLayers) const
861         {
862             (void)rViewShell;
863             SdPage* pPageToPrint = rDocument.GetSdPage(mnPageIndex, mePageKind);
864             if (pPageToPrint==NULL)
865                 return;
866             MapMode aMap (rPrinter.GetMapMode());
867 
868             const Size aPageSize (pPageToPrint->GetSize());
869             const Size aPrintSize (rPrinter.GetOutputSize());
870 
871             const sal_Int32 nPageWidth (aPageSize.Width() + mnGap
872                 - pPageToPrint->GetLftBorder() - pPageToPrint->GetRgtBorder());
873             const sal_Int32 nPageHeight (aPageSize.Height() + mnGap
874                 - pPageToPrint->GetUppBorder() - pPageToPrint->GetLwrBorder());
875             if (nPageWidth<=0 || nPageHeight<=0)
876                 return;
877 
878             // Print at least two rows and columns.  More if the document
879             // page fits completely onto the printer page.
880             const sal_Int32 nColumnCount (::std::max(sal_Int32(2),
881                     sal_Int32(aPrintSize.Width() / nPageWidth)));
882             const sal_Int32 nRowCount (::std::max(sal_Int32(2),
883                     sal_Int32(aPrintSize.Height() / nPageHeight)));
884             Point aPrintOrigin;
885             for (sal_Int32 nRow=0; nRow<nRowCount; ++nRow)
886                 for (sal_Int32 nColumn=0; nColumn<nColumnCount; ++nColumn)
887                 {
888                     aMap.SetOrigin(Point(nColumn*nPageWidth,nRow*nPageHeight));
889                     rPrinter.SetMapMode(aMap);
890                     PrintPage(
891                         rPrinter,
892                         rPrintView,
893                         *pPageToPrint,
894                         pView,
895                         mbPrintMarkedOnly,
896                         rVisibleLayers,
897                         rPrintableLayers);
898                 }
899 
900             PrintMessage(
901                 rPrinter,
902                 msPageString,
903                 maPageStringOffset);
904         }
905 
906     private:
907         const sal_uInt16 mnPageIndex;
908         const sal_Int32 mnGap;
909     };
910 
911     /** Print two slides to one printer page so that the resulting pages
912         form a booklet.
913     */
914     class BookletPrinterPage : public PrinterPage
915     {
916     public:
917         BookletPrinterPage (
918             const sal_uInt16 nFirstPageIndex,
919             const sal_uInt16 nSecondPageIndex,
920             const Point& rFirstOffset,
921             const Point& rSecondOffset,
922             const PageKind ePageKind,
923             const MapMode& rMapMode,
924             const bool bPrintMarkedOnly,
925             const sal_uLong nDrawMode,
926             const Orientation eOrientation,
927             const sal_uInt16 nPaperTray)
928             : PrinterPage(ePageKind, rMapMode, bPrintMarkedOnly, ::rtl::OUString(),
929                 Point(), nDrawMode, eOrientation, nPaperTray),
930               mnFirstPageIndex(nFirstPageIndex),
931               mnSecondPageIndex(nSecondPageIndex),
932               maFirstOffset(rFirstOffset),
933               maSecondOffset(rSecondOffset)
934         {
935         }
936 
937         virtual ~BookletPrinterPage (void) {}
938 
939         virtual void Print (
940             Printer& rPrinter,
941             SdDrawDocument& rDocument,
942             ViewShell& rViewShell,
943             View* pView,
944             DrawView& rPrintView,
945             const SetOfByte& rVisibleLayers,
946             const SetOfByte& rPrintableLayers) const
947         {
948             (void)rViewShell;
949             MapMode aMap (maMap);
950             SdPage* pPageToPrint = rDocument.GetSdPage(mnFirstPageIndex, mePageKind);
951             if (pPageToPrint)
952             {
953                 aMap.SetOrigin(maFirstOffset);
954                 rPrinter.SetMapMode(aMap);
955                 PrintPage(
956                     rPrinter,
957                     rPrintView,
958                     *pPageToPrint,
959                     pView,
960                     mbPrintMarkedOnly,
961                     rVisibleLayers,
962                     rPrintableLayers);
963             }
964 
965             pPageToPrint = rDocument.GetSdPage(mnSecondPageIndex, mePageKind);
966             if( pPageToPrint )
967             {
968                 aMap.SetOrigin(maSecondOffset);
969                 rPrinter.SetMapMode(aMap);
970                 PrintPage(
971                     rPrinter,
972                     rPrintView,
973                     *pPageToPrint,
974                     pView,
975                     mbPrintMarkedOnly,
976                     rVisibleLayers,
977                     rPrintableLayers);
978             }
979         }
980 
981     private:
982         const sal_uInt16 mnFirstPageIndex;
983         const sal_uInt16 mnSecondPageIndex;
984         const Point maFirstOffset;
985         const Point maSecondOffset;
986     };
987 
988 
989 
990 
991     /** One handout page displays one to nine slides.
992     */
993     class HandoutPrinterPage : public PrinterPage
994     {
995     public:
996         HandoutPrinterPage (
997             const sal_uInt16 nHandoutPageIndex,
998             const ::std::vector<sal_uInt16>& rPageIndices,
999             const MapMode& rMapMode,
1000             const ::rtl::OUString& rsPageString,
1001             const Point& rPageStringOffset,
1002             const sal_uLong nDrawMode,
1003             const Orientation eOrientation,
1004             const sal_uInt16 nPaperTray)
1005             : PrinterPage(PK_HANDOUT, rMapMode, false, rsPageString,
1006                 rPageStringOffset, nDrawMode, eOrientation, nPaperTray),
1007               mnHandoutPageIndex(nHandoutPageIndex),
1008               maPageIndices(rPageIndices)
1009         {
1010         }
1011 
1012         virtual void Print (
1013             Printer& rPrinter,
1014             SdDrawDocument& rDocument,
1015             ViewShell& rViewShell,
1016             View* pView,
1017             DrawView& rPrintView,
1018             const SetOfByte& rVisibleLayers,
1019             const SetOfByte& rPrintableLayers) const
1020         {
1021             SdPage& rHandoutPage (*rDocument.GetSdPage(0, PK_HANDOUT));
1022 
1023 			Reference< com::sun::star::beans::XPropertySet > xHandoutPage( rHandoutPage.getUnoPage(), UNO_QUERY );
1024 			const rtl::OUString sPageNumber( RTL_CONSTASCII_USTRINGPARAM( "Number" ) );
1025 
1026             // Collect the page objects of the handout master.
1027             std::vector<SdrPageObj*> aHandoutPageObjects;
1028             SdrObjListIter aShapeIter (rHandoutPage);
1029             while (aShapeIter.IsMore())
1030             {
1031                 SdrPageObj* pPageObj = dynamic_cast<SdrPageObj*>(aShapeIter.Next());
1032                 if (pPageObj)
1033                     aHandoutPageObjects.push_back(pPageObj);
1034             }
1035             if (aHandoutPageObjects.empty())
1036                 return;
1037 
1038             // Connect page objects with pages.
1039             std::vector<SdrPageObj*>::iterator aPageObjIter (aHandoutPageObjects.begin());
1040             for (std::vector<sal_uInt16>::const_iterator
1041                      iPageIndex(maPageIndices.begin()),
1042                      iEnd(maPageIndices.end());
1043                  iPageIndex!=iEnd && aPageObjIter!=aHandoutPageObjects.end();
1044                  ++iPageIndex)
1045             {
1046                 // Check if the page still exists.
1047                 if (*iPageIndex >= rDocument.GetSdPageCount(PK_STANDARD))
1048                     continue;
1049 
1050                 SdrPageObj* pPageObj = (*aPageObjIter++);
1051                 pPageObj->SetReferencedPage(rDocument.GetSdPage(*iPageIndex, PK_STANDARD));
1052             }
1053 
1054             // if there are more page objects than pages left, set the rest to invisible
1055             int nHangoverCount = 0;
1056             while (aPageObjIter != aHandoutPageObjects.end())
1057             {
1058                 (*aPageObjIter++)->SetReferencedPage(0L);
1059                 nHangoverCount++;
1060             }
1061 
1062             // Hide outlines for objects that have pages attached.
1063             if (nHangoverCount > 0)
1064             {
1065                 int nSkip = aHandoutPageObjects.size() - nHangoverCount;
1066                 aShapeIter.Reset();
1067                 while (aShapeIter.IsMore())
1068                 {
1069                     SdrPathObj* pPathObj = dynamic_cast<SdrPathObj*>(aShapeIter.Next());
1070                     if (pPathObj)
1071                     {
1072                         if (nSkip > 0)
1073                             --nSkip;
1074                         else
1075                             pPathObj->SetMergedItem(XLineStyleItem(XLINE_NONE));
1076                     }
1077                 }
1078             }
1079 
1080 			if( xHandoutPage.is() ) try
1081 			{
1082 				xHandoutPage->setPropertyValue( sPageNumber, Any( static_cast<sal_Int16>(mnHandoutPageIndex) ) );
1083 			}
1084 			catch( Exception& )
1085 			{
1086 			}
1087 			rViewShell.SetPrintedHandoutPageNum( mnHandoutPageIndex + 1 );
1088 
1089             MapMode aMap (rPrinter.GetMapMode());
1090             rPrinter.SetMapMode(maMap);
1091 
1092             PrintPage(
1093                 rPrinter,
1094                 rPrintView,
1095                 rHandoutPage,
1096                 pView,
1097                 false,
1098                 rVisibleLayers,
1099                 rPrintableLayers);
1100             PrintMessage(
1101                 rPrinter,
1102                 msPageString,
1103                 maPageStringOffset);
1104 
1105 			if( xHandoutPage.is() ) try
1106 			{
1107 				xHandoutPage->setPropertyValue( sPageNumber, Any( static_cast<sal_Int16>(0) ) );
1108 			}
1109 			catch( Exception& )
1110 			{
1111 			}
1112 			rViewShell.SetPrintedHandoutPageNum(1);
1113 
1114             // Restore outlines.
1115             if (nHangoverCount > 0)
1116             {
1117                 aShapeIter.Reset();
1118                 while (aShapeIter.IsMore())
1119                 {
1120                     SdrPathObj* pPathObj = dynamic_cast<SdrPathObj*>(aShapeIter.Next());
1121                     if (pPathObj != NULL)
1122                         pPathObj->SetMergedItem(XLineStyleItem(XLINE_SOLID));
1123                 }
1124             }
1125 
1126        }
1127 
1128     private:
1129         const sal_uInt16 mnHandoutPageIndex;
1130         const ::std::vector<sal_uInt16> maPageIndices;
1131     };
1132 
1133 
1134 
1135 
1136     /** The outline information (title, subtitle, outline objects) of the
1137         document.  There is no fixed mapping of slides to printer pages.
1138     */
1139     class OutlinerPrinterPage : public PrinterPage
1140     {
1141     public:
1142         OutlinerPrinterPage (
1143             OutlinerParaObject* pParaObject,
1144             const MapMode& rMapMode,
1145             const ::rtl::OUString& rsPageString,
1146             const Point& rPageStringOffset,
1147             const sal_uLong nDrawMode,
1148             const Orientation eOrientation,
1149             const sal_uInt16 nPaperTray)
1150             : PrinterPage(PK_HANDOUT, rMapMode, false, rsPageString,
1151                 rPageStringOffset, nDrawMode, eOrientation, nPaperTray),
1152               mpParaObject(pParaObject)
1153         {
1154         }
1155 
1156         ~OutlinerPrinterPage (void)
1157         {
1158             mpParaObject.reset();
1159         }
1160 
1161         virtual void Print (
1162             Printer& rPrinter,
1163             SdDrawDocument& rDocument,
1164             ViewShell& rViewShell,
1165             View* pView,
1166             DrawView& rPrintView,
1167             const SetOfByte& rVisibleLayers,
1168             const SetOfByte& rPrintableLayers) const
1169         {
1170             (void)rViewShell;
1171             (void)pView;
1172             (void)rPrintView;
1173             (void)rVisibleLayers;
1174             (void)rPrintableLayers;
1175 
1176             // Set up the printer.
1177             rPrinter.SetMapMode(maMap);
1178 
1179             // Get and set up the outliner.
1180             const Rectangle aOutRect (rPrinter.GetPageOffset(), rPrinter.GetOutputSize());
1181             Outliner* pOutliner = rDocument.GetInternalOutliner();
1182             const sal_uInt16 nSavedOutlMode (pOutliner->GetMode());
1183             const sal_Bool bSavedUpdateMode (pOutliner->GetUpdateMode());
1184             const Size aSavedPaperSize (pOutliner->GetPaperSize());
1185 
1186             pOutliner->Init(OUTLINERMODE_OUTLINEVIEW);
1187             pOutliner->SetPaperSize(aOutRect.GetSize());
1188             pOutliner->SetUpdateMode(sal_True);
1189             pOutliner->Clear();
1190             pOutliner->SetText(*mpParaObject);
1191 
1192             pOutliner->Draw(&rPrinter, aOutRect);
1193 
1194             PrintMessage(
1195                 rPrinter,
1196                 msPageString,
1197                 maPageStringOffset);
1198 
1199             // Restore outliner and printer.
1200             pOutliner->Clear();
1201             pOutliner->SetUpdateMode(bSavedUpdateMode);
1202             pOutliner->SetPaperSize(aSavedPaperSize);
1203             pOutliner->Init(nSavedOutlMode);
1204         }
1205 
1206     private:
1207         ::boost::scoped_ptr<OutlinerParaObject> mpParaObject;
1208     };
1209 }
1210 
1211 
1212 //===== DocumentRenderer::Implementation ======================================
1213 
1214 class DocumentRenderer::Implementation
1215     : public SfxListener,
1216       public vcl::PrinterOptionsHelper
1217 {
1218 public:
1219     Implementation (ViewShellBase& rBase)
1220         : mrBase(rBase),
1221           mbIsDisposed(false),
1222           mpPrinter(NULL),
1223           mpOptions(),
1224           maPrinterPages(),
1225           mpPrintView(),
1226           mbHasOrientationWarningBeenShown(false)
1227     {
1228         DialogCreator aCreator( mrBase.GetDocShell()->GetDocumentType() == DOCUMENT_TYPE_IMPRESS );
1229         m_aUIProperties = aCreator.GetDialogControls();
1230         maSlidesPerPage = aCreator.GetSlidesPerPage();
1231 
1232         StartListening(mrBase);
1233     }
1234 
1235 
1236 
1237 
1238     virtual ~Implementation (void)
1239     {
1240         EndListening(mrBase);
1241     }
1242 
1243 
1244 
1245 
1246     virtual void Notify (SfxBroadcaster& rBroadcaster, const SfxHint& rHint)
1247     {
1248         const SfxSimpleHint* pSimpleHint = dynamic_cast<const SfxSimpleHint*>(&rHint);
1249         if (pSimpleHint != NULL
1250             && pSimpleHint->GetId() == SFX_HINT_DYING
1251             && &rBroadcaster == &static_cast<SfxBroadcaster&>(mrBase))
1252         {
1253             Dispose();
1254         }
1255     }
1256 
1257 
1258 
1259     /** Process the sequence of properties given to one of the XRenderable
1260         methods.
1261     */
1262     void ProcessProperties (const css::uno::Sequence<css::beans::PropertyValue >& rOptions)
1263     {
1264         OSL_ASSERT(!mbIsDisposed);
1265         if (mbIsDisposed)
1266             return;
1267 
1268         bool bIsValueChanged = processProperties( rOptions );
1269         bool bIsPaperChanged = false;
1270 
1271         // The RenderDevice property is handled specially: its value is
1272         // stored in mpPrinter instead of being retrieved on demand.
1273         Any aDev( getValue( OUString( RTL_CONSTASCII_USTRINGPARAM( "RenderDevice" ) ) ) );
1274         Reference<awt::XDevice> xRenderDevice;
1275 
1276         if (aDev >>= xRenderDevice)
1277         {
1278             VCLXDevice* pDevice = VCLXDevice::GetImplementation(xRenderDevice);
1279             OutputDevice* pOut = pDevice ? pDevice->GetOutputDevice() : NULL;
1280             mpPrinter = dynamic_cast<Printer*>(pOut);
1281             Size aPageSizePixel = mpPrinter ? mpPrinter->GetPaperSizePixel() : Size();
1282             if( aPageSizePixel != maPrinterPageSizePixel )
1283             {
1284                 bIsPaperChanged = true;
1285                 maPrinterPageSizePixel = aPageSizePixel;
1286             }
1287         }
1288 
1289         if (bIsValueChanged)
1290         {
1291             if ( ! mpOptions )
1292                 mpOptions.reset(new PrintOptions(*this, maSlidesPerPage));
1293         }
1294         if( bIsValueChanged || bIsPaperChanged )
1295             PreparePages();
1296     }
1297 
1298 
1299 
1300     /** Return the number of pages that are to be printed.
1301     */
1302     sal_Int32 GetPrintPageCount (void)
1303     {
1304         OSL_ASSERT(!mbIsDisposed);
1305         if (mbIsDisposed)
1306             return 0;
1307         else
1308             return maPrinterPages.size();
1309     }
1310 
1311 
1312 
1313     /** Return a sequence of properties that can be returned by the
1314         XRenderable::getRenderer() method.
1315     */
1316     css::uno::Sequence<css::beans::PropertyValue> GetProperties (
1317         const css::uno::Sequence<css::beans::PropertyValue>& rOptions)
1318     {
1319         (void)rOptions;
1320 
1321         css::uno::Sequence<css::beans::PropertyValue> aProperties (3);
1322 
1323         aProperties[0].Name = A2S("ExtraPrintUIOptions");
1324         aProperties[0].Value <<= m_aUIProperties;
1325 
1326         aProperties[1].Name = A2S("PageSize");
1327         aProperties[1].Value <<= maPrintSize;
1328 
1329         // FIXME: is this always true ?
1330         aProperties[2].Name = A2S("PageIncludesNonprintableArea");
1331         aProperties[2].Value = makeAny( sal_True );
1332 
1333         return aProperties;
1334     }
1335 
1336 
1337 
1338 
1339     /** Print one of the prepared pages.
1340     */
1341     void PrintPage (const sal_Int32 nIndex)
1342     {
1343         OSL_ASSERT(!mbIsDisposed);
1344         if (mbIsDisposed)
1345             return;
1346 
1347         Printer& rPrinter (*mpPrinter);
1348 
1349         ::boost::shared_ptr<ViewShell> pViewShell (mrBase.GetMainViewShell());
1350         if ( ! pViewShell)
1351             return;
1352 
1353         SdDrawDocument* pDocument = pViewShell->GetDoc();
1354         OSL_ASSERT(pDocument!=NULL);
1355 
1356         ::boost::shared_ptr<DrawViewShell> pDrawViewShell(
1357             ::boost::dynamic_pointer_cast<DrawViewShell>(mrBase.GetMainViewShell()));
1358 
1359         if ( ! mpPrintView)
1360             mpPrintView.reset(new DrawView(mrBase.GetDocShell(), &rPrinter, pDrawViewShell.get()));
1361 
1362         if (nIndex<0 || sal::static_int_cast<sal_uInt32>(nIndex)>=maPrinterPages.size())
1363             return;
1364 
1365         const ::boost::shared_ptr<PrinterPage> pPage (maPrinterPages[nIndex]);
1366         OSL_ASSERT(pPage);
1367         if ( ! pPage)
1368             return;
1369 
1370         const Orientation eSavedOrientation (rPrinter.GetOrientation());
1371         const sal_uLong nSavedDrawMode (rPrinter.GetDrawMode());
1372         const MapMode aSavedMapMode (rPrinter.GetMapMode());
1373         const sal_uInt16 nSavedPaperBin (rPrinter.GetPaperBin());
1374 
1375 
1376         // Set page orientation.
1377         if ( ! rPrinter.SetOrientation(pPage->GetOrientation()))
1378         {
1379             if ( ! mbHasOrientationWarningBeenShown
1380                 && mpOptions->IsWarningOrientation())
1381             {
1382                 mbHasOrientationWarningBeenShown = true;
1383                 // Show warning that the orientation could not be set.
1384                 if (pViewShell)
1385                 {
1386                     WarningBox aWarnBox(
1387                         pViewShell->GetActiveWindow(),
1388                         (WinBits)(WB_OK_CANCEL | WB_DEF_CANCEL),
1389                         String(SdResId(STR_WARN_PRINTFORMAT_FAILURE)));
1390                     if (aWarnBox.Execute() != RET_OK)
1391                         return;
1392                 }
1393             }
1394         }
1395 
1396         // Set the draw mode.
1397         rPrinter.SetDrawMode(pPage->GetDrawMode());
1398 
1399         // Set paper tray.
1400         rPrinter.SetPaperBin(pPage->GetPaperTray());
1401 
1402         // Print the actual page.
1403         pPage->Print(
1404             rPrinter,
1405             *pDocument,
1406             *pViewShell,
1407             pDrawViewShell ? pDrawViewShell->GetView() : NULL,
1408             *mpPrintView,
1409             pViewShell->GetFrameView()->GetVisibleLayers(),
1410             pViewShell->GetFrameView()->GetPrintableLayers());
1411 
1412         rPrinter.SetOrientation(eSavedOrientation);
1413         rPrinter.SetDrawMode(nSavedDrawMode);
1414         rPrinter.SetMapMode(aSavedMapMode);
1415         rPrinter.SetPaperBin(nSavedPaperBin);
1416     }
1417 
1418 
1419 
1420 
1421 private:
1422     ViewShellBase& mrBase;
1423     bool mbIsDisposed;
1424     Printer* mpPrinter;
1425     Size maPrinterPageSizePixel;
1426     ::boost::scoped_ptr<PrintOptions> mpOptions;
1427     ::std::vector< ::boost::shared_ptr< ::sd::PrinterPage> > maPrinterPages;
1428     ::boost::scoped_ptr<DrawView> mpPrintView;
1429     bool mbHasOrientationWarningBeenShown;
1430     ::std::vector<sal_Int32> maSlidesPerPage;
1431     awt::Size maPrintSize;
1432 
1433     void Dispose (void)
1434     {
1435         mbIsDisposed = true;
1436     }
1437 
1438 
1439 
1440     /** Determine and set the paper orientation.
1441     */
1442     bool SetupPaperOrientation (
1443         const PageKind ePageKind,
1444         PrintInfo& rInfo)
1445     {
1446         SdDrawDocument* pDocument = mrBase.GetMainViewShell()->GetDoc();
1447         rInfo.meOrientation = ORIENTATION_PORTRAIT;
1448 
1449         if( ! mpOptions->IsBooklet())
1450         {
1451             rInfo.meOrientation = pDocument->GetSdPage(0, ePageKind)->GetOrientation();
1452         }
1453         else if (rInfo.maPageSize.Width() < rInfo.maPageSize.Height())
1454             rInfo.meOrientation = ORIENTATION_LANDSCAPE;
1455 
1456         const Size aPaperSize (rInfo.mpPrinter->GetPaperSize());
1457         if( (rInfo.meOrientation == ORIENTATION_LANDSCAPE &&
1458               (aPaperSize.Width() < aPaperSize.Height()))
1459            ||
1460             (rInfo.meOrientation == ORIENTATION_PORTRAIT &&
1461               (aPaperSize.Width() > aPaperSize.Height()))
1462           )
1463         {
1464             maPrintSize = awt::Size(aPaperSize.Height(), aPaperSize.Width());
1465             //            rInfo.maPrintSize = Size(rInfo.maPrintSize.Height(), rInfo.maPrintSize.Width());
1466         }
1467         else
1468         {
1469             maPrintSize = awt::Size(aPaperSize.Width(), aPaperSize.Height());
1470         }
1471 
1472         return true;
1473     }
1474 
1475 
1476 
1477     /** Top most method for preparing printer pages.  In this and the other
1478         Prepare... methods the various special cases are detected and
1479         handled.
1480         For every page that is to be printed (that may contain several
1481         slides) one PrinterPage object is created and inserted into
1482         maPrinterPages.
1483     */
1484     void PreparePages (void)
1485     {
1486         mpPrintView.reset();
1487         maPrinterPages.clear();
1488         mbHasOrientationWarningBeenShown = false;
1489 
1490         ViewShell* pShell = mrBase.GetMainViewShell().get();
1491 
1492         PrintInfo aInfo (mpPrinter, mpOptions->GetPrinterSelection(), mrBase.GetMainViewShell());
1493 
1494         if (aInfo.mpPrinter!=NULL && pShell!=NULL)
1495         {
1496 
1497             MapMode aMap (aInfo.mpPrinter->GetMapMode());
1498             aMap.SetMapUnit(MAP_100TH_MM);
1499             aInfo.maMap = aMap;
1500             mpPrinter->SetMapMode(aMap);
1501 
1502             ::Outliner& rOutliner = mrBase.GetDocument()->GetDrawOutliner();
1503             const sal_uLong nSavedControlWord (rOutliner.GetControlWord());
1504             sal_uLong nCntrl = nSavedControlWord;
1505             nCntrl &= ~EE_CNTRL_MARKFIELDS;
1506             nCntrl &= ~EE_CNTRL_ONLINESPELLING;
1507             rOutliner.SetControlWord( nCntrl );
1508 
1509             // When in outline view then apply all pending changes to the model.
1510             if (pShell->ISA(OutlineViewShell))
1511                 static_cast<OutlineViewShell*>(pShell)->PrepareClose (sal_False, sal_False);
1512 
1513             // Collect some frequently used data.
1514             if (mpOptions->IsDate())
1515             {
1516                 aInfo.msTimeDate += GetSdrGlobalData().GetLocaleData()->getDate( Date() );
1517                 aInfo.msTimeDate += ::rtl::OUString((sal_Unicode)' ');
1518             }
1519 
1520             if (mpOptions->IsTime())
1521                 aInfo.msTimeDate += GetSdrGlobalData().GetLocaleData()->getTime( Time(), sal_False, sal_False );
1522             aInfo.maPrintSize = aInfo.mpPrinter->GetOutputSize();
1523             maPrintSize = awt::Size(
1524                 aInfo.mpPrinter->GetPaperSize().Width(),
1525                 aInfo.mpPrinter->GetPaperSize().Height());
1526 
1527             switch (mpOptions->GetOutputQuality())
1528             {
1529                 case 1:
1530                     aInfo.mnDrawMode = DRAWMODE_GRAYLINE | DRAWMODE_GRAYFILL
1531                         | DRAWMODE_BLACKTEXT | DRAWMODE_GRAYBITMAP
1532                         | DRAWMODE_GRAYGRADIENT;
1533                     break;
1534 
1535                 case 2:
1536                     aInfo.mnDrawMode = DRAWMODE_BLACKLINE | DRAWMODE_BLACKTEXT
1537                         | DRAWMODE_WHITEFILL | DRAWMODE_GRAYBITMAP
1538                         | DRAWMODE_WHITEGRADIENT;
1539                     break;
1540 
1541                 default:
1542                     aInfo.mnDrawMode = DRAWMODE_DEFAULT;
1543             }
1544 
1545             // check if selected range of pages contains transparent objects
1546             /*
1547             const bool bPrintPages (bPrintNotes || bPrintDraw || bPrintHandout);
1548             const bool bContainsTransparency (bPrintPages && ContainsTransparency());
1549             if (pPrinter->InitJob (mrBase.GetWindow(), !bIsAPI && bContainsTransparency))
1550             */
1551 
1552             if (mpOptions->IsDraw())
1553                 PrepareStdOrNotes(PK_STANDARD, aInfo);
1554             if (mpOptions->IsNotes())
1555                 PrepareStdOrNotes(PK_NOTES, aInfo);
1556 			if (mpOptions->IsHandout())
1557             {
1558 				InitHandoutTemplate();
1559                 PrepareHandout(aInfo);
1560             }
1561             if (mpOptions->IsOutline())
1562                 PrepareOutline(aInfo);
1563 
1564             rOutliner.SetControlWord(nSavedControlWord);
1565         }
1566     }
1567 
1568 
1569 
1570 
1571     /** Create the page objects of the handout template.  When the actual
1572         printing takes place then the page objects are assigned different
1573         sets of slides for each printed page (see HandoutPrinterPage::Print).
1574     */
1575     void InitHandoutTemplate (void)
1576     {
1577         const sal_Int32 nSlidesPerHandout (mpOptions->GetHandoutPageCount());
1578         const bool bHandoutHorizontal (mpOptions->IsHandoutHorizontal());
1579 
1580         AutoLayout eLayout = AUTOLAYOUT_HANDOUT6;
1581         switch (nSlidesPerHandout)
1582         {
1583             case 0: eLayout = AUTOLAYOUT_NONE; break; // AUTOLAYOUT_HANDOUT1; break;
1584             case 1: eLayout = AUTOLAYOUT_HANDOUT1; break;
1585             case 2: eLayout = AUTOLAYOUT_HANDOUT2; break;
1586             case 3: eLayout = AUTOLAYOUT_HANDOUT3; break;
1587             case 4: eLayout = AUTOLAYOUT_HANDOUT4; break;
1588             default:
1589             case 6: eLayout = AUTOLAYOUT_HANDOUT6; break;
1590             case 9: eLayout = AUTOLAYOUT_HANDOUT9; break;
1591         }
1592 
1593         if( !mrBase.GetDocument() )
1594             return;
1595 
1596         SdDrawDocument& rModel = *mrBase.GetDocument();
1597 
1598         // first, prepare handout page (not handout master)
1599 
1600         SdPage* pHandout = rModel.GetSdPage(0, PK_HANDOUT);
1601         if( !pHandout )
1602             return;
1603 
1604         // delete all previous shapes from handout page
1605         while( pHandout->GetObjCount() )
1606         {
1607             SdrObject* pObj = pHandout->NbcRemoveObject(0);
1608             if( pObj )
1609                 SdrObject::Free( pObj  );
1610         }
1611 
1612         const bool bDrawLines (eLayout == AUTOLAYOUT_HANDOUT3);
1613 
1614         std::vector< Rectangle > aAreas;
1615         SdPage::CalculateHandoutAreas( rModel, eLayout, bHandoutHorizontal, aAreas );
1616 
1617         std::vector< Rectangle >::iterator iter( aAreas.begin() );
1618         while( iter != aAreas.end() )
1619         {
1620             pHandout->NbcInsertObject( new SdrPageObj((*iter++)) );
1621 
1622             if( bDrawLines && (iter != aAreas.end())  )
1623             {
1624                 Rectangle aRect( (*iter++) );
1625 
1626                 basegfx::B2DPolygon aPoly;
1627                 aPoly.insert(0, basegfx::B2DPoint( aRect.Left(), aRect.Top() ) );
1628                 aPoly.insert(1, basegfx::B2DPoint( aRect.Right(), aRect.Top() ) );
1629 
1630                 basegfx::B2DHomMatrix aMatrix;
1631                 aMatrix.translate( 0.0, static_cast< double >( aRect.GetHeight() / 7 ) );
1632 
1633                 basegfx::B2DPolyPolygon aPathPoly;
1634                 for( sal_uInt16 nLine = 0; nLine < 7; nLine++ )
1635                 {
1636                     aPoly.transform( aMatrix );
1637                     aPathPoly.append( aPoly );
1638                 }
1639 
1640                 SdrPathObj* pPathObj = new SdrPathObj(OBJ_PATHLINE, aPathPoly );
1641                 pPathObj->SetMergedItem(XLineStyleItem(XLINE_SOLID));
1642                 pPathObj->SetMergedItem(XLineColorItem(String(), Color(COL_BLACK)));
1643 
1644                 pHandout->NbcInsertObject( pPathObj );
1645             }
1646         }
1647     }
1648 
1649 
1650 
1651 
1652     /** Detect whether any of the slides that are to be printed contains
1653         partially transparent or translucent shapes.
1654     */
1655     bool ContainsTransparency (const PrintInfo& rInfo) const
1656     {
1657         // const bool bPrintExcluded (mpOptions->IsPrintExcluded());
1658         bool bContainsTransparency = false;
1659 
1660         for (sal_uInt16
1661                  nIndex=0,
1662                  nCount=mrBase.GetDocument()->GetSdPageCount(PK_STANDARD);
1663              nIndex < nCount && !bContainsTransparency;
1664              ++nIndex)
1665         {
1666             SdPage* pPage = GetFilteredPage(nIndex, PK_STANDARD, rInfo);
1667             if (pPage == NULL)
1668                 continue;
1669 
1670             bContainsTransparency = pPage->HasTransparentObjects();
1671             if ( ! bContainsTransparency && pPage->TRG_HasMasterPage())
1672                 bContainsTransparency = pPage->TRG_GetMasterPage().HasTransparentObjects();
1673         }
1674 
1675         return bContainsTransparency;
1676     }
1677 
1678 
1679 
1680 
1681     /** Detect whether the specified slide is to be printed.
1682         @return
1683             When the slide is not to be printed then <NULL/> is returned.
1684             Otherwise a pointer to the slide is returned.
1685     */
1686     SdPage* GetFilteredPage (
1687         const sal_Int32 nPageIndex,
1688         const PageKind ePageKind,
1689         const PrintInfo& rInfo) const
1690     {
1691         OSL_ASSERT(mrBase.GetDocument() != NULL);
1692         OSL_ASSERT(nPageIndex>=0);
1693         if ( ! rInfo.maSelection.IsSelected(nPageIndex))
1694             return NULL;
1695         SdPage* pPage = mrBase.GetDocument()->GetSdPage(
1696             sal::static_int_cast<sal_uInt16>(nPageIndex),
1697             ePageKind);
1698         if (pPage == NULL)
1699             return NULL;
1700         if ( ! pPage->IsExcluded() || mpOptions->IsPrintExcluded())
1701             return pPage;
1702         else
1703             return NULL;
1704     }
1705 
1706 
1707 
1708 
1709     /** Prepare the outline of the document for printing.  There is no fixed
1710         number of slides whose outline data is put onto one printer page.
1711         If the current printer page has enough room for the outline of the
1712         current slide then that is added.  Otherwise a new printer page is
1713         started.
1714     */
1715     void PrepareOutline (PrintInfo& rInfo)
1716     {
1717         MapMode aMap (rInfo.maMap);
1718         Point aPageOfs (rInfo.mpPrinter->GetPageOffset() );
1719         // aMap.SetOrigin(Point() - aPageOfs);
1720         aMap.SetScaleX(Fraction(1,2));
1721         aMap.SetScaleY(Fraction(1,2));
1722         mpPrinter->SetMapMode(aMap);
1723 
1724         Rectangle aOutRect(aPageOfs, rInfo.mpPrinter->GetOutputSize());
1725         if( aOutRect.GetWidth() > aOutRect.GetHeight() )
1726         {
1727             Size aPaperSize( rInfo.mpPrinter->PixelToLogic( rInfo.mpPrinter->GetPaperSizePixel(), MapMode( MAP_100TH_MM ) ) );
1728             maPrintSize.Width  = aPaperSize.Height();
1729             maPrintSize.Height = aPaperSize.Width();
1730             aOutRect = Rectangle( Point( aPageOfs.Y(), aPageOfs.X() ),
1731                                   Size( aOutRect.GetHeight(), aOutRect.GetWidth() ) );
1732         }
1733 
1734         Link aOldLink;
1735         Outliner* pOutliner = mrBase.GetDocument()->GetInternalOutliner();
1736         pOutliner->Init(OUTLINERMODE_OUTLINEVIEW);
1737         const sal_uInt16 nSavedOutlMode (pOutliner->GetMode());
1738         const sal_Bool bSavedUpdateMode (pOutliner->GetUpdateMode());
1739         const Size aSavedPaperSize (pOutliner->GetPaperSize());
1740         const MapMode aSavedMapMode (pOutliner->GetRefMapMode());
1741         pOutliner->SetPaperSize(aOutRect.GetSize());
1742         pOutliner->SetUpdateMode(sal_True);
1743 
1744         long nPageH = aOutRect.GetHeight();
1745 
1746         for (sal_uInt16
1747                  nIndex=0,
1748                  nCount=mrBase.GetDocument()->GetSdPageCount(PK_STANDARD);
1749              nIndex < nCount;
1750              )
1751         {
1752             pOutliner->Clear();
1753             pOutliner->SetFirstPageNumber(nIndex+1);
1754 
1755             Paragraph* pPara = NULL;
1756             sal_Int32 nH (0);
1757             while (nH < nPageH && nIndex<nCount)
1758             {
1759                 SdPage* pPage = GetFilteredPage(nIndex, PK_STANDARD, rInfo);
1760                 ++nIndex;
1761                 if (pPage == NULL)
1762                     continue;
1763 
1764                 SdrTextObj* pTextObj = NULL;
1765                 sal_uInt32 nObj (0);
1766 
1767                 while (pTextObj==NULL && nObj < pPage->GetObjCount())
1768                 {
1769                     SdrObject* pObj = pPage->GetObj(nObj++);
1770                     if (pObj->GetObjInventor() == SdrInventor
1771                         && pObj->GetObjIdentifier() == OBJ_TITLETEXT)
1772                     {
1773                         pTextObj = dynamic_cast<SdrTextObj*>(pObj);
1774                     }
1775                 }
1776 
1777                 pPara = pOutliner->GetParagraph(pOutliner->GetParagraphCount() - 1);
1778 
1779                 if (pTextObj!=NULL
1780                     && !pTextObj->IsEmptyPresObj()
1781                     && pTextObj->GetOutlinerParaObject())
1782                 {
1783                     pOutliner->AddText(*(pTextObj->GetOutlinerParaObject()));
1784                 }
1785                 else
1786                     pOutliner->Insert(String());
1787 
1788                 pTextObj = NULL;
1789                 nObj = 0;
1790 
1791                 while (pTextObj==NULL && nObj<pPage->GetObjCount())
1792                 {
1793                     SdrObject* pObj = pPage->GetObj(nObj++);
1794                     if (pObj->GetObjInventor() == SdrInventor
1795                         && pObj->GetObjIdentifier() == OBJ_OUTLINETEXT)
1796                     {
1797                         pTextObj = dynamic_cast<SdrTextObj*>(pObj);
1798                     }
1799                 }
1800 
1801                 bool bSubTitle (false);
1802                 if (!pTextObj)
1803                 {
1804                     bSubTitle = true;
1805                     pTextObj = dynamic_cast<SdrTextObj*>(pPage->GetPresObj(PRESOBJ_TEXT));  // Untertitel vorhanden?
1806                 }
1807 
1808                 sal_uLong nParaCount1 = pOutliner->GetParagraphCount();
1809 
1810                 if (pTextObj!=NULL
1811                     && !pTextObj->IsEmptyPresObj()
1812                     && pTextObj->GetOutlinerParaObject())
1813                 {
1814                     pOutliner->AddText(*(pTextObj->GetOutlinerParaObject()));
1815                 }
1816 
1817                 if (bSubTitle )
1818                 {
1819                     const sal_Int32 nParaCount2 (pOutliner->GetParagraphCount());
1820                     for (sal_Int32 nPara=nParaCount1; nPara<nParaCount2; ++nPara)
1821                     {
1822                         Paragraph* pP = pOutliner->GetParagraph(nPara);
1823                         if (pP!=NULL && pOutliner->GetDepth((sal_uInt16)nPara) > 0)
1824                             pOutliner->SetDepth(pP, 0);
1825                     }
1826                 }
1827 
1828                 nH = pOutliner->GetTextHeight();
1829             }
1830 
1831             // Remove the last paragraph when that does not fit completely on
1832             // the current page.
1833             if (nH > nPageH && pPara!=NULL)
1834             {
1835                 sal_uLong nCnt = pOutliner->GetAbsPos(
1836                     pOutliner->GetParagraph( pOutliner->GetParagraphCount() - 1 ) );
1837                 sal_uLong nParaPos = pOutliner->GetAbsPos( pPara );
1838                 nCnt -= nParaPos;
1839                 pPara = pOutliner->GetParagraph( ++nParaPos );
1840                 if ( nCnt && pPara )
1841                 {
1842                     pOutliner->Remove(pPara, nCnt);
1843                     --nIndex;
1844                 }
1845             }
1846 
1847             maPrinterPages.push_back(
1848                 ::boost::shared_ptr<PrinterPage>(
1849                     new OutlinerPrinterPage(
1850                         pOutliner->CreateParaObject(),
1851                         aMap,
1852                         rInfo.msTimeDate,
1853                         aPageOfs,
1854                         rInfo.mnDrawMode,
1855                         rInfo.meOrientation,
1856                         rInfo.mpPrinter->GetPaperBin())));
1857         }
1858 
1859         pOutliner->SetRefMapMode(aSavedMapMode);
1860         pOutliner->SetUpdateMode(bSavedUpdateMode);
1861         pOutliner->SetPaperSize(aSavedPaperSize);
1862         pOutliner->Init(nSavedOutlMode);
1863     }
1864 
1865 
1866 
1867 
1868     /** Prepare handout pages for slides that are to be printed.
1869     */
1870     void PrepareHandout (PrintInfo& rInfo)
1871     {
1872         SdDrawDocument* pDocument = mrBase.GetDocument();
1873         OSL_ASSERT(pDocument != NULL);
1874         SdPage& rHandoutPage (*pDocument->GetSdPage(0, PK_HANDOUT));
1875 
1876         const bool bScalePage (mpOptions->IsPageSize());
1877 
1878         sal_uInt16 nPaperBin;
1879         if ( ! mpOptions->IsPaperBin())
1880             nPaperBin = rHandoutPage.GetPaperBin();
1881         else
1882             nPaperBin = rInfo.mpPrinter->GetPaperBin();
1883 
1884         // Change orientation?
1885         SdPage& rMaster (dynamic_cast<SdPage&>(rHandoutPage.TRG_GetMasterPage()));
1886         rInfo.meOrientation = rMaster.GetOrientation();
1887 
1888         const Size aPaperSize (rInfo.mpPrinter->GetPaperSize());
1889         if( (rInfo.meOrientation == ORIENTATION_LANDSCAPE &&
1890               (aPaperSize.Width() < aPaperSize.Height()))
1891            ||
1892             (rInfo.meOrientation == ORIENTATION_PORTRAIT &&
1893               (aPaperSize.Width() > aPaperSize.Height()))
1894           )
1895         {
1896             maPrintSize = awt::Size(aPaperSize.Height(), aPaperSize.Width());
1897         }
1898         else
1899         {
1900             maPrintSize = awt::Size(aPaperSize.Width(), aPaperSize.Height());
1901         }
1902 
1903         MapMode aMap (rInfo.maMap);
1904         const Point aPageOfs (rInfo.mpPrinter->GetPageOffset());
1905         //DrawView* pPrintView;
1906 
1907         // aMap.SetOrigin(Point() - aPageOfs);
1908 
1909         if ( bScalePage )
1910         {
1911             const Size aPageSize (rHandoutPage.GetSize());
1912             const Size aPrintSize (rInfo.mpPrinter->GetOutputSize());
1913 
1914             const double fHorz = (double) aPrintSize.Width()	/ aPageSize.Width();
1915             const double fVert = (double) aPrintSize.Height() / aPageSize.Height();
1916 
1917             Fraction	aFract;
1918             if ( fHorz < fVert )
1919                 aFract = Fraction(aPrintSize.Width(), aPageSize.Width());
1920             else
1921                 aFract = Fraction(aPrintSize.Height(), aPageSize.Height());
1922 
1923             aMap.SetScaleX(aFract);
1924             aMap.SetScaleY(aFract);
1925             aMap.SetOrigin(Point());
1926         }
1927 
1928         ::boost::shared_ptr<ViewShell> pViewShell (mrBase.GetMainViewShell());
1929         pViewShell->WriteFrameViewData();
1930 
1931         // Count page shapes.
1932         sal_uInt32 nShapeCount (0);
1933         SdrObjListIter aShapeIter (rHandoutPage);
1934         while (aShapeIter.IsMore())
1935         {
1936             SdrPageObj* pPageObj = dynamic_cast<SdrPageObj*>(aShapeIter.Next());
1937             if (pPageObj)
1938                 ++nShapeCount;
1939         }
1940 
1941 		const sal_uInt16 nPageCount = mrBase.GetDocument()->GetSdPageCount(PK_STANDARD);
1942 		const sal_uInt16 nHandoutPageCount = nShapeCount ? (nPageCount + nShapeCount - 1) / nShapeCount : 0;
1943 		pViewShell->SetPrintedHandoutPageCount( nHandoutPageCount );
1944 		mrBase.GetDocument()->setHandoutPageCount( nHandoutPageCount );
1945 
1946         // Distribute pages to handout pages.
1947         ::std::vector<sal_uInt16> aPageIndices;
1948         for (sal_uInt16
1949                  nIndex=0,
1950                  nCount= nPageCount,
1951                  nHandoutPageIndex=0;
1952              nIndex <= nCount;
1953              ++nIndex)
1954         {
1955             if (nIndex < nCount)
1956             {
1957                 if (GetFilteredPage(nIndex, PK_STANDARD, rInfo) == NULL)
1958                     continue;
1959                 aPageIndices.push_back(nIndex);
1960             }
1961 
1962             // Create a printer page when we have found one page for each
1963             // placeholder or when this is the last (and special) loop.
1964             if (!aPageIndices.empty() && (aPageIndices.size() == nShapeCount || nIndex==nCount))
1965             {
1966                 maPrinterPages.push_back(
1967                     ::boost::shared_ptr<PrinterPage>(
1968                         new HandoutPrinterPage(
1969                             nHandoutPageIndex++,
1970                             aPageIndices,
1971                             aMap,
1972                             rInfo.msTimeDate,
1973                             aPageOfs,
1974                             rInfo.mnDrawMode,
1975                             rInfo.meOrientation,
1976                             nPaperBin)));
1977                 aPageIndices.clear();
1978             }
1979         }
1980 	}
1981 
1982 
1983 
1984 
1985     /** Prepare the notes pages or regular slides.
1986     */
1987     void PrepareStdOrNotes (
1988         const PageKind ePageKind,
1989         PrintInfo& rInfo)
1990     {
1991         OSL_ASSERT(rInfo.mpPrinter != NULL);
1992 
1993         // Fill in page kind specific data.
1994         SdDrawDocument* pDocument = mrBase.GetMainViewShell()->GetDoc();
1995         if (pDocument->GetSdPageCount(ePageKind) == 0)
1996             return;
1997         SdPage* pRefPage = pDocument->GetSdPage(0, ePageKind);
1998         rInfo.maPageSize = pRefPage->GetSize();
1999 
2000         if ( ! SetupPaperOrientation(ePageKind, rInfo))
2001             return;
2002 
2003         MapMode aMap (rInfo.maMap);
2004         // aMap.SetOrigin(Point() - rInfo.mpPrinter->GetPageOffset());
2005         rInfo.maMap = aMap;
2006 
2007         if (mpOptions->IsBooklet())
2008             PrepareBooklet(ePageKind, rInfo);
2009         else
2010             PrepareRegularPages(ePageKind, rInfo);
2011     }
2012 
2013 
2014 
2015 
2016     /** Prepare slides in a non-booklet way: one slide per one to many
2017         printer pages.
2018     */
2019     void PrepareRegularPages (
2020         const PageKind ePageKind,
2021         PrintInfo& rInfo)
2022     {
2023         ::boost::shared_ptr<ViewShell> pViewShell (mrBase.GetMainViewShell());
2024         pViewShell->WriteFrameViewData();
2025         Point aPtZero;
2026 
2027         for (sal_uInt16
2028                  nIndex=0,
2029                  nCount=mrBase.GetDocument()->GetSdPageCount(PK_STANDARD);
2030              nIndex < nCount;
2031              ++nIndex)
2032         {
2033             SdPage* pPage = GetFilteredPage(nIndex, ePageKind, rInfo);
2034             if (pPage == NULL)
2035                 continue;
2036 
2037             MapMode aMap (rInfo.maMap);
2038             // Kann sich die Seitengroesse geaendert haben?
2039             const Size aPageSize = pPage->GetSize();
2040 
2041             if (mpOptions->IsPageSize())
2042             {
2043                 const double fHorz ((double) rInfo.maPrintSize.Width()	/ aPageSize.Width());
2044                 const double fVert ((double) rInfo.maPrintSize.Height() / aPageSize.Height());
2045 
2046                 Fraction aFract;
2047                 if (fHorz < fVert)
2048                     aFract = Fraction(rInfo.maPrintSize.Width(), aPageSize.Width());
2049                 else
2050                     aFract = Fraction(rInfo.maPrintSize.Height(), aPageSize.Height());
2051 
2052                 aMap.SetScaleX(aFract);
2053                 aMap.SetScaleY(aFract);
2054                 aMap.SetOrigin(Point());
2055             }
2056 
2057             if (mpOptions->IsPrintPageName())
2058             {
2059                 rInfo.msPageString = pPage->GetName();
2060                 rInfo.msPageString += ::rtl::OUString(sal_Unicode(' '));
2061             }
2062             else
2063                 rInfo.msPageString = ::rtl::OUString();
2064             rInfo.msPageString += rInfo.msTimeDate;
2065 
2066             long aPageWidth   = aPageSize.Width() - pPage->GetLftBorder() - pPage->GetRgtBorder();
2067             long aPageHeight  = aPageSize.Height() - pPage->GetUppBorder() - pPage->GetLwrBorder();
2068             // Bugfix zu 44530:
2069             // Falls implizit umgestellt wurde (Landscape/Portrait)
2070             // wird dies beim Kacheln, bzw. aufteilen (Poster) beruecksichtigt
2071             sal_Bool bSwitchPageSize = sal_False;
2072             if( ( rInfo.maPrintSize.Width() > rInfo.maPrintSize.Height()
2073                     && aPageWidth < aPageHeight )
2074                 || ( rInfo.maPrintSize.Width() < rInfo.maPrintSize.Height()
2075                     && aPageWidth > aPageHeight ) )
2076             {
2077                 bSwitchPageSize = sal_True;
2078                 const sal_Int32 nTmp (rInfo.maPrintSize.Width());
2079                 rInfo.maPrintSize.Width() = rInfo.maPrintSize.Height();
2080                 rInfo.maPrintSize.Height() = nTmp;
2081             }
2082 
2083             if (mpOptions->IsTilePage()
2084                 && aPageWidth < rInfo.maPrintSize.Width()
2085                 && aPageHeight < rInfo.maPrintSize.Height())
2086             {
2087                 // Put multiple slides on one printer page.
2088                 PrepareTiledPage(nIndex, *pPage, ePageKind, rInfo);
2089             }
2090             else
2091             {
2092                 rInfo.maMap = aMap;
2093                 PrepareScaledPage(nIndex, *pPage, ePageKind, rInfo);
2094             }
2095         }
2096     }
2097 
2098 
2099 
2100 
2101     /** Put two slides on one printer page.
2102     */
2103     void PrepareBooklet (
2104         const PageKind ePageKind,
2105         const PrintInfo& rInfo)
2106     {
2107         MapMode aStdMap (rInfo.maMap);
2108         Point aOffset;
2109         Size aPrintSize_2 (rInfo.maPrintSize);
2110         Size aPageSize_2 (rInfo.maPageSize);
2111 
2112         if (rInfo.meOrientation == ORIENTATION_LANDSCAPE)
2113             aPrintSize_2.Width() >>= 1;
2114         else
2115             aPrintSize_2.Height() >>= 1;
2116 
2117         const double fPageWH = (double) aPageSize_2.Width() / aPageSize_2.Height();
2118         const double fPrintWH = (double) aPrintSize_2.Width() / aPrintSize_2.Height();
2119 
2120         if( fPageWH < fPrintWH )
2121         {
2122             aPageSize_2.Width() = (long) ( aPrintSize_2.Height() * fPageWH );
2123             aPageSize_2.Height()= aPrintSize_2.Height();
2124         }
2125         else
2126         {
2127             aPageSize_2.Width() = aPrintSize_2.Width();
2128             aPageSize_2.Height() = (long) ( aPrintSize_2.Width() / fPageWH );
2129         }
2130 
2131         MapMode aMap (rInfo.maMap);
2132         aMap.SetScaleX( Fraction( aPageSize_2.Width(), rInfo.maPageSize.Width() ) );
2133         aMap.SetScaleY( Fraction( aPageSize_2.Height(), rInfo.maPageSize.Height() ) );
2134 
2135         // calculate adjusted print size
2136         const Size aAdjustedPrintSize (OutputDevice::LogicToLogic(
2137             rInfo.maPrintSize,
2138             aStdMap,
2139             aMap));
2140 
2141         if (rInfo.meOrientation == ORIENTATION_LANDSCAPE)
2142         {
2143             aOffset.X() = ( ( aAdjustedPrintSize.Width() >> 1 ) - rInfo.maPageSize.Width() ) >> 1;
2144             aOffset.Y() = ( aAdjustedPrintSize.Height() - rInfo.maPageSize.Height() ) >> 1;
2145         }
2146         else
2147         {
2148             aOffset.X() = ( aAdjustedPrintSize.Width() - rInfo.maPageSize.Width() ) >> 1;
2149             aOffset.Y() = ( ( aAdjustedPrintSize.Height() >> 1 ) - rInfo.maPageSize.Height() ) >> 1;
2150         }
2151 
2152         // create vector of pages to print
2153         ::std::vector< sal_uInt16 > aPageVector;
2154         for (sal_uInt16
2155                  nIndex=0,
2156                  nCount=mrBase.GetDocument()->GetSdPageCount(ePageKind);
2157              nIndex < nCount;
2158              ++nIndex)
2159         {
2160             SdPage* pPage = GetFilteredPage(nIndex, ePageKind, rInfo);
2161             if (pPage != NULL)
2162                 aPageVector.push_back(nIndex);
2163         }
2164 
2165         // create pairs of pages to print on each page
2166         typedef ::std::vector< ::std::pair< sal_uInt16, sal_uInt16 > > PairVector;
2167         PairVector aPairVector;
2168         if ( ! aPageVector.empty())
2169         {
2170             sal_uInt32 nFirstIndex = 0, nLastIndex = aPageVector.size() - 1;
2171 
2172             if( aPageVector.size() & 1 )
2173                 aPairVector.push_back( ::std::make_pair( (sal_uInt16) 65535, aPageVector[ nFirstIndex++ ] ) );
2174             else
2175                 aPairVector.push_back( ::std::make_pair( aPageVector[ nLastIndex-- ], aPageVector[ nFirstIndex++ ] ) );
2176 
2177             while( nFirstIndex < nLastIndex )
2178             {
2179                 if( nFirstIndex & 1 )
2180                     aPairVector.push_back( ::std::make_pair( aPageVector[ nFirstIndex++ ], aPageVector[ nLastIndex-- ] ) );
2181                 else
2182                     aPairVector.push_back( ::std::make_pair( aPageVector[ nLastIndex-- ], aPageVector[ nFirstIndex++ ] ) );
2183             }
2184         }
2185 
2186         for (sal_uInt32
2187                  nIndex=0,
2188                  nCount=aPairVector.size();
2189              nIndex < nCount;
2190              ++nIndex)
2191         {
2192             const bool bIsIndexOdd (nIndex & 1);
2193             if ((!bIsIndexOdd && mpOptions->IsPrintFrontPage())
2194                 || (bIsIndexOdd && mpOptions->IsPrintBackPage()))
2195             {
2196                 const ::std::pair<sal_uInt16, sal_uInt16> aPair (aPairVector[nIndex]);
2197                 Point aSecondOffset (aOffset);
2198                 if (rInfo.meOrientation == ORIENTATION_LANDSCAPE)
2199                     aSecondOffset.X() += aAdjustedPrintSize.Width() / 2;
2200                 else
2201                     aSecondOffset.Y() += aAdjustedPrintSize.Height() / 2;
2202                 maPrinterPages.push_back(
2203                     ::boost::shared_ptr<PrinterPage>(
2204                         new BookletPrinterPage(
2205                             aPair.first,
2206                             aPair.second,
2207                             aOffset,
2208                             aSecondOffset,
2209                             ePageKind,
2210                             aMap,
2211                             rInfo.mbPrintMarkedOnly,
2212                             rInfo.mnDrawMode,
2213                             rInfo.meOrientation,
2214                             rInfo.mpPrinter->GetPaperBin())));
2215 
2216             }
2217         }
2218     }
2219 
2220 
2221 
2222 
2223     /** Print one slide multiple times on one printer page so that the whole
2224         printer page is covered.
2225     */
2226     void PrepareTiledPage (
2227         const sal_Int32 nPageIndex,
2228         const SdPage& rPage,
2229         const PageKind ePageKind,
2230         const PrintInfo& rInfo)
2231     {
2232         sal_uInt16 nPaperBin;
2233         if ( ! mpOptions->IsPaperBin())
2234             nPaperBin = rPage.GetPaperBin();
2235         else
2236             nPaperBin = rInfo.mpPrinter->GetPaperBin();
2237 
2238         maPrinterPages.push_back(
2239             ::boost::shared_ptr<PrinterPage>(
2240                 new TiledPrinterPage(
2241                     sal::static_int_cast<sal_uInt16>(nPageIndex),
2242                     ePageKind,
2243                     500,
2244                     rInfo.mbPrintMarkedOnly,
2245                     rInfo.msPageString,
2246                     rInfo.mpPrinter->GetPageOffset(),
2247                     rInfo.mnDrawMode,
2248                     rInfo.meOrientation,
2249                     nPaperBin)));
2250     }
2251 
2252 
2253 
2254     /** Print one standard slide or notes page on one to many printer
2255         pages.  More than on printer page is used when the slide is larger
2256         than the printable area.
2257     */
2258     void PrepareScaledPage (
2259         const sal_Int32 nPageIndex,
2260         const SdPage& rPage,
2261         const PageKind ePageKind,
2262         const PrintInfo& rInfo)
2263     {
2264         const Point aPageOffset (rInfo.mpPrinter->GetPageOffset());
2265 
2266         sal_uInt16 nPaperBin;
2267         if ( ! mpOptions->IsPaperBin())
2268             nPaperBin = rPage.GetPaperBin();
2269         else
2270             nPaperBin = rInfo.mpPrinter->GetPaperBin();
2271 
2272         // For pages larger then the printable area there
2273         // are three options:
2274         // 1. Scale down to the page to the printable area.
2275         // 2. Print only the upper left part of the page
2276         //    (without the unprintable borders).
2277         // 3. Split the page into parts of the size of the
2278         // printable area.
2279         const bool bScalePage (mpOptions->IsPageSize());
2280         const bool bCutPage (mpOptions->IsCutPage());
2281         MapMode aMap (rInfo.maMap);
2282         if (bScalePage || bCutPage)
2283         {
2284             // Handle 1 and 2.
2285 
2286             // if CutPage is set then do not move it, otherwise move the
2287             // scaled page to printable area
2288             #if 0
2289             if (bCutPage)
2290                 aMap.SetOrigin(Point(-aPageOffset.X(), -aPageOffset.Y()));
2291             else
2292                 aMap.SetOrigin(Point(0,0));
2293             #endif
2294             maPrinterPages.push_back(
2295                 ::boost::shared_ptr<PrinterPage>(
2296                     new RegularPrinterPage(
2297                         sal::static_int_cast<sal_uInt16>(nPageIndex),
2298                         ePageKind,
2299                         aMap,
2300                         rInfo.mbPrintMarkedOnly,
2301                         rInfo.msPageString,
2302                         aPageOffset,
2303                         rInfo.mnDrawMode,
2304                         rInfo.meOrientation,
2305                         nPaperBin)));
2306         }
2307         else
2308         {
2309             // Handle 3.  Print parts of the page in the size of the
2310             // printable area until the whole page is covered.
2311 
2312             // keep the page content at its position if it fits, otherwise
2313             // move it to the printable area
2314             const long nPageWidth (
2315                 rInfo.maPageSize.Width() - rPage.GetLftBorder() - rPage.GetRgtBorder());
2316             const long nPageHeight (
2317                 rInfo.maPageSize.Height() - rPage.GetUppBorder() - rPage.GetLwrBorder());
2318             #if 0
2319             Point aOrigin (
2320                 nPageWidth < rInfo.maPrintSize.Width() ? -aPageOffset.X() : 0,
2321                 nPageHeight < rInfo.maPrintSize.Height() ? -aPageOffset.Y() : 0);
2322             #else
2323             Point aOrigin ( 0, 0 );
2324             #endif
2325             for (Point aPageOrigin = aOrigin;
2326                  -aPageOrigin.Y()<nPageHeight;
2327                  aPageOrigin.Y() -= rInfo.maPrintSize.Height())
2328             {
2329                 for (aPageOrigin.X()=aOrigin.X();
2330                      -aPageOrigin.X()<nPageWidth;
2331                      aPageOrigin.X() -= rInfo.maPrintSize.Width())
2332                 {
2333                     aMap.SetOrigin(aPageOrigin);
2334                     maPrinterPages.push_back(
2335                         ::boost::shared_ptr<PrinterPage>(
2336                             new RegularPrinterPage(
2337                                 sal::static_int_cast<sal_uInt16>(nPageIndex),
2338                                 ePageKind,
2339                                 aMap,
2340                                 rInfo.mbPrintMarkedOnly,
2341                                 rInfo.msPageString,
2342                                 aPageOffset,
2343                                 rInfo.mnDrawMode,
2344                                 rInfo.meOrientation,
2345                                 nPaperBin)));
2346                 }
2347             }
2348         }
2349     }
2350 };
2351 
2352 
2353 
2354 
2355 //===== DocumentRenderer ======================================================
2356 
2357 DocumentRenderer::DocumentRenderer (ViewShellBase& rBase)
2358     : DocumentRendererInterfaceBase(m_aMutex),
2359       mpImpl(new Implementation(rBase))
2360 {
2361 }
2362 
2363 
2364 
2365 
2366 DocumentRenderer::~DocumentRenderer (void)
2367 {
2368 }
2369 
2370 
2371 
2372 
2373 //----- XRenderable -----------------------------------------------------------
2374 
2375 sal_Int32 SAL_CALL DocumentRenderer::getRendererCount (
2376     const css::uno::Any& aSelection,
2377     const css::uno::Sequence<css::beans::PropertyValue >& rOptions)
2378     throw (css::lang::IllegalArgumentException, css::uno::RuntimeException)
2379 {
2380     (void)aSelection;
2381     mpImpl->ProcessProperties(rOptions);
2382     return mpImpl->GetPrintPageCount();
2383 }
2384 
2385 
2386 
2387 
2388 Sequence<beans::PropertyValue> SAL_CALL DocumentRenderer::getRenderer (
2389     sal_Int32 nRenderer,
2390     const css::uno::Any& rSelection,
2391     const css::uno::Sequence<css::beans::PropertyValue>& rOptions)
2392     throw (css::lang::IllegalArgumentException, css::uno::RuntimeException)
2393 {
2394     (void)nRenderer;
2395     (void)rSelection;
2396     mpImpl->ProcessProperties(rOptions);
2397     return mpImpl->GetProperties(rOptions);
2398 }
2399 
2400 
2401 
2402 
2403 void SAL_CALL DocumentRenderer::render (
2404     sal_Int32 nRenderer,
2405     const css::uno::Any& rSelection,
2406     const css::uno::Sequence<css::beans::PropertyValue>& rOptions)
2407     throw (css::lang::IllegalArgumentException, css::uno::RuntimeException)
2408 {
2409     (void)rSelection;
2410     mpImpl->ProcessProperties(rOptions);
2411     mpImpl->PrintPage(nRenderer);
2412 }
2413 
2414 
2415 
2416 } // end of namespace sd
2417