diff options
author | Noel Grandin <noel.grandin@collabora.co.uk> | 2023-06-19 12:23:55 +0200 |
---|---|---|
committer | Noel Grandin <noel.grandin@collabora.co.uk> | 2023-06-19 18:26:03 +0200 |
commit | e27d4cc31e04be4c47b5085dfa2363ee45457e8a (patch) | |
tree | 75aebab2bd9a0c2d0f152fb545757af0513f5aec | |
parent | 281bdb5a979f0516fe7a9cd35585fd3daf4021b1 (diff) |
tdf#150534 reduce the memory consumption of cells when calculating
borders - if there are a lot hidden columns, we end up
allocating a lot of cells. Since most of the cells have the same
settings, use a SfxItemPool to consolidate them.
Change-Id: If2dd77b3eabc5d37eb8804953f8c89ffa04f2400
Reviewed-on: https://gerrit.libreoffice.org/c/core/+/153258
Tested-by: Jenkins
Reviewed-by: Noel Grandin <noel.grandin@collabora.co.uk>
-rw-r--r-- | svx/source/dialog/framelinkarray.cxx | 151 |
1 files changed, 113 insertions, 38 deletions
diff --git a/svx/source/dialog/framelinkarray.cxx b/svx/source/dialog/framelinkarray.cxx index 918aa76dc7ae..5ad6c03c770d 100644 --- a/svx/source/dialog/framelinkarray.cxx +++ b/svx/source/dialog/framelinkarray.cxx @@ -24,6 +24,7 @@ #include <set> #include <unordered_set> #include <algorithm> +#include <svl/itempool.hxx> #include <tools/debug.hxx> #include <tools/gen.hxx> #include <vcl/canvastools.hxx> @@ -42,7 +43,7 @@ namespace svx::frame { namespace { -class Cell +class Cell final : public SfxPoolItem { private: Style maLeft; @@ -69,7 +70,11 @@ public: bool mbOverlapY; public: - explicit Cell(); + explicit Cell(); + explicit Cell(const Cell&) = default; + + virtual bool operator==( const SfxPoolItem& ) const override; + virtual Cell* Clone( SfxItemPool *pPool = nullptr ) const override; void SetStyleLeft(const Style& rStyle) { maLeft = rStyle; } void SetStyleRight(const Style& rStyle) { maRight = rStyle; } @@ -98,7 +103,7 @@ public: } -typedef std::vector< Cell > CellVec; +typedef std::vector< const Cell* > CellVec; basegfx::B2DHomMatrix Cell::HelperCreateB2DHomMatrixFromB2DRange( const basegfx::B2DRange& rRange ) const @@ -197,6 +202,7 @@ basegfx::B2DHomMatrix Cell::CreateCoordinateSystemMergedCell( } Cell::Cell() : + SfxPoolItem(10), mnAddLeft( 0 ), mnAddRight( 0 ), mnAddTop( 0 ), @@ -209,6 +215,32 @@ Cell::Cell() : { } +Cell* Cell::Clone(SfxItemPool* /*pPool*/) const +{ + return new Cell(*this); +} + +bool Cell::operator==(const SfxPoolItem& rItem) const +{ + if (!SfxPoolItem::operator==(rItem)) + return false; + const Cell& rOther = static_cast<const Cell&>(rItem); + return maLeft == rOther.maLeft + && maRight == rOther.maRight + && maTop == rOther.maTop + && maBottom == rOther.maBottom + && maTLBR == rOther.maTLBR + && maBLTR == rOther.maBLTR + && mnAddLeft == rOther.mnAddLeft + && mnAddRight == rOther.mnAddRight + && mnAddTop == rOther.mnAddTop + && mnAddBottom == rOther.mnAddBottom + && meRotMode == rOther.meRotMode + && mbMergeOrig == rOther.mbMergeOrig + && mbOverlapX == rOther.mbOverlapX + && mbOverlapY == rOther.mbOverlapY; +} + void Cell::MirrorSelfX() { std::swap( maLeft, maRight ); @@ -230,19 +262,23 @@ static void lclRecalcCoordVec( std::vector<sal_Int32>& rCoords, const std::vecto } } -static void lclSetMergedRange( CellVec& rCells, sal_Int32 nWidth, sal_Int32 nFirstCol, sal_Int32 nFirstRow, sal_Int32 nLastCol, sal_Int32 nLastRow ) +static void lclSetMergedRange( SfxItemPool& rPool, CellVec& rCells, sal_Int32 nWidth, sal_Int32 nFirstCol, sal_Int32 nFirstRow, sal_Int32 nLastCol, sal_Int32 nLastRow ) { for( sal_Int32 nCol = nFirstCol; nCol <= nLastCol; ++nCol ) { for( sal_Int32 nRow = nFirstRow; nRow <= nLastRow; ++nRow ) { - Cell& rCell = rCells[ nRow * nWidth + nCol ]; - rCell.mbMergeOrig = false; - rCell.mbOverlapX = nCol > nFirstCol; - rCell.mbOverlapY = nRow > nFirstRow; + const Cell* pCell = rCells[ nRow * nWidth + nCol ]; + Cell aTempCell(*pCell); + aTempCell.mbMergeOrig = false; + aTempCell.mbOverlapX = nCol > nFirstCol; + aTempCell.mbOverlapY = nRow > nFirstRow; + rCells[ nRow * nWidth + nCol ] = &rPool.Put(aTempCell); } } - rCells[ nFirstRow * nWidth + nFirstCol ].mbMergeOrig = true; + Cell aTempCell(*rCells[ nFirstRow * nWidth + nFirstCol ]); + aTempCell.mbMergeOrig = false; + rCells[ nFirstRow * nWidth + nFirstCol ] = &rPool.Put(aTempCell); } @@ -251,6 +287,9 @@ const Cell OBJ_CELL_NONE; struct ArrayImpl { + // used to reduce the memory consumption of cells + rtl::Reference<SfxItemPool> mxPool; + const Cell* mpDefaultCell; CellVec maCells; std::vector<sal_Int32> maWidths; std::vector<sal_Int32> maHeights; @@ -274,7 +313,7 @@ struct ArrayImpl { return nRow * mnWidth + nCol; } const Cell& GetCell( sal_Int32 nCol, sal_Int32 nRow ) const; - Cell& GetCellAcc( sal_Int32 nCol, sal_Int32 nRow ); + void PutCell( sal_Int32 nCol, sal_Int32 nRow, const Cell& ); sal_Int32 GetMergedFirstCol( sal_Int32 nCol, sal_Int32 nRow ) const; sal_Int32 GetMergedFirstRow( sal_Int32 nCol, sal_Int32 nRow ) const; @@ -303,7 +342,11 @@ struct ArrayImpl bool HasCellRotation() const; }; +const SfxItemInfo maItemInfos[] { + {0, true} +}; ArrayImpl::ArrayImpl( sal_Int32 nWidth, sal_Int32 nHeight ) : + mxPool(new SfxItemPool("Mine", 10, 10, maItemInfos)), mnWidth( nWidth ), mnHeight( nHeight ), mnFirstClipCol( 0 ), @@ -314,8 +357,9 @@ ArrayImpl::ArrayImpl( sal_Int32 nWidth, sal_Int32 nHeight ) : mbYCoordsDirty( false ), mbMayHaveCellRotation( false ) { + mpDefaultCell = &mxPool->Put(Cell()); // default-construct all vectors - maCells.resize( mnWidth * mnHeight ); + maCells.resize( mnWidth * mnHeight, mpDefaultCell ); maWidths.resize( mnWidth, 0 ); maHeights.resize( mnHeight, 0 ); maXCoords.resize( mnWidth + 1, 0 ); @@ -324,13 +368,13 @@ ArrayImpl::ArrayImpl( sal_Int32 nWidth, sal_Int32 nHeight ) : const Cell& ArrayImpl::GetCell( sal_Int32 nCol, sal_Int32 nRow ) const { - return IsValidPos( nCol, nRow ) ? maCells[ GetIndex( nCol, nRow ) ] : OBJ_CELL_NONE; + return IsValidPos( nCol, nRow ) ? *maCells[ GetIndex( nCol, nRow ) ] : OBJ_CELL_NONE; } -Cell& ArrayImpl::GetCellAcc( sal_Int32 nCol, sal_Int32 nRow ) +void ArrayImpl::PutCell( sal_Int32 nCol, sal_Int32 nRow, const Cell & rCell ) { - static Cell aDummy; - return IsValidPos( nCol, nRow ) ? maCells[ GetIndex( nCol, nRow ) ] : aDummy; + if (IsValidPos( nCol, nRow )) + maCells[ GetIndex( nCol, nRow ) ] = &mxPool->Put(rCell); } sal_Int32 ArrayImpl::GetMergedFirstCol( sal_Int32 nCol, sal_Int32 nRow ) const @@ -450,7 +494,7 @@ bool ArrayImpl::HasCellRotation() const // check cell array for (const auto& aCell : maCells) { - if (aCell.IsRotated()) + if (aCell->IsRotated()) { return true; } @@ -512,7 +556,7 @@ MergedCellIterator& MergedCellIterator::operator++() #define CELL( col, row ) mxImpl->GetCell( col, row ) -#define CELLACC( col, row ) mxImpl->GetCellAcc( col, row ) +#define PUTCELL( col, row, cell ) mxImpl->PutCell( col, row, cell ) #define ORIGCELL( col, row ) mxImpl->GetMergedOriginCell( col, row ) #define LASTCELL( col, row ) mxImpl->GetMergedLastCell( col, row ) @@ -559,45 +603,58 @@ sal_Int32 Array::GetCellIndex( sal_Int32 nCol, sal_Int32 nRow, bool bRTL ) const void Array::SetCellStyleLeft( sal_Int32 nCol, sal_Int32 nRow, const Style& rStyle ) { DBG_FRAME_CHECK_COLROW( nCol, nRow, "SetCellStyleLeft" ); - CELLACC( nCol, nRow ).SetStyleLeft(rStyle); + Cell aTempCell(CELL(nCol, nRow)); + aTempCell.SetStyleLeft(rStyle); + PUTCELL( nCol, nRow, aTempCell ); } void Array::SetCellStyleRight( sal_Int32 nCol, sal_Int32 nRow, const Style& rStyle ) { DBG_FRAME_CHECK_COLROW( nCol, nRow, "SetCellStyleRight" ); - CELLACC( nCol, nRow ).SetStyleRight(rStyle); + Cell aTempCell(CELL(nCol, nRow)); + aTempCell.SetStyleRight(rStyle); + PUTCELL( nCol, nRow, aTempCell ); } void Array::SetCellStyleTop( sal_Int32 nCol, sal_Int32 nRow, const Style& rStyle ) { DBG_FRAME_CHECK_COLROW( nCol, nRow, "SetCellStyleTop" ); - CELLACC( nCol, nRow ).SetStyleTop(rStyle); + Cell aTempCell(CELL(nCol, nRow)); + aTempCell.SetStyleTop(rStyle); + PUTCELL( nCol, nRow, aTempCell ); } void Array::SetCellStyleBottom( sal_Int32 nCol, sal_Int32 nRow, const Style& rStyle ) { DBG_FRAME_CHECK_COLROW( nCol, nRow, "SetCellStyleBottom" ); - CELLACC( nCol, nRow ).SetStyleBottom(rStyle); + Cell aTempCell(CELL(nCol, nRow)); + aTempCell.SetStyleBottom(rStyle); + PUTCELL( nCol, nRow, aTempCell ); } void Array::SetCellStyleTLBR( sal_Int32 nCol, sal_Int32 nRow, const Style& rStyle ) { DBG_FRAME_CHECK_COLROW( nCol, nRow, "SetCellStyleTLBR" ); - CELLACC( nCol, nRow ).SetStyleTLBR(rStyle); + Cell aTempCell(CELL(nCol, nRow)); + aTempCell.SetStyleTLBR(rStyle); + PUTCELL( nCol, nRow, aTempCell ); } void Array::SetCellStyleBLTR( sal_Int32 nCol, sal_Int32 nRow, const Style& rStyle ) { DBG_FRAME_CHECK_COLROW( nCol, nRow, "SetCellStyleBLTR" ); - CELLACC( nCol, nRow ).SetStyleBLTR(rStyle); + Cell aTempCell(CELL(nCol, nRow)); + aTempCell.SetStyleBLTR(rStyle); + PUTCELL( nCol, nRow, aTempCell ); } void Array::SetCellStyleDiag( sal_Int32 nCol, sal_Int32 nRow, const Style& rTLBR, const Style& rBLTR ) { DBG_FRAME_CHECK_COLROW( nCol, nRow, "SetCellStyleDiag" ); - Cell& rCell = CELLACC( nCol, nRow ); - rCell.SetStyleTLBR(rTLBR); - rCell.SetStyleBLTR(rBLTR); + Cell aTempCell(CELL(nCol, nRow)); + aTempCell.SetStyleTLBR(rTLBR); + aTempCell.SetStyleBLTR(rBLTR); + PUTCELL( nCol, nRow, aTempCell ); } void Array::SetColumnStyleLeft( sal_Int32 nCol, const Style& rStyle ) @@ -631,15 +688,16 @@ void Array::SetRowStyleBottom( sal_Int32 nRow, const Style& rStyle ) void Array::SetCellRotation(sal_Int32 nCol, sal_Int32 nRow, SvxRotateMode eRotMode, double fOrientation) { DBG_FRAME_CHECK_COLROW(nCol, nRow, "SetCellRotation"); - Cell& rTarget = CELLACC(nCol, nRow); - rTarget.meRotMode = eRotMode; - rTarget.mfOrientation = fOrientation; + Cell aTempCell(CELL(nCol, nRow)); + aTempCell.meRotMode = eRotMode; + aTempCell.mfOrientation = fOrientation; + PUTCELL( nCol, nRow, aTempCell ); if (!mxImpl->mbMayHaveCellRotation) { // activate once when a cell gets actually rotated to allow fast // answering HasCellRotation() calls - mxImpl->mbMayHaveCellRotation = rTarget.IsRotated(); + mxImpl->mbMayHaveCellRotation = aTempCell.IsRotated(); } } @@ -799,7 +857,7 @@ void Array::SetMergedRange( sal_Int32 nFirstCol, sal_Int32 nFirstRow, sal_Int32 } #endif if( mxImpl->IsValidPos( nFirstCol, nFirstRow ) && mxImpl->IsValidPos( nLastCol, nLastRow ) ) - lclSetMergedRange( mxImpl->maCells, mxImpl->mnWidth, nFirstCol, nFirstRow, nLastCol, nLastRow ); + lclSetMergedRange( *mxImpl->mxPool, mxImpl->maCells, mxImpl->mnWidth, nFirstCol, nFirstRow, nLastCol, nLastRow ); } void Array::SetAddMergedLeftSize( sal_Int32 nCol, sal_Int32 nRow, sal_Int32 nAddSize ) @@ -807,7 +865,11 @@ void Array::SetAddMergedLeftSize( sal_Int32 nCol, sal_Int32 nRow, sal_Int32 nAdd DBG_FRAME_CHECK_COLROW( nCol, nRow, "SetAddMergedLeftSize" ); DBG_FRAME_CHECK( mxImpl->GetMergedFirstCol( nCol, nRow ) == 0, "SetAddMergedLeftSize", "additional border inside array" ); for( MergedCellIterator aIt( *this, nCol, nRow ); aIt.Is(); ++aIt ) - CELLACC( aIt.Col(), aIt.Row() ).mnAddLeft = nAddSize; + { + Cell aTempCell(CELL(aIt.Col(), aIt.Row())); + aTempCell.mnAddLeft = nAddSize; + PUTCELL( nCol, nRow, aTempCell ); + } } void Array::SetAddMergedRightSize( sal_Int32 nCol, sal_Int32 nRow, sal_Int32 nAddSize ) @@ -815,7 +877,11 @@ void Array::SetAddMergedRightSize( sal_Int32 nCol, sal_Int32 nRow, sal_Int32 nAd DBG_FRAME_CHECK_COLROW( nCol, nRow, "SetAddMergedRightSize" ); DBG_FRAME_CHECK( mxImpl->GetMergedLastCol( nCol, nRow ) + 1 == mxImpl->mnWidth, "SetAddMergedRightSize", "additional border inside array" ); for( MergedCellIterator aIt( *this, nCol, nRow ); aIt.Is(); ++aIt ) - CELLACC( aIt.Col(), aIt.Row() ).mnAddRight = nAddSize; + { + Cell aTempCell(CELL(aIt.Col(), aIt.Row())); + aTempCell.mnAddRight = nAddSize; + PUTCELL( nCol, nRow, aTempCell ); + } } void Array::SetAddMergedTopSize( sal_Int32 nCol, sal_Int32 nRow, sal_Int32 nAddSize ) @@ -823,7 +889,11 @@ void Array::SetAddMergedTopSize( sal_Int32 nCol, sal_Int32 nRow, sal_Int32 nAddS DBG_FRAME_CHECK_COLROW( nCol, nRow, "SetAddMergedTopSize" ); DBG_FRAME_CHECK( mxImpl->GetMergedFirstRow( nCol, nRow ) == 0, "SetAddMergedTopSize", "additional border inside array" ); for( MergedCellIterator aIt( *this, nCol, nRow ); aIt.Is(); ++aIt ) - CELLACC( aIt.Col(), aIt.Row() ).mnAddTop = nAddSize; + { + Cell aTempCell(CELL(aIt.Col(), aIt.Row())); + aTempCell.mnAddTop = nAddSize; + PUTCELL( nCol, nRow, aTempCell ); + } } void Array::SetAddMergedBottomSize( sal_Int32 nCol, sal_Int32 nRow, sal_Int32 nAddSize ) @@ -831,7 +901,11 @@ void Array::SetAddMergedBottomSize( sal_Int32 nCol, sal_Int32 nRow, sal_Int32 nA DBG_FRAME_CHECK_COLROW( nCol, nRow, "SetAddMergedBottomSize" ); DBG_FRAME_CHECK( mxImpl->GetMergedLastRow( nCol, nRow ) + 1 == mxImpl->mnHeight, "SetAddMergedBottomSize", "additional border inside array" ); for( MergedCellIterator aIt( *this, nCol, nRow ); aIt.Is(); ++aIt ) - CELLACC( aIt.Col(), aIt.Row() ).mnAddBottom = nAddSize; + { + Cell aTempCell(CELL(aIt.Col(), aIt.Row())); + aTempCell.mnAddBottom = nAddSize; + PUTCELL( nCol, nRow, aTempCell ); + } } bool Array::IsMerged( sal_Int32 nCol, sal_Int32 nRow ) const @@ -992,8 +1066,9 @@ void Array::MirrorSelfX() { for( nCol = 0; nCol < mxImpl->mnWidth; ++nCol ) { - aNewCells.push_back( CELL( mxImpl->GetMirrorCol( nCol ), nRow ) ); - aNewCells.back().MirrorSelfX(); + Cell aTempCell(CELL(mxImpl->GetMirrorCol( nCol ), nRow)); + aTempCell.MirrorSelfX(); + aNewCells.push_back( &mxImpl->mxPool->Put(aTempCell) ); } } for( nRow = 0; nRow < mxImpl->mnHeight; ++nRow ) @@ -1004,7 +1079,7 @@ void Array::MirrorSelfX() { sal_Int32 nLastCol = mxImpl->GetMergedLastCol( nCol, nRow ); sal_Int32 nLastRow = mxImpl->GetMergedLastRow( nCol, nRow ); - lclSetMergedRange( aNewCells, mxImpl->mnWidth, + lclSetMergedRange( *mxImpl->mxPool, aNewCells, mxImpl->mnWidth, mxImpl->GetMirrorCol( nLastCol ), nRow, mxImpl->GetMirrorCol( nCol ), nLastRow ); } |