/************************************************************************* * * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * Copyright 2000, 2010 Oracle and/or its affiliates. * * OpenOffice.org - a multi-platform office productivity suite * * This file is part of OpenOffice.org. * * OpenOffice.org is free software: you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License version 3 * only, as published by the Free Software Foundation. * * OpenOffice.org is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Lesser General Public License version 3 for more details * (a copy is included in the LICENSE file that accompanied this code). * * You should have received a copy of the GNU Lesser General Public License * version 3 along with OpenOffice.org. If not, see * * for a copy of the LGPLv3 License. * ************************************************************************/ // MARKER(update_precomp.py): autogen include statement, do not remove #include "precompiled_sc.hxx" #include #include #include #include "dapiuno.hxx" #include "datauno.hxx" #include "miscuno.hxx" #include "convuno.hxx" #include "docsh.hxx" #include "tabvwsh.hxx" #include "pivot.hxx" #include "rangeutl.hxx" #include "unoguard.hxx" #include "dpobject.hxx" #include "dpshttab.hxx" #include "dpsdbtab.hxx" #include "dpsave.hxx" #include "dbdocfun.hxx" #include "unonames.hxx" #include "dpgroup.hxx" #include "dpdimsave.hxx" #include "hints.hxx" #include #include #include #include #include #include #include #include #include #include #include using namespace com::sun::star; using namespace com::sun::star::sheet; using ::rtl::OUString; using ::com::sun::star::uno::Any; using ::com::sun::star::uno::Exception; using ::com::sun::star::uno::Reference; using ::com::sun::star::uno::RuntimeException; using ::com::sun::star::uno::Sequence; using ::com::sun::star::uno::UNO_QUERY; using ::com::sun::star::uno::UNO_QUERY_THROW; using ::com::sun::star::container::ElementExistException; using ::com::sun::star::container::NoSuchElementException; using ::com::sun::star::container::XEnumeration; using ::com::sun::star::container::XIndexAccess; using ::com::sun::star::container::XNameAccess; using ::com::sun::star::container::XNamed; using ::com::sun::star::beans::PropertyVetoException; using ::com::sun::star::beans::UnknownPropertyException; using ::com::sun::star::beans::XPropertyChangeListener; using ::com::sun::star::beans::XPropertySet; using ::com::sun::star::beans::XPropertySetInfo; using ::com::sun::star::beans::XVetoableChangeListener; using ::com::sun::star::lang::IllegalArgumentException; using ::com::sun::star::lang::IndexOutOfBoundsException; using ::com::sun::star::lang::WrappedTargetException; using ::com::sun::star::table::CellAddress; using ::com::sun::star::table::CellRangeAddress; // ============================================================================ namespace { const SfxItemPropertyMapEntry* lcl_GetDataPilotDescriptorBaseMap() { static SfxItemPropertyMapEntry aDataPilotDescriptorBaseMap_Impl[] = { {MAP_CHAR_LEN(SC_UNO_COLGRAND), 0, &getBooleanCppuType(), 0, 0 }, {MAP_CHAR_LEN(SC_UNO_DRILLDOWN), 0, &getBooleanCppuType(), 0, 0 }, {MAP_CHAR_LEN(SC_UNO_GRANDTOTAL_NAME),0,&getCppuType((rtl::OUString*)0), beans::PropertyAttribute::MAYBEVOID, 0 }, {MAP_CHAR_LEN(SC_UNO_IGNEMPROWS), 0, &getBooleanCppuType(), 0, 0 }, {MAP_CHAR_LEN(SC_UNO_IMPORTDESC), 0, &getCppuType((uno::Sequence*)0), 0, 0 }, {MAP_CHAR_LEN(SC_UNO_RPTEMPTY), 0, &getBooleanCppuType(), 0, 0 }, {MAP_CHAR_LEN(SC_UNO_ROWGRAND), 0, &getBooleanCppuType(), 0, 0 }, {MAP_CHAR_LEN(SC_UNO_SERVICEARG), 0, &getCppuType((uno::Sequence*)0), 0, 0 }, {MAP_CHAR_LEN(SC_UNO_SHOWFILT), 0, &getBooleanCppuType(), 0, 0 }, {MAP_CHAR_LEN(SC_UNO_SOURCESERV), 0, &getCppuType((rtl::OUString*)0), 0, 0 }, {0,0,0,0,0,0} }; return aDataPilotDescriptorBaseMap_Impl; } // ---------------------------------------------------------------------------- const SfxItemPropertyMapEntry* lcl_GetDataPilotFieldMap() { using namespace ::com::sun::star::beans::PropertyAttribute; static SfxItemPropertyMapEntry aDataPilotFieldMap_Impl[] = { {MAP_CHAR_LEN(SC_UNONAME_AUTOSHOW), 0, &getCppuType((DataPilotFieldAutoShowInfo*)0), MAYBEVOID, 0 }, {MAP_CHAR_LEN(SC_UNONAME_FUNCTION), 0, &getCppuType((GeneralFunction*)0), 0, 0 }, {MAP_CHAR_LEN(SC_UNONAME_GROUPINFO), 0, &getCppuType((DataPilotFieldGroupInfo*)0), MAYBEVOID, 0 }, {MAP_CHAR_LEN(SC_UNONAME_HASAUTOSHOW), 0, &getBooleanCppuType(), 0, 0 }, {MAP_CHAR_LEN(SC_UNONAME_HASLAYOUTINFO),0, &getBooleanCppuType(), 0, 0 }, {MAP_CHAR_LEN(SC_UNONAME_HASREFERENCE), 0, &getBooleanCppuType(), 0, 0 }, {MAP_CHAR_LEN(SC_UNONAME_HASSORTINFO), 0, &getBooleanCppuType(), 0, 0 }, {MAP_CHAR_LEN(SC_UNONAME_ISGROUP), 0, &getBooleanCppuType(), 0, 0 }, {MAP_CHAR_LEN(SC_UNONAME_LAYOUTINFO), 0, &getCppuType((DataPilotFieldLayoutInfo*)0), MAYBEVOID, 0 }, {MAP_CHAR_LEN(SC_UNONAME_ORIENT), 0, &getCppuType((DataPilotFieldOrientation*)0), MAYBEVOID, 0 }, {MAP_CHAR_LEN(SC_UNONAME_REFERENCE), 0, &getCppuType((DataPilotFieldReference*)0), MAYBEVOID, 0 }, {MAP_CHAR_LEN(SC_UNONAME_SELPAGE), 0, &getCppuType((OUString*)0), 0, 0 }, {MAP_CHAR_LEN(SC_UNONAME_SHOWEMPTY), 0, &getBooleanCppuType(), 0, 0 }, {MAP_CHAR_LEN(SC_UNONAME_SORTINFO), 0, &getCppuType((DataPilotFieldSortInfo*)0), MAYBEVOID, 0 }, {MAP_CHAR_LEN(SC_UNONAME_SUBTOTALS), 0, &getCppuType((Sequence*)0), 0, 0 }, {MAP_CHAR_LEN(SC_UNONAME_USESELPAGE), 0, &getBooleanCppuType(), 0, 0 }, {0,0,0,0,0,0} }; return aDataPilotFieldMap_Impl; } // ---------------------------------------------------------------------------- const SfxItemPropertyMapEntry* lcl_GetDataPilotItemMap() { static SfxItemPropertyMapEntry aDataPilotItemMap_Impl[] = { {MAP_CHAR_LEN(SC_UNONAME_ISHIDDEN), 0, &getBooleanCppuType(), 0, 0 }, {MAP_CHAR_LEN(SC_UNONAME_POS), 0, &getCppuType((sal_Int32*)0), 0, 0 }, {MAP_CHAR_LEN(SC_UNONAME_SHOWDETAIL), 0, &getBooleanCppuType(), 0, 0 }, {0,0,0,0,0,0} }; return aDataPilotItemMap_Impl; } // ---------------------------------------------------------------------------- inline bool lclCheckValidDouble( double fValue, sal_Bool bAuto ) { return bAuto || ::rtl::math::isFinite( fValue ); } bool lclCheckMinMaxStep( const DataPilotFieldGroupInfo& rInfo ) { return lclCheckValidDouble( rInfo.Start, rInfo.HasAutoStart ) && lclCheckValidDouble( rInfo.End, rInfo.HasAutoEnd ) && (rInfo.HasAutoStart || rInfo.HasAutoEnd || (rInfo.Start <= rInfo.End)) && lclCheckValidDouble( rInfo.Step, sal_False ) && (0.0 <= rInfo.Step); } } // namespace // ============================================================================ SC_SIMPLE_SERVICE_INFO( ScDataPilotDescriptor, "ScDataPilotDescriptor", "stardiv::one::sheet::DataPilotDescriptor" ) SC_SIMPLE_SERVICE_INFO( ScDataPilotFieldObj, "ScDataPilotFieldObj", "com.sun.star.sheet.DataPilotField" ) SC_SIMPLE_SERVICE_INFO( ScDataPilotFieldsObj, "ScDataPilotFieldsObj", "com.sun.star.sheet.DataPilotFields" ) SC_SIMPLE_SERVICE_INFO( ScDataPilotTableObj, "ScDataPilotTableObj", "com.sun.star.sheet.DataPilotTable" ) SC_SIMPLE_SERVICE_INFO( ScDataPilotTablesObj, "ScDataPilotTablesObj", "com.sun.star.sheet.DataPilotTables" ) SC_SIMPLE_SERVICE_INFO( ScDataPilotItemsObj, "ScDataPilotItemsObj", "com.sun.star.sheet.DataPilotItems" ) SC_SIMPLE_SERVICE_INFO( ScDataPilotItemObj, "ScDataPilotItemObj", "com.sun.star.sheet.DataPilotItem" ) SC_SIMPLE_SERVICE_INFO( ScDataPilotFieldGroupsObj, "ScDataPilotFieldGroupsObj", "com.sun.star.sheet.DataPilotFieldGroups" ) SC_SIMPLE_SERVICE_INFO( ScDataPilotFieldGroupObj, "ScDataPilotFieldGroupObj", "com.sun.star.sheet.DataPilotFieldGroup" ) SC_SIMPLE_SERVICE_INFO( ScDataPilotFieldGroupItemObj, "ScDataPilotFieldGroupItemObj", "com.sun.star.sheet.DataPilotFieldGroupItem" ) //------------------------------------------------------------------------ // name that is used in the API for the data layout field #define SC_DATALAYOUT_NAME "Data" //------------------------------------------------------------------------ GeneralFunction ScDataPilotConversion::FirstFunc( sal_uInt16 nBits ) { if ( nBits & PIVOT_FUNC_SUM ) return GeneralFunction_SUM; if ( nBits & PIVOT_FUNC_COUNT ) return GeneralFunction_COUNT; if ( nBits & PIVOT_FUNC_AVERAGE ) return GeneralFunction_AVERAGE; if ( nBits & PIVOT_FUNC_MAX ) return GeneralFunction_MAX; if ( nBits & PIVOT_FUNC_MIN ) return GeneralFunction_MIN; if ( nBits & PIVOT_FUNC_PRODUCT ) return GeneralFunction_PRODUCT; if ( nBits & PIVOT_FUNC_COUNT_NUM ) return GeneralFunction_COUNTNUMS; if ( nBits & PIVOT_FUNC_STD_DEV ) return GeneralFunction_STDEV; if ( nBits & PIVOT_FUNC_STD_DEVP ) return GeneralFunction_STDEVP; if ( nBits & PIVOT_FUNC_STD_VAR ) return GeneralFunction_VAR; if ( nBits & PIVOT_FUNC_STD_VARP ) return GeneralFunction_VARP; if ( nBits & PIVOT_FUNC_AUTO ) return GeneralFunction_AUTO; return GeneralFunction_NONE; } sal_uInt16 ScDataPilotConversion::FunctionBit( GeneralFunction eFunc ) { sal_uInt16 nRet = PIVOT_FUNC_NONE; // 0 switch (eFunc) { case GeneralFunction_SUM: nRet = PIVOT_FUNC_SUM; break; case GeneralFunction_COUNT: nRet = PIVOT_FUNC_COUNT; break; case GeneralFunction_AVERAGE: nRet = PIVOT_FUNC_AVERAGE; break; case GeneralFunction_MAX: nRet = PIVOT_FUNC_MAX; break; case GeneralFunction_MIN: nRet = PIVOT_FUNC_MIN; break; case GeneralFunction_PRODUCT: nRet = PIVOT_FUNC_PRODUCT; break; case GeneralFunction_COUNTNUMS: nRet = PIVOT_FUNC_COUNT_NUM; break; case GeneralFunction_STDEV: nRet = PIVOT_FUNC_STD_DEV; break; case GeneralFunction_STDEVP: nRet = PIVOT_FUNC_STD_DEVP; break; case GeneralFunction_VAR: nRet = PIVOT_FUNC_STD_VAR; break; case GeneralFunction_VARP: nRet = PIVOT_FUNC_STD_VARP; break; case GeneralFunction_AUTO: nRet = PIVOT_FUNC_AUTO; break; default: { // added to avoid warnings } } return nRet; } void ScDataPilotConversion::FillGroupInfo( DataPilotFieldGroupInfo& rInfo, const ScDPNumGroupInfo& rGroupInfo ) { rInfo.HasDateValues = rGroupInfo.DateValues; rInfo.HasAutoStart = rGroupInfo.AutoStart; rInfo.Start = rGroupInfo.Start; rInfo.HasAutoEnd = rGroupInfo.AutoEnd; rInfo.End = rGroupInfo.End; rInfo.Step = rGroupInfo.Step; } //------------------------------------------------------------------------ ScDPObject* lcl_GetDPObject( ScDocShell* pDocShell, SCTAB nTab, const String& rName ) { if (pDocShell) { ScDocument* pDoc = pDocShell->GetDocument(); ScDPCollection* pColl = pDoc->GetDPCollection(); if ( pColl ) { sal_uInt16 nCount = pColl->GetCount(); for (sal_uInt16 i=0; iGetOutRange().aStart.Tab() == nTab && pDPObj->GetName() == rName ) return pDPObj; } } } return NULL; // nicht gefunden } String lcl_CreatePivotName( ScDocShell* pDocShell ) { if (pDocShell) { ScDocument* pDoc = pDocShell->GetDocument(); ScDPCollection* pColl = pDoc->GetDPCollection(); if ( pColl ) return pColl->CreateNewName(); } return String(); // sollte nicht vorkommen } sal_Int32 lcl_GetObjectIndex( ScDPObject* pDPObj, const ScFieldIdentifier& rFieldId ) { // used for items - nRepeat in identifier can be ignored if ( pDPObj ) { sal_Int32 nCount = pDPObj->GetDimCount(); for ( sal_Int32 nDim = 0; nDim < nCount; ++nDim ) { sal_Bool bIsDataLayout = sal_False; OUString aDimName( pDPObj->GetDimName( nDim, bIsDataLayout ) ); if ( rFieldId.mbDataLayout ? bIsDataLayout : (aDimName == rFieldId.maFieldName) ) return nDim; } } return -1; // none } //------------------------------------------------------------------------ ScDataPilotTablesObj::ScDataPilotTablesObj(ScDocShell* pDocSh, SCTAB nT) : pDocShell( pDocSh ), nTab( nT ) { pDocShell->GetDocument()->AddUnoObject(*this); } ScDataPilotTablesObj::~ScDataPilotTablesObj() { if (pDocShell) pDocShell->GetDocument()->RemoveUnoObject(*this); } void ScDataPilotTablesObj::Notify( SfxBroadcaster&, const SfxHint& rHint ) { //! Referenz-Update if ( rHint.ISA( SfxSimpleHint ) && ((const SfxSimpleHint&)rHint).GetId() == SFX_HINT_DYING ) { pDocShell = NULL; // ungueltig geworden } } // XDataPilotTables ScDataPilotTableObj* ScDataPilotTablesObj::GetObjectByIndex_Impl( sal_Int32 nIndex ) { if (pDocShell) { ScDocument* pDoc = pDocShell->GetDocument(); ScDPCollection* pColl = pDoc->GetDPCollection(); if ( pColl ) { // count tables on this sheet sal_Int32 nFound = 0; sal_uInt16 nCount = pColl->GetCount(); for (sal_uInt16 i=0; iGetOutRange().aStart.Tab() == nTab ) { if ( nFound == nIndex ) { String aName = pDPObj->GetName(); return new ScDataPilotTableObj( pDocShell, nTab, aName ); } ++nFound; } } } } return NULL; } ScDataPilotTableObj* ScDataPilotTablesObj::GetObjectByName_Impl(const OUString& rName) { if (hasByName(rName)) return new ScDataPilotTableObj( pDocShell, nTab, rName ); return 0; } Reference SAL_CALL ScDataPilotTablesObj::createDataPilotDescriptor() throw(RuntimeException) { ScUnoGuard aGuard; if (pDocShell) return new ScDataPilotDescriptor(pDocShell); return NULL; } bool lcl_IsDuplicated( const Reference xDimProps ) { try { Any aAny = xDimProps->getPropertyValue( OUString( RTL_CONSTASCII_USTRINGPARAM( SC_UNO_ORIGINAL ) ) ); Reference< XNamed > xOriginal( aAny, UNO_QUERY ); return xOriginal.is(); } catch( Exception& ) { } return false; } OUString lcl_GetOriginalName( const Reference< XNamed > xDim ) { Reference< XNamed > xOriginal; Reference< XPropertySet > xDimProps( xDim, UNO_QUERY ); if ( xDimProps.is() ) { try { Any aAny = xDimProps->getPropertyValue(OUString(RTL_CONSTASCII_USTRINGPARAM(SC_UNO_ORIGINAL))); aAny >>= xOriginal; } catch( Exception& ) { } } if ( !xOriginal.is() ) xOriginal = xDim; return xOriginal->getName(); } void SAL_CALL ScDataPilotTablesObj::insertNewByName( const OUString& aNewName, const CellAddress& aOutputAddress, const Reference& xDescriptor ) throw(RuntimeException) { ScUnoGuard aGuard; if (!xDescriptor.is()) return; // inserting with already existing name? if ( aNewName.getLength() && hasByName( aNewName ) ) throw RuntimeException(); // no other exceptions specified sal_Bool bDone = sal_False; ScDataPilotDescriptorBase* pImp = ScDataPilotDescriptorBase::getImplementation( xDescriptor ); if ( pDocShell && pImp ) { ScDPObject* pNewObj = pImp->GetDPObject(); if (pNewObj) { ScRange aOutputRange((SCCOL)aOutputAddress.Column, (SCROW)aOutputAddress.Row, (SCTAB)aOutputAddress.Sheet, (SCCOL)aOutputAddress.Column, (SCROW)aOutputAddress.Row, (SCTAB)aOutputAddress.Sheet); pNewObj->SetOutRange(aOutputRange); String aName = aNewName; if (!aName.Len()) aName = lcl_CreatePivotName( pDocShell ); pNewObj->SetName(aName); String aTag = xDescriptor->getTag(); pNewObj->SetTag(aTag); // todo: handle double fields (for more information see ScDPObject ScDBDocFunc aFunc(*pDocShell); bDone = aFunc.DataPilotUpdate( NULL, pNewObj, sal_True, sal_True ); } } if (!bDone) throw RuntimeException(); // no other exceptions specified } void SAL_CALL ScDataPilotTablesObj::removeByName( const OUString& aName ) throw(RuntimeException) { ScUnoGuard aGuard; String aNameStr(aName); ScDPObject* pDPObj = lcl_GetDPObject( pDocShell, nTab, aNameStr ); if (pDPObj && pDocShell) { ScDBDocFunc aFunc(*pDocShell); aFunc.DataPilotUpdate( pDPObj, NULL, sal_True, sal_True ); // remove - incl. undo etc. } else throw RuntimeException(); // no other exceptions specified } // XEnumerationAccess Reference< XEnumeration > SAL_CALL ScDataPilotTablesObj::createEnumeration() throw(RuntimeException) { ScUnoGuard aGuard; return new ScIndexEnumeration(this, OUString(RTL_CONSTASCII_USTRINGPARAM("com.sun.star.sheet.DataPilotTablesEnumeration"))); } // XIndexAccess sal_Int32 SAL_CALL ScDataPilotTablesObj::getCount() throw(RuntimeException) { ScUnoGuard aGuard; if ( pDocShell ) { ScDocument* pDoc = pDocShell->GetDocument(); ScDPCollection* pColl = pDoc->GetDPCollection(); if ( pColl ) { // count tables on this sheet sal_uInt16 nFound = 0; sal_uInt16 nCount = pColl->GetCount(); for (sal_uInt16 i=0; iGetOutRange().aStart.Tab() == nTab ) ++nFound; } return nFound; } } return 0; } Any SAL_CALL ScDataPilotTablesObj::getByIndex( sal_Int32 nIndex ) throw(IndexOutOfBoundsException, WrappedTargetException, RuntimeException) { ScUnoGuard aGuard; Reference xTable(GetObjectByIndex_Impl(nIndex)); if (!xTable.is()) throw IndexOutOfBoundsException(); return Any( xTable ); } uno::Type SAL_CALL ScDataPilotTablesObj::getElementType() throw(RuntimeException) { ScUnoGuard aGuard; return getCppuType((Reference*)0); } sal_Bool SAL_CALL ScDataPilotTablesObj::hasElements() throw(RuntimeException) { ScUnoGuard aGuard; return ( getCount() != 0 ); } // XNameAccess Any SAL_CALL ScDataPilotTablesObj::getByName( const OUString& aName ) throw(NoSuchElementException, WrappedTargetException, RuntimeException) { ScUnoGuard aGuard; Reference xTable(GetObjectByName_Impl(aName)); if (!xTable.is()) throw NoSuchElementException(); return Any( xTable ); } Sequence SAL_CALL ScDataPilotTablesObj::getElementNames() throw(RuntimeException) { ScUnoGuard aGuard; if (pDocShell) { ScDocument* pDoc = pDocShell->GetDocument(); ScDPCollection* pColl = pDoc->GetDPCollection(); if ( pColl ) { // count tables on this sheet sal_uInt16 nFound = 0; sal_uInt16 nCount = pColl->GetCount(); sal_uInt16 i; for (i=0; iGetOutRange().aStart.Tab() == nTab ) ++nFound; } sal_uInt16 nPos = 0; Sequence aSeq(nFound); OUString* pAry = aSeq.getArray(); for (i=0; iGetOutRange().aStart.Tab() == nTab ) pAry[nPos++] = pDPObj->GetName(); } return aSeq; } } return Sequence(0); } sal_Bool SAL_CALL ScDataPilotTablesObj::hasByName( const OUString& aName ) throw(RuntimeException) { ScUnoGuard aGuard; if (pDocShell) { ScDocument* pDoc = pDocShell->GetDocument(); ScDPCollection* pColl = pDoc->GetDPCollection(); if ( pColl ) { String aNamStr(aName); sal_uInt16 nCount = pColl->GetCount(); for (sal_uInt16 i=0; iGetOutRange().aStart.Tab() == nTab && pDPObj->GetName() == aNamStr ) return sal_True; } } } return sal_False; } //------------------------------------------------------------------------ ScDataPilotDescriptorBase::ScDataPilotDescriptorBase(ScDocShell* pDocSh) : maPropSet( lcl_GetDataPilotDescriptorBaseMap() ), pDocShell( pDocSh ) { pDocShell->GetDocument()->AddUnoObject(*this); } ScDataPilotDescriptorBase::~ScDataPilotDescriptorBase() { if (pDocShell) pDocShell->GetDocument()->RemoveUnoObject(*this); } Any SAL_CALL ScDataPilotDescriptorBase::queryInterface( const uno::Type& rType ) throw(RuntimeException) { SC_QUERYINTERFACE( XDataPilotDescriptor ) SC_QUERYINTERFACE( XPropertySet ) SC_QUERYINTERFACE( XDataPilotDataLayoutFieldSupplier ) SC_QUERYINTERFACE( XNamed ) // base of XDataPilotDescriptor SC_QUERYINTERFACE( lang::XUnoTunnel ) SC_QUERYINTERFACE( lang::XTypeProvider ) SC_QUERYINTERFACE( lang::XServiceInfo ) return OWeakObject::queryInterface( rType ); } void SAL_CALL ScDataPilotDescriptorBase::acquire() throw() { OWeakObject::acquire(); } void SAL_CALL ScDataPilotDescriptorBase::release() throw() { OWeakObject::release(); } Sequence< uno::Type > SAL_CALL ScDataPilotDescriptorBase::getTypes() throw(RuntimeException) { static Sequence< uno::Type > aTypes; if ( aTypes.getLength() == 0 ) { aTypes.realloc( 6 ); uno::Type* pPtr = aTypes.getArray(); pPtr[ 0 ] = getCppuType( (const Reference< XDataPilotDescriptor >*)0 ); pPtr[ 1 ] = getCppuType( (const Reference< XPropertySet >*)0 ); pPtr[ 2 ] = getCppuType( (const Reference< XDataPilotDataLayoutFieldSupplier >*)0 ); pPtr[ 3 ] = getCppuType( (const Reference< lang::XUnoTunnel >*)0 ); pPtr[ 4 ] = getCppuType( (const Reference< lang::XTypeProvider >*)0 ); pPtr[ 5 ] = getCppuType( (const Reference< lang::XServiceInfo >*)0 ); } return aTypes; } Sequence SAL_CALL ScDataPilotDescriptorBase::getImplementationId() throw(RuntimeException) { static Sequence< sal_Int8 > aId; if( aId.getLength() == 0 ) { aId.realloc( 16 ); rtl_createUuid( (sal_uInt8 *)aId.getArray(), 0, sal_True ); } return aId; } void ScDataPilotDescriptorBase::Notify( SfxBroadcaster&, const SfxHint& rHint ) { //! Referenz-Update? if ( rHint.ISA( SfxSimpleHint ) && ((const SfxSimpleHint&)rHint).GetId() == SFX_HINT_DYING ) { pDocShell = NULL; // ungueltig geworden } } // XDataPilotDescriptor CellRangeAddress SAL_CALL ScDataPilotDescriptorBase::getSourceRange() throw(RuntimeException) { ScUnoGuard aGuard; ScDPObject* pDPObject(GetDPObject()); if (!pDPObject) throw RuntimeException(); CellRangeAddress aRet; if (pDPObject->IsSheetData()) ScUnoConversion::FillApiRange( aRet, pDPObject->GetSheetDesc()->aSourceRange ); return aRet; } void SAL_CALL ScDataPilotDescriptorBase::setSourceRange( const CellRangeAddress& aSourceRange ) throw(RuntimeException) { ScUnoGuard aGuard; ScDPObject* pDPObject = GetDPObject(); if (!pDPObject) throw RuntimeException(); ScSheetSourceDesc aSheetDesc; if (pDPObject->IsSheetData()) aSheetDesc = *pDPObject->GetSheetDesc(); ScUnoConversion::FillScRange( aSheetDesc.aSourceRange, aSourceRange ); pDPObject->SetSheetDesc( aSheetDesc ); SetDPObject( pDPObject ); } Reference SAL_CALL ScDataPilotDescriptorBase::getFilterDescriptor() throw(RuntimeException) { ScUnoGuard aGuard; return new ScDataPilotFilterDescriptor( pDocShell, this ); } Reference SAL_CALL ScDataPilotDescriptorBase::getDataPilotFields() throw(RuntimeException) { ScUnoGuard aGuard; return new ScDataPilotFieldsObj( *this ); } Reference SAL_CALL ScDataPilotDescriptorBase::getColumnFields() throw(RuntimeException) { ScUnoGuard aGuard; return new ScDataPilotFieldsObj( *this, DataPilotFieldOrientation_COLUMN ); } Reference SAL_CALL ScDataPilotDescriptorBase::getRowFields() throw(RuntimeException) { ScUnoGuard aGuard; return new ScDataPilotFieldsObj( *this, DataPilotFieldOrientation_ROW ); } Reference SAL_CALL ScDataPilotDescriptorBase::getPageFields() throw(RuntimeException) { ScUnoGuard aGuard; return new ScDataPilotFieldsObj( *this, DataPilotFieldOrientation_PAGE ); } Reference SAL_CALL ScDataPilotDescriptorBase::getDataFields() throw(RuntimeException) { ScUnoGuard aGuard; return new ScDataPilotFieldsObj( *this, DataPilotFieldOrientation_DATA ); } Reference SAL_CALL ScDataPilotDescriptorBase::getHiddenFields() throw(RuntimeException) { ScUnoGuard aGuard; return new ScDataPilotFieldsObj( *this, DataPilotFieldOrientation_HIDDEN ); } // XPropertySet Reference< XPropertySetInfo > SAL_CALL ScDataPilotDescriptorBase::getPropertySetInfo( ) throw(RuntimeException) { ScUnoGuard aGuard; static Reference aRef = new SfxItemPropertySetInfo( maPropSet.getPropertyMap() ); return aRef; } void SAL_CALL ScDataPilotDescriptorBase::setPropertyValue( const OUString& aPropertyName, const Any& aValue ) throw(UnknownPropertyException, PropertyVetoException, IllegalArgumentException, WrappedTargetException, RuntimeException) { ScUnoGuard aGuard; ScDPObject* pDPObject = GetDPObject(); if (pDPObject) { ScDPSaveData* pOldData = pDPObject->GetSaveData(); DBG_ASSERT(pOldData, "Here should be a SaveData"); if ( pOldData ) { ScDPSaveData aNewData( *pOldData ); String aNameString = aPropertyName; if ( aNameString.EqualsAscii( SC_UNO_COLGRAND ) ) { aNewData.SetColumnGrand(::cppu::any2bool( aValue )); } else if ( aNameString.EqualsAscii( SC_UNO_IGNEMPROWS ) ) { aNewData.SetIgnoreEmptyRows(::cppu::any2bool( aValue )); } else if ( aNameString.EqualsAscii( SC_UNO_RPTEMPTY ) ) { aNewData.SetRepeatIfEmpty(::cppu::any2bool( aValue )); } else if ( aNameString.EqualsAscii( SC_UNO_ROWGRAND ) ) { aNewData.SetRowGrand(::cppu::any2bool( aValue )); } else if ( aNameString.EqualsAscii( SC_UNO_SHOWFILT ) ) { aNewData.SetFilterButton(::cppu::any2bool( aValue )); } else if ( aNameString.EqualsAscii( SC_UNO_DRILLDOWN ) ) { aNewData.SetDrillDown(::cppu::any2bool( aValue )); } else if ( aNameString.EqualsAscii( SC_UNO_GRANDTOTAL_NAME ) ) { rtl::OUString aStrVal; if ( aValue >>= aStrVal ) aNewData.SetGrandTotalName(aStrVal); } else if ( aNameString.EqualsAscii( SC_UNO_IMPORTDESC ) ) { uno::Sequence aArgSeq; if ( aValue >>= aArgSeq ) { ScImportSourceDesc aImportDesc; const ScImportSourceDesc* pOldDesc = pDPObject->GetImportSourceDesc(); if (pOldDesc) aImportDesc = *pOldDesc; ScImportParam aParam; ScImportDescriptor::FillImportParam( aParam, aArgSeq ); sal_uInt16 nNewType = sheet::DataImportMode_NONE; if ( aParam.bImport ) { if ( aParam.bSql ) nNewType = sheet::DataImportMode_SQL; else if ( aParam.nType == ScDbQuery ) nNewType = sheet::DataImportMode_QUERY; else nNewType = sheet::DataImportMode_TABLE; } aImportDesc.nType = nNewType; aImportDesc.aDBName = aParam.aDBName; aImportDesc.aObject = aParam.aStatement; aImportDesc.bNative = aParam.bNative; pDPObject->SetImportDesc( aImportDesc ); } } else if ( aNameString.EqualsAscii( SC_UNO_SOURCESERV ) ) { rtl::OUString aStrVal; if ( aValue >>= aStrVal ) { String aEmpty; ScDPServiceDesc aServiceDesc(aEmpty, aEmpty, aEmpty, aEmpty, aEmpty); const ScDPServiceDesc* pOldDesc = pDPObject->GetDPServiceDesc(); if (pOldDesc) aServiceDesc = *pOldDesc; aServiceDesc.aServiceName = aStrVal; pDPObject->SetServiceData( aServiceDesc ); } } else if ( aNameString.EqualsAscii( SC_UNO_SERVICEARG ) ) { uno::Sequence aArgSeq; if ( aValue >>= aArgSeq ) { String aEmpty; ScDPServiceDesc aServiceDesc(aEmpty, aEmpty, aEmpty, aEmpty, aEmpty); const ScDPServiceDesc* pOldDesc = pDPObject->GetDPServiceDesc(); if (pOldDesc) aServiceDesc = *pOldDesc; rtl::OUString aStrVal; sal_Int32 nArgs = aArgSeq.getLength(); for (sal_Int32 nArgPos=0; nArgPos>= aStrVal ) aServiceDesc.aParSource = aStrVal; } else if (aPropName.EqualsAscii( SC_UNO_OBJECTNAME )) { if ( rProp.Value >>= aStrVal ) aServiceDesc.aParName = aStrVal; } else if (aPropName.EqualsAscii( SC_UNO_USERNAME )) { if ( rProp.Value >>= aStrVal ) aServiceDesc.aParUser = aStrVal; } else if (aPropName.EqualsAscii( SC_UNO_PASSWORD )) { if ( rProp.Value >>= aStrVal ) aServiceDesc.aParPass = aStrVal; } } pDPObject->SetServiceData( aServiceDesc ); } } else throw UnknownPropertyException(); pDPObject->SetSaveData( aNewData ); } SetDPObject(pDPObject); } } Any SAL_CALL ScDataPilotDescriptorBase::getPropertyValue( const OUString& aPropertyName ) throw(UnknownPropertyException, WrappedTargetException, RuntimeException) { ScUnoGuard aGuard; Any aRet; ScDPObject* pDPObject(GetDPObject()); if (pDPObject) { ScDPSaveData* pOldData = pDPObject->GetSaveData(); DBG_ASSERT(pOldData, "Here should be a SaveData"); if ( pOldData ) { ScDPSaveData aNewData( *pOldData ); String aNameString = aPropertyName; if ( aNameString.EqualsAscii( SC_UNO_COLGRAND ) ) { aRet = ::cppu::bool2any( aNewData.GetColumnGrand() ); } else if ( aNameString.EqualsAscii( SC_UNO_IGNEMPROWS ) ) { aRet = ::cppu::bool2any( aNewData.GetIgnoreEmptyRows() ); } else if ( aNameString.EqualsAscii( SC_UNO_RPTEMPTY ) ) { aRet = ::cppu::bool2any( aNewData.GetRepeatIfEmpty() ); } else if ( aNameString.EqualsAscii( SC_UNO_ROWGRAND ) ) { aRet = ::cppu::bool2any( aNewData.GetRowGrand() ); } else if ( aNameString.EqualsAscii( SC_UNO_SHOWFILT ) ) { aRet = ::cppu::bool2any( aNewData.GetFilterButton() ); } else if ( aNameString.EqualsAscii( SC_UNO_DRILLDOWN ) ) { aRet = ::cppu::bool2any( aNewData.GetDrillDown() ); } else if ( aNameString.EqualsAscii( SC_UNO_GRANDTOTAL_NAME ) ) { const rtl::OUString* pGrandTotalName = aNewData.GetGrandTotalName(); if (pGrandTotalName) aRet <<= *pGrandTotalName; // same behavior as in ScDPSource } else if ( aNameString.EqualsAscii( SC_UNO_IMPORTDESC ) ) { const ScImportSourceDesc* pImportDesc = pDPObject->GetImportSourceDesc(); if ( pImportDesc ) { // fill ScImportParam so ScImportDescriptor::FillProperties can be used ScImportParam aParam; aParam.bImport = ( pImportDesc->nType != sheet::DataImportMode_NONE ); aParam.aDBName = pImportDesc->aDBName; aParam.aStatement = pImportDesc->aObject; aParam.bNative = pImportDesc->bNative; aParam.bSql = ( pImportDesc->nType == sheet::DataImportMode_SQL ); aParam.nType = static_cast(( pImportDesc->nType == sheet::DataImportMode_QUERY ) ? ScDbQuery : ScDbTable); uno::Sequence aSeq( ScImportDescriptor::GetPropertyCount() ); ScImportDescriptor::FillProperties( aSeq, aParam ); aRet <<= aSeq; } else { // empty sequence uno::Sequence aEmpty(0); aRet <<= aEmpty; } } else if ( aNameString.EqualsAscii( SC_UNO_SOURCESERV ) ) { rtl::OUString aServiceName; const ScDPServiceDesc* pServiceDesc = pDPObject->GetDPServiceDesc(); if (pServiceDesc) aServiceName = pServiceDesc->aServiceName; aRet <<= aServiceName; // empty string if no ServiceDesc set } else if ( aNameString.EqualsAscii( SC_UNO_SERVICEARG ) ) { const ScDPServiceDesc* pServiceDesc = pDPObject->GetDPServiceDesc(); if (pServiceDesc) { uno::Sequence aSeq( 4 ); beans::PropertyValue* pArray = aSeq.getArray(); pArray[0].Name = rtl::OUString::createFromAscii( SC_UNO_SOURCENAME ); pArray[0].Value <<= rtl::OUString( pServiceDesc->aParSource ); pArray[1].Name = rtl::OUString::createFromAscii( SC_UNO_OBJECTNAME ); pArray[1].Value <<= rtl::OUString( pServiceDesc->aParName ); pArray[2].Name = rtl::OUString::createFromAscii( SC_UNO_USERNAME ); pArray[2].Value <<= rtl::OUString( pServiceDesc->aParUser ); pArray[3].Name = rtl::OUString::createFromAscii( SC_UNO_PASSWORD ); pArray[3].Value <<= rtl::OUString( pServiceDesc->aParPass ); aRet <<= aSeq; } else { // empty sequence uno::Sequence aEmpty(0); aRet <<= aEmpty; } } else throw UnknownPropertyException(); } } return aRet; } void SAL_CALL ScDataPilotDescriptorBase::addPropertyChangeListener( const OUString& /* aPropertyName */, const Reference& /* xListener */ ) throw(UnknownPropertyException, WrappedTargetException, RuntimeException) { } void SAL_CALL ScDataPilotDescriptorBase::removePropertyChangeListener( const OUString& /* aPropertyName */, const Reference& /* aListener */ ) throw(UnknownPropertyException, WrappedTargetException, RuntimeException) { } void SAL_CALL ScDataPilotDescriptorBase::addVetoableChangeListener( const OUString& /* PropertyName */, const Reference& /* aListener */ ) throw(UnknownPropertyException, WrappedTargetException, RuntimeException) { } void SAL_CALL ScDataPilotDescriptorBase::removeVetoableChangeListener( const OUString& /* PropertyName */, const Reference& /* aListener */ ) throw(UnknownPropertyException, WrappedTargetException, RuntimeException) { } // XDataPilotDataLayoutFieldSupplier Reference< XDataPilotField > SAL_CALL ScDataPilotDescriptorBase::getDataLayoutField() throw(RuntimeException) { ScUnoGuard aGuard; if( ScDPObject* pDPObject = GetDPObject() ) { if( ScDPSaveData* pSaveData = pDPObject->GetSaveData() ) { if( /*ScDPSaveDimension* pDataDim =*/ pSaveData->GetDataLayoutDimension() ) { ScFieldIdentifier aFieldId( OUString( RTL_CONSTASCII_USTRINGPARAM( SC_DATALAYOUT_NAME ) ), 0, true ); return new ScDataPilotFieldObj( *this, aFieldId ); } } } return 0; } // XUnoTunnel sal_Int64 SAL_CALL ScDataPilotDescriptorBase::getSomething( const Sequence& rId ) throw(RuntimeException) { if ( rId.getLength() == 16 && 0 == rtl_compareMemory( getUnoTunnelId().getConstArray(), rId.getConstArray(), 16 ) ) { return sal::static_int_cast(reinterpret_cast(this)); } return 0; } // static const Sequence& ScDataPilotDescriptorBase::getUnoTunnelId() { static Sequence * pSeq = 0; if( !pSeq ) { osl::Guard< osl::Mutex > aGuard( osl::Mutex::getGlobalMutex() ); if( !pSeq ) { static Sequence< sal_Int8 > aSeq( 16 ); rtl_createUuid( (sal_uInt8*)aSeq.getArray(), 0, sal_True ); pSeq = &aSeq; } } return *pSeq; } // static ScDataPilotDescriptorBase* ScDataPilotDescriptorBase::getImplementation( const Reference xObj ) { ScDataPilotDescriptorBase* pRet = NULL; Reference xUT( xObj, UNO_QUERY ); if (xUT.is()) pRet = reinterpret_cast(sal::static_int_cast(xUT->getSomething(getUnoTunnelId()))); return pRet; } //------------------------------------------------------------------------ ScDataPilotTableObj::ScDataPilotTableObj(ScDocShell* pDocSh, SCTAB nT, const String& rN) : ScDataPilotDescriptorBase( pDocSh ), nTab( nT ), aName( rN ), aModifyListeners( 0 ) { } ScDataPilotTableObj::~ScDataPilotTableObj() { } Any SAL_CALL ScDataPilotTableObj::queryInterface( const uno::Type& rType ) throw(RuntimeException) { // since we manually do resolve the query for XDataPilotTable2 // we also need to do the same for XDataPilotTable SC_QUERYINTERFACE( XDataPilotTable ) SC_QUERYINTERFACE( XDataPilotTable2 ) SC_QUERYINTERFACE( XModifyBroadcaster ) return ScDataPilotDescriptorBase::queryInterface( rType ); } void SAL_CALL ScDataPilotTableObj::acquire() throw() { ScDataPilotDescriptorBase::acquire(); } void SAL_CALL ScDataPilotTableObj::release() throw() { ScDataPilotDescriptorBase::release(); } Sequence< uno::Type > SAL_CALL ScDataPilotTableObj::getTypes() throw(RuntimeException) { static Sequence< uno::Type > aTypes; if ( aTypes.getLength() == 0 ) { Sequence< uno::Type > aParentTypes = ScDataPilotDescriptorBase::getTypes(); sal_Int32 nParentLen = aParentTypes.getLength(); const uno::Type* pParentPtr = aParentTypes.getConstArray(); aTypes.realloc( nParentLen + 2 ); uno::Type* pPtr = aTypes.getArray(); for (sal_Int32 i = 0; i < nParentLen; ++i) pPtr[ i ] = pParentPtr[ i ]; // parent types first pPtr[ nParentLen ] = getCppuType( (const Reference< XDataPilotTable2 >*)0 ); pPtr[ nParentLen+1 ] = getCppuType( (const Reference< XModifyBroadcaster >*)0 ); } return aTypes; } Sequence SAL_CALL ScDataPilotTableObj::getImplementationId() throw(RuntimeException) { static Sequence< sal_Int8 > aId; if( aId.getLength() == 0 ) { aId.realloc( 16 ); rtl_createUuid( (sal_uInt8 *)aId.getArray(), 0, sal_True ); } return aId; } // --- ScDPObject* ScDataPilotTableObj::GetDPObject() const { return lcl_GetDPObject(GetDocShell(), nTab, aName); } void ScDataPilotTableObj::SetDPObject( ScDPObject* pDPObject ) { ScDocShell* pDocSh = GetDocShell(); ScDPObject* pDPObj = lcl_GetDPObject(pDocSh, nTab, aName); if ( pDPObj && pDocSh ) { ScDBDocFunc aFunc(*pDocSh); aFunc.DataPilotUpdate( pDPObj, pDPObject, sal_True, sal_True ); } } // "rest of XDataPilotDescriptor" OUString SAL_CALL ScDataPilotTableObj::getName() throw(RuntimeException) { ScUnoGuard aGuard; ScDPObject* pDPObj = lcl_GetDPObject(GetDocShell(), nTab, aName); if (pDPObj) return pDPObj->GetName(); return OUString(); } void SAL_CALL ScDataPilotTableObj::setName( const OUString& aNewName ) throw(RuntimeException) { ScUnoGuard aGuard; ScDPObject* pDPObj = lcl_GetDPObject(GetDocShell(), nTab, aName); if (pDPObj) { //! test for existing names !!! String aString(aNewName); pDPObj->SetName( aString ); //! Undo - DBDocFunc ??? aName = aString; // DataPilotUpdate would do too much (output table is not changed) GetDocShell()->SetDocumentModified(); } } OUString SAL_CALL ScDataPilotTableObj::getTag() throw(RuntimeException) { ScUnoGuard aGuard; ScDPObject* pDPObj = lcl_GetDPObject(GetDocShell(), nTab, aName); if (pDPObj) return pDPObj->GetTag(); return OUString(); } void SAL_CALL ScDataPilotTableObj::setTag( const OUString& aNewTag ) throw(RuntimeException) { ScUnoGuard aGuard; ScDPObject* pDPObj = lcl_GetDPObject(GetDocShell(), nTab, aName); if (pDPObj) { String aString(aNewTag); pDPObj->SetTag( aString ); //! Undo - DBDocFunc ??? // DataPilotUpdate would do too much (output table is not changed) GetDocShell()->SetDocumentModified(); } } // XDataPilotTable CellRangeAddress SAL_CALL ScDataPilotTableObj::getOutputRange() throw(RuntimeException) { ScUnoGuard aGuard; CellRangeAddress aRet; ScDPObject* pDPObj = lcl_GetDPObject(GetDocShell(), nTab, aName); if (pDPObj) { ScRange aRange(pDPObj->GetOutRange()); aRet.Sheet = aRange.aStart.Tab(); aRet.StartColumn = aRange.aStart.Col(); aRet.StartRow = aRange.aStart.Row(); aRet.EndColumn = aRange.aEnd.Col(); aRet.EndRow = aRange.aEnd.Row(); } return aRet; } sal_uLong RefreshDPObject( ScDPObject *pDPObj, ScDocument *pDoc, ScDocShell *pDocSh, sal_Bool bRecord, sal_Bool bApi ); void SAL_CALL ScDataPilotTableObj::refresh() throw(RuntimeException) { ScUnoGuard aGuard; if( ScDPObject* pDPObj = lcl_GetDPObject(GetDocShell(), nTab, aName) ) RefreshDPObject( pDPObj, NULL, GetDocShell(), sal_True, sal_True ); //if (pDPObj) //{ // ScDPObject* pNew = new ScDPObject(*pDPObj); // ScDBDocFunc aFunc(*GetDocShell()); // aFunc.DataPilotUpdate( pDPObj, pNew, sal_True, sal_True ); // delete pNew; // DataPilotUpdate copies settings from "new" object //} } Sequence< Sequence > SAL_CALL ScDataPilotTableObj::getDrillDownData(const CellAddress& aAddr) throw (RuntimeException) { ScUnoGuard aGuard; Sequence< Sequence > aTabData; ScAddress aAddr2(static_cast(aAddr.Column), static_cast(aAddr.Row), aAddr.Sheet); ScDPObject* pObj = GetDPObject(); if (!pObj) throw RuntimeException(); pObj->GetDrillDownData(aAddr2, aTabData); return aTabData; } DataPilotTablePositionData SAL_CALL ScDataPilotTableObj::getPositionData(const CellAddress& aAddr) throw (RuntimeException) { ScUnoGuard aGuard; DataPilotTablePositionData aPosData; ScAddress aAddr2(static_cast(aAddr.Column), static_cast(aAddr.Row), aAddr.Sheet); ScDPObject* pObj = GetDPObject(); if (!pObj) throw RuntimeException(); pObj->GetPositionData(aAddr2, aPosData); return aPosData; } void SAL_CALL ScDataPilotTableObj::insertDrillDownSheet(const CellAddress& aAddr) throw (RuntimeException) { ScUnoGuard aGuard; ScDPObject* pDPObj = GetDPObject(); if (!pDPObj) throw RuntimeException(); Sequence aFilters; pDPObj->GetDataFieldPositionData( ScAddress(static_cast(aAddr.Column), static_cast(aAddr.Row), aAddr.Sheet), aFilters); GetDocShell()->GetBestViewShell()->ShowDataPilotSourceData(*pDPObj, aFilters); } CellRangeAddress SAL_CALL ScDataPilotTableObj::getOutputRangeByType( sal_Int32 nType ) throw (IllegalArgumentException, RuntimeException) { ScUnoGuard aGuard; if (nType < 0 || nType > DataPilotOutputRangeType::RESULT) throw IllegalArgumentException(); CellRangeAddress aRet; if (ScDPObject* pDPObj = lcl_GetDPObject(GetDocShell(), nTab, aName)) ScUnoConversion::FillApiRange( aRet, pDPObj->GetOutputRangeByType( nType ) ); return aRet; } void SAL_CALL ScDataPilotTableObj::addModifyListener( const uno::Reference& aListener ) throw (uno::RuntimeException) { ScUnoGuard aGuard; uno::Reference *pObj = new uno::Reference( aListener ); aModifyListeners.Insert( pObj, aModifyListeners.Count() ); if ( aModifyListeners.Count() == 1 ) { acquire(); // don't lose this object (one ref for all listeners) } } void SAL_CALL ScDataPilotTableObj::removeModifyListener( const uno::Reference& aListener ) throw (uno::RuntimeException) { ScUnoGuard aGuard; acquire(); // in case the listeners have the last ref - released below sal_uInt16 nCount = aModifyListeners.Count(); for ( sal_uInt16 n=nCount; n--; ) { uno::Reference *pObj = aModifyListeners[n]; if ( *pObj == aListener ) { aModifyListeners.DeleteAndDestroy( n ); if ( aModifyListeners.Count() == 0 ) { release(); // release the ref for the listeners } break; } } release(); // might delete this object } void ScDataPilotTableObj::Notify( SfxBroadcaster& rBC, const SfxHint& rHint ) { if ( rHint.ISA(ScDataPilotModifiedHint) && static_cast(rHint).GetName() == aName ) { Refreshed_Impl(); } else if ( rHint.ISA( ScUpdateRefHint ) ) { ScRange aRange( 0, 0, nTab ); ScRangeList aRanges; aRanges.Append( aRange ); const ScUpdateRefHint& rRef = static_cast< const ScUpdateRefHint& >( rHint ); if ( aRanges.UpdateReference( rRef.GetMode(), GetDocShell()->GetDocument(), rRef.GetRange(), rRef.GetDx(), rRef.GetDy(), rRef.GetDz() ) && aRanges.Count() == 1 ) { const ScRange* pRange = aRanges.GetObject( 0 ); if ( pRange ) { nTab = pRange->aStart.Tab(); } } } ScDataPilotDescriptorBase::Notify( rBC, rHint ); } void ScDataPilotTableObj::Refreshed_Impl() { lang::EventObject aEvent; aEvent.Source.set((cppu::OWeakObject*)this); // the EventObject holds a Ref to this object until after the listener calls ScDocument* pDoc = GetDocShell()->GetDocument(); for ( sal_uInt16 n=0; nAddUnoListenerCall( *aModifyListeners[n], aEvent ); } // ============================================================================ ScDataPilotDescriptor::ScDataPilotDescriptor(ScDocShell* pDocSh) : ScDataPilotDescriptorBase( pDocSh ), mpDPObject(new ScDPObject(pDocSh ? pDocSh->GetDocument() : NULL) ) { mpDPObject->SetAlive(sal_True); ScDPSaveData aSaveData; // set defaults like in ScPivotParam constructor aSaveData.SetColumnGrand( sal_True ); aSaveData.SetRowGrand( sal_True ); aSaveData.SetIgnoreEmptyRows( sal_False ); aSaveData.SetRepeatIfEmpty( sal_False ); mpDPObject->SetSaveData(aSaveData); ScSheetSourceDesc aSheetDesc; mpDPObject->SetSheetDesc(aSheetDesc); mpDPObject->GetSource(); } ScDataPilotDescriptor::~ScDataPilotDescriptor() { delete mpDPObject; } ScDPObject* ScDataPilotDescriptor::GetDPObject() const { return mpDPObject; } void ScDataPilotDescriptor::SetDPObject( ScDPObject* pDPObject ) { if (mpDPObject != pDPObject) { delete mpDPObject; mpDPObject = pDPObject; DBG_ERROR("replace DPObject should not happen"); } } // "rest of XDataPilotDescriptor" OUString SAL_CALL ScDataPilotDescriptor::getName() throw(RuntimeException) { ScUnoGuard aGuard; return mpDPObject->GetName(); } void SAL_CALL ScDataPilotDescriptor::setName( const OUString& aNewName ) throw(RuntimeException) { ScUnoGuard aGuard; mpDPObject->SetName( aNewName ); } OUString SAL_CALL ScDataPilotDescriptor::getTag() throw(RuntimeException) { ScUnoGuard aGuard; return mpDPObject->GetTag(); } void SAL_CALL ScDataPilotDescriptor::setTag( const OUString& aNewTag ) throw(RuntimeException) { ScUnoGuard aGuard; mpDPObject->SetTag( aNewTag ); } // ============================================================================ ScDataPilotChildObjBase::ScDataPilotChildObjBase( ScDataPilotDescriptorBase& rParent ) : mrParent( rParent ) { mrParent.acquire(); } ScDataPilotChildObjBase::ScDataPilotChildObjBase( ScDataPilotDescriptorBase& rParent, const ScFieldIdentifier& rFieldId ) : mrParent( rParent ), maFieldId( rFieldId ) { mrParent.acquire(); } ScDataPilotChildObjBase::~ScDataPilotChildObjBase() { mrParent.release(); } ScDPObject* ScDataPilotChildObjBase::GetDPObject() const { return mrParent.GetDPObject(); } void ScDataPilotChildObjBase::SetDPObject( ScDPObject* pDPObject ) { mrParent.SetDPObject( pDPObject ); } ScDPSaveDimension* ScDataPilotChildObjBase::GetDPDimension( ScDPObject** ppDPObject ) const { if( ScDPObject* pDPObj = GetDPObject() ) { if( ppDPObject ) *ppDPObject = pDPObj; if( ScDPSaveData* pSaveData = pDPObj->GetSaveData() ) { if( maFieldId.mbDataLayout ) return pSaveData->GetDataLayoutDimension(); if( maFieldId.mnFieldIdx == 0 ) return pSaveData->GetDimensionByName( maFieldId.maFieldName ); // find dimension with specified index (search in duplicated dimensions) String aFieldName = maFieldId.maFieldName; // needed for comparison const List& rDimensions = pSaveData->GetDimensions(); sal_uLong nDimCount = rDimensions.Count(); sal_Int32 nFoundIdx = 0; for( sal_uLong nDim = 0; nDim < nDimCount; ++nDim ) { ScDPSaveDimension* pDim = static_cast< ScDPSaveDimension* >( rDimensions.GetObject( nDim ) ); if( !pDim->IsDataLayout() && (pDim->GetName() == aFieldName) ) { if( nFoundIdx == maFieldId.mnFieldIdx ) return pDim; ++nFoundIdx; } } } } return 0; } sal_Int32 ScDataPilotChildObjBase::GetMemberCount() const { sal_Int32 nRet = 0; Reference xMembersNA = GetMembers(); if (xMembersNA.is()) { Reference< XIndexAccess > xMembersIA( new ScNameToIndexAccess( xMembersNA ) ); nRet = xMembersIA->getCount(); } return nRet; } Reference< XNameAccess > ScDataPilotChildObjBase::GetMembers() const { Reference< XNameAccess > xMembersNA; if( ScDPObject* pDPObj = GetDPObject() ) pDPObj->GetMembersNA( lcl_GetObjectIndex( pDPObj, maFieldId ), xMembersNA ); return xMembersNA; } // ============================================================================ ScDataPilotFieldsObj::ScDataPilotFieldsObj( ScDataPilotDescriptorBase& rParent ) : ScDataPilotChildObjBase( rParent ) { } ScDataPilotFieldsObj::ScDataPilotFieldsObj( ScDataPilotDescriptorBase& rParent, DataPilotFieldOrientation eOrient ) : ScDataPilotChildObjBase( rParent ), maOrient( eOrient ) { } ScDataPilotFieldsObj::~ScDataPilotFieldsObj() { } sal_Int32 lcl_GetFieldCount( const Reference& rSource, const Any& rOrient ) { sal_Int32 nRet = 0; Reference xDimsName(rSource->getDimensions()); Reference xIntDims(new ScNameToIndexAccess( xDimsName )); sal_Int32 nIntCount = xIntDims->getCount(); if (rOrient.hasValue()) { // all fields of the specified orientation, including duplicated Reference xDim; for (sal_Int32 i = 0; i < nIntCount; ++i) { xDim.set(xIntDims->getByIndex(i), UNO_QUERY); if (xDim.is() && (xDim->getPropertyValue(OUString(RTL_CONSTASCII_USTRINGPARAM(SC_UNO_ORIENTAT))) == rOrient)) ++nRet; } } else { // count all non-duplicated fields Reference xDim; for (sal_Int32 i = 0; i < nIntCount; ++i) { xDim.set(xIntDims->getByIndex(i), UNO_QUERY); if ( xDim.is() && !lcl_IsDuplicated( xDim ) ) ++nRet; } } return nRet; } sal_Bool lcl_GetFieldDataByIndex( const Reference& rSource, const Any& rOrient, SCSIZE nIndex, ScFieldIdentifier& rFieldId ) { sal_Bool bOk = sal_False; SCSIZE nPos = 0; sal_Int32 nDimIndex = 0; Reference xDimsName(rSource->getDimensions()); Reference xIntDims(new ScNameToIndexAccess( xDimsName )); sal_Int32 nIntCount = xIntDims->getCount(); Reference xDim; if (rOrient.hasValue()) { sal_Int32 i = 0; while (i < nIntCount && !bOk) { xDim.set(xIntDims->getByIndex(i), UNO_QUERY); if (xDim.is() && (xDim->getPropertyValue(OUString(RTL_CONSTASCII_USTRINGPARAM(SC_UNO_ORIENTAT))) == rOrient)) { if (nPos == nIndex) { bOk = sal_True; nDimIndex = i; } else ++nPos; } ++i; } } else { sal_Int32 i = 0; while (i < nIntCount && !bOk) { xDim.set(xIntDims->getByIndex(i), UNO_QUERY); if ( xDim.is() && !lcl_IsDuplicated( xDim ) ) { if (nPos == nIndex) { bOk = sal_True; nDimIndex = i; } else ++nPos; } ++i; } } if ( bOk ) { xDim.set( xIntDims->getByIndex(nDimIndex), UNO_QUERY ); Reference xDimName( xDim, UNO_QUERY ); if ( xDimName.is() ) { OUString sOriginalName( lcl_GetOriginalName( xDimName ) ); rFieldId.maFieldName = sOriginalName; rFieldId.mbDataLayout = ScUnoHelpFunctions::GetBoolProperty( xDim, OUString(RTL_CONSTASCII_USTRINGPARAM(SC_UNO_ISDATALA)) ); sal_Int32 nRepeat = 0; if ( rOrient.hasValue() && lcl_IsDuplicated( xDim ) ) { // find the repeat count // (this relies on the original dimension always being before the duplicates) Reference xPrevName; for (sal_Int32 i = 0; i < nDimIndex; ++i) { xPrevName.set( xIntDims->getByIndex(i), UNO_QUERY ); if ( xPrevName.is() && lcl_GetOriginalName( xPrevName ) == sOriginalName ) ++nRepeat; } } rFieldId.mnFieldIdx = nRepeat; } else bOk = sal_False; } return bOk; } sal_Bool lcl_GetFieldDataByName( ScDPObject* pDPObj, const OUString& rFieldName, ScFieldIdentifier& rFieldId ) { // "By name" is always the first match. // The name "Data" always refers to the data layout field. rFieldId.maFieldName = rFieldName; rFieldId.mnFieldIdx = 0; rFieldId.mbDataLayout = rFieldName.equalsAscii( SC_DATALAYOUT_NAME ); pDPObj->GetSource(); // IsDimNameInUse doesn't update source data // check if the named field exists (not for data layout) return rFieldId.mbDataLayout || pDPObj->IsDimNameInUse( rFieldName ); } // XDataPilotFields ScDataPilotFieldObj* ScDataPilotFieldsObj::GetObjectByIndex_Impl( sal_Int32 nIndex ) const { // TODO if (ScDPObject* pObj = GetDPObject()) { ScFieldIdentifier aFieldId; if (lcl_GetFieldDataByIndex( pObj->GetSource(), maOrient, nIndex, aFieldId )) return new ScDataPilotFieldObj( mrParent, aFieldId, maOrient ); } return 0; } ScDataPilotFieldObj* ScDataPilotFieldsObj::GetObjectByName_Impl(const OUString& aName) const { if (ScDPObject* pDPObj = GetDPObject()) { ScFieldIdentifier aFieldId; if (lcl_GetFieldDataByName( pDPObj, aName, aFieldId )) return new ScDataPilotFieldObj( mrParent, aFieldId, maOrient ); } return 0; } // XEnumerationAccess Reference SAL_CALL ScDataPilotFieldsObj::createEnumeration() throw(RuntimeException) { ScUnoGuard aGuard; return new ScIndexEnumeration(this, OUString(RTL_CONSTASCII_USTRINGPARAM("com.sun.star.sheet.DataPilotFieldsEnumeration"))); } // XIndexAccess sal_Int32 SAL_CALL ScDataPilotFieldsObj::getCount() throw(RuntimeException) { ScUnoGuard aGuard; // TODO ScDPObject* pDPObj = GetDPObject(); return pDPObj ? lcl_GetFieldCount( pDPObj->GetSource(), maOrient ) : 0; } Any SAL_CALL ScDataPilotFieldsObj::getByIndex( sal_Int32 nIndex ) throw(IndexOutOfBoundsException, WrappedTargetException, RuntimeException) { ScUnoGuard aGuard; Reference< XPropertySet > xField( GetObjectByIndex_Impl( nIndex ) ); if (!xField.is()) throw IndexOutOfBoundsException(); return Any( xField ); } uno::Type SAL_CALL ScDataPilotFieldsObj::getElementType() throw(RuntimeException) { ScUnoGuard aGuard; return getCppuType((Reference*)0); } sal_Bool SAL_CALL ScDataPilotFieldsObj::hasElements() throw(RuntimeException) { ScUnoGuard aGuard; return ( getCount() != 0 ); } Any SAL_CALL ScDataPilotFieldsObj::getByName( const OUString& aName ) throw(NoSuchElementException, WrappedTargetException, RuntimeException) { ScUnoGuard aGuard; Reference xField(GetObjectByName_Impl(aName)); if (!xField.is()) throw NoSuchElementException(); return Any( xField ); } Sequence SAL_CALL ScDataPilotFieldsObj::getElementNames() throw(RuntimeException) { ScUnoGuard aGuard; // TODO if (ScDPObject* pDPObj = GetDPObject()) { Sequence< OUString > aSeq( lcl_GetFieldCount( pDPObj->GetSource(), maOrient ) ); OUString* pAry = aSeq.getArray(); const List& rDimensions = pDPObj->GetSaveData()->GetDimensions(); sal_Int32 nDimCount = rDimensions.Count(); for (sal_Int32 nDim = 0; nDim < nDimCount; nDim++) { ScDPSaveDimension* pDim = (ScDPSaveDimension*)rDimensions.GetObject(nDim); if(maOrient.hasValue() && (pDim->GetOrientation() == maOrient.get< DataPilotFieldOrientation >())) { *pAry = pDim->GetName(); ++pAry; } } return aSeq; } return Sequence(); } sal_Bool SAL_CALL ScDataPilotFieldsObj::hasByName( const OUString& aName ) throw(RuntimeException) { ScUnoGuard aGuard; return GetObjectByName_Impl(aName) != NULL; } //------------------------------------------------------------------------ ScDataPilotFieldObj::ScDataPilotFieldObj( ScDataPilotDescriptorBase& rParent, const ScFieldIdentifier& rFieldId ) : ScDataPilotChildObjBase( rParent, rFieldId ), maPropSet( lcl_GetDataPilotFieldMap() ) { } ScDataPilotFieldObj::ScDataPilotFieldObj( ScDataPilotDescriptorBase& rParent, const ScFieldIdentifier& rFieldId, const Any& rOrient ) : ScDataPilotChildObjBase( rParent, rFieldId ), maPropSet( lcl_GetDataPilotFieldMap() ), maOrient( rOrient ) { } ScDataPilotFieldObj::~ScDataPilotFieldObj() { } // XNamed OUString SAL_CALL ScDataPilotFieldObj::getName() throw(RuntimeException) { ScUnoGuard aGuard; OUString aName; if( ScDPSaveDimension* pDim = GetDPDimension() ) { if( pDim->IsDataLayout() ) aName = OUString( RTL_CONSTASCII_USTRINGPARAM( SC_DATALAYOUT_NAME ) ); else { const rtl::OUString* pLayoutName = pDim->GetLayoutName(); if (pLayoutName) aName = *pLayoutName; else aName = pDim->GetName(); } } return aName; } void SAL_CALL ScDataPilotFieldObj::setName( const OUString& rName ) throw(RuntimeException) { ScUnoGuard aGuard; ScDPObject* pDPObj = 0; ScDPSaveDimension* pDim = GetDPDimension( &pDPObj ); if( pDim && !pDim->IsDataLayout() ) { String aName( rName ); pDim->SetLayoutName(aName); SetDPObject( pDPObj ); } } // XPropertySet Reference SAL_CALL ScDataPilotFieldObj::getPropertySetInfo() throw(RuntimeException) { ScUnoGuard aGuard; static Reference aRef( new SfxItemPropertySetInfo( maPropSet.getPropertyMap() )); return aRef; } void SAL_CALL ScDataPilotFieldObj::setPropertyValue( const OUString& aPropertyName, const Any& aValue ) throw(UnknownPropertyException, PropertyVetoException, IllegalArgumentException, WrappedTargetException, RuntimeException) { ScUnoGuard aGuard; String aNameString(aPropertyName); if ( aNameString.EqualsAscii( SC_UNONAME_FUNCTION ) ) { // #i109350# use GetEnumFromAny because it also allows sal_Int32 GeneralFunction eFunction = (GeneralFunction) ScUnoHelpFunctions::GetEnumFromAny( aValue ); setFunction( eFunction ); } else if ( aNameString.EqualsAscii( SC_UNONAME_SUBTOTALS ) ) { Sequence< GeneralFunction > aSubtotals; if( aValue >>= aSubtotals ) setSubtotals( aSubtotals ); } else if ( aNameString.EqualsAscii( SC_UNONAME_ORIENT ) ) { //! test for correct enum type? DataPilotFieldOrientation eOrient = (DataPilotFieldOrientation) ScUnoHelpFunctions::GetEnumFromAny( aValue ); setOrientation( eOrient ); } else if ( aNameString.EqualsAscii( SC_UNONAME_SELPAGE ) ) { OUString sCurrentPage; if (aValue >>= sCurrentPage) setCurrentPage(sCurrentPage); } else if ( aNameString.EqualsAscii( SC_UNONAME_USESELPAGE ) ) { setUseCurrentPage(cppu::any2bool(aValue)); } else if ( aNameString.EqualsAscii( SC_UNONAME_HASAUTOSHOW ) ) { if (!cppu::any2bool(aValue)) setAutoShowInfo(NULL); } else if ( aNameString.EqualsAscii( SC_UNONAME_AUTOSHOW ) ) { DataPilotFieldAutoShowInfo aInfo; if (aValue >>= aInfo) setAutoShowInfo(&aInfo); } else if ( aNameString.EqualsAscii( SC_UNONAME_HASLAYOUTINFO ) ) { if (!cppu::any2bool(aValue)) setLayoutInfo(NULL); } else if ( aNameString.EqualsAscii( SC_UNONAME_LAYOUTINFO ) ) { DataPilotFieldLayoutInfo aInfo; if (aValue >>= aInfo) setLayoutInfo(&aInfo); } else if ( aNameString.EqualsAscii( SC_UNONAME_HASREFERENCE ) ) { if (!cppu::any2bool(aValue)) setReference(NULL); } else if ( aNameString.EqualsAscii( SC_UNONAME_REFERENCE ) ) { DataPilotFieldReference aRef; if (aValue >>= aRef) setReference(&aRef); } else if ( aNameString.EqualsAscii( SC_UNONAME_HASSORTINFO ) ) { if (!cppu::any2bool(aValue)) setSortInfo(NULL); } else if ( aNameString.EqualsAscii( SC_UNONAME_SORTINFO ) ) { DataPilotFieldSortInfo aInfo; if (aValue >>= aInfo) setSortInfo(&aInfo); } else if ( aNameString.EqualsAscii( SC_UNONAME_ISGROUP ) ) { if (!cppu::any2bool(aValue)) setGroupInfo(NULL); } else if ( aNameString.EqualsAscii( SC_UNONAME_GROUPINFO ) ) { DataPilotFieldGroupInfo aInfo; if (aValue >>= aInfo) setGroupInfo(&aInfo); } else if ( aNameString.EqualsAscii( SC_UNONAME_SHOWEMPTY ) ) { setShowEmpty(cppu::any2bool(aValue)); } } Any SAL_CALL ScDataPilotFieldObj::getPropertyValue( const OUString& aPropertyName ) throw(UnknownPropertyException, WrappedTargetException, RuntimeException) { ScUnoGuard aGuard; String aNameString(aPropertyName); Any aRet; if ( aNameString.EqualsAscii( SC_UNONAME_FUNCTION ) ) aRet <<= getFunction(); else if ( aNameString.EqualsAscii( SC_UNONAME_SUBTOTALS ) ) aRet <<= getSubtotals(); else if ( aNameString.EqualsAscii( SC_UNONAME_ORIENT ) ) aRet <<= getOrientation(); else if ( aNameString.EqualsAscii( SC_UNONAME_SELPAGE ) ) aRet <<= getCurrentPage(); else if ( aNameString.EqualsAscii( SC_UNONAME_USESELPAGE ) ) aRet <<= getUseCurrentPage(); else if ( aNameString.EqualsAscii( SC_UNONAME_HASAUTOSHOW ) ) aRet = ::cppu::bool2any(getAutoShowInfo() != NULL); else if ( aNameString.EqualsAscii( SC_UNONAME_AUTOSHOW ) ) { const DataPilotFieldAutoShowInfo* pInfo = getAutoShowInfo(); if (pInfo) aRet <<= DataPilotFieldAutoShowInfo(*pInfo); } else if ( aNameString.EqualsAscii( SC_UNONAME_HASLAYOUTINFO ) ) aRet = ::cppu::bool2any(getLayoutInfo() != NULL); else if ( aNameString.EqualsAscii( SC_UNONAME_LAYOUTINFO ) ) { const DataPilotFieldLayoutInfo* pInfo = getLayoutInfo(); if (pInfo) aRet <<= DataPilotFieldLayoutInfo(*pInfo); } else if ( aNameString.EqualsAscii( SC_UNONAME_HASREFERENCE ) ) aRet = ::cppu::bool2any(getReference() != NULL); else if ( aNameString.EqualsAscii( SC_UNONAME_REFERENCE ) ) { const DataPilotFieldReference* pRef = getReference(); if (pRef) aRet <<= DataPilotFieldReference(*pRef); } else if ( aNameString.EqualsAscii( SC_UNONAME_HASSORTINFO ) ) aRet = ::cppu::bool2any(getSortInfo() != NULL); else if ( aNameString.EqualsAscii( SC_UNONAME_SORTINFO ) ) { const DataPilotFieldSortInfo* pInfo = getSortInfo(); if (pInfo) aRet <<= DataPilotFieldSortInfo(*pInfo); } else if ( aNameString.EqualsAscii( SC_UNONAME_ISGROUP ) ) aRet = ::cppu::bool2any(hasGroupInfo()); else if ( aNameString.EqualsAscii( SC_UNONAME_GROUPINFO ) ) { aRet <<= getGroupInfo(); } else if ( aNameString.EqualsAscii( SC_UNONAME_SHOWEMPTY ) ) aRet <<= getShowEmpty(); return aRet; } // XDatePilotField Reference SAL_CALL ScDataPilotFieldObj::getItems() throw (RuntimeException) { ScUnoGuard aGuard; if (!mxItems.is()) mxItems.set( new ScDataPilotItemsObj( mrParent, maFieldId ) ); return mxItems; } SC_IMPL_DUMMY_PROPERTY_LISTENER( ScDataPilotFieldObj ) DataPilotFieldOrientation ScDataPilotFieldObj::getOrientation() const { ScUnoGuard aGuard; ScDPSaveDimension* pDim = GetDPDimension(); return pDim ? static_cast< DataPilotFieldOrientation >( pDim->GetOrientation() ) : DataPilotFieldOrientation_HIDDEN; } void ScDataPilotFieldObj::setOrientation(DataPilotFieldOrientation eNew) { ScUnoGuard aGuard; if (maOrient.hasValue() && (eNew == maOrient.get< DataPilotFieldOrientation >())) return; ScDPObject* pDPObj = 0; if( ScDPSaveDimension* pDim = GetDPDimension( &pDPObj ) ) { ScDPSaveData* pSaveData = pDPObj->GetSaveData(); /* If the field was taken from getDataPilotFields(), don't reset the orientation for an existing use, but create a duplicated field instead (for "Data" orientation only). */ if ( !maOrient.hasValue() && !maFieldId.mbDataLayout && (pDim->GetOrientation() != DataPilotFieldOrientation_HIDDEN) && (eNew == DataPilotFieldOrientation_DATA) ) { ScDPSaveDimension* pNewDim = 0; // look for existing duplicate with orientation "hidden" String aNameStr( maFieldId.maFieldName ); const List& rDimensions = pSaveData->GetDimensions(); sal_Int32 nDimCount = rDimensions.Count(); sal_Int32 nFound = 0; for ( sal_Int32 nDim = 0; nDim < nDimCount && !pNewDim; nDim++ ) { ScDPSaveDimension* pOneDim = static_cast(rDimensions.GetObject(nDim)); if ( !pOneDim->IsDataLayout() && (pOneDim->GetName() == aNameStr) ) { if ( pOneDim->GetOrientation() == DataPilotFieldOrientation_HIDDEN ) pNewDim = pOneDim; // use this one else ++nFound; // count existing non-hidden occurences } } if ( !pNewDim ) // if none found, create a new duplicated dimension pNewDim = &pSaveData->DuplicateDimension( *pDim ); maFieldId.mnFieldIdx = nFound; // keep accessing the new one pDim = pNewDim; } pDim->SetOrientation(sal::static_int_cast(eNew)); // move changed field behind all other fields (make it the last field in dimension) pSaveData->SetPosition( pDim, pSaveData->GetDimensions().Count() ); SetDPObject( pDPObj ); maOrient <<= eNew; // modifying the same object's orientation again doesn't create another duplicate } } GeneralFunction ScDataPilotFieldObj::getFunction() const { ScUnoGuard aGuard; GeneralFunction eRet = GeneralFunction_NONE; if( ScDPSaveDimension* pDim = GetDPDimension() ) { if( pDim->GetOrientation() != DataPilotFieldOrientation_DATA ) { // for non-data fields, property Function is the subtotals long nSubCount = pDim->GetSubTotalsCount(); if ( nSubCount > 0 ) eRet = (GeneralFunction)pDim->GetSubTotalFunc(0); // always use the first one // else keep NONE } else eRet = (GeneralFunction)pDim->GetFunction(); } return eRet; } void ScDataPilotFieldObj::setFunction(GeneralFunction eNewFunc) { ScUnoGuard aGuard; ScDPObject* pDPObj = 0; if( ScDPSaveDimension* pDim = GetDPDimension( &pDPObj ) ) { if( pDim->GetOrientation() != DataPilotFieldOrientation_DATA ) { // for non-data fields, property Function is the subtotals if ( eNewFunc == GeneralFunction_NONE ) pDim->SetSubTotals( 0, NULL ); else { sal_uInt16 nFunc = sal::static_int_cast( eNewFunc ); pDim->SetSubTotals( 1, &nFunc ); } } else pDim->SetFunction( sal::static_int_cast( eNewFunc ) ); SetDPObject( pDPObj ); } } Sequence< GeneralFunction > ScDataPilotFieldObj::getSubtotals() const { ScUnoGuard aGuard; Sequence< GeneralFunction > aRet; if( ScDPSaveDimension* pDim = GetDPDimension() ) { if( pDim->GetOrientation() != DataPilotFieldOrientation_DATA ) { // for non-data fields, property Functions is the sequence of subtotals sal_Int32 nCount = static_cast< sal_Int32 >( pDim->GetSubTotalsCount() ); if ( nCount > 0 ) { aRet.realloc( nCount ); for( sal_Int32 nIdx = 0; nIdx < nCount; ++nIdx ) aRet[ nIdx ] = (GeneralFunction)pDim->GetSubTotalFunc( nIdx ); } } } return aRet; } void ScDataPilotFieldObj::setSubtotals( const Sequence< GeneralFunction >& rSubtotals ) { ScUnoGuard aGuard; ScDPObject* pDPObj = 0; if( ScDPSaveDimension* pDim = GetDPDimension( &pDPObj ) ) { if( pDim->GetOrientation() != DataPilotFieldOrientation_DATA ) { sal_Int32 nCount = rSubtotals.getLength(); if( nCount == 1 ) { // count 1: all values are allowed (including NONE and AUTO) if( rSubtotals[ 0 ] == GeneralFunction_NONE ) pDim->SetSubTotals( 0, NULL ); else { sal_uInt16 nFunc = sal::static_int_cast( rSubtotals[ 0 ] ); pDim->SetSubTotals( 1, &nFunc ); } } else if( nCount > 1 ) { // set multiple functions, ignore NONE and AUTO in this case ::std::vector< sal_uInt16 > aSubt; for( sal_Int32 nIdx = 0; nIdx < nCount; ++nIdx ) { GeneralFunction eFunc = rSubtotals[ nIdx ]; if( (eFunc != GeneralFunction_NONE) && (eFunc != GeneralFunction_AUTO) ) { // do not insert functions twice sal_uInt16 nFunc = static_cast< sal_uInt16 >( eFunc ); if( ::std::find( aSubt.begin(), aSubt.end(), nFunc ) == aSubt.end() ) aSubt.push_back( nFunc ); } } // set values from vector to ScDPSaveDimension if ( aSubt.empty() ) pDim->SetSubTotals( 0, NULL ); else pDim->SetSubTotals( static_cast< long >( aSubt.size() ), &aSubt.front() ); } } SetDPObject( pDPObj ); } } OUString ScDataPilotFieldObj::getCurrentPage() const { ScUnoGuard aGuard; ScDPSaveDimension* pDim = GetDPDimension(); if( pDim && pDim->HasCurrentPage() ) return pDim->GetCurrentPage(); return OUString(); } void ScDataPilotFieldObj::setCurrentPage( const OUString& rPage ) { ScUnoGuard aGuard; ScDPObject* pDPObj = 0; if( ScDPSaveDimension* pDim = GetDPDimension( &pDPObj ) ) { String aPage( rPage ); pDim->SetCurrentPage( &aPage ); SetDPObject( pDPObj ); } } sal_Bool ScDataPilotFieldObj::getUseCurrentPage() const { ScUnoGuard aGuard; ScDPSaveDimension* pDim = GetDPDimension(); return pDim && pDim->HasCurrentPage(); } void ScDataPilotFieldObj::setUseCurrentPage( sal_Bool bUse ) { ScUnoGuard aGuard; ScDPObject* pDPObj = 0; if( ScDPSaveDimension* pDim = GetDPDimension( &pDPObj ) ) { if( bUse ) { /* It is somehow useless to set the property "HasSelectedPage" to true, because it is still needed to set an explicit page name. */ if( !pDim->HasCurrentPage() ) { String aPage; pDim->SetCurrentPage( &aPage ); } } else pDim->SetCurrentPage( 0 ); SetDPObject( pDPObj ); } } const DataPilotFieldAutoShowInfo* ScDataPilotFieldObj::getAutoShowInfo() { ScUnoGuard aGuard; ScDPSaveDimension* pDim = GetDPDimension(); return pDim ? pDim->GetAutoShowInfo() : 0; } void ScDataPilotFieldObj::setAutoShowInfo( const DataPilotFieldAutoShowInfo* pInfo ) { ScUnoGuard aGuard; ScDPObject* pDPObj = 0; if( ScDPSaveDimension* pDim = GetDPDimension( &pDPObj ) ) { pDim->SetAutoShowInfo( pInfo ); SetDPObject( pDPObj ); } } const DataPilotFieldLayoutInfo* ScDataPilotFieldObj::getLayoutInfo() { ScUnoGuard aGuard; ScDPSaveDimension* pDim = GetDPDimension(); return pDim ? pDim->GetLayoutInfo() : 0; } void ScDataPilotFieldObj::setLayoutInfo( const DataPilotFieldLayoutInfo* pInfo ) { ScUnoGuard aGuard; ScDPObject* pDPObj = 0; if( ScDPSaveDimension* pDim = GetDPDimension( &pDPObj ) ) { pDim->SetLayoutInfo( pInfo ); SetDPObject( pDPObj ); } } const DataPilotFieldReference* ScDataPilotFieldObj::getReference() { ScUnoGuard aGuard; ScDPSaveDimension* pDim = GetDPDimension(); return pDim ? pDim->GetReferenceValue() : 0; } void ScDataPilotFieldObj::setReference( const DataPilotFieldReference* pInfo ) { ScUnoGuard aGuard; ScDPObject* pDPObj = 0; if( ScDPSaveDimension* pDim = GetDPDimension( &pDPObj ) ) { pDim->SetReferenceValue( pInfo ); SetDPObject( pDPObj ); } } const DataPilotFieldSortInfo* ScDataPilotFieldObj::getSortInfo() { ScUnoGuard aGuard; ScDPSaveDimension* pDim = GetDPDimension(); return pDim ? pDim->GetSortInfo() : 0; } void ScDataPilotFieldObj::setSortInfo( const DataPilotFieldSortInfo* pInfo ) { ScUnoGuard aGuard; ScDPObject* pDPObj = 0; if( ScDPSaveDimension* pDim = GetDPDimension( &pDPObj ) ) { pDim->SetSortInfo( pInfo ); SetDPObject( pDPObj ); } } sal_Bool ScDataPilotFieldObj::getShowEmpty() const { ScUnoGuard aGuard; ScDPSaveDimension* pDim = GetDPDimension(); return pDim && pDim->GetShowEmpty(); } void ScDataPilotFieldObj::setShowEmpty( sal_Bool bShow ) { ScUnoGuard aGuard; ScDPObject* pDPObj = 0; if( ScDPSaveDimension* pDim = GetDPDimension( &pDPObj ) ) { pDim->SetShowEmpty( bShow ); SetDPObject( pDPObj ); } } sal_Bool ScDataPilotFieldObj::hasGroupInfo() { ScUnoGuard aGuard; ScDPObject* pDPObj = 0; if( ScDPSaveDimension* pDim = GetDPDimension( &pDPObj ) ) if( const ScDPDimensionSaveData* pDimData = pDPObj->GetSaveData()->GetExistingDimensionData() ) return pDimData->GetNamedGroupDim( pDim->GetName() ) || pDimData->GetNumGroupDim( pDim->GetName() ); return sal_False; } DataPilotFieldGroupInfo ScDataPilotFieldObj::getGroupInfo() { ScUnoGuard aGuard; DataPilotFieldGroupInfo aInfo; ScDPObject* pDPObj = 0; if( ScDPSaveDimension* pDim = GetDPDimension( &pDPObj ) ) { if( const ScDPDimensionSaveData* pDimData = pDPObj->GetSaveData()->GetExistingDimensionData() ) { if( const ScDPSaveGroupDimension* pGroupDim = pDimData->GetNamedGroupDim( pDim->GetName() ) ) { // grouped by ... aInfo.GroupBy = pGroupDim->GetDatePart(); // find source field try { Reference< XNameAccess > xFields( mrParent.getDataPilotFields(), UNO_QUERY_THROW ); aInfo.SourceField.set( xFields->getByName( pGroupDim->GetSourceDimName() ), UNO_QUERY ); } catch( Exception& ) { } ScDataPilotConversion::FillGroupInfo( aInfo, pGroupDim->GetDateInfo() ); if( pGroupDim->GetDatePart() == 0 ) { // fill vector of group and group member information ScFieldGroups aGroups; for( sal_Int32 nIdx = 0, nCount = pGroupDim->GetGroupCount(); nIdx < nCount; ++nIdx ) { if( const ScDPSaveGroupItem* pGroup = pGroupDim->GetGroupByIndex( nIdx ) ) { ScFieldGroup aGroup; aGroup.maName = pGroup->GetGroupName(); for( sal_Int32 nMemIdx = 0, nMemCount = pGroup->GetElementCount(); nMemIdx < nMemCount; ++nMemIdx ) if( const String* pMem = pGroup->GetElementByIndex( nMemIdx ) ) aGroup.maMembers.push_back( *pMem ); aGroups.push_back( aGroup ); } } aInfo.Groups = new ScDataPilotFieldGroupsObj( aGroups ); } } else if( const ScDPSaveNumGroupDimension* pNumGroupDim = pDimData->GetNumGroupDim( pDim->GetName() ) ) { if (pNumGroupDim->GetDatePart()) { ScDataPilotConversion::FillGroupInfo( aInfo, pNumGroupDim->GetDateInfo() ); aInfo.GroupBy = pNumGroupDim->GetDatePart(); } else { ScDataPilotConversion::FillGroupInfo( aInfo, pNumGroupDim->GetInfo() ); } } } } return aInfo; } void ScDataPilotFieldObj::setGroupInfo( const DataPilotFieldGroupInfo* pInfo ) { ScUnoGuard aGuard; ScDPObject* pDPObj = 0; if( /*ScDPSaveDimension* pDim =*/ GetDPDimension( &pDPObj ) ) { ScDPSaveData* pSaveData = pDPObj->GetSaveData(); if( pInfo && lclCheckMinMaxStep( *pInfo ) ) { ScDPNumGroupInfo aInfo; aInfo.Enable = sal_True; aInfo.DateValues = pInfo->HasDateValues; aInfo.AutoStart = pInfo->HasAutoStart; aInfo.AutoEnd = pInfo->HasAutoEnd; aInfo.Start = pInfo->Start; aInfo.End = pInfo->End; aInfo.Step = pInfo->Step; Reference< XNamed > xNamed( pInfo->SourceField, UNO_QUERY ); if( xNamed.is() ) { ScDPSaveGroupDimension aGroupDim( xNamed->getName(), getName() ); if( pInfo->GroupBy ) aGroupDim.SetDateInfo(aInfo, pInfo->GroupBy); else { Reference xIndex(pInfo->Groups, UNO_QUERY); if (xIndex.is()) { sal_Int32 nCount(xIndex->getCount()); for(sal_Int32 i = 0; i < nCount; i++) { Reference xGroupNamed(xIndex->getByIndex(i), UNO_QUERY); if (xGroupNamed.is()) { ScDPSaveGroupItem aItem(xGroupNamed->getName()); Reference xGroupIndex(xGroupNamed, UNO_QUERY); if (xGroupIndex.is()) { sal_Int32 nItemCount(xGroupIndex->getCount()); for (sal_Int32 j = 0; j < nItemCount; ++j) { Reference xItemNamed(xGroupIndex->getByIndex(j), UNO_QUERY); if (xItemNamed.is()) aItem.AddElement(xItemNamed->getName()); } } aGroupDim.AddGroupItem(aItem); } } } } // get dimension savedata or create new if none ScDPDimensionSaveData& rDimSaveData = *pSaveData->GetDimensionData(); rDimSaveData.ReplaceGroupDimension( aGroupDim ); } else // no source field in group info -> numeric group { ScDPDimensionSaveData* pDimData = pSaveData->GetDimensionData(); // created if not there ScDPSaveNumGroupDimension* pExisting = pDimData->GetNumGroupDimAcc( getName() ); if ( pExisting ) { if (pInfo->GroupBy) pExisting->SetDateInfo(aInfo, pInfo->GroupBy); // modify existing group dimension pExisting->SetGroupInfo( aInfo ); } else if (pInfo->GroupBy) { // create new group dimension ScDPSaveNumGroupDimension aNumGroupDim( getName(), aInfo, pInfo->GroupBy ); pDimData->AddNumGroupDimension( aNumGroupDim ); } else { // create new group dimension ScDPSaveNumGroupDimension aNumGroupDim( getName(), aInfo ); pDimData->AddNumGroupDimension( aNumGroupDim ); } } } else // null passed as argument { pSaveData->SetDimensionData( 0 ); } pDPObj->SetSaveData( *pSaveData ); SetDPObject( pDPObj ); } } sal_Bool ScDataPilotFieldObj::HasString(const Sequence< OUString >& rItems, const OUString& aString) { sal_Bool bRet = sal_False; sal_Int32 nCount(rItems.getLength()); sal_Int32 nItem(0); while (nItem < nCount && !bRet) { bRet = rItems[nItem] == aString; ++nItem; } return bRet; } // XDataPilotFieldGrouping Reference< XDataPilotField > SAL_CALL ScDataPilotFieldObj::createNameGroup( const Sequence< OUString >& rItems ) throw (RuntimeException, IllegalArgumentException) { ScUnoGuard aGuard; Reference< XDataPilotField > xRet; OUString sNewDim; if( !rItems.hasElements() ) throw IllegalArgumentException(); ScDPObject* pDPObj = 0; if( ScDPSaveDimension* pDim = GetDPDimension( &pDPObj ) ) { String aDimName = pDim->GetName(); ScDPSaveData aSaveData = *pDPObj->GetSaveData(); ScDPDimensionSaveData* pDimData = aSaveData.GetDimensionData(); // created if not there // find original base String aBaseDimName( aDimName ); const ScDPSaveGroupDimension* pBaseGroupDim = pDimData->GetNamedGroupDim( aDimName ); if ( pBaseGroupDim ) { // any entry's SourceDimName is the original base aBaseDimName = pBaseGroupDim->GetSourceDimName(); } // find existing group dimension // (using the selected dim, can be intermediate group dim) ScDPSaveGroupDimension* pGroupDimension = pDimData->GetGroupDimAccForBase( aDimName ); // remove the selected items from their groups // (empty groups are removed, too) sal_Int32 nEntryCount = rItems.getLength(); sal_Int32 nEntry; if ( pGroupDimension ) { for (nEntry=0; nEntryGetNamedGroup( aEntryName ); if ( pBaseGroup ) pBaseGroup->RemoveElementsFromGroups( *pGroupDimension ); // remove all elements else pGroupDimension->RemoveFromGroups( aEntryName ); } else pGroupDimension->RemoveFromGroups( aEntryName ); } } ScDPSaveGroupDimension* pNewGroupDim = 0; if ( !pGroupDimension ) { // create a new group dimension String aGroupDimName = pDimData->CreateGroupDimName( aBaseDimName, *pDPObj, false, NULL ); pNewGroupDim = new ScDPSaveGroupDimension( aBaseDimName, aGroupDimName ); sNewDim = aGroupDimName; pGroupDimension = pNewGroupDim; // make changes to the new dim if none existed if ( pBaseGroupDim ) { // If it's a higher-order group dimension, pre-allocate groups for all // non-selected original groups, so the individual base members aren't // used for automatic groups (this would make the original groups hard // to find). //! Also do this when removing groups? //! Handle this case dynamically with automatic groups? long nGroupCount = pBaseGroupDim->GetGroupCount(); for ( long nGroup = 0; nGroup < nGroupCount; nGroup++ ) { const ScDPSaveGroupItem* pBaseGroup = pBaseGroupDim->GetGroupByIndex( nGroup ); StrData aStrData( pBaseGroup->GetGroupName() ); if ( !HasString(rItems, aStrData.GetString()) ) //! ignore case? { // add an additional group for each item that is not in the selection ScDPSaveGroupItem aGroup( pBaseGroup->GetGroupName() ); aGroup.AddElementsFromGroup( *pBaseGroup ); pGroupDimension->AddGroupItem( aGroup ); } } } } String aGroupDimName = pGroupDimension->GetGroupDimName(); //! localized prefix string String aGroupName = pGroupDimension->CreateGroupName( String( RTL_CONSTASCII_USTRINGPARAM( "Group" ) ) ); ScDPSaveGroupItem aGroup( aGroupName ); Reference< XNameAccess > xMembers = GetMembers(); if (!xMembers.is()) { delete pNewGroupDim; throw RuntimeException(); } for (nEntry=0; nEntryhasByName(aEntryName)) { delete pNewGroupDim; throw IllegalArgumentException(); } if ( pBaseGroupDim ) { // for each selected (intermediate) group, add all its items const ScDPSaveGroupItem* pBaseGroup = pBaseGroupDim->GetNamedGroup( aEntryName ); if ( pBaseGroup ) aGroup.AddElementsFromGroup( *pBaseGroup ); else aGroup.AddElement( aEntryName ); // no group found -> automatic group, add the item itself } else aGroup.AddElement( aEntryName ); // no group dimension, add all items directly } pGroupDimension->AddGroupItem( aGroup ); if ( pNewGroupDim ) { pDimData->AddGroupDimension( *pNewGroupDim ); delete pNewGroupDim; // AddGroupDimension copies the object // don't access pGroupDimension after here } pGroupDimension = pNewGroupDim = NULL; // set orientation ScDPSaveDimension* pSaveDimension = aSaveData.GetDimensionByName( aGroupDimName ); if ( pSaveDimension->GetOrientation() == DataPilotFieldOrientation_HIDDEN ) { ScDPSaveDimension* pOldDimension = aSaveData.GetDimensionByName( aDimName ); pSaveDimension->SetOrientation( pOldDimension->GetOrientation() ); long nPosition = 0; //! before (immediate) base aSaveData.SetPosition( pSaveDimension, nPosition ); } // apply changes pDPObj->SetSaveData( aSaveData ); SetDPObject( pDPObj ); } // if new grouping field has been created (on first group), return it if( sNewDim.getLength() > 0 ) { Reference< XNameAccess > xFields(mrParent.getDataPilotFields(), UNO_QUERY); if (xFields.is()) { xRet.set(xFields->getByName(sNewDim), UNO_QUERY); DBG_ASSERT(xRet.is(), "there is a name, so there should be also a field"); } } return xRet; } Reference < XDataPilotField > SAL_CALL ScDataPilotFieldObj::createDateGroup( const DataPilotFieldGroupInfo& rInfo ) throw (RuntimeException, IllegalArgumentException) { ScUnoGuard aGuard; using namespace ::com::sun::star::sheet::DataPilotFieldGroupBy; // check min/max/step, HasDateValues must be set always if( !rInfo.HasDateValues || !lclCheckMinMaxStep( rInfo ) ) throw IllegalArgumentException(); // only a single date flag is allowed if( (rInfo.GroupBy == 0) || (rInfo.GroupBy > YEARS) || ((rInfo.GroupBy & (rInfo.GroupBy - 1)) != 0) ) throw IllegalArgumentException(); // step must be zero, if something else than DAYS is specified if( rInfo.Step >= ((rInfo.GroupBy == DAYS) ? 32768.0 : 1.0) ) throw IllegalArgumentException(); String aGroupDimName; ScDPObject* pDPObj = 0; if( ScDPSaveDimension* pDim = GetDPDimension( &pDPObj ) ) { ScDPNumGroupInfo aInfo; aInfo.Enable = sal_True; aInfo.DateValues = (rInfo.GroupBy == DAYS) && (rInfo.Step >= 1.0); aInfo.AutoStart = rInfo.HasAutoStart; aInfo.AutoEnd = rInfo.HasAutoEnd; aInfo.Start = rInfo.Start; aInfo.End = rInfo.End; aInfo.Step = static_cast< sal_Int32 >( rInfo.Step ); // create a local copy of the entire save data (will be written back below) ScDPSaveData aSaveData = *pDPObj->GetSaveData(); // get or create dimension save data ScDPDimensionSaveData& rDimData = *aSaveData.GetDimensionData(); // find source dimension name const String& rDimName = pDim->GetName(); const ScDPSaveGroupDimension* pGroupDim = rDimData.GetNamedGroupDim( rDimName ); String aSrcDimName = pGroupDim ? pGroupDim->GetSourceDimName() : rDimName; // find a group dimension for the base field, or get numeric grouping pGroupDim = rDimData.GetFirstNamedGroupDim( aSrcDimName ); const ScDPSaveNumGroupDimension* pNumGroupDim = rDimData.GetNumGroupDim( aSrcDimName ); // do not group by dates, if named groups or numeric grouping is present bool bHasNamedGrouping = pGroupDim && !pGroupDim->GetDateInfo().Enable; bool bHasNumGrouping = pNumGroupDim && pNumGroupDim->GetInfo().Enable && !pNumGroupDim->GetInfo().DateValues && !pNumGroupDim->GetDateInfo().Enable; if( bHasNamedGrouping || bHasNumGrouping ) throw IllegalArgumentException(); if( aInfo.DateValues ) // create day ranges grouping { // first remove all named group dimensions while( pGroupDim ) { String aGroupDimName2 = pGroupDim->GetGroupDimName(); // find next group dimension before deleting this group pGroupDim = rDimData.GetNextNamedGroupDim( aGroupDimName2 ); // remove from dimension save data rDimData.RemoveGroupDimension( aGroupDimName2 ); // also remove save data settings for the dimension that no longer exists aSaveData.RemoveDimensionByName( aGroupDimName2 ); } // create or replace the number grouping dimension ScDPSaveNumGroupDimension aNumGroupDim( aSrcDimName, aInfo ); rDimData.ReplaceNumGroupDimension( aNumGroupDim ); } else // create date grouping { // collect all existing date flags sal_Int32 nDateParts = rDimData.CollectDateParts( aSrcDimName ); if( nDateParts == 0 ) { // insert numeric group dimension, if no date groups exist yet (or replace day range grouping) ScDPSaveNumGroupDimension aNumGroupDim( aSrcDimName, aInfo, rInfo.GroupBy ); rDimData.ReplaceNumGroupDimension( aNumGroupDim ); } else if( (nDateParts & rInfo.GroupBy) == 0 ) // do nothing if date field exists already { // create new named group dimension for additional date groups aGroupDimName = rDimData.CreateDateGroupDimName( rInfo.GroupBy, *pDPObj, true, 0 ); ScDPSaveGroupDimension aGroupDim( aSrcDimName, aGroupDimName, aInfo, rInfo.GroupBy ); rDimData.AddGroupDimension( aGroupDim ); // set orientation of new named group dimension ScDPSaveDimension& rSaveDim = *aSaveData.GetDimensionByName( aGroupDimName ); if( rSaveDim.GetOrientation() == DataPilotFieldOrientation_HIDDEN ) { ScDPSaveDimension& rOldDim = *aSaveData.GetDimensionByName( aSrcDimName ); rSaveDim.SetOrientation( rOldDim.GetOrientation() ); aSaveData.SetPosition( &rSaveDim, 0 ); //! before (immediate) base } } } // apply changes pDPObj->SetSaveData( aSaveData ); SetDPObject( pDPObj ); } // return the UNO object of the new dimension, after writing back saved data Reference< XDataPilotField > xRet; if( aGroupDimName.Len() > 0 ) try { Reference< XNameAccess > xFields( mrParent.getDataPilotFields(), UNO_QUERY_THROW ); xRet.set( xFields->getByName( aGroupDimName ), UNO_QUERY ); } catch( Exception& ) { } return xRet; } // ============================================================================ namespace { bool lclExtractGroupMembers( ScFieldGroupMembers& rMembers, const Any& rElement ) { // allow empty value to create a new group if( !rElement.hasValue() ) return true; // try to extract a simple sequence of strings Sequence< OUString > aSeq; if( rElement >>= aSeq ) { if( aSeq.hasElements() ) rMembers.insert( rMembers.end(), aSeq.getConstArray(), aSeq.getConstArray() + aSeq.getLength() ); return true; } // try to use XIndexAccess providing objects that support XNamed Reference< XIndexAccess > xItemsIA( rElement, UNO_QUERY ); if( xItemsIA.is() ) { for( sal_Int32 nIdx = 0, nCount = xItemsIA->getCount(); nIdx < nCount; ++nIdx ) { try // getByIndex() should not throw, but we cannot be sure { Reference< XNamed > xItemName( xItemsIA->getByIndex( nIdx ), UNO_QUERY_THROW ); rMembers.push_back( xItemName->getName() ); } catch( Exception& ) { // ignore exceptions, go ahead with next element in the array } } return true; } // nothing valid inside the Any -> return false return false; } } // namespace // ---------------------------------------------------------------------------- ScDataPilotFieldGroupsObj::ScDataPilotFieldGroupsObj( const ScFieldGroups& rGroups ) : maGroups( rGroups ) { } ScDataPilotFieldGroupsObj::~ScDataPilotFieldGroupsObj() { } // XNameAccess Any SAL_CALL ScDataPilotFieldGroupsObj::getByName( const OUString& rName ) throw(NoSuchElementException, WrappedTargetException, RuntimeException) { ScUnoGuard aGuard; if( implFindByName( rName ) == maGroups.end() ) throw NoSuchElementException(); return Any( Reference< XNameAccess >( new ScDataPilotFieldGroupObj( *this, rName ) ) ); } Sequence< OUString > SAL_CALL ScDataPilotFieldGroupsObj::getElementNames() throw(RuntimeException) { ScUnoGuard aGuard; Sequence< OUString > aSeq; if( !maGroups.empty() ) { aSeq.realloc( static_cast< sal_Int32 >( maGroups.size() ) ); OUString* pName = aSeq.getArray(); for( ScFieldGroups::iterator aIt = maGroups.begin(), aEnd = maGroups.end(); aIt != aEnd; ++aIt, ++pName ) *pName = aIt->maName; } return aSeq; } sal_Bool SAL_CALL ScDataPilotFieldGroupsObj::hasByName( const OUString& rName ) throw(RuntimeException) { ScUnoGuard aGuard; return implFindByName( rName ) != maGroups.end(); } // XNameReplace void SAL_CALL ScDataPilotFieldGroupsObj::replaceByName( const OUString& rName, const Any& rElement ) throw (IllegalArgumentException, NoSuchElementException, WrappedTargetException, RuntimeException) { ScUnoGuard aGuard; if( rName.getLength() == 0 ) throw IllegalArgumentException(); ScFieldGroups::iterator aIt = implFindByName( rName ); if( aIt == maGroups.end() ) throw NoSuchElementException(); // read all item names provided by the passed object ScFieldGroupMembers aMembers; if( !lclExtractGroupMembers( aMembers, rElement ) ) throw IllegalArgumentException(); // copy and forget, faster than vector assignment aIt->maMembers.swap( aMembers ); } // XNameContainer void SAL_CALL ScDataPilotFieldGroupsObj::insertByName( const OUString& rName, const Any& rElement ) throw (IllegalArgumentException, ElementExistException, WrappedTargetException, RuntimeException) { ScUnoGuard aGuard; if( rName.getLength() == 0 ) throw IllegalArgumentException(); ScFieldGroups::iterator aIt = implFindByName( rName ); if( aIt != maGroups.end() ) throw ElementExistException(); // read all item names provided by the passed object ScFieldGroupMembers aMembers; if( !lclExtractGroupMembers( aMembers, rElement ) ) throw IllegalArgumentException(); // create the new entry if no error has been occured maGroups.resize( maGroups.size() + 1 ); ScFieldGroup& rGroup = maGroups.back(); rGroup.maName = rName; rGroup.maMembers.swap( aMembers ); } void SAL_CALL ScDataPilotFieldGroupsObj::removeByName( const OUString& rName ) throw (NoSuchElementException, WrappedTargetException, RuntimeException) { ScUnoGuard aGuard; if( rName.getLength() == 0 ) throw IllegalArgumentException(); ScFieldGroups::iterator aIt = implFindByName( rName ); if( aIt == maGroups.end() ) throw NoSuchElementException(); maGroups.erase( aIt ); } // XIndexAccess sal_Int32 SAL_CALL ScDataPilotFieldGroupsObj::getCount() throw(RuntimeException) { ScUnoGuard aGuard; return static_cast< sal_Int32 >( maGroups.size() ); } Any SAL_CALL ScDataPilotFieldGroupsObj::getByIndex( sal_Int32 nIndex ) throw(IndexOutOfBoundsException, WrappedTargetException, RuntimeException) { ScUnoGuard aGuard; if ((nIndex < 0) || (nIndex >= static_cast< sal_Int32 >( maGroups.size() ))) throw IndexOutOfBoundsException(); return Any( Reference< XNameAccess >( new ScDataPilotFieldGroupObj( *this, maGroups[ nIndex ].maName ) ) ); } // XEnumerationAccess Reference SAL_CALL ScDataPilotFieldGroupsObj::createEnumeration() throw(RuntimeException) { ScUnoGuard aGuard; return new ScIndexEnumeration( this, OUString( RTL_CONSTASCII_USTRINGPARAM( "com.sun.star.sheet.DataPilotFieldGroupsEnumeration" ) ) ); } // XElementAccess uno::Type SAL_CALL ScDataPilotFieldGroupsObj::getElementType() throw(RuntimeException) { ScUnoGuard aGuard; return getCppuType( (Reference< XNameAccess >*)0 ); } sal_Bool SAL_CALL ScDataPilotFieldGroupsObj::hasElements() throw(RuntimeException) { ScUnoGuard aGuard; return !maGroups.empty(); } // implementation ScFieldGroup& ScDataPilotFieldGroupsObj::getFieldGroup( const OUString& rName ) throw(RuntimeException) { ScUnoGuard aGuard; ScFieldGroups::iterator aIt = implFindByName( rName ); if( aIt == maGroups.end() ) throw RuntimeException(); return *aIt; } void ScDataPilotFieldGroupsObj::renameFieldGroup( const OUString& rOldName, const OUString& rNewName ) throw(RuntimeException) { ScUnoGuard aGuard; ScFieldGroups::iterator aOldIt = implFindByName( rOldName ); ScFieldGroups::iterator aNewIt = implFindByName( rNewName ); // new name must not exist yet if( (aOldIt == maGroups.end()) || ((aNewIt != maGroups.end()) && (aNewIt != aOldIt)) ) throw RuntimeException(); aOldIt->maName = rNewName; } // private ScFieldGroups::iterator ScDataPilotFieldGroupsObj::implFindByName( const OUString& rName ) { for( ScFieldGroups::iterator aIt = maGroups.begin(), aEnd = maGroups.end(); aIt != aEnd; ++aIt ) if( aIt->maName == rName ) return aIt; return maGroups.end(); } // ============================================================================ namespace { OUString lclExtractMember( const Any& rElement ) { if( rElement.has< OUString >() ) return rElement.get< OUString >(); Reference< XNamed > xNamed( rElement, UNO_QUERY ); if( xNamed.is() ) return xNamed->getName(); return OUString(); } } // namespace // ---------------------------------------------------------------------------- ScDataPilotFieldGroupObj::ScDataPilotFieldGroupObj( ScDataPilotFieldGroupsObj& rParent, const OUString& rGroupName ) : mrParent( rParent ), maGroupName( rGroupName ) { mrParent.acquire(); } ScDataPilotFieldGroupObj::~ScDataPilotFieldGroupObj() { mrParent.release(); } // XNameAccess Any SAL_CALL ScDataPilotFieldGroupObj::getByName( const OUString& rName ) throw(NoSuchElementException, WrappedTargetException, RuntimeException) { ScUnoGuard aGuard; ScFieldGroupMembers& rMembers = mrParent.getFieldGroup( maGroupName ).maMembers; ScFieldGroupMembers::iterator aIt = ::std::find( rMembers.begin(), rMembers.end(), rName ); if( aIt == rMembers.end() ) throw NoSuchElementException(); return Any( Reference< XNamed >( new ScDataPilotFieldGroupItemObj( *this, *aIt ) ) ); } Sequence< OUString > SAL_CALL ScDataPilotFieldGroupObj::getElementNames() throw(RuntimeException) { ScUnoGuard aGuard; return ::comphelper::containerToSequence( mrParent.getFieldGroup( maGroupName ).maMembers ); } sal_Bool SAL_CALL ScDataPilotFieldGroupObj::hasByName( const OUString& rName ) throw(RuntimeException) { ScUnoGuard aGuard; ScFieldGroupMembers& rMembers = mrParent.getFieldGroup( maGroupName ).maMembers; return ::std::find( rMembers.begin(), rMembers.end(), rName ) != rMembers.end(); } // XNameReplace void SAL_CALL ScDataPilotFieldGroupObj::replaceByName( const OUString& rName, const Any& rElement ) throw (IllegalArgumentException, NoSuchElementException, WrappedTargetException, RuntimeException) { ScUnoGuard aGuard; // it should be possible to quickly rename an item -> accept string or XNamed OUString aNewName = lclExtractMember( rElement ); if( (rName.getLength() == 0) || (aNewName.getLength() == 0) ) throw IllegalArgumentException(); if( rName == aNewName ) return; ScFieldGroupMembers& rMembers = mrParent.getFieldGroup( maGroupName ).maMembers; ScFieldGroupMembers::iterator aOldIt = ::std::find( rMembers.begin(), rMembers.end(), rName ); ScFieldGroupMembers::iterator aNewIt = ::std::find( rMembers.begin(), rMembers.end(), aNewName ); // throw if passed member name does not exist if( aOldIt == rMembers.end() ) throw NoSuchElementException(); // throw if new name already exists if( aNewIt != rMembers.end() ) throw IllegalArgumentException(); *aOldIt = aNewName; } // XNameContainer void SAL_CALL ScDataPilotFieldGroupObj::insertByName( const OUString& rName, const Any& /*rElement*/ ) throw (IllegalArgumentException, ElementExistException, WrappedTargetException, RuntimeException) { ScUnoGuard aGuard; // we will ignore the passed element and just try to insert the name if( rName.getLength() == 0 ) throw IllegalArgumentException(); ScFieldGroupMembers& rMembers = mrParent.getFieldGroup( maGroupName ).maMembers; ScFieldGroupMembers::iterator aIt = ::std::find( rMembers.begin(), rMembers.end(), rName ); // throw if passed name already exists if( aIt != rMembers.end() ) throw IllegalArgumentException(); rMembers.push_back( rName ); } void SAL_CALL ScDataPilotFieldGroupObj::removeByName( const OUString& rName ) throw (NoSuchElementException, WrappedTargetException, RuntimeException) { ScUnoGuard aGuard; if( rName.getLength() == 0 ) throw IllegalArgumentException(); ScFieldGroupMembers& rMembers = mrParent.getFieldGroup( maGroupName ).maMembers; ScFieldGroupMembers::iterator aIt = ::std::find( rMembers.begin(), rMembers.end(), rName ); // throw if passed name does not exist if( aIt == rMembers.end() ) throw NoSuchElementException(); rMembers.erase( aIt ); } // XIndexAccess sal_Int32 SAL_CALL ScDataPilotFieldGroupObj::getCount() throw(RuntimeException) { ScUnoGuard aGuard; return static_cast< sal_Int32 >( mrParent.getFieldGroup( maGroupName ).maMembers.size() ); } Any SAL_CALL ScDataPilotFieldGroupObj::getByIndex( sal_Int32 nIndex ) throw(IndexOutOfBoundsException, WrappedTargetException, RuntimeException) { ScUnoGuard aGuard; ScFieldGroupMembers& rMembers = mrParent.getFieldGroup( maGroupName ).maMembers; if ((nIndex < 0) || (nIndex >= static_cast< sal_Int32 >( rMembers.size() ))) throw IndexOutOfBoundsException(); return Any( Reference< XNamed >( new ScDataPilotFieldGroupItemObj( *this, rMembers[ nIndex ] ) ) ); } // XEnumerationAccess Reference< XEnumeration > SAL_CALL ScDataPilotFieldGroupObj::createEnumeration() throw(RuntimeException) { ScUnoGuard aGuard; return new ScIndexEnumeration( this, OUString( RTL_CONSTASCII_USTRINGPARAM( "com.sun.star.sheet.DataPilotFieldGroupEnumeration" ) ) ); } // XElementAccess uno::Type SAL_CALL ScDataPilotFieldGroupObj::getElementType() throw(RuntimeException) { ScUnoGuard aGuard; return getCppuType( (Reference< XNamed >*)0 ); } sal_Bool SAL_CALL ScDataPilotFieldGroupObj::hasElements() throw(RuntimeException) { ScUnoGuard aGuard; return !mrParent.getFieldGroup( maGroupName ).maMembers.empty(); } // XNamed OUString SAL_CALL ScDataPilotFieldGroupObj::getName() throw(RuntimeException) { ScUnoGuard aGuard; return maGroupName; } void SAL_CALL ScDataPilotFieldGroupObj::setName( const OUString& rName ) throw(RuntimeException) { ScUnoGuard aGuard; mrParent.renameFieldGroup( maGroupName, rName ); // if call to renameFieldGroup() did not throw, remember the new name maGroupName = rName; } // ============================================================================ ScDataPilotFieldGroupItemObj::ScDataPilotFieldGroupItemObj( ScDataPilotFieldGroupObj& rParent, const OUString& rName ) : mrParent( rParent ), maName( rName ) { mrParent.acquire(); } ScDataPilotFieldGroupItemObj::~ScDataPilotFieldGroupItemObj() { mrParent.release(); } // XNamed OUString SAL_CALL ScDataPilotFieldGroupItemObj::getName() throw(RuntimeException) { ScUnoGuard aGuard; return maName; } void SAL_CALL ScDataPilotFieldGroupItemObj::setName( const OUString& rName ) throw(RuntimeException) { ScUnoGuard aGuard; mrParent.replaceByName( maName, Any( rName ) ); // if call to replaceByName() did not throw, remember the new name maName = rName; } // ============================================================================ ScDataPilotItemsObj::ScDataPilotItemsObj( ScDataPilotDescriptorBase& rParent, const ScFieldIdentifier& rFieldId ) : ScDataPilotChildObjBase( rParent, rFieldId ) { } ScDataPilotItemsObj::~ScDataPilotItemsObj() { } // XDataPilotItems ScDataPilotItemObj* ScDataPilotItemsObj::GetObjectByIndex_Impl( sal_Int32 nIndex ) const { return ((0 <= nIndex) && (nIndex < GetMemberCount())) ? new ScDataPilotItemObj( mrParent, maFieldId, nIndex ) : 0; } // XNameAccess Any SAL_CALL ScDataPilotItemsObj::getByName( const OUString& aName ) throw(NoSuchElementException, WrappedTargetException, RuntimeException) { ScUnoGuard aGuard; Reference xMembers = GetMembers(); if (xMembers.is()) { Reference xMembersIndex(new ScNameToIndexAccess( xMembers )); sal_Int32 nCount = xMembersIndex->getCount(); sal_Bool bFound(sal_False); sal_Int32 nItem = 0; while (nItem < nCount && !bFound ) { Reference xMember(xMembersIndex->getByIndex(nItem), UNO_QUERY); if (xMember.is() && (aName == xMember->getName())) return Any( Reference< XPropertySet >( GetObjectByIndex_Impl( nItem ) ) ); ++nItem; } if (!bFound) throw NoSuchElementException(); } return Any(); } Sequence SAL_CALL ScDataPilotItemsObj::getElementNames() throw(RuntimeException) { ScUnoGuard aGuard; Sequence< OUString > aSeq; if( ScDPObject* pDPObj = GetDPObject() ) pDPObj->GetMemberNames( lcl_GetObjectIndex( pDPObj, maFieldId ), aSeq ); return aSeq; } sal_Bool SAL_CALL ScDataPilotItemsObj::hasByName( const OUString& aName ) throw(RuntimeException) { ScUnoGuard aGuard; sal_Bool bFound = sal_False; Reference xMembers = GetMembers(); if (xMembers.is()) { Reference xMembersIndex(new ScNameToIndexAccess( xMembers )); sal_Int32 nCount = xMembersIndex->getCount(); sal_Int32 nItem = 0; while (nItem < nCount && !bFound ) { Reference xMember(xMembersIndex->getByIndex(nItem), UNO_QUERY); if (xMember.is() && aName == xMember->getName()) bFound = sal_True; else nItem++; } } return bFound; } // XEnumerationAccess Reference SAL_CALL ScDataPilotItemsObj::createEnumeration() throw(RuntimeException) { ScUnoGuard aGuard; return new ScIndexEnumeration(this, OUString(RTL_CONSTASCII_USTRINGPARAM("com.sun.star.sheet.DataPilotItemsEnumeration"))); } // XIndexAccess sal_Int32 SAL_CALL ScDataPilotItemsObj::getCount() throw(RuntimeException) { ScUnoGuard aGuard; return GetMemberCount(); } Any SAL_CALL ScDataPilotItemsObj::getByIndex( sal_Int32 nIndex ) throw(IndexOutOfBoundsException, WrappedTargetException, RuntimeException) { ScUnoGuard aGuard; Reference< XPropertySet > xItem( GetObjectByIndex_Impl( nIndex ) ); if (!xItem.is()) throw IndexOutOfBoundsException(); return Any( xItem ); } uno::Type SAL_CALL ScDataPilotItemsObj::getElementType() throw(RuntimeException) { ScUnoGuard aGuard; return getCppuType((Reference*)0); } sal_Bool SAL_CALL ScDataPilotItemsObj::hasElements() throw(RuntimeException) { ScUnoGuard aGuard; return ( getCount() != 0 ); } //------------------------------------------------------------------------ ScDataPilotItemObj::ScDataPilotItemObj( ScDataPilotDescriptorBase& rParent, const ScFieldIdentifier& rFieldId, sal_Int32 nIndex ) : ScDataPilotChildObjBase( rParent, rFieldId ), maPropSet( lcl_GetDataPilotItemMap() ), mnIndex( nIndex ) { } ScDataPilotItemObj::~ScDataPilotItemObj() { } // XNamed OUString SAL_CALL ScDataPilotItemObj::getName() throw(RuntimeException) { ScUnoGuard aGuard; OUString sRet; Reference xMembers = GetMembers(); if (xMembers.is()) { Reference xMembersIndex(new ScNameToIndexAccess( xMembers )); sal_Int32 nCount = xMembersIndex->getCount(); if (mnIndex < nCount) { Reference xMember(xMembersIndex->getByIndex(mnIndex), UNO_QUERY); sRet = xMember->getName(); } } return sRet; } void SAL_CALL ScDataPilotItemObj::setName( const OUString& /* aName */ ) throw(RuntimeException) { } // XPropertySet Reference< XPropertySetInfo > SAL_CALL ScDataPilotItemObj::getPropertySetInfo( ) throw(RuntimeException) { ScUnoGuard aGuard; static Reference aRef = new SfxItemPropertySetInfo( maPropSet.getPropertyMap() ); return aRef; } void SAL_CALL ScDataPilotItemObj::setPropertyValue( const OUString& aPropertyName, const Any& aValue ) throw(UnknownPropertyException, PropertyVetoException, IllegalArgumentException, WrappedTargetException, RuntimeException) { ScUnoGuard aGuard; ScDPObject* pDPObj = 0; if( ScDPSaveDimension* pDim = GetDPDimension( &pDPObj ) ) { Reference xMembers = GetMembers(); if( xMembers.is() ) { Reference xMembersIndex( new ScNameToIndexAccess( xMembers ) ); sal_Int32 nCount = xMembersIndex->getCount(); if( mnIndex < nCount ) { Reference xMember(xMembersIndex->getByIndex(mnIndex), UNO_QUERY); String sName(xMember->getName()); ScDPSaveMember* pMember = pDim->GetMemberByName(sName); if (pMember) { bool bGetNewIndex = false; if ( aPropertyName.equalsAscii( SC_UNONAME_SHOWDETAIL ) ) pMember->SetShowDetails(cppu::any2bool(aValue)); else if ( aPropertyName.equalsAscii( SC_UNONAME_ISHIDDEN ) ) pMember->SetIsVisible(!cppu::any2bool(aValue)); else if ( aPropertyName.equalsAscii( SC_UNONAME_POS ) ) { sal_Int32 nNewPos = 0; if ( ( aValue >>= nNewPos ) && nNewPos >= 0 && nNewPos < nCount ) { pDim->SetMemberPosition( sName, nNewPos ); // get new effective index (depends on sorting mode, which isn't modified) bGetNewIndex = true; } else throw IllegalArgumentException(); } SetDPObject( pDPObj ); if ( bGetNewIndex ) // after SetDPObject, get the new index { OUString aOUName( sName ); Sequence< OUString > aItemNames = xMembers->getElementNames(); sal_Int32 nItemCount = aItemNames.getLength(); for (sal_Int32 nItem=0; nItem xMembers = GetMembers(); if( xMembers.is() ) { Reference< XIndexAccess > xMembersIndex( new ScNameToIndexAccess( xMembers ) ); sal_Int32 nCount = xMembersIndex->getCount(); if( mnIndex < nCount ) { Reference< XNamed > xMember( xMembersIndex->getByIndex( mnIndex ), UNO_QUERY ); String sName( xMember->getName() ); ScDPSaveMember* pMember = pDim->GetExistingMemberByName( sName ); if( aPropertyName.equalsAscii( SC_UNONAME_SHOWDETAIL ) ) { if (pMember && pMember->HasShowDetails()) { aRet <<= (bool)pMember->GetShowDetails(); } else { Reference< XPropertySet > xMemberProps( xMember, UNO_QUERY ); if( xMemberProps.is() ) aRet = xMemberProps->getPropertyValue( OUString( RTL_CONSTASCII_USTRINGPARAM( SC_UNO_SHOWDETA ) ) ); else aRet <<= true; } } else if ( aPropertyName.equalsAscii( SC_UNONAME_ISHIDDEN ) ) { if (pMember && pMember->HasIsVisible()) { aRet <<= !pMember->GetIsVisible(); } else { Reference< XPropertySet > xMemberProps( xMember, UNO_QUERY ); if( xMemberProps.is() ) aRet <<= !cppu::any2bool( xMemberProps->getPropertyValue( OUString( RTL_CONSTASCII_USTRINGPARAM( SC_UNO_ISVISIBL ) ) ) ); else aRet <<= false; } } else if ( aPropertyName.equalsAscii( SC_UNONAME_POS ) ) { aRet <<= mnIndex; } } } } return aRet; } void SAL_CALL ScDataPilotItemObj::addPropertyChangeListener( const OUString& /* aPropertyName */, const Reference< XPropertyChangeListener >& /* xListener */ ) throw(UnknownPropertyException, WrappedTargetException, RuntimeException) { } void SAL_CALL ScDataPilotItemObj::removePropertyChangeListener( const OUString& /* aPropertyName */, const Reference< XPropertyChangeListener >& /* aListener */ ) throw(UnknownPropertyException, WrappedTargetException, RuntimeException) { } void SAL_CALL ScDataPilotItemObj::addVetoableChangeListener( const OUString& /* PropertyName */, const Reference< XVetoableChangeListener >& /* aListener */ ) throw(UnknownPropertyException, WrappedTargetException, RuntimeException) { } void SAL_CALL ScDataPilotItemObj::removeVetoableChangeListener( const OUString& /* PropertyName */, const Reference< XVetoableChangeListener >& /* aListener */ ) throw(UnknownPropertyException, WrappedTargetException, RuntimeException) { } //------------------------------------------------------------------------