Kaydet (Commit) e9e5fee7 authored tarafından Daniel Robertson's avatar Daniel Robertson Kaydeden (comit) Noel Grandin

tdf#62525 editeng: OutlinerParaObject

Convert the pimpled copy-on-write OutlinerParaObject class from
editeng to use ::o3tl::cow_wrapper using the default reference
counting policy.

Change-Id: I7009b9623d5be1bdf806ccaa6781cba82553c302
Reviewed-on: https://gerrit.libreoffice.org/18391Tested-by: 's avatarJenkins <ci@libreoffice.org>
Reviewed-by: 's avatarNoel Grandin <noelgrandin@gmail.com>
üst ee7da352
...@@ -32,169 +32,117 @@ ...@@ -32,169 +32,117 @@
#include <vcl/bitmap.hxx> #include <vcl/bitmap.hxx>
#include <tools/stream.hxx> #include <tools/stream.hxx>
#include <boost/intrusive_ptr.hpp> #include <o3tl/cow_wrapper.hxx>
#include <libxml/xmlwriter.h> #include <libxml/xmlwriter.h>
/** OutlinerParaObjData::OutlinerParaObjData( EditTextObject* pEditTextObject, const ParagraphDataVector& rParagraphDataVector, bool bIsEditDoc ) :
* This is the guts of OutlinerParaObject, refcounted and shared among mpEditTextObject(pEditTextObject),
* multiple instances of OutlinerParaObject. maParagraphDataVector(rParagraphDataVector),
*/ mbIsEditDoc(bIsEditDoc)
struct OutlinerParaObjData
{ {
// data members if( maParagraphDataVector.empty() && (pEditTextObject->GetParagraphCount() != 0) )
EditTextObject* mpEditTextObject; maParagraphDataVector.resize(pEditTextObject->GetParagraphCount());
ParagraphDataVector maParagraphDataVector; }
bool mbIsEditDoc;
// refcounter
mutable size_t mnRefCount;
// constuctor
OutlinerParaObjData( EditTextObject* pEditTextObject, const ParagraphDataVector& rParagraphDataVector, bool bIsEditDoc ) :
mpEditTextObject(pEditTextObject),
maParagraphDataVector(rParagraphDataVector),
mbIsEditDoc(bIsEditDoc),
mnRefCount(0)
{
if( maParagraphDataVector.empty() && (pEditTextObject->GetParagraphCount() != 0) )
maParagraphDataVector.resize(pEditTextObject->GetParagraphCount());
}
OutlinerParaObjData( const OutlinerParaObjData& r ) :
mpEditTextObject(r.mpEditTextObject->Clone()),
maParagraphDataVector(r.maParagraphDataVector),
mbIsEditDoc(r.mbIsEditDoc),
mnRefCount(0)
{}
// destructor
~OutlinerParaObjData()
{
delete mpEditTextObject;
}
bool operator==(const OutlinerParaObjData& rCandidate) const
{
return (*mpEditTextObject == *rCandidate.mpEditTextObject
&& maParagraphDataVector == rCandidate.maParagraphDataVector
&& mbIsEditDoc == rCandidate.mbIsEditDoc);
}
// #i102062#
bool isWrongListEqual(const OutlinerParaObjData& rCompare) const
{
return mpEditTextObject->isWrongListEqual(*rCompare.mpEditTextObject);
}
};
inline void intrusive_ptr_add_ref(const OutlinerParaObjData* p) OutlinerParaObjData::OutlinerParaObjData( const OutlinerParaObjData& r ):
mpEditTextObject(r.mpEditTextObject->Clone()),
maParagraphDataVector(r.maParagraphDataVector),
mbIsEditDoc(r.mbIsEditDoc)
{ {
++p->mnRefCount;
} }
inline void intrusive_ptr_release(const OutlinerParaObjData* p) OutlinerParaObjData::~OutlinerParaObjData()
{ {
--p->mnRefCount; delete mpEditTextObject;
if (!p->mnRefCount)
delete p;
} }
struct OutlinerParaObject::Impl bool OutlinerParaObjData::operator==(const OutlinerParaObjData& rCandidate) const
{ {
typedef boost::intrusive_ptr<OutlinerParaObjData> DataRef; return (*mpEditTextObject == *rCandidate.mpEditTextObject
DataRef mxData; && maParagraphDataVector == rCandidate.maParagraphDataVector
&& mbIsEditDoc == rCandidate.mbIsEditDoc);
Impl( const EditTextObject& rTextObj, const ParagraphDataVector& rParaData, bool bIsEditDoc ) : }
mxData(new OutlinerParaObjData(rTextObj.Clone(), rParaData, bIsEditDoc)) {}
explicit Impl(const EditTextObject& rTextObj) :
mxData(new OutlinerParaObjData(rTextObj.Clone(), ParagraphDataVector(), true)) {}
Impl( const Impl& r ) : mxData(r.mxData) {}
~Impl() {}
};
void OutlinerParaObject::ImplMakeUnique() bool OutlinerParaObjData::isWrongListEqual(const OutlinerParaObjData& rCompare) const
{ {
mpImpl->mxData.reset(new OutlinerParaObjData(*mpImpl->mxData)); return mpEditTextObject->isWrongListEqual(*rCompare.mpEditTextObject);
} }
OutlinerParaObject::OutlinerParaObject( OutlinerParaObject::OutlinerParaObject(
const EditTextObject& rTextObj, const ParagraphDataVector& rParagraphDataVector, bool bIsEditDoc ) : const EditTextObject& rTextObj, const ParagraphDataVector& rParagraphDataVector, bool bIsEditDoc ) :
mpImpl(new Impl(rTextObj, rParagraphDataVector, bIsEditDoc)) {} mpImpl(OutlinerParaObjData(rTextObj.Clone(), rParagraphDataVector, bIsEditDoc))
{
}
OutlinerParaObject::OutlinerParaObject( const EditTextObject& rTextObj ) : OutlinerParaObject::OutlinerParaObject( const EditTextObject& rTextObj ) :
mpImpl(new Impl(rTextObj)) mpImpl(OutlinerParaObjData(rTextObj.Clone(), ParagraphDataVector(), true))
{ {
} }
OutlinerParaObject::OutlinerParaObject( const OutlinerParaObject& r ) : OutlinerParaObject::OutlinerParaObject( const OutlinerParaObject& r ) :
mpImpl(new Impl(*r.mpImpl)) {} mpImpl(r.mpImpl)
{
}
OutlinerParaObject::~OutlinerParaObject() OutlinerParaObject::~OutlinerParaObject()
{ {
delete mpImpl;
} }
OutlinerParaObject& OutlinerParaObject::operator=( const OutlinerParaObject& r ) OutlinerParaObject& OutlinerParaObject::operator=( const OutlinerParaObject& r )
{ {
mpImpl->mxData = r.mpImpl->mxData; mpImpl = r.mpImpl;
return *this; return *this;
} }
bool OutlinerParaObject::operator==( const OutlinerParaObject& r ) const bool OutlinerParaObject::operator==( const OutlinerParaObject& r ) const
{ {
if (r.mpImpl->mxData.get() == mpImpl->mxData.get()) return r.mpImpl == mpImpl;
{
return true;
}
return (*r.mpImpl->mxData == *mpImpl->mxData);
} }
// #i102062# // #i102062#
bool OutlinerParaObject::isWrongListEqual( const OutlinerParaObject& r ) const bool OutlinerParaObject::isWrongListEqual( const OutlinerParaObject& r ) const
{ {
if (r.mpImpl->mxData.get() == mpImpl->mxData.get()) if (r.mpImpl.same_object(mpImpl))
{ {
return true; return true;
} }
return mpImpl->mxData->isWrongListEqual(*r.mpImpl->mxData); return mpImpl->isWrongListEqual(*r.mpImpl);
} }
sal_uInt16 OutlinerParaObject::GetOutlinerMode() const sal_uInt16 OutlinerParaObject::GetOutlinerMode() const
{ {
return mpImpl->mxData->mpEditTextObject->GetUserType(); return mpImpl->mpEditTextObject->GetUserType();
} }
void OutlinerParaObject::SetOutlinerMode(sal_uInt16 nNew) void OutlinerParaObject::SetOutlinerMode(sal_uInt16 nNew)
{ {
if (mpImpl->mxData->mpEditTextObject->GetUserType() != nNew) // create a const pointer to avoid an early call to
// make_unique() in the dereference of mpImpl
const ::o3tl::cow_wrapper< OutlinerParaObjData >* pImpl = &mpImpl;
if ( ( *pImpl )->mpEditTextObject->GetUserType() != nNew )
{ {
ImplMakeUnique(); mpImpl->mpEditTextObject->SetUserType(nNew);
mpImpl->mxData->mpEditTextObject->SetUserType(nNew);
} }
} }
bool OutlinerParaObject::IsVertical() const bool OutlinerParaObject::IsVertical() const
{ {
return mpImpl->mxData->mpEditTextObject->IsVertical(); return mpImpl->mpEditTextObject->IsVertical();
} }
void OutlinerParaObject::SetVertical(bool bNew) void OutlinerParaObject::SetVertical(bool bNew)
{ {
if (mpImpl->mxData->mpEditTextObject->IsVertical() != bNew) const ::o3tl::cow_wrapper< OutlinerParaObjData >* pImpl = &mpImpl;
if ( ( *pImpl )->mpEditTextObject->IsVertical() != bNew )
{ {
ImplMakeUnique(); mpImpl->mpEditTextObject->SetVertical(bNew);
mpImpl->mxData->mpEditTextObject->SetVertical(bNew);
} }
} }
sal_Int32 OutlinerParaObject::Count() const sal_Int32 OutlinerParaObject::Count() const
{ {
size_t nSize = mpImpl->mxData->maParagraphDataVector.size(); size_t nSize = mpImpl->maParagraphDataVector.size();
if (nSize > EE_PARA_MAX_COUNT) if (nSize > EE_PARA_MAX_COUNT)
{ {
SAL_WARN( "editeng", "OutlinerParaObject::Count - overflow " << nSize); SAL_WARN( "editeng", "OutlinerParaObject::Count - overflow " << nSize);
...@@ -205,9 +153,9 @@ sal_Int32 OutlinerParaObject::Count() const ...@@ -205,9 +153,9 @@ sal_Int32 OutlinerParaObject::Count() const
sal_Int16 OutlinerParaObject::GetDepth(sal_Int32 nPara) const sal_Int16 OutlinerParaObject::GetDepth(sal_Int32 nPara) const
{ {
if(0 <= nPara && static_cast<size_t>(nPara) < mpImpl->mxData->maParagraphDataVector.size()) if(0 <= nPara && static_cast<size_t>(nPara) < mpImpl->maParagraphDataVector.size())
{ {
return mpImpl->mxData->maParagraphDataVector[nPara].getDepth(); return mpImpl->maParagraphDataVector[nPara].getDepth();
} }
else else
{ {
...@@ -217,19 +165,19 @@ sal_Int16 OutlinerParaObject::GetDepth(sal_Int32 nPara) const ...@@ -217,19 +165,19 @@ sal_Int16 OutlinerParaObject::GetDepth(sal_Int32 nPara) const
const EditTextObject& OutlinerParaObject::GetTextObject() const const EditTextObject& OutlinerParaObject::GetTextObject() const
{ {
return *mpImpl->mxData->mpEditTextObject; return *mpImpl->mpEditTextObject;
} }
bool OutlinerParaObject::IsEditDoc() const bool OutlinerParaObject::IsEditDoc() const
{ {
return mpImpl->mxData->mbIsEditDoc; return mpImpl->mbIsEditDoc;
} }
const ParagraphData& OutlinerParaObject::GetParagraphData(sal_Int32 nIndex) const const ParagraphData& OutlinerParaObject::GetParagraphData(sal_Int32 nIndex) const
{ {
if(0 <= nIndex && static_cast<size_t>(nIndex) < mpImpl->mxData->maParagraphDataVector.size()) if(0 <= nIndex && static_cast<size_t>(nIndex) < mpImpl->maParagraphDataVector.size())
{ {
return mpImpl->mxData->maParagraphDataVector[nIndex]; return mpImpl->maParagraphDataVector[nIndex];
} }
else else
{ {
...@@ -241,22 +189,19 @@ const ParagraphData& OutlinerParaObject::GetParagraphData(sal_Int32 nIndex) cons ...@@ -241,22 +189,19 @@ const ParagraphData& OutlinerParaObject::GetParagraphData(sal_Int32 nIndex) cons
void OutlinerParaObject::ClearPortionInfo() void OutlinerParaObject::ClearPortionInfo()
{ {
ImplMakeUnique(); mpImpl->mpEditTextObject->ClearPortionInfo();
mpImpl->mxData->mpEditTextObject->ClearPortionInfo();
} }
bool OutlinerParaObject::ChangeStyleSheets(const OUString& rOldName, bool OutlinerParaObject::ChangeStyleSheets(const OUString& rOldName,
SfxStyleFamily eOldFamily, const OUString& rNewName, SfxStyleFamily eNewFamily) SfxStyleFamily eOldFamily, const OUString& rNewName, SfxStyleFamily eNewFamily)
{ {
ImplMakeUnique(); return mpImpl->mpEditTextObject->ChangeStyleSheets(rOldName, eOldFamily, rNewName, eNewFamily);
return mpImpl->mxData->mpEditTextObject->ChangeStyleSheets(rOldName, eOldFamily, rNewName, eNewFamily);
} }
void OutlinerParaObject::ChangeStyleSheetName(SfxStyleFamily eFamily, void OutlinerParaObject::ChangeStyleSheetName(SfxStyleFamily eFamily,
const OUString& rOldName, const OUString& rNewName) const OUString& rOldName, const OUString& rNewName)
{ {
ImplMakeUnique(); mpImpl->mpEditTextObject->ChangeStyleSheetName(eFamily, rOldName, rNewName);
mpImpl->mxData->mpEditTextObject->ChangeStyleSheetName(eFamily, rOldName, rNewName);
} }
void OutlinerParaObject::SetStyleSheets(sal_uInt16 nLevel, const OUString& rNewName, void OutlinerParaObject::SetStyleSheets(sal_uInt16 nLevel, const OUString& rNewName,
...@@ -266,14 +211,13 @@ void OutlinerParaObject::SetStyleSheets(sal_uInt16 nLevel, const OUString& rNewN ...@@ -266,14 +211,13 @@ void OutlinerParaObject::SetStyleSheets(sal_uInt16 nLevel, const OUString& rNewN
if(nCount) if(nCount)
{ {
ImplMakeUnique();
sal_Int32 nDecrementer(nCount); sal_Int32 nDecrementer(nCount);
for(;nDecrementer;) while(nDecrementer > 0)
{ {
if(GetDepth(--nDecrementer) == nLevel) if(GetDepth(--nDecrementer) == nLevel)
{ {
mpImpl->mxData->mpEditTextObject->SetStyleSheet(nDecrementer, rNewName, rNewFamily); mpImpl->mpEditTextObject->SetStyleSheet(nDecrementer, rNewName, rNewFamily);
} }
} }
} }
......
...@@ -24,15 +24,38 @@ ...@@ -24,15 +24,38 @@
#include <editeng/editengdllapi.h> #include <editeng/editengdllapi.h>
#include <rtl/ustring.hxx> #include <rtl/ustring.hxx>
#include <rsc/rscsfx.hxx> #include <rsc/rscsfx.hxx>
#include <o3tl/cow_wrapper.hxx>
class EditTextObject; class EditTextObject;
class EDITENG_DLLPUBLIC OutlinerParaObject /**
* This is the guts of OutlinerParaObject, refcounted and shared among
* multiple instances of OutlinerParaObject.
*/
struct OutlinerParaObjData
{ {
struct Impl; // data members
Impl* mpImpl; EditTextObject* mpEditTextObject;
ParagraphDataVector maParagraphDataVector;
bool mbIsEditDoc;
// constuctor
OutlinerParaObjData( EditTextObject* pEditTextObject, const ParagraphDataVector& rParagraphDataVector, bool bIsEditDoc );
OutlinerParaObjData( const OutlinerParaObjData& r );
// destructor
~OutlinerParaObjData();
bool operator==(const OutlinerParaObjData& rCandidate) const;
void ImplMakeUnique(); // #i102062#
bool isWrongListEqual(const OutlinerParaObjData& rCompare) const;
};
class EDITENG_DLLPUBLIC OutlinerParaObject
{
::o3tl::cow_wrapper< OutlinerParaObjData > mpImpl;
public: public:
// constructors/destructor // constructors/destructor
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment