1
0
Fork 0
libreoffice/sw/source/uibase/app/docstyle.cxx
Daniel Baumann 8e63e14cf6
Adding upstream version 4:25.2.3.
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
2025-06-22 16:20:04 +02:00

3403 lines
118 KiB
C++

/* -*- 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 <memory>
#include <sal/config.h>
#include <sal/log.hxx>
#include <osl/diagnose.h>
#include <cstdlib>
#include <hintids.hxx>
#include <rtl/ustrbuf.hxx>
#include <svl/itemiter.hxx>
#include <svl/eitem.hxx>
#include <unotools/syslocale.hxx>
#include <editeng/boxitem.hxx>
#include <editeng/numitem.hxx>
#include <editeng/lrspitem.hxx>
#include <drawdoc.hxx>
#include <fmtcol.hxx>
#include <uitool.hxx>
#include <wrtsh.hxx>
#include <docsh.hxx>
#include <frmfmt.hxx>
#include <charfmt.hxx>
#include <tblafmt.hxx>
#include <poolfmt.hxx>
#include <pagedesc.hxx>
#include <docstyle.hxx>
#include <docary.hxx>
#include <ccoll.hxx>
#include <doc.hxx>
#include <IDocumentUndoRedo.hxx>
#include <IDocumentStylePoolAccess.hxx>
#include <IDocumentSettingAccess.hxx>
#include <IDocumentDrawModelAccess.hxx>
#include <IDocumentState.hxx>
#include <cmdid.h>
#include <strings.hrc>
#include <paratr.hxx>
#include <SwStyleNameMapper.hxx>
#include <svl/cjkoptions.hxx>
#include <svl/ctloptions.hxx>
#include <unotools/intlwrapper.hxx>
#include <numrule.hxx>
#include <svx/xdef.hxx>
#include <SwRewriter.hxx>
#include <hints.hxx>
#include <frameformats.hxx>
#include <editeng/brushitem.hxx>
#include <editeng/colritem.hxx>
#include <editeng/eeitem.hxx>
#include <svx/xfillit0.hxx>
#include <svx/xflftrit.hxx>
#include <svx/drawitem.hxx>
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<svl::StyleSheetUser*>(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<SfxStyleSheetBase> Create(const OUString& rName, SfxStyleFamily eFamily,
SfxStyleSearchBits nMask,
const OUString& rParentStyleSheetName) override
{
return new EEStyleSheet(rName, *this, eFamily, nMask, rParentStyleSheetName);
}
void Notify(SfxBroadcaster&, const SfxHint& rHint) override
{
auto nId = rHint.GetId();
if (nId != SfxHintId::StyleSheetModified && nId != SfxHintId::StyleSheetModifiedExtended
&& nId != SfxHintId::StyleSheetErased)
return;
auto pHint = static_cast<const SfxStyleSheetHint*>(&rHint);
auto pDocStyleSheet = pHint->GetStyleSheet();
const SfxStyleSheetModifiedHint* pExtendedHint = nullptr;
if (nId == SfxHintId::StyleSheetModifiedExtended)
pExtendedHint = static_cast<const SfxStyleSheetModifiedHint*>(&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<SfxStyleSheet*>(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<SwDocStyleSheet*>(pSourceSheet)->GetCollection();
SfxItemIter aIter(pCol->GetAttrSet());
std::optional<SvxLRSpaceItem> 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<const SvxFirstLineIndentItem*>(pItem);
(*oLRSpaceItem).SetTextFirstLineOffset(pFirstLineItem->GetTextFirstLineOffset());
(*oLRSpaceItem).SetAutoFirst(pFirstLineItem->IsAutoFirst());
}
else if (nWhich == RES_MARGIN_TEXTLEFT)
{
if (!oLRSpaceItem)
oLRSpaceItem.emplace(EE_PARA_LRSPACE);
(*oLRSpaceItem)
.SetTextLeft(static_cast<const SvxTextLeftMarginItem*>(pItem)->GetTextLeft());
}
else if (nWhich == RES_MARGIN_RIGHT)
{
if (!oLRSpaceItem)
oLRSpaceItem.emplace(EE_PARA_LRSPACE);
(*oLRSpaceItem).SetRight(static_cast<const SvxRightMarginItem*>(pItem)->GetRight());
}
else if (nWhich == RES_CHRATR_BACKGROUND)
{
auto pBrushItem = static_cast<const SvxBrushItem*>(pItem);
rItemSet.Put(SvxColorItem(pBrushItem->GetColor(), EE_CHAR_BKGCOLOR));
}
}
if (oLRSpaceItem)
rItemSet.Put(*oLRSpaceItem);
}
};
class SwImplShellAction
{
SwWrtShell* m_pSh;
std::unique_ptr<CurrShell> 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<SfxStyleFamily,OUString>{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<SfxStyleFamily,OUString>{eFam, rStr});
if (it != maUnique.end())
return;
maUnique.emplace(std::pair<SfxStyleFamily,OUString>{eFam, rStr}, static_cast<sal_uInt32>(maImpl.size()));
maImpl.push_back(std::pair<SfxStyleFamily,OUString>{eFam, rStr});
}
// UI-sided implementation of StyleSheets
// uses the Core-Engine
SwDocStyleSheet::SwDocStyleSheet( SwDoc& rDocument,
SwDocStyleSheetPool& rPool) :
SfxStyleSheetBase( OUString(), &rPool, SfxStyleFamily::Char, SfxStyleSearchBits::Auto, u""_ustr ),
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<SwDocStyleSheetPool&>(*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<SwDocStyleSheetPool&>(*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:
{
return EMPTY_OUSTRING; // 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<SwDocStyleSheet*>(this);
pThis->aParent = sTmp;
}
return aParent;
}
// Follower
const OUString& SwDocStyleSheet::GetFollow() const
{
if( !m_bPhysical )
{
SwDocStyleSheet* pThis = const_cast<SwDocStyleSheet*>(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<SwDocStyleSheet*>(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<SwPageDesc*>(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<SwNumRule*>(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<SfxItemSet>& pRet)
{
// note: we don't add the odd items that GetItemSet() would add
// because they don't seem relevant for preview
std::vector<SfxItemSet const*> 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<SfxItemSet> SwDocStyleSheet::GetItemSetForPreview()
{
if (SfxStyleFamily::Page == nFamily || SfxStyleFamily::Pseudo == nFamily || SfxStyleFamily::Table == nFamily)
{
SAL_INFO("sw.ui", "GetItemSetForPreview not implemented for page or number or table style");
return std::optional<SfxItemSet>();
}
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<SfxItemSet> 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<SfxItemSet> 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 )
{
assert(m_pCharFormat && "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 )
{
assert(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
{
assert(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());
}
assert(m_pDesc && "No PageDescriptor");
::PageDescToItemSet(*const_cast<SwPageDesc*>(m_pDesc), m_aCoreSet);
}
break;
case SfxStyleFamily::Pseudo:
{
assert(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;
}
assert(m_pColl && "<SwDocStyleSheet::MergeIndentAttrsOfListStyle(..)> - 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,
"<SwDocStyleSheet::MergeIndentAttrsOfListStyle(..)> - 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(
SvxIndentValue{ static_cast<double>(rFormat.GetFirstLineIndent()),
rFormat.GetFirstLineIndentUnit() },
RES_MARGIN_FIRSTLINE);
rSet.Put(firstLine);
}
if (indents & ::sw::ListLevelIndents::LeftMargin)
{
SvxTextLeftMarginItem const leftMargin(SvxIndentValue::twips(rFormat.GetIndentAt()),
RES_MARGIN_TEXTLEFT);
rSet.Put(leftMargin);
}
}
}
}
// handling of parameter <bResetIndentAttrsAtParagraphStyle>
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<sal_uInt16> aWhichIdsToReset;
std::unique_ptr<SwPageDesc> 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<SwConditionTextFormatColl*>(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<SwConditionTextFormatColl*>(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<SwTextFormatColl*>(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<const SwNumRuleItem*>(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<SfxStyleSearchBits>(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<const SfxBoolItem*>(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();
// tdf#134166: Changing page style can affect toolbar button state.
if (SwEditShell* pSh = m_rDoc.GetEditShell())
pSh->CallChgLnk();
}
}
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<SvxNumRule&>(static_cast<const SvxNumBulletItem*>(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())
{
{
// Own scope for SfxItemIter needed to have it registered at rSet as short as possible
// due to m_aCoreSet.ClearItem() below - rSet *is* equal to m_aCoreSet here sometimes.
// This should *not* be the case (see OSL_ENSURE &rSet != &m_aCoreSet at the beginning
// of this method) but happens because lcl_setLineNumbering calls GetItemSet/SetItemSet
// at the *same* xStyleSheet. You can see that SwDocStyleSheet::GetItemSet() above
// does return m_aCoreSet. I guess that lcl_setLineNumbering should not do that...
SfxItemIter aIter( rSet );
const SfxPoolItem* pItem = aIter.GetCurItem();
do
{
if( IsInvalidItem( pItem ) ) // Clear
{
// use method <SwDoc::ResetAttrAtFormat(..)> in order to
// create an Undo object for the attribute reset.
aWhichIdsToReset.emplace_back(aIter.GetCurWhich());
}
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<void*>& 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<void*>& rArr, const void* p)
{
return std::find( rArr.begin(), rArr.end(), p ) != rArr.end();
}
static void lcl_DeleteInfoStyles( SfxStyleFamily nFamily, std::vector<void*> const & rArr, SwDoc& rDoc )
{
size_t n, nCnt;
switch( nFamily )
{
case SfxStyleFamily::Char:
{
std::deque<sal_uInt16> 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<sal_uInt16> 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<SwFrameFormat*> 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<size_t> 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<SwNumRule*> 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<SfxItemSet> *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<void*> 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);
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,
// #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<SwDocStyleSheet*>(this);
pThis->FillStyleSheet( FillOnlyName );
}
if( !m_bPhysical )
return false;
const sw::BroadcastingModify* pMod;
switch( nFamily )
{
case SfxStyleFamily::Para : pMod = m_pColl; break;
case SfxStyleFamily::Frame: pMod = m_pFrameFormat; break;
case SfxStyleFamily::Char:
return m_pCharFormat->IsUsed();
case SfxStyleFamily::Page:
return m_pDesc->IsUsed();
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<SwDocStyleSheetPool*>(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<SwPageDesc*>(m_pDesc)->SetPoolHelpId( nId );
const_cast<SwPageDesc*>(m_pDesc)->SetPoolHlpFileId( nFileId );
break;
case SfxStyleFamily::Pseudo:
const_cast<SwNumRule*>(m_pNumRule)->SetPoolHelpId( nId );
const_cast<SwNumRule*>(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,
const OUString& rParentStyleSheetName)
{
mxStyleSheet->PresetName(rName);
mxStyleSheet->PresetParent(OUString());
mxStyleSheet->PresetFollow(OUString());
mxStyleSheet->SetMask(_nMask) ;
mxStyleSheet->SetFamily(eFam);
mxStyleSheet->SetPhysical(true);
mxStyleSheet->SetParent(rParentStyleSheetName);
mxStyleSheet->Create();
return *mxStyleSheet;
}
rtl::Reference<SfxStyleSheetBase> SwDocStyleSheetPool::Create( const SfxStyleSheetBase& /*rOrg*/)
{
OSL_ENSURE(false , "Create in SW-Stylesheet-Pool not possible" );
return nullptr;
}
rtl::Reference<SfxStyleSheetBase> SwDocStyleSheetPool::Create( const OUString &,
SfxStyleFamily, SfxStyleSearchBits, const OUString& )
{
OSL_ENSURE( false, "Create in SW-Stylesheet-Pool not possible" );
return nullptr;
}
std::unique_ptr<SfxStyleSheetIterator> SwDocStyleSheetPool::CreateIterator( SfxStyleFamily eFam, SfxStyleSearchBits _nMask )
{
return std::make_unique<SwStyleSheetIterator>(*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<const SwPageDesc*>(pMod)->GetPoolFormatId()
: static_cast<const SwFormat*>(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<const SwDocStyleSheetPool*>(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<const SwDocStyleSheetPool*>(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 ];
// tdf#159168: default char format should be considered 'applied'/used, even if it apparently is not
const bool bUsed = bIsSearchUsed && (bOrganizer || pFormat->IsUsed() || pFormat == rDoc.GetDfltCharFormat());
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 || rDesc.IsUsed() );
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<OUString>& rList,
bool bTestUsed, bool bTestHidden, bool bOnlyHidden,
SwGetPoolIdFromName nSection, SfxStyleFamily eFamily )
{
const SwDoc& rDoc = static_cast<const SwDocStyleSheetPool*>(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<const SwDocStyleSheetPool*>(pBasePool)->GetEEStyleSheetPool();
SfxStyleSheetIterator aIter(pPool, GetSearchFamily(), SfxStyleSearchBits::Used);
return aIter.Find(rName) != nullptr;
}
void SwDocStyleSheetPool::InvalidateIterator()
{
if (SfxStyleSheetIterator* pIter = GetCachedIterator())
dynamic_cast<SwStyleSheetIterator&>(*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!!
if( SfxHintId::StyleSheetErased == rHint.GetId() )
{
const SfxStyleSheetHint* pStyleSheetHint = static_cast<const SfxStyleSheetHint*>(&rHint);
SfxStyleSheetBase* pStyle = pStyleSheetHint->GetStyleSheet();
if (pStyle)
m_aLst.RemoveName(pStyle->GetFamily(), pStyle->GetName());
}
}
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */