/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /************************************************************************* * * 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_svl.hxx" #include #include #include #include "whassert.hxx" #include #include #include "poolio.hxx" //======================================================================== void SfxItemPool::AddSfxItemPoolUser(SfxItemPoolUser& rNewUser) { pImp->maSfxItemPoolUsers.push_back(&rNewUser); } void SfxItemPool::RemoveSfxItemPoolUser(SfxItemPoolUser& rOldUser) { const std::vector::iterator aFindResult = ::std::find( pImp->maSfxItemPoolUsers.begin(), pImp->maSfxItemPoolUsers.end(), &rOldUser); if(aFindResult != pImp->maSfxItemPoolUsers.end()) { pImp->maSfxItemPoolUsers.erase(aFindResult); } } const SfxPoolItem* SfxItemPool::GetPoolDefaultItem( sal_uInt16 nWhich ) const { DBG_CHKTHIS(SfxItemPool, 0); const SfxPoolItem* pRet; if( IsInRange( nWhich ) ) pRet = *(pImp->ppPoolDefaults + GetIndex_Impl( nWhich )); else if( pImp->mpSecondary ) pRet = pImp->mpSecondary->GetPoolDefaultItem( nWhich ); else { SFX_ASSERT( 0, nWhich, "unknown Which-Id - cannot get pool default" ); pRet = 0; } return pRet; } // ----------------------------------------------------------------------- bool SfxItemPool::IsItemFlag_Impl( sal_uInt16 nPos, sal_uInt16 nFlag ) const { sal_uInt16 nItemFlag = pItemInfos[nPos]._nFlags; return nFlag == (nItemFlag & nFlag); } // ----------------------------------------------------------------------- bool SfxItemPool::IsItemFlag( sal_uInt16 nWhich, sal_uInt16 nFlag ) const { for ( const SfxItemPool *pPool = this; pPool; pPool = pPool->pImp->mpSecondary ) { if ( pPool->IsInRange(nWhich) ) return pPool->IsItemFlag_Impl( pPool->GetIndex_Impl(nWhich), nFlag); } DBG_ASSERT( !IsWhich(nWhich), "unknown which-id" ); return sal_False; } // ----------------------------------------------------------------------- SfxBroadcaster& SfxItemPool::BC() { return pImp->aBC; } // ----------------------------------------------------------------------- SfxItemPool::SfxItemPool ( UniString const & rName, /* Name des Pools zur Idetifikation im File-Format */ sal_uInt16 nStartWhich, /* erste Which-Id des Pools */ sal_uInt16 nEndWhich, /* letzte Which-Id des Pools */ const SfxItemInfo* pInfos, /* SID-Map und Item-Flags */ SfxPoolItem** pDefaults, /* Pointer auf statische Defaults, wird direkt vom Pool referenziert, jedoch kein Eigent"umer"ubergang */ bool bLoadRefCounts /* Ref-Counts mitladen oder auf 1 setzen */ ) : /* [Beschreibung] Der im Normalfall verwendete Konstruktor der Klasse SfxItemPool. Es wird eine SfxItemPool-Instanz initialisiert, die Items im b"undigen Which-Bereich von 'nStartWhich' bis 'nEndWhich' verwalten kann. F"ur jede dieser Which-Ids mu\s ein statischer Default im Array 'pDefaults' vorhanden sein, die dort beginnend mit einem mit der Which-Id 'nStartWhich' nach Which-Ids sortiert aufeinanderfolgend eingetragen sein m"ussen. 'pItemInfos' ist ein identisch angeordnetes Array von USHORTs, die Slot-Ids darstellen und Flags. Die Slot-Ids k"onnen 0 sein, wenn die betreffenden Items ausschlie\slich in der Core verwendet werden. "Uber die Flags kann z.B. bestimmt werden, ob Value-Sharing (SFX_ITEM_POOLABLE) stattfinden soll. [Anmerkung] Wenn der Pool s enthalten soll, k"onnen im Konstruktor noch keine static-Defaults angegeben werden. Dies mu\s dann nachtr"aglich mit geschehen. [Querverweise] */ pItemInfos(pInfos), pImp( new SfxItemPool_Impl( this, rName, nStartWhich, nEndWhich ) ) { DBG_CTOR(SfxItemPool, 0); pImp->eDefMetric = SFX_MAPUNIT_TWIP; pImp->nVersion = 0; pImp->bStreaming = false; pImp->nLoadingVersion = 0; pImp->nInitRefCount = 1; pImp->nVerStart = pImp->mnStart; pImp->nVerEnd = pImp->mnEnd; pImp->bInSetItem = false; pImp->nStoringStart = nStartWhich; pImp->nStoringEnd = nEndWhich; pImp->mbPersistentRefCounts = bLoadRefCounts; if ( pDefaults ) SetDefaults(pDefaults); } // ----------------------------------------------------------------------- SfxItemPool::SfxItemPool ( const SfxItemPool& rPool, // von dieser Instanz kopieren sal_Bool bCloneStaticDefaults /* sal_True statische Defaults kopieren sal_False statische Defaults "ubernehehmen */ ) : /* [Beschreibung] Copy-Konstruktor der Klasse SfxItemPool. [Querverweise] */ pItemInfos(rPool.pItemInfos), pImp( new SfxItemPool_Impl( this, rPool.pImp->aName, rPool.pImp->mnStart, rPool.pImp->mnEnd ) ) { DBG_CTOR(SfxItemPool, 0); pImp->eDefMetric = rPool.pImp->eDefMetric; pImp->nVersion = rPool.pImp->nVersion; pImp->bStreaming = sal_False; pImp->nLoadingVersion = 0; pImp->nInitRefCount = 1; pImp->nVerStart = rPool.pImp->nVerStart; pImp->nVerEnd = rPool.pImp->nVerEnd; pImp->bInSetItem = false; pImp->nStoringStart = pImp->mnStart; pImp->nStoringEnd = pImp->mnEnd; pImp->mbPersistentRefCounts = rPool.pImp->mbPersistentRefCounts; // Static Defaults "ubernehmen if ( bCloneStaticDefaults ) { SfxPoolItem **ppDefaults = new SfxPoolItem*[pImp->mnEnd-pImp->mnStart+1]; for ( sal_uInt16 n = 0; n <= pImp->mnEnd - pImp->mnStart; ++n ) { (*( ppDefaults + n )) = (*( rPool.pImp->ppStaticDefaults + n ))->Clone(this); (*( ppDefaults + n ))->SetKind( SFX_ITEMS_STATICDEFAULT ); } SetDefaults( ppDefaults ); } else SetDefaults( rPool.pImp->ppStaticDefaults ); // Pool Defaults kopieren for ( sal_uInt16 n = 0; n <= pImp->mnEnd - pImp->mnStart; ++n ) if ( (*( rPool.pImp->ppPoolDefaults + n )) ) { (*( pImp->ppPoolDefaults + n )) = (*( rPool.pImp->ppPoolDefaults + n ))->Clone(this); (*( pImp->ppPoolDefaults + n ))->SetKind( SFX_ITEMS_POOLDEFAULT ); } // Copy Version-Map for ( size_t nVer = 0; nVer < rPool.pImp->aVersions.size(); ++nVer ) { const SfxPoolVersion_ImplPtr pOld = rPool.pImp->aVersions[nVer]; SfxPoolVersion_ImplPtr pNew = SfxPoolVersion_ImplPtr( new SfxPoolVersion_Impl( *pOld ) ); pImp->aVersions.push_back( pNew ); } // Verkettung wiederherstellen if ( rPool.pImp->mpSecondary ) SetSecondaryPool( rPool.pImp->mpSecondary->Clone() ); } // ----------------------------------------------------------------------- void SfxItemPool::SetDefaults( SfxPoolItem **pDefaults ) { DBG_CHKTHIS(SfxItemPool, 0); DBG_ASSERT( pDefaults, "erst wollen, dann nichts geben..." ); DBG_ASSERT( !pImp->ppStaticDefaults, "habe schon defaults" ); pImp->ppStaticDefaults = pDefaults; //! if ( (*ppStaticDefaults)->GetKind() != SFX_ITEMS_STATICDEFAULT ) //! geht wohl nicht im Zshg mit SetItems, die hinten stehen { DBG_ASSERT( (*pImp->ppStaticDefaults)->GetRefCount() == 0 || IsDefaultItem( (*pImp->ppStaticDefaults) ), "das sind keine statics" ); for ( sal_uInt16 n = 0; n <= pImp->mnEnd - pImp->mnStart; ++n ) { SFX_ASSERT( (*( pImp->ppStaticDefaults + n ))->Which() == n + pImp->mnStart, n + pImp->mnStart, "static defaults not sorted" ); (*( pImp->ppStaticDefaults + n ))->SetKind( SFX_ITEMS_STATICDEFAULT ); DBG_ASSERT( !(pImp->maPoolItems[n]), "defaults with setitems with items?!" ); } } } // ----------------------------------------------------------------------- void SfxItemPool::ReleaseDefaults ( sal_Bool bDelete /* sal_True l"oscht sowohl das Array als auch die einzelnen statischen Defaults sal_False l"oscht weder das Array noch die einzelnen statischen Defaults */ ) /* [Beschreibung] Gibt die statischen Defaults der betreffenden SfxItemPool-Instanz frei und l"oscht ggf. die statischen Defaults. Nach Aufruf dieser Methode darf die SfxItemPool-Instanz nicht mehr verwendet werden, einzig ist der Aufruf des Destruktors zu"lassig. */ { DBG_ASSERT( pImp->ppStaticDefaults, "keine Arme keine Kekse" ); ReleaseDefaults( pImp->ppStaticDefaults, pImp->mnEnd - pImp->mnStart + 1, bDelete ); // KSO (22.10.98): ppStaticDefaults zeigt auf geloeschten Speicher, // wenn bDelete == sal_True. if ( bDelete ) pImp->ppStaticDefaults = 0; } // ----------------------------------------------------------------------- void SfxItemPool::ReleaseDefaults ( SfxPoolItem** pDefaults, /* freizugebende statische Defaults */ sal_uInt16 nCount, /* Anzahl der statischen Defaults */ sal_Bool bDelete /* sal_True l"oscht sowohl das Array als auch die einzelnen statischen Defaults sal_False l"oscht weder das Array noch die einzelnen statischen Defaults */ ) /* [Beschreibung] Gibt die angegebenen statischen Defaults frei und l"oscht ggf. die statischen Defaults. Diese Methode darf erst nach Zerst"orung aller SfxItemPool-Instanzen, welche die angegebenen statischen Defaults 'pDefault' verwenden, aufgerufen werden. */ { DBG_ASSERT( pDefaults, "erst wollen, dann nichts geben..." ); for ( sal_uInt16 n = 0; n < nCount; ++n ) { SFX_ASSERT( IsStaticDefaultItem( *(pDefaults+n) ), n, "das ist kein static-default" ); (*( pDefaults + n ))->SetRefCount( 0 ); if ( bDelete ) { delete *( pDefaults + n ); *(pDefaults + n) = 0; } } if ( bDelete ) { delete[] pDefaults; pDefaults = 0; } } // ----------------------------------------------------------------------- SfxItemPool::~SfxItemPool() { DBG_DTOR(SfxItemPool, 0); DBG_ASSERT( pImp->mpMaster == this, "destroying active Secondary-Pool" ); if ( !pImp->maPoolItems.empty() && pImp->ppPoolDefaults ) Delete(); delete pImp; } void SfxItemPool::Free(SfxItemPool* pPool) { if(pPool) { // tell all the registered SfxItemPoolUsers that the pool is in destruction std::vector aListCopy(pPool->pImp->maSfxItemPoolUsers.begin(), pPool->pImp->maSfxItemPoolUsers.end()); for(std::vector::iterator aIterator = aListCopy.begin(); aIterator != aListCopy.end(); ++aIterator) { SfxItemPoolUser* pSfxItemPoolUser = *aIterator; DBG_ASSERT(pSfxItemPoolUser, "corrupt SfxItemPoolUser list (!)"); pSfxItemPoolUser->ObjectInDestruction(*pPool); } // Clear the vector. This means that user do not need to call RemoveSfxItemPoolUser() // when they get called from ObjectInDestruction(). pPool->pImp->maSfxItemPoolUsers.clear(); // delete pool delete pPool; } } // ----------------------------------------------------------------------- void SfxItemPool::SetSecondaryPool( SfxItemPool *pPool ) { // ggf. an abgeh"angten Pools den Master zur"ucksetzen if ( pImp->mpSecondary ) { #ifdef DBG_UTIL HACK( "fuer Image, dort gibt es derzeit keine Statics - Bug" ) if ( pImp->ppStaticDefaults ) { // Delete() ist noch nicht gelaufen? if ( !pImp->maPoolItems.empty() && !pImp->mpSecondary->pImp->maPoolItems.empty() ) { // hat der master SetItems? sal_Bool bHasSetItems = sal_False; for ( sal_uInt16 i = 0; !bHasSetItems && i < pImp->mnEnd - pImp->mnStart; ++i ) bHasSetItems = pImp->ppStaticDefaults[i]->ISA(SfxSetItem); // abgehaengte Pools muessen leer sein bool bOK = bHasSetItems; for ( sal_uInt16 n = 0; bOK && n <= pImp->mpSecondary->pImp->mnEnd - pImp->mpSecondary->pImp->mnStart; ++n ) { SfxPoolItemArray_Impl* pItemArr = pImp->mpSecondary->pImp->maPoolItems[n]; if ( pItemArr ) { SfxPoolItemArrayBase_Impl::iterator ppHtArr = pItemArr->begin(); for( size_t i = pItemArr->size(); i; ++ppHtArr, --i ) if ( !(*ppHtArr) ) { OSL_FAIL( "old secondary pool must be empty" ); bOK = false; break; } } } } } #endif pImp->mpSecondary->pImp->mpMaster = pImp->mpSecondary; for ( SfxItemPool *p = pImp->mpSecondary->pImp->mpSecondary; p; p = p->pImp->mpSecondary ) p->pImp->mpMaster = pImp->mpSecondary; } // ggf. den Master der neuen Secondary-Pools setzen DBG_ASSERT( !pPool || pPool->pImp->mpMaster == pPool, "Secondary tanzt auf zwei Hochzeiten " ); SfxItemPool *pNewMaster = pImp->mpMaster ? pImp->mpMaster : this; for ( SfxItemPool *p = pPool; p; p = p->pImp->mpSecondary ) p->pImp->mpMaster = pNewMaster; // neuen Secondary-Pool merken pImp->mpSecondary = pPool; } // ----------------------------------------------------------------------- SfxMapUnit SfxItemPool::GetMetric( sal_uInt16 ) const { DBG_CHKTHIS(SfxItemPool, 0); return pImp->eDefMetric; } // ----------------------------------------------------------------------- void SfxItemPool::SetDefaultMetric( SfxMapUnit eNewMetric ) { DBG_CHKTHIS(SfxItemPool, 0); pImp->eDefMetric = eNewMetric; } const UniString& SfxItemPool::GetName() const { return pImp->aName; } // ----------------------------------------------------------------------- SfxItemPresentation SfxItemPool::GetPresentation ( const SfxPoolItem& rItem, /* IN: , dessen textuelle Wert-Darstellung geliefert werden soll */ SfxItemPresentation ePresent, /* IN: gew"unschte Art der Darstellung; siehe */ SfxMapUnit eMetric, /* IN: gew"unschte Ma\seinheit der Darstellung */ XubString& rText, /* OUT: textuelle Darstellung von 'rItem' */ const IntlWrapper * pIntlWrapper ) const /* [Beschreibung] "Uber diese virtuelle Methode k"onnen textuelle Darstellungen der von der jeweilige SfxItemPool-Subklasse verwalteten SfxPoolItems angefordert werden. In Ableitungen sollte diese Methode "uberladen werden und auf SfxPoolItems reagiert werden, die bei keine vollst"andige Information liefern k"onnen. Die Basisklasse liefert die unver"anderte Presentation von 'rItem'. */ { DBG_CHKTHIS(SfxItemPool, 0); return rItem.GetPresentation( ePresent, GetMetric(rItem.Which()), eMetric, rText, pIntlWrapper ); } // ----------------------------------------------------------------------- SfxItemPool* SfxItemPool::Clone() const { DBG_CHKTHIS(SfxItemPool, 0); SfxItemPool *pPool = new SfxItemPool( *this ); return pPool; } // ---------------------------------------------------------------------- void SfxItemPool::Delete() { DBG_CHKTHIS(SfxItemPool, 0); // schon deleted? if ( pImp->maPoolItems.empty() || !pImp->ppPoolDefaults ) return; // z.B. laufenden Requests bescheidsagen pImp->aBC.Broadcast( SfxSimpleHint( SFX_HINT_DYING ) ); //MA 16. Apr. 97: Zweimal durchlaufen, in der ersten Runde fuer die SetItems. //Der Klarheit halber wird das jetzt in zwei besser lesbare Schleifen aufgeteilt. std::vector::iterator itrItemArr = pImp->maPoolItems.begin(); SfxPoolItem** ppDefaultItem = pImp->ppPoolDefaults; SfxPoolItem** ppStaticDefaultItem = pImp->ppStaticDefaults; sal_uInt16 nArrCnt; //Erst die SetItems abraeumen HACK( "fuer Image, dort gibt es derzeit keine Statics - Bug" ) if ( pImp->ppStaticDefaults ) { for ( nArrCnt = GetSize_Impl(); nArrCnt; --nArrCnt, ++itrItemArr, ++ppDefaultItem, ++ppStaticDefaultItem ) { // KSO (22.10.98): *ppStaticDefaultItem kann im dtor einer // von SfxItemPool abgeleiteten Klasse bereits geloescht worden // sein! -> CHAOS Itempool if ( *ppStaticDefaultItem && (*ppStaticDefaultItem)->ISA(SfxSetItem) ) { if ( *itrItemArr ) { SfxPoolItemArrayBase_Impl::iterator ppHtArr = (*itrItemArr)->begin(); for ( size_t n = (*itrItemArr)->size(); n; --n, ++ppHtArr ) if (*ppHtArr) { #ifdef DBG_UTIL ReleaseRef( **ppHtArr, (*ppHtArr)->GetRefCount() ); #endif delete *ppHtArr; } DELETEZ( *itrItemArr ); } if ( *ppDefaultItem ) { #ifdef DBG_UTIL SetRefCount( **ppDefaultItem, 0 ); #endif DELETEZ( *ppDefaultItem ); } } } } itrItemArr = pImp->maPoolItems.begin(); ppDefaultItem = pImp->ppPoolDefaults; //Jetzt die 'einfachen' Items for ( nArrCnt = GetSize_Impl(); nArrCnt; --nArrCnt, ++itrItemArr, ++ppDefaultItem ) { if ( *itrItemArr ) { SfxPoolItemArrayBase_Impl::iterator ppHtArr = (*itrItemArr)->begin(); for ( size_t n = (*itrItemArr)->size(); n; --n, ++ppHtArr ) if (*ppHtArr) { #ifdef DBG_UTIL ReleaseRef( **ppHtArr, (*ppHtArr)->GetRefCount() ); #endif delete *ppHtArr; } DELETEZ( *itrItemArr ); } if ( *ppDefaultItem ) { #ifdef DBG_UTIL SetRefCount( **ppDefaultItem, 0 ); #endif delete *ppDefaultItem; } } pImp->DeleteItems(); } // ---------------------------------------------------------------------- void SfxItemPool::SetPoolDefaultItem(const SfxPoolItem &rItem) { DBG_CHKTHIS(SfxItemPool, 0); if ( IsInRange(rItem.Which()) ) { SfxPoolItem **ppOldDefault = pImp->ppPoolDefaults + GetIndex_Impl(rItem.Which()); SfxPoolItem *pNewDefault = rItem.Clone(this); pNewDefault->SetKind(SFX_ITEMS_POOLDEFAULT); if ( *ppOldDefault ) { (*ppOldDefault)->SetRefCount(0); DELETEZ( *ppOldDefault ); } *ppOldDefault = pNewDefault; } else if ( pImp->mpSecondary ) pImp->mpSecondary->SetPoolDefaultItem(rItem); else { SFX_ASSERT( 0, rItem.Which(), "unknown Which-Id - cannot set pool default" ); } } /* * Resets the default of the given back to the static default. * If a pool default exists it is removed. */ void SfxItemPool::ResetPoolDefaultItem( sal_uInt16 nWhichId ) { DBG_CHKTHIS(SfxItemPool, 0); if ( IsInRange(nWhichId) ) { SfxPoolItem **ppOldDefault = pImp->ppPoolDefaults + GetIndex_Impl( nWhichId ); if ( *ppOldDefault ) { (*ppOldDefault)->SetRefCount(0); DELETEZ( *ppOldDefault ); } } else if ( pImp->mpSecondary ) pImp->mpSecondary->ResetPoolDefaultItem(nWhichId); else { SFX_ASSERT( 0, nWhichId, "unknown Which-Id - cannot set pool default" ); } } // ----------------------------------------------------------------------- const SfxPoolItem& SfxItemPool::Put( const SfxPoolItem& rItem, sal_uInt16 nWhich ) { DBG_ASSERT( !rItem.ISA(SfxSetItem) || 0 != &((const SfxSetItem&)rItem).GetItemSet(), "SetItem without ItemSet" ); DBG_CHKTHIS(SfxItemPool, 0); if ( 0 == nWhich ) nWhich = rItem.Which(); // richtigen Secondary-Pool finden bool bSID = nWhich > SFX_WHICH_MAX; if ( !bSID && !IsInRange(nWhich) ) { if ( pImp->mpSecondary ) return pImp->mpSecondary->Put( rItem, nWhich ); OSL_FAIL( "unknown Which-Id - cannot put item" ); } // SID oder nicht poolable (neue Definition)? sal_uInt16 nIndex = bSID ? USHRT_MAX : GetIndex_Impl(nWhich); if ( USHRT_MAX == nIndex || IsItemFlag_Impl( nIndex, SFX_ITEM_NOT_POOLABLE ) ) { SFX_ASSERT( USHRT_MAX != nIndex || rItem.Which() != nWhich || !IsDefaultItem(&rItem) || rItem.GetKind() == SFX_ITEMS_DELETEONIDLE, nWhich, "ein nicht Pool-Item ist Default?!" ); SfxPoolItem *pPoolItem = rItem.Clone(pImp->mpMaster); pPoolItem->SetWhich(nWhich); AddRef( *pPoolItem ); return *pPoolItem; } SFX_ASSERT( rItem.IsA(GetDefaultItem(nWhich).Type()), nWhich, "SFxItemPool: wrong item type in Put" ); SfxPoolItemArray_Impl* pItemArr = pImp->maPoolItems[nIndex]; if (!pItemArr) { pImp->maPoolItems[nIndex] = new SfxPoolItemArray_Impl; pItemArr = pImp->maPoolItems[nIndex]; } SfxPoolItemArrayBase_Impl::iterator ppFree; bool ppFreeIsSet = false; if ( IsItemFlag_Impl( nIndex, SFX_ITEM_POOLABLE ) ) { // wenn es ueberhaupt gepoolt ist, koennte es schon drin sein if ( IsPooledItem(&rItem) ) { // 1. Schleife: teste ob der Pointer vorhanden ist. SfxPoolItemArrayBase_Impl::iterator itr = std::find(pItemArr->begin(), pItemArr->end(), &rItem); if (itr != pItemArr->end()) { AddRef(**itr); return **itr; } } // 2. Schleife: dann muessen eben die Attribute verglichen werden SfxPoolItemArrayBase_Impl::iterator itr = pItemArr->begin(); for (; itr != pItemArr->end(); ++itr) { if (*itr) { if (**itr == rItem) { AddRef(**itr); return **itr; } } else { if (!ppFreeIsSet) { ppFree = itr; ppFreeIsSet = true; } } } } else { // freien Platz suchen SfxPoolItemArrayBase_Impl::iterator itr = pItemArr->begin(); std::advance(itr, pItemArr->nFirstFree); for (; itr != pItemArr->end(); ++itr) { if (!*itr) { ppFree = itr; ppFreeIsSet = true; break; } } // naechstmoeglichen freien Platz merken pItemArr->nFirstFree = std::distance(pItemArr->begin(), itr); } // nicht vorhanden, also im PtrArray eintragen SfxPoolItem* pNewItem = rItem.Clone(pImp->mpMaster); pNewItem->SetWhich(nWhich); #ifdef DBG_UTIL SFX_ASSERT( rItem.Type() == pNewItem->Type(), nWhich, "unequal types in Put(): no Clone()?" ) if ( !rItem.ISA(SfxSetItem) ) { SFX_ASSERT( !IsItemFlag(nWhich, SFX_ITEM_POOLABLE) || rItem == *pNewItem, nWhich, "unequal items in Put(): no operator==?" ); SFX_ASSERT( !IsItemFlag(*pNewItem, SFX_ITEM_POOLABLE) || *pNewItem == rItem, nWhich, "unequal items in Put(): no operator==?" ); } #endif AddRef( *pNewItem, pImp->nInitRefCount ); if ( ppFreeIsSet == false ) pItemArr->push_back( pNewItem ); else { DBG_ASSERT( *ppFree == 0, "using surrogate in use" ); *ppFree = pNewItem; } return *pNewItem; } // ----------------------------------------------------------------------- void SfxItemPool::Remove( const SfxPoolItem& rItem ) { DBG_CHKTHIS(SfxItemPool, 0); DBG_ASSERT( !rItem.ISA(SfxSetItem) || 0 != &((const SfxSetItem&)rItem).GetItemSet(), "SetItem without ItemSet" ); SFX_ASSERT( !IsPoolDefaultItem(&rItem), rItem.Which(), "wo kommt denn hier ein Pool-Default her" ); // richtigen Secondary-Pool finden const sal_uInt16 nWhich = rItem.Which(); bool bSID = nWhich > SFX_WHICH_MAX; if ( !bSID && !IsInRange(nWhich) ) { if ( pImp->mpSecondary ) { pImp->mpSecondary->Remove( rItem ); return; } OSL_FAIL( "unknown Which-Id - cannot remove item" ); } // SID oder nicht poolable (neue Definition)? sal_uInt16 nIndex = bSID ? USHRT_MAX : GetIndex_Impl(nWhich); if ( bSID || IsItemFlag_Impl( nIndex, SFX_ITEM_NOT_POOLABLE ) ) { SFX_ASSERT( USHRT_MAX != nIndex || !IsDefaultItem(&rItem), rItem.Which(), "ein nicht Pool-Item ist Default?!" ); if ( 0 == ReleaseRef(rItem) ) { SfxPoolItem *pItem = &(SfxPoolItem &)rItem; delete pItem; } return; } SFX_ASSERT( rItem.GetRefCount(), rItem.Which(), "RefCount == 0, Remove unmoeglich" ); // statische Defaults sind eben einfach da if ( rItem.GetKind() == SFX_ITEMS_STATICDEFAULT && &rItem == *( pImp->ppStaticDefaults + GetIndex_Impl(nWhich) ) ) return; // Item im eigenen Pool suchen SfxPoolItemArray_Impl* pItemArr = pImp->maPoolItems[nIndex]; SFX_ASSERT( pItemArr, rItem.Which(), "removing Item not in Pool" ); SfxPoolItemArrayBase_Impl::iterator ppHtArrBeg = pItemArr->begin(), ppHtArrEnd = pItemArr->end(); for (SfxPoolItemArrayBase_Impl::iterator ppHtArr = ppHtArrBeg; ppHtArr != ppHtArrEnd; ++ppHtArr) { SfxPoolItem*& p = *ppHtArr; if (p == &rItem) { if ( p->GetRefCount() ) //! ReleaseRef( *p ); else { SFX_ASSERT( 0, rItem.Which(), "removing Item without ref" ); SFX_TRACE( "to be removed, but not no refs: ", p ); } // ggf. kleinstmoegliche freie Position merken size_t nPos = std::distance(ppHtArrBeg, ppHtArr); if ( pItemArr->nFirstFree > nPos ) pItemArr->nFirstFree = nPos; //! MI: Hack, solange wir das Problem mit dem Outliner haben //! siehe anderes MI-REF if ( 0 == p->GetRefCount() && nWhich < 4000 ) DELETEZ(p); return; } } // nicht vorhanden SFX_ASSERT( 0, rItem.Which(), "removing Item not in Pool" ); SFX_TRACE( "to be removed, but not in pool: ", &rItem ); } // ----------------------------------------------------------------------- const SfxPoolItem& SfxItemPool::GetDefaultItem( sal_uInt16 nWhich ) const { DBG_CHKTHIS(SfxItemPool, 0); if ( !IsInRange(nWhich) ) { if ( pImp->mpSecondary ) return pImp->mpSecondary->GetDefaultItem( nWhich ); SFX_ASSERT( 0, nWhich, "unknown which - dont ask me for defaults" ); } DBG_ASSERT( pImp->ppStaticDefaults, "no defaults known - dont ask me for defaults" ); sal_uInt16 nPos = GetIndex_Impl(nWhich); SfxPoolItem *pDefault = *(pImp->ppPoolDefaults + nPos); if ( pDefault ) return *pDefault; return **(pImp->ppStaticDefaults + nPos); } SfxItemPool* SfxItemPool::GetSecondaryPool() const { return pImp->mpSecondary; } SfxItemPool* SfxItemPool::GetMasterPool() const { return pImp->mpMaster; } void SfxItemPool::FreezeIdRanges() /* [Beschreibung] This method should be called at the master pool, when all secondary pools are appended to it. It calculates the ranges of 'which-ids' for fast construction of item-sets, which contains all 'which-ids'. */ { FillItemIdRanges_Impl( pImp->mpPoolRanges ); } // ----------------------------------------------------------------------- void SfxItemPool::FillItemIdRanges_Impl( sal_uInt16*& pWhichRanges ) const { DBG_CHKTHIS(SfxItemPool, 0); DBG_ASSERT( !pImp->mpPoolRanges, "GetFrozenRanges() would be faster!" ); const SfxItemPool *pPool; sal_uInt16 nLevel = 0; for( pPool = this; pPool; pPool = pPool->pImp->mpSecondary ) ++nLevel; pWhichRanges = new sal_uInt16[ 2*nLevel + 1 ]; nLevel = 0; for( pPool = this; pPool; pPool = pPool->pImp->mpSecondary ) { *(pWhichRanges+(nLevel++)) = pPool->pImp->mnStart; *(pWhichRanges+(nLevel++)) = pPool->pImp->mnEnd; *(pWhichRanges+nLevel) = 0; } } const sal_uInt16* SfxItemPool::GetFrozenIdRanges() const { return pImp->mpPoolRanges; } const SfxPoolItem *SfxItemPool::GetItem2(sal_uInt16 nWhich, sal_uInt32 nOfst) const { DBG_CHKTHIS(SfxItemPool, 0); if ( !IsInRange(nWhich) ) { if ( pImp->mpSecondary ) return pImp->mpSecondary->GetItem2( nWhich, nOfst ); SFX_ASSERT( 0, nWhich, "unknown Which-Id - cannot resolve surrogate" ); return 0; } // dflt-Attribut? if ( nOfst == SFX_ITEMS_DEFAULT ) return *(pImp->ppStaticDefaults + GetIndex_Impl(nWhich)); SfxPoolItemArray_Impl* pItemArr = pImp->maPoolItems[GetIndex_Impl(nWhich)]; if( pItemArr && nOfst < pItemArr->size() ) return (*pItemArr)[nOfst]; return 0; } // ----------------------------------------------------------------------- sal_uInt32 SfxItemPool::GetItemCount2(sal_uInt16 nWhich) const { DBG_CHKTHIS(SfxItemPool, 0); if ( !IsInRange(nWhich) ) { if ( pImp->mpSecondary ) return pImp->mpSecondary->GetItemCount2( nWhich ); SFX_ASSERT( 0, nWhich, "unknown Which-Id - cannot resolve surrogate" ); return 0; } SfxPoolItemArray_Impl* pItemArr = pImp->maPoolItems[GetIndex_Impl(nWhich)]; if ( pItemArr ) return pItemArr->size(); return 0; } // ----------------------------------------------------------------------- sal_uInt16 SfxItemPool::GetWhich( sal_uInt16 nSlotId, sal_Bool bDeep ) const { if ( !IsSlot(nSlotId) ) return nSlotId; sal_uInt16 nCount = pImp->mnEnd - pImp->mnStart + 1; for ( sal_uInt16 nOfs = 0; nOfs < nCount; ++nOfs ) if ( pItemInfos[nOfs]._nSID == nSlotId ) return nOfs + pImp->mnStart; if ( pImp->mpSecondary && bDeep ) return pImp->mpSecondary->GetWhich(nSlotId); return nSlotId; } // ----------------------------------------------------------------------- sal_uInt16 SfxItemPool::GetSlotId( sal_uInt16 nWhich, sal_Bool bDeep ) const { if ( !IsWhich(nWhich) ) return nWhich; if ( !IsInRange( nWhich ) ) { if ( pImp->mpSecondary && bDeep ) return pImp->mpSecondary->GetSlotId(nWhich); SFX_ASSERT( 0, nWhich, "unknown Which-Id - cannot get slot-id" ); return 0; } sal_uInt16 nSID = pItemInfos[nWhich - pImp->mnStart]._nSID; return nSID ? nSID : nWhich; } // ----------------------------------------------------------------------- sal_uInt16 SfxItemPool::GetTrueWhich( sal_uInt16 nSlotId, sal_Bool bDeep ) const { if ( !IsSlot(nSlotId) ) return 0; sal_uInt16 nCount = pImp->mnEnd - pImp->mnStart + 1; for ( sal_uInt16 nOfs = 0; nOfs < nCount; ++nOfs ) if ( pItemInfos[nOfs]._nSID == nSlotId ) return nOfs + pImp->mnStart; if ( pImp->mpSecondary && bDeep ) return pImp->mpSecondary->GetTrueWhich(nSlotId); return 0; } // ----------------------------------------------------------------------- sal_uInt16 SfxItemPool::GetTrueSlotId( sal_uInt16 nWhich, sal_Bool bDeep ) const { if ( !IsWhich(nWhich) ) return 0; if ( !IsInRange( nWhich ) ) { if ( pImp->mpSecondary && bDeep ) return pImp->mpSecondary->GetTrueSlotId(nWhich); SFX_ASSERT( 0, nWhich, "unknown Which-Id - cannot get slot-id" ); return 0; } return pItemInfos[nWhich - pImp->mnStart]._nSID; } void SfxItemPool::SetFileFormatVersion( sal_uInt16 nFileFormatVersion ) /* [Description] You must call this function to set the file format version after concatenating your secondary-pools but before you store any pool, itemset or item. Only set the version at the master pool, never at any secondary pool. */ { DBG_ASSERT( this == pImp->mpMaster, "SfxItemPool::SetFileFormatVersion() but not a master pool" ); for ( SfxItemPool *pPool = this; pPool; pPool = pPool->pImp->mpSecondary ) pPool->pImp->mnFileFormatVersion = nFileFormatVersion; } const SfxItemPool* SfxItemPool::pStoringPool_ = 0; /* vim:set shiftwidth=4 softtabstop=4 expandtab: */