xref: /AOO41X/main/sc/source/ui/view/printfun.cxx (revision 590a31a18e48070419a1bd47d2fa79552e22b8de)
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 // MARKER(update_precomp.py): autogen include statement, do not remove
25 #include "precompiled_sc.hxx"
26 
27 // INCLUDE ---------------------------------------------------------------
28 
29 #include "scitems.hxx"
30 #include <editeng/eeitem.hxx>
31 
32 #include "printfun.hxx"
33 
34 #include <svx/svxids.hrc>
35 #include <editeng/adjitem.hxx>
36 #include <editeng/boxitem.hxx>
37 #include <editeng/brshitem.hxx>
38 #include <svtools/colorcfg.hxx>
39 #include <editeng/editstat.hxx>     // EE_CNTRL_RTFSTYLESHEETS
40 #include <svx/fmview.hxx>
41 #include <editeng/frmdiritem.hxx>
42 #include <editeng/lrspitem.hxx>
43 #include <editeng/paperinf.hxx>
44 #include <editeng/pbinitem.hxx>
45 #include <editeng/shaditem.hxx>
46 #include <editeng/sizeitem.hxx>
47 #include <svx/svdpagv.hxx>
48 #include <editeng/ulspitem.hxx>
49 #include <sfx2/app.hxx>
50 #include <sfx2/printer.hxx>
51 #include <tools/multisel.hxx>
52 #include <sfx2/docfile.hxx>
53 #include <tools/urlobj.hxx>
54 #include <svx/xoutbmp.hxx>
55 
56 #include "editutil.hxx"
57 #include "docsh.hxx"
58 #include "output.hxx"
59 #include "viewdata.hxx"
60 #include "viewopti.hxx"
61 #include "stlpool.hxx"
62 #include "pagepar.hxx"
63 #include "attrib.hxx"
64 #include "patattr.hxx"
65 #include "docpool.hxx"
66 #include "dociter.hxx"
67 #include "cell.hxx"
68 #include "drawutil.hxx"
69 #include "globstr.hrc"
70 #include "scresid.hxx"
71 #include "sc.hrc"
72 #include "pagedata.hxx"
73 #include "printopt.hxx"
74 #include "prevloc.hxx"
75 #include "scmod.hxx"
76 #include "drwlayer.hxx"
77 #include "fillinfo.hxx"
78 #include "postit.hxx"
79 
80 #include <vcl/lineinfo.hxx>
81 #include <tools/pstm.hxx>
82 
83 #include <boost/scoped_ptr.hpp>
84 
85 #define ZOOM_MIN    10
86 
87 #define GET_BOOL(set,which)   ((const SfxBoolItem&)(set)->Get((which))).GetValue()
88 #define GET_USHORT(set,which) ((const SfxUInt16Item&)(set)->Get((which))).GetValue()
89 #define GET_SHOW(set,which)   ( VOBJ_MODE_SHOW == ScVObjMode( ((const ScViewObjectModeItem&)(set)->Get((which))).GetValue()) )
90 
91 //------------------------------------------------------------------------
92 
ScPageRowEntry(const ScPageRowEntry & r)93 ScPageRowEntry::ScPageRowEntry(const ScPageRowEntry& r)
94 {
95     nStartRow = r.nStartRow;
96     nEndRow   = r.nEndRow;
97     nPagesX   = r.nPagesX;
98     if (r.pHidden && nPagesX)
99     {
100         pHidden = new sal_Bool[nPagesX];
101         memcpy( pHidden, r.pHidden, nPagesX * sizeof(sal_Bool) );
102     }
103     else
104         pHidden = NULL;
105 }
106 
operator =(const ScPageRowEntry & r)107 const ScPageRowEntry& ScPageRowEntry::operator=(const ScPageRowEntry& r)
108 {
109     delete[] pHidden;
110 
111     nStartRow = r.nStartRow;
112     nEndRow   = r.nEndRow;
113     nPagesX   = r.nPagesX;
114     if (r.pHidden && nPagesX)
115     {
116         pHidden = new sal_Bool[nPagesX];
117         memcpy( pHidden, r.pHidden, nPagesX * sizeof(sal_Bool) );
118     }
119     else
120         pHidden = NULL;
121 
122     return *this;
123 }
124 
SetPagesX(size_t nNew)125 void ScPageRowEntry::SetPagesX(size_t nNew)
126 {
127     if (pHidden)
128     {
129         DBG_ERROR("SetPagesX nicht nach SetHidden");
130         delete[] pHidden;
131         pHidden = NULL;
132     }
133     nPagesX = nNew;
134 }
135 
SetHidden(size_t nX)136 void ScPageRowEntry::SetHidden(size_t nX)
137 {
138     if ( nX < nPagesX )
139     {
140         if ( nX+1 == nPagesX )  // letzte Seite?
141             --nPagesX;
142         else
143         {
144             if (!pHidden)
145             {
146                 pHidden = new sal_Bool[nPagesX];
147                 memset( pHidden, sal_False, nPagesX * sizeof(sal_Bool) );
148             }
149             pHidden[nX] = sal_True;
150         }
151     }
152 }
153 
IsHidden(size_t nX) const154 sal_Bool ScPageRowEntry::IsHidden(size_t nX) const
155 {
156     return nX>=nPagesX || ( pHidden && pHidden[nX] );       //! inline?
157 }
158 
CountVisible() const159 size_t ScPageRowEntry::CountVisible() const
160 {
161     if ( pHidden )
162     {
163         size_t nVis = 0;
164         for (size_t i=0; i<nPagesX; i++)
165             if (!pHidden[i])
166                 ++nVis;
167         return nVis;
168     }
169     else
170         return nPagesX;
171 }
172 
173 //------------------------------------------------------------------------
174 
lcl_LineTotal(const SvxBorderLine * pLine)175 long lcl_LineTotal(const SvxBorderLine* pLine)
176 {
177     return pLine ? ( pLine->GetOutWidth() + pLine->GetInWidth() + pLine->GetDistance() ) : 0;
178 }
179 
Construct(const ScPrintOptions * pOptions)180 void ScPrintFunc::Construct( const ScPrintOptions* pOptions )
181 {
182     pDocShell->UpdatePendingRowHeights( nPrintTab );
183     pDoc = pDocShell->GetDocument();
184 
185     SfxPrinter* pDocPrinter = pDoc->GetPrinter();   // auch fuer Preview den Drucker nehmen
186     if (pDocPrinter)
187         aOldPrinterMode = pDocPrinter->GetMapMode();
188 
189     //  einheitlicher MapMode ueber alle Aufrufe (z.B. Repaint !!!),
190     //  weil die EditEngine sonst unterschiedliche Texthoehen liefert
191     pDev->SetMapMode(MAP_PIXEL);
192 
193     pBorderItem = NULL;
194     pBackgroundItem = NULL;
195     pShadowItem = NULL;
196 
197     pEditEngine = NULL;
198     pEditDefaults = NULL;
199 
200     ScStyleSheetPool* pStylePool    = pDoc->GetStyleSheetPool();
201     SfxStyleSheetBase* pStyleSheet  = pStylePool->Find(
202                                             pDoc->GetPageStyle( nPrintTab ),
203                                             SFX_STYLE_FAMILY_PAGE );
204     if (pStyleSheet)
205         pParamSet = &pStyleSheet->GetItemSet();
206     else
207     {
208         DBG_ERROR("Seitenvorlage nicht gefunden" );
209         pParamSet = NULL;
210     }
211 
212     if (!bState)
213         nZoom = 100;
214     nManualZoom = 100;
215     bClearWin = sal_False;
216     bUseStyleColor = sal_False;
217     bIsRender = sal_False;
218 
219     InitParam(pOptions);
220 
221     pPageData = NULL;       // wird nur zur Initialisierung gebraucht
222 }
223 
ScPrintFunc(ScDocShell * pShell,SfxPrinter * pNewPrinter,SCTAB nTab,long nPage,long nDocP,const ScRange * pArea,const ScPrintOptions * pOptions,ScPageBreakData * pData)224 ScPrintFunc::ScPrintFunc( ScDocShell* pShell, SfxPrinter* pNewPrinter, SCTAB nTab,
225                             long nPage, long nDocP, const ScRange* pArea,
226                             const ScPrintOptions* pOptions,
227                             ScPageBreakData* pData )
228     :   pDocShell           ( pShell ),
229         pPrinter            ( pNewPrinter ),
230         pDrawView           ( NULL ),
231         nPrintTab           ( nTab ),
232         nPageStart          ( nPage ),
233         nDocPages           ( nDocP ),
234         pUserArea           ( pArea ),
235         bState              ( sal_False ),
236         bSourceRangeValid   ( sal_False ),
237         bPrintCurrentTable  ( sal_False ),
238         bMultiArea          ( sal_False ),
239         nTabPages           ( 0 ),
240         nTotalPages         ( 0 ),
241         pPageData           ( pData )
242 {
243     pDev = pPrinter;
244     aSrcOffset = pPrinter->PixelToLogic( pPrinter->GetPageOffsetPixel(), MAP_100TH_MM );
245     Construct( pOptions );
246 }
247 
ScPrintFunc(OutputDevice * pOutDev,ScDocShell * pShell,SCTAB nTab,long nPage,long nDocP,const ScRange * pArea,const ScPrintOptions * pOptions)248 ScPrintFunc::ScPrintFunc( OutputDevice* pOutDev, ScDocShell* pShell, SCTAB nTab,
249                             long nPage, long nDocP, const ScRange* pArea,
250                             const ScPrintOptions* pOptions )
251     :   pDocShell           ( pShell ),
252         pPrinter            ( NULL ),
253         pDrawView           ( NULL ),
254         nPrintTab           ( nTab ),
255         nPageStart          ( nPage ),
256         nDocPages           ( nDocP ),
257         pUserArea           ( pArea ),
258         bState              ( sal_False ),
259         bSourceRangeValid   ( sal_False ),
260         bPrintCurrentTable  ( sal_False ),
261         bMultiArea          ( sal_False ),
262         nTabPages           ( 0 ),
263         nTotalPages         ( 0 ),
264         pPageData           ( NULL )
265 {
266     pDev = pOutDev;
267     Construct( pOptions );
268 }
269 
ScPrintFunc(OutputDevice * pOutDev,ScDocShell * pShell,const ScPrintState & rState,const ScPrintOptions * pOptions)270 ScPrintFunc::ScPrintFunc( OutputDevice* pOutDev, ScDocShell* pShell,
271                              const ScPrintState& rState, const ScPrintOptions* pOptions )
272     :   pDocShell           ( pShell ),
273         pPrinter            ( NULL ),
274         pDrawView           ( NULL ),
275         pUserArea           ( NULL ),
276         bSourceRangeValid   ( sal_False ),
277         bPrintCurrentTable  ( sal_False ),
278         bMultiArea          ( sal_False ),
279         pPageData           ( NULL )
280 {
281     pDev = pOutDev;
282 
283     nPrintTab   = rState.nPrintTab;
284     nStartCol   = rState.nStartCol;
285     nStartRow   = rState.nStartRow;
286     nEndCol     = rState.nEndCol;
287     nEndRow     = rState.nEndRow;
288     nZoom       = rState.nZoom;
289     nPagesX     = rState.nPagesX;
290     nPagesY     = rState.nPagesY;
291     nTabPages   = rState.nTabPages;
292     nTotalPages = rState.nTotalPages;
293     nPageStart  = rState.nPageStart;
294     nDocPages   = rState.nDocPages;
295     bState      = sal_True;
296 
297     Construct( pOptions );
298 }
299 
GetPrintState(ScPrintState & rState)300 void ScPrintFunc::GetPrintState( ScPrintState& rState )
301 {
302     rState.nPrintTab    = nPrintTab;
303     rState.nStartCol    = nStartCol;
304     rState.nStartRow    = nStartRow;
305     rState.nEndCol      = nEndCol;
306     rState.nEndRow      = nEndRow;
307     rState.nZoom        = nZoom;
308     rState.nPagesX      = nPagesX;
309     rState.nPagesY      = nPagesY;
310     rState.nTabPages    = nTabPages;
311     rState.nTotalPages  = nTotalPages;
312     rState.nPageStart   = nPageStart;
313     rState.nDocPages    = nDocPages;
314 }
315 
GetLastSourceRange(ScRange & rRange) const316 sal_Bool ScPrintFunc::GetLastSourceRange( ScRange& rRange ) const
317 {
318     rRange = aLastSourceRange;
319     return bSourceRangeValid;
320 }
321 
FillPageData()322 void ScPrintFunc::FillPageData()
323 {
324     if (pPageData)
325     {
326         sal_uInt16 nCount = sal::static_int_cast<sal_uInt16>( pPageData->GetCount() );
327         ScPrintRangeData& rData = pPageData->GetData(nCount);       // hochzaehlen
328 
329         rData.SetPrintRange( ScRange( nStartCol, nStartRow, nPrintTab,
330                                         nEndCol, nEndRow, nPrintTab ) );
331         // #123672#
332         if(maPageEndX.empty())
333         {
334             OSL_ENSURE(false, "vector access error for maPageEndX (!)");
335         }
336         else
337         {
338             rData.SetPagesX( nPagesX, &maPageEndX[0]);
339         }
340 
341         // #123672#
342         if(maPageEndY.empty())
343         {
344             OSL_ENSURE(false, "vector access error for maPageEndY (!)");
345         }
346         else
347         {
348             rData.SetPagesY( nTotalY, &maPageEndY[0]);
349         }
350 
351         //  Einstellungen
352         rData.SetTopDown( aTableParam.bTopDown );
353         rData.SetAutomatic( !aAreaParam.bPrintArea );
354     }
355 }
356 
~ScPrintFunc()357 ScPrintFunc::~ScPrintFunc()
358 {
359     ScAddress* pTripel = (ScAddress*) aNotePosList.First();
360     while (pTripel)
361     {
362         delete pTripel;
363         pTripel = (ScAddress*) aNotePosList.Next();
364     }
365     aNotePosList.Clear();
366 
367     delete pEditDefaults;
368     delete pEditEngine;
369 
370     //  Druckereinstellungen werden jetzt von aussen wiederhergestellt
371 
372     //  #64294# Fuer DrawingLayer/Charts muss der MapMode am Drucker (RefDevice) immer stimmen
373     SfxPrinter* pDocPrinter = pDoc->GetPrinter();   // auch fuer Preview den Drucker nehmen
374     if (pDocPrinter)
375         pDocPrinter->SetMapMode(aOldPrinterMode);
376 }
377 
SetDrawView(FmFormView * pNew)378 void ScPrintFunc::SetDrawView( FmFormView* pNew )
379 {
380     pDrawView = pNew;
381 }
382 
lcl_HidePrint(ScTableInfo & rTabInfo,SCCOL nX1,SCCOL nX2)383 void lcl_HidePrint( ScTableInfo& rTabInfo, SCCOL nX1, SCCOL nX2 )
384 {
385     for (SCSIZE nArrY=1; nArrY+1<rTabInfo.mnArrCount; nArrY++)
386     {
387         RowInfo* pThisRowInfo = &rTabInfo.mpRowInfo[nArrY];
388         for (SCCOL nX=nX1; nX<=nX2; nX++)
389         {
390             const CellInfo& rCellInfo = pThisRowInfo->pCellInfo[nX+1];
391             if (!rCellInfo.bEmptyCellText)
392                 if (((const ScProtectionAttr&)rCellInfo.pPatternAttr->
393                             GetItem(ATTR_PROTECTION, rCellInfo.pConditionSet)).GetHidePrint())
394                 {
395                     pThisRowInfo->pCellInfo[nX+1].pCell          = NULL;
396                     pThisRowInfo->pCellInfo[nX+1].bEmptyCellText = sal_True;
397                 }
398         }
399     }
400 }
401 
402 //
403 //          Ausgabe auf Device (static)
404 //
405 //      wird benutzt fuer:
406 //      -   Clipboard/Bitmap
407 //      -   Ole-Object (DocShell::Draw)
408 //      -   Vorschau bei Vorlagen
409 
DrawToDev(ScDocument * pDoc,OutputDevice * pDev,double,const Rectangle & rBound,ScViewData * pViewData,sal_Bool bMetaFile)410 void ScPrintFunc::DrawToDev( ScDocument* pDoc, OutputDevice* pDev, double /* nPrintFactor */,
411                             const Rectangle& rBound, ScViewData* pViewData, sal_Bool bMetaFile )
412 {
413     //! nPrintFactor auswerten !!!
414 
415     SCTAB nTab = 0;
416     if (pViewData)
417         nTab = pViewData->GetTabNo();
418 
419     sal_Bool bDoGrid, bNullVal, bFormula;
420     ScStyleSheetPool* pStylePool = pDoc->GetStyleSheetPool();
421     SfxStyleSheetBase* pStyleSheet = pStylePool->Find( pDoc->GetPageStyle( nTab ), SFX_STYLE_FAMILY_PAGE );
422     if (pStyleSheet)
423     {
424         SfxItemSet& rSet = pStyleSheet->GetItemSet();
425         bDoGrid  = ((const SfxBoolItem&)rSet.Get(ATTR_PAGE_GRID)).GetValue();
426         bNullVal = ((const SfxBoolItem&)rSet.Get(ATTR_PAGE_NULLVALS)).GetValue();
427         bFormula = ((const SfxBoolItem&)rSet.Get(ATTR_PAGE_FORMULAS)).GetValue();
428     }
429     else
430     {
431         const ScViewOptions& rOpt = pDoc->GetViewOptions();
432         bDoGrid  = rOpt.GetOption(VOPT_GRID);
433         bNullVal = rOpt.GetOption(VOPT_NULLVALS);
434         bFormula = rOpt.GetOption(VOPT_FORMULAS);
435     }
436 
437     MapMode aMode = pDev->GetMapMode();
438 
439     Rectangle aRect = rBound;
440 
441     if (aRect.Right() < aRect.Left() || aRect.Bottom() < aRect.Top())
442         aRect = Rectangle( Point(), pDev->GetOutputSize() );
443 
444     SCCOL nX1 = 0;
445     SCROW nY1 = 0;
446     SCCOL nX2 = OLE_STD_CELLS_X - 1;
447     SCROW nY2 = OLE_STD_CELLS_Y - 1;
448     if (bMetaFile)
449     {
450         ScRange aRange = pDoc->GetRange( nTab, rBound );
451         nX1 = aRange.aStart.Col();
452         nY1 = aRange.aStart.Row();
453         nX2 = aRange.aEnd.Col();
454         nY2 = aRange.aEnd.Row();
455     }
456     else if (pViewData)
457     {
458         ScSplitPos eWhich = pViewData->GetActivePart();
459         ScHSplitPos eHWhich = WhichH(eWhich);
460         ScVSplitPos eVWhich = WhichV(eWhich);
461         nX1 = pViewData->GetPosX(eHWhich);
462         nY1 = pViewData->GetPosY(eVWhich);
463         nX2 = nX1 + pViewData->VisibleCellsX(eHWhich);
464         if (nX2>nX1) --nX2;
465         nY2 = nY1 + pViewData->VisibleCellsY(eVWhich);
466         if (nY2>nY1) --nY2;
467     }
468 
469     if (nX1 > MAXCOL) nX1 = MAXCOL;
470     if (nX2 > MAXCOL) nX2 = MAXCOL;
471     if (nY1 > MAXROW) nY1 = MAXROW;
472     if (nY2 > MAXROW) nY2 = MAXROW;
473 
474     long nDevSizeX = aRect.Right()-aRect.Left()+1;
475     long nDevSizeY = aRect.Bottom()-aRect.Top()+1;
476 
477     Rectangle aLines;
478     ScRange aRange( nX1,nY1,nTab, nX2,nY2,nTab );
479 //    sal_Bool bAddLines = pDoc->HasLines( aRange, aLines );
480 
481     long nTwipsSizeX = 0;
482     for (SCCOL i=nX1; i<=nX2; i++)
483         nTwipsSizeX += pDoc->GetColWidth( i, nTab );
484     long nTwipsSizeY = (long) pDoc->GetRowHeight( nY1, nY2, nTab );
485 
486     //  wenn keine Linien, dann trotzdem Platz fuer den Aussenrahmen (20 Twips = 1pt)
487     //  (HasLines initalisiert aLines auf 0,0,0,0)
488     nTwipsSizeX += aLines.Left() + Max( aLines.Right(), 20L );
489     nTwipsSizeY += aLines.Top() +  Max( aLines.Bottom(), 20L );
490 
491     double nScaleX = (double) nDevSizeX / nTwipsSizeX;
492     double nScaleY = (double) nDevSizeY / nTwipsSizeY;
493 
494                             //!     Flag bei FillInfo uebergeben !!!!!
495     ScRange aERange;
496     sal_Bool bEmbed = pDoc->IsEmbedded();
497     if (bEmbed)
498     {
499         pDoc->GetEmbedded(aERange);
500         pDoc->ResetEmbedded();
501     }
502 
503     //  Daten zusammenstellen
504 
505     ScTableInfo aTabInfo;
506     pDoc->FillInfo( aTabInfo, nX1, nY1, nX2, nY2, nTab,
507                                         nScaleX, nScaleY, sal_False, bFormula );
508     lcl_HidePrint( aTabInfo, nX1, nX2 );
509 
510     if (bEmbed)
511         pDoc->SetEmbedded(aERange);
512 
513 /*  if (!bMetaFile)
514         pDev->SetMapMode(MAP_PIXEL);
515 */
516     long nScrX = aRect.Left();
517     long nScrY = aRect.Top();
518 
519     //  Wenn keine Linien, trotzdem Platz fuer Gitterlinien lassen
520     //  (werden sonst abgeschnitten)
521     long nAddX = (long)( aLines.Left() * nScaleX );
522     nScrX += ( nAddX ? nAddX : 1 );
523     long nAddY = (long)( aLines.Top() * nScaleY );
524     nScrY += ( nAddY ? nAddY : 1 );
525 
526     ScOutputData aOutputData( pDev, OUTTYPE_PRINTER, aTabInfo, pDoc, nTab,
527                                 nScrX, nScrY, nX1, nY1, nX2, nY2, nScaleX, nScaleY );
528     aOutputData.SetMetaFileMode(bMetaFile);
529     aOutputData.SetShowNullValues(bNullVal);
530     aOutputData.SetShowFormulas(bFormula);
531 
532     // #114135#
533     ScDrawLayer* pModel = pDoc->GetDrawLayer();
534     FmFormView* pDrawView = NULL;
535 
536     if( pModel )
537     {
538         pDrawView = new FmFormView( pModel, pDev );
539         pDrawView->ShowSdrPage(pDrawView->GetModel()->GetPage(nTab));
540         pDrawView->SetPrintPreview( sal_True );
541         aOutputData.SetDrawView( pDrawView );
542     }
543 
544     //! SetUseStyleColor ??
545 
546     if ( bMetaFile && pDev->GetOutDevType() == OUTDEV_VIRDEV )
547         aOutputData.SetSnapPixel();
548 
549     Point aLogStart = pDev->PixelToLogic( Point(nScrX,nScrY), MAP_100TH_MM );
550     long nLogStX = aLogStart.X();
551     long nLogStY = aLogStart.Y();
552 
553     //!     nZoom fuer GetFont in OutputData ???
554 
555     if (!bMetaFile && pViewData)
556         pDev->SetMapMode(pViewData->GetLogicMode(pViewData->GetActivePart()));
557 
558     // #i72502#
559     const Point aMMOffset(aOutputData.PrePrintDrawingLayer(nLogStX, nLogStY));
560     aOutputData.PrintDrawingLayer(SC_LAYER_BACK, aMMOffset);
561 
562     if (!bMetaFile && pViewData)
563         pDev->SetMapMode(aMode);
564 
565     aOutputData.DrawBackground();
566 
567 #ifdef OS2
568     if (bMetaFile && !bDoGrid)
569     {
570                     // unter OS2 fuer Metafiles gesamte Flaeche benutzen,
571                     // weil sonst die Groesse nicht erkannt wird
572         pDev->SetLineColor();
573         pDev->SetFillColor();
574         pDev->DrawRect( Rectangle( nScrX,nScrY,
575                         nScrX+aOutputData.GetScrW(), nScrY+aOutputData.GetScrH() ) );
576     }
577 #endif
578 
579     aOutputData.DrawShadow();
580     aOutputData.DrawFrame();
581     aOutputData.DrawStrings();
582 
583     if (!bMetaFile && pViewData)
584         pDev->SetMapMode(pViewData->GetLogicMode(pViewData->GetActivePart()));
585 
586     aOutputData.DrawEdit(!bMetaFile);
587 
588     if (bDoGrid)
589     {
590         if (!bMetaFile && pViewData)
591             pDev->SetMapMode(aMode);
592 
593         aOutputData.DrawGrid( sal_True, sal_False );    // keine Seitenumbrueche
594 
595         pDev->SetLineColor( COL_BLACK );
596 
597         Size aOne = pDev->PixelToLogic( Size(1,1) );
598         if (bMetaFile)
599             aOne = Size(1,1);   // compatible with DrawGrid
600         long nRight = nScrX + aOutputData.GetScrW() - aOne.Width();
601         long nBottom = nScrY + aOutputData.GetScrH() - aOne.Height();
602 
603         sal_Bool bLayoutRTL = pDoc->IsLayoutRTL( nTab );
604 
605         // extra line at the left edge for left-to-right, right for right-to-left
606         if ( bLayoutRTL )
607             pDev->DrawLine( Point(nRight,nScrY), Point(nRight,nBottom) );
608         else
609             pDev->DrawLine( Point(nScrX,nScrY), Point(nScrX,nBottom) );
610         // extra line at the top in both cases
611         pDev->DrawLine( Point(nScrX,nScrY), Point(nRight,nScrY) );
612     }
613 
614     // #i72502#
615     aOutputData.PrintDrawingLayer(SC_LAYER_FRONT, aMMOffset);
616     aOutputData.PrintDrawingLayer(SC_LAYER_INTERN, aMMOffset);
617     aOutputData.PostPrintDrawingLayer(aMMOffset); // #i74768#
618 
619     // #114135#
620     delete pDrawView;
621 }
622 
623 //
624 //          Drucken
625 //
626 
lcl_FillHFParam(ScPrintHFParam & rParam,const SfxItemSet * pHFSet)627 void lcl_FillHFParam( ScPrintHFParam& rParam, const SfxItemSet* pHFSet )
628 {
629     //  nDistance muss vorher unterschiedlich initalisiert sein
630 
631     if ( pHFSet == NULL )
632     {
633         rParam.bEnable  = sal_False;
634         rParam.pBorder  = NULL;
635         rParam.pBack    = NULL;
636         rParam.pShadow  = NULL;
637     }
638     else
639     {
640         rParam.bEnable  = ((const SfxBoolItem&) pHFSet->Get(ATTR_PAGE_ON)).GetValue();
641         rParam.bDynamic = ((const SfxBoolItem&) pHFSet->Get(ATTR_PAGE_DYNAMIC)).GetValue();
642         rParam.bShared  = ((const SfxBoolItem&) pHFSet->Get(ATTR_PAGE_SHARED)).GetValue();
643         rParam.nHeight  = ((const SvxSizeItem&) pHFSet->Get(ATTR_PAGE_SIZE)).GetSize().Height();
644         const SvxLRSpaceItem* pHFLR = &(const SvxLRSpaceItem&) pHFSet->Get(ATTR_LRSPACE);
645         long nTmp;
646         nTmp = pHFLR->GetLeft();
647         rParam.nLeft = nTmp < 0 ? 0 : sal_uInt16(nTmp);
648         nTmp = pHFLR->GetRight();
649         rParam.nRight = nTmp < 0 ? 0 : sal_uInt16(nTmp);
650         rParam.pBorder  = (const SvxBoxItem*)   &pHFSet->Get(ATTR_BORDER);
651         rParam.pBack    = (const SvxBrushItem*) &pHFSet->Get(ATTR_BACKGROUND);
652         rParam.pShadow  = (const SvxShadowItem*)&pHFSet->Get(ATTR_SHADOW);;
653 
654 //  jetzt doch wieder schon im Dialog:
655 //      rParam.nHeight += rParam.nDistance;             // nicht mehr im Dialog ???
656 
657         if (rParam.pBorder)
658             rParam.nHeight += lcl_LineTotal( rParam.pBorder->GetTop() ) +
659                               lcl_LineTotal( rParam.pBorder->GetBottom() );
660 
661         rParam.nManHeight = rParam.nHeight;
662     }
663 
664     if (!rParam.bEnable)
665         rParam.nHeight = 0;
666 }
667 
668 //  bNew = TRUE:    benutzten Bereich aus dem Dokument suchen
669 //  bNew = FALSE:   nur ganze Zeilen/Spalten begrenzen
670 
AdjustPrintArea(sal_Bool bNew)671 sal_Bool ScPrintFunc::AdjustPrintArea( sal_Bool bNew )
672 {
673     SCCOL nOldEndCol = nEndCol; // nur wichtig bei !bNew
674     SCROW nOldEndRow = nEndRow;
675     sal_Bool bChangeCol = sal_True;         // bei bNew werden beide angepasst
676     sal_Bool bChangeRow = sal_True;
677 
678     sal_Bool bNotes = aTableParam.bNotes;
679     if ( bNew )
680     {
681         nStartCol = 0;
682         nStartRow = 0;
683         if (!pDoc->GetPrintArea( nPrintTab, nEndCol, nEndRow, bNotes ))
684             return sal_False;   // nix
685     }
686     else
687     {
688         sal_Bool bFound = sal_True;
689         bChangeCol = ( nStartCol == 0 && nEndCol == MAXCOL );
690         bChangeRow = ( nStartRow == 0 && nEndRow == MAXROW );
691         sal_Bool bForcedChangeRow = sal_False;
692 
693         // #i53558# Crop entire column of old row limit to real print area with
694         // some fuzzyness.
695         if (!bChangeRow && nStartRow == 0)
696         {
697             SCROW nPAEndRow;
698             bFound = pDoc->GetPrintAreaVer( nPrintTab, nStartCol, nEndCol, nPAEndRow, bNotes );
699             // Say we don't want to print more than ~1000 empty rows, which are
700             // about 14 pages intentionally left blank..
701             const SCROW nFuzzy = 23*42;
702             if (nPAEndRow + nFuzzy < nEndRow)
703             {
704                 bForcedChangeRow = sal_True;
705                 nEndRow = nPAEndRow;
706             }
707             else
708                 bFound = sal_True;  // user seems to _want_ to print some empty rows
709         }
710         // TODO: in case we extend the number of columns we may have to do the
711         // same for horizontal cropping.
712 
713         if ( bChangeCol && bChangeRow )
714             bFound = pDoc->GetPrintArea( nPrintTab, nEndCol, nEndRow, bNotes );
715         else if ( bChangeCol )
716             bFound = pDoc->GetPrintAreaHor( nPrintTab, nStartRow, nEndRow, nEndCol, bNotes );
717         else if ( bChangeRow )
718             bFound = pDoc->GetPrintAreaVer( nPrintTab, nStartCol, nEndCol, nEndRow, bNotes );
719 
720         if (!bFound)
721             return sal_False;   // leer
722 
723         if (bForcedChangeRow)
724             bChangeRow = sal_True;
725     }
726 
727     pDoc->ExtendMerge( nStartCol,nStartRow, nEndCol,nEndRow, nPrintTab,
728                         sal_False, sal_True );      // kein Refresh, incl. Attrs
729 
730     if ( bChangeCol )
731     {
732         OutputDevice* pRefDev = pDoc->GetPrinter();     // auch fuer Preview den Drucker nehmen
733         pRefDev->SetMapMode( MAP_PIXEL );               // wichtig fuer GetNeededSize
734 
735         pDoc->ExtendPrintArea( pRefDev,
736                             nPrintTab, nStartCol, nStartRow, nEndCol, nEndRow );
737         //  nEndCol wird veraendert
738     }
739 
740     if ( nEndCol < MAXCOL && pDoc->HasAttrib(
741                     nEndCol,nStartRow,nPrintTab, nEndCol,nEndRow,nPrintTab, HASATTR_SHADOW_RIGHT ) )
742         ++nEndCol;
743     if ( nEndRow < MAXROW && pDoc->HasAttrib(
744                     nStartCol,nEndRow,nPrintTab, nEndCol,nEndRow,nPrintTab, HASATTR_SHADOW_DOWN ) )
745         ++nEndRow;
746 
747     if (!bChangeCol) nEndCol = nOldEndCol;
748     if (!bChangeRow) nEndRow = nOldEndRow;
749 
750     return sal_True;
751 }
752 
TextHeight(const EditTextObject * pObject)753 long ScPrintFunc::TextHeight( const EditTextObject* pObject )
754 {
755     if (!pObject)
756         return 0;
757 
758 //  pEditEngine->SetPageNo( nTotalPages );
759     pEditEngine->SetTextNewDefaults( *pObject, *pEditDefaults, sal_False );
760 
761     return (long) pEditEngine->GetTextHeight();
762 }
763 
764 //  nZoom muss gesetzt sein !!!
765 //  und der entsprechende Twip-MapMode eingestellt
766 
UpdateHFHeight(ScPrintHFParam & rParam)767 void ScPrintFunc::UpdateHFHeight( ScPrintHFParam& rParam )
768 {
769     DBG_ASSERT( aPageSize.Width(), "UpdateHFHeight ohne aPageSize");
770 
771     if (rParam.bEnable && rParam.bDynamic)
772     {
773         //  nHeight aus Inhalten berechnen
774 
775         MakeEditEngine();
776         long nPaperWidth = ( aPageSize.Width() - nLeftMargin - nRightMargin -
777                                 rParam.nLeft - rParam.nRight ) * 100 / nZoom;
778         if (rParam.pBorder)
779             nPaperWidth -= ( rParam.pBorder->GetDistance(BOX_LINE_LEFT) +
780                              rParam.pBorder->GetDistance(BOX_LINE_RIGHT) +
781                              lcl_LineTotal(rParam.pBorder->GetLeft()) +
782                              lcl_LineTotal(rParam.pBorder->GetRight()) ) * 100 / nZoom;
783 
784         if (rParam.pShadow && rParam.pShadow->GetLocation() != SVX_SHADOW_NONE)
785             nPaperWidth -= ( rParam.pShadow->CalcShadowSpace(SHADOW_LEFT) +
786                              rParam.pShadow->CalcShadowSpace(SHADOW_RIGHT) ) * 100L / nZoom;
787 
788         pEditEngine->SetPaperSize( Size( nPaperWidth, 10000 ) );
789 
790         long nMaxHeight = 0;
791         if ( rParam.pLeft )
792         {
793             nMaxHeight = Max( nMaxHeight, TextHeight( rParam.pLeft->GetLeftArea() ) );
794             nMaxHeight = Max( nMaxHeight, TextHeight( rParam.pLeft->GetCenterArea() ) );
795             nMaxHeight = Max( nMaxHeight, TextHeight( rParam.pLeft->GetRightArea() ) );
796         }
797         if ( rParam.pRight )
798         {
799             nMaxHeight = Max( nMaxHeight, TextHeight( rParam.pRight->GetLeftArea() ) );
800             nMaxHeight = Max( nMaxHeight, TextHeight( rParam.pRight->GetCenterArea() ) );
801             nMaxHeight = Max( nMaxHeight, TextHeight( rParam.pRight->GetRightArea() ) );
802         }
803 
804         rParam.nHeight = nMaxHeight + rParam.nDistance;
805         if (rParam.pBorder)
806             rParam.nHeight += rParam.pBorder->GetDistance(BOX_LINE_TOP) +
807                               rParam.pBorder->GetDistance(BOX_LINE_BOTTOM) +
808                               lcl_LineTotal( rParam.pBorder->GetTop() ) +
809                               lcl_LineTotal( rParam.pBorder->GetBottom() );
810         if (rParam.pShadow && rParam.pShadow->GetLocation() != SVX_SHADOW_NONE)
811             rParam.nHeight += rParam.pShadow->CalcShadowSpace(SHADOW_TOP) +
812                               rParam.pShadow->CalcShadowSpace(SHADOW_BOTTOM);
813 
814         if (rParam.nHeight < rParam.nManHeight)
815             rParam.nHeight = rParam.nManHeight;         // eingestelltes Minimum
816     }
817 }
818 
InitParam(const ScPrintOptions * pOptions)819 void ScPrintFunc::InitParam( const ScPrintOptions* pOptions )
820 {
821     if (!pParamSet)
822         return;
823 
824                                 // TabPage "Seite"
825     const SvxLRSpaceItem* pLRItem = (const SvxLRSpaceItem*) &pParamSet->Get( ATTR_LRSPACE );
826     long nTmp;
827     nTmp = pLRItem->GetLeft();
828     nLeftMargin = nTmp < 0 ? 0 : sal_uInt16(nTmp);
829     nTmp = pLRItem->GetRight();
830     nRightMargin = nTmp < 0 ? 0 : sal_uInt16(nTmp);
831     const SvxULSpaceItem* pULItem = (const SvxULSpaceItem*) &pParamSet->Get( ATTR_ULSPACE );
832     nTopMargin    = pULItem->GetUpper();
833     nBottomMargin = pULItem->GetLower();
834 
835     const SvxPageItem* pPageItem = (const SvxPageItem*) &pParamSet->Get( ATTR_PAGE );
836     nPageUsage          = pPageItem->GetPageUsage();
837     bLandscape          = pPageItem->IsLandscape();
838     aFieldData.eNumType = pPageItem->GetNumType();
839 
840     bCenterHor = ((const SfxBoolItem&) pParamSet->Get(ATTR_PAGE_HORCENTER)).GetValue();
841     bCenterVer = ((const SfxBoolItem&) pParamSet->Get(ATTR_PAGE_VERCENTER)).GetValue();
842 
843     aPageSize = ((const SvxSizeItem&) pParamSet->Get(ATTR_PAGE_SIZE)).GetSize();
844     if ( !aPageSize.Width() || !aPageSize.Height() )
845     {
846         DBG_ERROR("PageSize Null ?!?!?");
847         aPageSize = SvxPaperInfo::GetPaperSize( PAPER_A4 );
848     }
849 
850     pBorderItem     = (const SvxBoxItem*)    &pParamSet->Get(ATTR_BORDER);
851     pBackgroundItem = (const SvxBrushItem*)  &pParamSet->Get(ATTR_BACKGROUND);
852     pShadowItem     = (const SvxShadowItem*) &pParamSet->Get(ATTR_SHADOW);
853 
854                                 // TabPage "Kopfzeile"
855 
856     aHdr.pLeft      = (const ScPageHFItem*) &pParamSet->Get(ATTR_PAGE_HEADERLEFT);      // Inhalt
857     aHdr.pRight     = (const ScPageHFItem*) &pParamSet->Get(ATTR_PAGE_HEADERRIGHT);
858 
859     const SvxSetItem* pHeaderSetItem;
860     const SfxItemSet* pHeaderSet = NULL;
861     if ( pParamSet->GetItemState( ATTR_PAGE_HEADERSET, sal_False,
862                             (const SfxPoolItem**)&pHeaderSetItem ) == SFX_ITEM_SET )
863     {
864         pHeaderSet = &pHeaderSetItem->GetItemSet();
865                                                         // Kopfzeile hat unteren Abstand
866         aHdr.nDistance  = ((const SvxULSpaceItem&) pHeaderSet->Get(ATTR_ULSPACE)).GetLower();
867     }
868     lcl_FillHFParam( aHdr, pHeaderSet );
869 
870                                 // TabPage "Fusszeile"
871 
872     aFtr.pLeft      = (const ScPageHFItem*) &pParamSet->Get(ATTR_PAGE_FOOTERLEFT);      // Inhalt
873     aFtr.pRight     = (const ScPageHFItem*) &pParamSet->Get(ATTR_PAGE_FOOTERRIGHT);
874 
875     const SvxSetItem* pFooterSetItem;
876     const SfxItemSet* pFooterSet = NULL;
877     if ( pParamSet->GetItemState( ATTR_PAGE_FOOTERSET, sal_False,
878                             (const SfxPoolItem**)&pFooterSetItem ) == SFX_ITEM_SET )
879     {
880         pFooterSet = &pFooterSetItem->GetItemSet();
881                                                         // Fusszeile hat oberen Abstand
882         aFtr.nDistance  = ((const SvxULSpaceItem&) pFooterSet->Get(ATTR_ULSPACE)).GetUpper();
883     }
884     lcl_FillHFParam( aFtr, pFooterSet );
885 
886     //------------------------------------------------------
887     // Table-/Area-Params aus einzelnen Items zusammenbauen:
888     //------------------------------------------------------
889     // TabPage "Tabelle"
890 
891     const SfxUInt16Item*     pScaleItem          = NULL;
892     const ScPageScaleToItem* pScaleToItem        = NULL;
893     const SfxUInt16Item*     pScaleToPagesItem   = NULL;
894     SfxItemState             eState;
895 
896     eState = pParamSet->GetItemState( ATTR_PAGE_SCALE, sal_False,
897                                       (const SfxPoolItem**)&pScaleItem );
898     if ( SFX_ITEM_DEFAULT == eState )
899         pScaleItem = (const SfxUInt16Item*)
900                     &pParamSet->GetPool()->GetDefaultItem( ATTR_PAGE_SCALE );
901 
902     eState = pParamSet->GetItemState( ATTR_PAGE_SCALETO, sal_False,
903                                       (const SfxPoolItem**)&pScaleToItem );
904     if ( SFX_ITEM_DEFAULT == eState )
905         pScaleToItem = (const ScPageScaleToItem*)
906                     &pParamSet->GetPool()->GetDefaultItem( ATTR_PAGE_SCALETO );
907 
908     eState = pParamSet->GetItemState( ATTR_PAGE_SCALETOPAGES, sal_False,
909                                       (const SfxPoolItem**)&pScaleToPagesItem );
910     if ( SFX_ITEM_DEFAULT == eState )
911         pScaleToPagesItem = (const SfxUInt16Item*)
912                     &pParamSet->GetPool()->GetDefaultItem( ATTR_PAGE_SCALETOPAGES );
913 
914     DBG_ASSERT( pScaleItem && pScaleToItem && pScaleToPagesItem, "Missing ScaleItem! :-/" );
915 
916     aTableParam.bCellContent    = sal_True;
917     aTableParam.bNotes          = GET_BOOL(pParamSet,ATTR_PAGE_NOTES);
918     aTableParam.bGrid           = GET_BOOL(pParamSet,ATTR_PAGE_GRID);
919     aTableParam.bHeaders        = GET_BOOL(pParamSet,ATTR_PAGE_HEADERS);
920     aTableParam.bFormulas       = GET_BOOL(pParamSet,ATTR_PAGE_FORMULAS);
921     aTableParam.bNullVals       = GET_BOOL(pParamSet,ATTR_PAGE_NULLVALS);
922     aTableParam.bCharts         = GET_SHOW(pParamSet,ATTR_PAGE_CHARTS);
923     aTableParam.bObjects        = GET_SHOW(pParamSet,ATTR_PAGE_OBJECTS);
924     aTableParam.bDrawings       = GET_SHOW(pParamSet,ATTR_PAGE_DRAWINGS);
925     aTableParam.bTopDown        = GET_BOOL(pParamSet,ATTR_PAGE_TOPDOWN);
926     aTableParam.bLeftRight      = !aTableParam.bLeftRight;
927     aTableParam.nFirstPageNo    = GET_USHORT(pParamSet,ATTR_PAGE_FIRSTPAGENO);
928     if (!aTableParam.nFirstPageNo)
929         aTableParam.nFirstPageNo = (sal_uInt16) nPageStart;     // von vorheriger Tabelle
930 
931     if ( pScaleItem && pScaleToItem && pScaleToPagesItem )
932     {
933         sal_uInt16  nScaleAll     = pScaleItem->GetValue();
934         sal_uInt16  nScaleToPages = pScaleToPagesItem->GetValue();
935 
936         aTableParam.bScaleNone      = (nScaleAll     == 100);
937         aTableParam.bScaleAll       = (nScaleAll      > 0  );
938         aTableParam.bScaleTo        = pScaleToItem->IsValid();
939         aTableParam.bScalePageNum   = (nScaleToPages  > 0  );
940         aTableParam.nScaleAll       = nScaleAll;
941         aTableParam.nScaleWidth     = pScaleToItem->GetWidth();
942         aTableParam.nScaleHeight    = pScaleToItem->GetHeight();
943         aTableParam.nScalePageNum   = nScaleToPages;
944     }
945     else
946     {
947         aTableParam.bScaleNone      = sal_True;
948         aTableParam.bScaleAll       = sal_False;
949         aTableParam.bScaleTo        = sal_False;
950         aTableParam.bScalePageNum   = sal_False;
951         aTableParam.nScaleAll       = 0;
952         aTableParam.nScaleWidth     = 0;
953         aTableParam.nScaleHeight    = 0;
954         aTableParam.nScalePageNum   = 0;
955     }
956 
957     //  skip empty pages only if options with that flag are passed
958     aTableParam.bSkipEmpty = pOptions && pOptions->GetSkipEmpty();
959     if ( pPageData )
960         aTableParam.bSkipEmpty = sal_False;
961     // Wenn pPageData gesetzt ist, interessieren fuer die Umbruch-Vorschau
962     // nur die Umbrueche, leere Seiten werden nicht speziell behandelt
963 
964     //------------------------------------------------------
965     // TabPage "Bereiche":
966     //------------------------------------------------------
967 
968     //! alle PrintAreas der Tabelle durchgehen !!!
969     const ScRange*  pPrintArea = pDoc->GetPrintRange( nPrintTab, 0 );
970     const ScRange*  pRepeatCol = pDoc->GetRepeatColRange( nPrintTab );
971     const ScRange*  pRepeatRow = pDoc->GetRepeatRowRange( nPrintTab );
972 
973     //  ATTR_PAGE_PRINTTABLES wird ignoriert
974 
975     if ( pUserArea )                // UserArea (Selektion) hat Vorrang
976     {
977         bPrintCurrentTable    =
978         aAreaParam.bPrintArea = sal_True;                   // Selektion
979         aAreaParam.aPrintArea = *pUserArea;
980 
981         //  Die Tabellen-Abfrage ist schon in DocShell::Print, hier immer
982         aAreaParam.aPrintArea.aStart.SetTab(nPrintTab);
983         aAreaParam.aPrintArea.aEnd.SetTab(nPrintTab);
984 
985 //      lcl_LimitRange( aAreaParam.aPrintArea, nPrintTab );         // ganze Zeilen/Spalten...
986     }
987     else if ( pDoc->HasPrintRange() )
988     {
989         if ( pPrintArea )                               // mindestens eine gesetzt ?
990         {
991             bPrintCurrentTable    =
992             aAreaParam.bPrintArea = sal_True;
993             aAreaParam.aPrintArea = *pPrintArea;
994 
995             bMultiArea = ( pDoc->GetPrintRangeCount(nPrintTab) > 1 );
996         }
997         else
998         {
999             // do not print hidden sheets with "Print entire sheet" flag
1000             bPrintCurrentTable = pDoc->IsPrintEntireSheet( nPrintTab ) && pDoc->IsVisible( nPrintTab );
1001             aAreaParam.bPrintArea = !bPrintCurrentTable;    // otherwise the table is always counted
1002         }
1003     }
1004     else
1005     {
1006         //  #74834# don't print hidden tables if there's no print range defined there
1007         if ( pDoc->IsVisible( nPrintTab ) )
1008         {
1009             aAreaParam.bPrintArea = sal_False;
1010             bPrintCurrentTable = sal_True;
1011         }
1012         else
1013         {
1014             aAreaParam.bPrintArea = sal_True;   // otherwise the table is always counted
1015             bPrintCurrentTable = sal_False;
1016         }
1017     }
1018 
1019     if ( pRepeatCol )
1020     {
1021         aAreaParam.bRepeatCol = sal_True;
1022         aAreaParam.aRepeatCol = *pRepeatCol;
1023         nRepeatStartCol = pRepeatCol->aStart.Col();
1024         nRepeatEndCol   = pRepeatCol->aEnd  .Col();
1025     }
1026     else
1027     {
1028         aAreaParam.bRepeatCol = sal_False;
1029         nRepeatStartCol = nRepeatEndCol = SCCOL_REPEAT_NONE;
1030     }
1031 
1032     if ( pRepeatRow )
1033     {
1034         aAreaParam.bRepeatRow = sal_True;
1035         aAreaParam.aRepeatRow = *pRepeatRow;
1036         nRepeatStartRow = pRepeatRow->aStart.Row();
1037         nRepeatEndRow   = pRepeatRow->aEnd  .Row();
1038     }
1039     else
1040     {
1041         aAreaParam.bRepeatRow = sal_False;
1042         nRepeatStartRow = nRepeatEndRow = SCROW_REPEAT_NONE;
1043     }
1044 
1045             //
1046             //  Seiten aufteilen
1047             //
1048 
1049     if (!bState)
1050     {
1051         nTabPages = CountPages();                                   // berechnet auch Zoom
1052         nTotalPages = nTabPages;
1053         nTotalPages += CountNotePages();
1054     }
1055     else
1056     {
1057         CalcPages();            // nur Umbrueche suchen
1058         CountNotePages();       // Notizen zaehlen, auch wenn Seitenzahl schon bekannt
1059     }
1060 
1061     if (nDocPages)
1062         aFieldData.nTotalPages = nDocPages;
1063     else
1064         aFieldData.nTotalPages = nTotalPages;
1065 
1066     SetDateTime( Date(), Time() );
1067 
1068     aFieldData.aTitle       = pDocShell->GetTitle();
1069     const INetURLObject& rURLObj = pDocShell->GetMedium()->GetURLObject();
1070     aFieldData.aLongDocName = rURLObj.GetMainURL( INetURLObject::DECODE_UNAMBIGUOUS );
1071     if ( aFieldData.aLongDocName.Len() )
1072         aFieldData.aShortDocName = rURLObj.GetName( INetURLObject::DECODE_UNAMBIGUOUS );
1073     else
1074         aFieldData.aShortDocName = aFieldData.aLongDocName = aFieldData.aTitle;
1075 
1076     //  Druckereinstellungen (Orientation, Paper) jetzt erst bei DoPrint
1077 }
1078 
GetDataSize() const1079 Size ScPrintFunc::GetDataSize() const
1080 {
1081     Size aSize = aPageSize;
1082     aSize.Width()  -= nLeftMargin + nRightMargin;
1083     aSize.Height() -= nTopMargin + nBottomMargin;
1084     aSize.Height() -= aHdr.nHeight + aFtr.nHeight;
1085     return aSize;
1086 }
1087 
GetScaleData(Size & rPhysSize,long & rDocHdr,long & rDocFtr)1088 void ScPrintFunc::GetScaleData( Size& rPhysSize, long& rDocHdr, long& rDocFtr )
1089 {
1090     rPhysSize = aPageSize;
1091     rPhysSize.Width()  -= nLeftMargin + nRightMargin;
1092     rPhysSize.Height() -= nTopMargin + nBottomMargin;
1093 
1094     rDocHdr = aHdr.nHeight;
1095     rDocFtr = aFtr.nHeight;
1096 }
1097 
SetDateTime(const Date & rDate,const Time & rTime)1098 void ScPrintFunc::SetDateTime( const Date& rDate, const Time& rTime )
1099 {
1100     aFieldData.aDate = rDate;
1101     aFieldData.aTime = rTime;
1102 }
1103 
lcl_DrawGraphic(const Graphic & rGraphic,OutputDevice * pOut,const Rectangle & rGrf,const Rectangle & rOut)1104 void lcl_DrawGraphic( const Graphic &rGraphic, OutputDevice *pOut,
1105                       const Rectangle &rGrf, const Rectangle &rOut )
1106 {
1107     const FASTBOOL bNotInside = !rOut.IsInside( rGrf );
1108     if ( bNotInside )
1109     {
1110         pOut->Push();
1111         pOut->IntersectClipRegion( rOut );
1112     }
1113 
1114     ((Graphic&)rGraphic).Draw( pOut, rGrf.TopLeft(), rGrf.GetSize() );
1115 
1116     if ( bNotInside )
1117         pOut->Pop();
1118 }
1119 
lcl_DrawGraphic(const SvxBrushItem & rBrush,OutputDevice * pOut,OutputDevice * pRefDev,const Rectangle & rOrg,const Rectangle & rOut)1120 void lcl_DrawGraphic( const SvxBrushItem &rBrush, OutputDevice *pOut, OutputDevice* pRefDev,
1121                         const Rectangle &rOrg, const Rectangle &rOut )
1122 {
1123     Size aGrfSize(0,0);
1124     const Graphic *pGraphic = rBrush.GetGraphic();
1125     SvxGraphicPosition ePos;
1126     if ( pGraphic && pGraphic->IsSupportedGraphic() )
1127     {
1128         const MapMode aMapMM( MAP_100TH_MM );
1129         if ( pGraphic->GetPrefMapMode().GetMapUnit() == MAP_PIXEL )
1130             aGrfSize = pRefDev->PixelToLogic( pGraphic->GetPrefSize(), aMapMM );
1131         else
1132             aGrfSize = OutputDevice::LogicToLogic( pGraphic->GetPrefSize(),
1133                                     pGraphic->GetPrefMapMode(), aMapMM );
1134         ePos = rBrush.GetGraphicPos();
1135     }
1136     else
1137         ePos = GPOS_NONE;
1138 
1139     Point aPos;
1140     Size aDrawSize = aGrfSize;
1141 
1142     FASTBOOL bDraw = sal_True;
1143 //  FASTBOOL bRetouche = sal_True;
1144     switch ( ePos )
1145     {
1146         case GPOS_LT: aPos = rOrg.TopLeft();
1147                       break;
1148         case GPOS_MT: aPos.Y() = rOrg.Top();
1149                       aPos.X() = rOrg.Left() + rOrg.GetSize().Width()/2 - aGrfSize.Width()/2;
1150                       break;
1151         case GPOS_RT: aPos.Y() = rOrg.Top();
1152                       aPos.X() = rOrg.Right() - aGrfSize.Width();
1153                       break;
1154 
1155         case GPOS_LM: aPos.Y() = rOrg.Top() + rOrg.GetSize().Height()/2 - aGrfSize.Height()/2;
1156                       aPos.X() = rOrg.Left();
1157                       break;
1158         case GPOS_MM: aPos.Y() = rOrg.Top() + rOrg.GetSize().Height()/2 - aGrfSize.Height()/2;
1159                       aPos.X() = rOrg.Left() + rOrg.GetSize().Width()/2 - aGrfSize.Width()/2;
1160                       break;
1161         case GPOS_RM: aPos.Y() = rOrg.Top() + rOrg.GetSize().Height()/2 - aGrfSize.Height()/2;
1162                       aPos.X() = rOrg.Right() - aGrfSize.Width();
1163                       break;
1164 
1165         case GPOS_LB: aPos.Y() = rOrg.Bottom() - aGrfSize.Height();
1166                       aPos.X() = rOrg.Left();
1167                       break;
1168         case GPOS_MB: aPos.Y() = rOrg.Bottom() - aGrfSize.Height();
1169                       aPos.X() = rOrg.Left() + rOrg.GetSize().Width()/2 - aGrfSize.Width()/2;
1170                       break;
1171         case GPOS_RB: aPos.Y() = rOrg.Bottom() - aGrfSize.Height();
1172                       aPos.X() = rOrg.Right() - aGrfSize.Width();
1173                       break;
1174 
1175         case GPOS_AREA:
1176                       aPos = rOrg.TopLeft();
1177                       aDrawSize = rOrg.GetSize();
1178 //                    bRetouche = sal_False;
1179                       break;
1180         case GPOS_TILED:
1181                     {
1182                         //  #104004# use GraphicObject::DrawTiled instead of an own loop
1183                         //  (pixel rounding is handled correctly, and a very small bitmap
1184                         //  is duplicated into a bigger one for better performance)
1185 
1186                         GraphicObject aObject( *pGraphic );
1187 
1188                         if( pOut->GetPDFWriter() &&
1189                             (aObject.GetType() == GRAPHIC_BITMAP || aObject.GetType() == GRAPHIC_DEFAULT) )
1190                         {
1191                             // #104004# For PDF export, every draw
1192                             // operation for bitmaps takes a noticeable
1193                             // amount of place (~50 characters). Thus,
1194                             // optimize between tile bitmap size and
1195                             // number of drawing operations here.
1196                             //
1197                             //                  A_out
1198                             // n_chars = k1 *  ---------- + k2 * A_bitmap
1199                             //                  A_bitmap
1200                             //
1201                             // minimum n_chars is obtained for (derive for
1202                             // A_bitmap, set to 0, take positive
1203                             // solution):
1204                             //                   k1
1205                             // A_bitmap = Sqrt( ---- A_out )
1206                             //                   k2
1207                             //
1208                             // where k1 is the number of chars per draw
1209                             // operation, and k2 is the number of chars
1210                             // per bitmap pixel. This is approximately 50
1211                             // and 7 for current PDF writer, respectively.
1212                             //
1213                             const double    k1( 50 );
1214                             const double    k2( 7 );
1215                             const Size      aSize( rOrg.GetSize() );
1216                             const double    Abitmap( k1/k2 * aSize.Width()*aSize.Height() );
1217 
1218                             aObject.DrawTiled( pOut, rOrg, aGrfSize, Size(0,0),
1219                                                NULL, GRFMGR_DRAW_STANDARD,
1220                                                ::std::max( 128, static_cast<int>( sqrt(sqrt( Abitmap)) + .5 ) ) );
1221                         }
1222                         else
1223                         {
1224                             aObject.DrawTiled( pOut, rOrg, aGrfSize, Size(0,0) );
1225                         }
1226 
1227                         bDraw = sal_False;
1228 //                      bRetouche = sal_False;
1229                     }
1230                     break;
1231 
1232         case GPOS_NONE:
1233                       bDraw = sal_False;
1234                       break;
1235 
1236         default: DBG_ASSERT( !pOut, "new Graphic position?" );
1237     }
1238     Rectangle aGrf( aPos,aDrawSize );
1239     if ( bDraw && aGrf.IsOver( rOut ) )
1240     {
1241         lcl_DrawGraphic( *pGraphic, pOut, aGrf, rOut );
1242     }
1243 }
1244 
1245 //  Rahmen wird nach innen gezeichnet
1246 
DrawBorder(long nScrX,long nScrY,long nScrW,long nScrH,const SvxBoxItem * pBorderData,const SvxBrushItem * pBackground,const SvxShadowItem * pShadow)1247 void ScPrintFunc::DrawBorder( long nScrX, long nScrY, long nScrW, long nScrH,
1248                                 const SvxBoxItem* pBorderData, const SvxBrushItem* pBackground,
1249                                 const SvxShadowItem* pShadow )
1250 {
1251     //!     direkte Ausgabe aus SvxBoxItem !!!
1252 
1253     if (pBorderData)
1254         if ( !pBorderData->GetTop() && !pBorderData->GetBottom() && !pBorderData->GetLeft() &&
1255                                         !pBorderData->GetRight() )
1256             pBorderData = NULL;
1257 
1258     if (!pBorderData && !pBackground && !pShadow)
1259         return;                                     // nichts zu tun
1260 
1261     long nLeft   = 0;
1262     long nRight  = 0;
1263     long nTop    = 0;
1264     long nBottom = 0;
1265 
1266     //  aFrameRect - aussen um die Umrandung, ohne Schatten
1267     if ( pShadow && pShadow->GetLocation() != SVX_SHADOW_NONE )
1268     {
1269         nLeft   += (long) ( pShadow->CalcShadowSpace(SHADOW_LEFT)   * nScaleX );
1270         nRight  += (long) ( pShadow->CalcShadowSpace(SHADOW_RIGHT)  * nScaleX );
1271         nTop    += (long) ( pShadow->CalcShadowSpace(SHADOW_TOP)    * nScaleY );
1272         nBottom += (long) ( pShadow->CalcShadowSpace(SHADOW_BOTTOM) * nScaleY );
1273     }
1274     Rectangle aFrameRect( Point(nScrX+nLeft, nScrY+nTop),
1275                           Size(nScrW-nLeft-nRight, nScrH-nTop-nBottom) );
1276 
1277     //  Mitte der Umrandung, um Linien ueber OutputData zu zeichnen:
1278     if (pBorderData)
1279     {
1280         nLeft   += (long) ( lcl_LineTotal(pBorderData->GetLeft())   * nScaleX / 2 );
1281         nRight  += (long) ( lcl_LineTotal(pBorderData->GetRight())  * nScaleX / 2 );
1282         nTop    += (long) ( lcl_LineTotal(pBorderData->GetTop())    * nScaleY / 2 );
1283         nBottom += (long) ( lcl_LineTotal(pBorderData->GetBottom()) * nScaleY / 2 );
1284     }
1285     long nEffHeight = nScrH - nTop - nBottom;
1286     long nEffWidth = nScrW - nLeft - nRight;
1287     if (nEffHeight<=0 || nEffWidth<=0)
1288         return;                                         // leer
1289 
1290     //  #105733# SvtAccessibilityOptions::GetIsForBorders is no longer used (always assumed sal_True)
1291     sal_Bool bCellContrast = bUseStyleColor &&
1292             Application::GetSettings().GetStyleSettings().GetHighContrastMode();
1293 
1294     if ( pBackground && !bCellContrast )
1295     {
1296 //      Rectangle aBackRect( Point(nScrX+nLeft, nScrY+nTop), Size(nEffWidth,nEffHeight) );
1297         if (pBackground->GetGraphicPos() != GPOS_NONE)
1298         {
1299             OutputDevice* pRefDev;
1300             if ( bIsRender )
1301                 pRefDev = pDev;                 // don't use printer for PDF
1302             else
1303                 pRefDev = pDoc->GetPrinter();   // use printer also for preview
1304 
1305             lcl_DrawGraphic( *pBackground, pDev, pRefDev, aFrameRect, aFrameRect );
1306         }
1307         else
1308         {
1309             pDev->SetFillColor(pBackground->GetColor());
1310             pDev->SetLineColor();
1311             pDev->DrawRect(aFrameRect);
1312         }
1313     }
1314 
1315     if ( pShadow && pShadow->GetLocation() != SVX_SHADOW_NONE )
1316     {
1317         if ( bCellContrast )
1318             pDev->SetFillColor( SC_MOD()->GetColorConfig().GetColorValue(svtools::FONTCOLOR).nColor );
1319         else
1320             pDev->SetFillColor(pShadow->GetColor());
1321         pDev->SetLineColor();
1322         long nShadowX = (long) ( pShadow->GetWidth() * nScaleX );
1323         long nShadowY = (long) ( pShadow->GetWidth() * nScaleY );
1324         switch (pShadow->GetLocation())
1325         {
1326             case SVX_SHADOW_TOPLEFT:
1327                 pDev->DrawRect( Rectangle(
1328                         aFrameRect.Left()-nShadowX, aFrameRect.Top()-nShadowY,
1329                         aFrameRect.Right()-nShadowX, aFrameRect.Top() ) );
1330                 pDev->DrawRect( Rectangle(
1331                         aFrameRect.Left()-nShadowX, aFrameRect.Top()-nShadowY,
1332                         aFrameRect.Left(), aFrameRect.Bottom()-nShadowY ) );
1333                 break;
1334             case SVX_SHADOW_TOPRIGHT:
1335                 pDev->DrawRect( Rectangle(
1336                         aFrameRect.Left()+nShadowX, aFrameRect.Top()-nShadowY,
1337                         aFrameRect.Right()+nShadowX, aFrameRect.Top() ) );
1338                 pDev->DrawRect( Rectangle(
1339                         aFrameRect.Right(), aFrameRect.Top()-nShadowY,
1340                         aFrameRect.Right()+nShadowX, aFrameRect.Bottom()-nShadowY ) );
1341                 break;
1342             case SVX_SHADOW_BOTTOMLEFT:
1343                 pDev->DrawRect( Rectangle(
1344                         aFrameRect.Left()-nShadowX, aFrameRect.Bottom(),
1345                         aFrameRect.Right()-nShadowX, aFrameRect.Bottom()+nShadowY ) );
1346                 pDev->DrawRect( Rectangle(
1347                         aFrameRect.Left()-nShadowX, aFrameRect.Top()+nShadowY,
1348                         aFrameRect.Left(), aFrameRect.Bottom()+nShadowY ) );
1349                 break;
1350             case SVX_SHADOW_BOTTOMRIGHT:
1351                 pDev->DrawRect( Rectangle(
1352                         aFrameRect.Left()+nShadowX, aFrameRect.Bottom(),
1353                         aFrameRect.Right()+nShadowX, aFrameRect.Bottom()+nShadowY ) );
1354                 pDev->DrawRect( Rectangle(
1355                         aFrameRect.Right(), aFrameRect.Top()+nShadowY,
1356                         aFrameRect.Right()+nShadowX, aFrameRect.Bottom()+nShadowY ) );
1357                 break;
1358             default:
1359             {
1360                 // added to avoid warnings
1361             }
1362         }
1363     }
1364 
1365     if (pBorderData)
1366     {
1367         ScDocument* pBorderDoc = new ScDocument( SCDOCMODE_UNDO );
1368         pBorderDoc->InitUndo( pDoc, 0,0, sal_True,sal_True );
1369         if (pBorderData)
1370             pBorderDoc->ApplyAttr( 0,0,0, *pBorderData );
1371 
1372         ScTableInfo aTabInfo;
1373         pBorderDoc->FillInfo( aTabInfo, 0,0, 0,0, 0,
1374                                             nScaleX, nScaleY, sal_False, sal_False );
1375         DBG_ASSERT(aTabInfo.mnArrCount,"nArrCount == 0");
1376 
1377         aTabInfo.mpRowInfo[1].nHeight = (sal_uInt16) nEffHeight;
1378         aTabInfo.mpRowInfo[0].pCellInfo[1].nWidth =
1379             aTabInfo.mpRowInfo[1].pCellInfo[1].nWidth = (sal_uInt16) nEffWidth;
1380 
1381         ScOutputData aOutputData( pDev, OUTTYPE_PRINTER, aTabInfo, pBorderDoc, 0,
1382                                     nScrX+nLeft, nScrY+nTop, 0,0, 0,0, nScaleX, nScaleY );
1383         aOutputData.SetUseStyleColor( bUseStyleColor );
1384 
1385 //      pDev->SetMapMode(aTwipMode);
1386 
1387         if (pBorderData)
1388             aOutputData.DrawFrame();
1389 
1390         delete pBorderDoc;
1391     }
1392 }
1393 
PrintColHdr(SCCOL nX1,SCCOL nX2,long nScrX,long nScrY)1394 void ScPrintFunc::PrintColHdr( SCCOL nX1, SCCOL nX2, long nScrX, long nScrY )
1395 {
1396     sal_Bool bLayoutRTL = pDoc->IsLayoutRTL( nPrintTab );
1397     long nLayoutSign = bLayoutRTL ? -1 : 1;
1398 
1399     Size aOnePixel = pDev->PixelToLogic(Size(1,1));
1400     long nOneX = aOnePixel.Width();
1401     long nOneY = aOnePixel.Height();
1402     SCCOL nCol;
1403 
1404     long nHeight = (long) (PRINT_HEADER_HEIGHT * nScaleY);
1405     long nEndY = nScrY + nHeight - nOneY;
1406 
1407     long nPosX = nScrX;
1408     if ( bLayoutRTL )
1409     {
1410         for (nCol=nX1; nCol<=nX2; nCol++)
1411             nPosX += (long)( pDoc->GetColWidth( nCol, nPrintTab ) * nScaleX );
1412     }
1413     else
1414         nPosX -= nOneX;
1415     long nPosY = nScrY - nOneY;
1416     String aText;
1417 
1418     for (nCol=nX1; nCol<=nX2; nCol++)
1419     {
1420         sal_uInt16 nDocW = pDoc->GetColWidth( nCol, nPrintTab );
1421         if (nDocW)
1422         {
1423             long nWidth = (long) (nDocW * nScaleX);
1424             long nEndX = nPosX + nWidth * nLayoutSign;
1425 
1426             pDev->DrawRect( Rectangle( nPosX,nPosY,nEndX,nEndY ) );
1427 
1428             aText = ::ScColToAlpha( nCol);
1429             long nTextWidth = pDev->GetTextWidth(aText);
1430             long nTextHeight = pDev->GetTextHeight();
1431             long nAddX = ( nWidth  - nTextWidth  ) / 2;
1432             long nAddY = ( nHeight - nTextHeight ) / 2;
1433             long nTextPosX = nPosX+nAddX;
1434             if ( bLayoutRTL )
1435                 nTextPosX -= nWidth;
1436             pDev->DrawText( Point( nTextPosX,nPosY+nAddY ), aText );
1437 
1438             nPosX = nEndX;
1439         }
1440     }
1441 }
1442 
PrintRowHdr(SCROW nY1,SCROW nY2,long nScrX,long nScrY)1443 void ScPrintFunc::PrintRowHdr( SCROW nY1, SCROW nY2, long nScrX, long nScrY )
1444 {
1445     Size aOnePixel = pDev->PixelToLogic(Size(1,1));
1446     long nOneX = aOnePixel.Width();
1447     long nOneY = aOnePixel.Height();
1448 
1449     sal_Bool bLayoutRTL = pDoc->IsLayoutRTL( nPrintTab );
1450 
1451     long nWidth = (long) (PRINT_HEADER_WIDTH * nScaleX);
1452     long nEndX = nScrX + nWidth;
1453     long nPosX = nScrX;
1454     if ( !bLayoutRTL )
1455     {
1456         nEndX -= nOneX;
1457         nPosX -= nOneX;
1458     }
1459     long nPosY = nScrY - nOneY;
1460     String aText;
1461 
1462     for (SCROW nRow=nY1; nRow<=nY2; nRow++)
1463     {
1464         sal_uInt16 nDocH = pDoc->GetRowHeight( nRow, nPrintTab );
1465         if (nDocH)
1466         {
1467             long nHeight = (long) (nDocH * nScaleY);
1468             long nEndY = nPosY + nHeight;
1469 
1470             pDev->DrawRect( Rectangle( nPosX,nPosY,nEndX,nEndY ) );
1471 
1472             aText = String::CreateFromInt32( nRow+1 );
1473             long nTextWidth = pDev->GetTextWidth(aText);
1474             long nTextHeight = pDev->GetTextHeight();
1475             long nAddX = ( nWidth  - nTextWidth  ) / 2;
1476             long nAddY = ( nHeight - nTextHeight ) / 2;
1477             pDev->DrawText( Point( nPosX+nAddX,nPosY+nAddY ), aText );
1478 
1479             nPosY = nEndY;
1480         }
1481     }
1482 }
1483 
LocateColHdr(SCCOL nX1,SCCOL nX2,long nScrX,long nScrY,sal_Bool bRepCol,ScPreviewLocationData & rLocationData)1484 void ScPrintFunc::LocateColHdr( SCCOL nX1, SCCOL nX2, long nScrX, long nScrY,
1485                                 sal_Bool bRepCol, ScPreviewLocationData& rLocationData )
1486 {
1487     Size aOnePixel = pDev->PixelToLogic(Size(1,1));
1488     long nOneX = aOnePixel.Width();
1489     long nOneY = aOnePixel.Height();
1490 
1491     long nHeight = (long) (PRINT_HEADER_HEIGHT * nScaleY);
1492     long nEndY = nScrY + nHeight - nOneY;
1493 
1494     long nPosX = nScrX - nOneX;
1495     for (SCCOL nCol=nX1; nCol<=nX2; nCol++)
1496     {
1497         sal_uInt16 nDocW = pDoc->GetColWidth( nCol, nPrintTab );
1498         if (nDocW)
1499             nPosX += (long) (nDocW * nScaleX);
1500     }
1501     Rectangle aCellRect( nScrX, nScrY, nPosX, nEndY );
1502     rLocationData.AddColHeaders( aCellRect, nX1, nX2, bRepCol );
1503 }
1504 
LocateRowHdr(SCROW nY1,SCROW nY2,long nScrX,long nScrY,sal_Bool bRepRow,ScPreviewLocationData & rLocationData)1505 void ScPrintFunc::LocateRowHdr( SCROW nY1, SCROW nY2, long nScrX, long nScrY,
1506                                 sal_Bool bRepRow, ScPreviewLocationData& rLocationData )
1507 {
1508     Size aOnePixel = pDev->PixelToLogic(Size(1,1));
1509     long nOneX = aOnePixel.Width();
1510     long nOneY = aOnePixel.Height();
1511 
1512     sal_Bool bLayoutRTL = pDoc->IsLayoutRTL( nPrintTab );
1513 
1514     long nWidth = (long) (PRINT_HEADER_WIDTH * nScaleX);
1515     long nEndX = nScrX + nWidth;
1516     if ( !bLayoutRTL )
1517         nEndX -= nOneX;
1518 
1519     long nPosY = nScrY - nOneY;
1520     nPosY += pDoc->GetScaledRowHeight( nY1, nY2, nPrintTab, nScaleY);
1521     Rectangle aCellRect( nScrX, nScrY, nEndX, nPosY );
1522     rLocationData.AddRowHeaders( aCellRect, nY1, nY2, bRepRow );
1523 }
1524 
LocateArea(SCCOL nX1,SCROW nY1,SCCOL nX2,SCROW nY2,long nScrX,long nScrY,sal_Bool bRepCol,sal_Bool bRepRow,ScPreviewLocationData & rLocationData)1525 void ScPrintFunc::LocateArea( SCCOL nX1, SCROW nY1, SCCOL nX2, SCROW nY2,
1526                                 long nScrX, long nScrY, sal_Bool bRepCol, sal_Bool bRepRow,
1527                                 ScPreviewLocationData& rLocationData )
1528 {
1529     //  get MapMode for drawing objects (same MapMode as in ScOutputData::PrintDrawingLayer)
1530 
1531     Point aLogPos = OutputDevice::LogicToLogic(Point(nScrX,nScrY), aOffsetMode, aLogicMode);
1532     long nLogStX = aLogPos.X();
1533     long nLogStY = aLogPos.Y();
1534 
1535     SCCOL nCol;
1536     Point aTwipOffset;
1537     for (nCol=0; nCol<nX1; nCol++)
1538         aTwipOffset.X() -= pDoc->GetColWidth( nCol, nPrintTab );
1539     aTwipOffset.Y() -= pDoc->GetRowHeight( 0, nY1-1, nPrintTab );
1540 
1541     Point aMMOffset( aTwipOffset );
1542     aMMOffset.X() = (long)(aMMOffset.X() * HMM_PER_TWIPS);
1543     aMMOffset.Y() = (long)(aMMOffset.Y() * HMM_PER_TWIPS);
1544     aMMOffset += Point( nLogStX, nLogStY );
1545     MapMode aDrawMapMode( MAP_100TH_MM, aMMOffset, aLogicMode.GetScaleX(), aLogicMode.GetScaleY() );
1546 
1547     //  get pixel rectangle
1548 
1549     Size aOnePixel = pDev->PixelToLogic(Size(1,1));
1550     long nOneX = aOnePixel.Width();
1551     long nOneY = aOnePixel.Height();
1552 
1553     long nPosX = nScrX - nOneX;
1554     for (nCol=nX1; nCol<=nX2; nCol++)
1555     {
1556         sal_uInt16 nDocW = pDoc->GetColWidth( nCol, nPrintTab );
1557         if (nDocW)
1558             nPosX += (long) (nDocW * nScaleX);
1559     }
1560 
1561     long nPosY = nScrY - nOneY;
1562     nPosY += pDoc->GetScaledRowHeight( nY1, nY2, nPrintTab, nScaleY);
1563     Rectangle aCellRect( nScrX, nScrY, nPosX, nPosY );
1564     rLocationData.AddCellRange( aCellRect, ScRange( nX1,nY1,nPrintTab, nX2,nY2,nPrintTab ),
1565                                 bRepCol, bRepRow, aDrawMapMode );
1566 }
1567 
PrintArea(SCCOL nX1,SCROW nY1,SCCOL nX2,SCROW nY2,long nScrX,long nScrY,sal_Bool bShLeft,sal_Bool bShTop,sal_Bool bShRight,sal_Bool bShBottom)1568 void ScPrintFunc::PrintArea( SCCOL nX1, SCROW nY1, SCCOL nX2, SCROW nY2,
1569                                 long nScrX, long nScrY,
1570                                 sal_Bool bShLeft, sal_Bool bShTop, sal_Bool bShRight, sal_Bool bShBottom )
1571 {
1572     // #i47547# nothing to do if the end of the print area is before the end of
1573     // the repeat columns/rows (don't use negative size for ScOutputData)
1574     if ( nX2 < nX1 || nY2 < nY1 )
1575         return;
1576 
1577                             //!     Flag bei FillInfo uebergeben !!!!!
1578     ScRange aERange;
1579     sal_Bool bEmbed = pDoc->IsEmbedded();
1580     if (bEmbed)
1581     {
1582         pDoc->GetEmbedded(aERange);
1583         pDoc->ResetEmbedded();
1584     }
1585 
1586     Point aPos = OutputDevice::LogicToLogic(Point(nScrX,nScrY), aOffsetMode, aLogicMode);
1587     long nLogStX = aPos.X();
1588     long nLogStY = aPos.Y();
1589 
1590                     //  Daten zusammenstellen
1591 
1592     ScTableInfo aTabInfo;
1593     pDoc->FillInfo( aTabInfo, nX1, nY1, nX2, nY2, nPrintTab,
1594                                         nScaleX, nScaleY, sal_True, aTableParam.bFormulas );
1595     lcl_HidePrint( aTabInfo, nX1, nX2 );
1596 
1597     if (bEmbed)
1598         pDoc->SetEmbedded(aERange);
1599 
1600     ScOutputData aOutputData( pDev, OUTTYPE_PRINTER, aTabInfo, pDoc, nPrintTab,
1601                                 nScrX, nScrY, nX1, nY1, nX2, nY2, nScaleX, nScaleY );
1602 
1603     // #114135#
1604     aOutputData.SetDrawView( pDrawView );
1605 
1606     // test if all paint parts are hidden, then a paint is not necessary at all
1607     const Point aMMOffset(aOutputData.PrePrintDrawingLayer(nLogStX, nLogStY));
1608     const bool bHideAllDrawingLayer( pDrawView && pDrawView->getHideOle() && pDrawView->getHideChart()
1609             && pDrawView->getHideDraw() && pDrawView->getHideFormControl() );
1610 
1611     if(!bHideAllDrawingLayer)
1612     {
1613         pDev->SetMapMode(aLogicMode);
1614         //  hier kein Clipping setzen (Mapmode wird verschoben)
1615 
1616         // #i72502#
1617         aOutputData.PrintDrawingLayer(SC_LAYER_BACK, aMMOffset);
1618     }
1619 
1620     pDev->SetMapMode(aOffsetMode);
1621 
1622     aOutputData.SetShowFormulas( aTableParam.bFormulas );
1623     aOutputData.SetShowNullValues( aTableParam.bNullVals );
1624     aOutputData.SetUseStyleColor( bUseStyleColor );
1625 
1626     Color aGridColor( COL_BLACK );
1627     if ( bUseStyleColor )
1628         aGridColor.SetColor( SC_MOD()->GetColorConfig().GetColorValue(svtools::FONTCOLOR).nColor );
1629     aOutputData.SetGridColor( aGridColor );
1630 
1631     if ( !pPrinter )
1632     {
1633         OutputDevice* pRefDev = pDoc->GetPrinter();     // auch fuer Preview den Drucker nehmen
1634         Fraction aPrintFrac( nZoom, 100 );              // ohne nManualZoom
1635         //  MapMode, wie er beim Drucken herauskommen wuerde:
1636         pRefDev->SetMapMode( MapMode( MAP_100TH_MM, Point(), aPrintFrac, aPrintFrac ) );
1637 
1638         //  when rendering (PDF), don't use printer as ref device, but printer's MapMode
1639         //  has to be set anyway, as charts still use it (#106409#)
1640         if ( !bIsRender )
1641             aOutputData.SetRefDevice( pRefDev );
1642     }
1643 
1644 //  aOutputData.SetMetaFileMode(sal_True);
1645     if( aTableParam.bCellContent )
1646         aOutputData.DrawBackground();
1647 
1648     pDev->SetClipRegion( Rectangle( aPos, Size( aOutputData.GetScrW(), aOutputData.GetScrH() ) ) );
1649     pDev->SetClipRegion();
1650 
1651 //  aOutputData.SetMetaFileMode(sal_False);
1652     if( aTableParam.bCellContent )
1653     {
1654         aOutputData.DrawExtraShadow( bShLeft, bShTop, bShRight, bShBottom );
1655         aOutputData.DrawFrame();
1656         aOutputData.DrawStrings();
1657 
1658     //  pDev->SetMapMode(aLogicMode);
1659         aOutputData.DrawEdit(sal_False);
1660     }
1661 
1662 //  pDev->SetMapMode(aOffsetMode);
1663     if (aTableParam.bGrid)
1664         aOutputData.DrawGrid( sal_True, sal_False );    // keine Seitenumbrueche
1665 
1666 /*!!!!!!!!!!!       Notizen in Tabelle markieren ??????????????????????????
1667 
1668     if (aTableParam.bNotes)
1669     {
1670         pDev->SetMapMode(aOffsetMode);
1671         aOutputData.PrintNoteMarks(aNotePosList);
1672         pDev->SetMapMode(aLogicMode);
1673     }
1674 */
1675 
1676     aOutputData.AddPDFNotes();      // has no effect if not rendering PDF with notes enabled
1677 
1678 //  pDev->SetMapMode(aDrawMode);
1679 
1680     // test if all paint parts are hidden, then a paint is not necessary at all
1681     if(!bHideAllDrawingLayer)
1682     {
1683         // #i72502#
1684         aOutputData.PrintDrawingLayer(SC_LAYER_FRONT, aMMOffset);
1685     }
1686 
1687     // #i72502#
1688     aOutputData.PrintDrawingLayer(SC_LAYER_INTERN, aMMOffset);
1689     aOutputData.PostPrintDrawingLayer(aMMOffset); // #i74768#
1690 }
1691 
IsMirror(long nPageNo)1692 sal_Bool ScPrintFunc::IsMirror( long nPageNo )          // Raender spiegeln ?
1693 {
1694     SvxPageUsage eUsage = (SvxPageUsage) ( nPageUsage & 0x000f );
1695     return ( eUsage == SVX_PAGE_MIRROR && (nPageNo & 1) );
1696 }
1697 
IsLeft(long nPageNo)1698 sal_Bool ScPrintFunc::IsLeft( long nPageNo )            // linke Fussnoten ?
1699 {
1700     SvxPageUsage eUsage = (SvxPageUsage) ( nPageUsage & 0x000f );
1701     sal_Bool bLeft;
1702     if (eUsage == SVX_PAGE_LEFT)
1703         bLeft = sal_True;
1704     else if (eUsage == SVX_PAGE_RIGHT)
1705         bLeft = sal_False;
1706     else
1707         bLeft = (nPageNo & 1) != 0;
1708     return bLeft;
1709 }
1710 
MakeTableString()1711 void ScPrintFunc::MakeTableString()
1712 {
1713     pDoc->GetName( nPrintTab, aFieldData.aTabName );
1714 }
1715 
MakeEditEngine()1716 void ScPrintFunc::MakeEditEngine()
1717 {
1718     if (!pEditEngine)
1719     {
1720         //  can't use document's edit engine pool here,
1721         //  because pool must have twips as default metric
1722         pEditEngine = new ScHeaderEditEngine( EditEngine::CreatePool(), sal_True );
1723 
1724         pEditEngine->EnableUndo(sal_False);
1725         pEditEngine->SetRefDevice( pDev );
1726         pEditEngine->SetWordDelimiters(
1727                 ScEditUtil::ModifyDelimiters( pEditEngine->GetWordDelimiters() ) );
1728         pEditEngine->SetControlWord( pEditEngine->GetControlWord() & ~EE_CNTRL_RTFSTYLESHEETS );
1729         pDoc->ApplyAsianEditSettings( *pEditEngine );
1730         pEditEngine->EnableAutoColor( bUseStyleColor );
1731 
1732         //  Default-Set fuer Ausrichtung
1733         pEditDefaults = new SfxItemSet( pEditEngine->GetEmptyItemSet() );
1734 
1735         const ScPatternAttr& rPattern = (const ScPatternAttr&)pDoc->GetPool()->GetDefaultItem(ATTR_PATTERN);
1736         rPattern.FillEditItemSet( pEditDefaults );
1737         //  FillEditItemSet adjusts font height to 1/100th mm,
1738         //  but for header/footer twips is needed, as in the PatternAttr:
1739         pEditDefaults->Put( rPattern.GetItem(ATTR_FONT_HEIGHT), EE_CHAR_FONTHEIGHT );
1740         pEditDefaults->Put( rPattern.GetItem(ATTR_CJK_FONT_HEIGHT), EE_CHAR_FONTHEIGHT_CJK );
1741         pEditDefaults->Put( rPattern.GetItem(ATTR_CTL_FONT_HEIGHT), EE_CHAR_FONTHEIGHT_CTL );
1742         //  #69193# dont use font color, because background color is not used
1743         //! there's no way to set the background for note pages
1744         pEditDefaults->ClearItem( EE_CHAR_COLOR );
1745         if (ScGlobal::IsSystemRTL())
1746             pEditDefaults->Put( SvxFrameDirectionItem( FRMDIR_HORI_RIGHT_TOP, EE_PARA_WRITINGDIR ) );
1747     }
1748 
1749     pEditEngine->SetData( aFieldData );     // Seitennummer etc. setzen
1750 }
1751 
1752 //  nStartY = logic
PrintHF(long nPageNo,sal_Bool bHeader,long nStartY,sal_Bool bDoPrint,ScPreviewLocationData * pLocationData)1753 void ScPrintFunc::PrintHF( long nPageNo, sal_Bool bHeader, long nStartY,
1754                             sal_Bool bDoPrint, ScPreviewLocationData* pLocationData )
1755 {
1756     const ScPrintHFParam& rParam = bHeader ? aHdr : aFtr;
1757 
1758     pDev->SetMapMode( aTwipMode );          // Kopf-/Fusszeilen in Twips
1759 
1760     sal_Bool bLeft = IsLeft(nPageNo) && !rParam.bShared;
1761     const ScPageHFItem* pHFItem = bLeft ? rParam.pLeft : rParam.pRight;
1762 
1763     long nLineStartX = aPageRect.Left()  + rParam.nLeft;
1764     long nLineEndX   = aPageRect.Right() - rParam.nRight;
1765     long nLineWidth  = nLineEndX - nLineStartX + 1;
1766 
1767     //  Edit-Engine
1768 
1769     Point aStart( nLineStartX, nStartY );
1770     Size aPaperSize( nLineWidth, rParam.nHeight-rParam.nDistance );
1771     if ( rParam.pBorder )
1772     {
1773         long nLeft = lcl_LineTotal( rParam.pBorder->GetLeft() ) + rParam.pBorder->GetDistance(BOX_LINE_LEFT);
1774         long nTop = lcl_LineTotal( rParam.pBorder->GetTop() ) + rParam.pBorder->GetDistance(BOX_LINE_TOP);
1775         aStart.X() += nLeft;
1776         aStart.Y() += nTop;
1777         aPaperSize.Width() -= nLeft + lcl_LineTotal( rParam.pBorder->GetRight() ) + rParam.pBorder->GetDistance(BOX_LINE_RIGHT);
1778         aPaperSize.Height() -= nTop + lcl_LineTotal( rParam.pBorder->GetBottom() ) + rParam.pBorder->GetDistance(BOX_LINE_BOTTOM);
1779     }
1780 
1781     if ( rParam.pShadow && rParam.pShadow->GetLocation() != SVX_SHADOW_NONE )
1782     {
1783         long nLeft  = rParam.pShadow->CalcShadowSpace(SHADOW_LEFT);
1784         long nTop   = rParam.pShadow->CalcShadowSpace(SHADOW_TOP);
1785         aStart.X() += nLeft;
1786         aStart.Y() += nTop;
1787         aPaperSize.Width() -= nLeft + rParam.pShadow->CalcShadowSpace(SHADOW_RIGHT);
1788         aPaperSize.Height() -= nTop + rParam.pShadow->CalcShadowSpace(SHADOW_BOTTOM);
1789     }
1790 
1791     aFieldData.nPageNo = nPageNo+aTableParam.nFirstPageNo;
1792     MakeEditEngine();
1793 
1794     pEditEngine->SetPaperSize(aPaperSize);
1795     const EditTextObject* pObject;
1796 
1797     //  Rahmen / Hintergrund
1798 
1799     Point aBorderStart( nLineStartX, nStartY );
1800     Size aBorderSize( nLineWidth, rParam.nHeight-rParam.nDistance );
1801     if ( rParam.bDynamic )
1802     {
1803         //  hier nochmal anpassen, wegen geraden/ungeraden Kopf/Fusszeilen
1804         //  und evtl. anderen Umbruechen durch Variablen (Seitennummer etc.)
1805 
1806         long nMaxHeight = 0;
1807         nMaxHeight = Max( nMaxHeight, TextHeight( pHFItem->GetLeftArea() ) );
1808         nMaxHeight = Max( nMaxHeight, TextHeight( pHFItem->GetCenterArea() ) );
1809         nMaxHeight = Max( nMaxHeight, TextHeight( pHFItem->GetRightArea() ) );
1810         if (rParam.pBorder)
1811             nMaxHeight += lcl_LineTotal( rParam.pBorder->GetTop() ) +
1812                           lcl_LineTotal( rParam.pBorder->GetBottom() ) +
1813                                     rParam.pBorder->GetDistance(BOX_LINE_TOP) +
1814                                     rParam.pBorder->GetDistance(BOX_LINE_BOTTOM);
1815         if (rParam.pShadow && rParam.pShadow->GetLocation() != SVX_SHADOW_NONE)
1816             nMaxHeight += rParam.pShadow->CalcShadowSpace(SHADOW_TOP) +
1817                           rParam.pShadow->CalcShadowSpace(SHADOW_BOTTOM);
1818 
1819         if (nMaxHeight < rParam.nManHeight-rParam.nDistance)
1820             nMaxHeight = rParam.nManHeight-rParam.nDistance;        // eingestelltes Minimum
1821 
1822         aBorderSize.Height() = nMaxHeight;
1823     }
1824 
1825     if ( bDoPrint )
1826     {
1827         double nOldScaleX = nScaleX;
1828         double nOldScaleY = nScaleY;
1829         nScaleX = nScaleY = 1.0;            // direkt in Twips ausgeben
1830         DrawBorder( aBorderStart.X(), aBorderStart.Y(), aBorderSize.Width(), aBorderSize.Height(),
1831                         rParam.pBorder, rParam.pBack, rParam.pShadow );
1832         nScaleX = nOldScaleX;
1833         nScaleY = nOldScaleY;
1834 
1835         //  Clipping fuer Text
1836 
1837         pDev->SetClipRegion( Rectangle( aStart, aPaperSize ) );
1838 
1839         //  links
1840 
1841         pObject = pHFItem->GetLeftArea();
1842         if (pObject)
1843         {
1844             pEditDefaults->Put( SvxAdjustItem( SVX_ADJUST_LEFT, EE_PARA_JUST ) );
1845             pEditEngine->SetTextNewDefaults( *pObject, *pEditDefaults, sal_False );
1846             Point aDraw = aStart;
1847             long nDif = aPaperSize.Height() - (long) pEditEngine->GetTextHeight();
1848             if (nDif > 0)
1849                 aDraw.Y() += nDif / 2;
1850             pEditEngine->Draw( pDev, aDraw, 0 );
1851         }
1852 
1853         //  Mitte
1854 
1855         pObject = pHFItem->GetCenterArea();
1856         if (pObject)
1857         {
1858             pEditDefaults->Put( SvxAdjustItem( SVX_ADJUST_CENTER, EE_PARA_JUST ) );
1859             pEditEngine->SetTextNewDefaults( *pObject, *pEditDefaults, sal_False );
1860             Point aDraw = aStart;
1861             long nDif = aPaperSize.Height() - (long) pEditEngine->GetTextHeight();
1862             if (nDif > 0)
1863                 aDraw.Y() += nDif / 2;
1864             pEditEngine->Draw( pDev, aDraw, 0 );
1865         }
1866 
1867         //  rechts
1868 
1869         pObject = pHFItem->GetRightArea();
1870         if (pObject)
1871         {
1872             pEditDefaults->Put( SvxAdjustItem( SVX_ADJUST_RIGHT, EE_PARA_JUST ) );
1873             pEditEngine->SetTextNewDefaults( *pObject, *pEditDefaults, sal_False );
1874             Point aDraw = aStart;
1875             long nDif = aPaperSize.Height() - (long) pEditEngine->GetTextHeight();
1876             if (nDif > 0)
1877                 aDraw.Y() += nDif / 2;
1878             pEditEngine->Draw( pDev, aDraw, 0 );
1879         }
1880 
1881         pDev->SetClipRegion();
1882     }
1883 
1884     if ( pLocationData )
1885     {
1886         Rectangle aHeaderRect( aBorderStart, aBorderSize );
1887         pLocationData->AddHeaderFooter( aHeaderRect, bHeader, bLeft );
1888     }
1889 }
1890 
DoNotes(long nNoteStart,sal_Bool bDoPrint,ScPreviewLocationData * pLocationData)1891 long ScPrintFunc::DoNotes( long nNoteStart, sal_Bool bDoPrint, ScPreviewLocationData* pLocationData )
1892 {
1893     if (bDoPrint)
1894         pDev->SetMapMode(aTwipMode);
1895 
1896     MakeEditEngine();
1897     pEditDefaults->Put( SvxAdjustItem( SVX_ADJUST_LEFT, EE_PARA_JUST ) );
1898     pEditEngine->SetDefaults( *pEditDefaults );
1899 
1900     Font aMarkFont;
1901     ScAutoFontColorMode eColorMode = bUseStyleColor ? SC_AUTOCOL_DISPLAY : SC_AUTOCOL_PRINT;
1902     ((const ScPatternAttr&)pDoc->GetPool()->GetDefaultItem(ATTR_PATTERN)).GetFont( aMarkFont, eColorMode );
1903 //? aMarkFont.SetWeight( WEIGHT_BOLD );
1904     pDev->SetFont( aMarkFont );
1905     long nMarkLen = pDev->GetTextWidth(
1906             String::CreateFromAscii(RTL_CONSTASCII_STRINGPARAM("GW99999:")));
1907     // ohne Space, weil's eh selten so weit kommt
1908 
1909     Size aDataSize = aPageRect.GetSize();
1910     if ( nMarkLen > aDataSize.Width() / 2 )     // alles viel zu klein?
1911         nMarkLen = aDataSize.Width() / 2;       // Seite bruederlich aufteilen
1912     aDataSize.Width() -= nMarkLen;
1913 
1914     pEditEngine->SetPaperSize( aDataSize );
1915     long nPosX = aPageRect.Left() + nMarkLen;
1916     long nPosY = aPageRect.Top();
1917 
1918     long nCount = 0;
1919     sal_Bool bOk;
1920     do
1921     {
1922         bOk = sal_False;
1923         ScAddress* pPos = (ScAddress*) aNotePosList.GetObject( nNoteStart+nCount );
1924         if (pPos)
1925         {
1926             ScBaseCell* pCell = pDoc->GetCell( *pPos);
1927             if( const ScPostIt* pNote = pCell->GetNote() )
1928             {
1929                 if(const EditTextObject *pEditText = pNote->GetEditTextObject())
1930                     pEditEngine->SetText(*pEditText);
1931                 long nTextHeight = pEditEngine->GetTextHeight();
1932                 if ( nPosY + nTextHeight < aPageRect.Bottom() )
1933                 {
1934                     if (bDoPrint)
1935                     {
1936                         pEditEngine->Draw( pDev, Point( nPosX, nPosY ), 0 );
1937 
1938                         String aMarkStr;
1939                         pPos->Format( aMarkStr, SCA_VALID, pDoc, pDoc->GetAddressConvention() );
1940                         aMarkStr += ':';
1941 
1942                         //  Zellposition auch per EditEngine, damit die Position stimmt
1943                         pEditEngine->SetText(aMarkStr);
1944                         pEditEngine->Draw( pDev, Point( aPageRect.Left(), nPosY ), 0 );
1945                     }
1946 
1947                     if ( pLocationData )
1948                     {
1949                         Rectangle aTextRect( Point( nPosX, nPosY ), Size( aDataSize.Width(), nTextHeight ) );
1950                         pLocationData->AddNoteText( aTextRect, *pPos );
1951                         Rectangle aMarkRect( Point( aPageRect.Left(), nPosY ), Size( nMarkLen, nTextHeight ) );
1952                         pLocationData->AddNoteMark( aMarkRect, *pPos );
1953                     }
1954 
1955                     nPosY += nTextHeight;
1956                     nPosY += 200;                   // Abstand
1957                     ++nCount;
1958                     bOk = sal_True;
1959                 }
1960             }
1961         }
1962     }
1963     while (bOk);
1964 
1965     return nCount;
1966 }
1967 
PrintNotes(long nPageNo,long nNoteStart,sal_Bool bDoPrint,ScPreviewLocationData * pLocationData)1968 long ScPrintFunc::PrintNotes( long nPageNo, long nNoteStart, sal_Bool bDoPrint, ScPreviewLocationData* pLocationData )
1969 {
1970     if ( nNoteStart >= (long) aNotePosList.Count() || !aTableParam.bNotes )
1971         return 0;
1972 
1973     if ( bDoPrint && bClearWin )
1974     {
1975         //! mit PrintPage zusammenfassen !!!
1976 
1977         Color aBackgroundColor( COL_WHITE );
1978         if ( bUseStyleColor )
1979             aBackgroundColor.SetColor( SC_MOD()->GetColorConfig().GetColorValue(svtools::DOCCOLOR).nColor );
1980 
1981         pDev->SetMapMode(aOffsetMode);
1982         pDev->SetLineColor();
1983         pDev->SetFillColor(aBackgroundColor);
1984         pDev->DrawRect(Rectangle(Point(),
1985                 Size((long)(aPageSize.Width() * nScaleX * 100 / nZoom),
1986                      (long)(aPageSize.Height() * nScaleY * 100 / nZoom))));
1987     }
1988 
1989 
1990     //      aPageRect auf linke / rechte Seiten anpassen
1991 
1992     Rectangle aTempRect = Rectangle( Point(), aPageSize );
1993     if (IsMirror(nPageNo))
1994     {
1995         aPageRect.Left()  = ( aTempRect.Left()  + nRightMargin ) * 100 / nZoom;
1996         aPageRect.Right() = ( aTempRect.Right() - nLeftMargin  ) * 100 / nZoom;
1997     }
1998     else
1999     {
2000         aPageRect.Left()  = ( aTempRect.Left()  + nLeftMargin  ) * 100 / nZoom;
2001         aPageRect.Right() = ( aTempRect.Right() - nRightMargin ) * 100 / nZoom;
2002     }
2003 
2004     if ( pPrinter && bDoPrint )
2005     {
2006         DBG_ERROR( "StartPage does not exist anymore" );
2007         // pPrinter->StartPage();
2008     }
2009 
2010     if ( bDoPrint || pLocationData )
2011     {
2012         //  Kopf- und Fusszeilen
2013 
2014         if (aHdr.bEnable)
2015         {
2016             long nHeaderY = aPageRect.Top()-aHdr.nHeight;
2017             PrintHF( nPageNo, sal_True, nHeaderY, bDoPrint, pLocationData );
2018         }
2019         if (aFtr.bEnable)
2020         {
2021             long nFooterY = aPageRect.Bottom()+aFtr.nDistance;
2022             PrintHF( nPageNo, sal_False, nFooterY, bDoPrint, pLocationData );
2023         }
2024     }
2025 
2026     long nCount = DoNotes( nNoteStart, bDoPrint, pLocationData );
2027 
2028     if ( pPrinter && bDoPrint )
2029     {
2030         DBG_ERROR( "EndPage does not exist anymore" );
2031         // pPrinter->EndPage();
2032     }
2033 
2034     return nCount;
2035 }
2036 
PrintPage(long nPageNo,SCCOL nX1,SCROW nY1,SCCOL nX2,SCROW nY2,sal_Bool bDoPrint,ScPreviewLocationData * pLocationData)2037 void ScPrintFunc::PrintPage( long nPageNo, SCCOL nX1, SCROW nY1, SCCOL nX2, SCROW nY2,
2038                                 sal_Bool bDoPrint, ScPreviewLocationData* pLocationData )
2039 {
2040     sal_Bool bLayoutRTL = pDoc->IsLayoutRTL( nPrintTab );
2041     long nLayoutSign = bLayoutRTL ? -1 : 1;
2042 
2043     //  nPageNo is the page number within all sheets of one "start page" setting
2044 
2045     if ( bClearWin && bDoPrint )
2046     {
2047         //  muss genau zum Zeichnen des Rahmens in preview.cxx passen !!!
2048 
2049         Color aBackgroundColor( COL_WHITE );
2050         if ( bUseStyleColor )
2051             aBackgroundColor.SetColor( SC_MOD()->GetColorConfig().GetColorValue(svtools::DOCCOLOR).nColor );
2052 
2053         pDev->SetMapMode(aOffsetMode);
2054         pDev->SetLineColor();
2055         pDev->SetFillColor(aBackgroundColor);
2056         pDev->DrawRect(Rectangle(Point(),
2057                 Size((long)(aPageSize.Width() * nScaleX * 100 / nZoom),
2058                      (long)(aPageSize.Height() * nScaleY * 100 / nZoom))));
2059     }
2060 
2061 
2062     //      aPageRect auf linke / rechte Seiten anpassen
2063 
2064     Rectangle aTempRect = Rectangle( Point(), aPageSize );
2065     if (IsMirror(nPageNo))
2066     {
2067         aPageRect.Left()  = ( aTempRect.Left()  + nRightMargin ) * 100 / nZoom;
2068         aPageRect.Right() = ( aTempRect.Right() - nLeftMargin  ) * 100 / nZoom;
2069     }
2070     else
2071     {
2072         aPageRect.Left()  = ( aTempRect.Left()  + nLeftMargin  ) * 100 / nZoom;
2073         aPageRect.Right() = ( aTempRect.Right() - nRightMargin ) * 100 / nZoom;
2074     }
2075 
2076     if ( aAreaParam.bRepeatCol )
2077         if ( nX1 > nRepeatStartCol && nX1 <= nRepeatEndCol )
2078             nX1 = nRepeatEndCol + 1;
2079     sal_Bool bDoRepCol = (aAreaParam.bRepeatCol && nX1 > nRepeatEndCol);
2080     if ( aAreaParam.bRepeatRow )
2081         if ( nY1 > nRepeatStartRow && nY1 <= nRepeatEndRow )
2082             nY1 = nRepeatEndRow + 1;
2083     sal_Bool bDoRepRow = (aAreaParam.bRepeatRow && nY1 > nRepeatEndRow);
2084 
2085     // use new object hide flags in SdrPaintView
2086     if(pDrawView)
2087     {
2088         pDrawView->setHideOle(!aTableParam.bObjects);
2089         pDrawView->setHideChart(!aTableParam.bCharts);
2090         pDrawView->setHideDraw(!aTableParam.bDrawings);
2091         pDrawView->setHideFormControl(!aTableParam.bDrawings);
2092     }
2093 
2094     if ( pPrinter && bDoPrint )
2095     {
2096         DBG_ERROR( "StartPage does not exist anymore" );
2097         // pPrinter->StartPage();
2098     }
2099 
2100     //  Kopf- und Fusszeilen (ohne Zentrierung)
2101 
2102     if (aHdr.bEnable)
2103     {
2104         long nHeaderY = aPageRect.Top()-aHdr.nHeight;
2105         PrintHF( nPageNo, sal_True, nHeaderY, bDoPrint, pLocationData );
2106     }
2107     if (aFtr.bEnable)
2108     {
2109         long nFooterY = aPageRect.Bottom()+aFtr.nDistance;
2110         PrintHF( nPageNo, sal_False, nFooterY, bDoPrint, pLocationData );
2111     }
2112 
2113     //  Position ( Raender / zentrieren )
2114 
2115     long nLeftSpace = aPageRect.Left();     // Document-Twips
2116     long nTopSpace  = aPageRect.Top();
2117     if ( bCenterHor || bLayoutRTL )
2118     {
2119         long nDataWidth = 0;
2120         SCCOL i;
2121         for (i=nX1; i<=nX2; i++)
2122             nDataWidth += pDoc->GetColWidth( i,nPrintTab );
2123         if (bDoRepCol)
2124             for (i=nRepeatStartCol; i<=nRepeatEndCol; i++)
2125                 nDataWidth += pDoc->GetColWidth( i,nPrintTab );
2126         if (aTableParam.bHeaders)
2127             nDataWidth += (long) PRINT_HEADER_WIDTH;
2128         if (pBorderItem)
2129             nDataWidth += pBorderItem->GetDistance(BOX_LINE_LEFT) +
2130                            pBorderItem->GetDistance(BOX_LINE_RIGHT);        //! Line width?
2131         if (pShadowItem && pShadowItem->GetLocation() != SVX_SHADOW_NONE)
2132             nDataWidth += pShadowItem->CalcShadowSpace(SHADOW_LEFT) +
2133                            pShadowItem->CalcShadowSpace(SHADOW_RIGHT);
2134         if ( bCenterHor )
2135         {
2136             nLeftSpace += ( aPageRect.GetWidth() - nDataWidth ) / 2;        // LTR or RTL
2137             if (pBorderItem)
2138                 nLeftSpace -= lcl_LineTotal(pBorderItem->GetLeft());
2139         }
2140         else if ( bLayoutRTL )
2141             nLeftSpace += aPageRect.GetWidth() - nDataWidth;                // align to the right edge of the page
2142     }
2143     if ( bCenterVer )
2144     {
2145         long nDataHeight = pDoc->GetRowHeight( nY1, nY2, nPrintTab);
2146         if (bDoRepRow)
2147             nDataHeight += pDoc->GetRowHeight( nRepeatStartRow,
2148                     nRepeatEndRow, nPrintTab);
2149         if (aTableParam.bHeaders)
2150             nDataHeight += (long) PRINT_HEADER_HEIGHT;
2151         if (pBorderItem)
2152             nDataHeight += pBorderItem->GetDistance(BOX_LINE_TOP) +
2153                            pBorderItem->GetDistance(BOX_LINE_BOTTOM);       //! Line width?
2154         if (pShadowItem && pShadowItem->GetLocation() != SVX_SHADOW_NONE)
2155             nDataHeight += pShadowItem->CalcShadowSpace(SHADOW_TOP) +
2156                            pShadowItem->CalcShadowSpace(SHADOW_BOTTOM);
2157         nTopSpace += ( aPageRect.GetHeight() - nDataHeight ) / 2;
2158         if (pBorderItem)
2159             nTopSpace -= lcl_LineTotal(pBorderItem->GetTop());
2160     }
2161 
2162     //  calculate sizes of the elements for partitioning
2163     //  (header, repeat, data)
2164 
2165     long nHeaderWidth   = 0;
2166     long nHeaderHeight  = 0;
2167     long nRepeatWidth   = 0;
2168     long nRepeatHeight  = 0;
2169     long nContentWidth  = 0;        // scaled - not the same as nDataWidth above
2170     long nContentHeight = 0;
2171     if (aTableParam.bHeaders)
2172     {
2173         nHeaderWidth  = (long) (PRINT_HEADER_WIDTH * nScaleX);
2174         nHeaderHeight = (long) (PRINT_HEADER_HEIGHT * nScaleY);
2175     }
2176     if (bDoRepCol)
2177         for (SCCOL i=nRepeatStartCol; i<=nRepeatEndCol; i++)
2178             nRepeatWidth += (long) (pDoc->GetColWidth(i,nPrintTab) * nScaleX);
2179     if (bDoRepRow)
2180         nRepeatHeight += pDoc->GetScaledRowHeight( nRepeatStartRow,
2181                 nRepeatEndRow, nPrintTab, nScaleY);
2182     for (SCCOL i=nX1; i<=nX2; i++)
2183         nContentWidth += (long) (pDoc->GetColWidth(i,nPrintTab) * nScaleX);
2184     nContentHeight += pDoc->GetScaledRowHeight( nY1, nY2, nPrintTab,
2185             nScaleY);
2186 
2187     //  partition the page
2188 
2189     long nStartX = ((long) ( nLeftSpace * nScaleX ));
2190     long nStartY = ((long) ( nTopSpace  * nScaleY ));
2191 //      nStartX -= aOffset.X();         // schon im MapMode
2192 //      nStartY -= aOffset.Y();
2193 
2194     long nInnerStartX = nStartX;
2195     long nInnerStartY = nStartY;
2196     if (pBorderItem)
2197     {
2198         nInnerStartX += (long) ( ( lcl_LineTotal(pBorderItem->GetLeft()) +
2199                                     pBorderItem->GetDistance(BOX_LINE_LEFT) ) * nScaleX );
2200         nInnerStartY += (long) ( ( lcl_LineTotal(pBorderItem->GetTop()) +
2201                                     pBorderItem->GetDistance(BOX_LINE_TOP) ) * nScaleY );
2202     }
2203     if (pShadowItem && pShadowItem->GetLocation() != SVX_SHADOW_NONE)
2204     {
2205         nInnerStartX += (long) ( pShadowItem->CalcShadowSpace(SHADOW_LEFT) * nScaleX );
2206         nInnerStartY += (long) ( pShadowItem->CalcShadowSpace(SHADOW_TOP) * nScaleY );
2207     }
2208 
2209     if ( bLayoutRTL )
2210     {
2211         //  arrange elements starting from the right edge
2212         nInnerStartX += nHeaderWidth + nRepeatWidth + nContentWidth;
2213 
2214         //  make rounding easier so the elements are really next to each other in preview
2215         Size aOffsetOnePixel = pDev->PixelToLogic( Size(1,1), aOffsetMode );
2216         long nOffsetOneX = aOffsetOnePixel.Width();
2217         nInnerStartX += nOffsetOneX / 2;
2218     }
2219 
2220     long nFrameStartX = nInnerStartX;
2221     long nFrameStartY = nInnerStartY;
2222 
2223     long nRepStartX = nInnerStartX + nHeaderWidth * nLayoutSign;    // widths/heights are 0 if not used
2224     long nRepStartY = nInnerStartY + nHeaderHeight;
2225     long nDataX = nRepStartX + nRepeatWidth * nLayoutSign;
2226     long nDataY = nRepStartY + nRepeatHeight;
2227     long nEndX = nDataX + nContentWidth * nLayoutSign;
2228     long nEndY = nDataY + nContentHeight;
2229     long nFrameEndX = nEndX;
2230     long nFrameEndY = nEndY;
2231 
2232     if ( bLayoutRTL )
2233     {
2234         //  each element's start position is its left edge
2235         //! subtract one pixel less?
2236         nInnerStartX -= nHeaderWidth;       // used for header
2237         nRepStartX   -= nRepeatWidth;
2238         nDataX       -= nContentWidth;
2239 
2240         //  continue right of the main elements again
2241         nEndX += nHeaderWidth + nRepeatWidth + nContentWidth;
2242     }
2243 
2244     //  Seiten-Rahmen / Hintergrund
2245 
2246     //! nEndX/Y anpassen
2247 
2248     long nBorderEndX = nEndX;
2249     long nBorderEndY = nEndY;
2250     if (pBorderItem)
2251     {
2252         nBorderEndX += (long) ( ( lcl_LineTotal(pBorderItem->GetRight()) +
2253                                     pBorderItem->GetDistance(BOX_LINE_RIGHT) ) * nScaleX );
2254         nBorderEndY += (long) ( ( lcl_LineTotal(pBorderItem->GetBottom()) +
2255                                     pBorderItem->GetDistance(BOX_LINE_BOTTOM) ) * nScaleY );
2256     }
2257     if (pShadowItem && pShadowItem->GetLocation() != SVX_SHADOW_NONE)
2258     {
2259         nBorderEndX += (long) ( pShadowItem->CalcShadowSpace(SHADOW_RIGHT) * nScaleX );
2260         nBorderEndY += (long) ( pShadowItem->CalcShadowSpace(SHADOW_BOTTOM) * nScaleY );
2261     }
2262 
2263     if ( bDoPrint )
2264     {
2265         pDev->SetMapMode( aOffsetMode );
2266         DrawBorder( nStartX, nStartY, nBorderEndX-nStartX, nBorderEndY-nStartY,
2267                         pBorderItem, pBackgroundItem, pShadowItem );
2268 
2269         pDev->SetMapMode( aTwipMode );
2270     }
2271 
2272     pDev->SetMapMode( aOffsetMode );
2273 
2274     //  Wiederholungszeilen/Spalten ausgeben
2275 
2276     if (bDoRepCol && bDoRepRow)
2277     {
2278         if ( bDoPrint )
2279             PrintArea( nRepeatStartCol,nRepeatStartRow, nRepeatEndCol,nRepeatEndRow,
2280                             nRepStartX,nRepStartY, sal_True,sal_True,sal_False,sal_False );
2281         if ( pLocationData )
2282             LocateArea( nRepeatStartCol,nRepeatStartRow, nRepeatEndCol,nRepeatEndRow,
2283                             nRepStartX,nRepStartY, sal_True,sal_True, *pLocationData );
2284     }
2285     if (bDoRepCol)
2286     {
2287         if ( bDoPrint )
2288             PrintArea( nRepeatStartCol,nY1, nRepeatEndCol,nY2, nRepStartX,nDataY,
2289                         sal_True,!bDoRepRow,sal_False,sal_True );
2290         if ( pLocationData )
2291             LocateArea( nRepeatStartCol,nY1, nRepeatEndCol,nY2, nRepStartX,nDataY, sal_True,sal_False, *pLocationData );
2292     }
2293     if (bDoRepRow)
2294     {
2295         if ( bDoPrint )
2296             PrintArea( nX1,nRepeatStartRow, nX2,nRepeatEndRow, nDataX,nRepStartY,
2297                         !bDoRepCol,sal_True,sal_True,sal_False );
2298         if ( pLocationData )
2299             LocateArea( nX1,nRepeatStartRow, nX2,nRepeatEndRow, nDataX,nRepStartY, sal_False,sal_True, *pLocationData );
2300     }
2301 
2302     //  Daten ausgeben
2303 
2304     if ( bDoPrint )
2305         PrintArea( nX1,nY1, nX2,nY2, nDataX,nDataY, !bDoRepCol,!bDoRepRow,sal_True,sal_True );
2306     if ( pLocationData )
2307         LocateArea( nX1,nY1, nX2,nY2, nDataX,nDataY, sal_False,sal_False, *pLocationData );
2308 
2309     //  Spalten-/Zeilenkoepfe ausgeben
2310     //  nach den Daten (ueber evtl. weitergezeichneten Schatten)
2311 
2312     Color aGridColor( COL_BLACK );
2313     if ( bUseStyleColor )
2314         aGridColor.SetColor( SC_MOD()->GetColorConfig().GetColorValue(svtools::FONTCOLOR).nColor );
2315 
2316     if (aTableParam.bHeaders)
2317     {
2318         if ( bDoPrint )
2319         {
2320             pDev->SetLineColor( aGridColor );
2321             pDev->SetFillColor();
2322             pDev->SetMapMode(aOffsetMode);
2323         }
2324 
2325         ScPatternAttr aPattern( pDoc->GetPool() );
2326         Font aFont;
2327         ScAutoFontColorMode eColorMode = bUseStyleColor ? SC_AUTOCOL_DISPLAY : SC_AUTOCOL_PRINT;
2328         aPattern.GetFont( aFont, eColorMode, pDev );
2329         pDev->SetFont( aFont );
2330 
2331         if (bDoRepCol)
2332         {
2333             if ( bDoPrint )
2334                 PrintColHdr( nRepeatStartCol,nRepeatEndCol, nRepStartX,nInnerStartY );
2335             if ( pLocationData )
2336                 LocateColHdr( nRepeatStartCol,nRepeatEndCol, nRepStartX,nInnerStartY, sal_True, *pLocationData );
2337         }
2338         if ( bDoPrint )
2339             PrintColHdr( nX1,nX2, nDataX,nInnerStartY );
2340         if ( pLocationData )
2341             LocateColHdr( nX1,nX2, nDataX,nInnerStartY, sal_False, *pLocationData );
2342         if (bDoRepRow)
2343         {
2344             if ( bDoPrint )
2345                 PrintRowHdr( nRepeatStartRow,nRepeatEndRow, nInnerStartX,nRepStartY );
2346             if ( pLocationData )
2347                 LocateRowHdr( nRepeatStartRow,nRepeatEndRow, nInnerStartX,nRepStartY, sal_True, *pLocationData );
2348         }
2349         if ( bDoPrint )
2350             PrintRowHdr( nY1,nY2, nInnerStartX,nDataY );
2351         if ( pLocationData )
2352             LocateRowHdr( nY1,nY2, nInnerStartX,nDataY, sal_False, *pLocationData );
2353     }
2354 
2355     //  einfacher Rahmen
2356 
2357     if ( bDoPrint && ( aTableParam.bGrid || aTableParam.bHeaders ) )
2358     {
2359         Size aOnePixel = pDev->PixelToLogic(Size(1,1));
2360         long nOneX = aOnePixel.Width();
2361         long nOneY = aOnePixel.Height();
2362 
2363         long nLeftX   = nFrameStartX;
2364         long nTopY    = nFrameStartY - nOneY;
2365         long nRightX  = nFrameEndX;
2366         long nBottomY = nFrameEndY - nOneY;
2367         if ( !bLayoutRTL )
2368         {
2369             nLeftX   -= nOneX;
2370             nRightX  -= nOneX;
2371         }
2372         pDev->SetMapMode(aOffsetMode);
2373         pDev->SetLineColor( aGridColor );
2374         pDev->SetFillColor();
2375         pDev->DrawRect( Rectangle( nLeftX, nTopY, nRightX, nBottomY ) );
2376         //  nEndX/Y ohne Rahmen-Anpassung
2377     }
2378 
2379     if ( pPrinter && bDoPrint )
2380     {
2381         DBG_ERROR( "EndPage does not exist anymore" );
2382         // pPrinter->EndPage();
2383     }
2384 
2385     aLastSourceRange = ScRange( nX1, nY1, nPrintTab, nX2, nY2, nPrintTab );
2386     bSourceRangeValid = sal_True;
2387 }
2388 
SetOffset(const Point & rOfs)2389 void ScPrintFunc::SetOffset( const Point& rOfs )
2390 {
2391     aSrcOffset = rOfs;
2392 }
2393 
SetManualZoom(sal_uInt16 nNewZoom)2394 void ScPrintFunc::SetManualZoom( sal_uInt16 nNewZoom )
2395 {
2396     nManualZoom = nNewZoom;
2397 }
2398 
SetClearFlag(sal_Bool bFlag)2399 void ScPrintFunc::SetClearFlag( sal_Bool bFlag )
2400 {
2401     bClearWin = bFlag;
2402 }
2403 
SetUseStyleColor(sal_Bool bFlag)2404 void ScPrintFunc::SetUseStyleColor( sal_Bool bFlag )
2405 {
2406     bUseStyleColor = bFlag;
2407     if (pEditEngine)
2408         pEditEngine->EnableAutoColor( bUseStyleColor );
2409 }
2410 
SetRenderFlag(sal_Bool bFlag)2411 void ScPrintFunc::SetRenderFlag( sal_Bool bFlag )
2412 {
2413     bIsRender = bFlag;      // set when using XRenderable (PDF)
2414 }
2415 
SetExclusivelyDrawOleAndDrawObjects()2416 void ScPrintFunc::SetExclusivelyDrawOleAndDrawObjects()
2417 {
2418     aTableParam.bCellContent = false;
2419     aTableParam.bNotes = false;
2420     aTableParam.bGrid = false;
2421     aTableParam.bHeaders = false;
2422     aTableParam.bFormulas = false;
2423     aTableParam.bNullVals = false;
2424 }
2425 
2426 //
2427 //  UpdatePages wird nur von aussen gerufen, um die Umbrueche fuer die Anzeige
2428 //  richtig zu setzen - immer ohne UserArea
2429 //
2430 
UpdatePages()2431 sal_Bool ScPrintFunc::UpdatePages()
2432 {
2433     if (!pParamSet)
2434         return sal_False;
2435 
2436     //  Zoom
2437 
2438     nZoom = 100;
2439     if (aTableParam.bScalePageNum || aTableParam.bScaleTo)
2440         nZoom = ZOOM_MIN;                       // stimmt fuer Umbrueche
2441     else if (aTableParam.bScaleAll)
2442     {
2443         nZoom = aTableParam.nScaleAll;
2444         if ( nZoom <= ZOOM_MIN )
2445             nZoom = ZOOM_MIN;
2446     }
2447 
2448     String aName = pDoc->GetPageStyle( nPrintTab );
2449     SCTAB nTabCount = pDoc->GetTableCount();
2450     for (SCTAB nTab=0; nTab<nTabCount; nTab++)
2451         if ( nTab==nPrintTab || pDoc->GetPageStyle(nTab)==aName )
2452         {
2453             //  Wiederholungszeilen / Spalten
2454             pDoc->SetRepeatArea( nTab, nRepeatStartCol,nRepeatEndCol, nRepeatStartRow,nRepeatEndRow );
2455 
2456             //  Umbrueche setzen
2457             ResetBreaks(nTab);
2458             pDocShell->PostPaint(0,0,nTab, MAXCOL,MAXROW,nTab, PAINT_GRID);
2459         }
2460 
2461     return sal_True;
2462 }
2463 
CountPages()2464 long ScPrintFunc::CountPages()                          // setzt auch nPagesX, nPagesY
2465 {
2466     sal_Bool bAreaOk = sal_False;
2467 
2468     if (pDoc->HasTable( nPrintTab ))
2469     {
2470         if (aAreaParam.bPrintArea)                          // Druckbereich angegeben?
2471         {
2472             if ( bPrintCurrentTable )
2473             {
2474                 ScRange& rRange = aAreaParam.aPrintArea;
2475 
2476                 //  hier kein Vergleich der Tabellen mehr, die Area gilt immer fuer diese Tabelle
2477                 //  wenn hier verglichen werden soll, muss die Tabelle der Druckbereiche beim
2478                 //  Einfuegen von Tabellen etc. angepasst werden !
2479 
2480                 nStartCol = rRange.aStart.Col();
2481                 nStartRow = rRange.aStart.Row();
2482                 nEndCol   = rRange.aEnd  .Col();
2483                 nEndRow   = rRange.aEnd  .Row();
2484                 bAreaOk   = AdjustPrintArea(sal_False);         // begrenzen
2485             }
2486             else
2487                 bAreaOk = sal_False;
2488         }
2489         else                                                // aus Dokument suchen
2490             bAreaOk = AdjustPrintArea(sal_True);
2491     }
2492 
2493     if (bAreaOk)
2494     {
2495         long nPages = 0;
2496         size_t nY;
2497         if (bMultiArea)
2498         {
2499             sal_uInt16 nRCount = pDoc->GetPrintRangeCount( nPrintTab );
2500             for (sal_uInt16 i=0; i<nRCount; i++)
2501             {
2502                 CalcZoom(i);
2503                 if ( aTableParam.bSkipEmpty )
2504                     for (nY=0; nY<nPagesY; nY++)
2505                     {
2506                         OSL_ENSURE(nY < maPageRows.size(), "vector access error for maPageRows (!)");
2507                         nPages += maPageRows[nY].CountVisible();
2508                     }
2509                 else
2510                     nPages += ((long) nPagesX) * nPagesY;
2511                 if ( pPageData )
2512                     FillPageData();
2513             }
2514         }
2515         else
2516         {
2517             CalcZoom(RANGENO_NORANGE);                      // Zoom berechnen
2518             if ( aTableParam.bSkipEmpty )
2519                 for (nY=0; nY<nPagesY; nY++)
2520                 {
2521                     OSL_ENSURE(nY < maPageRows.size(), "vector access error for maPageRows (!)");
2522                     nPages += maPageRows[nY].CountVisible();
2523                 }
2524             else
2525                 nPages += ((long) nPagesX) * nPagesY;
2526             if ( pPageData )
2527                 FillPageData();
2528         }
2529         return nPages;
2530     }
2531     else
2532     {
2533 //      nZoom = 100;                        // nZoom auf letztem Wert stehenlassen !!!
2534         nPagesX = nPagesY = nTotalY = 0;
2535         return 0;
2536     }
2537 }
2538 
CountNotePages()2539 long ScPrintFunc::CountNotePages()
2540 {
2541     if ( !aTableParam.bNotes || !bPrintCurrentTable )
2542         return 0;
2543 
2544     long nCount=0;
2545     SCCOL nCol;
2546     SCROW nRow;
2547 
2548     sal_Bool bError = sal_False;
2549     if (!aAreaParam.bPrintArea)
2550         bError = !AdjustPrintArea(sal_True);            // komplett aus Dok suchen
2551 
2552     sal_uInt16 nRepeats = 1;                            // wie oft durchgehen ?
2553     if (bMultiArea)
2554         nRepeats = pDoc->GetPrintRangeCount(nPrintTab);
2555     if (bError)
2556         nRepeats = 0;
2557 
2558     for (sal_uInt16 nStep=0; nStep<nRepeats; nStep++)
2559     {
2560         sal_Bool bDoThis = sal_True;
2561         if (bMultiArea)             // alle Areas durchgehen
2562         {
2563             const ScRange* pThisRange = pDoc->GetPrintRange( nPrintTab, nStep );
2564             if ( pThisRange )
2565             {
2566                 nStartCol = pThisRange->aStart.Col();
2567                 nStartRow = pThisRange->aStart.Row();
2568                 nEndCol   = pThisRange->aEnd  .Col();
2569                 nEndRow   = pThisRange->aEnd  .Row();
2570                 bDoThis = AdjustPrintArea(sal_False);
2571             }
2572         }
2573 
2574         if (bDoThis)
2575         {
2576             ScHorizontalCellIterator aIter( pDoc, nPrintTab, nStartCol,nStartRow, nEndCol,nEndRow );
2577             ScBaseCell* pCell = aIter.GetNext( nCol, nRow );
2578             while (pCell)
2579             {
2580                 if (pCell->HasNote())
2581                 {
2582                     aNotePosList.Insert( new ScAddress( nCol,nRow,nPrintTab ), LIST_APPEND );
2583                     ++nCount;
2584                 }
2585 
2586                 pCell = aIter.GetNext( nCol, nRow );
2587             }
2588         }
2589     }
2590 
2591     long nPages = 0;
2592     long nNoteNr = 0;
2593     long nNoteAdd;
2594     do
2595     {
2596         nNoteAdd = PrintNotes( nPages, nNoteNr, sal_False, NULL );
2597         if (nNoteAdd)
2598         {
2599             nNoteNr += nNoteAdd;
2600             ++nPages;
2601         }
2602     }
2603     while (nNoteAdd);
2604 
2605     return nPages;
2606 }
2607 
InitModes()2608 void ScPrintFunc::InitModes()               // aus nZoom etc. die MapModes setzen
2609 {
2610     aOffset = Point( aSrcOffset.X()*100/nZoom, aSrcOffset.Y()*100/nZoom );
2611 
2612     long nEffZoom = nZoom * (long) nManualZoom;
2613 
2614 //  nScaleX = nScaleY = 1.0;            // Ausgabe in Twips
2615     nScaleX = nScaleY = HMM_PER_TWIPS;  // Ausgabe in 1/100 mm
2616 
2617     Fraction aZoomFract( nEffZoom,10000 );
2618     Fraction aHorFract = aZoomFract;
2619 
2620     if ( !pPrinter && !bIsRender )                          // adjust scale for preview
2621     {
2622         double nFact = pDocShell->GetOutputFactor();
2623         aHorFract = Fraction( (long)( nEffZoom / nFact ), 10000 );
2624     }
2625 
2626     aLogicMode = MapMode( MAP_100TH_MM, Point(), aHorFract, aZoomFract );
2627 
2628     Point aLogicOfs( -aOffset.X(), -aOffset.Y() );
2629     aOffsetMode = MapMode( MAP_100TH_MM, aLogicOfs, aHorFract, aZoomFract );
2630 
2631     Point aTwipsOfs( (long) ( -aOffset.X() / nScaleX + 0.5 ), (long) ( -aOffset.Y() / nScaleY + 0.5 ) );
2632     aTwipMode = MapMode( MAP_TWIP, aTwipsOfs, aHorFract, aZoomFract );
2633 }
2634 
2635 //--------------------------------------------------------------------
2636 
ApplyPrintSettings()2637 void ScPrintFunc::ApplyPrintSettings()
2638 {
2639     if ( pPrinter )
2640     {
2641         //
2642         //  Printer zum Drucken umstellen
2643         //
2644 
2645         Size aEnumSize = aPageSize;
2646 
2647 
2648         pPrinter->SetOrientation( bLandscape ? ORIENTATION_LANDSCAPE : ORIENTATION_PORTRAIT );
2649         if ( bLandscape )
2650         {
2651                 // landscape is always interpreted as a rotation by 90 degrees !
2652                 // this leads to non WYSIWIG but at least it prints!
2653                 // #i21775#
2654                 long nTemp = aEnumSize.Width();
2655                 aEnumSize.Width() = aEnumSize.Height();
2656                 aEnumSize.Height() = nTemp;
2657         }
2658         Paper ePaper = SvxPaperInfo::GetSvxPaper( aEnumSize, MAP_TWIP, sal_True );
2659         sal_uInt16 nPaperBin = ((const SvxPaperBinItem&)pParamSet->Get(ATTR_PAGE_PAPERBIN)).GetValue();
2660 
2661         pPrinter->SetPaper( ePaper );
2662         if ( PAPER_USER == ePaper )
2663         {
2664             MapMode aPrinterMode = pPrinter->GetMapMode();
2665             MapMode aLocalMode( MAP_TWIP );
2666             pPrinter->SetMapMode( aLocalMode );
2667             pPrinter->SetPaperSizeUser( aEnumSize );
2668             pPrinter->SetMapMode( aPrinterMode );
2669         }
2670 
2671         pPrinter->SetPaperBin( nPaperBin );
2672     }
2673 }
2674 
2675 //--------------------------------------------------------------------
2676 //  rPageRanges   = Range fuer alle Tabellen
2677 //  nStartPage    = in rPageRanges beginnen bei nStartPage
2678 //  nDisplayStart = lfd. Nummer fuer Anzeige der Seitennummer
2679 
DoPrint(const MultiSelection & rPageRanges,long nStartPage,long nDisplayStart,sal_Bool bDoPrint,ScPreviewLocationData * pLocationData)2680 long ScPrintFunc::DoPrint( const MultiSelection& rPageRanges,
2681                                 long nStartPage, long nDisplayStart, sal_Bool bDoPrint,
2682                                 ScPreviewLocationData* pLocationData )
2683 {
2684     DBG_ASSERT(pDev,"Device == NULL");
2685     if (!pParamSet)
2686         return 0;
2687 
2688     if ( pPrinter && bDoPrint )
2689         ApplyPrintSettings();
2690 
2691     //--------------------------------------------------------------------
2692 
2693     InitModes();
2694     if ( pLocationData )
2695     {
2696         pLocationData->SetCellMapMode( aOffsetMode );
2697         pLocationData->SetPrintTab( nPrintTab );
2698     }
2699 
2700     MakeTableString();
2701 
2702     //--------------------------------------------------------------------
2703 
2704     long nPageNo = 0;
2705     long nPrinted = 0;
2706     long nEndPage = rPageRanges.GetTotalRange().Max();
2707 
2708     sal_uInt16 nRepeats = 1;                    // wie oft durchgehen ?
2709     if (bMultiArea)
2710         nRepeats = pDoc->GetPrintRangeCount(nPrintTab);
2711     for (sal_uInt16 nStep=0; nStep<nRepeats; nStep++)
2712     {
2713         if (bMultiArea)                     // Bereich neu belegen ?
2714         {
2715             CalcZoom(nStep);                // setzt auch nStartCol etc. neu
2716             InitModes();
2717         }
2718 
2719         SCCOL nX1;
2720         SCROW nY1;
2721         SCCOL nX2;
2722         SCROW nY2;
2723         size_t nCountX;
2724         size_t nCountY;
2725 
2726         if (aTableParam.bTopDown)                           // von oben nach unten
2727         {
2728             nX1 = nStartCol;
2729             for (nCountX=0; nCountX<nPagesX; nCountX++)
2730             {
2731                 OSL_ENSURE(nCountX < maPageEndX.size(), "vector access error for maPageEndX (!)");
2732                 nX2 = maPageEndX[nCountX];
2733                 for (nCountY=0; nCountY<nPagesY; nCountY++)
2734                 {
2735                     OSL_ENSURE(nCountY < maPageRows.size(), "vector access error for maPageRows (!)");
2736                     nY1 = maPageRows[nCountY].GetStartRow();
2737                     nY2 = maPageRows[nCountY].GetEndRow();
2738                     if ( !aTableParam.bSkipEmpty || !maPageRows[nCountY].IsHidden(nCountX) )
2739                     {
2740                         if ( rPageRanges.IsSelected( nPageNo+nStartPage+1 ) )
2741                         {
2742                             PrintPage( nPageNo+nDisplayStart, nX1, nY1, nX2, nY2,
2743                                         bDoPrint, pLocationData );
2744                             ++nPrinted;
2745                         }
2746                         ++nPageNo;
2747                     }
2748                 }
2749                 nX1 = nX2 + 1;
2750             }
2751         }
2752         else                                                // von links nach rechts
2753         {
2754             for (nCountY=0; nCountY<nPagesY; nCountY++)
2755             {
2756                 OSL_ENSURE(nCountY < maPageRows.size(), "vector access error for maPageRows (!)");
2757                 nY1 = maPageRows[nCountY].GetStartRow();
2758                 nY2 = maPageRows[nCountY].GetEndRow();
2759                 nX1 = nStartCol;
2760                 for (nCountX=0; nCountX<nPagesX; nCountX++)
2761                 {
2762                     OSL_ENSURE(nCountX < maPageEndX.size(), "vector access error for maPageEndX (!)");
2763                     nX2 = maPageEndX[nCountX];
2764                     if ( !aTableParam.bSkipEmpty || !maPageRows[nCountY].IsHidden(nCountX) )
2765                     {
2766                         if ( rPageRanges.IsSelected( nPageNo+nStartPage+1 ) )
2767                         {
2768                             PrintPage( nPageNo+nDisplayStart, nX1, nY1, nX2, nY2,
2769                                         bDoPrint, pLocationData );
2770                             ++nPrinted;
2771                         }
2772                         ++nPageNo;
2773                     }
2774                     nX1 = nX2 + 1;
2775                 }
2776             }
2777         }
2778     }
2779 
2780     aFieldData.aTabName = ScGlobal::GetRscString( STR_NOTES );
2781 
2782     long nNoteNr = 0;
2783     long nNoteAdd;
2784     do
2785     {
2786         if ( nPageNo+nStartPage <= nEndPage )
2787         {
2788             sal_Bool bPageSelected = rPageRanges.IsSelected( nPageNo+nStartPage+1 );
2789             nNoteAdd = PrintNotes( nPageNo+nStartPage, nNoteNr, bDoPrint && bPageSelected,
2790                                     ( bPageSelected ? pLocationData : NULL ) );
2791             if ( nNoteAdd )
2792             {
2793                 nNoteNr += nNoteAdd;
2794                 if (bPageSelected)
2795                 {
2796                     ++nPrinted;
2797                     bSourceRangeValid = sal_False;      // last page was no cell range
2798                 }
2799                 ++nPageNo;
2800             }
2801         }
2802         else
2803             nNoteAdd = 0;
2804     }
2805     while (nNoteAdd);
2806 
2807     if ( bMultiArea )
2808         ResetBreaks(nPrintTab);                         // Breaks fuer Anzeige richtig
2809 
2810     return nPrinted;
2811 }
2812 
CalcZoom(sal_uInt16 nRangeNo)2813 void ScPrintFunc::CalcZoom( sal_uInt16 nRangeNo )                       // Zoom berechnen
2814 {
2815     sal_uInt16 nRCount = pDoc->GetPrintRangeCount( nPrintTab );
2816     const ScRange* pThisRange = NULL;
2817     if ( nRangeNo != RANGENO_NORANGE || nRangeNo < nRCount )
2818         pThisRange = pDoc->GetPrintRange( nPrintTab, nRangeNo );
2819     if ( pThisRange )
2820     {
2821         nStartCol = pThisRange->aStart.Col();
2822         nStartRow = pThisRange->aStart.Row();
2823         nEndCol   = pThisRange->aEnd  .Col();
2824         nEndRow   = pThisRange->aEnd  .Row();
2825     }
2826 
2827     if (!AdjustPrintArea(sal_False))                        // leer
2828     {
2829         nZoom = 100;
2830         nPagesX = nPagesY = nTotalY = 0;
2831         return;
2832     }
2833 
2834     pDoc->SetRepeatArea( nPrintTab, nRepeatStartCol,nRepeatEndCol, nRepeatStartRow,nRepeatEndRow );
2835 
2836     if (aTableParam.bScalePageNum)
2837     {
2838         nZoom = 100;
2839         sal_uInt16 nPagesToFit = aTableParam.nScalePageNum;
2840 
2841         sal_uInt16 nLastFitZoom = 0, nLastNonFitZoom = 0;
2842         while (true)
2843         {
2844             if (nZoom <= ZOOM_MIN)
2845                 break;
2846 
2847             CalcPages();
2848             bool bFitsPage = (nPagesX * nPagesY <= nPagesToFit);
2849 
2850             if (bFitsPage)
2851             {
2852                 if (nZoom == 100)
2853                     // If it fits at 100 %, it's good enough for me.
2854                     break;
2855 
2856                 nLastFitZoom = nZoom;
2857                 nZoom = (nLastNonFitZoom + nZoom) / 2;
2858 
2859                 if (nLastFitZoom == nZoom)
2860                     // It converged.  Use this zoom level.
2861                     break;
2862             }
2863             else
2864             {
2865                 if (nZoom - nLastFitZoom <= 1)
2866                 {
2867                     nZoom = nLastFitZoom;
2868                     CalcPages();
2869                     break;
2870                 }
2871 
2872                 nLastNonFitZoom = nZoom;
2873                 nZoom = (nLastFitZoom + nZoom) / 2;
2874             }
2875         }
2876     }
2877     else if (aTableParam.bScaleTo)
2878     {
2879         nZoom = 100;
2880         sal_uInt16 nW = aTableParam.nScaleWidth;
2881         sal_uInt16 nH = aTableParam.nScaleHeight;
2882 
2883         sal_uInt16 nLastFitZoom = 0, nLastNonFitZoom = 0;
2884         while (true)
2885         {
2886             if (nZoom <= ZOOM_MIN)
2887                 break;
2888 
2889             CalcPages();
2890             bool bFitsPage = ((!nW || (nPagesX <= nW)) && (!nH || (nPagesY <= nH)));
2891 
2892             if (bFitsPage)
2893             {
2894                 if (nZoom == 100)
2895                     // If it fits at 100 %, it's good enough for me.
2896                     break;
2897 
2898                 nLastFitZoom = nZoom;
2899                 nZoom = (nLastNonFitZoom + nZoom) / 2;
2900 
2901                 if (nLastFitZoom == nZoom)
2902                     // It converged.  Use this zoom level.
2903                     break;
2904             }
2905             else
2906             {
2907                 if (nZoom - nLastFitZoom <= 1)
2908                 {
2909                     nZoom = nLastFitZoom;
2910                     CalcPages();
2911                     break;
2912                 }
2913 
2914                 nLastNonFitZoom = nZoom;
2915                 nZoom = (nLastFitZoom + nZoom) / 2;
2916             }
2917         }
2918     }
2919     else if (aTableParam.bScaleAll)
2920     {
2921         nZoom = aTableParam.nScaleAll;
2922         if ( nZoom <= ZOOM_MIN )
2923             nZoom = ZOOM_MIN;
2924         CalcPages();
2925     }
2926     else
2927     {
2928         DBG_ASSERT( aTableParam.bScaleNone, "kein Scale-Flag gesetzt" );
2929         nZoom = 100;
2930         CalcPages();
2931     }
2932 }
2933 
GetDocPageSize()2934 Size ScPrintFunc::GetDocPageSize()
2935 {
2936                         // Hoehe Kopf-/Fusszeile anpassen
2937 
2938     InitModes();                            // aTwipMode aus nZoom initialisieren
2939     pDev->SetMapMode( aTwipMode );          // Kopf-/Fusszeilen in Twips
2940     UpdateHFHeight( aHdr );
2941     UpdateHFHeight( aFtr );
2942 
2943                         // Seitengroesse in Document-Twips
2944                         //  Berechnung Left / Right auch in PrintPage
2945 
2946     aPageRect = Rectangle( Point(), aPageSize );
2947     aPageRect.Left()   = ( aPageRect.Left()   + nLeftMargin                  ) * 100 / nZoom;
2948     aPageRect.Right()  = ( aPageRect.Right()  - nRightMargin                 ) * 100 / nZoom;
2949     aPageRect.Top()    = ( aPageRect.Top()    + nTopMargin    ) * 100 / nZoom + aHdr.nHeight;
2950     aPageRect.Bottom() = ( aPageRect.Bottom() - nBottomMargin ) * 100 / nZoom - aFtr.nHeight;
2951 
2952     Size aDocPageSize = aPageRect.GetSize();
2953     if (aTableParam.bHeaders)
2954     {
2955         aDocPageSize.Width()  -= (long) PRINT_HEADER_WIDTH;
2956         aDocPageSize.Height() -= (long) PRINT_HEADER_HEIGHT;
2957     }
2958     if (pBorderItem)
2959     {
2960         aDocPageSize.Width()  -= lcl_LineTotal(pBorderItem->GetLeft()) +
2961                                  lcl_LineTotal(pBorderItem->GetRight()) +
2962                                  pBorderItem->GetDistance(BOX_LINE_LEFT) +
2963                                  pBorderItem->GetDistance(BOX_LINE_RIGHT);
2964         aDocPageSize.Height() -= lcl_LineTotal(pBorderItem->GetTop()) +
2965                                  lcl_LineTotal(pBorderItem->GetBottom()) +
2966                                  pBorderItem->GetDistance(BOX_LINE_TOP) +
2967                                  pBorderItem->GetDistance(BOX_LINE_BOTTOM);
2968     }
2969     if (pShadowItem && pShadowItem->GetLocation() != SVX_SHADOW_NONE)
2970     {
2971         aDocPageSize.Width()  -= pShadowItem->CalcShadowSpace(SHADOW_LEFT) +
2972                                  pShadowItem->CalcShadowSpace(SHADOW_RIGHT);
2973         aDocPageSize.Height() -= pShadowItem->CalcShadowSpace(SHADOW_TOP) +
2974                                  pShadowItem->CalcShadowSpace(SHADOW_BOTTOM);
2975     }
2976     return aDocPageSize;
2977 }
2978 
ResetBreaks(SCTAB nTab)2979 void ScPrintFunc::ResetBreaks( SCTAB nTab )         // Breaks fuer Anzeige richtig setzen
2980 {
2981     pDoc->SetPageSize( nTab, GetDocPageSize() );
2982     pDoc->UpdatePageBreaks( nTab, NULL );
2983 }
2984 
lcl_SetHidden(ScDocument * pDoc,SCTAB nPrintTab,ScPageRowEntry & rPageRowEntry,SCCOL nStartCol,const std::vector<SCCOL> & aPageEndX)2985 void lcl_SetHidden( ScDocument* pDoc, SCTAB nPrintTab, ScPageRowEntry& rPageRowEntry, SCCOL nStartCol, const std::vector< SCCOL >& aPageEndX)
2986 {
2987     size_t nPagesX   = rPageRowEntry.GetPagesX();
2988     SCROW nStartRow = rPageRowEntry.GetStartRow();
2989     SCROW nEndRow   = rPageRowEntry.GetEndRow();
2990 
2991     sal_Bool bLeftIsEmpty = sal_False;
2992     ScRange aTempRange;
2993     Rectangle aTempRect = pDoc->GetMMRect( 0,0, 0,0, 0 );
2994 
2995     for (size_t i=0; i<nPagesX; i++)
2996     {
2997         OSL_ENSURE(i < aPageEndX.size(), "vector access error for maPageEndX (!)");
2998         SCCOL nEndCol = aPageEndX[i];
2999         if ( pDoc->IsPrintEmpty( nPrintTab, nStartCol, nStartRow, nEndCol, nEndRow,
3000                                     bLeftIsEmpty, &aTempRange, &aTempRect ) )
3001         {
3002             rPageRowEntry.SetHidden(i);
3003             bLeftIsEmpty = sal_True;
3004         }
3005         else
3006             bLeftIsEmpty = sal_False;
3007 
3008         nStartCol = nEndCol+1;
3009     }
3010 }
3011 
CalcPages()3012 void ScPrintFunc::CalcPages()               // berechnet aPageRect und Seiten aus nZoom
3013 {
3014     // #123672# use dynamic mem to react on size changes
3015     if(maPageEndX.size() < MAXCOL+1)
3016     {
3017         maPageEndX.resize(MAXCOL+1, SCCOL());
3018     }
3019 
3020     pDoc->SetPageSize( nPrintTab, GetDocPageSize() );
3021     if (aAreaParam.bPrintArea)
3022     {
3023         ScRange aRange( nStartCol, nStartRow, nPrintTab, nEndCol, nEndRow, nPrintTab );
3024         pDoc->UpdatePageBreaks( nPrintTab, &aRange );
3025     }
3026     else
3027     {
3028         pDoc->UpdatePageBreaks( nPrintTab, NULL );      // sonst wird das Ende markiert
3029     }
3030 
3031     const SCROW nRealCnt = nEndRow-nStartRow+1;
3032 
3033     // #123672# use dynamic mem to react on size changes
3034     if(maPageEndY.size() < nRealCnt+1)
3035     {
3036         maPageEndY.resize(nRealCnt+1, SCROW());
3037     }
3038 
3039     // #123672# use dynamic mem to react on size changes
3040     if(maPageRows.size() < nRealCnt+1)
3041     {
3042         maPageRows.resize(nRealCnt+1, ScPageRowEntry());
3043     }
3044 
3045     //
3046     //  Seiteneinteilung nach Umbruechen in Col/RowFlags
3047     //  Von mehreren Umbruechen in einem ausgeblendeten Bereich zaehlt nur einer.
3048     //
3049 
3050     nPagesX = 0;
3051     nPagesY = 0;
3052     nTotalY = 0;
3053 
3054     bool bVisCol = false;
3055     SCCOL nLastCol = -1;
3056     for (SCCOL i=nStartCol; i<=nEndCol; i++)
3057     {
3058         bool bHidden = pDoc->ColHidden(i, nPrintTab, nLastCol);
3059         bool bPageBreak = (pDoc->HasColBreak(i, nPrintTab) & BREAK_PAGE);
3060         if ( i>nStartCol && bVisCol && bPageBreak )
3061         {
3062             OSL_ENSURE(nPagesX < maPageEndX.size(), "vector access error for maPageEndX (!)");
3063             maPageEndX[nPagesX] = i-1;
3064             ++nPagesX;
3065             bVisCol = false;
3066         }
3067         if (!bHidden)
3068             bVisCol = true;
3069     }
3070     if (bVisCol)    // auch am Ende keine leeren Seiten
3071     {
3072         OSL_ENSURE(nPagesX < maPageEndX.size(), "vector access error for maPageEndX (!)");
3073         maPageEndX[nPagesX] = nEndCol;
3074         ++nPagesX;
3075     }
3076 
3077     bool bVisRow = false;
3078     SCROW nPageStartRow = nStartRow;
3079     SCROW nLastVisibleRow = -1;
3080 
3081     ::boost::scoped_ptr<ScRowBreakIterator> pRowBreakIter(pDoc->GetRowBreakIterator(nPrintTab));
3082     SCROW nNextPageBreak = pRowBreakIter->first();
3083     while (nNextPageBreak != ScRowBreakIterator::NOT_FOUND && nNextPageBreak < nStartRow)
3084         // Skip until the page break position is at the start row or greater.
3085         nNextPageBreak = pRowBreakIter->next();
3086 
3087     for (SCROW nRow = nStartRow; nRow <= nEndRow; ++nRow)
3088     {
3089         bool bPageBreak = (nNextPageBreak == nRow);
3090         if (bPageBreak)
3091             nNextPageBreak = pRowBreakIter->next();
3092 
3093         if (nRow > nStartRow && bVisRow && bPageBreak )
3094         {
3095             OSL_ENSURE(nTotalY < maPageEndY.size(), "vector access error for maPageEndY (!)");
3096             maPageEndY[nTotalY] = nRow-1;
3097             ++nTotalY;
3098 
3099             if ( !aTableParam.bSkipEmpty ||
3100                     !pDoc->IsPrintEmpty( nPrintTab, nStartCol, nPageStartRow, nEndCol, nRow-1 ) )
3101             {
3102                 OSL_ENSURE(nPagesY < maPageRows.size(), "vector access error for maPageRows (!)");
3103                 maPageRows[nPagesY].SetStartRow( nPageStartRow );
3104                 maPageRows[nPagesY].SetEndRow( nRow-1 );
3105                 maPageRows[nPagesY].SetPagesX( nPagesX );
3106                 if (aTableParam.bSkipEmpty)
3107                     lcl_SetHidden( pDoc, nPrintTab, maPageRows[nPagesY], nStartCol, maPageEndX );
3108                 ++nPagesY;
3109             }
3110 
3111             nPageStartRow = nRow;
3112             bVisRow = false;
3113         }
3114 
3115         if (nRow <= nLastVisibleRow)
3116         {
3117             // This row is still visible.  Don't bother calling RowHidden() to
3118             // find out, for speed optimization.
3119             bVisRow = true;
3120             continue;
3121         }
3122 
3123         SCROW nLastRow = -1;
3124         if (!pDoc->RowHidden(nRow, nPrintTab, NULL, &nLastRow))
3125         {
3126             bVisRow = true;
3127             nLastVisibleRow = nLastRow;
3128         }
3129         else
3130             // skip all hidden rows.
3131             nRow = nLastRow;
3132     }
3133 
3134     if (bVisRow)
3135     {
3136         OSL_ENSURE(nTotalY < maPageEndY.size(), "vector access error for maPageEndY (!)");
3137         maPageEndY[nTotalY] = nEndRow;
3138         ++nTotalY;
3139 
3140         if ( !aTableParam.bSkipEmpty ||
3141                 !pDoc->IsPrintEmpty( nPrintTab, nStartCol, nPageStartRow, nEndCol, nEndRow ) )
3142         {
3143             OSL_ENSURE(nPagesY < maPageRows.size(), "vector access error for maPageRows (!)");
3144             maPageRows[nPagesY].SetStartRow( nPageStartRow );
3145             maPageRows[nPagesY].SetEndRow( nEndRow );
3146             maPageRows[nPagesY].SetPagesX( nPagesX );
3147             if (aTableParam.bSkipEmpty)
3148                 lcl_SetHidden( pDoc, nPrintTab, maPageRows[nPagesY], nStartCol, maPageEndX );
3149             ++nPagesY;
3150         }
3151     }
3152 }
3153 
3154 //------------------------------------------------------------------------
3155 //  class ScJobSetup
3156 //------------------------------------------------------------------------
3157 
ScJobSetup(SfxPrinter * pPrinter)3158 ScJobSetup::ScJobSetup( SfxPrinter* pPrinter )
3159 {
3160     eOrientation = pPrinter->GetOrientation();
3161     nPaperBin    = pPrinter->GetPaperBin();
3162     ePaper       = pPrinter->GetPaper();
3163 
3164     if ( PAPER_USER == ePaper )
3165     {
3166         aUserSize = pPrinter->GetPaperSize();
3167         aUserMapMode = pPrinter->GetMapMode();
3168     }
3169 };
3170 
3171 
3172 
3173 
3174 
3175