xref: /trunk/main/sc/source/ui/view/pfuncache.cxx (revision cdf0e10c4e3984b49a9502b011690b615761d4a3)
1*cdf0e10cSrcweir /*************************************************************************
2*cdf0e10cSrcweir  *
3*cdf0e10cSrcweir  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4*cdf0e10cSrcweir  *
5*cdf0e10cSrcweir  * Copyright 2000, 2010 Oracle and/or its affiliates.
6*cdf0e10cSrcweir  *
7*cdf0e10cSrcweir  * OpenOffice.org - a multi-platform office productivity suite
8*cdf0e10cSrcweir  *
9*cdf0e10cSrcweir  * This file is part of OpenOffice.org.
10*cdf0e10cSrcweir  *
11*cdf0e10cSrcweir  * OpenOffice.org is free software: you can redistribute it and/or modify
12*cdf0e10cSrcweir  * it under the terms of the GNU Lesser General Public License version 3
13*cdf0e10cSrcweir  * only, as published by the Free Software Foundation.
14*cdf0e10cSrcweir  *
15*cdf0e10cSrcweir  * OpenOffice.org is distributed in the hope that it will be useful,
16*cdf0e10cSrcweir  * but WITHOUT ANY WARRANTY; without even the implied warranty of
17*cdf0e10cSrcweir  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
18*cdf0e10cSrcweir  * GNU Lesser General Public License version 3 for more details
19*cdf0e10cSrcweir  * (a copy is included in the LICENSE file that accompanied this code).
20*cdf0e10cSrcweir  *
21*cdf0e10cSrcweir  * You should have received a copy of the GNU Lesser General Public License
22*cdf0e10cSrcweir  * version 3 along with OpenOffice.org.  If not, see
23*cdf0e10cSrcweir  * <http://www.openoffice.org/license.html>
24*cdf0e10cSrcweir  * for a copy of the LGPLv3 License.
25*cdf0e10cSrcweir  *
26*cdf0e10cSrcweir  ************************************************************************/
27*cdf0e10cSrcweir 
28*cdf0e10cSrcweir // MARKER(update_precomp.py): autogen include statement, do not remove
29*cdf0e10cSrcweir #include "precompiled_sc.hxx"
30*cdf0e10cSrcweir 
31*cdf0e10cSrcweir 
32*cdf0e10cSrcweir 
33*cdf0e10cSrcweir // INCLUDE ---------------------------------------------------------------
34*cdf0e10cSrcweir 
35*cdf0e10cSrcweir #include <tools/multisel.hxx>
36*cdf0e10cSrcweir 
37*cdf0e10cSrcweir #include "pfuncache.hxx"
38*cdf0e10cSrcweir #include "printfun.hxx"
39*cdf0e10cSrcweir #include "docsh.hxx"
40*cdf0e10cSrcweir #include "markdata.hxx"
41*cdf0e10cSrcweir #include "prevloc.hxx"
42*cdf0e10cSrcweir 
43*cdf0e10cSrcweir //------------------------------------------------------------------------
44*cdf0e10cSrcweir 
45*cdf0e10cSrcweir ScPrintFuncCache::ScPrintFuncCache( ScDocShell* pD, const ScMarkData& rMark,
46*cdf0e10cSrcweir                                     const ScPrintSelectionStatus& rStatus ) :
47*cdf0e10cSrcweir     aSelection( rStatus ),
48*cdf0e10cSrcweir     pDocSh( pD ),
49*cdf0e10cSrcweir     nTotalPages( 0 ),
50*cdf0e10cSrcweir     bLocInitialized( false )
51*cdf0e10cSrcweir {
52*cdf0e10cSrcweir     //  page count uses the stored cell widths for the printer anyway,
53*cdf0e10cSrcweir     //  so ScPrintFunc with the document's printer can be used to count
54*cdf0e10cSrcweir 
55*cdf0e10cSrcweir     SfxPrinter* pPrinter = pDocSh->GetPrinter();
56*cdf0e10cSrcweir 
57*cdf0e10cSrcweir     ScRange aRange;
58*cdf0e10cSrcweir     const ScRange* pSelRange = NULL;
59*cdf0e10cSrcweir     if ( rMark.IsMarked() )
60*cdf0e10cSrcweir     {
61*cdf0e10cSrcweir         rMark.GetMarkArea( aRange );
62*cdf0e10cSrcweir         pSelRange = &aRange;
63*cdf0e10cSrcweir     }
64*cdf0e10cSrcweir 
65*cdf0e10cSrcweir     ScDocument* pDoc = pDocSh->GetDocument();
66*cdf0e10cSrcweir     SCTAB nTabCount = pDoc->GetTableCount();
67*cdf0e10cSrcweir 
68*cdf0e10cSrcweir     // avoid repeated progress bars if row heights for all sheets are needed
69*cdf0e10cSrcweir     if ( nTabCount > 1 && rMark.GetSelectCount() == nTabCount )
70*cdf0e10cSrcweir         pDocSh->UpdatePendingRowHeights( nTabCount-1, true );
71*cdf0e10cSrcweir 
72*cdf0e10cSrcweir     SCTAB nTab;
73*cdf0e10cSrcweir     for ( nTab=0; nTab<nTabCount; nTab++ )
74*cdf0e10cSrcweir     {
75*cdf0e10cSrcweir         long nAttrPage = nTab > 0 ? nFirstAttr[nTab-1] : 1;
76*cdf0e10cSrcweir 
77*cdf0e10cSrcweir         long nThisTab = 0;
78*cdf0e10cSrcweir         if ( rMark.GetTableSelect( nTab ) )
79*cdf0e10cSrcweir         {
80*cdf0e10cSrcweir             ScPrintFunc aFunc( pDocSh, pPrinter, nTab, nAttrPage, 0, pSelRange, &aSelection.GetOptions() );
81*cdf0e10cSrcweir             nThisTab = aFunc.GetTotalPages();
82*cdf0e10cSrcweir             nFirstAttr[nTab] = aFunc.GetFirstPageNo();          // from page style or previous sheet
83*cdf0e10cSrcweir         }
84*cdf0e10cSrcweir         else
85*cdf0e10cSrcweir             nFirstAttr[nTab] = nAttrPage;
86*cdf0e10cSrcweir 
87*cdf0e10cSrcweir         nPages[nTab] = nThisTab;
88*cdf0e10cSrcweir         nTotalPages += nThisTab;
89*cdf0e10cSrcweir     }
90*cdf0e10cSrcweir }
91*cdf0e10cSrcweir 
92*cdf0e10cSrcweir ScPrintFuncCache::~ScPrintFuncCache()
93*cdf0e10cSrcweir {
94*cdf0e10cSrcweir }
95*cdf0e10cSrcweir 
96*cdf0e10cSrcweir void ScPrintFuncCache::InitLocations( const ScMarkData& rMark, OutputDevice* pDev )
97*cdf0e10cSrcweir {
98*cdf0e10cSrcweir     if ( bLocInitialized )
99*cdf0e10cSrcweir         return;                 // initialize only once
100*cdf0e10cSrcweir 
101*cdf0e10cSrcweir     ScRange aRange;
102*cdf0e10cSrcweir     const ScRange* pSelRange = NULL;
103*cdf0e10cSrcweir     if ( rMark.IsMarked() )
104*cdf0e10cSrcweir     {
105*cdf0e10cSrcweir         rMark.GetMarkArea( aRange );
106*cdf0e10cSrcweir         pSelRange = &aRange;
107*cdf0e10cSrcweir     }
108*cdf0e10cSrcweir 
109*cdf0e10cSrcweir     long nRenderer = 0;     // 0-based physical page number across sheets
110*cdf0e10cSrcweir     long nTabStart = 0;
111*cdf0e10cSrcweir 
112*cdf0e10cSrcweir     ScDocument* pDoc = pDocSh->GetDocument();
113*cdf0e10cSrcweir     SCTAB nTabCount = pDoc->GetTableCount();
114*cdf0e10cSrcweir     for ( SCTAB nTab=0; nTab<nTabCount; nTab++ )
115*cdf0e10cSrcweir     {
116*cdf0e10cSrcweir         if ( rMark.GetTableSelect( nTab ) )
117*cdf0e10cSrcweir         {
118*cdf0e10cSrcweir             ScPrintFunc aFunc( pDev, pDocSh, nTab, nFirstAttr[nTab], nTotalPages, pSelRange, &aSelection.GetOptions() );
119*cdf0e10cSrcweir             aFunc.SetRenderFlag( sal_True );
120*cdf0e10cSrcweir 
121*cdf0e10cSrcweir             long nDisplayStart = GetDisplayStart( nTab );
122*cdf0e10cSrcweir 
123*cdf0e10cSrcweir             for ( long nPage=0; nPage<nPages[nTab]; nPage++ )
124*cdf0e10cSrcweir             {
125*cdf0e10cSrcweir                 Range aPageRange( nRenderer+1, nRenderer+1 );
126*cdf0e10cSrcweir                 MultiSelection aPage( aPageRange );
127*cdf0e10cSrcweir                 aPage.SetTotalRange( Range(0,RANGE_MAX) );
128*cdf0e10cSrcweir                 aPage.Select( aPageRange );
129*cdf0e10cSrcweir 
130*cdf0e10cSrcweir                 ScPreviewLocationData aLocData( pDoc, pDev );
131*cdf0e10cSrcweir                 aFunc.DoPrint( aPage, nTabStart, nDisplayStart, sal_False, &aLocData );
132*cdf0e10cSrcweir 
133*cdf0e10cSrcweir                 ScRange aCellRange;
134*cdf0e10cSrcweir                 Rectangle aPixRect;
135*cdf0e10cSrcweir                 if ( aLocData.GetMainCellRange( aCellRange, aPixRect ) )
136*cdf0e10cSrcweir                     aLocations.push_back( ScPrintPageLocation( nRenderer, aCellRange, aPixRect ) );
137*cdf0e10cSrcweir 
138*cdf0e10cSrcweir                 ++nRenderer;
139*cdf0e10cSrcweir             }
140*cdf0e10cSrcweir 
141*cdf0e10cSrcweir             nTabStart += nPages[nTab];
142*cdf0e10cSrcweir         }
143*cdf0e10cSrcweir     }
144*cdf0e10cSrcweir 
145*cdf0e10cSrcweir     bLocInitialized = true;
146*cdf0e10cSrcweir }
147*cdf0e10cSrcweir 
148*cdf0e10cSrcweir bool ScPrintFuncCache::FindLocation( const ScAddress& rCell, ScPrintPageLocation& rLocation ) const
149*cdf0e10cSrcweir {
150*cdf0e10cSrcweir     for ( std::vector<ScPrintPageLocation>::const_iterator aIter(aLocations.begin());
151*cdf0e10cSrcweir           aIter != aLocations.end(); aIter++ )
152*cdf0e10cSrcweir     {
153*cdf0e10cSrcweir         if ( aIter->aCellRange.In( rCell ) )
154*cdf0e10cSrcweir         {
155*cdf0e10cSrcweir             rLocation = *aIter;
156*cdf0e10cSrcweir             return true;
157*cdf0e10cSrcweir         }
158*cdf0e10cSrcweir     }
159*cdf0e10cSrcweir     return false;   // not found
160*cdf0e10cSrcweir }
161*cdf0e10cSrcweir 
162*cdf0e10cSrcweir sal_Bool ScPrintFuncCache::IsSameSelection( const ScPrintSelectionStatus& rStatus ) const
163*cdf0e10cSrcweir {
164*cdf0e10cSrcweir     return aSelection == rStatus;
165*cdf0e10cSrcweir }
166*cdf0e10cSrcweir 
167*cdf0e10cSrcweir SCTAB ScPrintFuncCache::GetTabForPage( long nPage ) const
168*cdf0e10cSrcweir {
169*cdf0e10cSrcweir     ScDocument* pDoc = pDocSh->GetDocument();
170*cdf0e10cSrcweir     SCTAB nTabCount = pDoc->GetTableCount();
171*cdf0e10cSrcweir     SCTAB nTab = 0;
172*cdf0e10cSrcweir     while ( nTab < nTabCount && nPage >= nPages[nTab] )
173*cdf0e10cSrcweir         nPage -= nPages[nTab++];
174*cdf0e10cSrcweir     return nTab;
175*cdf0e10cSrcweir }
176*cdf0e10cSrcweir 
177*cdf0e10cSrcweir long ScPrintFuncCache::GetTabStart( SCTAB nTab ) const
178*cdf0e10cSrcweir {
179*cdf0e10cSrcweir     long nRet = 0;
180*cdf0e10cSrcweir     for ( SCTAB i=0; i<nTab; i++ )
181*cdf0e10cSrcweir         nRet += nPages[i];
182*cdf0e10cSrcweir     return nRet;
183*cdf0e10cSrcweir }
184*cdf0e10cSrcweir 
185*cdf0e10cSrcweir long ScPrintFuncCache::GetDisplayStart( SCTAB nTab ) const
186*cdf0e10cSrcweir {
187*cdf0e10cSrcweir     //! merge with lcl_GetDisplayStart in preview?
188*cdf0e10cSrcweir 
189*cdf0e10cSrcweir     long nDisplayStart = 0;
190*cdf0e10cSrcweir     ScDocument* pDoc = pDocSh->GetDocument();
191*cdf0e10cSrcweir     for (SCTAB i=0; i<nTab; i++)
192*cdf0e10cSrcweir     {
193*cdf0e10cSrcweir         if ( pDoc->NeedPageResetAfterTab(i) )
194*cdf0e10cSrcweir             nDisplayStart = 0;
195*cdf0e10cSrcweir         else
196*cdf0e10cSrcweir             nDisplayStart += nPages[i];
197*cdf0e10cSrcweir     }
198*cdf0e10cSrcweir     return nDisplayStart;
199*cdf0e10cSrcweir }
200*cdf0e10cSrcweir 
201*cdf0e10cSrcweir 
202