/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed * with this work for additional information regarding copyright * ownership. The ASF licenses this file to you under the Apache * License, Version 2.0 (the "License"); you may not use this file * except in compliance with the License. You may obtain a copy of * the License at http://www.apache.org/licenses/LICENSE-2.0 . */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace com::sun::star; // At the names' publication, this character is removed again and the // family is newly generated. // In addition now there is the Bit bPhysical. In case this Bit is // TRUE, the Pool-Formatnames are not being submitted. namespace { class EEStyleSheet : public SfxStyleSheet { public: using SfxStyleSheet::SfxStyleSheet; bool IsUsed() const override { bool bResult = false; ForAllListeners( [&bResult] (SfxListener* pListener) { auto pUser(dynamic_cast(pListener)); bResult = pUser && pUser->isUsedByModel(); return bResult; }); return bResult; } void Notify(SfxBroadcaster& rBC, const SfxHint& rHint) override { if (rHint.GetId() == SfxHintId::DataChanged) Broadcast(rHint); else SfxStyleSheet::Notify(rBC, rHint); } bool SetParent(const OUString& rName) override { if (SfxStyleSheet::SetParent(rName)) { auto pStyle = m_pPool->Find(rName, nFamily); pSet->SetParent(pStyle ? &pStyle->GetItemSet() : nullptr); return true; } return false; } }; class EEStyleSheetPool : public SfxStyleSheetPool, public SfxListener { SfxStyleSheetBasePool* m_pOwner; public: explicit EEStyleSheetPool(SfxStyleSheetBasePool* pOwner) : SfxStyleSheetPool(pOwner->GetPool()) , m_pOwner(pOwner) { StartListening(*m_pOwner); } using SfxStyleSheetPool::Create; rtl::Reference Create(const OUString& rName, SfxStyleFamily eFamily, SfxStyleSearchBits nMask) override { return new EEStyleSheet(rName, *this, eFamily, nMask); } void Notify(SfxBroadcaster&, const SfxHint& rHint) override { auto pHint = dynamic_cast(&rHint); if (!pHint) return; auto nId = pHint->GetId(); auto pDocStyleSheet = pHint->GetStyleSheet(); const SfxStyleSheetModifiedHint* pExtendedHint = nullptr; if (nId == SfxHintId::StyleSheetModifiedExtended) pExtendedHint = static_cast(&rHint); const OUString aName = pExtendedHint ? pExtendedHint->GetOldName() : pDocStyleSheet->GetName(); auto pStyleSheet = SfxStyleSheetPool::Find(aName, pDocStyleSheet->GetFamily()); if (!pStyleSheet) return; if (nId == SfxHintId::StyleSheetModified || nId == SfxHintId::StyleSheetModifiedExtended) { pStyleSheet->SetName(pDocStyleSheet->GetName()); UpdateStyleHierarchyFrom(pStyleSheet, pDocStyleSheet); static_cast(pStyleSheet)->Broadcast(SfxHint(SfxHintId::DataChanged)); } else if (nId == SfxHintId::StyleSheetErased) Remove(pStyleSheet); } SfxStyleSheetBase* Find(const OUString& rName, SfxStyleFamily eFamily, SfxStyleSearchBits = SfxStyleSearchBits::All) override { auto pStyleSheet = SfxStyleSheetPool::Find(rName, eFamily); if (auto pDocStyleSheet = pStyleSheet ? nullptr : m_pOwner->Find(rName, eFamily)) { pStyleSheet = &Make(pDocStyleSheet->GetName(), pDocStyleSheet->GetFamily()); UpdateStyleHierarchyFrom(pStyleSheet, pDocStyleSheet); } return pStyleSheet; } void UpdateStyleHierarchyFrom(SfxStyleSheetBase* pStyleSheet, SfxStyleSheetBase* pDocStyleSheet) { FillItemSet(pStyleSheet, pDocStyleSheet); // Remember now, as the next calls will invalidate pDocStyleSheet. const OUString aParent = pDocStyleSheet->GetParent(); const OUString aFollow = pDocStyleSheet->GetFollow(); if (pStyleSheet->GetParent() != aParent) pStyleSheet->SetParent(aParent); if (pStyleSheet->GetFollow() != aFollow) pStyleSheet->SetFollow(aFollow); } void FillItemSet(SfxStyleSheetBase* pDestSheet, SfxStyleSheetBase* pSourceSheet) { auto& rItemSet = pDestSheet->GetItemSet(); rItemSet.ClearItem(); auto pCol = static_cast(pSourceSheet)->GetCollection(); SfxItemIter aIter(pCol->GetAttrSet()); std::optional oLRSpaceItem; for (auto pItem = aIter.GetCurItem(); pItem; pItem = aIter.NextItem()) { if (aIter.GetItemState(false) != SfxItemState::SET) continue; auto nWhich = pItem->Which(); auto nSlotId = rPool.GetSlotId(nWhich); auto nNewWhich = rPool.GetSecondaryPool()->GetWhichIDFromSlotID(nSlotId); if (nNewWhich != nSlotId) rItemSet.Put(pItem->CloneSetWhich(nNewWhich)); else if (nWhich == RES_MARGIN_FIRSTLINE) { if (!oLRSpaceItem) oLRSpaceItem.emplace(EE_PARA_LRSPACE); auto pFirstLineItem = static_cast(pItem); (*oLRSpaceItem).SetTextFirstLineOffsetValue(pFirstLineItem->GetTextFirstLineOffset()); (*oLRSpaceItem).SetAutoFirst(pFirstLineItem->IsAutoFirst()); } else if (nWhich == RES_MARGIN_TEXTLEFT) { if (!oLRSpaceItem) oLRSpaceItem.emplace(EE_PARA_LRSPACE); (*oLRSpaceItem).SetTextLeft(static_cast(pItem)->GetTextLeft()); } else if (nWhich == RES_MARGIN_RIGHT) { if (!oLRSpaceItem) oLRSpaceItem.emplace(EE_PARA_LRSPACE); (*oLRSpaceItem).SetRight(static_cast(pItem)->GetRight()); } else if (nWhich == RES_CHRATR_BACKGROUND) { auto pBrushItem = static_cast(pItem); rItemSet.Put(SvxColorItem(pBrushItem->GetColor(), EE_CHAR_BKGCOLOR)); } } if (oLRSpaceItem) rItemSet.Put(*oLRSpaceItem); } }; class SwImplShellAction { SwWrtShell* m_pSh; std::unique_ptr m_pCurrSh; public: explicit SwImplShellAction( SwDoc& rDoc ); ~SwImplShellAction() COVERITY_NOEXCEPT_FALSE; SwImplShellAction(const SwImplShellAction&) = delete; SwImplShellAction& operator=(const SwImplShellAction&) = delete; }; } SwImplShellAction::SwImplShellAction( SwDoc& rDoc ) { if( rDoc.GetDocShell() ) m_pSh = rDoc.GetDocShell()->GetWrtShell(); else m_pSh = nullptr; if( m_pSh ) { m_pCurrSh.reset( new CurrShell( m_pSh ) ); m_pSh->StartAllAction(); } } SwImplShellAction::~SwImplShellAction() COVERITY_NOEXCEPT_FALSE { if( m_pCurrSh ) { m_pSh->EndAllAction(); m_pCurrSh.reset(); } } // find/create SwCharFormate // possibly fill Style static SwCharFormat* lcl_FindCharFormat( SwDoc& rDoc, const OUString& rName, SwDocStyleSheet* pStyle = nullptr, bool bCreate = true ) { SwCharFormat* pFormat = nullptr; if (!rName.isEmpty()) { pFormat = rDoc.FindCharFormatByName( rName ); if( !pFormat && rName == SwResId(STR_POOLCHR_STANDARD)) { // Standard-Character template pFormat = rDoc.GetDfltCharFormat(); } if( !pFormat && bCreate ) { // explore Pool const sal_uInt16 nId = SwStyleNameMapper::GetPoolIdFromUIName(rName, SwGetPoolIdFromName::ChrFmt); if(nId != USHRT_MAX) pFormat = rDoc.getIDocumentStylePoolAccess().GetCharFormatFromPool(nId); } } if(pStyle) { if(pFormat) { pStyle->SetPhysical(true); SwFormat* p = pFormat->DerivedFrom(); if( p && !p->IsDefault() ) pStyle->PresetParent( p->GetName() ); else pStyle->PresetParent( OUString() ); } else pStyle->SetPhysical(false); } return pFormat; } // find/create ParaFormats // fill Style static SwTextFormatColl* lcl_FindParaFormat( SwDoc& rDoc, const OUString& rName, SwDocStyleSheet* pStyle = nullptr, bool bCreate = true ) { SwTextFormatColl* pColl = nullptr; if (!rName.isEmpty()) { pColl = rDoc.FindTextFormatCollByName( rName ); if( !pColl && bCreate ) { // explore Pool const sal_uInt16 nId = SwStyleNameMapper::GetPoolIdFromUIName(rName, SwGetPoolIdFromName::TxtColl); if(nId != USHRT_MAX) pColl = rDoc.getIDocumentStylePoolAccess().GetTextCollFromPool(nId); } } if(pStyle) { if(pColl) { pStyle->SetPhysical(true); if( pColl->DerivedFrom() && !pColl->DerivedFrom()->IsDefault() ) pStyle->PresetParent( pColl->DerivedFrom()->GetName() ); else pStyle->PresetParent( OUString() ); SwTextFormatColl& rNext = pColl->GetNextTextFormatColl(); pStyle->PresetFollow(rNext.GetName()); } else pStyle->SetPhysical(false); } return pColl; } // Border formats static SwFrameFormat* lcl_FindFrameFormat( SwDoc& rDoc, const OUString& rName, SwDocStyleSheet* pStyle = nullptr, bool bCreate = true ) { SwFrameFormat* pFormat = nullptr; if( !rName.isEmpty() ) { pFormat = rDoc.FindFrameFormatByName( rName ); if( !pFormat && bCreate ) { // explore Pool const sal_uInt16 nId = SwStyleNameMapper::GetPoolIdFromUIName(rName, SwGetPoolIdFromName::FrmFmt); if(nId != USHRT_MAX) pFormat = rDoc.getIDocumentStylePoolAccess().GetFrameFormatFromPool(nId); } } if(pStyle) { if(pFormat) { pStyle->SetPhysical(true); if( pFormat->DerivedFrom() && !pFormat->DerivedFrom()->IsDefault() ) pStyle->PresetParent( pFormat->DerivedFrom()->GetName() ); else pStyle->PresetParent( OUString() ); } else pStyle->SetPhysical(false); } return pFormat; } // Page descriptors static const SwPageDesc* lcl_FindPageDesc( SwDoc& rDoc, const OUString& rName, SwDocStyleSheet* pStyle = nullptr, bool bCreate = true ) { const SwPageDesc* pDesc = nullptr; if (!rName.isEmpty()) { pDesc = rDoc.FindPageDesc(rName); if( !pDesc && bCreate ) { sal_uInt16 nId = SwStyleNameMapper::GetPoolIdFromUIName(rName, SwGetPoolIdFromName::PageDesc); if(nId != USHRT_MAX) pDesc = rDoc.getIDocumentStylePoolAccess().GetPageDescFromPool(nId); } } if(pStyle) { if(pDesc) { pStyle->SetPhysical(true); if(pDesc->GetFollow()) pStyle->PresetFollow(pDesc->GetFollow()->GetName()); else pStyle->PresetParent( OUString() ); } else pStyle->SetPhysical(false); } return pDesc; } static const SwNumRule* lcl_FindNumRule( SwDoc& rDoc, const OUString& rName, SwDocStyleSheet* pStyle = nullptr, bool bCreate = true ) { const SwNumRule* pRule = nullptr; if (!rName.isEmpty()) { pRule = rDoc.FindNumRulePtr( rName ); if( !pRule && bCreate ) { sal_uInt16 nId = SwStyleNameMapper::GetPoolIdFromUIName(rName, SwGetPoolIdFromName::NumRule); if(nId != USHRT_MAX) pRule = rDoc.getIDocumentStylePoolAccess().GetNumRuleFromPool(nId); } } if(pStyle) { if(pRule) { pStyle->SetPhysical(true); pStyle->PresetParent( OUString() ); } else pStyle->SetPhysical(false); } return pRule; } static SwTableAutoFormat* lcl_FindTableStyle(SwDoc& rDoc, const OUString& rName, SwDocStyleSheet *pStyle = nullptr, bool bCreate = true) { SwTableAutoFormat* pFormat = nullptr; if (!rName.isEmpty()) { pFormat = rDoc.GetTableStyles().FindAutoFormat(rName); if (!pFormat && bCreate) { SwTableAutoFormat aNew(rName); rDoc.GetTableStyles().AddAutoFormat(aNew); } } if(pStyle) { if(pFormat) { pStyle->SetPhysical(true); pStyle->PresetParent(OUString()); } else pStyle->SetPhysical(false); } return pFormat; } static const SwBoxAutoFormat* lcl_FindCellStyle(SwDoc& rDoc, std::u16string_view rName, SwDocStyleSheet *pStyle) { const SwBoxAutoFormat* pFormat = rDoc.GetCellStyles().GetBoxFormat(rName); if (!pFormat) { const auto& aTableTemplateMap = SwTableAutoFormat::GetTableTemplateMap(); SwTableAutoFormatTable& rTableStyles = rDoc.GetTableStyles(); for (size_t i=0; i < rTableStyles.size() && !pFormat; ++i) { const SwTableAutoFormat& rTableStyle = rTableStyles[i]; for (size_t nBoxFormat=0; nBoxFormat < aTableTemplateMap.size() && !pFormat; ++nBoxFormat) { const sal_uInt32 nBoxIndex = aTableTemplateMap[nBoxFormat]; const SwBoxAutoFormat& rBoxFormat = rTableStyle.GetBoxFormat(nBoxIndex); OUString sBoxFormatName; SwStyleNameMapper::FillProgName(rTableStyle.GetName(), sBoxFormatName, SwGetPoolIdFromName::TabStyle); sBoxFormatName += rTableStyle.GetTableTemplateCellSubName(rBoxFormat); if (rName == sBoxFormatName) pFormat = &rBoxFormat; } } } if(pStyle) { if(pFormat) { pStyle->SetPhysical(true); pStyle->PresetParent(OUString()); } else pStyle->SetPhysical(false); } return pFormat; } sal_uInt32 SwStyleSheetIterator::SwPoolFormatList::FindName(SfxStyleFamily eFam, const OUString& rName) { if(!maImpl.empty()) { UniqueHash::const_iterator it = maUnique.find(std::pair{eFam, rName}); if (it != maUnique.end()) { sal_uInt32 nIdx = it->second; assert (nIdx < maImpl.size()); assert (maImpl.size() == maUnique.size()); return nIdx; } } return SAL_MAX_UINT32; } void SwStyleSheetIterator::SwPoolFormatList::rehash() { maUnique.clear(); for (size_t i = 0; i < maImpl.size(); i++) maUnique[maImpl[i]] = i; assert (maImpl.size() == maUnique.size()); } void SwStyleSheetIterator::SwPoolFormatList::RemoveName(SfxStyleFamily eFam, const OUString& rName) { sal_uInt32 nTmpPos = FindName( eFam, rName ); if (nTmpPos != SAL_MAX_UINT32) maImpl.erase(maImpl.begin() + nTmpPos); // assumption: this seldom occurs, the iterator is built, then emptied. rehash(); assert (maImpl.size() == maUnique.size()); } // Add Strings to the list of templates void SwStyleSheetIterator::SwPoolFormatList::Append( SfxStyleFamily eFam, const OUString& rStr ) { UniqueHash::const_iterator it = maUnique.find(std::pair{eFam, rStr}); if (it != maUnique.end()) return; maUnique.emplace(std::pair{eFam, rStr}, static_cast(maImpl.size())); maImpl.push_back(std::pair{eFam, rStr}); } // UI-sided implementation of StyleSheets // uses the Core-Engine SwDocStyleSheet::SwDocStyleSheet( SwDoc& rDocument, SwDocStyleSheetPool& rPool) : SfxStyleSheetBase( OUString(), &rPool, SfxStyleFamily::Char, SfxStyleSearchBits::Auto ), m_pCharFormat(nullptr), m_pColl(nullptr), m_pFrameFormat(nullptr), m_pDesc(nullptr), m_pNumRule(nullptr), m_pTableFormat(nullptr), m_pBoxFormat(nullptr), m_rDoc(rDocument), m_aCoreSet( rPool.GetPool(), svl::Items< RES_CHRATR_BEGIN, RES_CHRATR_END - 1, RES_PARATR_BEGIN, RES_FRMATR_END - 1, RES_UNKNOWNATR_BEGIN, RES_UNKNOWNATR_END - 1, // FillAttribute support: XATTR_FILL_FIRST, XATTR_FILL_LAST, SID_ATTR_BORDER_INNER, SID_ATTR_BORDER_INNER, SID_ATTR_PAGE, SID_ATTR_PAGE_EXT1, SID_ATTR_PAGE_HEADERSET, SID_ATTR_PAGE_FOOTERSET, SID_ATTR_PARA_MODEL, SID_ATTR_PARA_MODEL, // Items to hand over XPropertyList things like XColorList, // XHatchList, XGradientList, and XBitmapList to the Area TabPage: SID_COLOR_TABLE, SID_PATTERN_LIST, SID_SWREGISTER_COLLECTION, SID_SWREGISTER_COLLECTION, SID_ATTR_PARA_PAGENUM, SID_ATTR_PARA_PAGENUM, SID_SWREGISTER_MODE, SID_SWREGISTER_MODE, SID_ATTR_BRUSH_CHAR, SID_ATTR_BRUSH_CHAR, SID_ATTR_NUMBERING_RULE, SID_ATTR_NUMBERING_RULE, SID_ATTR_CHAR_GRABBAG, SID_ATTR_CHAR_GRABBAG, SID_ATTR_AUTO_STYLE_UPDATE, SID_ATTR_AUTO_STYLE_UPDATE, FN_PARAM_FTN_INFO, FN_PARAM_FTN_INFO, FN_KEEP_ASPECT_RATIO, FN_KEEP_ASPECT_RATIO, FN_COND_COLL, FN_COND_COLL>), m_bPhysical(false) { nHelpId = UCHAR_MAX; } SwDocStyleSheet::SwDocStyleSheet( const SwDocStyleSheet& ) = default; SwDocStyleSheet::~SwDocStyleSheet() = default; void SwDocStyleSheet::Reset() { aName.clear(); aFollow.clear(); aParent.clear(); SetPhysical(false); } void SwDocStyleSheet::SetGrabBagItem(const uno::Any& rVal) { bool bChg = false; if (!m_bPhysical) FillStyleSheet(FillPhysical); SwFormat* pFormat = nullptr; switch (nFamily) { case SfxStyleFamily::Char: pFormat = m_rDoc.FindCharFormatByName(aName); if (pFormat) { pFormat->SetGrabBagItem(rVal); bChg = true; } break; case SfxStyleFamily::Para: pFormat = m_rDoc.FindTextFormatCollByName(aName); if (pFormat) { pFormat->SetGrabBagItem(rVal); bChg = true; } break; case SfxStyleFamily::Pseudo: { SwNumRule* pRule = m_rDoc.FindNumRulePtr(aName); if (pRule) { pRule->SetGrabBagItem(rVal); bChg = true; } } break; default: break; } if (bChg) { dynamic_cast(*m_pPool).InvalidateIterator(); m_pPool->Broadcast(SfxStyleSheetHint(SfxHintId::StyleSheetModified, *this)); if (SwEditShell* pSh = m_rDoc.GetEditShell()) pSh->CallChgLnk(); } } void SwDocStyleSheet::GetGrabBagItem(uno::Any& rVal) const { SwFormat* pFormat = nullptr; switch (nFamily) { case SfxStyleFamily::Char: pFormat = m_rDoc.FindCharFormatByName(aName); if (pFormat) pFormat->GetGrabBagItem(rVal); break; case SfxStyleFamily::Para: pFormat = m_rDoc.FindTextFormatCollByName(aName); if (pFormat) pFormat->GetGrabBagItem(rVal); break; case SfxStyleFamily::Pseudo: { SwNumRule* pRule = m_rDoc.FindNumRulePtr(aName); if (pRule) pRule->GetGrabBagItem(rVal); } break; default: break; } } // virtual methods void SwDocStyleSheet::SetHidden( bool bValue ) { bool bChg = false; if(!m_bPhysical) FillStyleSheet( FillPhysical ); SwFormat* pFormat = nullptr; switch(nFamily) { case SfxStyleFamily::Char: pFormat = m_rDoc.FindCharFormatByName( aName ); if ( pFormat ) { pFormat->SetHidden( bValue ); bChg = true; } break; case SfxStyleFamily::Para: pFormat = m_rDoc.FindTextFormatCollByName( aName ); if ( pFormat ) { pFormat->SetHidden( bValue ); bChg = true; } break; case SfxStyleFamily::Frame: pFormat = m_rDoc.FindFrameFormatByName( aName ); if ( pFormat ) { pFormat->SetHidden( bValue ); bChg = true; } break; case SfxStyleFamily::Page: { SwPageDesc* pPgDesc = m_rDoc.FindPageDesc(aName); if ( pPgDesc ) { pPgDesc->SetHidden( bValue ); bChg = true; } } break; case SfxStyleFamily::Pseudo: { SwNumRule* pRule = m_rDoc.FindNumRulePtr( aName ); if ( pRule ) { pRule->SetHidden( bValue ); bChg = true; } } break; case SfxStyleFamily::Table: { SwTableAutoFormat* pTableAutoFormat = m_rDoc.GetTableStyles().FindAutoFormat( aName ); if ( pTableAutoFormat ) { pTableAutoFormat->SetHidden( bValue ); bChg = true; } } break; default: break; } if( bChg ) { // calling pPool->First() here would be quite slow... dynamic_cast(*m_pPool).InvalidateIterator(); // internal list has to be updated m_pPool->Broadcast( SfxStyleSheetHint( SfxHintId::StyleSheetModified, *this ) ); if (SwEditShell* pSh = m_rDoc.GetEditShell()) pSh->CallChgLnk(); } } bool SwDocStyleSheet::IsHidden( ) const { bool bRet = false; SwFormat* pFormat = nullptr; switch(nFamily) { case SfxStyleFamily::Char: pFormat = m_rDoc.FindCharFormatByName( aName ); bRet = pFormat && pFormat->IsHidden( ); break; case SfxStyleFamily::Para: pFormat = m_rDoc.FindTextFormatCollByName( aName ); bRet = pFormat && pFormat->IsHidden( ); break; case SfxStyleFamily::Frame: pFormat = m_rDoc.FindFrameFormatByName( aName ); bRet = pFormat && pFormat->IsHidden( ); break; case SfxStyleFamily::Page: { SwPageDesc* pPgDesc = m_rDoc.FindPageDesc(aName); bRet = pPgDesc && pPgDesc->IsHidden( ); } break; case SfxStyleFamily::Pseudo: { SwNumRule* pRule = m_rDoc.FindNumRulePtr( aName ); bRet = pRule && pRule->IsHidden( ); } break; case SfxStyleFamily::Table: { SwTableAutoFormat* pTableAutoFormat = m_rDoc.GetTableStyles().FindAutoFormat( aName ); bRet = pTableAutoFormat && pTableAutoFormat->IsHidden( ); } break; default: break; } return bRet; } const OUString& SwDocStyleSheet::GetParent() const { if( !m_bPhysical ) { // check if it's already in document SwFormat* pFormat = nullptr; SwGetPoolIdFromName eGetType; switch(nFamily) { case SfxStyleFamily::Char: pFormat = m_rDoc.FindCharFormatByName( aName ); eGetType = SwGetPoolIdFromName::ChrFmt; break; case SfxStyleFamily::Para: pFormat = m_rDoc.FindTextFormatCollByName( aName ); eGetType = SwGetPoolIdFromName::TxtColl; break; case SfxStyleFamily::Frame: pFormat = m_rDoc.FindFrameFormatByName( aName ); eGetType = SwGetPoolIdFromName::FrmFmt; break; case SfxStyleFamily::Page: case SfxStyleFamily::Pseudo: default: { static const OUString sEmpty; return sEmpty; // there's no parent } } OUString sTmp; if( !pFormat ) // not yet there, so default Parent { sal_uInt16 i = SwStyleNameMapper::GetPoolIdFromUIName( aName, eGetType ); i = ::GetPoolParent( i ); if( i && USHRT_MAX != i ) SwStyleNameMapper::FillUIName( i, sTmp ); } else { SwFormat* p = pFormat->DerivedFrom(); if( p && !p->IsDefault() ) sTmp = p->GetName(); } SwDocStyleSheet* pThis = const_cast(this); pThis->aParent = sTmp; } return aParent; } // Follower const OUString& SwDocStyleSheet::GetFollow() const { if( !m_bPhysical ) { SwDocStyleSheet* pThis = const_cast(this); pThis->FillStyleSheet( FillAllInfo ); } return aFollow; } void SwDocStyleSheet::SetLink(const OUString& rStr) { SwImplShellAction aTmpSh(m_rDoc); switch (nFamily) { case SfxStyleFamily::Para: { if (m_pColl) { SwCharFormat* pLink = lcl_FindCharFormat(m_rDoc, rStr); if (pLink) { m_pColl->SetLinkedCharFormat(pLink); } } break; } case SfxStyleFamily::Char: { if (m_pCharFormat) { SwTextFormatColl* pLink = lcl_FindParaFormat(m_rDoc, rStr); if (pLink) { m_pCharFormat->SetLinkedParaFormat(pLink); } } break; } default: break; } } const OUString& SwDocStyleSheet::GetLink() const { if (!m_bPhysical) { SwDocStyleSheet* pThis = const_cast(this); pThis->FillStyleSheet(FillAllInfo); } return m_aLink; } // What Linkage is possible bool SwDocStyleSheet::HasFollowSupport() const { switch(nFamily) { case SfxStyleFamily::Para : case SfxStyleFamily::Page : return true; case SfxStyleFamily::Frame: case SfxStyleFamily::Char : case SfxStyleFamily::Pseudo: return false; default: OSL_ENSURE(false, "unknown style family"); } return false; } // Parent ? bool SwDocStyleSheet::HasParentSupport() const { bool bRet = false; switch(nFamily) { case SfxStyleFamily::Char : case SfxStyleFamily::Para : case SfxStyleFamily::Frame: bRet = true; break; default:; //prevent warning } return bRet; } bool SwDocStyleSheet::HasClearParentSupport() const { bool bRet = false; switch(nFamily) { case SfxStyleFamily::Para : case SfxStyleFamily::Char : case SfxStyleFamily::Frame: bRet = true; break; default:; //prevent warning } return bRet; } // determine textual description OUString SwDocStyleSheet::GetDescription(MapUnit eUnit) { IntlWrapper aIntlWrapper(SvtSysLocale().GetUILanguageTag()); static constexpr OUString sPlus(u" + "_ustr); if ( SfxStyleFamily::Page == nFamily ) { if( !pSet ) GetItemSet(); SfxItemIter aIter( *pSet ); OUStringBuffer aDesc; for (const SfxPoolItem* pItem = aIter.GetCurItem(); pItem; pItem = aIter.NextItem()) { if(!IsInvalidItem(pItem)) { switch ( pItem->Which() ) { case RES_LR_SPACE: case SID_ATTR_PAGE_SIZE: case SID_ATTR_PAGE_MAXSIZE: case SID_ATTR_PAGE_PAPERBIN: case SID_ATTR_BORDER_INNER: break; default: { OUString aItemPresentation; if ( !IsInvalidItem( pItem ) && m_pPool->GetPool().GetPresentation( *pItem, eUnit, aItemPresentation, aIntlWrapper ) ) { if ( !aDesc.isEmpty() && !aItemPresentation.isEmpty() ) aDesc.append(sPlus); aDesc.append(aItemPresentation); } } } } } return aDesc.makeStringAndClear(); } if ( SfxStyleFamily::Frame == nFamily || SfxStyleFamily::Para == nFamily || SfxStyleFamily::Char == nFamily ) { if( !pSet ) GetItemSet(); SfxItemIter aIter( *pSet ); OUStringBuffer aDesc; OUString sPageNum; OUString sModel; OUString sBreak; bool bHasWesternFontPrefix = false; bool bHasCJKFontPrefix = false; bool bHasCTLFontPrefix = false; SvtCTLOptions aCTLOptions; // Get currently used FillStyle and remember, also need the XFillFloatTransparenceItem // to decide if gradient transparence is used const drawing::FillStyle eFillStyle(pSet->Get(XATTR_FILLSTYLE).GetValue()); const bool bUseFloatTransparence(pSet->Get(XATTR_FILLFLOATTRANSPARENCE).IsEnabled()); for (const SfxPoolItem* pItem = aIter.GetCurItem(); pItem; pItem = aIter.NextItem()) { if(!IsInvalidItem(pItem)) { switch ( pItem->Which() ) { case SID_ATTR_AUTO_STYLE_UPDATE: case RES_PAGEDESC: break; default: { OUString aItemPresentation; if ( !IsInvalidItem( pItem ) && m_pPool->GetPool().GetPresentation( *pItem, eUnit, aItemPresentation, aIntlWrapper ) ) { bool bIsDefault = false; switch ( pItem->Which() ) { case XATTR_FILLCOLOR: { // only use active FillStyle information bIsDefault = (drawing::FillStyle_SOLID == eFillStyle); break; } case XATTR_FILLGRADIENT: { // only use active FillStyle information bIsDefault = (drawing::FillStyle_GRADIENT == eFillStyle); break; } case XATTR_FILLHATCH: { // only use active FillStyle information bIsDefault = (drawing::FillStyle_HATCH == eFillStyle); break; } case XATTR_FILLBITMAP: { // only use active FillStyle information bIsDefault = (drawing::FillStyle_BITMAP == eFillStyle); break; } case XATTR_FILLTRANSPARENCE: { // only active when not FloatTransparence bIsDefault = !bUseFloatTransparence; break; } case XATTR_FILLFLOATTRANSPARENCE: { // only active when FloatTransparence bIsDefault = bUseFloatTransparence; break; } case SID_ATTR_PARA_PAGENUM: sPageNum = aItemPresentation; break; case SID_ATTR_PARA_MODEL: sModel = aItemPresentation; break; case RES_BREAK: sBreak = aItemPresentation; break; case RES_CHRATR_CJK_FONT: case RES_CHRATR_CJK_FONTSIZE: case RES_CHRATR_CJK_LANGUAGE: case RES_CHRATR_CJK_POSTURE: case RES_CHRATR_CJK_WEIGHT: if(SvtCJKOptions::IsCJKFontEnabled()) bIsDefault = true; if(!bHasCJKFontPrefix) { aItemPresentation = SwResId(STR_CJK_FONT) + aItemPresentation; bHasCJKFontPrefix = true; } break; case RES_CHRATR_CTL_FONT: case RES_CHRATR_CTL_FONTSIZE: case RES_CHRATR_CTL_LANGUAGE: case RES_CHRATR_CTL_POSTURE: case RES_CHRATR_CTL_WEIGHT: if(SvtCTLOptions::IsCTLFontEnabled()) bIsDefault = true; if(!bHasCTLFontPrefix) { aItemPresentation = SwResId(STR_CTL_FONT) + aItemPresentation; bHasCTLFontPrefix = true; } break; case RES_CHRATR_FONT: case RES_CHRATR_FONTSIZE: case RES_CHRATR_LANGUAGE: case RES_CHRATR_POSTURE: case RES_CHRATR_WEIGHT: if(!bHasWesternFontPrefix) { aItemPresentation = SwResId(STR_WESTERN_FONT) + aItemPresentation; bHasWesternFontPrefix = true; } [[fallthrough]]; default: bIsDefault = true; } if(bIsDefault) { if ( !aDesc.isEmpty() && !aItemPresentation.isEmpty() ) aDesc.append(sPlus); aDesc.append(aItemPresentation); } } } } } } // Special treatment for Break, Page template and Site offset if (!sModel.isEmpty()) { if (!aDesc.isEmpty()) aDesc.append(sPlus); aDesc.append(SwResId(STR_PAGEBREAK) + sPlus + sModel); if (sPageNum != "0") { aDesc.append(sPlus + SwResId(STR_PAGEOFFSET) + sPageNum); } } else if (!sBreak.isEmpty()) // Break can be valid only when NO Model { if (!aDesc.isEmpty()) aDesc.append(sPlus); aDesc.append(sBreak); } return aDesc.makeStringAndClear(); } if( SfxStyleFamily::Pseudo == nFamily ) { return OUString(); } return SfxStyleSheetBase::GetDescription(eUnit); } // Set names bool SwDocStyleSheet::SetName(const OUString& rStr, bool bReindexNow) { if( rStr.isEmpty() ) return false; if( aName != rStr ) { if( !SfxStyleSheetBase::SetName(rStr, bReindexNow)) return false; } else if(!m_bPhysical) FillStyleSheet( FillPhysical ); bool bChg = false; switch(nFamily) { case SfxStyleFamily::Char : { OSL_ENSURE(m_pCharFormat, "SwCharFormat missing!"); if( m_pCharFormat && m_pCharFormat->GetName() != rStr ) { if (!m_pCharFormat->GetName().isEmpty()) m_rDoc.RenameFormat(*m_pCharFormat, rStr); else m_pCharFormat->SetFormatName(rStr); bChg = true; } break; } case SfxStyleFamily::Para : { OSL_ENSURE(m_pColl, "Collection missing!"); if( m_pColl && m_pColl->GetName() != rStr ) { if (!m_pColl->GetName().isEmpty()) m_rDoc.RenameFormat(*m_pColl, rStr); else m_pColl->SetFormatName(rStr); bChg = true; } break; } case SfxStyleFamily::Frame: { OSL_ENSURE(m_pFrameFormat, "FrameFormat missing!"); if( m_pFrameFormat && m_pFrameFormat->GetName() != rStr ) { if (!m_pFrameFormat->GetName().isEmpty()) m_rDoc.RenameFormat(*m_pFrameFormat, rStr); else m_pFrameFormat->SetFormatName( rStr ); bChg = true; } break; } case SfxStyleFamily::Page : OSL_ENSURE(m_pDesc, "PageDesc missing!"); if( m_pDesc && m_pDesc->GetName() != rStr ) { // Set PageDesc - copy with earlier one - probably not // necessary for setting the name. So here we allow a // cast. SwPageDesc aPageDesc(*const_cast(m_pDesc)); const OUString aOldName(aPageDesc.GetName()); aPageDesc.SetName( rStr ); bool const bDoesUndo = m_rDoc.GetIDocumentUndoRedo().DoesUndo(); m_rDoc.GetIDocumentUndoRedo().DoUndo(!aOldName.isEmpty()); m_rDoc.ChgPageDesc(aOldName, aPageDesc); m_rDoc.GetIDocumentUndoRedo().DoUndo(bDoesUndo); m_rDoc.getIDocumentState().SetModified(); bChg = true; } break; case SfxStyleFamily::Pseudo: OSL_ENSURE(m_pNumRule, "NumRule missing!"); if (m_pNumRule) { OUString aOldName = m_pNumRule->GetName(); if (!aOldName.isEmpty()) { if ( aOldName != rStr && m_rDoc.RenameNumRule(aOldName, rStr)) { m_pNumRule = m_rDoc.FindNumRulePtr(rStr); m_rDoc.getIDocumentState().SetModified(); bChg = true; } } else { // #i91400# const_cast(m_pNumRule)->SetName( rStr, m_rDoc.getIDocumentListsAccess() ); m_rDoc.getIDocumentState().SetModified(); bChg = true; } } break; default: OSL_ENSURE(false, "unknown style family"); } if( bChg ) { m_pPool->First(nFamily); // internal list has to be updated m_pPool->Broadcast( SfxStyleSheetHint( SfxHintId::StyleSheetModified, *this ) ); if (SwEditShell* pSh = m_rDoc.GetEditShell()) pSh->CallChgLnk(); } return true; } // hierarchy of deduction bool SwDocStyleSheet::SetParent( const OUString& rStr) { SwFormat* pFormat = nullptr, *pParent = nullptr; switch(nFamily) { case SfxStyleFamily::Char : OSL_ENSURE( m_pCharFormat, "SwCharFormat missing!" ); if( nullptr != ( pFormat = m_pCharFormat ) && !rStr.isEmpty() ) pParent = lcl_FindCharFormat(m_rDoc, rStr); break; case SfxStyleFamily::Para : OSL_ENSURE( m_pColl, "Collection missing!"); if( nullptr != ( pFormat = m_pColl ) && !rStr.isEmpty() ) pParent = lcl_FindParaFormat( m_rDoc, rStr ); break; case SfxStyleFamily::Frame: OSL_ENSURE(m_pFrameFormat, "FrameFormat missing!"); if( nullptr != ( pFormat = m_pFrameFormat ) && !rStr.isEmpty() ) pParent = lcl_FindFrameFormat( m_rDoc, rStr ); break; case SfxStyleFamily::Page: case SfxStyleFamily::Pseudo: break; default: OSL_ENSURE(false, "unknown style family"); } bool bRet = false; if( pFormat && pFormat->DerivedFrom() && pFormat->DerivedFrom()->GetName() != rStr ) { { SwImplShellAction aTmp( m_rDoc ); bRet = pFormat->SetDerivedFrom( pParent ); } if( bRet ) { aParent = rStr; m_pPool->Broadcast( SfxStyleSheetHint( SfxHintId::StyleSheetModified, *this ) ); } } return bRet; } // Set Follower bool SwDocStyleSheet::SetFollow( const OUString& rStr) { if( !rStr.isEmpty() && !SfxStyleSheetBase::SetFollow( rStr )) return false; SwImplShellAction aTmpSh( m_rDoc ); switch(nFamily) { case SfxStyleFamily::Para : { OSL_ENSURE(m_pColl, "Collection missing!"); if( m_pColl ) { SwTextFormatColl* pFollow = m_pColl; if( !rStr.isEmpty() && nullptr == (pFollow = lcl_FindParaFormat(m_rDoc, rStr) )) pFollow = m_pColl; m_pColl->SetNextTextFormatColl(*pFollow); } break; } case SfxStyleFamily::Page : { OSL_ENSURE(m_pDesc, "PageDesc missing!"); if( m_pDesc ) { const SwPageDesc* pFollowDesc = !rStr.isEmpty() ? lcl_FindPageDesc(m_rDoc, rStr) : nullptr; size_t nId = 0; if (pFollowDesc != m_pDesc->GetFollow() && m_rDoc.FindPageDesc(m_pDesc->GetName(), &nId)) { SwPageDesc aDesc( *m_pDesc ); aDesc.SetFollow( pFollowDesc ); m_rDoc.ChgPageDesc( nId, aDesc ); m_pDesc = &m_rDoc.GetPageDesc( nId ); } } break; } case SfxStyleFamily::Char: case SfxStyleFamily::Frame: case SfxStyleFamily::Pseudo: break; default: OSL_ENSURE(false, "unknown style family"); } return true; } static void lcl_SwFormatToFlatItemSet(SwFormat const *const pFormat, std::optional& pRet) { // note: we don't add the odd items that GetItemSet() would add // because they don't seem relevant for preview std::vector sets; sets.push_back(&pFormat->GetAttrSet()); while (SfxItemSet const*const pParent = sets.back()->GetParent()) { sets.push_back(pParent); } // start by copying top-level parent set pRet.emplace(*sets.back()); sets.pop_back(); for (auto iter = sets.rbegin(); iter != sets.rend(); ++iter) { // in reverse so child overrides parent pRet->Put(**iter); } } std::optional SwDocStyleSheet::GetItemSetForPreview() { if (SfxStyleFamily::Page == nFamily || SfxStyleFamily::Pseudo == nFamily || SfxStyleFamily::Table == nFamily) { SAL_WARN("sw.ui", "GetItemSetForPreview not implemented for page or number or table style"); return std::optional(); } if (!m_bPhysical) { // because not only this style, but also any number of its parents // (or follow style) may not actually exist in the document at this // time, return one "flattened" item set that contains all items from // all parents. std::optional pRet; bool bModifiedEnabled = m_rDoc.getIDocumentState().IsEnableSetModified(); m_rDoc.getIDocumentState().SetEnableSetModified(false); FillStyleSheet(FillPreview, &pRet); m_rDoc.getIDocumentState().SetEnableSetModified(bModifiedEnabled); assert(pRet); return pRet; } else { std::optional pRet; switch (nFamily) { case SfxStyleFamily::Char: lcl_SwFormatToFlatItemSet(m_pCharFormat, pRet); break; case SfxStyleFamily::Para: lcl_SwFormatToFlatItemSet(m_pColl, pRet); break; case SfxStyleFamily::Frame: lcl_SwFormatToFlatItemSet(m_pFrameFormat, pRet); break; default: std::abort(); } return pRet; } } // extract ItemSet to Name and Family, Mask SfxItemSet& SwDocStyleSheet::GetItemSet() { if(!m_bPhysical) FillStyleSheet( FillPhysical ); switch(nFamily) { case SfxStyleFamily::Char: case SfxStyleFamily::Para: case SfxStyleFamily::Frame: { SvxBoxInfoItem aBoxInfo( SID_ATTR_BORDER_INNER ); aBoxInfo.SetTable( false ); aBoxInfo.SetDist( true ); // always show gap field aBoxInfo.SetMinDist( true );// set minimum size in tables and paragraphs aBoxInfo.SetDefDist( MIN_BORDER_DIST );// always set Default-Gap // Single lines can only have DontCare-Status in tables aBoxInfo.SetValid( SvxBoxInfoItemValidFlags::DISABLE ); if( nFamily == SfxStyleFamily::Char ) { SAL_WARN_IF(!m_pCharFormat, "sw.ui", "Where's SwCharFormat"); m_aCoreSet.Put(m_pCharFormat->GetAttrSet()); m_aCoreSet.Put( aBoxInfo ); if(m_pCharFormat->DerivedFrom()) m_aCoreSet.SetParent(&m_pCharFormat->DerivedFrom()->GetAttrSet()); } else if ( nFamily == SfxStyleFamily::Para ) { OSL_ENSURE(m_pColl, "Where's Collection"); m_aCoreSet.Put(m_pColl->GetAttrSet()); m_aCoreSet.Put( aBoxInfo ); m_aCoreSet.Put(SfxBoolItem(SID_ATTR_AUTO_STYLE_UPDATE, m_pColl->IsAutoUpdateOnDirectFormat())); if(m_pColl->DerivedFrom()) m_aCoreSet.SetParent(&m_pColl->DerivedFrom()->GetAttrSet()); } else { OSL_ENSURE(m_pFrameFormat, "Where's FrameFormat"); m_aCoreSet.Put(m_pFrameFormat->GetAttrSet()); m_aCoreSet.Put( aBoxInfo ); m_aCoreSet.Put(SfxBoolItem(SID_ATTR_AUTO_STYLE_UPDATE, m_pFrameFormat->IsAutoUpdateOnDirectFormat())); if(m_pFrameFormat->DerivedFrom()) m_aCoreSet.SetParent(&m_pFrameFormat->DerivedFrom()->GetAttrSet()); // create needed items for XPropertyList entries from the DrawModel so that // the Area TabPage can access them const SwDrawModel* pDrawModel = m_rDoc.getIDocumentDrawModelAccess().GetDrawModel(); m_aCoreSet.Put(SvxColorListItem(pDrawModel->GetColorList(), SID_COLOR_TABLE)); m_aCoreSet.Put(SvxGradientListItem(pDrawModel->GetGradientList(), SID_GRADIENT_LIST)); m_aCoreSet.Put(SvxHatchListItem(pDrawModel->GetHatchList(), SID_HATCH_LIST)); m_aCoreSet.Put(SvxBitmapListItem(pDrawModel->GetBitmapList(), SID_BITMAP_LIST)); m_aCoreSet.Put(SvxPatternListItem(pDrawModel->GetPatternList(), SID_PATTERN_LIST)); } } break; case SfxStyleFamily::Page : { // set correct parent to get the drawing::FillStyle_NONE FillStyle as needed if(!m_aCoreSet.GetParent()) { m_aCoreSet.SetParent(&m_rDoc.GetDfltFrameFormat()->GetAttrSet()); } OSL_ENSURE(m_pDesc, "No PageDescriptor"); ::PageDescToItemSet(*const_cast(m_pDesc), m_aCoreSet); } break; case SfxStyleFamily::Pseudo: { OSL_ENSURE(m_pNumRule, "No NumRule"); SvxNumRule aRule = m_pNumRule->MakeSvxNumRule(); m_aCoreSet.Put(SvxNumBulletItem(std::move(aRule))); } break; default: OSL_ENSURE(false, "unknown style family"); } // Member of Baseclass pSet = &m_aCoreSet; return m_aCoreSet; } void SwDocStyleSheet::MergeIndentAttrsOfListStyle( SfxItemSet& rSet ) { if ( nFamily != SfxStyleFamily::Para ) { return; } OSL_ENSURE( m_pColl, " - missing paragraph style"); ::sw::ListLevelIndents const indents(m_pColl->AreListLevelIndentsApplicable()); if (indents == ::sw::ListLevelIndents::No) return; OSL_ENSURE( m_pColl->GetItemState( RES_PARATR_NUMRULE ) == SfxItemState::SET, " - list level indents are applicable at paragraph style, but no list style found. Serious defect." ); const OUString sNumRule = m_pColl->GetNumRule().GetValue(); if (sNumRule.isEmpty()) return; const SwNumRule* pRule = m_rDoc.FindNumRulePtr( sNumRule ); if( pRule ) { const SwNumFormat& rFormat = pRule->Get( 0 ); if ( rFormat.GetPositionAndSpaceMode() == SvxNumberFormat::LABEL_ALIGNMENT ) { if (indents & ::sw::ListLevelIndents::FirstLine) { SvxFirstLineIndentItem const firstLine(static_cast(rFormat.GetFirstLineIndent()), RES_MARGIN_FIRSTLINE); rSet.Put(firstLine); } if (indents & ::sw::ListLevelIndents::LeftMargin) { SvxTextLeftMarginItem const leftMargin(rFormat.GetIndentAt(), RES_MARGIN_TEXTLEFT); rSet.Put(leftMargin); } } } } // handling of parameter void SwDocStyleSheet::SetItemSet( const SfxItemSet& rSet, const bool bBroadcast, const bool bResetIndentAttrsAtParagraphStyle ) { // if applicable determine format first if(!m_bPhysical) FillStyleSheet( FillPhysical ); SwImplShellAction aTmpSh( m_rDoc ); OSL_ENSURE( &rSet != &m_aCoreSet, "SetItemSet with own Set is not allowed" ); if (m_rDoc.GetIDocumentUndoRedo().DoesUndo()) { SwRewriter aRewriter; aRewriter.AddRule( UndoArg1, GetName() ); m_rDoc.GetIDocumentUndoRedo().StartUndo( SwUndoId::INSFMTATTR, &aRewriter ); } SwFormat* pFormat = nullptr; std::vector aWhichIdsToReset; std::unique_ptr pNewDsc; size_t nPgDscPos = 0; switch(nFamily) { case SfxStyleFamily::Char : { OSL_ENSURE(m_pCharFormat, "Where's CharFormat"); pFormat = m_pCharFormat; } break; case SfxStyleFamily::Para : { OSL_ENSURE(m_pColl, "Where's Collection"); if(const SfxBoolItem* pAutoUpdate = rSet.GetItemIfSet(SID_ATTR_AUTO_STYLE_UPDATE,false)) { m_pColl->SetAutoUpdateOnDirectFormat(pAutoUpdate->GetValue()); } const SwCondCollItem* pCondItem = rSet.GetItemIfSet( FN_COND_COLL, false ); if( RES_CONDTXTFMTCOLL == m_pColl->Which() && pCondItem ) { const CommandStruct* pCmds = SwCondCollItem::GetCmds(); for(sal_uInt16 i = 0; i < COND_COMMAND_COUNT; i++) { SwCollCondition aCond( nullptr, pCmds[ i ].nCnd, pCmds[ i ].nSubCond ); static_cast(m_pColl)->RemoveCondition( aCond ); const OUString sStyle = pCondItem->GetStyle( i ); if (sStyle.isEmpty()) continue; SwFormat *const pFindFormat = lcl_FindParaFormat( m_rDoc, sStyle ); if (pFindFormat) { aCond.RegisterToFormat( *pFindFormat ); static_cast(m_pColl)->InsertCondition( aCond ); } } m_pColl->GetNotifier().Broadcast(sw::CondCollCondChg(*m_pColl)); } else if( pCondItem && !m_pColl->HasWriterListeners() ) { // no conditional template, then first create and adopt // all important values SwConditionTextFormatColl* pCColl = m_rDoc.MakeCondTextFormatColl( m_pColl->GetName(), static_cast(m_pColl->DerivedFrom()) ); if( m_pColl != &m_pColl->GetNextTextFormatColl() ) pCColl->SetNextTextFormatColl( m_pColl->GetNextTextFormatColl() ); if( m_pColl->IsAssignedToListLevelOfOutlineStyle()) pCColl->AssignToListLevelOfOutlineStyle(m_pColl->GetAssignedOutlineStyleLevel()); else pCColl->DeleteAssignmentToListLevelOfOutlineStyle(); const CommandStruct* pCmds = SwCondCollItem::GetCmds(); for( sal_uInt16 i = 0; i < COND_COMMAND_COUNT; ++i ) { const OUString sStyle = pCondItem->GetStyle( i ); if (sStyle.isEmpty()) continue; SwTextFormatColl *const pFindFormat = lcl_FindParaFormat( m_rDoc, sStyle ); if (pFindFormat) { pCColl->InsertCondition( SwCollCondition( pFindFormat, pCmds[ i ].nCnd, pCmds[ i ].nSubCond ) ); } } m_rDoc.DelTextFormatColl( m_pColl ); m_pColl = pCColl; } if ( bResetIndentAttrsAtParagraphStyle && rSet.GetItemState( RES_PARATR_NUMRULE, false ) == SfxItemState::SET && rSet.GetItemState(RES_MARGIN_FIRSTLINE, false) != SfxItemState::SET && m_pColl->GetItemState(RES_MARGIN_FIRSTLINE, false) == SfxItemState::SET) { aWhichIdsToReset.emplace_back(RES_MARGIN_FIRSTLINE); } if ( bResetIndentAttrsAtParagraphStyle && rSet.GetItemState( RES_PARATR_NUMRULE, false ) == SfxItemState::SET && rSet.GetItemState(RES_MARGIN_TEXTLEFT, false) != SfxItemState::SET && m_pColl->GetItemState(RES_MARGIN_TEXTLEFT, false) == SfxItemState::SET) { aWhichIdsToReset.emplace_back(RES_MARGIN_TEXTLEFT); } // #i56252: If a standard numbering style is assigned to a standard paragraph style // we have to create a physical instance of the numbering style. If we do not and // neither the paragraph style nor the numbering style is used in the document // the numbering style will not be saved with the document and the assignment got lost. if( const SfxPoolItem* pNumRuleItem = rSet.GetItemIfSet( RES_PARATR_NUMRULE, false ) ) { // Setting a numbering rule? const OUString sNumRule = static_cast(pNumRuleItem)->GetValue(); if (!sNumRule.isEmpty()) { SwNumRule* pRule = m_rDoc.FindNumRulePtr( sNumRule ); if( !pRule ) { // Numbering rule not in use yet. sal_uInt16 nPoolId = SwStyleNameMapper::GetPoolIdFromUIName( sNumRule, SwGetPoolIdFromName::NumRule ); if( USHRT_MAX != nPoolId ) // It's a standard numbering rule { m_rDoc.getIDocumentStylePoolAccess().GetNumRuleFromPool( nPoolId ); // Create numbering rule (physical) } } } } pFormat = m_pColl; sal_uInt16 nId = m_pColl->GetPoolFormatId() & ~ ( COLL_GET_RANGE_BITS | POOLGRP_NOCOLLID ); switch( GetMask() & ( static_cast(0x0fff) & ~SfxStyleSearchBits::SwCondColl ) ) { case SfxStyleSearchBits::SwText: nId |= COLL_TEXT_BITS; break; case SfxStyleSearchBits::SwChapter: nId |= COLL_DOC_BITS; break; case SfxStyleSearchBits::SwList: nId |= COLL_LISTS_BITS; break; case SfxStyleSearchBits::SwIndex: nId |= COLL_REGISTER_BITS; break; case SfxStyleSearchBits::SwExtra: nId |= COLL_EXTRA_BITS; break; case SfxStyleSearchBits::SwHtml: nId |= COLL_HTML_BITS; break; default: break; } m_pColl->SetPoolFormatId( nId ); break; } case SfxStyleFamily::Frame: { OSL_ENSURE(m_pFrameFormat, "Where's FrameFormat"); if(const SfxPoolItem* pAutoUpdate = rSet.GetItemIfSet(SID_ATTR_AUTO_STYLE_UPDATE,false)) { m_pFrameFormat->SetAutoUpdateOnDirectFormat(static_cast(pAutoUpdate)->GetValue()); } pFormat = m_pFrameFormat; } break; case SfxStyleFamily::Page : { OSL_ENSURE(m_pDesc, "Where's PageDescriptor"); if (m_rDoc.FindPageDesc(m_pDesc->GetName(), &nPgDscPos)) { pNewDsc.reset( new SwPageDesc( *m_pDesc ) ); // #i48949# - no undo actions for the // copy of the page style ::sw::UndoGuard const ug(m_rDoc.GetIDocumentUndoRedo()); m_rDoc.CopyPageDesc(*m_pDesc, *pNewDsc); // #i7983# pFormat = &pNewDsc->GetMaster(); } } break; case SfxStyleFamily::Pseudo: { OSL_ENSURE(m_pNumRule, "Where's NumRule"); if (!m_pNumRule) break; const SfxPoolItem* pItem; switch( rSet.GetItemState( SID_ATTR_NUMBERING_RULE, false, &pItem )) { case SfxItemState::SET: { SvxNumRule& rSetRule = const_cast(static_cast(pItem)->GetNumRule()); rSetRule.UnLinkGraphics(); SwNumRule aSetRule(*m_pNumRule); aSetRule.SetSvxRule(rSetRule, &m_rDoc); m_rDoc.ChgNumRuleFormats( aSetRule ); } break; case SfxItemState::INVALID: // set NumRule to default values // what are the default values? { SwNumRule aRule( m_pNumRule->GetName(), // #i89178# numfunc::GetDefaultPositionAndSpaceMode() ); m_rDoc.ChgNumRuleFormats( aRule ); } break; default: break; } } break; default: OSL_ENSURE(false, "unknown style family"); } if( pFormat && rSet.Count()) { SfxItemIter aIter( rSet ); const SfxPoolItem* pItem = aIter.GetCurItem(); do { if( IsInvalidItem( pItem ) ) // Clear { // use method in order to // create an Undo object for the attribute reset. aWhichIdsToReset.emplace_back(rSet.GetWhichByOffset(aIter.GetCurPos())); } pItem = aIter.NextItem(); } while (pItem); m_rDoc.ResetAttrAtFormat(aWhichIdsToReset, *pFormat); SfxItemSet aSet(rSet); aSet.ClearInvalidItems(); if(SfxStyleFamily::Frame == nFamily) { // Need to check for unique item for DrawingLayer items of type NameOrIndex // and evtl. correct that item to ensure unique names for that type. This call may // modify/correct entries inside of the given SfxItemSet m_rDoc.CheckForUniqueItemForLineFillNameOrIndex(aSet); } m_aCoreSet.ClearItem(); if( pNewDsc ) { ::ItemSetToPageDesc( aSet, *pNewDsc ); m_rDoc.ChgPageDesc( nPgDscPos, *pNewDsc ); m_pDesc = &m_rDoc.GetPageDesc( nPgDscPos ); m_rDoc.PreDelPageDesc(pNewDsc.get()); // #i7983# pNewDsc.reset(); } else { m_rDoc.ChgFormat(*pFormat, aSet); // put all that is set if (bBroadcast) m_pPool->Broadcast(SfxStyleSheetHint(SfxHintId::StyleSheetModified, *this)); } } else { m_aCoreSet.ClearItem(); if( pNewDsc ) // we still need to delete it { m_rDoc.PreDelPageDesc(pNewDsc.get()); // #i7983# pNewDsc.reset(); } } if (m_rDoc.GetIDocumentUndoRedo().DoesUndo()) { m_rDoc.GetIDocumentUndoRedo().EndUndo(SwUndoId::END, nullptr); } } static void lcl_SaveStyles( SfxStyleFamily nFamily, std::vector& rArr, SwDoc& rDoc ) { switch( nFamily ) { case SfxStyleFamily::Char: { const SwCharFormats& rTable = *rDoc.GetCharFormats(); for(auto const& rChar: rTable) { rArr.push_back(rChar); } } break; case SfxStyleFamily::Para: { const SwTextFormatColls& rTable = *rDoc.GetTextFormatColls(); for(auto const& rPara : rTable) { rArr.push_back(rPara); } } break; case SfxStyleFamily::Frame: { for(auto const& rFrame: *rDoc.GetFrameFormats()) { rArr.push_back(rFrame); } } break; case SfxStyleFamily::Page: { for( size_t n = 0, nCnt = rDoc.GetPageDescCnt(); n < nCnt; ++n ) { rArr.push_back( &rDoc.GetPageDesc( n ) ); } } break; case SfxStyleFamily::Pseudo: { const SwNumRuleTable& rTable = rDoc.GetNumRuleTable(); for(auto const& rPseudo: rTable) { rArr.push_back(rPseudo); } } break; default: break; } } static bool lcl_Contains(const std::vector& rArr, const void* p) { return std::find( rArr.begin(), rArr.end(), p ) != rArr.end(); } static void lcl_DeleteInfoStyles( SfxStyleFamily nFamily, std::vector const & rArr, SwDoc& rDoc ) { size_t n, nCnt; switch( nFamily ) { case SfxStyleFamily::Char: { std::deque aDelArr; const SwCharFormats& rTable = *rDoc.GetCharFormats(); for( n = 0, nCnt = rTable.size(); n < nCnt; ++n ) { if( !lcl_Contains( rArr, rTable[ n ] )) aDelArr.push_front( n ); } for(auto const& rDelArr: aDelArr) rDoc.DelCharFormat( rDelArr ); } break; case SfxStyleFamily::Para : { std::deque aDelArr; const SwTextFormatColls& rTable = *rDoc.GetTextFormatColls(); for( n = 0, nCnt = rTable.size(); n < nCnt; ++n ) { if( !lcl_Contains( rArr, rTable[ n ] )) aDelArr.push_front( n ); } for(auto const& rDelArr: aDelArr) rDoc.DelTextFormatColl( rDelArr ); } break; case SfxStyleFamily::Frame: { std::deque aDelArr; for(auto const& rFrame: *rDoc.GetFrameFormats()) { if( !lcl_Contains( rArr, rFrame )) aDelArr.push_front( rFrame ); } for( auto const& rDelArr: aDelArr) rDoc.DelFrameFormat( rDelArr ); } break; case SfxStyleFamily::Page: { std::deque aDelArr; for( n = 0, nCnt = rDoc.GetPageDescCnt(); n < nCnt; ++n ) { if( !lcl_Contains( rArr, &rDoc.GetPageDesc( n ) )) aDelArr.push_front( n ); } for( auto const& rDelArr: aDelArr ) rDoc.DelPageDesc( rDelArr); } break; case SfxStyleFamily::Pseudo: { std::deque aDelArr; const SwNumRuleTable& rTable = rDoc.GetNumRuleTable(); for( auto const& rPseudo: rTable) { if( !lcl_Contains( rArr, rPseudo )) aDelArr.push_front( rPseudo ); } for( auto const& rDelArr: aDelArr) rDoc.DelNumRule( rDelArr->GetName() ); } break; default: break; } } // determine the format bool SwDocStyleSheet::FillStyleSheet( FillStyleType const eFType, std::optional *const o_ppFlatSet) { bool bRet = false; sal_uInt16 nPoolId = USHRT_MAX; SwFormat* pFormat = nullptr; bool bCreate = FillPhysical == eFType; bool bDeleteInfo = false; bool bFillOnlyInfo = FillAllInfo == eFType || FillPreview == eFType; std::vector aDelArr; bool const isModified(m_rDoc.getIDocumentState().IsModified()); switch(nFamily) { case SfxStyleFamily::Char: m_pCharFormat = lcl_FindCharFormat(m_rDoc, aName, this, bCreate ); m_bPhysical = nullptr != m_pCharFormat; if( bFillOnlyInfo && !m_bPhysical ) { // create style (plus all needed parents) and clean it up // later - without affecting the undo/redo stack ::sw::UndoGuard const ug(m_rDoc.GetIDocumentUndoRedo()); bDeleteInfo = true; ::lcl_SaveStyles( nFamily, aDelArr, m_rDoc ); m_pCharFormat = lcl_FindCharFormat(m_rDoc, aName, this ); } pFormat = m_pCharFormat; m_aLink.clear(); if( !bCreate && !pFormat ) { if( aName == SwResId(STR_POOLCHR_STANDARD)) nPoolId = 0; else nPoolId = SwStyleNameMapper::GetPoolIdFromUIName( aName, SwGetPoolIdFromName::ChrFmt ); } if (m_pCharFormat) { const SwTextFormatColl* pParaFormat = m_pCharFormat->GetLinkedParaFormat(); if (pParaFormat) { m_aLink = pParaFormat->GetName(); } } bRet = nullptr != m_pCharFormat || USHRT_MAX != nPoolId; if( bDeleteInfo ) m_pCharFormat = nullptr; break; case SfxStyleFamily::Para: { m_pColl = lcl_FindParaFormat(m_rDoc, aName, this, bCreate); m_bPhysical = nullptr != m_pColl; if( bFillOnlyInfo && !m_bPhysical ) { ::sw::UndoGuard const ug(m_rDoc.GetIDocumentUndoRedo()); bDeleteInfo = true; ::lcl_SaveStyles( nFamily, aDelArr, m_rDoc ); m_pColl = lcl_FindParaFormat(m_rDoc, aName, this ); } pFormat = m_pColl; m_aLink.clear(); if( m_pColl ) { PresetFollow( m_pColl->GetNextTextFormatColl().GetName() ); const SwCharFormat* pCharFormat = m_pColl->GetLinkedCharFormat(); if (pCharFormat) { m_aLink = pCharFormat->GetName(); } } else if( !bCreate ) nPoolId = SwStyleNameMapper::GetPoolIdFromUIName( aName, SwGetPoolIdFromName::TxtColl ); bRet = nullptr != m_pColl || USHRT_MAX != nPoolId; if( bDeleteInfo ) m_pColl = nullptr; } break; case SfxStyleFamily::Frame: m_pFrameFormat = lcl_FindFrameFormat(m_rDoc, aName, this, bCreate); m_bPhysical = nullptr != m_pFrameFormat; if (bFillOnlyInfo && !m_bPhysical) { ::sw::UndoGuard const ug(m_rDoc.GetIDocumentUndoRedo()); bDeleteInfo = true; ::lcl_SaveStyles( nFamily, aDelArr, m_rDoc ); m_pFrameFormat = lcl_FindFrameFormat(m_rDoc, aName, this ); } pFormat = m_pFrameFormat; if( !bCreate && !pFormat ) nPoolId = SwStyleNameMapper::GetPoolIdFromUIName( aName, SwGetPoolIdFromName::FrmFmt ); bRet = nullptr != m_pFrameFormat || USHRT_MAX != nPoolId; if( bDeleteInfo ) m_pFrameFormat = nullptr; break; case SfxStyleFamily::Page: m_pDesc = lcl_FindPageDesc(m_rDoc, aName, this, bCreate); m_bPhysical = nullptr != m_pDesc; if( bFillOnlyInfo && !m_pDesc ) { ::sw::UndoGuard const ug(m_rDoc.GetIDocumentUndoRedo()); bDeleteInfo = true; ::lcl_SaveStyles( nFamily, aDelArr, m_rDoc ); m_pDesc = lcl_FindPageDesc( m_rDoc, aName, this ); } if( m_pDesc ) { nPoolId = m_pDesc->GetPoolFormatId(); nHelpId = m_pDesc->GetPoolHelpId(); if (const OUString* pattern = m_pDesc->GetPoolHlpFileId() != UCHAR_MAX ? m_rDoc.GetDocPattern(m_pDesc->GetPoolHlpFileId()) : nullptr) aHelpFile = *pattern; else aHelpFile.clear(); } else if( !bCreate ) nPoolId = SwStyleNameMapper::GetPoolIdFromUIName( aName, SwGetPoolIdFromName::PageDesc ); SetMask( (USER_FMT & nPoolId) ? SfxStyleSearchBits::UserDefined : SfxStyleSearchBits::Auto ); bRet = nullptr != m_pDesc || USHRT_MAX != nPoolId; if( bDeleteInfo ) m_pDesc = nullptr; break; case SfxStyleFamily::Pseudo: m_pNumRule = lcl_FindNumRule(m_rDoc, aName, this, bCreate); m_bPhysical = nullptr != m_pNumRule; if( bFillOnlyInfo && !m_pNumRule ) { ::sw::UndoGuard const ug(m_rDoc.GetIDocumentUndoRedo()); bDeleteInfo = true; ::lcl_SaveStyles( nFamily, aDelArr, m_rDoc ); m_pNumRule = lcl_FindNumRule( m_rDoc, aName, this ); } if( m_pNumRule ) { nPoolId = m_pNumRule->GetPoolFormatId(); nHelpId = m_pNumRule->GetPoolHelpId(); if (const OUString* pattern = m_pNumRule->GetPoolHlpFileId() != UCHAR_MAX ? m_rDoc.GetDocPattern(m_pNumRule->GetPoolHlpFileId()) : nullptr) aHelpFile = *pattern; else aHelpFile.clear(); } else if( !bCreate ) nPoolId = SwStyleNameMapper::GetPoolIdFromUIName( aName, SwGetPoolIdFromName::NumRule ); SetMask( (USER_FMT & nPoolId) ? SfxStyleSearchBits::UserDefined : SfxStyleSearchBits::Auto ); bRet = nullptr != m_pNumRule || USHRT_MAX != nPoolId; if( bDeleteInfo ) m_pNumRule = nullptr; break; case SfxStyleFamily::Table: m_pTableFormat = lcl_FindTableStyle(m_rDoc, aName, this, bCreate); SetMask((m_pTableFormat && m_pTableFormat->IsUserDefined()) ? SfxStyleSearchBits::UserDefined : SfxStyleSearchBits::Auto); bRet = m_bPhysical = (nullptr != m_pTableFormat); break; case SfxStyleFamily::Cell: m_pBoxFormat = lcl_FindCellStyle(m_rDoc, aName, this); bRet = m_bPhysical = (nullptr != m_pBoxFormat); break; default:; //prevent warning } if( SfxStyleFamily::Char == nFamily || SfxStyleFamily::Para == nFamily || SfxStyleFamily::Frame == nFamily ) { if( pFormat ) nPoolId = pFormat->GetPoolFormatId(); SfxStyleSearchBits _nMask = SfxStyleSearchBits::Auto; if( pFormat == m_rDoc.GetDfltCharFormat() ) _nMask |= SfxStyleSearchBits::ReadOnly; else if( USER_FMT & nPoolId ) _nMask |= SfxStyleSearchBits::UserDefined; switch ( COLL_GET_RANGE_BITS & nPoolId ) { case COLL_TEXT_BITS: _nMask |= SfxStyleSearchBits::SwText; break; case COLL_DOC_BITS : _nMask |= SfxStyleSearchBits::SwChapter; break; case COLL_LISTS_BITS: _nMask |= SfxStyleSearchBits::SwList; break; case COLL_REGISTER_BITS: _nMask |= SfxStyleSearchBits::SwIndex; break; case COLL_EXTRA_BITS: _nMask |= SfxStyleSearchBits::SwExtra; break; case COLL_HTML_BITS: _nMask |= SfxStyleSearchBits::SwHtml; break; } if( pFormat ) { OSL_ENSURE( m_bPhysical, "Format not found" ); nHelpId = pFormat->GetPoolHelpId(); if (const OUString* pattern = pFormat->GetPoolHlpFileId() != UCHAR_MAX ? m_rDoc.GetDocPattern(pFormat->GetPoolHlpFileId()) : nullptr) aHelpFile = *pattern; else aHelpFile.clear(); if( RES_CONDTXTFMTCOLL == pFormat->Which() ) _nMask |= SfxStyleSearchBits::SwCondColl; if (FillPreview == eFType) { assert(o_ppFlatSet); lcl_SwFormatToFlatItemSet(pFormat, *o_ppFlatSet); } } SetMask( _nMask ); } if( bDeleteInfo && bFillOnlyInfo ) { ::sw::UndoGuard const ug(m_rDoc.GetIDocumentUndoRedo()); ::lcl_DeleteInfoStyles( nFamily, aDelArr, m_rDoc ); if (!isModified) { m_rDoc.getIDocumentState().ResetModified(); } } return bRet; } // Create new format in Core void SwDocStyleSheet::Create() { switch(nFamily) { case SfxStyleFamily::Char : m_pCharFormat = lcl_FindCharFormat( m_rDoc, aName ); if( !m_pCharFormat ) m_pCharFormat = m_rDoc.MakeCharFormat(aName, m_rDoc.GetDfltCharFormat()); m_pCharFormat->SetAuto(false); break; case SfxStyleFamily::Para : m_pColl = lcl_FindParaFormat( m_rDoc, aName ); if( !m_pColl ) { SwTextFormatColl *pPar = (*m_rDoc.GetTextFormatColls())[0]; if( nMask & SfxStyleSearchBits::SwCondColl ) m_pColl = m_rDoc.MakeCondTextFormatColl( aName, pPar ); else m_pColl = m_rDoc.MakeTextFormatColl( aName, pPar ); } break; case SfxStyleFamily::Frame: m_pFrameFormat = lcl_FindFrameFormat( m_rDoc, aName ); if( !m_pFrameFormat ) m_pFrameFormat = m_rDoc.MakeFrameFormat(aName, m_rDoc.GetDfltFrameFormat(), false, false); break; case SfxStyleFamily::Page : m_pDesc = lcl_FindPageDesc( m_rDoc, aName ); if( !m_pDesc ) { m_pDesc = m_rDoc.MakePageDesc(aName); } break; case SfxStyleFamily::Pseudo: m_pNumRule = lcl_FindNumRule( m_rDoc, aName ); if( !m_pNumRule ) { const OUString sTmpNm( aName.isEmpty() ? m_rDoc.GetUniqueNumRuleName() : aName ); SwNumRule* pRule = m_rDoc.GetNumRuleTable()[ m_rDoc.MakeNumRule( sTmpNm, nullptr, false, // #i89178# numfunc::GetDefaultPositionAndSpaceMode() ) ]; pRule->SetAutoRule( false ); if( aName.isEmpty() ) { // #i91400# pRule->SetName( aName, m_rDoc.getIDocumentListsAccess() ); } m_pNumRule = pRule; } break; case SfxStyleFamily::Table: if (aName.isEmpty()) return; m_pTableFormat = lcl_FindTableStyle(m_rDoc, aName); if (!m_pTableFormat) { m_rDoc.MakeTableStyle(aName); m_pTableFormat = m_rDoc.GetTableStyles().FindAutoFormat(aName); SAL_WARN_IF(!m_pTableFormat, "sw.ui", "Recently added auto format not found"); } break; default:; //prevent warning } m_bPhysical = true; m_aCoreSet.ClearItem(); } SwCharFormat* SwDocStyleSheet::GetCharFormat() { if(!m_bPhysical) FillStyleSheet( FillPhysical ); return m_pCharFormat; } SwTextFormatColl* SwDocStyleSheet::GetCollection() { if(!m_bPhysical) FillStyleSheet( FillPhysical ); return m_pColl; } const SwPageDesc* SwDocStyleSheet::GetPageDesc() { if(!m_bPhysical) FillStyleSheet( FillPhysical ); return m_pDesc; } const SwNumRule * SwDocStyleSheet::GetNumRule() { if(!m_bPhysical) FillStyleSheet( FillPhysical ); return m_pNumRule; } void SwDocStyleSheet::SetNumRule(const SwNumRule& rRule) { OSL_ENSURE(m_pNumRule, "Where is the NumRule"); m_rDoc.ChgNumRuleFormats( rRule ); } SwTableAutoFormat* SwDocStyleSheet::GetTableFormat() { if(!m_bPhysical) FillStyleSheet( FillPhysical ); assert(m_pTableFormat && "SwDocStyleSheet table style, SwTableAutoFormat not found"); return m_pTableFormat; } // re-generate Name AND Family from String // First() and Next() (see below) insert an identification letter at Pos.1 void SwDocStyleSheet::PresetNameAndFamily(SfxStyleFamily eFamily, const OUString& rName) { this->nFamily = eFamily; this->aName = rName; } // Is the format physically present yet void SwDocStyleSheet::SetPhysical(bool bPhys) { m_bPhysical = bPhys; if(!bPhys) { m_pCharFormat = nullptr; m_pColl = nullptr; m_pFrameFormat = nullptr; m_pDesc = nullptr; } } SwFrameFormat* SwDocStyleSheet::GetFrameFormat() { if(!m_bPhysical) FillStyleSheet( FillPhysical ); return m_pFrameFormat; } bool SwDocStyleSheet::IsUsed() const { if( !m_bPhysical ) { SwDocStyleSheet* pThis = const_cast(this); pThis->FillStyleSheet( FillOnlyName ); } if( !m_bPhysical ) return false; const sw::BroadcastingModify* pMod; switch( nFamily ) { case SfxStyleFamily::Char : pMod = m_pCharFormat; break; case SfxStyleFamily::Para : pMod = m_pColl; break; case SfxStyleFamily::Frame: pMod = m_pFrameFormat; break; case SfxStyleFamily::Page : pMod = m_pDesc; break; case SfxStyleFamily::Pseudo: return m_pNumRule && m_rDoc.IsUsed(*m_pNumRule); case SfxStyleFamily::Table: return m_pTableFormat && m_rDoc.IsUsed( *m_pTableFormat ); default: OSL_ENSURE(false, "unknown style family"); return false; } if (m_rDoc.IsUsed(*pMod)) return true; SfxStyleSheetIterator aIter(static_cast(m_pPool)->GetEEStyleSheetPool(), nFamily, SfxStyleSearchBits::Used); return aIter.Find(GetName()) != nullptr; } OUString SwDocStyleSheet::GetUsedBy() { return m_pNumRule ? m_pNumRule->MakeParagraphStyleListString() : OUString(); } sal_uInt32 SwDocStyleSheet::GetHelpId( OUString& rFile ) { sal_uInt16 nId = 0; sal_uInt16 nPoolId = 0; unsigned char nFileId = UCHAR_MAX; rFile = "swrhlppi.hlp"; const SwFormat* pTmpFormat = nullptr; switch( nFamily ) { case SfxStyleFamily::Char : if( !m_pCharFormat && nullptr == (m_pCharFormat = lcl_FindCharFormat( m_rDoc, aName, nullptr, false )) ) { nId = SwStyleNameMapper::GetPoolIdFromUIName( aName, SwGetPoolIdFromName::ChrFmt ); return USHRT_MAX == nId ? 0 : nId; } pTmpFormat = m_pCharFormat; break; case SfxStyleFamily::Para: if( !m_pColl && nullptr == ( m_pColl = lcl_FindParaFormat( m_rDoc, aName, nullptr, false )) ) { nId = SwStyleNameMapper::GetPoolIdFromUIName( aName, SwGetPoolIdFromName::TxtColl ); return USHRT_MAX == nId ? 0 : nId; } pTmpFormat = m_pColl; break; case SfxStyleFamily::Frame: if( !m_pFrameFormat && nullptr == ( m_pFrameFormat = lcl_FindFrameFormat( m_rDoc, aName, nullptr, false ) ) ) { nId = SwStyleNameMapper::GetPoolIdFromUIName( aName, SwGetPoolIdFromName::FrmFmt ); return USHRT_MAX == nId ? 0 : nId; } pTmpFormat = m_pFrameFormat; break; case SfxStyleFamily::Page: if( !m_pDesc && nullptr == ( m_pDesc = lcl_FindPageDesc( m_rDoc, aName, nullptr, false ) ) ) { nId = SwStyleNameMapper::GetPoolIdFromUIName( aName, SwGetPoolIdFromName::PageDesc ); return USHRT_MAX == nId ? 0 : nId; } nId = m_pDesc->GetPoolHelpId(); nFileId = m_pDesc->GetPoolHlpFileId(); nPoolId = m_pDesc->GetPoolFormatId(); break; case SfxStyleFamily::Pseudo: if( !m_pNumRule && nullptr == ( m_pNumRule = lcl_FindNumRule( m_rDoc, aName, nullptr, false ) ) ) { nId = SwStyleNameMapper::GetPoolIdFromUIName( aName, SwGetPoolIdFromName::NumRule ); return USHRT_MAX == nId ? 0 : nId; } nId = m_pNumRule->GetPoolHelpId(); nFileId = m_pNumRule->GetPoolHlpFileId(); nPoolId = m_pNumRule->GetPoolFormatId(); break; default: OSL_ENSURE(false, "unknown style family"); return 0; } if( pTmpFormat ) { nId = pTmpFormat->GetPoolHelpId(); nFileId = pTmpFormat->GetPoolHlpFileId(); nPoolId = pTmpFormat->GetPoolFormatId(); } if( UCHAR_MAX != nFileId ) { const OUString *pTemplate = m_rDoc.GetDocPattern( nFileId ); if( pTemplate ) { rFile = *pTemplate; } } else if( !IsPoolUserFormat( nPoolId ) ) { nId = nPoolId; } // because SFX acts like that, with HelpId: if( USHRT_MAX == nId ) nId = 0; // don't show Help accordingly return nId; } void SwDocStyleSheet::SetHelpId( const OUString& r, sal_uInt32 nId ) { sal_uInt8 nFileId = static_cast< sal_uInt8 >(m_rDoc.SetDocPattern( r )); SwFormat* pTmpFormat = nullptr; switch( nFamily ) { case SfxStyleFamily::Char : pTmpFormat = m_pCharFormat; break; case SfxStyleFamily::Para : pTmpFormat = m_pColl; break; case SfxStyleFamily::Frame: pTmpFormat = m_pFrameFormat; break; case SfxStyleFamily::Page : const_cast(m_pDesc)->SetPoolHelpId( nId ); const_cast(m_pDesc)->SetPoolHlpFileId( nFileId ); break; case SfxStyleFamily::Pseudo: const_cast(m_pNumRule)->SetPoolHelpId( nId ); const_cast(m_pNumRule)->SetPoolHlpFileId( nFileId ); break; default: OSL_ENSURE(false, "unknown style family"); return ; } if( pTmpFormat ) { pTmpFormat->SetPoolHelpId( nId ); pTmpFormat->SetPoolHlpFileId( nFileId ); } } // methods for DocStyleSheetPool SwDocStyleSheetPool::SwDocStyleSheetPool( SwDoc& rDocument, bool bOrg ) : SfxStyleSheetBasePool(rDocument.GetAttrPool()) , mxStyleSheet(new SwDocStyleSheet(rDocument, *this)) , mxEEStyleSheetPool(new EEStyleSheetPool(this)) , m_rDoc(rDocument) { m_bOrganizer = bOrg; } SwDocStyleSheetPool::~SwDocStyleSheetPool() { } SfxStyleSheetBase& SwDocStyleSheetPool::Make( const OUString& rName, SfxStyleFamily eFam, SfxStyleSearchBits _nMask) { mxStyleSheet->PresetName(rName); mxStyleSheet->PresetParent(OUString()); mxStyleSheet->PresetFollow(OUString()); mxStyleSheet->SetMask(_nMask) ; mxStyleSheet->SetFamily(eFam); mxStyleSheet->SetPhysical(true); mxStyleSheet->Create(); return *mxStyleSheet; } rtl::Reference SwDocStyleSheetPool::Create( const SfxStyleSheetBase& /*rOrg*/) { OSL_ENSURE(false , "Create in SW-Stylesheet-Pool not possible" ); return nullptr; } rtl::Reference SwDocStyleSheetPool::Create( const OUString &, SfxStyleFamily, SfxStyleSearchBits ) { OSL_ENSURE( false, "Create in SW-Stylesheet-Pool not possible" ); return nullptr; } std::unique_ptr SwDocStyleSheetPool::CreateIterator( SfxStyleFamily eFam, SfxStyleSearchBits _nMask ) { return std::make_unique(*this, eFam, _nMask); } void SwDocStyleSheetPool::dispose() { mxStyleSheet.clear(); mxEEStyleSheetPool.clear(); } void SwDocStyleSheetPool::Remove( SfxStyleSheetBase* pStyle) { if( !pStyle ) return; bool bBroadcast = true; SwImplShellAction aTmpSh( m_rDoc ); const OUString sName = pStyle->GetName(); switch( pStyle->GetFamily() ) { case SfxStyleFamily::Char: { SwCharFormat* pFormat = lcl_FindCharFormat(m_rDoc, sName, nullptr, false ); if(pFormat) m_rDoc.DelCharFormat(pFormat); } break; case SfxStyleFamily::Para: { SwTextFormatColl* pColl = lcl_FindParaFormat(m_rDoc, sName, nullptr, false ); if(pColl) m_rDoc.DelTextFormatColl(pColl); } break; case SfxStyleFamily::Frame: { SwFrameFormat* pFormat = lcl_FindFrameFormat(m_rDoc, sName, nullptr, false ); if(pFormat) m_rDoc.DelFrameFormat(pFormat); } break; case SfxStyleFamily::Page : { m_rDoc.DelPageDesc(sName); } break; case SfxStyleFamily::Pseudo: { if( !m_rDoc.DelNumRule( sName ) ) // Only send Broadcast, when something was deleted bBroadcast = false; } break; case SfxStyleFamily::Table: { m_rDoc.DelTableStyle(sName); } break; default: OSL_ENSURE(false, "unknown style family"); bBroadcast = false; } if( bBroadcast ) Broadcast( SfxStyleSheetHint( SfxHintId::StyleSheetErased, *pStyle ) ); } SfxStyleSheetBase* SwDocStyleSheetPool::Find(const OUString& rName, SfxStyleFamily eFam, SfxStyleSearchBits n) { SfxStyleSearchBits nSMask = n; if( SfxStyleFamily::Para == eFam && m_rDoc.getIDocumentSettingAccess().get(DocumentSettingId::HTML_MODE) ) { // then only HTML-Templates are of interest if( SfxStyleSearchBits::All == nSMask ) nSMask = SfxStyleSearchBits::SwHtml | SfxStyleSearchBits::UserDefined | SfxStyleSearchBits::Used; else nSMask &= SfxStyleSearchBits::Used | SfxStyleSearchBits::UserDefined | SfxStyleSearchBits::SwCondColl | SfxStyleSearchBits::SwHtml; if( nSMask == SfxStyleSearchBits::Auto ) nSMask = SfxStyleSearchBits::SwHtml; } const bool bSearchUsed = ( n != SfxStyleSearchBits::All && n & SfxStyleSearchBits::Used ); const sw::BroadcastingModify* pMod = nullptr; mxStyleSheet->SetPhysical( false ); mxStyleSheet->PresetName( rName ); mxStyleSheet->SetFamily( eFam ); bool bFnd = mxStyleSheet->FillStyleSheet( SwDocStyleSheet::FillOnlyName ); if( mxStyleSheet->IsPhysical() ) { switch( eFam ) { case SfxStyleFamily::Char: pMod = mxStyleSheet->GetCharFormat(); break; case SfxStyleFamily::Para: pMod = mxStyleSheet->GetCollection(); break; case SfxStyleFamily::Frame: pMod = mxStyleSheet->GetFrameFormat(); break; case SfxStyleFamily::Page: pMod = mxStyleSheet->GetPageDesc(); break; case SfxStyleFamily::Pseudo: { const SwNumRule* pRule = mxStyleSheet->GetNumRule(); if( pRule && !bSearchUsed && (( nSMask & ~SfxStyleSearchBits::Used) == SfxStyleSearchBits::UserDefined ? !(pRule->GetPoolFormatId() & USER_FMT) // searched for used and found none : bSearchUsed )) bFnd = false; } break; case SfxStyleFamily::Table: case SfxStyleFamily::Cell: break; default: OSL_ENSURE(false, "unknown style family"); } } // then evaluate the mask: if( pMod && !bSearchUsed ) { const sal_uInt16 nId = SfxStyleFamily::Page == eFam ? static_cast(pMod)->GetPoolFormatId() : static_cast(pMod)->GetPoolFormatId(); if( ( nSMask & ~SfxStyleSearchBits::Used) == SfxStyleSearchBits::UserDefined && !(nId & USER_FMT) ) // searched for used and found none bFnd = false; } return bFnd ? mxStyleSheet.get() : nullptr; } SwStyleSheetIterator::SwStyleSheetIterator(SwDocStyleSheetPool& rBase, SfxStyleFamily eFam, SfxStyleSearchBits n ) : SfxStyleSheetIterator(&rBase, eFam, n) , mxIterSheet(new SwDocStyleSheet(rBase.GetDoc(), rBase)) , mxStyleSheet(new SwDocStyleSheet(rBase.GetDoc(), rBase)) { m_bFirstCalled = false; m_nLastPos = 0; StartListening(rBase); } SwStyleSheetIterator::~SwStyleSheetIterator() { EndListening( *mxIterSheet->GetPool() ); } sal_Int32 SwStyleSheetIterator::Count() { // let the list fill correctly!! if( !m_bFirstCalled ) First(); return m_aLst.size(); } SfxStyleSheetBase* SwStyleSheetIterator::operator[]( sal_Int32 nIdx ) { // found if( !m_bFirstCalled ) First(); auto const & rEntry = m_aLst[ nIdx ]; mxStyleSheet->PresetNameAndFamily( rEntry.first, rEntry.second ); mxStyleSheet->SetPhysical( false ); mxStyleSheet->FillStyleSheet( SwDocStyleSheet::FillOnlyName ); return mxStyleSheet.get(); } SfxStyleSheetBase* SwStyleSheetIterator::First() { // Delete old list m_bFirstCalled = true; m_nLastPos = 0; m_aLst.clear(); // Delete current mxIterSheet->Reset(); const SwDoc& rDoc = static_cast(pBasePool)->GetDoc(); const SfxStyleSearchBits nSrchMask = nMask; const bool bIsSearchUsed = SearchUsed(); bool bSearchHidden( nMask & SfxStyleSearchBits::Hidden ); bool bOnlyHidden = nMask == SfxStyleSearchBits::Hidden; const bool bOrganizer = static_cast(pBasePool)->IsOrganizerMode(); bool bAll = ( nSrchMask & SfxStyleSearchBits::AllVisible ) == SfxStyleSearchBits::AllVisible; if( nSearchFamily == SfxStyleFamily::Char || nSearchFamily == SfxStyleFamily::All ) { const size_t nArrLen = rDoc.GetCharFormats()->size(); for( size_t i = 0; i < nArrLen; i++ ) { SwCharFormat* pFormat = (*rDoc.GetCharFormats())[ i ]; const bool bUsed = bIsSearchUsed && (bOrganizer || rDoc.IsUsed(*pFormat)); if( ( !bSearchHidden && pFormat->IsHidden() && !bUsed ) || ( pFormat->IsDefault() && pFormat != rDoc.GetDfltCharFormat() ) ) continue; if ( nSrchMask == SfxStyleSearchBits::Hidden && !pFormat->IsHidden( ) ) continue; if( !bUsed ) { // Standard is no User template const sal_uInt16 nId = rDoc.GetDfltCharFormat() == pFormat ? sal_uInt16( RES_POOLCHR_INET_NORMAL ): pFormat->GetPoolFormatId(); if( (nSrchMask & ~SfxStyleSearchBits::Used) == SfxStyleSearchBits::UserDefined ? !(nId & USER_FMT) // searched for used and found none : bIsSearchUsed ) { continue; } if( rDoc.getIDocumentSettingAccess().get(DocumentSettingId::HTML_MODE) && !(nId & USER_FMT) && ( RES_POOLCHR_HTML_BEGIN > nId || nId >= RES_POOLCHR_HTML_END ) && RES_POOLCHR_INET_NORMAL != nId && RES_POOLCHR_INET_VISIT != nId && RES_POOLCHR_FOOTNOTE != nId && RES_POOLCHR_ENDNOTE != nId ) continue; } m_aLst.Append( SfxStyleFamily::Char, pFormat == rDoc.GetDfltCharFormat() ? SwResId(STR_POOLCHR_STANDARD) : pFormat->GetName() ); } // PoolFormat if( bAll ) { if( ! rDoc.getIDocumentSettingAccess().get(DocumentSettingId::HTML_MODE) ) AppendStyleList(SwStyleNameMapper::GetChrFormatUINameArray(), bIsSearchUsed, bSearchHidden, bOnlyHidden, SwGetPoolIdFromName::ChrFmt, SfxStyleFamily::Char); else { m_aLst.Append( SfxStyleFamily::Char, SwStyleNameMapper::GetChrFormatUINameArray()[ RES_POOLCHR_INET_NORMAL - RES_POOLCHR_BEGIN ] ); m_aLst.Append( SfxStyleFamily::Char, SwStyleNameMapper::GetChrFormatUINameArray()[ RES_POOLCHR_INET_VISIT - RES_POOLCHR_BEGIN ] ); m_aLst.Append( SfxStyleFamily::Char, SwStyleNameMapper::GetChrFormatUINameArray()[ RES_POOLCHR_ENDNOTE - RES_POOLCHR_BEGIN ] ); m_aLst.Append( SfxStyleFamily::Char, SwStyleNameMapper::GetChrFormatUINameArray()[ RES_POOLCHR_FOOTNOTE - RES_POOLCHR_BEGIN ] ); } AppendStyleList(SwStyleNameMapper::GetHTMLChrFormatUINameArray(), bIsSearchUsed, bSearchHidden, bOnlyHidden, SwGetPoolIdFromName::ChrFmt, SfxStyleFamily::Char); } } if( nSearchFamily == SfxStyleFamily::Para || nSearchFamily == SfxStyleFamily::All ) { SfxStyleSearchBits nSMask = nSrchMask; if( rDoc.getIDocumentSettingAccess().get(DocumentSettingId::HTML_MODE) ) { // then only HTML-Template are of interest if( SfxStyleSearchBits::AllVisible == ( nSMask & SfxStyleSearchBits::AllVisible ) ) nSMask = SfxStyleSearchBits::SwHtml | SfxStyleSearchBits::UserDefined | SfxStyleSearchBits::Used; else nSMask &= SfxStyleSearchBits::Used | SfxStyleSearchBits::UserDefined | SfxStyleSearchBits::SwCondColl | SfxStyleSearchBits::SwHtml; if( nSMask == SfxStyleSearchBits::Auto ) nSMask = SfxStyleSearchBits::SwHtml; } const size_t nArrLen = rDoc.GetTextFormatColls()->size(); for( size_t i = 0; i < nArrLen; i++ ) { SwTextFormatColl* pColl = (*rDoc.GetTextFormatColls())[ i ]; const bool bUsed = bOrganizer || rDoc.IsUsed(*pColl) || IsUsedInComments(pColl->GetName()); if ( ( !bSearchHidden && pColl->IsHidden( ) && !bUsed ) || pColl->IsDefault() ) continue; if ( nSMask == SfxStyleSearchBits::Hidden && !pColl->IsHidden( ) ) continue; if( !(bIsSearchUsed && bUsed )) { const sal_uInt16 nId = pColl->GetPoolFormatId(); auto tmpMask = nSMask & ~SfxStyleSearchBits::Used; if (tmpMask == SfxStyleSearchBits::UserDefined) { if(!IsPoolUserFormat(nId)) continue; } else if (tmpMask == SfxStyleSearchBits::SwText) { if((nId & COLL_GET_RANGE_BITS) != COLL_TEXT_BITS) continue; } else if (tmpMask == SfxStyleSearchBits::SwChapter) { if((nId & COLL_GET_RANGE_BITS) != COLL_DOC_BITS) continue; } else if (tmpMask == SfxStyleSearchBits::SwList) { if((nId & COLL_GET_RANGE_BITS) != COLL_LISTS_BITS) continue; } else if (tmpMask == SfxStyleSearchBits::SwIndex) { if((nId & COLL_GET_RANGE_BITS) != COLL_REGISTER_BITS) continue; } else if (tmpMask == SfxStyleSearchBits::SwExtra) { if((nId & COLL_GET_RANGE_BITS) != COLL_EXTRA_BITS) continue; } else if (tmpMask == (SfxStyleSearchBits::SwHtml | SfxStyleSearchBits::UserDefined) || tmpMask == SfxStyleSearchBits::SwHtml) { if((tmpMask & SfxStyleSearchBits::UserDefined) && IsPoolUserFormat(nId)) ; // do nothing else if( (nId & COLL_GET_RANGE_BITS) != COLL_HTML_BITS) { // but some we also want to see in this section bool bContinue = true; switch( nId ) { case RES_POOLCOLL_SEND_ADDRESS: // --> ADDRESS case RES_POOLCOLL_TABLE_HDLN: // --> TH case RES_POOLCOLL_TABLE: // --> TD case RES_POOLCOLL_TEXT: // --> P case RES_POOLCOLL_HEADLINE_BASE:// --> H case RES_POOLCOLL_HEADLINE1: // --> H1 case RES_POOLCOLL_HEADLINE2: // --> H2 case RES_POOLCOLL_HEADLINE3: // --> H3 case RES_POOLCOLL_HEADLINE4: // --> H4 case RES_POOLCOLL_HEADLINE5: // --> H5 case RES_POOLCOLL_HEADLINE6: // --> H6 case RES_POOLCOLL_STANDARD: // --> P case RES_POOLCOLL_FOOTNOTE: case RES_POOLCOLL_ENDNOTE: bContinue = false; break; } if( bContinue ) continue; } } else if (tmpMask == SfxStyleSearchBits::SwCondColl) { if( RES_CONDTXTFMTCOLL != pColl->Which() ) continue; } else { // searched for used and found none if( bIsSearchUsed ) continue; } } m_aLst.Append( SfxStyleFamily::Para, pColl->GetName() ); } bAll = ( nSMask & SfxStyleSearchBits::AllVisible ) == SfxStyleSearchBits::AllVisible; if ( bAll || (nSMask & ~SfxStyleSearchBits::Used) == SfxStyleSearchBits::SwText ) AppendStyleList(SwStyleNameMapper::GetTextUINameArray(), bIsSearchUsed, bSearchHidden, bOnlyHidden, SwGetPoolIdFromName::TxtColl, SfxStyleFamily::Para ); if ( bAll || (nSMask & ~SfxStyleSearchBits::Used) == SfxStyleSearchBits::SwChapter ) AppendStyleList(SwStyleNameMapper::GetDocUINameArray(), bIsSearchUsed, bSearchHidden, bOnlyHidden, SwGetPoolIdFromName::TxtColl, SfxStyleFamily::Para ) ; if ( bAll || (nSMask & ~SfxStyleSearchBits::Used) == SfxStyleSearchBits::SwList ) AppendStyleList(SwStyleNameMapper::GetListsUINameArray(), bIsSearchUsed, bSearchHidden, bOnlyHidden, SwGetPoolIdFromName::TxtColl, SfxStyleFamily::Para ) ; if ( bAll || (nSMask & ~SfxStyleSearchBits::Used) == SfxStyleSearchBits::SwIndex ) AppendStyleList(SwStyleNameMapper::GetRegisterUINameArray(), bIsSearchUsed, bSearchHidden, bOnlyHidden, SwGetPoolIdFromName::TxtColl, SfxStyleFamily::Para ) ; if ( bAll || (nSMask & ~SfxStyleSearchBits::Used) == SfxStyleSearchBits::SwExtra ) AppendStyleList(SwStyleNameMapper::GetExtraUINameArray(), bIsSearchUsed, bSearchHidden, bOnlyHidden, SwGetPoolIdFromName::TxtColl, SfxStyleFamily::Para ) ; if ( bAll || (nSMask & ~SfxStyleSearchBits::Used) == SfxStyleSearchBits::SwCondColl ) { if( !bIsSearchUsed || rDoc.getIDocumentStylePoolAccess().IsPoolTextCollUsed( RES_POOLCOLL_TEXT )) m_aLst.Append( SfxStyleFamily::Para, SwStyleNameMapper::GetTextUINameArray()[ RES_POOLCOLL_TEXT - RES_POOLCOLL_TEXT_BEGIN ] ); } if ( bAll || (nSMask & ~SfxStyleSearchBits::Used) == SfxStyleSearchBits::SwHtml || (nSMask & ~SfxStyleSearchBits::Used) == (SfxStyleSearchBits::SwHtml | SfxStyleSearchBits::UserDefined) ) { AppendStyleList(SwStyleNameMapper::GetHTMLUINameArray(), bIsSearchUsed, bSearchHidden, bOnlyHidden, SwGetPoolIdFromName::TxtColl, SfxStyleFamily::Para ) ; if( !bAll ) { // then also the ones, that we are mapping: static sal_uInt16 aPoolIds[] = { RES_POOLCOLL_SEND_ADDRESS, // --> ADDRESS RES_POOLCOLL_TABLE_HDLN, // --> TH RES_POOLCOLL_TABLE, // --> TD RES_POOLCOLL_STANDARD, // --> P RES_POOLCOLL_TEXT, // --> P RES_POOLCOLL_HEADLINE_BASE, // --> H RES_POOLCOLL_HEADLINE1, // --> H1 RES_POOLCOLL_HEADLINE2, // --> H2 RES_POOLCOLL_HEADLINE3, // --> H3 RES_POOLCOLL_HEADLINE4, // --> H4 RES_POOLCOLL_HEADLINE5, // --> H5 RES_POOLCOLL_HEADLINE6, // --> H6 RES_POOLCOLL_FOOTNOTE, RES_POOLCOLL_ENDNOTE, 0 }; sal_uInt16* pPoolIds = aPoolIds; OUString s; while( *pPoolIds ) { if( !bIsSearchUsed || rDoc.getIDocumentStylePoolAccess().IsPoolTextCollUsed( *pPoolIds ) ) { s = SwStyleNameMapper::GetUIName( *pPoolIds, s ); m_aLst.Append( SfxStyleFamily::Para, s); } ++pPoolIds; } } } } if( nSearchFamily == SfxStyleFamily::Frame || nSearchFamily == SfxStyleFamily::All ) { const size_t nArrLen = rDoc.GetFrameFormats()->size(); for( size_t i = 0; i < nArrLen; i++ ) { const SwFrameFormat* pFormat = (*rDoc.GetFrameFormats())[ i ]; bool bUsed = bIsSearchUsed && ( bOrganizer || rDoc.IsUsed(*pFormat)); if( ( !bSearchHidden && pFormat->IsHidden( ) && !bUsed ) || pFormat->IsDefault() || pFormat->IsAuto() ) continue; if ( nSrchMask == SfxStyleSearchBits::Hidden && !pFormat->IsHidden( ) ) continue; const sal_uInt16 nId = pFormat->GetPoolFormatId(); if( !bUsed ) { if( (nSrchMask & ~SfxStyleSearchBits::Used) == SfxStyleSearchBits::UserDefined ? !(nId & USER_FMT) // searched for used and found none : bIsSearchUsed ) { continue; } } m_aLst.Append( SfxStyleFamily::Frame, pFormat->GetName() ); } // PoolFormat if ( bAll ) AppendStyleList(SwStyleNameMapper::GetFrameFormatUINameArray(), bIsSearchUsed, bSearchHidden, bOnlyHidden, SwGetPoolIdFromName::FrmFmt, SfxStyleFamily::Frame); } if( nSearchFamily == SfxStyleFamily::Page || nSearchFamily == SfxStyleFamily::All ) { const size_t nCount = rDoc.GetPageDescCnt(); for(size_t i = 0; i < nCount; ++i) { const SwPageDesc& rDesc = rDoc.GetPageDesc(i); const sal_uInt16 nId = rDesc.GetPoolFormatId(); bool bUsed = bIsSearchUsed && ( bOrganizer || rDoc.IsUsed(rDesc)); if( !bUsed ) { if ( ( !bSearchHidden && rDesc.IsHidden() ) || ( (nSrchMask & ~SfxStyleSearchBits::Used) == SfxStyleSearchBits::UserDefined ? !(nId & USER_FMT) // searched for used and found none : bIsSearchUsed ) ) continue; } if ( nSrchMask == SfxStyleSearchBits::Hidden && !rDesc.IsHidden( ) ) continue; m_aLst.Append( SfxStyleFamily::Page, rDesc.GetName() ); } if ( bAll ) AppendStyleList(SwStyleNameMapper::GetPageDescUINameArray(), bIsSearchUsed, bSearchHidden, bOnlyHidden, SwGetPoolIdFromName::PageDesc, SfxStyleFamily::Page); } if( nSearchFamily == SfxStyleFamily::Pseudo || nSearchFamily == SfxStyleFamily::All ) { const SwNumRuleTable& rNumTable = rDoc.GetNumRuleTable(); for(auto const& rNum: rNumTable) { const SwNumRule& rRule = *rNum; if( !rRule.IsAutoRule() ) { if ( nSrchMask == SfxStyleSearchBits::Hidden && !rRule.IsHidden( ) ) continue; bool bUsed = bIsSearchUsed && (bOrganizer || rDoc.IsUsed(rRule)); if( !bUsed ) { if( ( !bSearchHidden && rRule.IsHidden() ) || ( (nSrchMask & ~SfxStyleSearchBits::Used) == SfxStyleSearchBits::UserDefined ? !(rRule.GetPoolFormatId() & USER_FMT) // searched for used and found none : bIsSearchUsed ) ) continue; } m_aLst.Append( SfxStyleFamily::Pseudo, rRule.GetName() ); } } if ( bAll ) AppendStyleList(SwStyleNameMapper::GetNumRuleUINameArray(), bIsSearchUsed, bSearchHidden, bOnlyHidden, SwGetPoolIdFromName::NumRule, SfxStyleFamily::Pseudo); } if( nSearchFamily == SfxStyleFamily::Table || nSearchFamily == SfxStyleFamily::All ) { const SwTableAutoFormatTable& rTableStyles = rDoc.GetTableStyles(); for(size_t i = 0; i < rTableStyles.size(); ++i) { const SwTableAutoFormat& rTableStyle = rTableStyles[i]; bool bUsed = bIsSearchUsed && (bOrganizer || rDoc.IsUsed(rTableStyle)); if(!bUsed) { if(nSrchMask == SfxStyleSearchBits::Hidden && !rTableStyle.IsHidden()) continue; if( (!bSearchHidden && rTableStyle.IsHidden() ) || ( (nSrchMask & ~SfxStyleSearchBits::Used) == SfxStyleSearchBits::UserDefined ? !rTableStyle.IsUserDefined() // searched for used and found none : bIsSearchUsed ) ) continue; } m_aLst.Append( SfxStyleFamily::Table, rTableStyle.GetName() ); } } if( nSearchFamily == SfxStyleFamily::Cell || nSearchFamily == SfxStyleFamily::All ) { const auto& aTableTemplateMap = SwTableAutoFormat::GetTableTemplateMap(); if (rDoc.HasTableStyles()) { const SwTableAutoFormatTable& rTableStyles = rDoc.GetTableStyles(); for(size_t i = 0; i < rTableStyles.size(); ++i) { const SwTableAutoFormat& rTableStyle = rTableStyles[i]; for(size_t nBoxFormat = 0; nBoxFormat < aTableTemplateMap.size(); ++nBoxFormat) { const sal_uInt32 nBoxIndex = aTableTemplateMap[nBoxFormat]; const SwBoxAutoFormat& rBoxFormat = rTableStyle.GetBoxFormat(nBoxIndex); OUString sBoxFormatName; SwStyleNameMapper::FillProgName(rTableStyle.GetName(), sBoxFormatName, SwGetPoolIdFromName::TabStyle); sBoxFormatName += rTableStyle.GetTableTemplateCellSubName(rBoxFormat); m_aLst.Append( SfxStyleFamily::Cell, sBoxFormatName ); } } } const SwCellStyleTable& rCellStyles = rDoc.GetCellStyles(); for(size_t i = 0; i < rCellStyles.size(); ++i) m_aLst.Append( SfxStyleFamily::Cell, rCellStyles[i].GetName() ); } if(!m_aLst.empty()) { m_nLastPos = SAL_MAX_UINT32; return Next(); } return nullptr; } SfxStyleSheetBase* SwStyleSheetIterator::Next() { assert(m_bFirstCalled); ++m_nLastPos; if(m_nLastPos < m_aLst.size()) { auto const & rEntry = m_aLst[m_nLastPos]; mxIterSheet->PresetNameAndFamily(rEntry.first, rEntry.second); mxIterSheet->SetPhysical( false ); mxIterSheet->SetMask( nMask ); if(mxIterSheet->pSet) { mxIterSheet->pSet->ClearItem(); mxIterSheet->pSet= nullptr; } return mxIterSheet.get(); } return nullptr; } SfxStyleSheetBase* SwStyleSheetIterator::Find(const OUString& rName) { // searching if( !m_bFirstCalled ) First(); m_nLastPos = m_aLst.FindName( nSearchFamily, rName ); if( SAL_MAX_UINT32 != m_nLastPos ) { // found auto const & rEntry = m_aLst[m_nLastPos]; mxStyleSheet->PresetNameAndFamily(rEntry.first, rEntry.second); // new name is set, so determine its Data mxStyleSheet->FillStyleSheet( SwDocStyleSheet::FillOnlyName ); if( !mxStyleSheet->IsPhysical() ) mxStyleSheet->SetPhysical( false ); return mxStyleSheet.get(); } return nullptr; } void SwStyleSheetIterator::AppendStyleList(const std::vector& rList, bool bTestUsed, bool bTestHidden, bool bOnlyHidden, SwGetPoolIdFromName nSection, SfxStyleFamily eFamily ) { const SwDoc& rDoc = static_cast(pBasePool)->GetDoc(); bool bUsed = false; for (const auto & i : rList) { bool bHidden = false; sal_uInt16 nId = SwStyleNameMapper::GetPoolIdFromUIName(i, nSection); switch ( nSection ) { case SwGetPoolIdFromName::TxtColl: { bUsed = rDoc.getIDocumentStylePoolAccess().IsPoolTextCollUsed( nId ); SwFormat* pFormat = rDoc.FindTextFormatCollByName( i ); bHidden = pFormat && pFormat->IsHidden( ); } break; case SwGetPoolIdFromName::ChrFmt: { bUsed = rDoc.getIDocumentStylePoolAccess().IsPoolFormatUsed( nId ); SwFormat* pFormat = rDoc.FindCharFormatByName( i ); bHidden = pFormat && pFormat->IsHidden( ); } break; case SwGetPoolIdFromName::FrmFmt: { bUsed = rDoc.getIDocumentStylePoolAccess().IsPoolFormatUsed( nId ); SwFormat* pFormat = rDoc.FindFrameFormatByName( i ); bHidden = pFormat && pFormat->IsHidden( ); } break; case SwGetPoolIdFromName::PageDesc: { bUsed = rDoc.getIDocumentStylePoolAccess().IsPoolPageDescUsed( nId ); SwPageDesc* pPgDesc = rDoc.FindPageDesc(i); bHidden = pPgDesc && pPgDesc->IsHidden( ); } break; case SwGetPoolIdFromName::NumRule: { SwNumRule* pRule = rDoc.FindNumRulePtr( i ); bUsed = pRule && rDoc.IsUsed(*pRule); bHidden = pRule && pRule->IsHidden( ); } break; default: OSL_ENSURE( false, "unknown PoolFormat-Id" ); } bool bMatchHidden = ( bTestHidden && ( bHidden || !bOnlyHidden ) ) || ( !bTestHidden && ( !bHidden || bUsed ) ); if ( ( !bTestUsed && bMatchHidden ) || ( bTestUsed && bUsed ) ) m_aLst.Append( eFamily, i ); } } bool SwStyleSheetIterator::IsUsedInComments(const OUString& rName) const { auto pPool = static_cast(pBasePool)->GetEEStyleSheetPool(); SfxStyleSheetIterator aIter(pPool, GetSearchFamily(), SfxStyleSearchBits::Used); return aIter.Find(rName) != nullptr; } void SwDocStyleSheetPool::InvalidateIterator() { if (SfxStyleSheetIterator* pIter = GetCachedIterator()) dynamic_cast(*pIter).InvalidateIterator(); } void SwStyleSheetIterator::InvalidateIterator() { // potentially we could send an SfxHint to Notify but currently it's // iterating over the vector anyway so would still be slow - why does // this iterator not use a map? m_bFirstCalled = false; m_nLastPos = 0; m_aLst.clear(); } void SwStyleSheetIterator::Notify( SfxBroadcaster&, const SfxHint& rHint ) { // search and remove from View-List!! const SfxStyleSheetHint* pStyleSheetHint = dynamic_cast(&rHint); if( pStyleSheetHint && SfxHintId::StyleSheetErased == pStyleSheetHint->GetId() ) { SfxStyleSheetBase* pStyle = pStyleSheetHint->GetStyleSheet(); if (pStyle) m_aLst.RemoveName(pStyle->GetFamily(), pStyle->GetName()); } } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */