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 10cdf0e10cSrcweir * 11*ca5ec200SAndrew Rist * http://www.apache.org/licenses/LICENSE-2.0 12cdf0e10cSrcweir * 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. 19cdf0e10cSrcweir * 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 */ 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 170cdf0e10cSrcweir PivotCacheItem::PivotCacheItem() : 171cdf0e10cSrcweir mnType( XML_m ) 172cdf0e10cSrcweir { 173cdf0e10cSrcweir } 174cdf0e10cSrcweir 175cdf0e10cSrcweir void PivotCacheItem::readString( const AttributeList& rAttribs ) 176cdf0e10cSrcweir { 177cdf0e10cSrcweir maValue <<= rAttribs.getXString( XML_v, OUString() ); 178cdf0e10cSrcweir mnType = XML_s; 179cdf0e10cSrcweir } 180cdf0e10cSrcweir 181cdf0e10cSrcweir void PivotCacheItem::readNumeric( const AttributeList& rAttribs ) 182cdf0e10cSrcweir { 183cdf0e10cSrcweir maValue <<= rAttribs.getDouble( XML_v, 0.0 ); 184cdf0e10cSrcweir mnType = XML_n; 185cdf0e10cSrcweir } 186cdf0e10cSrcweir 187cdf0e10cSrcweir void PivotCacheItem::readDate( const AttributeList& rAttribs ) 188cdf0e10cSrcweir { 189cdf0e10cSrcweir maValue <<= rAttribs.getDateTime( XML_v, DateTime() ); 190cdf0e10cSrcweir mnType = XML_d; 191cdf0e10cSrcweir } 192cdf0e10cSrcweir 193cdf0e10cSrcweir void PivotCacheItem::readBool( const AttributeList& rAttribs ) 194cdf0e10cSrcweir { 195cdf0e10cSrcweir maValue <<= rAttribs.getBool( XML_v, false ); 196cdf0e10cSrcweir mnType = XML_b; 197cdf0e10cSrcweir } 198cdf0e10cSrcweir 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 205cdf0e10cSrcweir void PivotCacheItem::readIndex( const AttributeList& rAttribs ) 206cdf0e10cSrcweir { 207cdf0e10cSrcweir maValue <<= rAttribs.getInteger( XML_v, -1 ); 208cdf0e10cSrcweir mnType = XML_x; 209cdf0e10cSrcweir } 210cdf0e10cSrcweir 211cdf0e10cSrcweir void PivotCacheItem::readString( SequenceInputStream& rStrm ) 212cdf0e10cSrcweir { 213cdf0e10cSrcweir maValue <<= BiffHelper::readString( rStrm ); 214cdf0e10cSrcweir mnType = XML_s; 215cdf0e10cSrcweir } 216cdf0e10cSrcweir 217cdf0e10cSrcweir void PivotCacheItem::readDouble( SequenceInputStream& rStrm ) 218cdf0e10cSrcweir { 219cdf0e10cSrcweir maValue <<= rStrm.readDouble(); 220cdf0e10cSrcweir mnType = XML_n; 221cdf0e10cSrcweir } 222cdf0e10cSrcweir 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 237cdf0e10cSrcweir void PivotCacheItem::readBool( SequenceInputStream& rStrm ) 238cdf0e10cSrcweir { 239cdf0e10cSrcweir maValue <<= (rStrm.readuInt8() != 0); 240cdf0e10cSrcweir mnType = XML_b; 241cdf0e10cSrcweir } 242cdf0e10cSrcweir 243cdf0e10cSrcweir void PivotCacheItem::readError( SequenceInputStream& rStrm ) 244cdf0e10cSrcweir { 245cdf0e10cSrcweir maValue <<= static_cast< sal_Int32 >( rStrm.readuInt8() ); 246cdf0e10cSrcweir mnType = XML_e; 247cdf0e10cSrcweir } 248cdf0e10cSrcweir 249cdf0e10cSrcweir void PivotCacheItem::readIndex( SequenceInputStream& rStrm ) 250cdf0e10cSrcweir { 251cdf0e10cSrcweir maValue <<= rStrm.readInt32(); 252cdf0e10cSrcweir mnType = XML_x; 253cdf0e10cSrcweir } 254cdf0e10cSrcweir 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 261cdf0e10cSrcweir void PivotCacheItem::readDouble( BiffInputStream& rStrm ) 262cdf0e10cSrcweir { 263cdf0e10cSrcweir maValue <<= rStrm.readDouble(); 264cdf0e10cSrcweir mnType = XML_n; 265cdf0e10cSrcweir } 266cdf0e10cSrcweir 267cdf0e10cSrcweir void PivotCacheItem::readInteger( BiffInputStream& rStrm ) 268cdf0e10cSrcweir { 269cdf0e10cSrcweir maValue <<= rStrm.readInt16(); 270cdf0e10cSrcweir mnType = XML_i; // fake, used for BIFF only 271cdf0e10cSrcweir } 272cdf0e10cSrcweir 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 287cdf0e10cSrcweir void PivotCacheItem::readBool( BiffInputStream& rStrm ) 288cdf0e10cSrcweir { 289cdf0e10cSrcweir maValue <<= (rStrm.readuInt8() != 0); 290cdf0e10cSrcweir mnType = XML_b; 291cdf0e10cSrcweir } 292cdf0e10cSrcweir 293cdf0e10cSrcweir void PivotCacheItem::readError( BiffInputStream& rStrm ) 294cdf0e10cSrcweir { 295cdf0e10cSrcweir maValue <<= static_cast< sal_Int32 >( rStrm.readuInt8() ); 296cdf0e10cSrcweir mnType = XML_e; 297cdf0e10cSrcweir } 298cdf0e10cSrcweir 299cdf0e10cSrcweir 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 317cdf0e10cSrcweir PivotCacheItemList::PivotCacheItemList( const WorkbookHelper& rHelper ) : 318cdf0e10cSrcweir WorkbookHelper( rHelper ) 319cdf0e10cSrcweir { 320cdf0e10cSrcweir } 321cdf0e10cSrcweir 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 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 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 385cdf0e10cSrcweir const PivotCacheItem* PivotCacheItemList::getCacheItem( sal_Int32 nItemIdx ) const 386cdf0e10cSrcweir { 387cdf0e10cSrcweir return ContainerHelper::getVectorElement( maItems, nItemIdx ); 388cdf0e10cSrcweir } 389cdf0e10cSrcweir 390cdf0e10cSrcweir 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 400cdf0e10cSrcweir PivotCacheItem& PivotCacheItemList::createItem() 401cdf0e10cSrcweir { 402cdf0e10cSrcweir maItems.resize( maItems.size() + 1 ); 403cdf0e10cSrcweir return maItems.back(); 404cdf0e10cSrcweir } 405cdf0e10cSrcweir 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 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 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 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 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 481cdf0e10cSrcweir PivotCacheField::PivotCacheField( const WorkbookHelper& rHelper, bool bIsDatabaseField ) : 482cdf0e10cSrcweir WorkbookHelper( rHelper ), 483cdf0e10cSrcweir maSharedItems( rHelper ), 484cdf0e10cSrcweir maGroupItems( rHelper ) 485cdf0e10cSrcweir { 486cdf0e10cSrcweir maFieldModel.mbDatabaseField = bIsDatabaseField; 487cdf0e10cSrcweir } 488cdf0e10cSrcweir 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 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 520cdf0e10cSrcweir void PivotCacheField::importSharedItem( sal_Int32 nElement, const AttributeList& rAttribs ) 521cdf0e10cSrcweir { 522cdf0e10cSrcweir maSharedItems.importItem( nElement, rAttribs ); 523cdf0e10cSrcweir } 524cdf0e10cSrcweir 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 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 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 552cdf0e10cSrcweir void PivotCacheField::importGroupItem( sal_Int32 nElement, const AttributeList& rAttribs ) 553cdf0e10cSrcweir { 554cdf0e10cSrcweir maGroupItems.importItem( nElement, rAttribs ); 555cdf0e10cSrcweir } 556cdf0e10cSrcweir 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 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 593cdf0e10cSrcweir void PivotCacheField::importPCDFSharedItem( sal_Int32 nRecId, SequenceInputStream& rStrm ) 594cdf0e10cSrcweir { 595cdf0e10cSrcweir maSharedItems.importItem( nRecId, rStrm ); 596cdf0e10cSrcweir } 597cdf0e10cSrcweir 598cdf0e10cSrcweir void PivotCacheField::importPCDFieldGroup( SequenceInputStream& rStrm ) 599cdf0e10cSrcweir { 600cdf0e10cSrcweir rStrm >> maFieldGroupModel.mnParentField >> maFieldGroupModel.mnBaseField; 601cdf0e10cSrcweir } 602cdf0e10cSrcweir 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 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 629cdf0e10cSrcweir void PivotCacheField::importPCDFGroupItem( sal_Int32 nRecId, SequenceInputStream& rStrm ) 630cdf0e10cSrcweir { 631cdf0e10cSrcweir maGroupItems.importItem( nRecId, rStrm ); 632cdf0e10cSrcweir } 633cdf0e10cSrcweir 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 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 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 741cdf0e10cSrcweir 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 750cdf0e10cSrcweir 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 758cdf0e10cSrcweir 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 776cdf0e10cSrcweir 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 820cdf0e10cSrcweir 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 933cdf0e10cSrcweir 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 940cdf0e10cSrcweir 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 950cdf0e10cSrcweir 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 969cdf0e10cSrcweir 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 978cdf0e10cSrcweir 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 999cdf0e10cSrcweir 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 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 1028cdf0e10cSrcweir PCSourceModel::PCSourceModel() : 1029cdf0e10cSrcweir mnSourceType( XML_TOKEN_INVALID ), 1030cdf0e10cSrcweir mnConnectionId( 0 ) 1031cdf0e10cSrcweir { 1032cdf0e10cSrcweir } 1033cdf0e10cSrcweir 1034cdf0e10cSrcweir // ---------------------------------------------------------------------------- 1035cdf0e10cSrcweir 1036cdf0e10cSrcweir PCWorksheetSourceModel::PCWorksheetSourceModel() 1037cdf0e10cSrcweir { 1038cdf0e10cSrcweir maRange.StartColumn = maRange.StartRow = maRange.EndColumn = maRange.EndRow = -1; 1039cdf0e10cSrcweir } 1040cdf0e10cSrcweir 1041cdf0e10cSrcweir // ---------------------------------------------------------------------------- 1042cdf0e10cSrcweir 1043cdf0e10cSrcweir PivotCache::PivotCache( const WorkbookHelper& rHelper ) : 1044cdf0e10cSrcweir WorkbookHelper( rHelper ), 1045cdf0e10cSrcweir mnCurrRow( -1 ), 1046cdf0e10cSrcweir mbValidSource( false ), 1047cdf0e10cSrcweir mbDummySheet( false ) 1048cdf0e10cSrcweir { 1049cdf0e10cSrcweir } 1050cdf0e10cSrcweir 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 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 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 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 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 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 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 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 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 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 1256cdf0e10cSrcweir sal_Int32 PivotCache::getCacheFieldCount() const 1257cdf0e10cSrcweir { 1258cdf0e10cSrcweir return static_cast< sal_Int32 >( maFields.size() ); 1259cdf0e10cSrcweir } 1260cdf0e10cSrcweir 1261cdf0e10cSrcweir const PivotCacheField* PivotCache::getCacheField( sal_Int32 nFieldIdx ) const 1262cdf0e10cSrcweir { 1263cdf0e10cSrcweir return maFields.get( nFieldIdx ).get(); 1264cdf0e10cSrcweir } 1265cdf0e10cSrcweir 1266cdf0e10cSrcweir sal_Int32 PivotCache::getCacheDatabaseIndex( sal_Int32 nFieldIdx ) const 1267cdf0e10cSrcweir { 1268cdf0e10cSrcweir return ContainerHelper::getVectorElement( maDatabaseIndexes, nFieldIdx, -1 ); 1269cdf0e10cSrcweir } 1270cdf0e10cSrcweir 1271cdf0e10cSrcweir 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 1284cdf0e10cSrcweir 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 1295cdf0e10cSrcweir 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 1305cdf0e10cSrcweir 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 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 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 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 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 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 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 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 1440cdf0e10cSrcweir 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 1451cdf0e10cSrcweir PivotCacheBuffer::PivotCacheBuffer( const WorkbookHelper& rHelper ) : 1452cdf0e10cSrcweir WorkbookHelper( rHelper ) 1453cdf0e10cSrcweir { 1454cdf0e10cSrcweir } 1455cdf0e10cSrcweir 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 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 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 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