xref: /trunk/main/sc/source/core/data/dpobject.cxx (revision 86e1cf34)
1 /**************************************************************
2  *
3  * Licensed to the Apache Software Foundation (ASF) under one
4  * or more contributor license agreements.  See the NOTICE file
5  * distributed with this work for additional information
6  * regarding copyright ownership.  The ASF licenses this file
7  * to you under the Apache License, Version 2.0 (the
8  * "License"); you may not use this file except in compliance
9  * with the License.  You may obtain a copy of the License at
10  *
11  *   http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing,
14  * software distributed under the License is distributed on an
15  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16  * KIND, either express or implied.  See the License for the
17  * specific language governing permissions and limitations
18  * under the License.
19  *
20  *************************************************************/
21 
22 
23 
24 // MARKER(update_precomp.py): autogen include statement, do not remove
25 #include "precompiled_sc.hxx"
26 
27 
28 
29 // INCLUDE ---------------------------------------------------------------
30 
31 #include "dpobject.hxx"
32 #include "dptabsrc.hxx"
33 #include "dpsave.hxx"
34 #include "dpdimsave.hxx"
35 #include "dpoutput.hxx"
36 #include "dpshttab.hxx"
37 #include "dpsdbtab.hxx"
38 #include "dpgroup.hxx"
39 #include "document.hxx"
40 #include "rechead.hxx"
41 #include "pivot.hxx"		// PIVOT_DATA_FIELD
42 #include "dapiuno.hxx"		// ScDataPilotConversion
43 #include "miscuno.hxx"
44 #include "scerrors.hxx"
45 #include "refupdat.hxx"
46 #include "scresid.hxx"
47 #include "sc.hrc"
48 #include "attrib.hxx"
49 #include "scitems.hxx"
50 #include "unonames.hxx"
51 // Wang Xu Ming -- 2009-8-17
52 // DataPilot Migration - Cache&&Performance
53 #include "dpglobal.hxx"
54 #include "globstr.hrc"
55 // End Comments
56 #include <com/sun/star/beans/XPropertySet.hpp>
57 #include <com/sun/star/sheet/GeneralFunction.hpp>
58 #include <com/sun/star/sheet/DataPilotFieldFilter.hpp>
59 #include <com/sun/star/sheet/DataPilotFieldOrientation.hpp>
60 #include <com/sun/star/sheet/DataPilotFieldReferenceType.hpp>
61 #include <com/sun/star/sheet/DataPilotTableHeaderData.hpp>
62 #include <com/sun/star/sheet/DataPilotTablePositionData.hpp>
63 #include <com/sun/star/sheet/DataPilotTablePositionType.hpp>
64 #include <com/sun/star/sheet/DimensionFlags.hpp>
65 #include <com/sun/star/lang/XMultiServiceFactory.hpp>
66 #include <com/sun/star/lang/XSingleServiceFactory.hpp>
67 #include <com/sun/star/lang/XSingleComponentFactory.hpp>
68 #include <com/sun/star/lang/XInitialization.hpp>
69 #include <com/sun/star/container/XContentEnumerationAccess.hpp>
70 #include <com/sun/star/sheet/XDrillDownDataSupplier.hpp>
71 
72 #include <comphelper/processfactory.hxx>
73 #include <tools/debug.hxx>
74 #include <tools/diagnose_ex.h>
75 #include <svl/zforlist.hxx>		// IsNumberFormat
76 
77 #include <vector>
78 #include <stdio.h>
79 
80 using namespace com::sun::star;
81 using ::std::vector;
82 using ::boost::shared_ptr;
83 using ::com::sun::star::uno::Sequence;
84 using ::com::sun::star::uno::Reference;
85 using ::com::sun::star::uno::UNO_QUERY;
86 using ::com::sun::star::uno::Any;
87 using ::com::sun::star::uno::Exception;
88 using ::com::sun::star::lang::XComponent;
89 using ::com::sun::star::sheet::DataPilotTableHeaderData;
90 using ::com::sun::star::sheet::DataPilotTablePositionData;
91 using ::com::sun::star::beans::XPropertySet;
92 using ::rtl::OUString;
93 
94 
95 // -----------------------------------------------------------------------
96 
97 #define SCDPSOURCE_SERVICE	"com.sun.star.sheet.DataPilotSource"
98 
99 // -----------------------------------------------------------------------
100 
101 // incompatible versions of data pilot files
102 #define SC_DP_VERSION_CURRENT	6
103 
104 // type of source data
105 #define SC_DP_SOURCE_SHEET		0
106 #define SC_DP_SOURCE_DATABASE	1
107 #define SC_DP_SOURCE_SERVICE	2
108 
109 // -----------------------------------------------------------------------
110 
111 //!	move to a header file
112 #define DP_PROP_COLUMNGRAND			"ColumnGrand"
113 #define DP_PROP_FUNCTION			"Function"
114 #define DP_PROP_IGNOREEMPTY			"IgnoreEmptyRows"
115 #define DP_PROP_ISDATALAYOUT		"IsDataLayoutDimension"
116 //#define DP_PROP_ISVISIBLE			"IsVisible"
117 #define DP_PROP_ORIENTATION			"Orientation"
118 #define DP_PROP_ORIGINAL			"Original"
119 #define DP_PROP_POSITION			"Position"
120 #define DP_PROP_REPEATIFEMPTY		"RepeatIfEmpty"
121 #define DP_PROP_ROWGRAND			"RowGrand"
122 #define DP_PROP_SHOWDETAILS			"ShowDetails"
123 #define DP_PROP_SHOWEMPTY			"ShowEmpty"
124 #define DP_PROP_SUBTOTALS			"SubTotals"
125 #define DP_PROP_USEDHIERARCHY		"UsedHierarchy"
126 
127 // -----------------------------------------------------------------------
128 
lcl_GetDataGetOrientation(const uno::Reference<sheet::XDimensionsSupplier> & xSource)129 sal_uInt16 lcl_GetDataGetOrientation( const uno::Reference<sheet::XDimensionsSupplier>& xSource )
130 {
131 	long nRet = sheet::DataPilotFieldOrientation_HIDDEN;
132 	if ( xSource.is() )
133 	{
134 		uno::Reference<container::XNameAccess> xDimsName = xSource->getDimensions();
135 		uno::Reference<container::XIndexAccess> xIntDims = new ScNameToIndexAccess( xDimsName );
136 		long nIntCount = xIntDims->getCount();
137 		sal_Bool bFound = sal_False;
138 		for (long nIntDim=0; nIntDim<nIntCount && !bFound; nIntDim++)
139 		{
140 			uno::Reference<uno::XInterface> xIntDim =
141 				ScUnoHelpFunctions::AnyToInterface( xIntDims->getByIndex(nIntDim) );
142 			uno::Reference<beans::XPropertySet> xDimProp( xIntDim, uno::UNO_QUERY );
143 			if ( xDimProp.is() )
144 			{
145 				bFound = ScUnoHelpFunctions::GetBoolProperty( xDimProp,
146 					rtl::OUString::createFromAscii(DP_PROP_ISDATALAYOUT) );
147 				//!	error checking -- is "IsDataLayoutDimension" property required??
148 				if (bFound)
149 					nRet = ScUnoHelpFunctions::GetEnumProperty(
150 							xDimProp, rtl::OUString::createFromAscii(DP_PROP_ORIENTATION),
151 							sheet::DataPilotFieldOrientation_HIDDEN );
152 			}
153 		}
154 	}
155     return static_cast< sal_uInt16 >( nRet );
156 }
157 
158 // -----------------------------------------------------------------------
159 
ScDPObject(ScDocument * pD)160 ScDPObject::ScDPObject( ScDocument* pD ) :
161 	pDoc( pD ),
162 	pSaveData( NULL ),
163 	pSheetDesc( NULL ),
164 	pImpDesc( NULL ),
165 	pServDesc( NULL ),
166     mpTableData(static_cast<ScDPTableData*>(NULL)),
167 	pOutput( NULL ),
168 	bSettingsChanged( sal_False ),
169 	bAlive( sal_False ),
170 	bAllowMove( sal_False ),
171 	nHeaderRows( 0 ),
172     mbHeaderLayout(false),
173 	bRefresh( sal_False ), // Wang Xu Ming - DataPilot migration
174     mnCacheId( -1 ), // Wang Xu Ming - DataPilot migration
175     mbCreatingTableData( false )
176 {
177 }
178 
ScDPObject(const ScDPObject & r)179 ScDPObject::ScDPObject(const ScDPObject& r) :
180     ScDataObject(),
181 	pDoc( r.pDoc ),
182 	pSaveData( NULL ),
183 	aTableName( r.aTableName ),
184 	aTableTag( r.aTableTag ),
185 	aOutRange( r.aOutRange ),
186 	pSheetDesc( NULL ),
187 	pImpDesc( NULL ),
188 	pServDesc( NULL ),
189     mpTableData(static_cast<ScDPTableData*>(NULL)),
190 	pOutput( NULL ),
191 	bSettingsChanged( sal_False ),
192 	bAlive( sal_False ),
193 	bAllowMove( sal_False ),
194 	nHeaderRows( r.nHeaderRows ),
195     mbHeaderLayout( r.mbHeaderLayout ),
196 	bRefresh( r.bRefresh ), // Wang Xu Ming - DataPilot migration
197     mnCacheId ( r.mnCacheId ), // Wang Xu Ming - DataPilot migration
198     mbCreatingTableData( false )
199 {
200 	if (r.pSaveData)
201 		pSaveData = new ScDPSaveData(*r.pSaveData);
202 	if (r.pSheetDesc)
203 		pSheetDesc = new ScSheetSourceDesc(*r.pSheetDesc);
204 	if (r.pImpDesc)
205 		pImpDesc = new ScImportSourceDesc(*r.pImpDesc);
206 	if (r.pServDesc)
207 		pServDesc = new ScDPServiceDesc(*r.pServDesc);
208 	// xSource (and pOutput) is not copied
209 }
210 
~ScDPObject()211 ScDPObject::~ScDPObject()
212 {
213 	delete pOutput;
214 	delete pSaveData;
215 	delete pSheetDesc;
216 	delete pImpDesc;
217 	delete pServDesc;
218 	mnCacheId = -1; // Wang Xu Ming - DataPilot migration
219     InvalidateSource();
220 }
221 
Clone() const222 ScDataObject* ScDPObject::Clone() const
223 {
224 	return new ScDPObject(*this);
225 }
226 
SetAlive(sal_Bool bSet)227 void ScDPObject::SetAlive(sal_Bool bSet)
228 {
229 	bAlive = bSet;
230 }
231 
SetAllowMove(sal_Bool bSet)232 void ScDPObject::SetAllowMove(sal_Bool bSet)
233 {
234 	bAllowMove = bSet;
235 }
236 
SetSaveData(const ScDPSaveData & rData)237 void ScDPObject::SetSaveData(const ScDPSaveData& rData)
238 {
239     if ( pSaveData != &rData )      // API implementation modifies the original SaveData object
240     {
241         delete pSaveData;
242         pSaveData = new ScDPSaveData( rData );
243         // Wang Xu Ming -- 2009-8-17
244         // DataPilot Migration - Cache&&Performance
245         if ( rData.GetCacheId() >= 0 )
246             mnCacheId = rData.GetCacheId();
247         else if ( mnCacheId >= 0 )
248             pSaveData->SetCacheId( mnCacheId );
249         // End Comments
250     }
251 
252 	InvalidateData();		// re-init source from SaveData
253 }
254 
SetHeaderLayout(bool bUseGrid)255 void ScDPObject::SetHeaderLayout (bool bUseGrid)
256 {
257     mbHeaderLayout = bUseGrid;
258 }
259 
GetHeaderLayout() const260 bool ScDPObject::GetHeaderLayout() const
261 {
262     return mbHeaderLayout;
263 }
264 
SetOutRange(const ScRange & rRange)265 void ScDPObject::SetOutRange(const ScRange& rRange)
266 {
267 	aOutRange = rRange;
268 
269 	if ( pOutput )
270 		pOutput->SetPosition( rRange.aStart );
271 }
272 
SetSheetDesc(const ScSheetSourceDesc & rDesc,bool bFromRefUpdate)273 void ScDPObject::SetSheetDesc(const ScSheetSourceDesc& rDesc, bool bFromRefUpdate)
274 {
275 	if ( pSheetDesc && rDesc == *pSheetDesc )
276 		return;				// nothing to do
277 
278 	DELETEZ( pImpDesc );
279 	DELETEZ( pServDesc );
280 
281     delete pSheetDesc;
282 	pSheetDesc = new ScSheetSourceDesc(rDesc);
283 
284 	//	make valid QueryParam
285 
286 	pSheetDesc->aQueryParam.nCol1 = pSheetDesc->aSourceRange.aStart.Col();
287 	pSheetDesc->aQueryParam.nRow1 = pSheetDesc->aSourceRange.aStart.Row();
288 	pSheetDesc->aQueryParam.nCol2 = pSheetDesc->aSourceRange.aEnd.Col();
289 	pSheetDesc->aQueryParam.nRow2 = pSheetDesc->aSourceRange.aEnd.Row();;
290 	pSheetDesc->aQueryParam.bHasHeader = sal_True;
291 
292 	InvalidateSource();		// new source must be created
293     if (!bFromRefUpdate)
294         SetCacheId( -1 );   // #i116504# don't use the same cache ID for a different range (except reference update)
295 }
296 
SetImportDesc(const ScImportSourceDesc & rDesc)297 void ScDPObject::SetImportDesc(const ScImportSourceDesc& rDesc)
298 {
299 	if ( pImpDesc && rDesc == *pImpDesc )
300 		return;				// nothing to do
301 
302 	DELETEZ( pSheetDesc );
303 	DELETEZ( pServDesc );
304 
305 	delete pImpDesc;
306 	pImpDesc = new ScImportSourceDesc(rDesc);
307 
308 	InvalidateSource();		// new source must be created
309     SetCacheId( -1 );
310 }
311 
SetServiceData(const ScDPServiceDesc & rDesc)312 void ScDPObject::SetServiceData(const ScDPServiceDesc& rDesc)
313 {
314 	if ( pServDesc && rDesc == *pServDesc )
315 		return;				// nothing to do
316 
317 	DELETEZ( pSheetDesc );
318 	DELETEZ( pImpDesc );
319 
320 	delete pServDesc;
321 	pServDesc = new ScDPServiceDesc(rDesc);
322 
323 	InvalidateSource();		// new source must be created
324 }
325 
WriteSourceDataTo(ScDPObject & rDest) const326 void ScDPObject::WriteSourceDataTo( ScDPObject& rDest ) const
327 {
328 	if ( pSheetDesc )
329 		rDest.SetSheetDesc( *pSheetDesc );
330 	else if ( pImpDesc )
331 		rDest.SetImportDesc( *pImpDesc );
332 	else if ( pServDesc )
333 		rDest.SetServiceData( *pServDesc );
334 
335 	//	name/tag are not source data, but needed along with source data
336 
337 	rDest.aTableName = aTableName;
338 	rDest.aTableTag  = aTableTag;
339 }
340 
WriteTempDataTo(ScDPObject & rDest) const341 void ScDPObject::WriteTempDataTo( ScDPObject& rDest ) const
342 {
343 	rDest.nHeaderRows = nHeaderRows;
344 }
345 
IsSheetData() const346 sal_Bool ScDPObject::IsSheetData() const
347 {
348 	return ( pSheetDesc != NULL );
349 }
350 
SetName(const String & rNew)351 void ScDPObject::SetName(const String& rNew)
352 {
353 	aTableName = rNew;
354 }
355 
SetTag(const String & rNew)356 void ScDPObject::SetTag(const String& rNew)
357 {
358 	aTableTag = rNew;
359 }
360 
IsDataDescriptionCell(const ScAddress & rPos)361 bool ScDPObject::IsDataDescriptionCell(const ScAddress& rPos)
362 {
363     if (!pSaveData)
364         return false;
365 
366     long nDataDimCount = pSaveData->GetDataDimensionCount();
367     if (nDataDimCount != 1)
368         // There has to be exactly one data dimension for the description to
369         // appear at top-left corner.
370         return false;
371 
372     CreateOutput();
373     ScRange aTabRange = pOutput->GetOutputRange(sheet::DataPilotOutputRangeType::TABLE);
374     return (rPos == aTabRange.aStart);
375 }
376 
GetSource()377 uno::Reference<sheet::XDimensionsSupplier> ScDPObject::GetSource()
378 {
379 	CreateObjects();
380 	return xSource;
381 }
382 
CreateOutput()383 void ScDPObject::CreateOutput()
384 {
385 	CreateObjects();
386 	if (!pOutput)
387 	{
388         sal_Bool bFilterButton = IsSheetData() && pSaveData && pSaveData->GetFilterButton();
389         pOutput = new ScDPOutput( pDoc, xSource, aOutRange.aStart, bFilterButton );
390         pOutput->SetHeaderLayout ( mbHeaderLayout );
391 
392 		long nOldRows = nHeaderRows;
393 		nHeaderRows = pOutput->GetHeaderRows();
394 
395 		if ( bAllowMove && nHeaderRows != nOldRows )
396 		{
397 			long nDiff = nOldRows - nHeaderRows;
398 			if ( nOldRows == 0 )
399 				--nDiff;
400 			if ( nHeaderRows == 0 )
401 				++nDiff;
402 
403 			long nNewRow = aOutRange.aStart.Row() + nDiff;
404 			if ( nNewRow < 0 )
405 				nNewRow = 0;
406 
407 			ScAddress aStart( aOutRange.aStart );
408             aStart.SetRow(nNewRow);
409 			pOutput->SetPosition( aStart );
410 
411 			//!	modify aOutRange?
412 
413 			bAllowMove = sal_False;		// use only once
414 		}
415 	}
416 }
417 
GetTableData()418 ScDPTableData* ScDPObject::GetTableData()
419 {
420     if (!mpTableData && !mbCreatingTableData)
421     {
422         // #i117239# While filling the cache, mpTableData is still null.
423         // Prevent nested calls from GetPivotData and similar functions.
424         mbCreatingTableData = true;
425 
426         shared_ptr<ScDPTableData> pData;
427         if ( pImpDesc )
428         {
429             // database data
430             pData.reset(new ScDatabaseDPData(pDoc, *pImpDesc, GetCacheId()));
431         }
432         else
433         {
434             // cell data
435             if (!pSheetDesc)
436             {
437                 DBG_ERROR("no source descriptor");
438                 pSheetDesc = new ScSheetSourceDesc;     // dummy defaults
439             }
440             // Wang Xu Ming -- 2009-8-17
441             // DataPilot Migration - Cache&&Performance
442             pData.reset(new ScSheetDPData(pDoc, *pSheetDesc, GetCacheId()));
443             // End Comments
444         }
445 
446         // grouping (for cell or database data)
447         if ( pSaveData && pSaveData->GetExistingDimensionData() )
448         {
449             shared_ptr<ScDPGroupTableData> pGroupData(new ScDPGroupTableData(pData, pDoc));
450             pSaveData->GetExistingDimensionData()->WriteToData(*pGroupData);
451             pData = pGroupData;
452         }
453 
454         // Wang Xu Ming -- 2009-8-17
455         // DataPilot Migration - Cache&&Performance
456         if ( pData )
457            SetCacheId( pData->GetCacheId());        // resets mpTableData
458         // End Comments
459 
460         mpTableData = pData;                        // after SetCacheId
461 
462         mbCreatingTableData = false;
463     }
464 
465     return mpTableData.get();
466 }
467 
CreateObjects()468 void ScDPObject::CreateObjects()
469 {
470     // if groups are involved, create a new source with the ScDPGroupTableData
471     if ( bSettingsChanged && pSaveData && pSaveData->GetExistingDimensionData() )
472         InvalidateSource();
473 
474     if (!xSource.is())
475     {
476         //!	cache DPSource and/or Output?
477 
478         DBG_ASSERT( bAlive, "CreateObjects on non-inserted DPObject" );
479 
480         DELETEZ( pOutput );		// not valid when xSource is changed
481 
482         if ( pServDesc )
483         {
484             xSource = CreateSource( *pServDesc );
485         }
486 
487         if ( !xSource.is() )    // database or sheet data, or error in CreateSource
488 		{
489 			DBG_ASSERT( !pServDesc, "DPSource could not be created" );
490             ScDPTableData* pData = GetTableData();
491 
492             if ( pData )    // nested GetTableData calls may return NULL
493             {
494                 ScDPSource* pSource = new ScDPSource( pData );
495                 xSource = pSource;
496 
497                 if ( pSaveData && bRefresh )
498                 {
499                     pSaveData->Refresh( xSource );
500                     bRefresh = sal_False;
501                 }
502             }
503 		}
504         if ( xSource.is() && pSaveData )
505             pSaveData->WriteToSource( xSource );
506     }
507     else if (bSettingsChanged)
508     {
509         DELETEZ( pOutput );		// not valid when xSource is changed
510 
511         uno::Reference<util::XRefreshable> xRef( xSource, uno::UNO_QUERY );
512         if (xRef.is())
513         {
514             try
515             {
516                 xRef->refresh();
517             }
518             catch(uno::Exception&)
519             {
520                 DBG_ERROR("exception in refresh");
521             }
522         }
523 
524         if (pSaveData)
525             pSaveData->WriteToSource( xSource );
526     }
527 	bSettingsChanged = sal_False;
528 }
529 
InvalidateData()530 void ScDPObject::InvalidateData()
531 {
532 	bSettingsChanged = sal_True;
533 }
534 
InvalidateSource()535 void ScDPObject::InvalidateSource()
536 {
537     Reference< XComponent > xObjectComp( xSource, UNO_QUERY );
538     if ( xObjectComp.is() )
539     {
540         try
541         {
542             xObjectComp->dispose();
543         }
544         catch( const Exception& )
545         {
546             DBG_UNHANDLED_EXCEPTION();
547         }
548     }
549     xSource = NULL;
550     mpTableData.reset();
551 }
552 
GetNewOutputRange(sal_Bool & rOverflow)553 ScRange ScDPObject::GetNewOutputRange( sal_Bool& rOverflow )
554 {
555 	CreateOutput();				// create xSource and pOutput if not already done
556 
557 	rOverflow = pOutput->HasError();		// range overflow or exception from source
558 	if ( rOverflow )
559 		return ScRange( aOutRange.aStart );
560 	else
561 	{
562 		//	don't store the result in aOutRange, because nothing has been output yet
563 		return pOutput->GetOutputRange();
564 	}
565 }
566 
Output(const ScAddress & rPos)567 void ScDPObject::Output( const ScAddress& rPos )
568 {
569 	//	clear old output area
570 	pDoc->DeleteAreaTab( aOutRange.aStart.Col(), aOutRange.aStart.Row(),
571 						 aOutRange.aEnd.Col(),   aOutRange.aEnd.Row(),
572 						 aOutRange.aStart.Tab(), IDF_ALL );
573 	pDoc->RemoveFlagsTab( aOutRange.aStart.Col(), aOutRange.aStart.Row(),
574 						  aOutRange.aEnd.Col(),   aOutRange.aEnd.Row(),
575 						  aOutRange.aStart.Tab(), SC_MF_AUTO );
576 
577 	CreateOutput();				// create xSource and pOutput if not already done
578 
579     pOutput->SetPosition( rPos );
580 
581 	pOutput->Output();
582 
583 	//	aOutRange is always the range that was last output to the document
584 	aOutRange = pOutput->GetOutputRange();
585     const ScAddress& s = aOutRange.aStart;
586     const ScAddress& e = aOutRange.aEnd;
587     pDoc->ApplyFlagsTab(s.Col(), s.Row(), e.Col(), e.Row(), s.Tab(), SC_MF_DP_TABLE);
588 }
589 
GetOutputRangeByType(sal_Int32 nType)590 const ScRange ScDPObject::GetOutputRangeByType( sal_Int32 nType )
591 {
592     CreateOutput();
593 
594     if (pOutput->HasError())
595         return ScRange(aOutRange.aStart);
596 
597     return pOutput->GetOutputRange(nType);
598 }
599 
lcl_HasButton(ScDocument * pDoc,SCCOL nCol,SCROW nRow,SCTAB nTab)600 sal_Bool lcl_HasButton( ScDocument* pDoc, SCCOL nCol, SCROW nRow, SCTAB nTab )
601 {
602 	return ((const ScMergeFlagAttr*)pDoc->GetAttr( nCol, nRow, nTab, ATTR_MERGE_FLAG ))->HasButton();
603 }
604 
RefreshAfterLoad()605 void ScDPObject::RefreshAfterLoad()
606 {
607 	// apply drop-down attribute, initialize nHeaderRows, without accessing the source
608 	// (button attribute must be present)
609 
610 	// simple test: block of button cells at the top, followed by an empty cell
611 
612 	SCCOL nFirstCol = aOutRange.aStart.Col();
613 	SCROW nFirstRow = aOutRange.aStart.Row();
614 	SCTAB nTab = aOutRange.aStart.Tab();
615 
616 	SCROW nInitial = 0;
617 	SCROW nOutRows = aOutRange.aEnd.Row() + 1 - aOutRange.aStart.Row();
618 	while ( nInitial + 1 < nOutRows && lcl_HasButton( pDoc, nFirstCol, nFirstRow + nInitial, nTab ) )
619 		++nInitial;
620 
621 	if ( nInitial + 1 < nOutRows &&
622 		pDoc->IsBlockEmpty( nTab, nFirstCol, nFirstRow + nInitial, nFirstCol, nFirstRow + nInitial ) &&
623 		aOutRange.aEnd.Col() > nFirstCol )
624 	{
625 		sal_Bool bFilterButton = IsSheetData();			// when available, filter button setting must be checked here
626 
627 		SCROW nSkip = bFilterButton ? 1 : 0;
628 		for (SCROW nPos=nSkip; nPos<nInitial; nPos++)
629 			pDoc->ApplyAttr( nFirstCol + 1, nFirstRow + nPos, nTab, ScMergeFlagAttr(SC_MF_AUTO) );
630 
631 		nHeaderRows = nInitial;
632 	}
633 	else
634 		nHeaderRows = 0;		// nothing found, no drop-down lists
635 }
636 
BuildAllDimensionMembers()637 void ScDPObject::BuildAllDimensionMembers()
638 {
639     if (!pSaveData)
640         return;
641 
642     // #i111857# don't always create empty mpTableData for external service.
643     // #163781# Initialize all members from xSource instead.
644     if (pServDesc)
645     {
646         pSaveData->BuildAllDimensionMembersFromSource( this );
647         return;
648     }
649 
650     pSaveData->BuildAllDimensionMembers(GetTableData());
651 }
652 
GetMemberNames(sal_Int32 nDim,Sequence<OUString> & rNames)653 bool ScDPObject::GetMemberNames( sal_Int32 nDim, Sequence<OUString>& rNames )
654 {
655     vector<ScDPLabelData::Member> aMembers;
656     if (!GetMembers(nDim, GetUsedHierarchy(nDim), aMembers))
657         return false;
658 
659     size_t n = aMembers.size();
660     rNames.realloc(n);
661     for (size_t i = 0; i < n; ++i)
662         rNames[i] = aMembers[i].maName;
663 
664     return true;
665 }
666 
GetMembers(sal_Int32 nDim,sal_Int32 nHier,vector<ScDPLabelData::Member> & rMembers)667 bool ScDPObject::GetMembers( sal_Int32 nDim, sal_Int32 nHier, vector<ScDPLabelData::Member>& rMembers )
668 {
669     Reference< container::XNameAccess > xMembersNA;
670     if (!GetMembersNA( nDim, nHier, xMembersNA ))
671         return false;
672 
673     Reference<container::XIndexAccess> xMembersIA( new ScNameToIndexAccess(xMembersNA) );
674     sal_Int32 nCount = xMembersIA->getCount();
675     vector<ScDPLabelData::Member> aMembers;
676     aMembers.reserve(nCount);
677 
678     for (sal_Int32 i = 0; i < nCount; ++i)
679     {
680         Reference<container::XNamed> xMember(xMembersIA->getByIndex(i), UNO_QUERY);
681         ScDPLabelData::Member aMem;
682 
683         if (xMember.is())
684             aMem.maName = xMember->getName();
685 
686         Reference<beans::XPropertySet> xMemProp(xMember, UNO_QUERY);
687         if (xMemProp.is())
688         {
689             aMem.mbVisible     = ScUnoHelpFunctions::GetBoolProperty(xMemProp, OUString::createFromAscii(SC_UNO_ISVISIBL));
690             aMem.mbShowDetails = ScUnoHelpFunctions::GetBoolProperty(xMemProp, OUString::createFromAscii(SC_UNO_SHOWDETA));
691 
692             aMem.maLayoutName = ScUnoHelpFunctions::GetStringProperty(
693                 xMemProp, OUString::createFromAscii(SC_UNO_LAYOUTNAME), OUString());
694         }
695 
696         aMembers.push_back(aMem);
697     }
698     rMembers.swap(aMembers);
699     return true;
700 }
701 
UpdateReference(UpdateRefMode eUpdateRefMode,const ScRange & rRange,SCsCOL nDx,SCsROW nDy,SCsTAB nDz)702 void ScDPObject::UpdateReference( UpdateRefMode eUpdateRefMode,
703 									 const ScRange& rRange, SCsCOL nDx, SCsROW nDy, SCsTAB nDz )
704 {
705 	// Output area
706 
707 	SCCOL nCol1 = aOutRange.aStart.Col();
708 	SCROW nRow1 = aOutRange.aStart.Row();
709 	SCTAB nTab1 = aOutRange.aStart.Tab();
710 	SCCOL nCol2 = aOutRange.aEnd.Col();
711 	SCROW nRow2 = aOutRange.aEnd.Row();
712 	SCTAB nTab2 = aOutRange.aEnd.Tab();
713 
714 	ScRefUpdateRes eRes =
715 		ScRefUpdate::Update( pDoc, eUpdateRefMode,
716 			rRange.aStart.Col(), rRange.aStart.Row(), rRange.aStart.Tab(),
717 			rRange.aEnd.Col(), rRange.aEnd.Row(), rRange.aEnd.Tab(), nDx, nDy, nDz,
718 			nCol1, nRow1, nTab1, nCol2, nRow2, nTab2 );
719 	if ( eRes != UR_NOTHING )
720 		SetOutRange( ScRange( nCol1, nRow1, nTab1, nCol2, nRow2, nTab2 ) );
721 
722 	// sheet source data
723 
724 	if ( pSheetDesc )
725 	{
726 		nCol1 = pSheetDesc->aSourceRange.aStart.Col();
727 		nRow1 = pSheetDesc->aSourceRange.aStart.Row();
728 		nTab1 = pSheetDesc->aSourceRange.aStart.Tab();
729 		nCol2 = pSheetDesc->aSourceRange.aEnd.Col();
730 		nRow2 = pSheetDesc->aSourceRange.aEnd.Row();
731 		nTab2 = pSheetDesc->aSourceRange.aEnd.Tab();
732 
733 		eRes = ScRefUpdate::Update( pDoc, eUpdateRefMode,
734 				rRange.aStart.Col(), rRange.aStart.Row(), rRange.aStart.Tab(),
735 				rRange.aEnd.Col(), rRange.aEnd.Row(), rRange.aEnd.Tab(), nDx, nDy, nDz,
736 				nCol1, nRow1, nTab1, nCol2, nRow2, nTab2 );
737 		if ( eRes != UR_NOTHING )
738 		{
739 			ScSheetSourceDesc aNewDesc;
740 			aNewDesc.aSourceRange = ScRange( nCol1, nRow1, nTab1, nCol2, nRow2, nTab2 );
741 
742 			SCsCOL nDiffX = nCol1 - (SCsCOL) pSheetDesc->aSourceRange.aStart.Col();
743 			SCsROW nDiffY = nRow1 - (SCsROW) pSheetDesc->aSourceRange.aStart.Row();
744 
745 			aNewDesc.aQueryParam = pSheetDesc->aQueryParam;
746             aNewDesc.aQueryParam.nCol1 = sal::static_int_cast<SCCOL>( aNewDesc.aQueryParam.nCol1 + nDiffX );
747             aNewDesc.aQueryParam.nCol2 = sal::static_int_cast<SCCOL>( aNewDesc.aQueryParam.nCol2 + nDiffX );
748 			aNewDesc.aQueryParam.nRow1 += nDiffY;	//! used?
749 			aNewDesc.aQueryParam.nRow2 += nDiffY;	//! used?
750 			SCSIZE nEC = aNewDesc.aQueryParam.GetEntryCount();
751 			for (SCSIZE i=0; i<nEC; i++)
752 				if (aNewDesc.aQueryParam.GetEntry(i).bDoQuery)
753 					aNewDesc.aQueryParam.GetEntry(i).nField += nDiffX;
754 
755             SetSheetDesc( aNewDesc, true );     // allocates new pSheetDesc
756 		}
757 	}
758 }
759 
RefsEqual(const ScDPObject & r) const760 sal_Bool ScDPObject::RefsEqual( const ScDPObject& r ) const
761 {
762 	if ( aOutRange != r.aOutRange )
763 		return sal_False;
764 
765 	if ( pSheetDesc && r.pSheetDesc )
766 	{
767 		if ( pSheetDesc->aSourceRange != r.pSheetDesc->aSourceRange )
768 			return sal_False;
769 	}
770 	else if ( pSheetDesc || r.pSheetDesc )
771 	{
772 		DBG_ERROR("RefsEqual: SheetDesc set at only one object");
773 		return sal_False;
774 	}
775 
776 	return sal_True;
777 }
778 
WriteRefsTo(ScDPObject & r) const779 void ScDPObject::WriteRefsTo( ScDPObject& r ) const
780 {
781 	r.SetOutRange( aOutRange );
782 	if ( pSheetDesc )
783         r.SetSheetDesc( *pSheetDesc, true );
784 }
785 
GetPositionData(const ScAddress & rPos,DataPilotTablePositionData & rPosData)786 void ScDPObject::GetPositionData(const ScAddress& rPos, DataPilotTablePositionData& rPosData)
787 {
788     CreateOutput();
789     pOutput->GetPositionData(rPos, rPosData);
790 }
791 
GetDataFieldPositionData(const ScAddress & rPos,Sequence<sheet::DataPilotFieldFilter> & rFilters)792 bool ScDPObject::GetDataFieldPositionData(
793     const ScAddress& rPos, Sequence<sheet::DataPilotFieldFilter>& rFilters)
794 {
795     CreateOutput();
796 
797     vector<sheet::DataPilotFieldFilter> aFilters;
798     if (!pOutput->GetDataResultPositionData(aFilters, rPos))
799         return false;
800 
801     sal_Int32 n = static_cast<sal_Int32>(aFilters.size());
802     rFilters.realloc(n);
803     for (sal_Int32 i = 0; i < n; ++i)
804         rFilters[i] = aFilters[i];
805 
806     return true;
807 }
808 
GetDrillDownData(const ScAddress & rPos,Sequence<Sequence<Any>> & rTableData)809 void ScDPObject::GetDrillDownData(const ScAddress& rPos, Sequence< Sequence<Any> >& rTableData)
810 {
811     CreateOutput();
812 
813     Reference<sheet::XDrillDownDataSupplier> xDrillDownData(xSource, UNO_QUERY);
814     if (!xDrillDownData.is())
815         return;
816 
817     Sequence<sheet::DataPilotFieldFilter> filters;
818     if (!GetDataFieldPositionData(rPos, filters))
819         return;
820 
821     rTableData = xDrillDownData->getDrillDownData(filters);
822 }
823 
IsDimNameInUse(const OUString & rName) const824 bool ScDPObject::IsDimNameInUse(const OUString& rName) const
825 {
826     if (!xSource.is())
827         return false;
828 
829     Reference<container::XNameAccess> xDims = xSource->getDimensions();
830     Sequence<OUString> aDimNames = xDims->getElementNames();
831     sal_Int32 n = aDimNames.getLength();
832     for (sal_Int32 i = 0; i < n; ++i)
833     {
834         const OUString& rDimName = aDimNames[i];
835         if (rDimName.equalsIgnoreAsciiCase(rName))
836             return true;
837 
838         Reference<beans::XPropertySet> xPropSet(xDims->getByName(rDimName), UNO_QUERY);
839         if (!xPropSet.is())
840             continue;
841 
842         OUString aLayoutName = ScUnoHelpFunctions::GetStringProperty(
843             xPropSet, OUString::createFromAscii(SC_UNO_LAYOUTNAME), OUString());
844         if (aLayoutName.equalsIgnoreAsciiCase(rName))
845             return true;
846     }
847     return false;
848 }
849 
GetDimName(long nDim,sal_Bool & rIsDataLayout,sal_Int32 * pFlags)850 String ScDPObject::GetDimName( long nDim, sal_Bool& rIsDataLayout, sal_Int32* pFlags )
851 {
852 	rIsDataLayout = sal_False;
853 	String aRet;
854 
855 	if ( xSource.is() )
856 	{
857 		uno::Reference<container::XNameAccess> xDimsName = xSource->getDimensions();
858 		uno::Reference<container::XIndexAccess> xDims = new ScNameToIndexAccess( xDimsName );
859 		long nDimCount = xDims->getCount();
860 		if ( nDim < nDimCount )
861 		{
862 			uno::Reference<uno::XInterface> xIntDim =
863 				ScUnoHelpFunctions::AnyToInterface( xDims->getByIndex(nDim) );
864 			uno::Reference<container::XNamed> xDimName( xIntDim, uno::UNO_QUERY );
865 			uno::Reference<beans::XPropertySet> xDimProp( xIntDim, uno::UNO_QUERY );
866 			if ( xDimName.is() && xDimProp.is() )
867 			{
868 				sal_Bool bData = ScUnoHelpFunctions::GetBoolProperty( xDimProp,
869 								rtl::OUString::createFromAscii(DP_PROP_ISDATALAYOUT) );
870 				//!	error checking -- is "IsDataLayoutDimension" property required??
871 
872 				rtl::OUString aName;
873 				try
874 				{
875 					aName = xDimName->getName();
876 				}
877 				catch(uno::Exception&)
878 				{
879 				}
880 				if ( bData )
881 					rIsDataLayout = sal_True;
882 				else
883 					aRet = String( aName );
884 
885                 if (pFlags)
886                     *pFlags = ScUnoHelpFunctions::GetLongProperty( xDimProp,
887                                 rtl::OUString::createFromAscii(SC_UNO_FLAGS), 0 );
888 			}
889 		}
890 	}
891 
892 	return aRet;
893 }
894 
IsDuplicated(long nDim)895 sal_Bool ScDPObject::IsDuplicated( long nDim )
896 {
897     sal_Bool bDuplicated = sal_False;
898     if ( xSource.is() )
899     {
900         uno::Reference<container::XNameAccess> xDimsName = xSource->getDimensions();
901         uno::Reference<container::XIndexAccess> xDims = new ScNameToIndexAccess( xDimsName );
902         long nDimCount = xDims->getCount();
903         if ( nDim < nDimCount )
904         {
905             uno::Reference<uno::XInterface> xIntDim =
906                 ScUnoHelpFunctions::AnyToInterface( xDims->getByIndex(nDim) );
907             uno::Reference<beans::XPropertySet> xDimProp( xIntDim, uno::UNO_QUERY );
908             if ( xDimProp.is() )
909             {
910                 try
911                 {
912                     uno::Any aOrigAny = xDimProp->getPropertyValue(
913                                 rtl::OUString::createFromAscii(DP_PROP_ORIGINAL) );
914                     uno::Reference<uno::XInterface> xIntOrig;
915                     if ( (aOrigAny >>= xIntOrig) && xIntOrig.is() )
916                         bDuplicated = sal_True;
917                 }
918                 catch(uno::Exception&)
919                 {
920                 }
921             }
922         }
923     }
924     return bDuplicated;
925 }
926 
GetDimCount()927 long ScDPObject::GetDimCount()
928 {
929     long nRet = 0;
930     if ( xSource.is() )
931     {
932         try
933         {
934             uno::Reference<container::XNameAccess> xDimsName = xSource->getDimensions();
935             if ( xDimsName.is() )
936                 nRet = xDimsName->getElementNames().getLength();
937         }
938         catch(uno::Exception&)
939         {
940         }
941     }
942     return nRet;
943 }
944 
FillPageList(TypedScStrCollection & rStrings,long nField)945 void ScDPObject::FillPageList( TypedScStrCollection& rStrings, long nField )
946 {
947 	//!	merge members access with ToggleDetails?
948 
949 	//!	convert field index to dimension index?
950 
951 	DBG_ASSERT( xSource.is(), "no source" );
952 	if ( !xSource.is() ) return;
953 
954 	uno::Reference<container::XNamed> xDim;
955 	uno::Reference<container::XNameAccess> xDimsName = xSource->getDimensions();
956 	uno::Reference<container::XIndexAccess> xIntDims = new ScNameToIndexAccess( xDimsName );
957 	long nIntCount = xIntDims->getCount();
958 	if ( nField < nIntCount )
959 	{
960 		uno::Reference<uno::XInterface> xIntDim = ScUnoHelpFunctions::AnyToInterface(
961 									xIntDims->getByIndex(nField) );
962 		xDim = uno::Reference<container::XNamed>( xIntDim, uno::UNO_QUERY );
963 	}
964 	DBG_ASSERT( xDim.is(), "dimension not found" );
965 	if ( !xDim.is() ) return;
966 
967 	uno::Reference<beans::XPropertySet> xDimProp( xDim, uno::UNO_QUERY );
968 	long nHierarchy = ScUnoHelpFunctions::GetLongProperty( xDimProp,
969 							rtl::OUString::createFromAscii(DP_PROP_USEDHIERARCHY) );
970 	long nLevel = 0;
971 
972 	long nHierCount = 0;
973 	uno::Reference<container::XIndexAccess> xHiers;
974 	uno::Reference<sheet::XHierarchiesSupplier> xHierSupp( xDim, uno::UNO_QUERY );
975 	if ( xHierSupp.is() )
976 	{
977 		uno::Reference<container::XNameAccess> xHiersName = xHierSupp->getHierarchies();
978 		xHiers = new ScNameToIndexAccess( xHiersName );
979 		nHierCount = xHiers->getCount();
980 	}
981 	uno::Reference<uno::XInterface> xHier;
982 	if ( nHierarchy < nHierCount )
983 		xHier = ScUnoHelpFunctions::AnyToInterface( xHiers->getByIndex(nHierarchy) );
984 	DBG_ASSERT( xHier.is(), "hierarchy not found" );
985 	if ( !xHier.is() ) return;
986 
987 	long nLevCount = 0;
988 	uno::Reference<container::XIndexAccess> xLevels;
989 	uno::Reference<sheet::XLevelsSupplier> xLevSupp( xHier, uno::UNO_QUERY );
990 	if ( xLevSupp.is() )
991 	{
992 		uno::Reference<container::XNameAccess> xLevsName = xLevSupp->getLevels();
993 		xLevels = new ScNameToIndexAccess( xLevsName );
994 		nLevCount = xLevels->getCount();
995 	}
996 	uno::Reference<uno::XInterface> xLevel;
997 	if ( nLevel < nLevCount )
998 		xLevel = ScUnoHelpFunctions::AnyToInterface( xLevels->getByIndex(nLevel) );
999 	DBG_ASSERT( xLevel.is(), "level not found" );
1000 	if ( !xLevel.is() ) return;
1001 
1002 	uno::Reference<container::XNameAccess> xMembers;
1003 	uno::Reference<sheet::XMembersSupplier> xMbrSupp( xLevel, uno::UNO_QUERY );
1004 	if ( xMbrSupp.is() )
1005 		xMembers = xMbrSupp->getMembers();
1006 	DBG_ASSERT( xMembers.is(), "members not found" );
1007 	if ( !xMembers.is() ) return;
1008 
1009 	uno::Sequence<rtl::OUString> aNames = xMembers->getElementNames();
1010 	long nNameCount = aNames.getLength();
1011 	const rtl::OUString* pNameArr = aNames.getConstArray();
1012 	for (long nPos = 0; nPos < nNameCount; ++nPos)
1013 	{
1014         // Make sure to insert only visible members.
1015         Reference<XPropertySet> xPropSet(xMembers->getByName(pNameArr[nPos]), UNO_QUERY);
1016         sal_Bool bVisible = false;
1017         if (xPropSet.is())
1018         {
1019             Any any = xPropSet->getPropertyValue(OUString::createFromAscii(SC_UNO_ISVISIBL));
1020             any >>= bVisible;
1021         }
1022 
1023         if (bVisible)
1024         {
1025             // use the order from getElementNames
1026             TypedStrData* pData = new TypedStrData( pNameArr[nPos] );
1027             if ( !rStrings.AtInsert( rStrings.GetCount(), pData ) )
1028                 delete pData;
1029         }
1030     }
1031 
1032 	//	add "-all-" entry to the top (unsorted)
1033 	TypedStrData* pAllData = new TypedStrData( String( ScResId( SCSTR_ALL ) ) );	//! separate string? (also output)
1034 	if ( !rStrings.AtInsert( 0, pAllData ) )
1035 		delete pAllData;
1036 }
1037 
GetHeaderPositionData(const ScAddress & rPos,DataPilotTableHeaderData & rData)1038 void ScDPObject::GetHeaderPositionData(const ScAddress& rPos, DataPilotTableHeaderData& rData)
1039 {
1040     using namespace ::com::sun::star::sheet::DataPilotTablePositionType;
1041 
1042 	CreateOutput();				// create xSource and pOutput if not already done
1043 
1044     // Reset member values to invalid state.
1045     rData.Dimension = rData.Hierarchy = rData.Level = -1;
1046     rData.Flags = 0;
1047 
1048     DataPilotTablePositionData aPosData;
1049     pOutput->GetPositionData(rPos, aPosData);
1050     const sal_Int32 nPosType = aPosData.PositionType;
1051     if (nPosType == COLUMN_HEADER || nPosType == ROW_HEADER)
1052         aPosData.PositionData >>= rData;
1053 }
1054 
1055 // Returns sal_True on success and stores the result in rTarget
GetPivotData(ScDPGetPivotDataField & rTarget,const std::vector<ScDPGetPivotDataField> & rFilters)1056 sal_Bool ScDPObject::GetPivotData( ScDPGetPivotDataField& rTarget,
1057                                const std::vector< ScDPGetPivotDataField >& rFilters )
1058 {
1059     // #i117239# Exit with an error if called from creating the cache for this object
1060     // (don't create an empty pOutput object)
1061     if (mbCreatingTableData)
1062         return sal_False;
1063 
1064     CreateOutput();             // create xSource and pOutput if not already done
1065 
1066     return pOutput->GetPivotData( rTarget, rFilters );
1067 }
1068 
IsFilterButton(const ScAddress & rPos)1069 sal_Bool ScDPObject::IsFilterButton( const ScAddress& rPos )
1070 {
1071 	CreateOutput();				// create xSource and pOutput if not already done
1072 
1073 	return pOutput->IsFilterButton( rPos );
1074 }
1075 
GetHeaderDim(const ScAddress & rPos,sal_uInt16 & rOrient)1076 long ScDPObject::GetHeaderDim( const ScAddress& rPos, sal_uInt16& rOrient )
1077 {
1078 	CreateOutput();				// create xSource and pOutput if not already done
1079 
1080 	return pOutput->GetHeaderDim( rPos, rOrient );
1081 }
1082 
GetHeaderDrag(const ScAddress & rPos,sal_Bool bMouseLeft,sal_Bool bMouseTop,long nDragDim,Rectangle & rPosRect,sal_uInt16 & rOrient,long & rDimPos)1083 sal_Bool ScDPObject::GetHeaderDrag( const ScAddress& rPos, sal_Bool bMouseLeft, sal_Bool bMouseTop, long nDragDim,
1084 								Rectangle& rPosRect, sal_uInt16& rOrient, long& rDimPos )
1085 {
1086 	CreateOutput();				// create xSource and pOutput if not already done
1087 
1088 	return pOutput->GetHeaderDrag( rPos, bMouseLeft, bMouseTop, nDragDim, rPosRect, rOrient, rDimPos );
1089 }
1090 
GetMemberResultNames(ScStrCollection & rNames,long nDimension)1091 void ScDPObject::GetMemberResultNames( ScStrCollection& rNames, long nDimension )
1092 {
1093     CreateOutput();             // create xSource and pOutput if not already done
1094 
1095     pOutput->GetMemberResultNames( rNames, nDimension );    // used only with table data -> level not needed
1096 }
1097 
lcl_Dequote(const String & rSource,xub_StrLen nStartPos,xub_StrLen & rEndPos,String & rResult)1098 bool lcl_Dequote( const String& rSource, xub_StrLen nStartPos, xub_StrLen& rEndPos, String& rResult )
1099 {
1100     // nStartPos has to point to opening quote
1101 
1102     bool bRet = false;
1103     const sal_Unicode cQuote = '\'';
1104 
1105     if ( rSource.GetChar(nStartPos) == cQuote )
1106     {
1107         rtl::OUStringBuffer aBuffer;
1108         xub_StrLen nPos = nStartPos + 1;
1109         const xub_StrLen nLen = rSource.Len();
1110 
1111         while ( nPos < nLen )
1112         {
1113             const sal_Unicode cNext = rSource.GetChar(nPos);
1114             if ( cNext == cQuote )
1115             {
1116                 if ( nPos+1 < nLen && rSource.GetChar(nPos+1) == cQuote )
1117                 {
1118                     // double quote is used for an embedded quote
1119                     aBuffer.append( cNext );    // append one quote
1120                     ++nPos;                     // skip the next one
1121                 }
1122                 else
1123                 {
1124                     // end of quoted string
1125                     rResult = aBuffer.makeStringAndClear();
1126                     rEndPos = nPos + 1;         // behind closing quote
1127                     return true;
1128                 }
1129             }
1130             else
1131                 aBuffer.append( cNext );
1132 
1133             ++nPos;
1134         }
1135         // no closing quote before the end of the string -> error (bRet still false)
1136     }
1137 
1138     return bRet;
1139 }
1140 
1141 struct ScGetPivotDataFunctionEntry
1142 {
1143     const sal_Char*         pName;
1144     sheet::GeneralFunction  eFunc;
1145 };
1146 
lcl_ParseFunction(const String & rList,xub_StrLen nStartPos,xub_StrLen & rEndPos,sheet::GeneralFunction & rFunc)1147 bool lcl_ParseFunction( const String& rList, xub_StrLen nStartPos, xub_StrLen& rEndPos, sheet::GeneralFunction& rFunc )
1148 {
1149     static const ScGetPivotDataFunctionEntry aFunctions[] =
1150     {
1151         // our names
1152         { "Sum",        sheet::GeneralFunction_SUM       },
1153         { "Count",      sheet::GeneralFunction_COUNT     },
1154         { "Average",    sheet::GeneralFunction_AVERAGE   },
1155         { "Max",        sheet::GeneralFunction_MAX       },
1156         { "Min",        sheet::GeneralFunction_MIN       },
1157         { "Product",    sheet::GeneralFunction_PRODUCT   },
1158         { "CountNums",  sheet::GeneralFunction_COUNTNUMS },
1159         { "StDev",      sheet::GeneralFunction_STDEV     },
1160         { "StDevp",     sheet::GeneralFunction_STDEVP    },
1161         { "Var",        sheet::GeneralFunction_VAR       },
1162         { "VarP",       sheet::GeneralFunction_VARP      },
1163         // compatibility names
1164         { "Count Nums", sheet::GeneralFunction_COUNTNUMS },
1165         { "StdDev",     sheet::GeneralFunction_STDEV     },
1166         { "StdDevp",    sheet::GeneralFunction_STDEVP    }
1167 	};
1168 
1169     const xub_StrLen nListLen = rList.Len();
1170     while ( nStartPos < nListLen && rList.GetChar(nStartPos) == ' ' )
1171         ++nStartPos;
1172 
1173     bool bParsed = false;
1174     bool bFound = false;
1175     String aFuncStr;
1176     xub_StrLen nFuncEnd = 0;
1177     if ( nStartPos < nListLen && rList.GetChar(nStartPos) == '\'' )
1178         bParsed = lcl_Dequote( rList, nStartPos, nFuncEnd, aFuncStr );
1179     else
1180     {
1181         nFuncEnd = rList.Search( static_cast<sal_Unicode>(']'), nStartPos );
1182         if ( nFuncEnd != STRING_NOTFOUND )
1183         {
1184             aFuncStr = rList.Copy( nStartPos, nFuncEnd - nStartPos );
1185             bParsed = true;
1186         }
1187     }
1188 
1189     if ( bParsed )
1190     {
1191         aFuncStr.EraseLeadingAndTrailingChars( ' ' );
1192 
1193         const sal_Int32 nFuncCount = sizeof(aFunctions) / sizeof(aFunctions[0]);
1194         for ( sal_Int32 nFunc=0; nFunc<nFuncCount && !bFound; nFunc++ )
1195         {
1196             if ( aFuncStr.EqualsIgnoreCaseAscii( aFunctions[nFunc].pName ) )
1197             {
1198                 rFunc = aFunctions[nFunc].eFunc;
1199                 bFound = true;
1200 
1201                 while ( nFuncEnd < nListLen && rList.GetChar(nFuncEnd) == ' ' )
1202                     ++nFuncEnd;
1203                 rEndPos = nFuncEnd;
1204             }
1205         }
1206     }
1207 
1208     return bFound;
1209 }
1210 
lcl_IsAtStart(const String & rList,const String & rSearch,sal_Int32 & rMatched,bool bAllowBracket,sheet::GeneralFunction * pFunc)1211 bool lcl_IsAtStart( const String& rList, const String& rSearch, sal_Int32& rMatched,
1212                     bool bAllowBracket, sheet::GeneralFunction* pFunc )
1213 {
1214     sal_Int32 nMatchList = 0;
1215     sal_Int32 nMatchSearch = 0;
1216     sal_Unicode cFirst = rList.GetChar(0);
1217     if ( cFirst == '\'' || cFirst == '[' )
1218     {
1219         // quoted string or string in brackets must match completely
1220 
1221         String aDequoted;
1222         xub_StrLen nQuoteEnd = 0;
1223         bool bParsed = false;
1224 
1225         if ( cFirst == '\'' )
1226             bParsed = lcl_Dequote( rList, 0, nQuoteEnd, aDequoted );
1227         else if ( cFirst == '[' )
1228         {
1229             // skip spaces after the opening bracket
1230 
1231             xub_StrLen nStartPos = 1;
1232             const xub_StrLen nListLen = rList.Len();
1233             while ( nStartPos < nListLen && rList.GetChar(nStartPos) == ' ' )
1234                 ++nStartPos;
1235 
1236             if ( rList.GetChar(nStartPos) == '\'' )         // quoted within the brackets?
1237             {
1238                 if ( lcl_Dequote( rList, nStartPos, nQuoteEnd, aDequoted ) )
1239                 {
1240                     // after the quoted string, there must be the closing bracket, optionally preceded by spaces,
1241                     // and/or a function name
1242                     while ( nQuoteEnd < nListLen && rList.GetChar(nQuoteEnd) == ' ' )
1243                         ++nQuoteEnd;
1244 
1245                     // semicolon separates function name
1246                     if ( nQuoteEnd < nListLen && rList.GetChar(nQuoteEnd) == ';' && pFunc )
1247                     {
1248                         xub_StrLen nFuncEnd = 0;
1249                         if ( lcl_ParseFunction( rList, nQuoteEnd + 1, nFuncEnd, *pFunc ) )
1250                             nQuoteEnd = nFuncEnd;
1251                     }
1252                     if ( nQuoteEnd < nListLen && rList.GetChar(nQuoteEnd) == ']' )
1253                     {
1254                         ++nQuoteEnd;        // include the closing bracket for the matched length
1255                         bParsed = true;
1256                     }
1257                 }
1258             }
1259             else
1260             {
1261                 // implicit quoting to the closing bracket
1262 
1263                 xub_StrLen nClosePos = rList.Search( static_cast<sal_Unicode>(']'), nStartPos );
1264                 if ( nClosePos != STRING_NOTFOUND )
1265                 {
1266                     xub_StrLen nNameEnd = nClosePos;
1267                     xub_StrLen nSemiPos = rList.Search( static_cast<sal_Unicode>(';'), nStartPos );
1268                     if ( nSemiPos != STRING_NOTFOUND && nSemiPos < nClosePos && pFunc )
1269                     {
1270                         xub_StrLen nFuncEnd = 0;
1271                         if ( lcl_ParseFunction( rList, nSemiPos + 1, nFuncEnd, *pFunc ) )
1272                             nNameEnd = nSemiPos;
1273                     }
1274 
1275                     aDequoted = rList.Copy( nStartPos, nNameEnd - nStartPos );
1276                     aDequoted.EraseTrailingChars( ' ' );        // spaces before the closing bracket or semicolon
1277                     nQuoteEnd = nClosePos + 1;
1278                     bParsed = true;
1279                 }
1280             }
1281         }
1282 
1283         if ( bParsed && ScGlobal::GetpTransliteration()->isEqual( aDequoted, rSearch ) )
1284         {
1285             nMatchList = nQuoteEnd;             // match count in the list string, including quotes
1286             nMatchSearch = rSearch.Len();
1287         }
1288     }
1289     else
1290     {
1291         // otherwise look for search string at the start of rList
1292         ScGlobal::GetpTransliteration()->equals( rList, 0, rList.Len(), nMatchList,
1293                                             rSearch, 0, rSearch.Len(), nMatchSearch );
1294     }
1295 
1296     if ( nMatchSearch == rSearch.Len() )
1297     {
1298         // search string is at start of rList - look for following space or end of string
1299 
1300         bool bValid = false;
1301         if ( sal::static_int_cast<xub_StrLen>(nMatchList) >= rList.Len() )
1302             bValid = true;
1303         else
1304         {
1305             sal_Unicode cNext = rList.GetChar(sal::static_int_cast<xub_StrLen>(nMatchList));
1306             if ( cNext == ' ' || ( bAllowBracket && cNext == '[' ) )
1307                 bValid = true;
1308         }
1309 
1310         if ( bValid )
1311         {
1312             rMatched = nMatchList;
1313             return true;
1314         }
1315     }
1316 
1317     return false;
1318 }
1319 
ParseFilters(ScDPGetPivotDataField & rTarget,std::vector<ScDPGetPivotDataField> & rFilters,const String & rFilterList)1320 sal_Bool ScDPObject::ParseFilters( ScDPGetPivotDataField& rTarget,
1321                                std::vector< ScDPGetPivotDataField >& rFilters,
1322                                const String& rFilterList )
1323 {
1324     // parse the string rFilterList into parameters for GetPivotData
1325 
1326     CreateObjects();            // create xSource if not already done
1327 
1328     std::vector<String> aDataNames;     // data fields (source name)
1329     std::vector<String> aGivenNames;    // data fields (compound name)
1330     std::vector<String> aFieldNames;    // column/row/data fields
1331     std::vector< uno::Sequence<rtl::OUString> > aFieldValues;
1332 
1333     //
1334     // get all the field and item names
1335     //
1336 
1337     uno::Reference<container::XNameAccess> xDimsName = xSource->getDimensions();
1338     uno::Reference<container::XIndexAccess> xIntDims = new ScNameToIndexAccess( xDimsName );
1339     sal_Int32 nDimCount = xIntDims->getCount();
1340     for ( sal_Int32 nDim = 0; nDim<nDimCount; nDim++ )
1341     {
1342         uno::Reference<uno::XInterface> xIntDim = ScUnoHelpFunctions::AnyToInterface( xIntDims->getByIndex(nDim) );
1343         uno::Reference<container::XNamed> xDim( xIntDim, uno::UNO_QUERY );
1344         uno::Reference<beans::XPropertySet> xDimProp( xDim, uno::UNO_QUERY );
1345         uno::Reference<sheet::XHierarchiesSupplier> xDimSupp( xDim, uno::UNO_QUERY );
1346         sal_Bool bDataLayout = ScUnoHelpFunctions::GetBoolProperty( xDimProp,
1347                             rtl::OUString::createFromAscii(DP_PROP_ISDATALAYOUT) );
1348         sal_Int32 nOrient = ScUnoHelpFunctions::GetEnumProperty(
1349                             xDimProp, rtl::OUString::createFromAscii(DP_PROP_ORIENTATION),
1350                             sheet::DataPilotFieldOrientation_HIDDEN );
1351         if ( !bDataLayout )
1352         {
1353             if ( nOrient == sheet::DataPilotFieldOrientation_DATA )
1354             {
1355                 String aSourceName;
1356                 String aGivenName;
1357                 ScDPOutput::GetDataDimensionNames( aSourceName, aGivenName, xIntDim );
1358                 aDataNames.push_back( aSourceName );
1359                 aGivenNames.push_back( aGivenName );
1360             }
1361             else if ( nOrient != sheet::DataPilotFieldOrientation_HIDDEN )
1362             {
1363                 // get level names, as in ScDPOutput
1364 
1365                 uno::Reference<container::XIndexAccess> xHiers = new ScNameToIndexAccess( xDimSupp->getHierarchies() );
1366                 sal_Int32 nHierarchy = ScUnoHelpFunctions::GetLongProperty( xDimProp,
1367                                                     rtl::OUString::createFromAscii(DP_PROP_USEDHIERARCHY) );
1368                 if ( nHierarchy >= xHiers->getCount() )
1369                     nHierarchy = 0;
1370 
1371                 uno::Reference<uno::XInterface> xHier = ScUnoHelpFunctions::AnyToInterface(
1372                                                     xHiers->getByIndex(nHierarchy) );
1373                 uno::Reference<sheet::XLevelsSupplier> xHierSupp( xHier, uno::UNO_QUERY );
1374                 if ( xHierSupp.is() )
1375                 {
1376                     uno::Reference<container::XIndexAccess> xLevels = new ScNameToIndexAccess( xHierSupp->getLevels() );
1377                     sal_Int32 nLevCount = xLevels->getCount();
1378                     for (sal_Int32 nLev=0; nLev<nLevCount; nLev++)
1379                     {
1380                         uno::Reference<uno::XInterface> xLevel = ScUnoHelpFunctions::AnyToInterface(
1381                                                             xLevels->getByIndex(nLev) );
1382                         uno::Reference<container::XNamed> xLevNam( xLevel, uno::UNO_QUERY );
1383                         uno::Reference<sheet::XMembersSupplier> xLevSupp( xLevel, uno::UNO_QUERY );
1384                         if ( xLevNam.is() && xLevSupp.is() )
1385                         {
1386                             uno::Reference<container::XNameAccess> xMembers = xLevSupp->getMembers();
1387 
1388                             String aFieldName( xLevNam->getName() );
1389                             uno::Sequence<rtl::OUString> aMemberNames( xMembers->getElementNames() );
1390 
1391                             aFieldNames.push_back( aFieldName );
1392                             aFieldValues.push_back( aMemberNames );
1393                         }
1394                     }
1395                 }
1396             }
1397         }
1398     }
1399 
1400     //
1401     // compare and build filters
1402     //
1403 
1404     SCSIZE nDataFields = aDataNames.size();
1405     SCSIZE nFieldCount = aFieldNames.size();
1406     DBG_ASSERT( aGivenNames.size() == nDataFields && aFieldValues.size() == nFieldCount, "wrong count" );
1407 
1408     bool bError = false;
1409     bool bHasData = false;
1410     String aRemaining( rFilterList );
1411     aRemaining.EraseLeadingAndTrailingChars( ' ' );
1412     while ( aRemaining.Len() && !bError )
1413     {
1414         bool bUsed = false;
1415 
1416         // look for data field name
1417 
1418         for ( SCSIZE nDataPos=0; nDataPos<nDataFields && !bUsed; nDataPos++ )
1419         {
1420             String aFound;
1421             sal_Int32 nMatched = 0;
1422             if ( lcl_IsAtStart( aRemaining, aDataNames[nDataPos], nMatched, false, NULL ) )
1423                 aFound = aDataNames[nDataPos];
1424             else if ( lcl_IsAtStart( aRemaining, aGivenNames[nDataPos], nMatched, false, NULL ) )
1425                 aFound = aGivenNames[nDataPos];
1426 
1427             if ( aFound.Len() )
1428             {
1429                 rTarget.maFieldName = aFound;
1430                 aRemaining.Erase( 0, sal::static_int_cast<xub_StrLen>(nMatched) );
1431                 bHasData = true;
1432                 bUsed = true;
1433             }
1434         }
1435 
1436         // look for field name
1437 
1438         String aSpecField;
1439         bool bHasFieldName = false;
1440         if ( !bUsed )
1441         {
1442             sal_Int32 nMatched = 0;
1443             for ( SCSIZE nField=0; nField<nFieldCount && !bHasFieldName; nField++ )
1444             {
1445                 if ( lcl_IsAtStart( aRemaining, aFieldNames[nField], nMatched, true, NULL ) )
1446                 {
1447                     aSpecField = aFieldNames[nField];
1448                     aRemaining.Erase( 0, sal::static_int_cast<xub_StrLen>(nMatched) );
1449                     aRemaining.EraseLeadingChars( ' ' );
1450 
1451                     // field name has to be followed by item name in brackets
1452                     if ( aRemaining.GetChar(0) == '[' )
1453                     {
1454                         bHasFieldName = true;
1455                         // bUsed remains false - still need the item
1456                     }
1457                     else
1458                     {
1459                         bUsed = true;
1460                         bError = true;
1461                     }
1462                 }
1463             }
1464         }
1465 
1466         // look for field item
1467 
1468         if ( !bUsed )
1469         {
1470             bool bItemFound = false;
1471             sal_Int32 nMatched = 0;
1472             String aFoundName;
1473             String aFoundValue;
1474             sheet::GeneralFunction eFunc = sheet::GeneralFunction_NONE;
1475             sheet::GeneralFunction eFoundFunc = sheet::GeneralFunction_NONE;
1476 
1477             for ( SCSIZE nField=0; nField<nFieldCount; nField++ )
1478             {
1479                 // If a field name is given, look in that field only, otherwise in all fields.
1480                 // aSpecField is initialized from aFieldNames array, so exact comparison can be used.
1481                 if ( !bHasFieldName || aFieldNames[nField] == aSpecField )
1482                 {
1483                     const uno::Sequence<rtl::OUString>& rItems = aFieldValues[nField];
1484                     sal_Int32 nItemCount = rItems.getLength();
1485                     const rtl::OUString* pItemArr = rItems.getConstArray();
1486                     for ( sal_Int32 nItem=0; nItem<nItemCount; nItem++ )
1487                     {
1488                         if ( lcl_IsAtStart( aRemaining, pItemArr[nItem], nMatched, false, &eFunc ) )
1489                         {
1490                             if ( bItemFound )
1491                                 bError = true;      // duplicate (also across fields)
1492                             else
1493                             {
1494                                 aFoundName = aFieldNames[nField];
1495                                 aFoundValue = pItemArr[nItem];
1496                                 eFoundFunc = eFunc;
1497                                 bItemFound = true;
1498                                 bUsed = true;
1499                             }
1500                         }
1501                     }
1502                 }
1503             }
1504 
1505             if ( bItemFound && !bError )
1506             {
1507                 ScDPGetPivotDataField aField;
1508                 aField.maFieldName = aFoundName;
1509                 aField.meFunction = eFoundFunc;
1510                 aField.mbValIsStr = true;
1511                 aField.maValStr = aFoundValue;
1512                 aField.mnValNum = 0.0;
1513                 rFilters.push_back( aField );
1514 
1515                 aRemaining.Erase( 0, sal::static_int_cast<xub_StrLen>(nMatched) );
1516             }
1517         }
1518 
1519         if ( !bUsed )
1520             bError = true;
1521 
1522         aRemaining.EraseLeadingChars( ' ' );        // remove any number of spaces between entries
1523     }
1524 
1525     if ( !bError && !bHasData && aDataNames.size() == 1 )
1526     {
1527         // if there's only one data field, its name need not be specified
1528         rTarget.maFieldName = aDataNames[0];
1529         bHasData = true;
1530     }
1531 
1532     return bHasData && !bError;
1533 }
1534 
ToggleDetails(const DataPilotTableHeaderData & rElemDesc,ScDPObject * pDestObj)1535 void ScDPObject::ToggleDetails(const DataPilotTableHeaderData& rElemDesc, ScDPObject* pDestObj)
1536 {
1537 	CreateObjects();			// create xSource if not already done
1538 
1539 	//	find dimension name
1540 
1541 	uno::Reference<container::XNamed> xDim;
1542 	uno::Reference<container::XNameAccess> xDimsName = xSource->getDimensions();
1543 	uno::Reference<container::XIndexAccess> xIntDims = new ScNameToIndexAccess( xDimsName );
1544 	long nIntCount = xIntDims->getCount();
1545 	if ( rElemDesc.Dimension < nIntCount )
1546 	{
1547 		uno::Reference<uno::XInterface> xIntDim = ScUnoHelpFunctions::AnyToInterface(
1548 									xIntDims->getByIndex(rElemDesc.Dimension) );
1549 		xDim = uno::Reference<container::XNamed>( xIntDim, uno::UNO_QUERY );
1550 	}
1551 	DBG_ASSERT( xDim.is(), "dimension not found" );
1552 	if ( !xDim.is() ) return;
1553 	String aDimName = xDim->getName();
1554 
1555 	uno::Reference<beans::XPropertySet> xDimProp( xDim, uno::UNO_QUERY );
1556 	sal_Bool bDataLayout = ScUnoHelpFunctions::GetBoolProperty( xDimProp,
1557 						rtl::OUString::createFromAscii(DP_PROP_ISDATALAYOUT) );
1558 	if (bDataLayout)
1559 	{
1560 		//	the elements of the data layout dimension can't be found by their names
1561 		//	-> don't change anything
1562 		return;
1563 	}
1564 
1565 	//	query old state
1566 
1567 	long nHierCount = 0;
1568 	uno::Reference<container::XIndexAccess> xHiers;
1569 	uno::Reference<sheet::XHierarchiesSupplier> xHierSupp( xDim, uno::UNO_QUERY );
1570 	if ( xHierSupp.is() )
1571 	{
1572 		uno::Reference<container::XNameAccess> xHiersName = xHierSupp->getHierarchies();
1573 		xHiers = new ScNameToIndexAccess( xHiersName );
1574 		nHierCount = xHiers->getCount();
1575 	}
1576 	uno::Reference<uno::XInterface> xHier;
1577 	if ( rElemDesc.Hierarchy < nHierCount )
1578 		xHier = ScUnoHelpFunctions::AnyToInterface( xHiers->getByIndex(rElemDesc.Hierarchy) );
1579 	DBG_ASSERT( xHier.is(), "hierarchy not found" );
1580 	if ( !xHier.is() ) return;
1581 
1582 	long nLevCount = 0;
1583 	uno::Reference<container::XIndexAccess> xLevels;
1584 	uno::Reference<sheet::XLevelsSupplier> xLevSupp( xHier, uno::UNO_QUERY );
1585 	if ( xLevSupp.is() )
1586 	{
1587 		uno::Reference<container::XNameAccess> xLevsName = xLevSupp->getLevels();
1588 		xLevels = new ScNameToIndexAccess( xLevsName );
1589 		nLevCount = xLevels->getCount();
1590 	}
1591 	uno::Reference<uno::XInterface> xLevel;
1592 	if ( rElemDesc.Level < nLevCount )
1593 		xLevel = ScUnoHelpFunctions::AnyToInterface( xLevels->getByIndex(rElemDesc.Level) );
1594 	DBG_ASSERT( xLevel.is(), "level not found" );
1595 	if ( !xLevel.is() ) return;
1596 
1597 	uno::Reference<container::XNameAccess> xMembers;
1598 	uno::Reference<sheet::XMembersSupplier> xMbrSupp( xLevel, uno::UNO_QUERY );
1599 	if ( xMbrSupp.is() )
1600 		xMembers = xMbrSupp->getMembers();
1601 
1602 	sal_Bool bFound = sal_False;
1603 	sal_Bool bShowDetails = sal_True;
1604 
1605 	if ( xMembers.is() )
1606 	{
1607 		if ( xMembers->hasByName(rElemDesc.MemberName) )
1608 		{
1609 			uno::Reference<uno::XInterface> xMemberInt = ScUnoHelpFunctions::AnyToInterface(
1610 											xMembers->getByName(rElemDesc.MemberName) );
1611 			uno::Reference<beans::XPropertySet> xMbrProp( xMemberInt, uno::UNO_QUERY );
1612 			if ( xMbrProp.is() )
1613 			{
1614 				bShowDetails = ScUnoHelpFunctions::GetBoolProperty( xMbrProp,
1615 									rtl::OUString::createFromAscii(DP_PROP_SHOWDETAILS) );
1616 				//! don't set bFound if property is unknown?
1617 				bFound = sal_True;
1618 			}
1619 		}
1620 	}
1621 
1622 	DBG_ASSERT( bFound, "member not found" );
1623 
1624 	//!	use Hierarchy and Level in SaveData !!!!
1625 
1626 	//	modify pDestObj if set, this object otherwise
1627 	ScDPSaveData* pModifyData = pDestObj ? ( pDestObj->pSaveData ) : pSaveData;
1628 	DBG_ASSERT( pModifyData, "no data?" );
1629 	if ( pModifyData )
1630 	{
1631         const String aName = rElemDesc.MemberName;
1632 		pModifyData->GetDimensionByName(aDimName)->
1633 			GetMemberByName(aName)->SetShowDetails( !bShowDetails );	// toggle
1634 
1635 		if ( pDestObj )
1636 			pDestObj->InvalidateData();		// re-init source from SaveData
1637 		else
1638 			InvalidateData();				// re-init source from SaveData
1639 	}
1640 }
1641 
lcl_FindName(const rtl::OUString & rString,const uno::Reference<container::XNameAccess> & xCollection)1642 long lcl_FindName( const rtl::OUString& rString, const uno::Reference<container::XNameAccess>& xCollection )
1643 {
1644 	if ( xCollection.is() )
1645 	{
1646 		uno::Sequence<rtl::OUString> aSeq = xCollection->getElementNames();
1647 		long nCount = aSeq.getLength();
1648 		const rtl::OUString* pArr = aSeq.getConstArray();
1649 		for (long nPos=0; nPos<nCount; nPos++)
1650 			if ( pArr[nPos] == rString )
1651 				return nPos;
1652 	}
1653 	return -1;		// not found
1654 }
1655 
lcl_FirstSubTotal(const uno::Reference<beans::XPropertySet> & xDimProp)1656 sal_uInt16 lcl_FirstSubTotal( const uno::Reference<beans::XPropertySet>& xDimProp )		// PIVOT_FUNC mask
1657 {
1658 	uno::Reference<sheet::XHierarchiesSupplier> xDimSupp( xDimProp, uno::UNO_QUERY );
1659 	if ( xDimProp.is() && xDimSupp.is() )
1660 	{
1661 		uno::Reference<container::XIndexAccess> xHiers = new ScNameToIndexAccess( xDimSupp->getHierarchies() );
1662 		long nHierarchy = ScUnoHelpFunctions::GetLongProperty( xDimProp,
1663 								rtl::OUString::createFromAscii(DP_PROP_USEDHIERARCHY) );
1664 		if ( nHierarchy >= xHiers->getCount() )
1665 			nHierarchy = 0;
1666 
1667 		uno::Reference<uno::XInterface> xHier = ScUnoHelpFunctions::AnyToInterface(
1668 									xHiers->getByIndex(nHierarchy) );
1669 		uno::Reference<sheet::XLevelsSupplier> xHierSupp( xHier, uno::UNO_QUERY );
1670 		if ( xHierSupp.is() )
1671 		{
1672 			uno::Reference<container::XIndexAccess> xLevels = new ScNameToIndexAccess( xHierSupp->getLevels() );
1673 			uno::Reference<uno::XInterface> xLevel =
1674 				ScUnoHelpFunctions::AnyToInterface( xLevels->getByIndex( 0 ) );
1675 			uno::Reference<beans::XPropertySet> xLevProp( xLevel, uno::UNO_QUERY );
1676 			if ( xLevProp.is() )
1677 			{
1678 				uno::Any aSubAny;
1679 				try
1680 				{
1681 					aSubAny = xLevProp->getPropertyValue(
1682 							rtl::OUString::createFromAscii(DP_PROP_SUBTOTALS) );
1683 				}
1684 				catch(uno::Exception&)
1685 				{
1686 				}
1687 				uno::Sequence<sheet::GeneralFunction> aSeq;
1688 				if ( aSubAny >>= aSeq )
1689 				{
1690 					sal_uInt16 nMask = 0;
1691 					const sheet::GeneralFunction* pArray = aSeq.getConstArray();
1692 					long nCount = aSeq.getLength();
1693 					for (long i=0; i<nCount; i++)
1694 						nMask |= ScDataPilotConversion::FunctionBit(pArray[i]);
1695 					return nMask;
1696 				}
1697 			}
1698 		}
1699 	}
1700 
1701 	DBG_ERROR("FirstSubTotal: NULL");
1702 	return 0;
1703 }
1704 
lcl_CountBits(sal_uInt16 nBits)1705 sal_uInt16 lcl_CountBits( sal_uInt16 nBits )
1706 {
1707 	if (!nBits) return 0;
1708 
1709 	sal_uInt16 nCount = 0;
1710 	sal_uInt16 nMask = 1;
1711 	for (sal_uInt16 i=0; i<16; i++)
1712 	{
1713 		if ( nBits & nMask )
1714 			++nCount;
1715 		nMask <<= 1;
1716 	}
1717 	return nCount;
1718 }
1719 
lcl_FillOldFields(ScPivotFieldVector & rFields,const uno::Reference<sheet::XDimensionsSupplier> & xSource,sal_uInt16 nOrient,SCCOL nColAdd,bool bAddData)1720 void lcl_FillOldFields( ScPivotFieldVector& rFields,
1721 							const uno::Reference<sheet::XDimensionsSupplier>& xSource,
1722 							sal_uInt16 nOrient, SCCOL nColAdd, bool bAddData )
1723 {
1724 	bool bDataFound = false;
1725 	rFields.clear();
1726 
1727 	//!	merge multiple occurrences (data field with different functions)
1728 	//!	force data field in one dimension
1729 
1730     std::vector< long > aPos;
1731 
1732 	uno::Reference<container::XNameAccess> xDimsName = xSource->getDimensions();
1733 	uno::Reference<container::XIndexAccess> xDims = new ScNameToIndexAccess( xDimsName );
1734 	long nDimCount = xDims->getCount();
1735     for (long nDim=0; nDim < nDimCount; nDim++)
1736 	{
1737 		uno::Reference<uno::XInterface> xIntDim =
1738 			ScUnoHelpFunctions::AnyToInterface( xDims->getByIndex(nDim) );
1739 		uno::Reference<beans::XPropertySet> xDimProp( xIntDim, uno::UNO_QUERY );
1740 		long nDimOrient = ScUnoHelpFunctions::GetEnumProperty(
1741 							xDimProp, rtl::OUString::createFromAscii(DP_PROP_ORIENTATION),
1742 							sheet::DataPilotFieldOrientation_HIDDEN );
1743 		if ( xDimProp.is() && nDimOrient == nOrient )
1744 		{
1745 			sal_uInt16 nMask = 0;
1746 			if ( nOrient == sheet::DataPilotFieldOrientation_DATA )
1747 			{
1748 				sheet::GeneralFunction eFunc = (sheet::GeneralFunction)ScUnoHelpFunctions::GetEnumProperty(
1749 											xDimProp, rtl::OUString::createFromAscii(DP_PROP_FUNCTION),
1750 											sheet::GeneralFunction_NONE );
1751 				if ( eFunc == sheet::GeneralFunction_AUTO )
1752 				{
1753 					//!	test for numeric data
1754 					eFunc = sheet::GeneralFunction_SUM;
1755 				}
1756 				nMask = ScDataPilotConversion::FunctionBit(eFunc);
1757 			}
1758 			else
1759 				nMask = lcl_FirstSubTotal( xDimProp );		// from first hierarchy
1760 
1761 			sal_Bool bDataLayout = ScUnoHelpFunctions::GetBoolProperty( xDimProp,
1762 									rtl::OUString::createFromAscii(DP_PROP_ISDATALAYOUT) );
1763 			uno::Any aOrigAny;
1764 			try
1765 			{
1766 				aOrigAny = xDimProp->getPropertyValue(
1767 								rtl::OUString::createFromAscii(DP_PROP_ORIGINAL) );
1768 			}
1769 			catch(uno::Exception&)
1770 			{
1771 			}
1772 
1773 			long nDupSource = -1;
1774 			uno::Reference<uno::XInterface> xIntOrig = ScUnoHelpFunctions::AnyToInterface( aOrigAny );
1775 			if ( xIntOrig.is() )
1776 			{
1777 				uno::Reference<container::XNamed> xNameOrig( xIntOrig, uno::UNO_QUERY );
1778 				if ( xNameOrig.is() )
1779 					nDupSource = lcl_FindName( xNameOrig->getName(), xDimsName );
1780 			}
1781 
1782 			bool bDupUsed = false;
1783 			if ( nDupSource >= 0 )
1784 			{
1785 				//	add function bit to previous entry
1786 
1787 				SCsCOL nCompCol;
1788 				if ( bDataLayout )
1789 					nCompCol = PIVOT_DATA_FIELD;
1790 				else
1791 					nCompCol = static_cast<SCsCOL>(nDupSource)+nColAdd;		//! seek source column from name
1792 
1793 				for (ScPivotFieldVector::iterator aIt = rFields.begin(), aEnd = rFields.end(); (aIt != aEnd) && !bDupUsed; ++aIt)
1794 					if ( aIt->nCol == nCompCol )
1795 					{
1796 						//	add to previous column only if new bits aren't already set there
1797 						if ( ( aIt->nFuncMask & nMask ) == 0 )
1798 						{
1799 							aIt->nFuncMask |= nMask;
1800 							aIt->nFuncCount = lcl_CountBits( aIt->nFuncMask );
1801 							bDupUsed = true;
1802 						}
1803 					}
1804 			}
1805 
1806 			if ( !bDupUsed )		// also for duplicated dim if original has different orientation
1807 			{
1808                 rFields.resize( rFields.size() + 1 );
1809                 ScPivotField& rField = rFields.back();
1810 
1811 				if ( bDataLayout )
1812 				{
1813 					rField.nCol = PIVOT_DATA_FIELD;
1814 					bDataFound = true;
1815 				}
1816 				else if ( nDupSource >= 0 )		// if source was not found (different orientation)
1817 					rField.nCol = static_cast<SCsCOL>(nDupSource)+nColAdd;		//! seek from name
1818 				else
1819 					rField.nCol = static_cast<SCsCOL>(nDim)+nColAdd;	//! seek source column from name
1820 
1821 				rField.nFuncMask = nMask;
1822 				rField.nFuncCount = lcl_CountBits( nMask );
1823 
1824                 aPos.push_back( ScUnoHelpFunctions::GetLongProperty( xDimProp,
1825 									rtl::OUString::createFromAscii(DP_PROP_POSITION) ) );
1826 
1827                 try
1828                 {
1829                     if( nOrient == sheet::DataPilotFieldOrientation_DATA )
1830                         xDimProp->getPropertyValue( rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( SC_UNO_REFVALUE ) ) )
1831                             >>= rFields.back().maFieldRef;
1832                 }
1833                 catch( uno::Exception& )
1834                 {
1835                 }
1836 			}
1837 		}
1838 	}
1839 
1840 	//	sort by getPosition() value
1841     size_t nSize = aPos.size();
1842 	for (size_t i=0; i+1<nSize; i++)
1843 	{
1844 		for (size_t j=0; j+i+1<nSize; j++)
1845             if ( aPos[j+1] < aPos[j] )
1846 			{
1847                 std::swap( aPos[j], aPos[j+1] );
1848                 std::swap( rFields[j], rFields[j+1] );
1849 			}
1850 	}
1851 
1852 	if ( bAddData && !bDataFound )
1853 	{
1854         rFields.resize( rFields.size() + 1 );
1855         ScPivotField& rField = rFields.back();
1856 		rField.nCol = PIVOT_DATA_FIELD;
1857 		rField.nFuncMask = 0;
1858 		rField.nFuncCount = 0;
1859 	}
1860 }
1861 
FillOldParam(ScPivotParam & rParam) const1862 sal_Bool ScDPObject::FillOldParam(ScPivotParam& rParam) const
1863 {
1864 	((ScDPObject*)this)->CreateObjects();		// xSource is needed for field numbers
1865 
1866 	rParam.nCol = aOutRange.aStart.Col();
1867 	rParam.nRow = aOutRange.aStart.Row();
1868 	rParam.nTab = aOutRange.aStart.Tab();
1869 	// ppLabelArr / nLabels is not changed
1870 
1871 	SCCOL nColAdd = 0;
1872 	bool bAddData = ( lcl_GetDataGetOrientation( xSource ) == sheet::DataPilotFieldOrientation_HIDDEN );
1873     lcl_FillOldFields( rParam.maPageArr, xSource, sheet::DataPilotFieldOrientation_PAGE,   nColAdd, false );
1874 	lcl_FillOldFields( rParam.maColArr,  xSource, sheet::DataPilotFieldOrientation_COLUMN, nColAdd, bAddData );
1875 	lcl_FillOldFields( rParam.maRowArr,  xSource, sheet::DataPilotFieldOrientation_ROW,    nColAdd, false );
1876 	lcl_FillOldFields( rParam.maDataArr, xSource, sheet::DataPilotFieldOrientation_DATA,   nColAdd, false );
1877 
1878 	uno::Reference<beans::XPropertySet> xProp( xSource, uno::UNO_QUERY );
1879 	if (xProp.is())
1880 	{
1881 		try
1882 		{
1883 			rParam.bMakeTotalCol = ScUnoHelpFunctions::GetBoolProperty( xProp,
1884 						rtl::OUString::createFromAscii(DP_PROP_COLUMNGRAND), sal_True );
1885 			rParam.bMakeTotalRow = ScUnoHelpFunctions::GetBoolProperty( xProp,
1886 						rtl::OUString::createFromAscii(DP_PROP_ROWGRAND), sal_True );
1887 
1888 			// following properties may be missing for external sources
1889 			rParam.bIgnoreEmptyRows = ScUnoHelpFunctions::GetBoolProperty( xProp,
1890 						rtl::OUString::createFromAscii(DP_PROP_IGNOREEMPTY) );
1891 			rParam.bDetectCategories = ScUnoHelpFunctions::GetBoolProperty( xProp,
1892 						rtl::OUString::createFromAscii(DP_PROP_REPEATIFEMPTY) );
1893 		}
1894 		catch(uno::Exception&)
1895 		{
1896 			// no error
1897 		}
1898 	}
1899 	return sal_True;
1900 }
1901 
lcl_FillLabelData(ScDPLabelData & rData,const uno::Reference<beans::XPropertySet> & xDimProp)1902 void lcl_FillLabelData( ScDPLabelData& rData, const uno::Reference< beans::XPropertySet >& xDimProp )
1903 {
1904 	uno::Reference<sheet::XHierarchiesSupplier> xDimSupp( xDimProp, uno::UNO_QUERY );
1905 	if ( xDimProp.is() && xDimSupp.is() )
1906 	{
1907 		uno::Reference<container::XIndexAccess> xHiers = new ScNameToIndexAccess( xDimSupp->getHierarchies() );
1908 		long nHierarchy = ScUnoHelpFunctions::GetLongProperty( xDimProp,
1909 								rtl::OUString::createFromAscii(DP_PROP_USEDHIERARCHY) );
1910 		if ( nHierarchy >= xHiers->getCount() )
1911 			nHierarchy = 0;
1912         rData.mnUsedHier = nHierarchy;
1913 
1914 		uno::Reference<uno::XInterface> xHier = ScUnoHelpFunctions::AnyToInterface(
1915 									xHiers->getByIndex(nHierarchy) );
1916 
1917 		uno::Reference<sheet::XLevelsSupplier> xHierSupp( xHier, uno::UNO_QUERY );
1918 		if ( xHierSupp.is() )
1919 		{
1920 			uno::Reference<container::XIndexAccess> xLevels = new ScNameToIndexAccess( xHierSupp->getLevels() );
1921 			uno::Reference<uno::XInterface> xLevel =
1922 				ScUnoHelpFunctions::AnyToInterface( xLevels->getByIndex( 0 ) );
1923 			uno::Reference<beans::XPropertySet> xLevProp( xLevel, uno::UNO_QUERY );
1924 			if ( xLevProp.is() )
1925             {
1926                 rData.mbShowAll = ScUnoHelpFunctions::GetBoolProperty( xLevProp,
1927 									rtl::OUString::createFromAscii(DP_PROP_SHOWEMPTY) );
1928 
1929                 try
1930                 {
1931                     xLevProp->getPropertyValue( rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( SC_UNO_SORTING ) ) )
1932                         >>= rData.maSortInfo;
1933                     xLevProp->getPropertyValue( rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( SC_UNO_LAYOUT ) ) )
1934                         >>= rData.maLayoutInfo;
1935                     xLevProp->getPropertyValue( rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( SC_UNO_AUTOSHOW ) ) )
1936                         >>= rData.maShowInfo;
1937                 }
1938                 catch(uno::Exception&)
1939                 {
1940                 }
1941             }
1942 		}
1943 	}
1944 }
1945 
FillLabelData(ScPivotParam & rParam)1946 sal_Bool ScDPObject::FillLabelData(ScPivotParam& rParam)
1947 {
1948     rParam.maLabelArray.clear();
1949 
1950 	((ScDPObject*)this)->CreateObjects();
1951 
1952 	uno::Reference<container::XNameAccess> xDimsName = xSource->getDimensions();
1953 	uno::Reference<container::XIndexAccess> xDims = new ScNameToIndexAccess( xDimsName );
1954 	long nDimCount = xDims->getCount();
1955 	if ( nDimCount > MAX_LABELS )
1956 		nDimCount = MAX_LABELS;
1957 	if (!nDimCount)
1958 		return sal_False;
1959 
1960 	for (long nDim=0; nDim < nDimCount; nDim++)
1961 	{
1962 		String aFieldName;
1963 		uno::Reference<uno::XInterface> xIntDim =
1964 			ScUnoHelpFunctions::AnyToInterface( xDims->getByIndex(nDim) );
1965 		uno::Reference<container::XNamed> xDimName( xIntDim, uno::UNO_QUERY );
1966 		uno::Reference<beans::XPropertySet> xDimProp( xIntDim, uno::UNO_QUERY );
1967 
1968 		if ( xDimName.is() && xDimProp.is() )
1969 		{
1970 			sal_Bool bDuplicated = sal_False;
1971 			sal_Bool bData = ScUnoHelpFunctions::GetBoolProperty( xDimProp,
1972 							rtl::OUString::createFromAscii(DP_PROP_ISDATALAYOUT) );
1973 			//!	error checking -- is "IsDataLayoutDimension" property required??
1974 
1975 			try
1976 			{
1977 				aFieldName = String( xDimName->getName() );
1978 
1979 				uno::Any aOrigAny = xDimProp->getPropertyValue(
1980 							rtl::OUString::createFromAscii(DP_PROP_ORIGINAL) );
1981 				uno::Reference<uno::XInterface> xIntOrig;
1982 				if ( (aOrigAny >>= xIntOrig) && xIntOrig.is() )
1983 					bDuplicated = sal_True;
1984 			}
1985 			catch(uno::Exception&)
1986 			{
1987 			}
1988 
1989             OUString aLayoutName = ScUnoHelpFunctions::GetStringProperty(
1990                 xDimProp, OUString::createFromAscii(SC_UNO_LAYOUTNAME), OUString());
1991 
1992 			if ( aFieldName.Len() && !bData && !bDuplicated )
1993 			{
1994                 SCsCOL nCol = static_cast< SCsCOL >( nDim );           //! ???
1995                 bool bIsValue = true;                               //! check
1996 
1997                 ScDPLabelData aNewLabel(aFieldName, nCol, bIsValue);
1998                 aNewLabel.maLayoutName = aLayoutName;
1999                 GetHierarchies(nDim, aNewLabel.maHiers);
2000                 GetMembers(nDim, GetUsedHierarchy(nDim), aNewLabel.maMembers);
2001                 lcl_FillLabelData(aNewLabel, xDimProp);
2002                 aNewLabel.mnFlags = ScUnoHelpFunctions::GetLongProperty( xDimProp,
2003                                         rtl::OUString::createFromAscii(SC_UNO_FLAGS), 0 );
2004                 rParam.maLabelArray.push_back(aNewLabel);
2005 			}
2006 		}
2007 	}
2008 
2009 	return sal_True;
2010 }
2011 
GetHierarchiesNA(sal_Int32 nDim,uno::Reference<container::XNameAccess> & xHiers)2012 sal_Bool ScDPObject::GetHierarchiesNA( sal_Int32 nDim, uno::Reference< container::XNameAccess >& xHiers )
2013 {
2014     sal_Bool bRet = sal_False;
2015     uno::Reference<container::XNameAccess> xDimsName( GetSource()->getDimensions() );
2016     uno::Reference<container::XIndexAccess> xIntDims(new ScNameToIndexAccess( xDimsName ));
2017     if( xIntDims.is() )
2018     {
2019         uno::Reference<sheet::XHierarchiesSupplier> xHierSup(xIntDims->getByIndex( nDim ), uno::UNO_QUERY);
2020         if (xHierSup.is())
2021         {
2022             xHiers.set( xHierSup->getHierarchies() );
2023             bRet = xHiers.is();
2024         }
2025     }
2026     return bRet;
2027 }
2028 
GetHierarchies(sal_Int32 nDim,uno::Sequence<rtl::OUString> & rHiers)2029 sal_Bool ScDPObject::GetHierarchies( sal_Int32 nDim, uno::Sequence< rtl::OUString >& rHiers )
2030 {
2031     sal_Bool bRet = sal_False;
2032     uno::Reference< container::XNameAccess > xHiersNA;
2033     if( GetHierarchiesNA( nDim, xHiersNA ) )
2034     {
2035         rHiers = xHiersNA->getElementNames();
2036         bRet = sal_True;
2037     }
2038     return bRet;
2039 }
2040 
GetUsedHierarchy(sal_Int32 nDim)2041 sal_Int32 ScDPObject::GetUsedHierarchy( sal_Int32 nDim )
2042 {
2043     sal_Int32 nHier = 0;
2044     uno::Reference<container::XNameAccess> xDimsName( GetSource()->getDimensions() );
2045     uno::Reference<container::XIndexAccess> xIntDims(new ScNameToIndexAccess( xDimsName ));
2046     uno::Reference<beans::XPropertySet> xDim(xIntDims->getByIndex( nDim ), uno::UNO_QUERY);
2047     if (xDim.is())
2048         nHier = ScUnoHelpFunctions::GetLongProperty( xDim, rtl::OUString(RTL_CONSTASCII_USTRINGPARAM( SC_UNO_USEDHIER ) ) );
2049     return nHier;
2050 }
2051 
GetMembersNA(sal_Int32 nDim,uno::Reference<container::XNameAccess> & xMembers)2052 sal_Bool ScDPObject::GetMembersNA( sal_Int32 nDim, uno::Reference< container::XNameAccess >& xMembers )
2053 {
2054     return GetMembersNA( nDim, GetUsedHierarchy( nDim ), xMembers );
2055 }
2056 
GetMembersNA(sal_Int32 nDim,sal_Int32 nHier,uno::Reference<container::XNameAccess> & xMembers)2057 sal_Bool ScDPObject::GetMembersNA( sal_Int32 nDim, sal_Int32 nHier, uno::Reference< container::XNameAccess >& xMembers )
2058 {
2059     sal_Bool bRet = sal_False;
2060     uno::Reference<container::XNameAccess> xDimsName( GetSource()->getDimensions() );
2061     uno::Reference<container::XIndexAccess> xIntDims(new ScNameToIndexAccess( xDimsName ));
2062     uno::Reference<beans::XPropertySet> xDim(xIntDims->getByIndex( nDim ), uno::UNO_QUERY);
2063     if (xDim.is())
2064     {
2065         uno::Reference<sheet::XHierarchiesSupplier> xHierSup(xDim, uno::UNO_QUERY);
2066         if (xHierSup.is())
2067         {
2068             uno::Reference<container::XIndexAccess> xHiers(new ScNameToIndexAccess(xHierSup->getHierarchies()));
2069             uno::Reference<sheet::XLevelsSupplier> xLevSupp( xHiers->getByIndex(nHier), uno::UNO_QUERY );
2070             if ( xLevSupp.is() )
2071             {
2072                 uno::Reference<container::XIndexAccess> xLevels(new ScNameToIndexAccess( xLevSupp->getLevels()));
2073                 if (xLevels.is())
2074                 {
2075                     sal_Int32 nLevCount = xLevels->getCount();
2076                     if (nLevCount > 0)
2077                     {
2078                         uno::Reference<sheet::XMembersSupplier> xMembSupp( xLevels->getByIndex(0), uno::UNO_QUERY );
2079                         if ( xMembSupp.is() )
2080                         {
2081                             xMembers.set(xMembSupp->getMembers());
2082                             bRet = sal_True;
2083                         }
2084                     }
2085                 }
2086             }
2087         }
2088     }
2089     return bRet;
2090 }
2091 
2092 //------------------------------------------------------------------------
2093 //	convert old pivot tables into new datapilot tables
2094 
lcl_GetDimName(const uno::Reference<sheet::XDimensionsSupplier> & xSource,long nDim)2095 String lcl_GetDimName( const uno::Reference<sheet::XDimensionsSupplier>& xSource, long nDim )
2096 {
2097 	rtl::OUString aName;
2098 	if ( xSource.is() )
2099 	{
2100 		uno::Reference<container::XNameAccess> xDimsName = xSource->getDimensions();
2101 		uno::Reference<container::XIndexAccess> xDims = new ScNameToIndexAccess( xDimsName );
2102 		long nDimCount = xDims->getCount();
2103 		if ( nDim < nDimCount )
2104 		{
2105 			uno::Reference<uno::XInterface> xIntDim =
2106 				ScUnoHelpFunctions::AnyToInterface( xDims->getByIndex(nDim) );
2107 			uno::Reference<container::XNamed> xDimName( xIntDim, uno::UNO_QUERY );
2108 			if (xDimName.is())
2109 			{
2110 				try
2111 				{
2112 					aName = xDimName->getName();
2113 				}
2114 				catch(uno::Exception&)
2115 				{
2116 				}
2117 			}
2118 		}
2119 	}
2120 	return aName;
2121 }
2122 
2123 // static
ConvertOrientation(ScDPSaveData & rSaveData,const ScPivotFieldVector & rFields,sal_uInt16 nOrient,ScDocument * pDoc,SCROW nRow,SCTAB nTab,const uno::Reference<sheet::XDimensionsSupplier> & xSource,bool bOldDefaults,const ScPivotFieldVector * pRefColFields,const ScPivotFieldVector * pRefRowFields,const ScPivotFieldVector * pRefPageFields)2124 void ScDPObject::ConvertOrientation( ScDPSaveData& rSaveData,
2125 							const ScPivotFieldVector& rFields, sal_uInt16 nOrient,
2126 							ScDocument* pDoc, SCROW nRow, SCTAB nTab,
2127 							const uno::Reference<sheet::XDimensionsSupplier>& xSource,
2128 							bool bOldDefaults,
2129 							const ScPivotFieldVector* pRefColFields,
2130                             const ScPivotFieldVector* pRefRowFields,
2131                             const ScPivotFieldVector* pRefPageFields )
2132 {
2133 	//	pDoc or xSource must be set
2134 	DBG_ASSERT( pDoc || xSource.is(), "missing string source" );
2135 
2136 	String aDocStr;
2137 	ScDPSaveDimension* pDim;
2138 
2139 	for (ScPivotFieldVector::const_iterator aIt = rFields.begin(), aEnd = rFields.end(); aIt != aEnd; ++aIt)
2140 	{
2141 		SCCOL nCol = aIt->nCol;
2142 		sal_uInt16 nFuncs = aIt->nFuncMask;
2143         const sheet::DataPilotFieldReference& rFieldRef = aIt->maFieldRef;
2144 
2145 		if ( nCol == PIVOT_DATA_FIELD )
2146 			pDim = rSaveData.GetDataLayoutDimension();
2147 		else
2148 		{
2149 			if ( pDoc )
2150 				pDoc->GetString( nCol, nRow, nTab, aDocStr );
2151 			else
2152 				aDocStr = lcl_GetDimName( xSource, nCol );	// cols must start at 0
2153 
2154 			if ( aDocStr.Len() )
2155 				pDim = rSaveData.GetDimensionByName(aDocStr);
2156 			else
2157 				pDim = NULL;
2158 		}
2159 
2160 		if ( pDim )
2161 		{
2162 			if ( nOrient == sheet::DataPilotFieldOrientation_DATA )		// set summary function
2163 			{
2164 				//	generate an individual entry for each function
2165 				bool bFirst = true;
2166 
2167                 //  if a dimension is used for column/row/page and data,
2168 				//	use duplicated dimensions for all data occurrences
2169 				if (pRefColFields)
2170                     for (ScPivotFieldVector::const_iterator aRefIt = pRefColFields->begin(), aRefEnd = pRefColFields->end(); bFirst && (aRefIt != aRefEnd); ++aRefIt)
2171 						if (aRefIt->nCol == nCol)
2172 							bFirst = false;
2173 				if (pRefRowFields)
2174                     for (ScPivotFieldVector::const_iterator aRefIt = pRefRowFields->begin(), aRefEnd = pRefRowFields->end(); bFirst && (aRefIt != aRefEnd); ++aRefIt)
2175 						if (aRefIt->nCol == nCol)
2176 							bFirst = false;
2177                 if (pRefPageFields)
2178                     for (ScPivotFieldVector::const_iterator aRefIt = pRefPageFields->begin(), aRefEnd = pRefPageFields->end(); bFirst && (aRefIt != aRefEnd); ++aRefIt)
2179 						if (aRefIt->nCol == nCol)
2180 							bFirst = false;
2181 
2182 				//	if set via api, a data column may occur several times
2183 				//	(if the function hasn't been changed yet) -> also look for duplicate data column
2184                 for (ScPivotFieldVector::const_iterator aRefIt = rFields.begin(); bFirst && (aRefIt != aIt); ++aRefIt)
2185 					if (aRefIt->nCol == nCol)
2186 						bFirst = false;
2187 
2188 				sal_uInt16 nMask = 1;
2189 				for (sal_uInt16 nBit=0; nBit<16; nBit++)
2190 				{
2191 					if ( nFuncs & nMask )
2192 					{
2193 						sheet::GeneralFunction eFunc = ScDataPilotConversion::FirstFunc( nMask );
2194                         ScDPSaveDimension* pCurrDim = bFirst ? pDim : rSaveData.DuplicateDimension(pDim->GetName());
2195                         pCurrDim->SetOrientation( nOrient );
2196                         pCurrDim->SetFunction( sal::static_int_cast<sal_uInt16>(eFunc) );
2197 
2198                         if( rFieldRef.ReferenceType == sheet::DataPilotFieldReferenceType::NONE )
2199                             pCurrDim->SetReferenceValue( 0 );
2200                         else
2201                             pCurrDim->SetReferenceValue( &rFieldRef );
2202 
2203                         bFirst = false;
2204 					}
2205 					nMask *= 2;
2206 				}
2207 			}
2208 			else											// set SubTotals
2209 			{
2210 				pDim->SetOrientation( nOrient );
2211 
2212 				sal_uInt16 nFuncArray[16];
2213 				sal_uInt16 nFuncCount = 0;
2214 				sal_uInt16 nMask = 1;
2215 				for (sal_uInt16 nBit=0; nBit<16; nBit++)
2216 				{
2217 					if ( nFuncs & nMask )
2218                         nFuncArray[nFuncCount++] = sal::static_int_cast<sal_uInt16>(ScDataPilotConversion::FirstFunc( nMask ));
2219 					nMask *= 2;
2220 				}
2221 				pDim->SetSubTotals( nFuncCount, nFuncArray );
2222 
2223 				//	ShowEmpty was implicit in old tables,
2224 				//	must be set for data layout dimension (not accessible in dialog)
2225 				if ( bOldDefaults || nCol == PIVOT_DATA_FIELD )
2226 					pDim->SetShowEmpty( sal_True );
2227 			}
2228 		}
2229 	}
2230 }
2231 
2232 // static
IsOrientationAllowed(sal_uInt16 nOrient,sal_Int32 nDimFlags)2233 bool ScDPObject::IsOrientationAllowed( sal_uInt16 nOrient, sal_Int32 nDimFlags )
2234 {
2235     bool bAllowed = true;
2236     switch (nOrient)
2237     {
2238         case sheet::DataPilotFieldOrientation_PAGE:
2239             bAllowed = ( nDimFlags & sheet::DimensionFlags::NO_PAGE_ORIENTATION ) == 0;
2240             break;
2241         case sheet::DataPilotFieldOrientation_COLUMN:
2242             bAllowed = ( nDimFlags & sheet::DimensionFlags::NO_COLUMN_ORIENTATION ) == 0;
2243             break;
2244         case sheet::DataPilotFieldOrientation_ROW:
2245             bAllowed = ( nDimFlags & sheet::DimensionFlags::NO_ROW_ORIENTATION ) == 0;
2246             break;
2247         case sheet::DataPilotFieldOrientation_DATA:
2248             bAllowed = ( nDimFlags & sheet::DimensionFlags::NO_DATA_ORIENTATION ) == 0;
2249             break;
2250         default:
2251             {
2252                 // allowed to remove from previous orientation
2253             }
2254     }
2255     return bAllowed;
2256 }
2257 
2258 // -----------------------------------------------------------------------
2259 
2260 //	static
HasRegisteredSources()2261 sal_Bool ScDPObject::HasRegisteredSources()
2262 {
2263 	sal_Bool bFound = sal_False;
2264 
2265 	uno::Reference<lang::XMultiServiceFactory> xManager = comphelper::getProcessServiceFactory();
2266 	uno::Reference<container::XContentEnumerationAccess> xEnAc( xManager, uno::UNO_QUERY );
2267 	if ( xEnAc.is() )
2268 	{
2269 		uno::Reference<container::XEnumeration> xEnum = xEnAc->createContentEnumeration(
2270 										rtl::OUString::createFromAscii( SCDPSOURCE_SERVICE ) );
2271 		if ( xEnum.is() && xEnum->hasMoreElements() )
2272 			bFound = sal_True;
2273 	}
2274 
2275 	return bFound;
2276 }
2277 
2278 //	static
GetRegisteredSources()2279 uno::Sequence<rtl::OUString> ScDPObject::GetRegisteredSources()
2280 {
2281 	long nCount = 0;
2282 	uno::Sequence<rtl::OUString> aSeq(0);
2283 
2284 	//	use implementation names...
2285 
2286 	uno::Reference<lang::XMultiServiceFactory> xManager = comphelper::getProcessServiceFactory();
2287 	uno::Reference<container::XContentEnumerationAccess> xEnAc( xManager, uno::UNO_QUERY );
2288 	if ( xEnAc.is() )
2289 	{
2290 		uno::Reference<container::XEnumeration> xEnum = xEnAc->createContentEnumeration(
2291 										rtl::OUString::createFromAscii( SCDPSOURCE_SERVICE ) );
2292 		if ( xEnum.is() )
2293 		{
2294 			while ( xEnum->hasMoreElements() )
2295 			{
2296 				uno::Any aAddInAny = xEnum->nextElement();
2297 //				if ( aAddInAny.getReflection()->getTypeClass() == TypeClass_INTERFACE )
2298 				{
2299 					uno::Reference<uno::XInterface> xIntFac;
2300 					aAddInAny >>= xIntFac;
2301 					if ( xIntFac.is() )
2302 					{
2303 						uno::Reference<lang::XServiceInfo> xInfo( xIntFac, uno::UNO_QUERY );
2304 						if ( xInfo.is() )
2305 						{
2306 							rtl::OUString sName = xInfo->getImplementationName();
2307 
2308 							aSeq.realloc( nCount+1 );
2309 							aSeq.getArray()[nCount] = sName;
2310 							++nCount;
2311 						}
2312 					}
2313 				}
2314 			}
2315 		}
2316 	}
2317 
2318 	return aSeq;
2319 }
2320 
2321 // use getContext from addincol.cxx
2322 uno::Reference<uno::XComponentContext> getContext(uno::Reference<lang::XMultiServiceFactory> xMSF);
2323 
2324 //	static
CreateSource(const ScDPServiceDesc & rDesc)2325 uno::Reference<sheet::XDimensionsSupplier> ScDPObject::CreateSource( const ScDPServiceDesc& rDesc )
2326 {
2327 	rtl::OUString aImplName = rDesc.aServiceName;
2328 	uno::Reference<sheet::XDimensionsSupplier> xRet = NULL;
2329 
2330 	uno::Reference<lang::XMultiServiceFactory> xManager = comphelper::getProcessServiceFactory();
2331 	uno::Reference<container::XContentEnumerationAccess> xEnAc( xManager, uno::UNO_QUERY );
2332 	if ( xEnAc.is() )
2333 	{
2334 		uno::Reference<container::XEnumeration> xEnum = xEnAc->createContentEnumeration(
2335 										rtl::OUString::createFromAscii( SCDPSOURCE_SERVICE ) );
2336 		if ( xEnum.is() )
2337 		{
2338 			while ( xEnum->hasMoreElements() && !xRet.is() )
2339 			{
2340 				uno::Any aAddInAny = xEnum->nextElement();
2341 //				if ( aAddInAny.getReflection()->getTypeClass() == TypeClass_INTERFACE )
2342 				{
2343 					uno::Reference<uno::XInterface> xIntFac;
2344 					aAddInAny >>= xIntFac;
2345 					if ( xIntFac.is() )
2346 					{
2347 						uno::Reference<lang::XServiceInfo> xInfo( xIntFac, uno::UNO_QUERY );
2348 						if ( xInfo.is() && xInfo->getImplementationName() == aImplName )
2349 						{
2350 							try
2351 							{
2352                                 // #i113160# try XSingleComponentFactory in addition to (old) XSingleServiceFactory,
2353                                 // passing the context to the component (see ScUnoAddInCollection::Initialize)
2354 
2355                                 uno::Reference<uno::XInterface> xInterface;
2356                                 uno::Reference<uno::XComponentContext> xCtx = getContext(xManager);
2357                                 uno::Reference<lang::XSingleComponentFactory> xCFac( xIntFac, uno::UNO_QUERY );
2358                                 if (xCtx.is() && xCFac.is())
2359                                     xInterface = xCFac->createInstanceWithContext(xCtx);
2360 
2361                                 if (!xInterface.is())
2362                                 {
2363                                     uno::Reference<lang::XSingleServiceFactory> xFac( xIntFac, uno::UNO_QUERY );
2364                                     if ( xFac.is() )
2365                                         xInterface = xFac->createInstance();
2366                                 }
2367 
2368 								uno::Reference<lang::XInitialization> xInit( xInterface, uno::UNO_QUERY );
2369 								if (xInit.is())
2370 								{
2371 									//	initialize
2372 									uno::Sequence<uno::Any> aSeq(4);
2373 									uno::Any* pArray = aSeq.getArray();
2374 									pArray[0] <<= rtl::OUString( rDesc.aParSource );
2375 									pArray[1] <<= rtl::OUString( rDesc.aParName );
2376 									pArray[2] <<= rtl::OUString( rDesc.aParUser );
2377 									pArray[3] <<= rtl::OUString( rDesc.aParPass );
2378 									xInit->initialize( aSeq );
2379 								}
2380 								xRet = uno::Reference<sheet::XDimensionsSupplier>( xInterface, uno::UNO_QUERY );
2381 							}
2382 							catch(uno::Exception&)
2383 							{
2384 							}
2385 						}
2386 					}
2387 				}
2388 			}
2389 		}
2390 	}
2391 
2392 	return xRet;
2393 }
2394 
2395 // ----------------------------------------------------------------------------
2396 
ScDPCollection(ScDocument * pDocument)2397 ScDPCollection::ScDPCollection(ScDocument* pDocument) :
2398 	pDoc( pDocument )
2399 {
2400 }
2401 
ScDPCollection(const ScDPCollection & r)2402 ScDPCollection::ScDPCollection(const ScDPCollection& r) :
2403 	ScCollection(r),
2404     pDoc(r.pDoc)
2405 {
2406 }
2407 
~ScDPCollection()2408 ScDPCollection::~ScDPCollection()
2409 {
2410 }
2411 
Clone() const2412 ScDataObject* ScDPCollection::Clone() const
2413 {
2414 	return new ScDPCollection(*this);
2415 }
2416 
DeleteOnTab(SCTAB nTab)2417 void ScDPCollection::DeleteOnTab( SCTAB nTab )
2418 {
2419     sal_uInt16 nPos = 0;
2420     while ( nPos < nCount )
2421     {
2422         // look for output positions on the deleted sheet
2423         if ( static_cast<const ScDPObject*>(At(nPos))->GetOutRange().aStart.Tab() == nTab )
2424             AtFree(nPos);
2425         else
2426             ++nPos;
2427     }
2428 }
2429 
UpdateReference(UpdateRefMode eUpdateRefMode,const ScRange & r,SCsCOL nDx,SCsROW nDy,SCsTAB nDz)2430 void ScDPCollection::UpdateReference( UpdateRefMode eUpdateRefMode,
2431 										 const ScRange& r, SCsCOL nDx, SCsROW nDy, SCsTAB nDz )
2432 {
2433 	for (sal_uInt16 i=0; i<nCount; i++)
2434 		((ScDPObject*)At(i))->UpdateReference( eUpdateRefMode, r, nDx, nDy, nDz );
2435 }
2436 
RefsEqual(const ScDPCollection & r) const2437 sal_Bool ScDPCollection::RefsEqual( const ScDPCollection& r ) const
2438 {
2439 	if ( nCount != r.nCount )
2440 		return sal_False;
2441 
2442 	for (sal_uInt16 i=0; i<nCount; i++)
2443 		if ( ! ((const ScDPObject*)At(i))->RefsEqual( *((const ScDPObject*)r.At(i)) ) )
2444 			return sal_False;
2445 
2446 	return sal_True;	// all equal
2447 }
2448 
WriteRefsTo(ScDPCollection & r) const2449 void ScDPCollection::WriteRefsTo( ScDPCollection& r ) const
2450 {
2451 	if ( nCount == r.nCount )
2452 	{
2453 		//!	assert equal names?
2454 		for (sal_uInt16 i=0; i<nCount; i++)
2455 			((const ScDPObject*)At(i))->WriteRefsTo( *((ScDPObject*)r.At(i)) );
2456 	}
2457 	else
2458     {
2459         // #i8180# If data pilot tables were deleted with their sheet,
2460         // this collection contains extra entries that must be restored.
2461         // Matching objects are found by their names.
2462 
2463         DBG_ASSERT( nCount >= r.nCount, "WriteRefsTo: missing entries in document" );
2464         for (sal_uInt16 nSourcePos=0; nSourcePos<nCount; nSourcePos++)
2465         {
2466             const ScDPObject* pSourceObj = static_cast<const ScDPObject*>(At(nSourcePos));
2467             String aName = pSourceObj->GetName();
2468             bool bFound = false;
2469             for (sal_uInt16 nDestPos=0; nDestPos<r.nCount && !bFound; nDestPos++)
2470             {
2471                 ScDPObject* pDestObj = static_cast<ScDPObject*>(r.At(nDestPos));
2472                 if ( pDestObj->GetName() == aName )
2473                 {
2474                     pSourceObj->WriteRefsTo( *pDestObj );     // found object, copy refs
2475                     bFound = true;
2476                 }
2477             }
2478             if ( !bFound )
2479             {
2480                 // none found, re-insert deleted object (see ScUndoDataPilot::Undo)
2481 
2482                 ScDPObject* pDestObj = new ScDPObject( *pSourceObj );
2483                 pDestObj->SetAlive(sal_True);
2484                 if ( !r.InsertNewTable(pDestObj) )
2485                 {
2486                     DBG_ERROR("cannot insert DPObject");
2487                     DELETEZ( pDestObj );
2488                 }
2489             }
2490         }
2491         DBG_ASSERT( nCount == r.nCount, "WriteRefsTo: couldn't restore all entries" );
2492     }
2493 }
2494 
GetByName(const String & rName) const2495 ScDPObject* ScDPCollection::GetByName(const String& rName) const
2496 {
2497     for (sal_uInt16 i=0; i<nCount; i++)
2498         if (static_cast<const ScDPObject*>(pItems[i])->GetName() == rName)
2499             return static_cast<ScDPObject*>(pItems[i]);
2500     return NULL;
2501 }
2502 
CreateNewName(sal_uInt16 nMin) const2503 String ScDPCollection::CreateNewName( sal_uInt16 nMin ) const
2504 {
2505 	String aBase( RTL_CONSTASCII_USTRINGPARAM( "Pivot" ) );
2506 	//!	from Resource?
2507 
2508 	for (sal_uInt16 nAdd=0; nAdd<=nCount; nAdd++)	//	nCount+1 tries
2509 	{
2510 		String aNewName = aBase;
2511 		aNewName += String::CreateFromInt32( nMin + nAdd );
2512 		sal_Bool bFound = sal_False;
2513 		for (sal_uInt16 i=0; i<nCount && !bFound; i++)
2514 			if (((const ScDPObject*)pItems[i])->GetName() == aNewName)
2515 				bFound = sal_True;
2516 		if (!bFound)
2517 			return aNewName;			// found unused Name
2518 	}
2519 	return String();					// should not happen
2520 }
2521 
2522 
2523 
2524 // Wang Xu Ming -- 2009-8-17
2525 // DataPilot Migration - Cache&&Performance
GetCacheId() const2526 long ScDPObject::GetCacheId() const
2527 {
2528     if ( GetSaveData() )
2529         return GetSaveData()->GetCacheId();
2530     else
2531         return mnCacheId;
2532 }
RefreshCache()2533 sal_uLong ScDPObject::RefreshCache()
2534 {
2535     if ( pServDesc )
2536     {
2537         // cache table isn't used for external service - do nothing, no error
2538         return 0;
2539     }
2540 
2541     CreateObjects();
2542     sal_uLong nErrId = 0;
2543     if ( pSheetDesc)
2544         nErrId =  pSheetDesc->CheckValidate( pDoc );
2545     if ( nErrId == 0 )
2546     {
2547         long nOldId = GetCacheId();
2548         long nNewId = pDoc->GetNewDPObjectCacheId();
2549         if ( nOldId >= 0 )
2550             pDoc->RemoveDPObjectCache( nOldId );
2551 
2552         ScDPTableDataCache* pCache  = NULL;
2553         if ( pSheetDesc )
2554             pCache = pSheetDesc->CreateCache( pDoc, nNewId );
2555         else if ( pImpDesc )
2556             pCache = pImpDesc->CreateCache( pDoc, nNewId );
2557 
2558         if ( pCache == NULL )
2559         {
2560             //cache failed
2561             DBG_ASSERT( pCache , " pCache == NULL" );
2562             return STR_ERR_DATAPILOTSOURCE;
2563         }
2564 
2565         nNewId = pCache->GetId();
2566 
2567         bRefresh = sal_True;
2568         ScDPCollection* pDPCollection = pDoc->GetDPCollection();
2569         sal_uInt16 nCount = pDPCollection->GetCount();
2570         for (sal_uInt16 i=0; i<nCount; i++)
2571         { //set new cache id
2572             if ( (*pDPCollection)[i]->GetCacheId() == nOldId  )
2573             {
2574                 (*pDPCollection)[i]->SetCacheId( nNewId );
2575                 (*pDPCollection)[i]->SetRefresh();
2576 
2577             }
2578         }
2579         DBG_ASSERT( GetCacheId() >= 0, " GetCacheId() >= 0 " );
2580     }
2581     return nErrId;
2582 }
SetCacheId(long nCacheId)2583 void ScDPObject::SetCacheId( long nCacheId )
2584 {
2585     if ( GetCacheId() != nCacheId )
2586     {
2587         InvalidateSource();
2588         if ( GetSaveData() )
2589             GetSaveData()->SetCacheId( nCacheId );
2590 
2591         mnCacheId = nCacheId;
2592     }
2593 }
GetCache() const2594 const ScDPTableDataCache* ScDPObject::GetCache() const
2595 {
2596     return pDoc->GetDPObjectCache( GetCacheId() );
2597 }
2598 // End Comments
2599 
FreeTable(ScDPObject * pDPObj)2600 void ScDPCollection::FreeTable(ScDPObject* pDPObj)
2601 {
2602     const ScRange& rOutRange = pDPObj->GetOutRange();
2603     const ScAddress& s = rOutRange.aStart;
2604     const ScAddress& e = rOutRange.aEnd;
2605     pDoc->RemoveFlagsTab(s.Col(), s.Row(), e.Col(), e.Row(), s.Tab(), SC_MF_DP_TABLE);
2606     Free(pDPObj);
2607 }
2608 
InsertNewTable(ScDPObject * pDPObj)2609 bool ScDPCollection::InsertNewTable(ScDPObject* pDPObj)
2610 {
2611     bool bSuccess = Insert(pDPObj);
2612     if (bSuccess)
2613     {
2614         const ScRange& rOutRange = pDPObj->GetOutRange();
2615         const ScAddress& s = rOutRange.aStart;
2616         const ScAddress& e = rOutRange.aEnd;
2617         pDoc->ApplyFlagsTab(s.Col(), s.Row(), e.Col(), e.Row(), s.Tab(), SC_MF_DP_TABLE);
2618     }
2619     return bSuccess;
2620 }
2621 
HasDPTable(SCCOL nCol,SCROW nRow,SCTAB nTab) const2622 bool ScDPCollection::HasDPTable(SCCOL nCol, SCROW nRow, SCTAB nTab) const
2623 {
2624     const ScMergeFlagAttr* pMergeAttr = static_cast<const ScMergeFlagAttr*>(
2625             pDoc->GetAttr(nCol, nRow, nTab, ATTR_MERGE_FLAG));
2626 
2627     if (!pMergeAttr)
2628         return false;
2629 
2630     return pMergeAttr->HasDPTable();
2631 }
2632 
2633 
2634