1*ca5ec200SAndrew Rist /**************************************************************
2cdf0e10cSrcweir  *
3*ca5ec200SAndrew Rist  * Licensed to the Apache Software Foundation (ASF) under one
4*ca5ec200SAndrew Rist  * or more contributor license agreements.  See the NOTICE file
5*ca5ec200SAndrew Rist  * distributed with this work for additional information
6*ca5ec200SAndrew Rist  * regarding copyright ownership.  The ASF licenses this file
7*ca5ec200SAndrew Rist  * to you under the Apache License, Version 2.0 (the
8*ca5ec200SAndrew Rist  * "License"); you may not use this file except in compliance
9*ca5ec200SAndrew Rist  * with the License.  You may obtain a copy of the License at
10*ca5ec200SAndrew Rist  *
11*ca5ec200SAndrew Rist  *   http://www.apache.org/licenses/LICENSE-2.0
12*ca5ec200SAndrew Rist  *
13*ca5ec200SAndrew Rist  * Unless required by applicable law or agreed to in writing,
14*ca5ec200SAndrew Rist  * software distributed under the License is distributed on an
15*ca5ec200SAndrew Rist  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16*ca5ec200SAndrew Rist  * KIND, either express or implied.  See the License for the
17*ca5ec200SAndrew Rist  * specific language governing permissions and limitations
18*ca5ec200SAndrew Rist  * under the License.
19*ca5ec200SAndrew Rist  *
20*ca5ec200SAndrew Rist  *************************************************************/
21*ca5ec200SAndrew Rist 
22*ca5ec200SAndrew Rist 
23cdf0e10cSrcweir 
24cdf0e10cSrcweir #include "oox/xls/pivotcachebuffer.hxx"
25cdf0e10cSrcweir 
26cdf0e10cSrcweir #include <set>
27cdf0e10cSrcweir #include <com/sun/star/container/XIndexAccess.hpp>
28cdf0e10cSrcweir #include <com/sun/star/container/XNameAccess.hpp>
29cdf0e10cSrcweir #include <com/sun/star/container/XNamed.hpp>
30cdf0e10cSrcweir #include <com/sun/star/sheet/DataPilotFieldGroupBy.hpp>
31cdf0e10cSrcweir #include <com/sun/star/sheet/DataPilotFieldGroupInfo.hpp>
32cdf0e10cSrcweir #include <com/sun/star/sheet/XDataPilotFieldGrouping.hpp>
33cdf0e10cSrcweir #include <rtl/ustrbuf.hxx>
34cdf0e10cSrcweir #include "oox/core/filterbase.hxx"
35cdf0e10cSrcweir #include "oox/helper/attributelist.hxx"
36cdf0e10cSrcweir #include "oox/helper/containerhelper.hxx"
37cdf0e10cSrcweir #include "oox/helper/propertyset.hxx"
38cdf0e10cSrcweir #include "oox/xls/biffinputstream.hxx"
39cdf0e10cSrcweir #include "oox/xls/defnamesbuffer.hxx"
40cdf0e10cSrcweir #include "oox/xls/excelhandlers.hxx"
41cdf0e10cSrcweir #include "oox/xls/pivotcachefragment.hxx"
42cdf0e10cSrcweir #include "oox/xls/sheetdatabuffer.hxx"
43cdf0e10cSrcweir #include "oox/xls/tablebuffer.hxx"
44cdf0e10cSrcweir #include "oox/xls/unitconverter.hxx"
45cdf0e10cSrcweir #include "oox/xls/worksheetbuffer.hxx"
46cdf0e10cSrcweir 
47cdf0e10cSrcweir namespace oox {
48cdf0e10cSrcweir namespace xls {
49cdf0e10cSrcweir 
50cdf0e10cSrcweir // ============================================================================
51cdf0e10cSrcweir 
52cdf0e10cSrcweir using namespace ::com::sun::star::container;
53cdf0e10cSrcweir using namespace ::com::sun::star::sheet;
54cdf0e10cSrcweir using namespace ::com::sun::star::table;
55cdf0e10cSrcweir using namespace ::com::sun::star::uno;
56cdf0e10cSrcweir using namespace ::com::sun::star::util;
57cdf0e10cSrcweir 
58cdf0e10cSrcweir using ::oox::core::Relations;
59cdf0e10cSrcweir using ::rtl::OUString;
60cdf0e10cSrcweir using ::rtl::OUStringBuffer;
61cdf0e10cSrcweir 
62cdf0e10cSrcweir // ============================================================================
63cdf0e10cSrcweir 
64cdf0e10cSrcweir namespace {
65cdf0e10cSrcweir 
66cdf0e10cSrcweir const sal_uInt16 BIFF12_PCDFIELD_SERVERFIELD        = 0x0001;
67cdf0e10cSrcweir const sal_uInt16 BIFF12_PCDFIELD_NOUNIQUEITEMS      = 0x0002;
68cdf0e10cSrcweir const sal_uInt16 BIFF12_PCDFIELD_DATABASEFIELD      = 0x0004;
69cdf0e10cSrcweir const sal_uInt16 BIFF12_PCDFIELD_HASCAPTION         = 0x0008;
70cdf0e10cSrcweir const sal_uInt16 BIFF12_PCDFIELD_MEMBERPROPFIELD    = 0x0010;
71cdf0e10cSrcweir const sal_uInt16 BIFF12_PCDFIELD_HASFORMULA         = 0x0100;
72cdf0e10cSrcweir const sal_uInt16 BIFF12_PCDFIELD_HASPROPERTYNAME    = 0x0200;
73cdf0e10cSrcweir 
74cdf0e10cSrcweir const sal_uInt16 BIFF12_PCDFSITEMS_HASSEMIMIXED     = 0x0001;
75cdf0e10cSrcweir const sal_uInt16 BIFF12_PCDFSITEMS_HASNONDATE       = 0x0002;
76cdf0e10cSrcweir const sal_uInt16 BIFF12_PCDFSITEMS_HASDATE          = 0x0004;
77cdf0e10cSrcweir const sal_uInt16 BIFF12_PCDFSITEMS_HASSTRING        = 0x0008;
78cdf0e10cSrcweir const sal_uInt16 BIFF12_PCDFSITEMS_HASBLANK         = 0x0010;
79cdf0e10cSrcweir const sal_uInt16 BIFF12_PCDFSITEMS_HASMIXED         = 0x0020;
80cdf0e10cSrcweir const sal_uInt16 BIFF12_PCDFSITEMS_ISNUMERIC        = 0x0040;
81cdf0e10cSrcweir const sal_uInt16 BIFF12_PCDFSITEMS_ISINTEGER        = 0x0080;
82cdf0e10cSrcweir const sal_uInt16 BIFF12_PCDFSITEMS_HASMINMAX        = 0x0100;
83cdf0e10cSrcweir const sal_uInt16 BIFF12_PCDFSITEMS_HASLONGTEXT      = 0x0200;
84cdf0e10cSrcweir 
85cdf0e10cSrcweir const sal_uInt16 BIFF12_PCITEM_ARRAY_DOUBLE         = 0x0001;
86cdf0e10cSrcweir const sal_uInt16 BIFF12_PCITEM_ARRAY_STRING         = 0x0002;
87cdf0e10cSrcweir const sal_uInt16 BIFF12_PCITEM_ARRAY_ERROR          = 0x0010;
88cdf0e10cSrcweir const sal_uInt16 BIFF12_PCITEM_ARRAY_DATE           = 0x0020;
89cdf0e10cSrcweir 
90cdf0e10cSrcweir const sal_uInt8 BIFF12_PCDFRANGEPR_AUTOSTART        = 0x01;
91cdf0e10cSrcweir const sal_uInt8 BIFF12_PCDFRANGEPR_AUTOEND          = 0x02;
92cdf0e10cSrcweir const sal_uInt8 BIFF12_PCDFRANGEPR_DATEGROUP        = 0x04;
93cdf0e10cSrcweir 
94cdf0e10cSrcweir const sal_uInt8 BIFF12_PCDEFINITION_SAVEDATA        = 0x01;
95cdf0e10cSrcweir const sal_uInt8 BIFF12_PCDEFINITION_INVALID         = 0x02;
96cdf0e10cSrcweir const sal_uInt8 BIFF12_PCDEFINITION_REFRESHONLOAD   = 0x04;
97cdf0e10cSrcweir const sal_uInt8 BIFF12_PCDEFINITION_OPTIMIZEMEMORY  = 0x08;
98cdf0e10cSrcweir const sal_uInt8 BIFF12_PCDEFINITION_ENABLEREFRESH   = 0x10;
99cdf0e10cSrcweir const sal_uInt8 BIFF12_PCDEFINITION_BACKGROUNDQUERY = 0x20;
100cdf0e10cSrcweir const sal_uInt8 BIFF12_PCDEFINITION_UPGRADEONREFR   = 0x40;
101cdf0e10cSrcweir const sal_uInt8 BIFF12_PCDEFINITION_TUPELCACHE      = 0x80;
102cdf0e10cSrcweir 
103cdf0e10cSrcweir const sal_uInt8 BIFF12_PCDEFINITION_HASUSERNAME     = 0x01;
104cdf0e10cSrcweir const sal_uInt8 BIFF12_PCDEFINITION_HASRELID        = 0x02;
105cdf0e10cSrcweir const sal_uInt8 BIFF12_PCDEFINITION_SUPPORTSUBQUERY = 0x04;
106cdf0e10cSrcweir const sal_uInt8 BIFF12_PCDEFINITION_SUPPORTDRILL    = 0x08;
107cdf0e10cSrcweir 
108cdf0e10cSrcweir const sal_uInt8 BIFF12_PCDWBSOURCE_HASRELID         = 0x01;
109cdf0e10cSrcweir const sal_uInt8 BIFF12_PCDWBSOURCE_HASSHEET         = 0x02;
110cdf0e10cSrcweir 
111cdf0e10cSrcweir // ----------------------------------------------------------------------------
112cdf0e10cSrcweir 
113cdf0e10cSrcweir const sal_uInt16 BIFF_PCDSOURCE_WORKSHEET           = 0x0001;
114cdf0e10cSrcweir const sal_uInt16 BIFF_PCDSOURCE_EXTERNAL            = 0x0002;
115cdf0e10cSrcweir const sal_uInt16 BIFF_PCDSOURCE_CONSOLIDATION       = 0x0004;
116cdf0e10cSrcweir const sal_uInt16 BIFF_PCDSOURCE_SCENARIO            = 0x0010;
117cdf0e10cSrcweir 
118cdf0e10cSrcweir const sal_uInt16 BIFF_PC_NOSTRING                   = 0xFFFF;
119cdf0e10cSrcweir 
120cdf0e10cSrcweir const sal_uInt16 BIFF_PCDFIELD_HASITEMS             = 0x0001;
121cdf0e10cSrcweir const sal_uInt16 BIFF_PCDFIELD_HASUNSHAREDITEMS     = 0x0002;
122cdf0e10cSrcweir const sal_uInt16 BIFF_PCDFIELD_CALCULATED           = 0x0004;
123cdf0e10cSrcweir const sal_uInt16 BIFF_PCDFIELD_HASPARENT            = 0x0008;
124cdf0e10cSrcweir const sal_uInt16 BIFF_PCDFIELD_RANGEGROUP           = 0x0010;
125cdf0e10cSrcweir const sal_uInt16 BIFF_PCDFIELD_ISNUMERIC            = 0x0020;
126cdf0e10cSrcweir const sal_uInt16 BIFF_PCDFIELD_HASSEMIMIXED         = 0x0080;
127cdf0e10cSrcweir const sal_uInt16 BIFF_PCDFIELD_HASMINMAX            = 0x0100;
128cdf0e10cSrcweir const sal_uInt16 BIFF_PCDFIELD_HASLONGINDEX         = 0x0200;
129cdf0e10cSrcweir const sal_uInt16 BIFF_PCDFIELD_HASNONDATE           = 0x0400;
130cdf0e10cSrcweir const sal_uInt16 BIFF_PCDFIELD_HASDATE              = 0x0800;
131cdf0e10cSrcweir const sal_uInt16 BIFF_PCDFIELD_SERVERFIELD          = 0x2000;
132cdf0e10cSrcweir const sal_uInt16 BIFF_PCDFIELD_NOUNIQUEITEMS        = 0x4000;
133cdf0e10cSrcweir 
134cdf0e10cSrcweir const sal_uInt16 BIFF_PCDFRANGEPR_AUTOSTART         = 0x0001;
135cdf0e10cSrcweir const sal_uInt16 BIFF_PCDFRANGEPR_AUTOEND           = 0x0002;
136cdf0e10cSrcweir 
137cdf0e10cSrcweir const sal_uInt16 BIFF_PCDEFINITION_SAVEDATA         = 0x0001;
138cdf0e10cSrcweir const sal_uInt16 BIFF_PCDEFINITION_INVALID          = 0x0002;
139cdf0e10cSrcweir const sal_uInt16 BIFF_PCDEFINITION_REFRESHONLOAD    = 0x0004;
140cdf0e10cSrcweir const sal_uInt16 BIFF_PCDEFINITION_OPTIMIZEMEMORY   = 0x0008;
141cdf0e10cSrcweir const sal_uInt16 BIFF_PCDEFINITION_BACKGROUNDQUERY  = 0x0010;
142cdf0e10cSrcweir const sal_uInt16 BIFF_PCDEFINITION_ENABLEREFRESH    = 0x0020;
143cdf0e10cSrcweir 
144cdf0e10cSrcweir // ----------------------------------------------------------------------------
145cdf0e10cSrcweir 
146cdf0e10cSrcweir /** Adjusts the weird date format read from binary streams.
147cdf0e10cSrcweir 
148cdf0e10cSrcweir     Dates before 1900-Mar-01 are stored including the non-existing leap day
149cdf0e10cSrcweir     1900-02-29. Time values (without date) are stored as times of day
150cdf0e10cSrcweir     1900-Jan-00. Nothing has to be done when the workbook is stored in 1904
151cdf0e10cSrcweir     date mode (dates before 1904-Jan-01 will not occur in this case).
152cdf0e10cSrcweir  */
lclAdjustBinDateTime(DateTime & orDateTime)153cdf0e10cSrcweir void lclAdjustBinDateTime( DateTime& orDateTime )
154cdf0e10cSrcweir {
155cdf0e10cSrcweir     if( (orDateTime.Year == 1900) && (orDateTime.Month <= 2) )
156cdf0e10cSrcweir     {
157cdf0e10cSrcweir         OSL_ENSURE( (orDateTime.Month == 1) || ((orDateTime.Month == 2) && (orDateTime.Day > 0)), "lclAdjustBinDateTime - invalid date" );
158cdf0e10cSrcweir         switch( orDateTime.Month )
159cdf0e10cSrcweir         {
160cdf0e10cSrcweir             case 2: if( orDateTime.Day > 1 ) --orDateTime.Day; else { orDateTime.Day += 30; --orDateTime.Month; }                       break;
161cdf0e10cSrcweir             case 1: if( orDateTime.Day > 1 ) --orDateTime.Day; else { orDateTime.Day += 30; orDateTime.Month = 12; --orDateTime.Year; } break;
162cdf0e10cSrcweir         }
163cdf0e10cSrcweir     }
164cdf0e10cSrcweir }
165cdf0e10cSrcweir 
166cdf0e10cSrcweir } // namespace
167cdf0e10cSrcweir 
168cdf0e10cSrcweir // ============================================================================
169cdf0e10cSrcweir 
PivotCacheItem()170cdf0e10cSrcweir PivotCacheItem::PivotCacheItem() :
171cdf0e10cSrcweir     mnType( XML_m )
172cdf0e10cSrcweir {
173cdf0e10cSrcweir }
174cdf0e10cSrcweir 
readString(const AttributeList & rAttribs)175cdf0e10cSrcweir void PivotCacheItem::readString( const AttributeList& rAttribs )
176cdf0e10cSrcweir {
177cdf0e10cSrcweir     maValue <<= rAttribs.getXString( XML_v, OUString() );
178cdf0e10cSrcweir     mnType = XML_s;
179cdf0e10cSrcweir }
180cdf0e10cSrcweir 
readNumeric(const AttributeList & rAttribs)181cdf0e10cSrcweir void PivotCacheItem::readNumeric( const AttributeList& rAttribs )
182cdf0e10cSrcweir {
183cdf0e10cSrcweir     maValue <<= rAttribs.getDouble( XML_v, 0.0 );
184cdf0e10cSrcweir     mnType = XML_n;
185cdf0e10cSrcweir }
186cdf0e10cSrcweir 
readDate(const AttributeList & rAttribs)187cdf0e10cSrcweir void PivotCacheItem::readDate( const AttributeList& rAttribs )
188cdf0e10cSrcweir {
189cdf0e10cSrcweir     maValue <<= rAttribs.getDateTime( XML_v, DateTime() );
190cdf0e10cSrcweir     mnType = XML_d;
191cdf0e10cSrcweir }
192cdf0e10cSrcweir 
readBool(const AttributeList & rAttribs)193cdf0e10cSrcweir void PivotCacheItem::readBool( const AttributeList& rAttribs )
194cdf0e10cSrcweir {
195cdf0e10cSrcweir     maValue <<= rAttribs.getBool( XML_v, false );
196cdf0e10cSrcweir     mnType = XML_b;
197cdf0e10cSrcweir }
198cdf0e10cSrcweir 
readError(const AttributeList & rAttribs,const UnitConverter & rUnitConverter)199cdf0e10cSrcweir void PivotCacheItem::readError( const AttributeList& rAttribs, const UnitConverter& rUnitConverter )
200cdf0e10cSrcweir {
201cdf0e10cSrcweir     maValue <<= static_cast< sal_Int32 >( rUnitConverter.calcBiffErrorCode( rAttribs.getXString( XML_v, OUString() ) ) );
202cdf0e10cSrcweir     mnType = XML_e;
203cdf0e10cSrcweir }
204cdf0e10cSrcweir 
readIndex(const AttributeList & rAttribs)205cdf0e10cSrcweir void PivotCacheItem::readIndex( const AttributeList& rAttribs )
206cdf0e10cSrcweir {
207cdf0e10cSrcweir     maValue <<= rAttribs.getInteger( XML_v, -1 );
208cdf0e10cSrcweir     mnType = XML_x;
209cdf0e10cSrcweir }
210cdf0e10cSrcweir 
readString(SequenceInputStream & rStrm)211cdf0e10cSrcweir void PivotCacheItem::readString( SequenceInputStream& rStrm )
212cdf0e10cSrcweir {
213cdf0e10cSrcweir     maValue <<= BiffHelper::readString( rStrm );
214cdf0e10cSrcweir     mnType = XML_s;
215cdf0e10cSrcweir }
216cdf0e10cSrcweir 
readDouble(SequenceInputStream & rStrm)217cdf0e10cSrcweir void PivotCacheItem::readDouble( SequenceInputStream& rStrm )
218cdf0e10cSrcweir {
219cdf0e10cSrcweir     maValue <<= rStrm.readDouble();
220cdf0e10cSrcweir     mnType = XML_n;
221cdf0e10cSrcweir }
222cdf0e10cSrcweir 
readDate(SequenceInputStream & rStrm)223cdf0e10cSrcweir void PivotCacheItem::readDate( SequenceInputStream& rStrm )
224cdf0e10cSrcweir {
225cdf0e10cSrcweir     DateTime aDateTime;
226cdf0e10cSrcweir     aDateTime.Year = rStrm.readuInt16();
227cdf0e10cSrcweir     aDateTime.Month = rStrm.readuInt16();
228cdf0e10cSrcweir     aDateTime.Day = rStrm.readuInt8();
229cdf0e10cSrcweir     aDateTime.Hours = rStrm.readuInt8();
230cdf0e10cSrcweir     aDateTime.Minutes = rStrm.readuInt8();
231cdf0e10cSrcweir     aDateTime.Seconds = rStrm.readuInt8();
232cdf0e10cSrcweir     lclAdjustBinDateTime( aDateTime );
233cdf0e10cSrcweir     maValue <<= aDateTime;
234cdf0e10cSrcweir     mnType = XML_d;
235cdf0e10cSrcweir }
236cdf0e10cSrcweir 
readBool(SequenceInputStream & rStrm)237cdf0e10cSrcweir void PivotCacheItem::readBool( SequenceInputStream& rStrm )
238cdf0e10cSrcweir {
239cdf0e10cSrcweir     maValue <<= (rStrm.readuInt8() != 0);
240cdf0e10cSrcweir     mnType = XML_b;
241cdf0e10cSrcweir }
242cdf0e10cSrcweir 
readError(SequenceInputStream & rStrm)243cdf0e10cSrcweir void PivotCacheItem::readError( SequenceInputStream& rStrm )
244cdf0e10cSrcweir {
245cdf0e10cSrcweir     maValue <<= static_cast< sal_Int32 >( rStrm.readuInt8() );
246cdf0e10cSrcweir     mnType = XML_e;
247cdf0e10cSrcweir }
248cdf0e10cSrcweir 
readIndex(SequenceInputStream & rStrm)249cdf0e10cSrcweir void PivotCacheItem::readIndex( SequenceInputStream& rStrm )
250cdf0e10cSrcweir {
251cdf0e10cSrcweir     maValue <<= rStrm.readInt32();
252cdf0e10cSrcweir     mnType = XML_x;
253cdf0e10cSrcweir }
254cdf0e10cSrcweir 
readString(BiffInputStream & rStrm,const WorkbookHelper & rHelper)255cdf0e10cSrcweir void PivotCacheItem::readString( BiffInputStream& rStrm, const WorkbookHelper& rHelper )
256cdf0e10cSrcweir {
257cdf0e10cSrcweir     maValue <<= (rHelper.getBiff() == BIFF8) ? rStrm.readUniString() : rStrm.readByteStringUC( true, rHelper.getTextEncoding() );
258cdf0e10cSrcweir     mnType = XML_s;
259cdf0e10cSrcweir }
260cdf0e10cSrcweir 
readDouble(BiffInputStream & rStrm)261cdf0e10cSrcweir void PivotCacheItem::readDouble( BiffInputStream& rStrm )
262cdf0e10cSrcweir {
263cdf0e10cSrcweir     maValue <<= rStrm.readDouble();
264cdf0e10cSrcweir     mnType = XML_n;
265cdf0e10cSrcweir }
266cdf0e10cSrcweir 
readInteger(BiffInputStream & rStrm)267cdf0e10cSrcweir void PivotCacheItem::readInteger( BiffInputStream& rStrm )
268cdf0e10cSrcweir {
269cdf0e10cSrcweir     maValue <<= rStrm.readInt16();
270cdf0e10cSrcweir     mnType = XML_i;                 // fake, used for BIFF only
271cdf0e10cSrcweir }
272cdf0e10cSrcweir 
readDate(BiffInputStream & rStrm)273cdf0e10cSrcweir void PivotCacheItem::readDate( BiffInputStream& rStrm )
274cdf0e10cSrcweir {
275cdf0e10cSrcweir     DateTime aDateTime;
276cdf0e10cSrcweir     aDateTime.Year = rStrm.readuInt16();
277cdf0e10cSrcweir     aDateTime.Month = rStrm.readuInt16();
278cdf0e10cSrcweir     aDateTime.Day = rStrm.readuInt8();
279cdf0e10cSrcweir     aDateTime.Hours = rStrm.readuInt8();
280cdf0e10cSrcweir     aDateTime.Minutes = rStrm.readuInt8();
281cdf0e10cSrcweir     aDateTime.Seconds = rStrm.readuInt8();
282cdf0e10cSrcweir     lclAdjustBinDateTime( aDateTime );
283cdf0e10cSrcweir     maValue <<= aDateTime;
284cdf0e10cSrcweir     mnType = XML_d;
285cdf0e10cSrcweir }
286cdf0e10cSrcweir 
readBool(BiffInputStream & rStrm)287cdf0e10cSrcweir void PivotCacheItem::readBool( BiffInputStream& rStrm )
288cdf0e10cSrcweir {
289cdf0e10cSrcweir     maValue <<= (rStrm.readuInt8() != 0);
290cdf0e10cSrcweir     mnType = XML_b;
291cdf0e10cSrcweir }
292cdf0e10cSrcweir 
readError(BiffInputStream & rStrm)293cdf0e10cSrcweir void PivotCacheItem::readError( BiffInputStream& rStrm )
294cdf0e10cSrcweir {
295cdf0e10cSrcweir     maValue <<= static_cast< sal_Int32 >( rStrm.readuInt8() );
296cdf0e10cSrcweir     mnType = XML_e;
297cdf0e10cSrcweir }
298cdf0e10cSrcweir 
getName() const299cdf0e10cSrcweir OUString PivotCacheItem::getName() const
300cdf0e10cSrcweir {
301cdf0e10cSrcweir     switch( mnType )
302cdf0e10cSrcweir     {
303cdf0e10cSrcweir         case XML_m: return OUString();
304cdf0e10cSrcweir         case XML_s: return maValue.get< OUString >();
305cdf0e10cSrcweir         case XML_n: return OUString::valueOf( maValue.get< double >() );                            // !TODO
306cdf0e10cSrcweir         case XML_i: return OUString::valueOf( maValue.get< sal_Int32 >() );
307cdf0e10cSrcweir         case XML_d: return OUString();                                                              // !TODO
308cdf0e10cSrcweir         case XML_b: return OUString::valueOf( static_cast< sal_Bool >( maValue.get< bool >() ) );   // !TODO
309cdf0e10cSrcweir         case XML_e: return OUString();                                                              // !TODO
310cdf0e10cSrcweir     }
311cdf0e10cSrcweir     OSL_ENSURE( false, "PivotCacheItem::getName - invalid data type" );
312cdf0e10cSrcweir     return OUString();
313cdf0e10cSrcweir }
314cdf0e10cSrcweir 
315cdf0e10cSrcweir // ----------------------------------------------------------------------------
316cdf0e10cSrcweir 
PivotCacheItemList(const WorkbookHelper & rHelper)317cdf0e10cSrcweir PivotCacheItemList::PivotCacheItemList( const WorkbookHelper& rHelper ) :
318cdf0e10cSrcweir     WorkbookHelper( rHelper )
319cdf0e10cSrcweir {
320cdf0e10cSrcweir }
321cdf0e10cSrcweir 
importItem(sal_Int32 nElement,const AttributeList & rAttribs)322cdf0e10cSrcweir void PivotCacheItemList::importItem( sal_Int32 nElement, const AttributeList& rAttribs )
323cdf0e10cSrcweir {
324cdf0e10cSrcweir     PivotCacheItem& rItem = createItem();
325cdf0e10cSrcweir     switch( nElement )
326cdf0e10cSrcweir     {
327cdf0e10cSrcweir         case XLS_TOKEN( m ):                                                        break;
328cdf0e10cSrcweir         case XLS_TOKEN( s ):    rItem.readString( rAttribs );                       break;
329cdf0e10cSrcweir         case XLS_TOKEN( n ):    rItem.readNumeric( rAttribs );                      break;
330cdf0e10cSrcweir         case XLS_TOKEN( d ):    rItem.readDate( rAttribs );                         break;
331cdf0e10cSrcweir         case XLS_TOKEN( b ):    rItem.readBool( rAttribs );                         break;
332cdf0e10cSrcweir         case XLS_TOKEN( e ):    rItem.readError( rAttribs, getUnitConverter() );    break;
333cdf0e10cSrcweir         default:    OSL_ENSURE( false, "PivotCacheItemList::importItem - unknown element type" );
334cdf0e10cSrcweir     }
335cdf0e10cSrcweir }
336cdf0e10cSrcweir 
importItem(sal_Int32 nRecId,SequenceInputStream & rStrm)337cdf0e10cSrcweir void PivotCacheItemList::importItem( sal_Int32 nRecId, SequenceInputStream& rStrm )
338cdf0e10cSrcweir {
339cdf0e10cSrcweir     if( nRecId == BIFF12_ID_PCITEM_ARRAY )
340cdf0e10cSrcweir     {
341cdf0e10cSrcweir         importArray( rStrm );
342cdf0e10cSrcweir         return;
343cdf0e10cSrcweir     }
344cdf0e10cSrcweir 
345cdf0e10cSrcweir     PivotCacheItem& rItem = createItem();
346cdf0e10cSrcweir     switch( nRecId )
347cdf0e10cSrcweir     {
348cdf0e10cSrcweir         case BIFF12_ID_PCITEM_MISSING:
349cdf0e10cSrcweir         case BIFF12_ID_PCITEMA_MISSING:                             break;
350cdf0e10cSrcweir         case BIFF12_ID_PCITEM_STRING:
351cdf0e10cSrcweir         case BIFF12_ID_PCITEMA_STRING:  rItem.readString( rStrm );  break;
352cdf0e10cSrcweir         case BIFF12_ID_PCITEM_DOUBLE:
353cdf0e10cSrcweir         case BIFF12_ID_PCITEMA_DOUBLE:  rItem.readDouble( rStrm );  break;
354cdf0e10cSrcweir         case BIFF12_ID_PCITEM_DATE:
355cdf0e10cSrcweir         case BIFF12_ID_PCITEMA_DATE:    rItem.readDate( rStrm );    break;
356cdf0e10cSrcweir         case BIFF12_ID_PCITEM_BOOL:
357cdf0e10cSrcweir         case BIFF12_ID_PCITEMA_BOOL:    rItem.readBool( rStrm );    break;
358cdf0e10cSrcweir         case BIFF12_ID_PCITEM_ERROR:
359cdf0e10cSrcweir         case BIFF12_ID_PCITEMA_ERROR:   rItem.readError( rStrm );   break;
360cdf0e10cSrcweir         default:    OSL_ENSURE( false, "PivotCacheItemList::importItem - unknown record type" );
361cdf0e10cSrcweir     }
362cdf0e10cSrcweir }
363cdf0e10cSrcweir 
importItemList(BiffInputStream & rStrm,sal_uInt16 nCount)364cdf0e10cSrcweir void PivotCacheItemList::importItemList( BiffInputStream& rStrm, sal_uInt16 nCount )
365cdf0e10cSrcweir {
366cdf0e10cSrcweir     bool bLoop = true;
367cdf0e10cSrcweir     for( sal_uInt16 nItemIdx = 0; bLoop && (nItemIdx < nCount); ++nItemIdx )
368cdf0e10cSrcweir     {
369cdf0e10cSrcweir         bLoop = rStrm.startNextRecord();
370cdf0e10cSrcweir         if( bLoop ) switch( rStrm.getRecId() )
371cdf0e10cSrcweir         {
372cdf0e10cSrcweir             case BIFF_ID_PCITEM_MISSING:    createItem();                               break;
373cdf0e10cSrcweir             case BIFF_ID_PCITEM_STRING:     createItem().readString( rStrm, *this );    break;
374cdf0e10cSrcweir             case BIFF_ID_PCITEM_DOUBLE:     createItem().readDouble( rStrm );           break;
375cdf0e10cSrcweir             case BIFF_ID_PCITEM_INTEGER:    createItem().readInteger( rStrm );          break;
376cdf0e10cSrcweir             case BIFF_ID_PCITEM_DATE:       createItem().readDate( rStrm );             break;
377cdf0e10cSrcweir             case BIFF_ID_PCITEM_BOOL:       createItem().readBool( rStrm );             break;
378cdf0e10cSrcweir             case BIFF_ID_PCITEM_ERROR:      createItem().readError( rStrm );            break;
379cdf0e10cSrcweir             default:                        rStrm.rewindRecord(); bLoop = false;
380cdf0e10cSrcweir         }
381cdf0e10cSrcweir     }
382cdf0e10cSrcweir     OSL_ENSURE( bLoop, "PivotCacheItemList::importItemList - could not read all cache item records" );
383cdf0e10cSrcweir }
384cdf0e10cSrcweir 
getCacheItem(sal_Int32 nItemIdx) const385cdf0e10cSrcweir const PivotCacheItem* PivotCacheItemList::getCacheItem( sal_Int32 nItemIdx ) const
386cdf0e10cSrcweir {
387cdf0e10cSrcweir     return ContainerHelper::getVectorElement( maItems, nItemIdx );
388cdf0e10cSrcweir }
389cdf0e10cSrcweir 
getCacheItemNames(::std::vector<OUString> & orItemNames) const390cdf0e10cSrcweir void PivotCacheItemList::getCacheItemNames( ::std::vector< OUString >& orItemNames ) const
391cdf0e10cSrcweir {
392cdf0e10cSrcweir     orItemNames.clear();
393cdf0e10cSrcweir     orItemNames.reserve( maItems.size() );
394cdf0e10cSrcweir     for( CacheItemVector::const_iterator aIt = maItems.begin(), aEnd = maItems.end(); aIt != aEnd; ++aIt )
395cdf0e10cSrcweir         orItemNames.push_back( aIt->getName() );
396cdf0e10cSrcweir }
397cdf0e10cSrcweir 
398cdf0e10cSrcweir // private --------------------------------------------------------------------
399cdf0e10cSrcweir 
createItem()400cdf0e10cSrcweir PivotCacheItem& PivotCacheItemList::createItem()
401cdf0e10cSrcweir {
402cdf0e10cSrcweir     maItems.resize( maItems.size() + 1 );
403cdf0e10cSrcweir     return maItems.back();
404cdf0e10cSrcweir }
405cdf0e10cSrcweir 
importArray(SequenceInputStream & rStrm)406cdf0e10cSrcweir void PivotCacheItemList::importArray( SequenceInputStream& rStrm )
407cdf0e10cSrcweir {
408cdf0e10cSrcweir     sal_uInt16 nType = rStrm.readuInt16();
409cdf0e10cSrcweir     sal_Int32 nCount = rStrm.readInt32();
410cdf0e10cSrcweir     for( sal_Int32 nIdx = 0; !rStrm.isEof() && (nIdx < nCount); ++nIdx )
411cdf0e10cSrcweir     {
412cdf0e10cSrcweir         switch( nType )
413cdf0e10cSrcweir         {
414cdf0e10cSrcweir             case BIFF12_PCITEM_ARRAY_DOUBLE: createItem().readDouble( rStrm );   break;
415cdf0e10cSrcweir             case BIFF12_PCITEM_ARRAY_STRING: createItem().readString( rStrm );   break;
416cdf0e10cSrcweir             case BIFF12_PCITEM_ARRAY_ERROR:  createItem().readError( rStrm );    break;
417cdf0e10cSrcweir             case BIFF12_PCITEM_ARRAY_DATE:   createItem().readDate( rStrm );     break;
418cdf0e10cSrcweir             default:
419cdf0e10cSrcweir                 OSL_ENSURE( false, "PivotCacheItemList::importArray - unknown data type" );
420cdf0e10cSrcweir                 nIdx = nCount;
421cdf0e10cSrcweir         }
422cdf0e10cSrcweir     }
423cdf0e10cSrcweir }
424cdf0e10cSrcweir 
425cdf0e10cSrcweir // ============================================================================
426cdf0e10cSrcweir 
PCFieldModel()427cdf0e10cSrcweir PCFieldModel::PCFieldModel() :
428cdf0e10cSrcweir     mnNumFmtId( 0 ),
429cdf0e10cSrcweir     mnSqlType( 0 ),
430cdf0e10cSrcweir     mnHierarchy( 0 ),
431cdf0e10cSrcweir     mnLevel( 0 ),
432cdf0e10cSrcweir     mnMappingCount( 0 ),
433cdf0e10cSrcweir     mbDatabaseField( true ),
434cdf0e10cSrcweir     mbServerField( false ),
435cdf0e10cSrcweir     mbUniqueList( true ),
436cdf0e10cSrcweir     mbMemberPropField( false )
437cdf0e10cSrcweir {
438cdf0e10cSrcweir }
439cdf0e10cSrcweir 
440cdf0e10cSrcweir // ----------------------------------------------------------------------------
441cdf0e10cSrcweir 
PCSharedItemsModel()442cdf0e10cSrcweir PCSharedItemsModel::PCSharedItemsModel() :
443cdf0e10cSrcweir     mbHasSemiMixed( true ),
444cdf0e10cSrcweir     mbHasNonDate( true ),
445cdf0e10cSrcweir     mbHasDate( false ),
446cdf0e10cSrcweir     mbHasString( true ),
447cdf0e10cSrcweir     mbHasBlank( false ),
448cdf0e10cSrcweir     mbHasMixed( false ),
449cdf0e10cSrcweir     mbIsNumeric( false ),
450cdf0e10cSrcweir     mbIsInteger( false ),
451cdf0e10cSrcweir     mbHasLongText( false ),
452cdf0e10cSrcweir     mbHasLongIndexes( false )
453cdf0e10cSrcweir {
454cdf0e10cSrcweir }
455cdf0e10cSrcweir 
456cdf0e10cSrcweir // ----------------------------------------------------------------------------
457cdf0e10cSrcweir 
PCFieldGroupModel()458cdf0e10cSrcweir PCFieldGroupModel::PCFieldGroupModel() :
459cdf0e10cSrcweir     mfStartValue( 0.0 ),
460cdf0e10cSrcweir     mfEndValue( 0.0 ),
461cdf0e10cSrcweir     mfInterval( 1.0 ),
462cdf0e10cSrcweir     mnParentField( -1 ),
463cdf0e10cSrcweir     mnBaseField( -1 ),
464cdf0e10cSrcweir     mnGroupBy( XML_range ),
465cdf0e10cSrcweir     mbRangeGroup( false ),
466cdf0e10cSrcweir     mbDateGroup( false ),
467cdf0e10cSrcweir     mbAutoStart( true ),
468cdf0e10cSrcweir     mbAutoEnd( true )
469cdf0e10cSrcweir {
470cdf0e10cSrcweir }
471cdf0e10cSrcweir 
setBiffGroupBy(sal_uInt8 nGroupBy)472cdf0e10cSrcweir void PCFieldGroupModel::setBiffGroupBy( sal_uInt8 nGroupBy )
473cdf0e10cSrcweir {
474cdf0e10cSrcweir     static const sal_Int32 spnGroupBy[] = { XML_range,
475cdf0e10cSrcweir         XML_seconds, XML_minutes, XML_hours, XML_days, XML_months, XML_quarters, XML_years };
476cdf0e10cSrcweir     mnGroupBy = STATIC_ARRAY_SELECT( spnGroupBy, nGroupBy, XML_range );
477cdf0e10cSrcweir }
478cdf0e10cSrcweir 
479cdf0e10cSrcweir // ----------------------------------------------------------------------------
480cdf0e10cSrcweir 
PivotCacheField(const WorkbookHelper & rHelper,bool bIsDatabaseField)481cdf0e10cSrcweir PivotCacheField::PivotCacheField( const WorkbookHelper& rHelper, bool bIsDatabaseField ) :
482cdf0e10cSrcweir     WorkbookHelper( rHelper ),
483cdf0e10cSrcweir     maSharedItems( rHelper ),
484cdf0e10cSrcweir     maGroupItems( rHelper )
485cdf0e10cSrcweir {
486cdf0e10cSrcweir     maFieldModel.mbDatabaseField = bIsDatabaseField;
487cdf0e10cSrcweir }
488cdf0e10cSrcweir 
importCacheField(const AttributeList & rAttribs)489cdf0e10cSrcweir void PivotCacheField::importCacheField( const AttributeList& rAttribs )
490cdf0e10cSrcweir {
491cdf0e10cSrcweir     maFieldModel.maName            = rAttribs.getXString( XML_name, OUString() );
492cdf0e10cSrcweir     maFieldModel.maCaption         = rAttribs.getXString( XML_caption, OUString() );
493cdf0e10cSrcweir     maFieldModel.maPropertyName    = rAttribs.getXString( XML_propertyName, OUString() );
494cdf0e10cSrcweir     maFieldModel.maFormula         = rAttribs.getXString( XML_formula, OUString() );
495cdf0e10cSrcweir     maFieldModel.mnNumFmtId        = rAttribs.getInteger( XML_numFmtId, 0 );
496cdf0e10cSrcweir     maFieldModel.mnSqlType         = rAttribs.getInteger( XML_sqlType, 0 );
497cdf0e10cSrcweir     maFieldModel.mnHierarchy       = rAttribs.getInteger( XML_hierarchy, 0 );
498cdf0e10cSrcweir     maFieldModel.mnLevel           = rAttribs.getInteger( XML_level, 0 );
499cdf0e10cSrcweir     maFieldModel.mnMappingCount    = rAttribs.getInteger( XML_mappingCount, 0 );
500cdf0e10cSrcweir     maFieldModel.mbDatabaseField   = rAttribs.getBool( XML_databaseField, true );
501cdf0e10cSrcweir     maFieldModel.mbServerField     = rAttribs.getBool( XML_serverField, false );
502cdf0e10cSrcweir     maFieldModel.mbUniqueList      = rAttribs.getBool( XML_uniqueList, true );
503cdf0e10cSrcweir     maFieldModel.mbMemberPropField = rAttribs.getBool( XML_memberPropertyField, false );
504cdf0e10cSrcweir }
505cdf0e10cSrcweir 
importSharedItems(const AttributeList & rAttribs)506cdf0e10cSrcweir void PivotCacheField::importSharedItems( const AttributeList& rAttribs )
507cdf0e10cSrcweir {
508cdf0e10cSrcweir     OSL_ENSURE( maSharedItems.empty(), "PivotCacheField::importSharedItems - multiple shared items elements" );
509cdf0e10cSrcweir     maSharedItemsModel.mbHasSemiMixed = rAttribs.getBool( XML_containsSemiMixedTypes, true );
510cdf0e10cSrcweir     maSharedItemsModel.mbHasNonDate   = rAttribs.getBool( XML_containsNonDate, true );
511cdf0e10cSrcweir     maSharedItemsModel.mbHasDate      = rAttribs.getBool( XML_containsDate, false );
512cdf0e10cSrcweir     maSharedItemsModel.mbHasString    = rAttribs.getBool( XML_containsString, true );
513cdf0e10cSrcweir     maSharedItemsModel.mbHasBlank     = rAttribs.getBool( XML_containsBlank, false );
514cdf0e10cSrcweir     maSharedItemsModel.mbHasMixed     = rAttribs.getBool( XML_containsMixedTypes, false );
515cdf0e10cSrcweir     maSharedItemsModel.mbIsNumeric    = rAttribs.getBool( XML_containsNumber, false );
516cdf0e10cSrcweir     maSharedItemsModel.mbIsInteger    = rAttribs.getBool( XML_containsInteger, false );
517cdf0e10cSrcweir     maSharedItemsModel.mbHasLongText  = rAttribs.getBool( XML_longText, false );
518cdf0e10cSrcweir }
519cdf0e10cSrcweir 
importSharedItem(sal_Int32 nElement,const AttributeList & rAttribs)520cdf0e10cSrcweir void PivotCacheField::importSharedItem( sal_Int32 nElement, const AttributeList& rAttribs )
521cdf0e10cSrcweir {
522cdf0e10cSrcweir     maSharedItems.importItem( nElement, rAttribs );
523cdf0e10cSrcweir }
524cdf0e10cSrcweir 
importFieldGroup(const AttributeList & rAttribs)525cdf0e10cSrcweir void PivotCacheField::importFieldGroup( const AttributeList& rAttribs )
526cdf0e10cSrcweir {
527cdf0e10cSrcweir     maFieldGroupModel.mnParentField = rAttribs.getInteger( XML_par, -1 );
528cdf0e10cSrcweir     maFieldGroupModel.mnBaseField   = rAttribs.getInteger( XML_base, -1 );
529cdf0e10cSrcweir }
530cdf0e10cSrcweir 
importRangePr(const AttributeList & rAttribs)531cdf0e10cSrcweir void PivotCacheField::importRangePr( const AttributeList& rAttribs )
532cdf0e10cSrcweir {
533cdf0e10cSrcweir     maFieldGroupModel.maStartDate    = rAttribs.getDateTime( XML_startDate, DateTime() );
534cdf0e10cSrcweir     maFieldGroupModel.maEndDate      = rAttribs.getDateTime( XML_endDate, DateTime() );
535cdf0e10cSrcweir     maFieldGroupModel.mfStartValue   = rAttribs.getDouble( XML_startNum, 0.0 );
536cdf0e10cSrcweir     maFieldGroupModel.mfEndValue     = rAttribs.getDouble( XML_endNum, 0.0 );
537cdf0e10cSrcweir     maFieldGroupModel.mfInterval     = rAttribs.getDouble( XML_groupInterval, 1.0 );
538cdf0e10cSrcweir     maFieldGroupModel.mnGroupBy      = rAttribs.getToken( XML_groupBy, XML_range );
539cdf0e10cSrcweir     maFieldGroupModel.mbRangeGroup   = true;
540cdf0e10cSrcweir     maFieldGroupModel.mbDateGroup    = maFieldGroupModel.mnGroupBy != XML_range;
541cdf0e10cSrcweir     maFieldGroupModel.mbAutoStart    = rAttribs.getBool( XML_autoStart, true );
542cdf0e10cSrcweir     maFieldGroupModel.mbAutoEnd      = rAttribs.getBool( XML_autoEnd, true );
543cdf0e10cSrcweir }
544cdf0e10cSrcweir 
importDiscretePrItem(sal_Int32 nElement,const AttributeList & rAttribs)545cdf0e10cSrcweir void PivotCacheField::importDiscretePrItem( sal_Int32 nElement, const AttributeList& rAttribs )
546cdf0e10cSrcweir {
547cdf0e10cSrcweir     OSL_ENSURE( nElement == XLS_TOKEN( x ), "PivotCacheField::importDiscretePrItem - unexpected element" );
548cdf0e10cSrcweir     if( nElement == XLS_TOKEN( x ) )
549cdf0e10cSrcweir         maDiscreteItems.push_back( rAttribs.getInteger( XML_v, -1 ) );
550cdf0e10cSrcweir }
551cdf0e10cSrcweir 
importGroupItem(sal_Int32 nElement,const AttributeList & rAttribs)552cdf0e10cSrcweir void PivotCacheField::importGroupItem( sal_Int32 nElement, const AttributeList& rAttribs )
553cdf0e10cSrcweir {
554cdf0e10cSrcweir     maGroupItems.importItem( nElement, rAttribs );
555cdf0e10cSrcweir }
556cdf0e10cSrcweir 
importPCDField(SequenceInputStream & rStrm)557cdf0e10cSrcweir void PivotCacheField::importPCDField( SequenceInputStream& rStrm )
558cdf0e10cSrcweir {
559cdf0e10cSrcweir     sal_uInt16 nFlags;
560cdf0e10cSrcweir     rStrm >> nFlags >> maFieldModel.mnNumFmtId;
561cdf0e10cSrcweir     maFieldModel.mnSqlType = rStrm.readInt16();
562cdf0e10cSrcweir     rStrm >> maFieldModel.mnHierarchy >> maFieldModel.mnLevel >> maFieldModel.mnMappingCount >> maFieldModel.maName;
563cdf0e10cSrcweir     if( getFlag( nFlags, BIFF12_PCDFIELD_HASCAPTION ) )
564cdf0e10cSrcweir         rStrm >> maFieldModel.maCaption;
565cdf0e10cSrcweir     if( getFlag( nFlags, BIFF12_PCDFIELD_HASFORMULA ) )
566cdf0e10cSrcweir         rStrm.skip( ::std::max< sal_Int32 >( rStrm.readInt32(), 0 ) );
567cdf0e10cSrcweir     if( maFieldModel.mnMappingCount > 0 )
568cdf0e10cSrcweir         rStrm.skip( ::std::max< sal_Int32 >( rStrm.readInt32(), 0 ) );
569cdf0e10cSrcweir     if( getFlag( nFlags, BIFF12_PCDFIELD_HASPROPERTYNAME ) )
570cdf0e10cSrcweir         rStrm >> maFieldModel.maPropertyName;
571cdf0e10cSrcweir 
572cdf0e10cSrcweir     maFieldModel.mbDatabaseField   = getFlag( nFlags, BIFF12_PCDFIELD_DATABASEFIELD );
573cdf0e10cSrcweir     maFieldModel.mbServerField     = getFlag( nFlags, BIFF12_PCDFIELD_SERVERFIELD );
574cdf0e10cSrcweir     maFieldModel.mbUniqueList      = !getFlag( nFlags, BIFF12_PCDFIELD_NOUNIQUEITEMS );
575cdf0e10cSrcweir     maFieldModel.mbMemberPropField = getFlag( nFlags, BIFF12_PCDFIELD_MEMBERPROPFIELD );
576cdf0e10cSrcweir }
577cdf0e10cSrcweir 
importPCDFSharedItems(SequenceInputStream & rStrm)578cdf0e10cSrcweir void PivotCacheField::importPCDFSharedItems( SequenceInputStream& rStrm )
579cdf0e10cSrcweir {
580cdf0e10cSrcweir     sal_uInt16 nFlags;
581cdf0e10cSrcweir     rStrm >> nFlags;
582cdf0e10cSrcweir     maSharedItemsModel.mbHasSemiMixed = getFlag( nFlags, BIFF12_PCDFSITEMS_HASSEMIMIXED );
583cdf0e10cSrcweir     maSharedItemsModel.mbHasNonDate   = getFlag( nFlags, BIFF12_PCDFSITEMS_HASNONDATE );
584cdf0e10cSrcweir     maSharedItemsModel.mbHasDate      = getFlag( nFlags, BIFF12_PCDFSITEMS_HASDATE );
585cdf0e10cSrcweir     maSharedItemsModel.mbHasString    = getFlag( nFlags, BIFF12_PCDFSITEMS_HASSTRING );
586cdf0e10cSrcweir     maSharedItemsModel.mbHasBlank     = getFlag( nFlags, BIFF12_PCDFSITEMS_HASBLANK );
587cdf0e10cSrcweir     maSharedItemsModel.mbHasMixed     = getFlag( nFlags, BIFF12_PCDFSITEMS_HASMIXED );
588cdf0e10cSrcweir     maSharedItemsModel.mbIsNumeric    = getFlag( nFlags, BIFF12_PCDFSITEMS_ISNUMERIC );
589cdf0e10cSrcweir     maSharedItemsModel.mbIsInteger    = getFlag( nFlags, BIFF12_PCDFSITEMS_ISINTEGER );
590cdf0e10cSrcweir     maSharedItemsModel.mbHasLongText  = getFlag( nFlags, BIFF12_PCDFSITEMS_HASLONGTEXT );
591cdf0e10cSrcweir }
592cdf0e10cSrcweir 
importPCDFSharedItem(sal_Int32 nRecId,SequenceInputStream & rStrm)593cdf0e10cSrcweir void PivotCacheField::importPCDFSharedItem( sal_Int32 nRecId, SequenceInputStream& rStrm )
594cdf0e10cSrcweir {
595cdf0e10cSrcweir     maSharedItems.importItem( nRecId, rStrm );
596cdf0e10cSrcweir }
597cdf0e10cSrcweir 
importPCDFieldGroup(SequenceInputStream & rStrm)598cdf0e10cSrcweir void PivotCacheField::importPCDFieldGroup( SequenceInputStream& rStrm )
599cdf0e10cSrcweir {
600cdf0e10cSrcweir     rStrm >> maFieldGroupModel.mnParentField >> maFieldGroupModel.mnBaseField;
601cdf0e10cSrcweir }
602cdf0e10cSrcweir 
importPCDFRangePr(SequenceInputStream & rStrm)603cdf0e10cSrcweir void PivotCacheField::importPCDFRangePr( SequenceInputStream& rStrm )
604cdf0e10cSrcweir {
605cdf0e10cSrcweir     sal_uInt8 nGroupBy, nFlags;
606cdf0e10cSrcweir     rStrm >> nGroupBy >> nFlags >> maFieldGroupModel.mfStartValue >> maFieldGroupModel.mfEndValue >> maFieldGroupModel.mfInterval;
607cdf0e10cSrcweir 
608cdf0e10cSrcweir     maFieldGroupModel.setBiffGroupBy( nGroupBy );
609cdf0e10cSrcweir     maFieldGroupModel.mbRangeGroup   = true;
610cdf0e10cSrcweir     maFieldGroupModel.mbDateGroup    = getFlag( nFlags, BIFF12_PCDFRANGEPR_DATEGROUP );
611cdf0e10cSrcweir     maFieldGroupModel.mbAutoStart    = getFlag( nFlags, BIFF12_PCDFRANGEPR_AUTOSTART );
612cdf0e10cSrcweir     maFieldGroupModel.mbAutoEnd      = getFlag( nFlags, BIFF12_PCDFRANGEPR_AUTOEND );
613cdf0e10cSrcweir 
614cdf0e10cSrcweir     OSL_ENSURE( maFieldGroupModel.mbDateGroup == (maFieldGroupModel.mnGroupBy != XML_range), "PivotCacheField::importPCDFRangePr - wrong date flag" );
615cdf0e10cSrcweir     if( maFieldGroupModel.mbDateGroup )
616cdf0e10cSrcweir     {
617cdf0e10cSrcweir         maFieldGroupModel.maStartDate = getUnitConverter().calcDateTimeFromSerial( maFieldGroupModel.mfStartValue );
618cdf0e10cSrcweir         maFieldGroupModel.maEndDate   = getUnitConverter().calcDateTimeFromSerial( maFieldGroupModel.mfEndValue );
619cdf0e10cSrcweir     }
620cdf0e10cSrcweir }
621cdf0e10cSrcweir 
importPCDFDiscretePrItem(sal_Int32 nRecId,SequenceInputStream & rStrm)622cdf0e10cSrcweir void PivotCacheField::importPCDFDiscretePrItem( sal_Int32 nRecId, SequenceInputStream& rStrm )
623cdf0e10cSrcweir {
624cdf0e10cSrcweir     OSL_ENSURE( nRecId == BIFF12_ID_PCITEM_INDEX, "PivotCacheField::importPCDFDiscretePrItem - unexpected record" );
625cdf0e10cSrcweir     if( nRecId == BIFF12_ID_PCITEM_INDEX )
626cdf0e10cSrcweir         maDiscreteItems.push_back( rStrm.readInt32() );
627cdf0e10cSrcweir }
628cdf0e10cSrcweir 
importPCDFGroupItem(sal_Int32 nRecId,SequenceInputStream & rStrm)629cdf0e10cSrcweir void PivotCacheField::importPCDFGroupItem( sal_Int32 nRecId, SequenceInputStream& rStrm )
630cdf0e10cSrcweir {
631cdf0e10cSrcweir     maGroupItems.importItem( nRecId, rStrm );
632cdf0e10cSrcweir }
633cdf0e10cSrcweir 
importPCDField(BiffInputStream & rStrm)634cdf0e10cSrcweir void PivotCacheField::importPCDField( BiffInputStream& rStrm )
635cdf0e10cSrcweir {
636cdf0e10cSrcweir     sal_uInt16 nFlags, nGroupItems, nBaseItems, nSharedItems;
637cdf0e10cSrcweir     rStrm >> nFlags;
638cdf0e10cSrcweir     maFieldGroupModel.mnParentField  = rStrm.readuInt16();
639cdf0e10cSrcweir     maFieldGroupModel.mnBaseField    = rStrm.readuInt16();
640cdf0e10cSrcweir     rStrm.skip( 2 );    // number of unique items (either shared or group)
641cdf0e10cSrcweir     rStrm >> nGroupItems >> nBaseItems >> nSharedItems;
642cdf0e10cSrcweir     maFieldModel.maName = (getBiff() == BIFF8) ? rStrm.readUniString() : rStrm.readByteStringUC( true, getTextEncoding() );
643cdf0e10cSrcweir 
644cdf0e10cSrcweir     maFieldModel.mbServerField          = getFlag( nFlags, BIFF_PCDFIELD_SERVERFIELD );
645cdf0e10cSrcweir     maFieldModel.mbUniqueList           = !getFlag( nFlags, BIFF_PCDFIELD_NOUNIQUEITEMS );
646cdf0e10cSrcweir     maSharedItemsModel.mbHasSemiMixed   = getFlag( nFlags, BIFF_PCDFIELD_HASSEMIMIXED );
647cdf0e10cSrcweir     maSharedItemsModel.mbHasNonDate     = getFlag( nFlags, BIFF_PCDFIELD_HASNONDATE );
648cdf0e10cSrcweir     maSharedItemsModel.mbHasDate        = getFlag( nFlags, BIFF_PCDFIELD_HASDATE );
649cdf0e10cSrcweir     maSharedItemsModel.mbIsNumeric      = getFlag( nFlags, BIFF_PCDFIELD_ISNUMERIC );
650cdf0e10cSrcweir     maSharedItemsModel.mbHasLongIndexes = getFlag( nFlags, BIFF_PCDFIELD_HASLONGINDEX );
651cdf0e10cSrcweir     maFieldGroupModel.mbRangeGroup      = getFlag( nFlags, BIFF_PCDFIELD_RANGEGROUP );
652cdf0e10cSrcweir 
653cdf0e10cSrcweir     // in BIFF, presence of parent group field is denoted by a flag
654cdf0e10cSrcweir     if( !getFlag( nFlags, BIFF_PCDFIELD_HASPARENT ) )
655cdf0e10cSrcweir         maFieldGroupModel.mnParentField = -1;
656cdf0e10cSrcweir 
657cdf0e10cSrcweir     // following PCDFSQLTYPE record contains SQL type
658cdf0e10cSrcweir     if( (rStrm.getNextRecId() == BIFF_ID_PCDFSQLTYPE) && rStrm.startNextRecord() )
659cdf0e10cSrcweir         maFieldModel.mnSqlType = rStrm.readInt16();
660cdf0e10cSrcweir 
661cdf0e10cSrcweir     // read group items, if any
662cdf0e10cSrcweir     if( nGroupItems > 0 )
663cdf0e10cSrcweir     {
664cdf0e10cSrcweir         OSL_ENSURE( getFlag( nFlags, BIFF_PCDFIELD_HASITEMS ), "PivotCacheField::importPCDField - missing items flag" );
665cdf0e10cSrcweir         maGroupItems.importItemList( rStrm, nGroupItems );
666cdf0e10cSrcweir 
667cdf0e10cSrcweir         sal_uInt16 nNextRecId = rStrm.getNextRecId();
668cdf0e10cSrcweir         bool bHasRangePr = nNextRecId == BIFF_ID_PCDFRANGEPR;
669cdf0e10cSrcweir         bool bHasDiscretePr = nNextRecId == BIFF_ID_PCDFDISCRETEPR;
670cdf0e10cSrcweir 
671cdf0e10cSrcweir         OSL_ENSURE( bHasRangePr || bHasDiscretePr, "PivotCacheField::importPCDField - missing group properties record" );
672cdf0e10cSrcweir         OSL_ENSURE( bHasRangePr == maFieldGroupModel.mbRangeGroup, "PivotCacheField::importPCDField - invalid range grouping flag" );
673cdf0e10cSrcweir         if( bHasRangePr && rStrm.startNextRecord() )
674cdf0e10cSrcweir             importPCDFRangePr( rStrm );
675cdf0e10cSrcweir         else if( bHasDiscretePr && rStrm.startNextRecord() )
676cdf0e10cSrcweir             importPCDFDiscretePr( rStrm );
677cdf0e10cSrcweir     }
678cdf0e10cSrcweir 
679cdf0e10cSrcweir     // read the shared items, if any
680cdf0e10cSrcweir     if( nSharedItems > 0 )
681cdf0e10cSrcweir     {
682cdf0e10cSrcweir         OSL_ENSURE( getFlag( nFlags, BIFF_PCDFIELD_HASITEMS ), "PivotCacheField::importPCDField - missing items flag" );
683cdf0e10cSrcweir         maSharedItems.importItemList( rStrm, nSharedItems );
684cdf0e10cSrcweir     }
685cdf0e10cSrcweir }
686cdf0e10cSrcweir 
importPCDFRangePr(BiffInputStream & rStrm)687cdf0e10cSrcweir void PivotCacheField::importPCDFRangePr( BiffInputStream& rStrm )
688cdf0e10cSrcweir {
689cdf0e10cSrcweir     sal_uInt16 nFlags;
690cdf0e10cSrcweir     rStrm >> nFlags;
691cdf0e10cSrcweir     maFieldGroupModel.setBiffGroupBy( extractValue< sal_uInt8 >( nFlags, 2, 3 ) );
692cdf0e10cSrcweir     maFieldGroupModel.mbRangeGroup = true;
693cdf0e10cSrcweir     maFieldGroupModel.mbDateGroup  = maFieldGroupModel.mnGroupBy != XML_range;
694cdf0e10cSrcweir     maFieldGroupModel.mbAutoStart  = getFlag( nFlags, BIFF_PCDFRANGEPR_AUTOSTART );
695cdf0e10cSrcweir     maFieldGroupModel.mbAutoEnd    = getFlag( nFlags, BIFF_PCDFRANGEPR_AUTOEND );
696cdf0e10cSrcweir 
697cdf0e10cSrcweir     /*  Start, end, and interval are stored in 3 separate item records. Type of
698cdf0e10cSrcweir         the items is dependent on numeric/date mode. Numeric groups expect
699cdf0e10cSrcweir         three PCITEM_DOUBLE records, date groups expect two PCITEM_DATE records
700cdf0e10cSrcweir         and one PCITEM_INT record. */
701cdf0e10cSrcweir     PivotCacheItemList aLimits( *this );
702cdf0e10cSrcweir     aLimits.importItemList( rStrm, 3 );
703cdf0e10cSrcweir     OSL_ENSURE( aLimits.size() == 3, "PivotCacheField::importPCDFRangePr - missing grouping records" );
704cdf0e10cSrcweir     const PivotCacheItem* pStartValue = aLimits.getCacheItem( 0 );
705cdf0e10cSrcweir     const PivotCacheItem* pEndValue = aLimits.getCacheItem( 1 );
706cdf0e10cSrcweir     const PivotCacheItem* pInterval = aLimits.getCacheItem( 2 );
707cdf0e10cSrcweir     if( pStartValue && pEndValue && pInterval )
708cdf0e10cSrcweir     {
709cdf0e10cSrcweir         if( maFieldGroupModel.mbDateGroup )
710cdf0e10cSrcweir         {
711cdf0e10cSrcweir             bool bHasTypes = (pStartValue->getType() == XML_d) && (pEndValue->getType() == XML_d) && (pInterval->getType() == XML_i);
712cdf0e10cSrcweir             OSL_ENSURE( bHasTypes, "PivotCacheField::importPCDFRangePr - wrong data types in grouping items" );
713cdf0e10cSrcweir             if( bHasTypes )
714cdf0e10cSrcweir             {
715cdf0e10cSrcweir                 maFieldGroupModel.maStartDate = pStartValue->getValue().get< DateTime >();
716cdf0e10cSrcweir                 maFieldGroupModel.maEndDate   = pEndValue->getValue().get< DateTime >();
717cdf0e10cSrcweir                 maFieldGroupModel.mfInterval  = pInterval->getValue().get< sal_Int16 >();
718cdf0e10cSrcweir             }
719cdf0e10cSrcweir         }
720cdf0e10cSrcweir         else
721cdf0e10cSrcweir         {
722cdf0e10cSrcweir             bool bHasTypes = (pStartValue->getType() == XML_n) && (pEndValue->getType() == XML_n) && (pInterval->getType() == XML_n);
723cdf0e10cSrcweir             OSL_ENSURE( bHasTypes, "PivotCacheField::importPCDFRangePr - wrong data types in grouping items" );
724cdf0e10cSrcweir             if( bHasTypes )
725cdf0e10cSrcweir             {
726cdf0e10cSrcweir                 maFieldGroupModel.mfStartValue = pStartValue->getValue().get< double >();
727cdf0e10cSrcweir                 maFieldGroupModel.mfEndValue   = pEndValue->getValue().get< double >();
728cdf0e10cSrcweir                 maFieldGroupModel.mfInterval   = pInterval->getValue().get< double >();
729cdf0e10cSrcweir             }
730cdf0e10cSrcweir         }
731cdf0e10cSrcweir     }
732cdf0e10cSrcweir }
733cdf0e10cSrcweir 
importPCDFDiscretePr(BiffInputStream & rStrm)734cdf0e10cSrcweir void PivotCacheField::importPCDFDiscretePr( BiffInputStream& rStrm )
735cdf0e10cSrcweir {
736cdf0e10cSrcweir     sal_Int32 nCount = static_cast< sal_Int32 >( rStrm.size() / 2 );
737cdf0e10cSrcweir     for( sal_Int32 nIndex = 0; !rStrm.isEof() && (nIndex < nCount); ++nIndex )
738cdf0e10cSrcweir         maDiscreteItems.push_back( rStrm.readuInt16() );
739cdf0e10cSrcweir }
740cdf0e10cSrcweir 
getCacheItem(sal_Int32 nItemIdx) const741cdf0e10cSrcweir const PivotCacheItem* PivotCacheField::getCacheItem( sal_Int32 nItemIdx ) const
742cdf0e10cSrcweir {
743cdf0e10cSrcweir     if( hasGroupItems() )
744cdf0e10cSrcweir         return maGroupItems.getCacheItem( nItemIdx );
745cdf0e10cSrcweir     if( hasSharedItems() )
746cdf0e10cSrcweir         return maSharedItems.getCacheItem( nItemIdx );
747cdf0e10cSrcweir     return 0;
748cdf0e10cSrcweir }
749cdf0e10cSrcweir 
getCacheItemNames(::std::vector<OUString> & orItemNames) const750cdf0e10cSrcweir void PivotCacheField::getCacheItemNames( ::std::vector< OUString >& orItemNames ) const
751cdf0e10cSrcweir {
752cdf0e10cSrcweir     if( hasGroupItems() )
753cdf0e10cSrcweir         maGroupItems.getCacheItemNames( orItemNames );
754cdf0e10cSrcweir     else if( hasSharedItems() )
755cdf0e10cSrcweir         maSharedItems.getCacheItemNames( orItemNames );
756cdf0e10cSrcweir }
757cdf0e10cSrcweir 
convertNumericGrouping(const Reference<XDataPilotField> & rxDPField) const758cdf0e10cSrcweir void PivotCacheField::convertNumericGrouping( const Reference< XDataPilotField >& rxDPField ) const
759cdf0e10cSrcweir {
760cdf0e10cSrcweir     OSL_ENSURE( hasGroupItems() && hasNumericGrouping(), "PivotCacheField::convertNumericGrouping - not a numeric group field" );
761cdf0e10cSrcweir     PropertySet aPropSet( rxDPField );
762cdf0e10cSrcweir     if( hasGroupItems() && hasNumericGrouping() && aPropSet.is() )
763cdf0e10cSrcweir     {
764cdf0e10cSrcweir         DataPilotFieldGroupInfo aGroupInfo;
765cdf0e10cSrcweir         aGroupInfo.HasAutoStart  = maFieldGroupModel.mbAutoStart;
766cdf0e10cSrcweir         aGroupInfo.HasAutoEnd    = maFieldGroupModel.mbAutoEnd;
767cdf0e10cSrcweir         aGroupInfo.HasDateValues = sal_False;
768cdf0e10cSrcweir         aGroupInfo.Start         = maFieldGroupModel.mfStartValue;
769cdf0e10cSrcweir         aGroupInfo.End           = maFieldGroupModel.mfEndValue;
770cdf0e10cSrcweir         aGroupInfo.Step          = maFieldGroupModel.mfInterval;
771cdf0e10cSrcweir         aGroupInfo.GroupBy       = 0;
772cdf0e10cSrcweir         aPropSet.setProperty( PROP_GroupInfo, aGroupInfo );
773cdf0e10cSrcweir     }
774cdf0e10cSrcweir }
775cdf0e10cSrcweir 
createDateGroupField(const Reference<XDataPilotField> & rxBaseDPField) const776cdf0e10cSrcweir OUString PivotCacheField::createDateGroupField( const Reference< XDataPilotField >& rxBaseDPField ) const
777cdf0e10cSrcweir {
778cdf0e10cSrcweir     OSL_ENSURE( hasGroupItems() && hasDateGrouping(), "PivotCacheField::createDateGroupField - not a numeric group field" );
779cdf0e10cSrcweir     Reference< XDataPilotField > xDPGroupField;
780cdf0e10cSrcweir     PropertySet aPropSet( rxBaseDPField );
781cdf0e10cSrcweir     if( hasGroupItems() && hasDateGrouping() && aPropSet.is() )
782cdf0e10cSrcweir     {
783cdf0e10cSrcweir         bool bDayRanges = (maFieldGroupModel.mnGroupBy == XML_days) && (maFieldGroupModel.mfInterval >= 2.0);
784cdf0e10cSrcweir 
785cdf0e10cSrcweir         DataPilotFieldGroupInfo aGroupInfo;
786cdf0e10cSrcweir         aGroupInfo.HasAutoStart  = maFieldGroupModel.mbAutoStart;
787cdf0e10cSrcweir         aGroupInfo.HasAutoEnd    = maFieldGroupModel.mbAutoEnd;
788cdf0e10cSrcweir         aGroupInfo.HasDateValues = sal_True;
789cdf0e10cSrcweir         aGroupInfo.Start         = getUnitConverter().calcSerialFromDateTime( maFieldGroupModel.maStartDate );
790cdf0e10cSrcweir         aGroupInfo.End           = getUnitConverter().calcSerialFromDateTime( maFieldGroupModel.maEndDate );
791cdf0e10cSrcweir         aGroupInfo.Step          = bDayRanges ? maFieldGroupModel.mfInterval : 0.0;
792cdf0e10cSrcweir 
793cdf0e10cSrcweir         using namespace ::com::sun::star::sheet::DataPilotFieldGroupBy;
794cdf0e10cSrcweir         switch( maFieldGroupModel.mnGroupBy )
795cdf0e10cSrcweir         {
796cdf0e10cSrcweir             case XML_years:     aGroupInfo.GroupBy = YEARS;     break;
797cdf0e10cSrcweir             case XML_quarters:  aGroupInfo.GroupBy = QUARTERS;  break;
798cdf0e10cSrcweir             case XML_months:    aGroupInfo.GroupBy = MONTHS;    break;
799cdf0e10cSrcweir             case XML_days:      aGroupInfo.GroupBy = DAYS;      break;
800cdf0e10cSrcweir             case XML_hours:     aGroupInfo.GroupBy = HOURS;     break;
801cdf0e10cSrcweir             case XML_minutes:   aGroupInfo.GroupBy = MINUTES;   break;
802cdf0e10cSrcweir             case XML_seconds:   aGroupInfo.GroupBy = SECONDS;   break;
803cdf0e10cSrcweir             default:    OSL_ENSURE( false, "PivotCacheField::convertRangeGrouping - unknown date/time interval" );
804cdf0e10cSrcweir         }
805cdf0e10cSrcweir 
806cdf0e10cSrcweir         try
807cdf0e10cSrcweir         {
808cdf0e10cSrcweir             Reference< XDataPilotFieldGrouping > xDPGrouping( rxBaseDPField, UNO_QUERY_THROW );
809cdf0e10cSrcweir             xDPGroupField = xDPGrouping->createDateGroup( aGroupInfo );
810cdf0e10cSrcweir         }
811cdf0e10cSrcweir         catch( Exception& )
812cdf0e10cSrcweir         {
813cdf0e10cSrcweir         }
814cdf0e10cSrcweir     }
815cdf0e10cSrcweir 
816cdf0e10cSrcweir     Reference< XNamed > xFieldName( xDPGroupField, UNO_QUERY );
817cdf0e10cSrcweir     return xFieldName.is() ? xFieldName->getName() : OUString();
818cdf0e10cSrcweir }
819cdf0e10cSrcweir 
createParentGroupField(const Reference<XDataPilotField> & rxBaseDPField,PivotCacheGroupItemVector & orItemNames) const820cdf0e10cSrcweir OUString PivotCacheField::createParentGroupField( const Reference< XDataPilotField >& rxBaseDPField, PivotCacheGroupItemVector& orItemNames ) const
821cdf0e10cSrcweir {
822cdf0e10cSrcweir     OSL_ENSURE( hasGroupItems() && !maDiscreteItems.empty(), "PivotCacheField::createParentGroupField - not a group field" );
823cdf0e10cSrcweir     OSL_ENSURE( maDiscreteItems.size() == orItemNames.size(), "PivotCacheField::createParentGroupField - number of item names does not match grouping info" );
824cdf0e10cSrcweir     Reference< XDataPilotFieldGrouping > xDPGrouping( rxBaseDPField, UNO_QUERY );
825cdf0e10cSrcweir     if( !xDPGrouping.is() ) return OUString();
826cdf0e10cSrcweir 
827cdf0e10cSrcweir     // map the group item indexes from maGroupItems to all item indexes from maDiscreteItems
828cdf0e10cSrcweir     typedef ::std::vector< sal_Int32 > GroupItemList;
829cdf0e10cSrcweir     typedef ::std::vector< GroupItemList > GroupItemMap;
830cdf0e10cSrcweir     GroupItemMap aItemMap( maGroupItems.size() );
831cdf0e10cSrcweir     for( IndexVector::const_iterator aBeg = maDiscreteItems.begin(), aIt = aBeg, aEnd = maDiscreteItems.end(); aIt != aEnd; ++aIt )
832cdf0e10cSrcweir         if( GroupItemList* pItems = ContainerHelper::getVectorElementAccess( aItemMap, *aIt ) )
833cdf0e10cSrcweir             pItems->push_back( static_cast< sal_Int32 >( aIt - aBeg ) );
834cdf0e10cSrcweir 
835cdf0e10cSrcweir     // process all groups
836cdf0e10cSrcweir     Reference< XDataPilotField > xDPGroupField;
837cdf0e10cSrcweir     for( GroupItemMap::iterator aBeg = aItemMap.begin(), aIt = aBeg, aEnd = aItemMap.end(); aIt != aEnd; ++aIt )
838cdf0e10cSrcweir     {
839cdf0e10cSrcweir         OSL_ENSURE( !aIt->empty(), "PivotCacheField::createParentGroupField - item/group should not be empty" );
840cdf0e10cSrcweir         // if the item count is greater than 1, the item is a group of items
841cdf0e10cSrcweir         if( aIt->size() > 1 )
842cdf0e10cSrcweir         {
843cdf0e10cSrcweir             /*  Insert the names of the items that are part of this group. Calc
844cdf0e10cSrcweir                 expects the names of the members of the field whose members are
845cdf0e10cSrcweir                 grouped (which may be the names of groups too). Excel provides
846cdf0e10cSrcweir                 the names of the base field items instead (no group names
847cdf0e10cSrcweir                 involved). Therefore, the passed collection of current item
848cdf0e10cSrcweir                 names as they are already grouped is used here to resolve the
849cdf0e10cSrcweir                 item names. */
850cdf0e10cSrcweir             ::std::vector< OUString > aMembers;
851cdf0e10cSrcweir             for( GroupItemList::iterator aBeg2 = aIt->begin(), aIt2 = aBeg2, aEnd2 = aIt->end(); aIt2 != aEnd2; ++aIt2 )
852cdf0e10cSrcweir                 if( const PivotCacheGroupItem* pName = ContainerHelper::getVectorElement( orItemNames, *aIt2 ) )
853cdf0e10cSrcweir                     if( ::std::find( aMembers.begin(), aMembers.end(), pName->maGroupName ) == aMembers.end() )
854cdf0e10cSrcweir                         aMembers.push_back( pName->maGroupName );
855cdf0e10cSrcweir 
856cdf0e10cSrcweir             /*  Check again, that this is not just a group that is not grouped
857cdf0e10cSrcweir                 further with other items. */
858cdf0e10cSrcweir             if( aMembers.size() > 1 ) try
859cdf0e10cSrcweir             {
860cdf0e10cSrcweir                 // only the first call of createNameGroup() returns the new field
861cdf0e10cSrcweir                 Reference< XDataPilotField > xDPNewField = xDPGrouping->createNameGroup( ContainerHelper::vectorToSequence( aMembers ) );
862cdf0e10cSrcweir                 OSL_ENSURE( xDPGroupField.is() != xDPNewField.is(), "PivotCacheField::createParentGroupField - missing group field" );
863cdf0e10cSrcweir                 if( !xDPGroupField.is() )
864cdf0e10cSrcweir                     xDPGroupField = xDPNewField;
865cdf0e10cSrcweir 
866cdf0e10cSrcweir                 // get current grouping info
867cdf0e10cSrcweir                 DataPilotFieldGroupInfo aGroupInfo;
868cdf0e10cSrcweir                 PropertySet aPropSet( xDPGroupField );
869cdf0e10cSrcweir                 aPropSet.getProperty( aGroupInfo, PROP_GroupInfo );
870cdf0e10cSrcweir 
871cdf0e10cSrcweir                 /*  Find the group object and the auto-generated group name.
872cdf0e10cSrcweir                     The returned field contains all groups derived from the
873cdf0e10cSrcweir                     previous field if that is grouped too. To find the correct
874cdf0e10cSrcweir                     group, the first item used to create the group is serached.
875cdf0e10cSrcweir                     Calc provides the original item names of the base field
876cdf0e10cSrcweir                     when the group is querried for its members. Its does not
877cdf0e10cSrcweir                     provide the names of members that are already groups in the
878cdf0e10cSrcweir                     field used to create the new groups. (Is this a bug?)
879cdf0e10cSrcweir                     Therefore, a name from the passed list of original item
880cdf0e10cSrcweir                     names is used to find the correct group. */
881cdf0e10cSrcweir                 OUString aFirstItem;
882cdf0e10cSrcweir                 if( const PivotCacheGroupItem* pName = ContainerHelper::getVectorElement( orItemNames, aIt->front() ) )
883cdf0e10cSrcweir                     aFirstItem = pName->maOrigName;
884cdf0e10cSrcweir                 Reference< XNamed > xGroupName;
885cdf0e10cSrcweir                 OUString aAutoName;
886cdf0e10cSrcweir                 Reference< XIndexAccess > xGroupsIA( aGroupInfo.Groups, UNO_QUERY_THROW );
887cdf0e10cSrcweir                 for( sal_Int32 nIdx = 0, nCount = xGroupsIA->getCount(); (nIdx < nCount) && (aAutoName.getLength() == 0); ++nIdx ) try
888cdf0e10cSrcweir                 {
889cdf0e10cSrcweir                     Reference< XNameAccess > xItemsNA( xGroupsIA->getByIndex( nIdx ), UNO_QUERY_THROW );
890cdf0e10cSrcweir                     if( xItemsNA->hasByName( aFirstItem ) )
891cdf0e10cSrcweir                     {
892cdf0e10cSrcweir                         xGroupName.set( xGroupsIA->getByIndex( nIdx ), UNO_QUERY_THROW );
893cdf0e10cSrcweir                         aAutoName = xGroupName->getName();
894cdf0e10cSrcweir                     }
895cdf0e10cSrcweir                 }
896cdf0e10cSrcweir                 catch( Exception& )
897cdf0e10cSrcweir                 {
898cdf0e10cSrcweir                 }
899cdf0e10cSrcweir                 OSL_ENSURE( aAutoName.getLength() > 0, "PivotCacheField::createParentGroupField - cannot find auto-generated group name" );
900cdf0e10cSrcweir 
901cdf0e10cSrcweir                 // get the real group name from the list of group items
902cdf0e10cSrcweir                 OUString aGroupName;
903cdf0e10cSrcweir                 if( const PivotCacheItem* pGroupItem = maGroupItems.getCacheItem( static_cast< sal_Int32 >( aIt - aBeg ) ) )
904cdf0e10cSrcweir                     aGroupName = pGroupItem->getName();
905cdf0e10cSrcweir                 OSL_ENSURE( aGroupName.getLength() > 0, "PivotCacheField::createParentGroupField - cannot find group name" );
906cdf0e10cSrcweir                 if( aGroupName.getLength() == 0 )
907cdf0e10cSrcweir                     aGroupName = aAutoName;
908cdf0e10cSrcweir 
909cdf0e10cSrcweir                 if( xGroupName.is() && (aGroupName.getLength() > 0) )
910cdf0e10cSrcweir                 {
911cdf0e10cSrcweir                     // replace the auto-generated group name with the real name
912cdf0e10cSrcweir                     if( aAutoName != aGroupName )
913cdf0e10cSrcweir                     {
914cdf0e10cSrcweir                         xGroupName->setName( aGroupName );
915cdf0e10cSrcweir                         aPropSet.setProperty( PROP_GroupInfo, aGroupInfo );
916cdf0e10cSrcweir                     }
917cdf0e10cSrcweir                     // replace original item names in passed vector with group name
918cdf0e10cSrcweir                     for( GroupItemList::iterator aIt2 = aIt->begin(), aEnd2 = aIt->end(); aIt2 != aEnd2; ++aIt2 )
919cdf0e10cSrcweir                         if( PivotCacheGroupItem* pName = ContainerHelper::getVectorElementAccess( orItemNames, *aIt2 ) )
920cdf0e10cSrcweir                             pName->maGroupName = aGroupName;
921cdf0e10cSrcweir                 }
922cdf0e10cSrcweir             }
923cdf0e10cSrcweir             catch( Exception& )
924cdf0e10cSrcweir             {
925cdf0e10cSrcweir             }
926cdf0e10cSrcweir         }
927cdf0e10cSrcweir     }
928cdf0e10cSrcweir 
929cdf0e10cSrcweir     Reference< XNamed > xFieldName( xDPGroupField, UNO_QUERY );
930cdf0e10cSrcweir     return xFieldName.is() ? xFieldName->getName() : OUString();
931cdf0e10cSrcweir }
932cdf0e10cSrcweir 
writeSourceHeaderCell(WorksheetHelper & rSheetHelper,sal_Int32 nCol,sal_Int32 nRow) const933cdf0e10cSrcweir void PivotCacheField::writeSourceHeaderCell( WorksheetHelper& rSheetHelper, sal_Int32 nCol, sal_Int32 nRow ) const
934cdf0e10cSrcweir {
935cdf0e10cSrcweir     CellModel aModel;
936cdf0e10cSrcweir     aModel.maCellAddr = CellAddress( rSheetHelper.getSheetIndex(), nCol, nRow );
937cdf0e10cSrcweir     rSheetHelper.getSheetData().setStringCell( aModel, maFieldModel.maName );
938cdf0e10cSrcweir }
939cdf0e10cSrcweir 
writeSourceDataCell(WorksheetHelper & rSheetHelper,sal_Int32 nCol,sal_Int32 nRow,const PivotCacheItem & rItem) const940cdf0e10cSrcweir void PivotCacheField::writeSourceDataCell( WorksheetHelper& rSheetHelper, sal_Int32 nCol, sal_Int32 nRow, const PivotCacheItem& rItem ) const
941cdf0e10cSrcweir {
942cdf0e10cSrcweir     bool bHasIndex = rItem.getType() == XML_x;
943cdf0e10cSrcweir     OSL_ENSURE( bHasIndex != maSharedItems.empty(), "PivotCacheField::writeSourceDataCell - shared items missing or not expected" );
944cdf0e10cSrcweir     if( bHasIndex )
945cdf0e10cSrcweir         writeSharedItemToSourceDataCell( rSheetHelper, nCol, nRow, rItem.getValue().get< sal_Int32 >() );
946cdf0e10cSrcweir     else
947cdf0e10cSrcweir         writeItemToSourceDataCell( rSheetHelper, nCol, nRow, rItem );
948cdf0e10cSrcweir }
949cdf0e10cSrcweir 
importPCRecordItem(SequenceInputStream & rStrm,WorksheetHelper & rSheetHelper,sal_Int32 nCol,sal_Int32 nRow) const950cdf0e10cSrcweir void PivotCacheField::importPCRecordItem( SequenceInputStream& rStrm, WorksheetHelper& rSheetHelper, sal_Int32 nCol, sal_Int32 nRow ) const
951cdf0e10cSrcweir {
952cdf0e10cSrcweir     if( hasSharedItems() )
953cdf0e10cSrcweir     {
954cdf0e10cSrcweir         writeSharedItemToSourceDataCell( rSheetHelper, nCol, nRow, rStrm.readInt32() );
955cdf0e10cSrcweir     }
956cdf0e10cSrcweir     else
957cdf0e10cSrcweir     {
958cdf0e10cSrcweir         PivotCacheItem aItem;
959cdf0e10cSrcweir         if( maSharedItemsModel.mbIsNumeric )
960cdf0e10cSrcweir            aItem.readDouble( rStrm );
961cdf0e10cSrcweir         else if( maSharedItemsModel.mbHasDate && !maSharedItemsModel.mbHasString )
962cdf0e10cSrcweir            aItem.readDate( rStrm );
963cdf0e10cSrcweir         else
964cdf0e10cSrcweir            aItem.readString( rStrm );
965cdf0e10cSrcweir         writeItemToSourceDataCell( rSheetHelper, nCol, nRow, aItem );
966cdf0e10cSrcweir     }
967cdf0e10cSrcweir }
968cdf0e10cSrcweir 
importPCItemIndex(BiffInputStream & rStrm,WorksheetHelper & rSheetHelper,sal_Int32 nCol,sal_Int32 nRow) const969cdf0e10cSrcweir void PivotCacheField::importPCItemIndex( BiffInputStream& rStrm, WorksheetHelper& rSheetHelper, sal_Int32 nCol, sal_Int32 nRow ) const
970cdf0e10cSrcweir {
971cdf0e10cSrcweir     OSL_ENSURE( hasSharedItems(), "PivotCacheField::importPCItemIndex - invalid call, no shared items found" );
972cdf0e10cSrcweir     sal_Int32 nIndex = maSharedItemsModel.mbHasLongIndexes ? rStrm.readuInt16() : rStrm.readuInt8();
973cdf0e10cSrcweir     writeSharedItemToSourceDataCell( rSheetHelper, nCol, nRow, nIndex );
974cdf0e10cSrcweir }
975cdf0e10cSrcweir 
976cdf0e10cSrcweir // private --------------------------------------------------------------------
977cdf0e10cSrcweir 
writeItemToSourceDataCell(WorksheetHelper & rSheetHelper,sal_Int32 nCol,sal_Int32 nRow,const PivotCacheItem & rItem) const978cdf0e10cSrcweir void PivotCacheField::writeItemToSourceDataCell( WorksheetHelper& rSheetHelper,
979cdf0e10cSrcweir         sal_Int32 nCol, sal_Int32 nRow, const PivotCacheItem& rItem ) const
980cdf0e10cSrcweir {
981cdf0e10cSrcweir     if( rItem.getType() != XML_m )
982cdf0e10cSrcweir     {
983cdf0e10cSrcweir         CellModel aModel;
984cdf0e10cSrcweir         aModel.maCellAddr = CellAddress( rSheetHelper.getSheetIndex(), nCol, nRow );
985cdf0e10cSrcweir         SheetDataBuffer& rSheetData = rSheetHelper.getSheetData();
986cdf0e10cSrcweir         switch( rItem.getType() )
987cdf0e10cSrcweir         {
988cdf0e10cSrcweir             case XML_s: rSheetData.setStringCell( aModel, rItem.getValue().get< OUString >() );                             break;
989cdf0e10cSrcweir             case XML_n: rSheetData.setValueCell( aModel, rItem.getValue().get< double >() );                                break;
990cdf0e10cSrcweir             case XML_i: rSheetData.setValueCell( aModel, rItem.getValue().get< sal_Int16 >() );                             break;
991cdf0e10cSrcweir             case XML_d: rSheetData.setDateTimeCell( aModel, rItem.getValue().get< DateTime >() );                           break;
992cdf0e10cSrcweir             case XML_b: rSheetData.setBooleanCell( aModel, rItem.getValue().get< bool >() );                                break;
993cdf0e10cSrcweir             case XML_e: rSheetData.setErrorCell( aModel, static_cast< sal_uInt8 >( rItem.getValue().get< sal_Int32 >() ) ); break;
994cdf0e10cSrcweir             default:    OSL_ENSURE( false, "PivotCacheField::writeItemToSourceDataCell - unexpected item data type" );
995cdf0e10cSrcweir         }
996cdf0e10cSrcweir     }
997cdf0e10cSrcweir }
998cdf0e10cSrcweir 
writeSharedItemToSourceDataCell(WorksheetHelper & rSheetHelper,sal_Int32 nCol,sal_Int32 nRow,sal_Int32 nItemIdx) const999cdf0e10cSrcweir void PivotCacheField::writeSharedItemToSourceDataCell(
1000cdf0e10cSrcweir         WorksheetHelper& rSheetHelper, sal_Int32 nCol, sal_Int32 nRow, sal_Int32 nItemIdx ) const
1001cdf0e10cSrcweir {
1002cdf0e10cSrcweir     if( const PivotCacheItem* pCacheItem = maSharedItems.getCacheItem( nItemIdx ) )
1003cdf0e10cSrcweir         writeItemToSourceDataCell( rSheetHelper, nCol, nRow, *pCacheItem );
1004cdf0e10cSrcweir }
1005cdf0e10cSrcweir 
1006cdf0e10cSrcweir // ============================================================================
1007cdf0e10cSrcweir 
PCDefinitionModel()1008cdf0e10cSrcweir PCDefinitionModel::PCDefinitionModel() :
1009cdf0e10cSrcweir     mfRefreshedDate( 0.0 ),
1010cdf0e10cSrcweir     mnRecords( 0 ),
1011cdf0e10cSrcweir     mnMissItemsLimit( 0 ),
1012cdf0e10cSrcweir     mnDatabaseFields( 0 ),
1013cdf0e10cSrcweir     mbInvalid( false ),
1014cdf0e10cSrcweir     mbSaveData( true ),
1015cdf0e10cSrcweir     mbRefreshOnLoad( false ),
1016cdf0e10cSrcweir     mbOptimizeMemory( false ),
1017cdf0e10cSrcweir     mbEnableRefresh( true ),
1018cdf0e10cSrcweir     mbBackgroundQuery( false ),
1019cdf0e10cSrcweir     mbUpgradeOnRefresh( false ),
1020cdf0e10cSrcweir     mbTupleCache( false ),
1021cdf0e10cSrcweir     mbSupportSubquery( false ),
1022cdf0e10cSrcweir     mbSupportDrill( false )
1023cdf0e10cSrcweir {
1024cdf0e10cSrcweir }
1025cdf0e10cSrcweir 
1026cdf0e10cSrcweir // ----------------------------------------------------------------------------
1027cdf0e10cSrcweir 
PCSourceModel()1028cdf0e10cSrcweir PCSourceModel::PCSourceModel() :
1029cdf0e10cSrcweir     mnSourceType( XML_TOKEN_INVALID ),
1030cdf0e10cSrcweir     mnConnectionId( 0 )
1031cdf0e10cSrcweir {
1032cdf0e10cSrcweir }
1033cdf0e10cSrcweir 
1034cdf0e10cSrcweir // ----------------------------------------------------------------------------
1035cdf0e10cSrcweir 
PCWorksheetSourceModel()1036cdf0e10cSrcweir PCWorksheetSourceModel::PCWorksheetSourceModel()
1037cdf0e10cSrcweir {
1038cdf0e10cSrcweir     maRange.StartColumn = maRange.StartRow = maRange.EndColumn = maRange.EndRow = -1;
1039cdf0e10cSrcweir }
1040cdf0e10cSrcweir 
1041cdf0e10cSrcweir // ----------------------------------------------------------------------------
1042cdf0e10cSrcweir 
PivotCache(const WorkbookHelper & rHelper)1043cdf0e10cSrcweir PivotCache::PivotCache( const WorkbookHelper& rHelper ) :
1044cdf0e10cSrcweir     WorkbookHelper( rHelper ),
1045cdf0e10cSrcweir     mnCurrRow( -1 ),
1046cdf0e10cSrcweir     mbValidSource( false ),
1047cdf0e10cSrcweir     mbDummySheet( false )
1048cdf0e10cSrcweir {
1049cdf0e10cSrcweir }
1050cdf0e10cSrcweir 
importPivotCacheDefinition(const AttributeList & rAttribs)1051cdf0e10cSrcweir void PivotCache::importPivotCacheDefinition( const AttributeList& rAttribs )
1052cdf0e10cSrcweir {
1053cdf0e10cSrcweir     maDefModel.maRelId            = rAttribs.getString( R_TOKEN( id ), OUString() );
1054cdf0e10cSrcweir     maDefModel.maRefreshedBy      = rAttribs.getXString( XML_refreshedBy, OUString() );
1055cdf0e10cSrcweir     maDefModel.mfRefreshedDate    = rAttribs.getDouble( XML_refreshedDate, 0.0 );
1056cdf0e10cSrcweir     maDefModel.mnRecords          = rAttribs.getInteger( XML_recordCount, 0 );
1057cdf0e10cSrcweir     maDefModel.mnMissItemsLimit   = rAttribs.getInteger( XML_missingItemsLimit, 0 );
1058cdf0e10cSrcweir     maDefModel.mbInvalid          = rAttribs.getBool( XML_invalid, false );
1059cdf0e10cSrcweir     maDefModel.mbSaveData         = rAttribs.getBool( XML_saveData, true );
1060cdf0e10cSrcweir     maDefModel.mbRefreshOnLoad    = rAttribs.getBool( XML_refreshOnLoad, false );
1061cdf0e10cSrcweir     maDefModel.mbOptimizeMemory   = rAttribs.getBool( XML_optimizeMemory, false );
1062cdf0e10cSrcweir     maDefModel.mbEnableRefresh    = rAttribs.getBool( XML_enableRefresh, true );
1063cdf0e10cSrcweir     maDefModel.mbBackgroundQuery  = rAttribs.getBool( XML_backgroundQuery, false );
1064cdf0e10cSrcweir     maDefModel.mbUpgradeOnRefresh = rAttribs.getBool( XML_upgradeOnRefresh, false );
1065cdf0e10cSrcweir     maDefModel.mbTupleCache       = rAttribs.getBool( XML_tupleCache, false );
1066cdf0e10cSrcweir     maDefModel.mbSupportSubquery  = rAttribs.getBool( XML_supportSubquery, false );
1067cdf0e10cSrcweir     maDefModel.mbSupportDrill     = rAttribs.getBool( XML_supportAdvancedDrill, false );
1068cdf0e10cSrcweir }
1069cdf0e10cSrcweir 
importCacheSource(const AttributeList & rAttribs)1070cdf0e10cSrcweir void PivotCache::importCacheSource( const AttributeList& rAttribs )
1071cdf0e10cSrcweir {
1072cdf0e10cSrcweir     maSourceModel.mnSourceType   = rAttribs.getToken( XML_type, XML_TOKEN_INVALID );
1073cdf0e10cSrcweir     maSourceModel.mnConnectionId = rAttribs.getInteger( XML_connectionId, 0 );
1074cdf0e10cSrcweir }
1075cdf0e10cSrcweir 
importWorksheetSource(const AttributeList & rAttribs,const Relations & rRelations)1076cdf0e10cSrcweir void PivotCache::importWorksheetSource( const AttributeList& rAttribs, const Relations& rRelations )
1077cdf0e10cSrcweir {
1078cdf0e10cSrcweir     maSheetSrcModel.maRelId   = rAttribs.getString( R_TOKEN( id ), OUString() );
1079cdf0e10cSrcweir     maSheetSrcModel.maSheet   = rAttribs.getXString( XML_sheet, OUString() );
1080cdf0e10cSrcweir     maSheetSrcModel.maDefName = rAttribs.getXString( XML_name, OUString() );
1081cdf0e10cSrcweir 
1082cdf0e10cSrcweir     // resolve URL of external document
1083cdf0e10cSrcweir     maTargetUrl = rRelations.getExternalTargetFromRelId( maSheetSrcModel.maRelId );
1084cdf0e10cSrcweir     // store range address unchecked with sheet index 0, will be resolved/checked later
1085cdf0e10cSrcweir     getAddressConverter().convertToCellRangeUnchecked( maSheetSrcModel.maRange, rAttribs.getString( XML_ref, OUString() ), 0 );
1086cdf0e10cSrcweir }
1087cdf0e10cSrcweir 
importPCDefinition(SequenceInputStream & rStrm)1088cdf0e10cSrcweir void PivotCache::importPCDefinition( SequenceInputStream& rStrm )
1089cdf0e10cSrcweir {
1090cdf0e10cSrcweir     sal_uInt8 nFlags1, nFlags2;
1091cdf0e10cSrcweir     rStrm.skip( 3 );    // create/refresh version id's
1092cdf0e10cSrcweir     rStrm >> nFlags1 >> maDefModel.mnMissItemsLimit >> maDefModel.mfRefreshedDate >> nFlags2 >> maDefModel.mnRecords;
1093cdf0e10cSrcweir     if( getFlag( nFlags2, BIFF12_PCDEFINITION_HASUSERNAME ) )
1094cdf0e10cSrcweir         rStrm >> maDefModel.maRefreshedBy;
1095cdf0e10cSrcweir     if( getFlag( nFlags2, BIFF12_PCDEFINITION_HASRELID ) )
1096cdf0e10cSrcweir         rStrm >> maDefModel.maRelId;
1097cdf0e10cSrcweir 
1098cdf0e10cSrcweir     maDefModel.mbInvalid          = getFlag( nFlags1, BIFF12_PCDEFINITION_INVALID );
1099cdf0e10cSrcweir     maDefModel.mbSaveData         = getFlag( nFlags1, BIFF12_PCDEFINITION_SAVEDATA );
1100cdf0e10cSrcweir     maDefModel.mbRefreshOnLoad    = getFlag( nFlags1, BIFF12_PCDEFINITION_REFRESHONLOAD );
1101cdf0e10cSrcweir     maDefModel.mbOptimizeMemory   = getFlag( nFlags1, BIFF12_PCDEFINITION_OPTIMIZEMEMORY );
1102cdf0e10cSrcweir     maDefModel.mbEnableRefresh    = getFlag( nFlags1, BIFF12_PCDEFINITION_ENABLEREFRESH );
1103cdf0e10cSrcweir     maDefModel.mbBackgroundQuery  = getFlag( nFlags1, BIFF12_PCDEFINITION_BACKGROUNDQUERY );
1104cdf0e10cSrcweir     maDefModel.mbUpgradeOnRefresh = getFlag( nFlags1, BIFF12_PCDEFINITION_UPGRADEONREFR );
1105cdf0e10cSrcweir     maDefModel.mbTupleCache       = getFlag( nFlags1, BIFF12_PCDEFINITION_TUPELCACHE );
1106cdf0e10cSrcweir     maDefModel.mbSupportSubquery  = getFlag( nFlags2, BIFF12_PCDEFINITION_SUPPORTSUBQUERY );
1107cdf0e10cSrcweir     maDefModel.mbSupportDrill     = getFlag( nFlags2, BIFF12_PCDEFINITION_SUPPORTDRILL );
1108cdf0e10cSrcweir }
1109cdf0e10cSrcweir 
importPCDSource(SequenceInputStream & rStrm)1110cdf0e10cSrcweir void PivotCache::importPCDSource( SequenceInputStream& rStrm )
1111cdf0e10cSrcweir {
1112cdf0e10cSrcweir     sal_Int32 nSourceType;
1113cdf0e10cSrcweir     rStrm >> nSourceType >> maSourceModel.mnConnectionId;
1114cdf0e10cSrcweir     static const sal_Int32 spnSourceTypes[] = { XML_worksheet, XML_external, XML_consolidation, XML_scenario };
1115cdf0e10cSrcweir     maSourceModel.mnSourceType = STATIC_ARRAY_SELECT( spnSourceTypes, nSourceType, XML_TOKEN_INVALID );
1116cdf0e10cSrcweir }
1117cdf0e10cSrcweir 
importPCDSheetSource(SequenceInputStream & rStrm,const Relations & rRelations)1118cdf0e10cSrcweir void PivotCache::importPCDSheetSource( SequenceInputStream& rStrm, const Relations& rRelations )
1119cdf0e10cSrcweir {
1120cdf0e10cSrcweir     sal_uInt8 nIsDefName, nIsBuiltinName, nFlags;
1121cdf0e10cSrcweir     rStrm >> nIsDefName >> nIsBuiltinName >> nFlags;
1122cdf0e10cSrcweir     if( getFlag( nFlags, BIFF12_PCDWBSOURCE_HASSHEET ) )
1123cdf0e10cSrcweir         rStrm >> maSheetSrcModel.maSheet;
1124cdf0e10cSrcweir     if( getFlag( nFlags, BIFF12_PCDWBSOURCE_HASRELID ) )
1125cdf0e10cSrcweir         rStrm >> maSheetSrcModel.maRelId;
1126cdf0e10cSrcweir 
1127cdf0e10cSrcweir     // read cell range or defined name
1128cdf0e10cSrcweir     if( nIsDefName == 0 )
1129cdf0e10cSrcweir     {
1130cdf0e10cSrcweir         BinRange aBinRange;
1131cdf0e10cSrcweir         rStrm >> aBinRange;
1132cdf0e10cSrcweir         // store range address unchecked with sheet index 0, will be resolved/checked later
1133cdf0e10cSrcweir         getAddressConverter().convertToCellRangeUnchecked( maSheetSrcModel.maRange, aBinRange, 0 );
1134cdf0e10cSrcweir     }
1135cdf0e10cSrcweir     else
1136cdf0e10cSrcweir     {
1137cdf0e10cSrcweir         rStrm >> maSheetSrcModel.maDefName;
1138cdf0e10cSrcweir         if( nIsBuiltinName != 0 )
1139cdf0e10cSrcweir             maSheetSrcModel.maDefName = CREATE_OUSTRING( "_xlnm." ) + maSheetSrcModel.maDefName;
1140cdf0e10cSrcweir     }
1141cdf0e10cSrcweir 
1142cdf0e10cSrcweir     // resolve URL of external document
1143cdf0e10cSrcweir     maTargetUrl = rRelations.getExternalTargetFromRelId( maSheetSrcModel.maRelId );
1144cdf0e10cSrcweir }
1145cdf0e10cSrcweir 
importPCDSource(BiffInputStream & rStrm)1146cdf0e10cSrcweir void PivotCache::importPCDSource( BiffInputStream& rStrm )
1147cdf0e10cSrcweir {
1148cdf0e10cSrcweir     switch( rStrm.readuInt16() )
1149cdf0e10cSrcweir     {
1150cdf0e10cSrcweir         case BIFF_PCDSOURCE_WORKSHEET:
1151cdf0e10cSrcweir         {
1152cdf0e10cSrcweir             maSourceModel.mnSourceType = XML_worksheet;
1153cdf0e10cSrcweir             sal_uInt16 nNextRecId = rStrm.getNextRecId();
1154cdf0e10cSrcweir             switch( nNextRecId )
1155cdf0e10cSrcweir             {
1156cdf0e10cSrcweir                 case BIFF_ID_DCONREF:       if( rStrm.startNextRecord() ) importDConRef( rStrm );       break;
1157cdf0e10cSrcweir                 case BIFF_ID_DCONNAME:      if( rStrm.startNextRecord() ) importDConName( rStrm );      break;
1158cdf0e10cSrcweir                 case BIFF_ID_DCONBINAME:    if( rStrm.startNextRecord() ) importDConBIName( rStrm );    break;
1159cdf0e10cSrcweir             }
1160cdf0e10cSrcweir         }
1161cdf0e10cSrcweir         break;
1162cdf0e10cSrcweir         case BIFF_PCDSOURCE_EXTERNAL:
1163cdf0e10cSrcweir             maSourceModel.mnSourceType = XML_external;
1164cdf0e10cSrcweir         break;
1165cdf0e10cSrcweir         case BIFF_PCDSOURCE_CONSOLIDATION:
1166cdf0e10cSrcweir             maSourceModel.mnSourceType = XML_consolidation;
1167cdf0e10cSrcweir         break;
1168cdf0e10cSrcweir         case BIFF_PCDSOURCE_SCENARIO:
1169cdf0e10cSrcweir             maSourceModel.mnSourceType = XML_scenario;
1170cdf0e10cSrcweir         break;
1171cdf0e10cSrcweir         default:
1172cdf0e10cSrcweir             maSourceModel.mnSourceType = XML_TOKEN_INVALID;
1173cdf0e10cSrcweir     }
1174cdf0e10cSrcweir }
1175cdf0e10cSrcweir 
importPCDefinition(BiffInputStream & rStrm)1176cdf0e10cSrcweir void PivotCache::importPCDefinition( BiffInputStream& rStrm )
1177cdf0e10cSrcweir {
1178cdf0e10cSrcweir     sal_uInt16 nFlags, nUserNameLen;
1179cdf0e10cSrcweir     rStrm >> maDefModel.mnRecords;
1180cdf0e10cSrcweir     rStrm.skip( 2 );    // repeated cache ID
1181cdf0e10cSrcweir     rStrm >> nFlags;
1182cdf0e10cSrcweir     rStrm.skip( 2 );    // unused
1183cdf0e10cSrcweir     rStrm >> maDefModel.mnDatabaseFields;
1184cdf0e10cSrcweir     rStrm.skip( 6 );    // total field count, report record count, (repeated) cache type
1185cdf0e10cSrcweir     rStrm >> nUserNameLen;
1186cdf0e10cSrcweir     if( nUserNameLen != BIFF_PC_NOSTRING )
1187cdf0e10cSrcweir         maDefModel.maRefreshedBy = (getBiff() == BIFF8) ?
1188cdf0e10cSrcweir             rStrm.readUniString( nUserNameLen ) :
1189cdf0e10cSrcweir             rStrm.readCharArrayUC( nUserNameLen, getTextEncoding() );
1190cdf0e10cSrcweir 
1191cdf0e10cSrcweir     maDefModel.mbInvalid          = getFlag( nFlags, BIFF_PCDEFINITION_INVALID );
1192cdf0e10cSrcweir     maDefModel.mbSaveData         = getFlag( nFlags, BIFF_PCDEFINITION_SAVEDATA );
1193cdf0e10cSrcweir     maDefModel.mbRefreshOnLoad    = getFlag( nFlags, BIFF_PCDEFINITION_REFRESHONLOAD );
1194cdf0e10cSrcweir     maDefModel.mbOptimizeMemory   = getFlag( nFlags, BIFF_PCDEFINITION_OPTIMIZEMEMORY );
1195cdf0e10cSrcweir     maDefModel.mbEnableRefresh    = getFlag( nFlags, BIFF_PCDEFINITION_ENABLEREFRESH );
1196cdf0e10cSrcweir     maDefModel.mbBackgroundQuery  = getFlag( nFlags, BIFF_PCDEFINITION_BACKGROUNDQUERY );
1197cdf0e10cSrcweir 
1198cdf0e10cSrcweir     if( (rStrm.getNextRecId() == BIFF_ID_PCDEFINITION2) && rStrm.startNextRecord() )
1199cdf0e10cSrcweir         rStrm >> maDefModel.mfRefreshedDate;
1200cdf0e10cSrcweir }
1201cdf0e10cSrcweir 
createCacheField(bool bInitDatabaseField)1202cdf0e10cSrcweir PivotCacheField& PivotCache::createCacheField( bool bInitDatabaseField )
1203cdf0e10cSrcweir {
1204cdf0e10cSrcweir     bool bIsDatabaseField = !bInitDatabaseField || (maFields.size() < maDefModel.mnDatabaseFields);
1205cdf0e10cSrcweir     PivotCacheFieldVector::value_type xCacheField( new PivotCacheField( *this, bIsDatabaseField ) );
1206cdf0e10cSrcweir     maFields.push_back( xCacheField );
1207cdf0e10cSrcweir     return *xCacheField;
1208cdf0e10cSrcweir }
1209cdf0e10cSrcweir 
finalizeImport()1210cdf0e10cSrcweir void PivotCache::finalizeImport()
1211cdf0e10cSrcweir {
1212cdf0e10cSrcweir     // collect all fields that are based on source data (needed to finalize source data below)
1213cdf0e10cSrcweir     OSL_ENSURE( !maFields.empty(), "PivotCache::finalizeImport - no pivot cache fields found" );
1214cdf0e10cSrcweir     for( PivotCacheFieldVector::const_iterator aIt = maFields.begin(), aEnd = maFields.end(); aIt != aEnd; ++aIt )
1215cdf0e10cSrcweir     {
1216cdf0e10cSrcweir         if( (*aIt)->isDatabaseField() )
1217cdf0e10cSrcweir         {
1218cdf0e10cSrcweir             OSL_ENSURE( (aIt == maFields.begin()) || (*(aIt - 1))->isDatabaseField(),
1219cdf0e10cSrcweir                 "PivotCache::finalizeImport - database field follows a calculated field" );
1220cdf0e10cSrcweir             maDatabaseIndexes.push_back( static_cast< sal_Int32 >( maDatabaseFields.size() ) );
1221cdf0e10cSrcweir             maDatabaseFields.push_back( *aIt );
1222cdf0e10cSrcweir         }
1223cdf0e10cSrcweir         else
1224cdf0e10cSrcweir         {
1225cdf0e10cSrcweir             maDatabaseIndexes.push_back( -1 );
1226cdf0e10cSrcweir         }
1227cdf0e10cSrcweir     }
1228cdf0e10cSrcweir     OSL_ENSURE( !maDatabaseFields.empty(), "PivotCache::finalizeImport - no pivot cache source fields found" );
1229cdf0e10cSrcweir 
1230cdf0e10cSrcweir     // finalize source data depending on source type
1231cdf0e10cSrcweir     switch( maSourceModel.mnSourceType )
1232cdf0e10cSrcweir     {
1233cdf0e10cSrcweir         case XML_worksheet:
1234cdf0e10cSrcweir         {
1235cdf0e10cSrcweir             // decide whether an external document is used
1236cdf0e10cSrcweir             bool bInternal = (maTargetUrl.getLength() == 0) && (maSheetSrcModel.maRelId.getLength() == 0);
1237cdf0e10cSrcweir             bool bExternal = maTargetUrl.getLength() > 0;   // relation ID may be empty, e.g. BIFF import
1238cdf0e10cSrcweir             OSL_ENSURE( bInternal || bExternal, "PivotCache::finalizeImport - invalid external document URL" );
1239cdf0e10cSrcweir             if( bInternal )
1240cdf0e10cSrcweir                 finalizeInternalSheetSource();
1241cdf0e10cSrcweir             else if( bExternal )
1242cdf0e10cSrcweir                 finalizeExternalSheetSource();
1243cdf0e10cSrcweir         }
1244cdf0e10cSrcweir         break;
1245cdf0e10cSrcweir 
1246cdf0e10cSrcweir         // currently, we only support worksheet data sources
1247cdf0e10cSrcweir         case XML_external:
1248cdf0e10cSrcweir         break;
1249cdf0e10cSrcweir         case XML_consolidation:
1250cdf0e10cSrcweir         break;
1251cdf0e10cSrcweir         case XML_scenario:
1252cdf0e10cSrcweir         break;
1253cdf0e10cSrcweir     }
1254cdf0e10cSrcweir }
1255cdf0e10cSrcweir 
getCacheFieldCount() const1256cdf0e10cSrcweir sal_Int32 PivotCache::getCacheFieldCount() const
1257cdf0e10cSrcweir {
1258cdf0e10cSrcweir     return static_cast< sal_Int32 >( maFields.size() );
1259cdf0e10cSrcweir }
1260cdf0e10cSrcweir 
getCacheField(sal_Int32 nFieldIdx) const1261cdf0e10cSrcweir const PivotCacheField* PivotCache::getCacheField( sal_Int32 nFieldIdx ) const
1262cdf0e10cSrcweir {
1263cdf0e10cSrcweir     return maFields.get( nFieldIdx ).get();
1264cdf0e10cSrcweir }
1265cdf0e10cSrcweir 
getCacheDatabaseIndex(sal_Int32 nFieldIdx) const1266cdf0e10cSrcweir sal_Int32 PivotCache::getCacheDatabaseIndex( sal_Int32 nFieldIdx ) const
1267cdf0e10cSrcweir {
1268cdf0e10cSrcweir     return ContainerHelper::getVectorElement( maDatabaseIndexes, nFieldIdx, -1 );
1269cdf0e10cSrcweir }
1270cdf0e10cSrcweir 
writeSourceHeaderCells(WorksheetHelper & rSheetHelper) const1271cdf0e10cSrcweir void PivotCache::writeSourceHeaderCells( WorksheetHelper& rSheetHelper ) const
1272cdf0e10cSrcweir {
1273cdf0e10cSrcweir     OSL_ENSURE( static_cast< size_t >( maSheetSrcModel.maRange.EndColumn - maSheetSrcModel.maRange.StartColumn + 1 ) == maDatabaseFields.size(),
1274cdf0e10cSrcweir         "PivotCache::writeSourceHeaderCells - source cell range width does not match number of source fields" );
1275cdf0e10cSrcweir     sal_Int32 nCol = maSheetSrcModel.maRange.StartColumn;
1276cdf0e10cSrcweir     sal_Int32 nMaxCol = getAddressConverter().getMaxApiAddress().Column;
1277cdf0e10cSrcweir     sal_Int32 nRow = maSheetSrcModel.maRange.StartRow;
1278cdf0e10cSrcweir     mnCurrRow = -1;
1279cdf0e10cSrcweir     updateSourceDataRow( rSheetHelper, nRow );
1280cdf0e10cSrcweir     for( PivotCacheFieldVector::const_iterator aIt = maDatabaseFields.begin(), aEnd = maDatabaseFields.end(); (aIt != aEnd) && (nCol <= nMaxCol); ++aIt, ++nCol )
1281cdf0e10cSrcweir         (*aIt)->writeSourceHeaderCell( rSheetHelper, nCol, nRow );
1282cdf0e10cSrcweir }
1283cdf0e10cSrcweir 
writeSourceDataCell(WorksheetHelper & rSheetHelper,sal_Int32 nColIdx,sal_Int32 nRowIdx,const PivotCacheItem & rItem) const1284cdf0e10cSrcweir void PivotCache::writeSourceDataCell( WorksheetHelper& rSheetHelper, sal_Int32 nColIdx, sal_Int32 nRowIdx, const PivotCacheItem& rItem ) const
1285cdf0e10cSrcweir {
1286cdf0e10cSrcweir     sal_Int32 nCol = maSheetSrcModel.maRange.StartColumn + nColIdx;
1287cdf0e10cSrcweir     OSL_ENSURE( (maSheetSrcModel.maRange.StartColumn <= nCol) && (nCol <= maSheetSrcModel.maRange.EndColumn), "PivotCache::writeSourceDataCell - invalid column index" );
1288cdf0e10cSrcweir     sal_Int32 nRow = maSheetSrcModel.maRange.StartRow + nRowIdx;
1289cdf0e10cSrcweir     OSL_ENSURE( (maSheetSrcModel.maRange.StartRow < nRow) && (nRow <= maSheetSrcModel.maRange.EndRow), "PivotCache::writeSourceDataCell - invalid row index" );
1290cdf0e10cSrcweir     updateSourceDataRow( rSheetHelper, nRow );
1291cdf0e10cSrcweir     if( const PivotCacheField* pCacheField = maDatabaseFields.get( nColIdx ).get() )
1292cdf0e10cSrcweir         pCacheField->writeSourceDataCell( rSheetHelper, nCol, nRow, rItem );
1293cdf0e10cSrcweir }
1294cdf0e10cSrcweir 
importPCRecord(SequenceInputStream & rStrm,WorksheetHelper & rSheetHelper,sal_Int32 nRowIdx) const1295cdf0e10cSrcweir void PivotCache::importPCRecord( SequenceInputStream& rStrm, WorksheetHelper& rSheetHelper, sal_Int32 nRowIdx ) const
1296cdf0e10cSrcweir {
1297cdf0e10cSrcweir     sal_Int32 nRow = maSheetSrcModel.maRange.StartRow + nRowIdx;
1298cdf0e10cSrcweir     OSL_ENSURE( (maSheetSrcModel.maRange.StartRow < nRow) && (nRow <= maSheetSrcModel.maRange.EndRow), "PivotCache::importPCRecord - invalid row index" );
1299cdf0e10cSrcweir     sal_Int32 nCol = maSheetSrcModel.maRange.StartColumn;
1300cdf0e10cSrcweir     sal_Int32 nMaxCol = getAddressConverter().getMaxApiAddress().Column;
1301cdf0e10cSrcweir     for( PivotCacheFieldVector::const_iterator aIt = maDatabaseFields.begin(), aEnd = maDatabaseFields.end(); !rStrm.isEof() && (aIt != aEnd) && (nCol <= nMaxCol); ++aIt, ++nCol )
1302cdf0e10cSrcweir         (*aIt)->importPCRecordItem( rStrm, rSheetHelper, nCol, nRow );
1303cdf0e10cSrcweir }
1304cdf0e10cSrcweir 
importPCItemIndexList(BiffInputStream & rStrm,WorksheetHelper & rSheetHelper,sal_Int32 nRowIdx) const1305cdf0e10cSrcweir void PivotCache::importPCItemIndexList( BiffInputStream& rStrm, WorksheetHelper& rSheetHelper, sal_Int32 nRowIdx ) const
1306cdf0e10cSrcweir {
1307cdf0e10cSrcweir     sal_Int32 nRow = maSheetSrcModel.maRange.StartRow + nRowIdx;
1308cdf0e10cSrcweir     OSL_ENSURE( (maSheetSrcModel.maRange.StartRow < nRow) && (nRow <= maSheetSrcModel.maRange.EndRow), "PivotCache::importPCItemIndexList - invalid row index" );
1309cdf0e10cSrcweir     sal_Int32 nCol = maSheetSrcModel.maRange.StartColumn;
1310cdf0e10cSrcweir     sal_Int32 nMaxCol = getAddressConverter().getMaxApiAddress().Column;
1311cdf0e10cSrcweir     for( PivotCacheFieldVector::const_iterator aIt = maDatabaseFields.begin(), aEnd = maDatabaseFields.end(); !rStrm.isEof() && (aIt != aEnd) && (nCol <= nMaxCol); ++aIt, ++nCol )
1312cdf0e10cSrcweir         if( (*aIt)->hasSharedItems() )
1313cdf0e10cSrcweir             (*aIt)->importPCItemIndex( rStrm, rSheetHelper, nCol, nRow );
1314cdf0e10cSrcweir }
1315cdf0e10cSrcweir 
1316cdf0e10cSrcweir // private --------------------------------------------------------------------
1317cdf0e10cSrcweir 
importDConRef(BiffInputStream & rStrm)1318cdf0e10cSrcweir void PivotCache::importDConRef( BiffInputStream& rStrm )
1319cdf0e10cSrcweir {
1320cdf0e10cSrcweir     BinRange aBinRange;
1321cdf0e10cSrcweir     aBinRange.read( rStrm, false );     // always 8-bit column indexes
1322cdf0e10cSrcweir     // store range address unchecked with sheet index 0, will be resolved/checked later
1323cdf0e10cSrcweir     getAddressConverter().convertToCellRangeUnchecked( maSheetSrcModel.maRange, aBinRange, 0 );
1324cdf0e10cSrcweir 
1325cdf0e10cSrcweir     // the URL with (required) sheet name and optional URL of an external document
1326cdf0e10cSrcweir     importDConUrl( rStrm );
1327cdf0e10cSrcweir     OSL_ENSURE( maSheetSrcModel.maSheet.getLength() > 0, "PivotCache::importDConRef - missing sheet name" );
1328cdf0e10cSrcweir }
1329cdf0e10cSrcweir 
importDConName(BiffInputStream & rStrm)1330cdf0e10cSrcweir void PivotCache::importDConName( BiffInputStream& rStrm )
1331cdf0e10cSrcweir {
1332cdf0e10cSrcweir     maSheetSrcModel.maDefName = (getBiff() == BIFF8) ? rStrm.readUniString() : rStrm.readByteStringUC( false, getTextEncoding() );
1333cdf0e10cSrcweir     OSL_ENSURE( maSheetSrcModel.maDefName.getLength() > 0, "PivotCache::importDConName - missing defined name" );
1334cdf0e10cSrcweir     importDConUrl( rStrm );
1335cdf0e10cSrcweir }
1336cdf0e10cSrcweir 
importDConBIName(BiffInputStream & rStrm)1337cdf0e10cSrcweir void PivotCache::importDConBIName( BiffInputStream& rStrm )
1338cdf0e10cSrcweir {
1339cdf0e10cSrcweir     sal_uInt8 nNameId = rStrm.readuInt8();
1340cdf0e10cSrcweir     rStrm.skip( 3 );
1341cdf0e10cSrcweir     maSheetSrcModel.maDefName = OUString( sal_Unicode( nNameId ) );
1342cdf0e10cSrcweir     importDConUrl( rStrm );
1343cdf0e10cSrcweir }
1344cdf0e10cSrcweir 
importDConUrl(BiffInputStream & rStrm)1345cdf0e10cSrcweir void PivotCache::importDConUrl( BiffInputStream& rStrm )
1346cdf0e10cSrcweir {
1347cdf0e10cSrcweir     // the URL with sheet name and optional URL of an external document
1348cdf0e10cSrcweir     OUString aEncodedUrl;
1349cdf0e10cSrcweir     if( getBiff() == BIFF8 )
1350cdf0e10cSrcweir     {
1351cdf0e10cSrcweir         // empty string does not contain a flags byte, cannot use simple readUniString() here...
1352cdf0e10cSrcweir         sal_uInt16 nChars = rStrm.readuInt16();
1353cdf0e10cSrcweir         if( nChars > 0 )
1354cdf0e10cSrcweir             aEncodedUrl = rStrm.readUniString( nChars );
1355cdf0e10cSrcweir     }
1356cdf0e10cSrcweir     else
1357cdf0e10cSrcweir     {
1358cdf0e10cSrcweir         aEncodedUrl = rStrm.readByteStringUC( false, getTextEncoding() );
1359cdf0e10cSrcweir     }
1360cdf0e10cSrcweir 
1361cdf0e10cSrcweir     if( aEncodedUrl.getLength() > 0 )
1362cdf0e10cSrcweir     {
1363cdf0e10cSrcweir         OUString aClassName;
1364cdf0e10cSrcweir         getAddressConverter().parseBiffTargetUrl( aClassName, maTargetUrl, maSheetSrcModel.maSheet, aEncodedUrl, true );
1365cdf0e10cSrcweir     }
1366cdf0e10cSrcweir }
1367cdf0e10cSrcweir 
finalizeInternalSheetSource()1368cdf0e10cSrcweir void PivotCache::finalizeInternalSheetSource()
1369cdf0e10cSrcweir {
1370cdf0e10cSrcweir     // resolve sheet name to sheet index
1371cdf0e10cSrcweir     sal_Int16 nSheet = getWorksheets().getCalcSheetIndex( maSheetSrcModel.maSheet );
1372cdf0e10cSrcweir 
1373cdf0e10cSrcweir     // if cache is based on a defined name or table, try to resolve to cell range
1374cdf0e10cSrcweir     if( maSheetSrcModel.maDefName.getLength() > 0 )
1375cdf0e10cSrcweir     {
1376cdf0e10cSrcweir         // local or global defined name
1377cdf0e10cSrcweir         if( const DefinedName* pDefName = getDefinedNames().getByModelName( maSheetSrcModel.maDefName, nSheet ).get() )
1378cdf0e10cSrcweir         {
1379cdf0e10cSrcweir             mbValidSource = pDefName->getAbsoluteRange( maSheetSrcModel.maRange );
1380cdf0e10cSrcweir         }
1381cdf0e10cSrcweir         // table
1382cdf0e10cSrcweir         else if( const Table* pTable = getTables().getTable( maSheetSrcModel.maDefName ).get() )
1383cdf0e10cSrcweir         {
1384cdf0e10cSrcweir             // get original range from table, but exclude the totals row(s)
1385cdf0e10cSrcweir             maSheetSrcModel.maRange = pTable->getOriginalRange();
1386cdf0e10cSrcweir             mbValidSource = (pTable->getHeight() - pTable->getTotalsRows()) > 1;
1387cdf0e10cSrcweir             if( mbValidSource )
1388cdf0e10cSrcweir                 maSheetSrcModel.maRange.EndRow -= pTable->getTotalsRows();
1389cdf0e10cSrcweir         }
1390cdf0e10cSrcweir     }
1391cdf0e10cSrcweir     // else try the cell range (if the sheet exists)
1392cdf0e10cSrcweir     else if( nSheet >= 0 )
1393cdf0e10cSrcweir     {
1394cdf0e10cSrcweir         // insert sheet index into the range, range address will be checked below
1395cdf0e10cSrcweir         maSheetSrcModel.maRange.Sheet = nSheet;
1396cdf0e10cSrcweir         mbValidSource = true;
1397cdf0e10cSrcweir     }
1398cdf0e10cSrcweir     // else sheet has been deleted, generate the source data from cache
1399cdf0e10cSrcweir     else if( maSheetSrcModel.maSheet.getLength() > 0 )
1400cdf0e10cSrcweir     {
1401cdf0e10cSrcweir         prepareSourceDataSheet();
1402cdf0e10cSrcweir         // return here to skip the source range check below
1403cdf0e10cSrcweir         return;
1404cdf0e10cSrcweir     }
1405cdf0e10cSrcweir 
1406cdf0e10cSrcweir     // check range location, do not allow ranges that overflow the sheet partly
1407cdf0e10cSrcweir     mbValidSource = mbValidSource &&
1408cdf0e10cSrcweir         getAddressConverter().checkCellRange( maSheetSrcModel.maRange, false, true ) &&
1409cdf0e10cSrcweir         (maSheetSrcModel.maRange.StartRow < maSheetSrcModel.maRange.EndRow);
1410cdf0e10cSrcweir }
1411cdf0e10cSrcweir 
finalizeExternalSheetSource()1412cdf0e10cSrcweir void PivotCache::finalizeExternalSheetSource()
1413cdf0e10cSrcweir {
1414cdf0e10cSrcweir     /*  If pivot cache is based on external sheet data, try to restore sheet
1415cdf0e10cSrcweir         data from cache records. No support for external defined names or tables,
1416cdf0e10cSrcweir         sheet name and path to cache records fragment (OOXML only) are required. */
1417cdf0e10cSrcweir     bool bHasRelation = (getFilterType() == FILTER_BIFF) || (maDefModel.maRelId.getLength() > 0);
1418cdf0e10cSrcweir     if( bHasRelation && (maSheetSrcModel.maDefName.getLength() == 0) && (maSheetSrcModel.maSheet.getLength() > 0) )
1419cdf0e10cSrcweir         prepareSourceDataSheet();
1420cdf0e10cSrcweir }
1421cdf0e10cSrcweir 
prepareSourceDataSheet()1422cdf0e10cSrcweir void PivotCache::prepareSourceDataSheet()
1423cdf0e10cSrcweir {
1424cdf0e10cSrcweir     CellRangeAddress& rRange = maSheetSrcModel.maRange;
1425cdf0e10cSrcweir     // data will be inserted in top-left cell, sheet index is still set to 0 (will be set below)
1426cdf0e10cSrcweir     rRange.EndColumn -= rRange.StartColumn;
1427cdf0e10cSrcweir     rRange.StartColumn = 0;
1428cdf0e10cSrcweir     rRange.EndRow -= rRange.StartRow;
1429cdf0e10cSrcweir     rRange.StartRow = 0;
1430cdf0e10cSrcweir     // check range location, do not allow ranges that overflow the sheet partly
1431cdf0e10cSrcweir     if( getAddressConverter().checkCellRange( rRange, false, true ) )
1432cdf0e10cSrcweir     {
1433cdf0e10cSrcweir         maColSpans.insert( ValueRange( rRange.StartColumn, rRange.EndColumn ) );
1434cdf0e10cSrcweir         OUString aSheetName = CREATE_OUSTRING( "DPCache_" ) + maSheetSrcModel.maSheet;
1435cdf0e10cSrcweir         rRange.Sheet = getWorksheets().insertEmptySheet( aSheetName, false );
1436cdf0e10cSrcweir         mbValidSource = mbDummySheet = rRange.Sheet >= 0;
1437cdf0e10cSrcweir     }
1438cdf0e10cSrcweir }
1439cdf0e10cSrcweir 
updateSourceDataRow(WorksheetHelper & rSheetHelper,sal_Int32 nRow) const1440cdf0e10cSrcweir void PivotCache::updateSourceDataRow( WorksheetHelper& rSheetHelper, sal_Int32 nRow ) const
1441cdf0e10cSrcweir {
1442cdf0e10cSrcweir     if( mnCurrRow != nRow )
1443cdf0e10cSrcweir     {
1444cdf0e10cSrcweir         rSheetHelper.getSheetData().setColSpans( nRow, maColSpans );
1445cdf0e10cSrcweir         mnCurrRow = nRow;
1446cdf0e10cSrcweir     }
1447cdf0e10cSrcweir }
1448cdf0e10cSrcweir 
1449cdf0e10cSrcweir // ============================================================================
1450cdf0e10cSrcweir 
PivotCacheBuffer(const WorkbookHelper & rHelper)1451cdf0e10cSrcweir PivotCacheBuffer::PivotCacheBuffer( const WorkbookHelper& rHelper ) :
1452cdf0e10cSrcweir     WorkbookHelper( rHelper )
1453cdf0e10cSrcweir {
1454cdf0e10cSrcweir }
1455cdf0e10cSrcweir 
registerPivotCacheFragment(sal_Int32 nCacheId,const OUString & rFragmentPath)1456cdf0e10cSrcweir void PivotCacheBuffer::registerPivotCacheFragment( sal_Int32 nCacheId, const OUString& rFragmentPath )
1457cdf0e10cSrcweir {
1458cdf0e10cSrcweir     OSL_ENSURE( nCacheId >= 0, "PivotCacheBuffer::registerPivotCacheFragment - invalid pivot cache identifier" );
1459cdf0e10cSrcweir     OSL_ENSURE( maFragmentPaths.count( nCacheId ) == 0, "PivotCacheBuffer::registerPivotCacheFragment - fragment path exists already" );
1460cdf0e10cSrcweir     if( (nCacheId >= 0) && (rFragmentPath.getLength() > 0) )
1461cdf0e10cSrcweir         maFragmentPaths[ nCacheId ] = rFragmentPath;
1462cdf0e10cSrcweir }
1463cdf0e10cSrcweir 
importPivotCacheRef(BiffInputStream & rStrm)1464cdf0e10cSrcweir void PivotCacheBuffer::importPivotCacheRef( BiffInputStream& rStrm )
1465cdf0e10cSrcweir {
1466cdf0e10cSrcweir     // read the PIVOTCACHE record that contains the stream ID
1467cdf0e10cSrcweir     sal_Int32 nCacheId = rStrm.readuInt16();
1468cdf0e10cSrcweir     OSL_ENSURE( maFragmentPaths.count( nCacheId ) == 0, "PivotCacheBuffer::importPivotCacheRef - cache stream exists already" );
1469cdf0e10cSrcweir     OUStringBuffer aStrmName;
1470cdf0e10cSrcweir     static const sal_Unicode spcHexChars[] = { '0','1','2','3','4','5','6','7','8','9','A','B','C','D','E','F' };
1471cdf0e10cSrcweir     for( sal_uInt8 nBit = 0; nBit < 16; nBit += 4 )
1472cdf0e10cSrcweir         aStrmName.insert( 0, spcHexChars[ extractValue< size_t >( nCacheId, nBit, 4 ) ] );
1473cdf0e10cSrcweir     aStrmName.insert( 0, (getBiff() == BIFF8) ? CREATE_OUSTRING( "_SX_DB_CUR/" ) : CREATE_OUSTRING( "_SX_DB/" ) );
1474cdf0e10cSrcweir     maFragmentPaths[ nCacheId ] = aStrmName.makeStringAndClear();
1475cdf0e10cSrcweir 
1476cdf0e10cSrcweir     // try to read PCDSOURCE record (will read following data location records too)
1477cdf0e10cSrcweir     sal_uInt16 nNextRecId = rStrm.getNextRecId();
1478cdf0e10cSrcweir     OSL_ENSURE( nNextRecId == BIFF_ID_PCDSOURCE, "PivotCacheBuffer::importPivotCacheRef - PCDSOURCE record expected" );
1479cdf0e10cSrcweir     if( (nNextRecId == BIFF_ID_PCDSOURCE) && rStrm.startNextRecord() )
1480cdf0e10cSrcweir         createPivotCache( nCacheId ).importPCDSource( rStrm );
1481cdf0e10cSrcweir }
1482cdf0e10cSrcweir 
importPivotCacheFragment(sal_Int32 nCacheId)1483cdf0e10cSrcweir PivotCache* PivotCacheBuffer::importPivotCacheFragment( sal_Int32 nCacheId )
1484cdf0e10cSrcweir {
1485cdf0e10cSrcweir     switch( getFilterType() )
1486cdf0e10cSrcweir     {
1487cdf0e10cSrcweir         /*  OOXML/BIFF12 filter: On first call for the cache ID, the pivot
1488cdf0e10cSrcweir             cache object is created and inserted into maCaches. Then, the cache
1489cdf0e10cSrcweir             definition fragment is read and the cache is returned. On
1490cdf0e10cSrcweir             subsequent calls, the created cache will be found in maCaches and
1491cdf0e10cSrcweir             returned immediately. */
1492cdf0e10cSrcweir         case FILTER_OOXML:
1493cdf0e10cSrcweir         {
1494cdf0e10cSrcweir             // try to find an imported pivot cache
1495cdf0e10cSrcweir             if( PivotCache* pCache = maCaches.get( nCacheId ).get() )
1496cdf0e10cSrcweir                 return pCache;
1497cdf0e10cSrcweir 
1498cdf0e10cSrcweir             // check if a fragment path exists for the passed cache identifier
1499cdf0e10cSrcweir             FragmentPathMap::iterator aIt = maFragmentPaths.find( nCacheId );
1500cdf0e10cSrcweir             if( aIt == maFragmentPaths.end() )
1501cdf0e10cSrcweir                 return 0;
1502cdf0e10cSrcweir 
1503cdf0e10cSrcweir             /*  Import the cache fragment. This may create a dummy data sheet
1504cdf0e10cSrcweir                 for external sheet sources. */
1505cdf0e10cSrcweir             PivotCache& rCache = createPivotCache( nCacheId );
1506cdf0e10cSrcweir             importOoxFragment( new PivotCacheDefinitionFragment( *this, aIt->second, rCache ) );
1507cdf0e10cSrcweir             return &rCache;
1508cdf0e10cSrcweir         }
1509cdf0e10cSrcweir 
1510cdf0e10cSrcweir         /*  BIFF filter: Pivot table provides 0-based index into list of pivot
1511cdf0e10cSrcweir             cache source links (PIVOTCACHE/PCDSOURCE/... record blocks in
1512cdf0e10cSrcweir             workbook stream). First, this index has to be resolved to the cache
1513cdf0e10cSrcweir             identifier that is used to manage the cache stream names (the
1514cdf0e10cSrcweir             maFragmentPaths member). The cache object itself exists already
1515cdf0e10cSrcweir             before the first call for the cache source index (see
1516cdf0e10cSrcweir             PivotCacheBuffer::importPivotCacheRef() above), because source data
1517cdf0e10cSrcweir             link is part of workbook data, not of the cache stream. To detect
1518cdf0e10cSrcweir             subsequent calls with an already initialized cache, the entry in
1519cdf0e10cSrcweir             maFragmentPaths will be removed after reading the cache stream. */
1520cdf0e10cSrcweir         case FILTER_BIFF:
1521cdf0e10cSrcweir         {
1522cdf0e10cSrcweir             /*  Resolve cache index to cache identifier and try to find pivot
1523cdf0e10cSrcweir                 cache. Cache must exist already for a valid cache index. */
1524cdf0e10cSrcweir             nCacheId = ContainerHelper::getVectorElement( maCacheIds, nCacheId, -1 );
1525cdf0e10cSrcweir             PivotCache* pCache = maCaches.get( nCacheId ).get();
1526cdf0e10cSrcweir             if( !pCache )
1527cdf0e10cSrcweir                 return 0;
1528cdf0e10cSrcweir 
1529cdf0e10cSrcweir             /*  Try to find fragment path entry (stream name). If missing, the
1530cdf0e10cSrcweir                 stream has been read already, and the cache can be returned. */
1531cdf0e10cSrcweir             FragmentPathMap::iterator aIt = maFragmentPaths.find( nCacheId );
1532cdf0e10cSrcweir             if( aIt != maFragmentPaths.end() )
1533cdf0e10cSrcweir             {
1534cdf0e10cSrcweir                 /*  Import the cache stream. This may create a dummy data sheet
1535cdf0e10cSrcweir                     for external sheet sources. */
1536cdf0e10cSrcweir                 BiffPivotCacheFragment( *this, aIt->second, *pCache ).importFragment();
1537cdf0e10cSrcweir                 // remove the fragment entry to mark that the cache is initialized
1538cdf0e10cSrcweir                 maFragmentPaths.erase( aIt );
1539cdf0e10cSrcweir             }
1540cdf0e10cSrcweir             return pCache;
1541cdf0e10cSrcweir         }
1542cdf0e10cSrcweir 
1543cdf0e10cSrcweir         case FILTER_UNKNOWN:
1544cdf0e10cSrcweir             OSL_ENSURE( false, "PivotCacheBuffer::importPivotCacheFragment - unknown filter type" );
1545cdf0e10cSrcweir     }
1546cdf0e10cSrcweir     return 0;
1547cdf0e10cSrcweir }
1548cdf0e10cSrcweir 
createPivotCache(sal_Int32 nCacheId)1549cdf0e10cSrcweir PivotCache& PivotCacheBuffer::createPivotCache( sal_Int32 nCacheId )
1550cdf0e10cSrcweir {
1551cdf0e10cSrcweir     maCacheIds.push_back( nCacheId );
1552cdf0e10cSrcweir     PivotCacheMap::mapped_type& rxCache = maCaches[ nCacheId ];
1553cdf0e10cSrcweir     rxCache.reset( new PivotCache( *this ) );
1554cdf0e10cSrcweir     return *rxCache;
1555cdf0e10cSrcweir }
1556cdf0e10cSrcweir 
1557cdf0e10cSrcweir // ============================================================================
1558cdf0e10cSrcweir 
1559cdf0e10cSrcweir } // namespace xls
1560cdf0e10cSrcweir } // namespace oox
1561