summaryrefslogtreecommitdiffstats
path: root/sw/source/core/table
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-07 09:06:44 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-07 09:06:44 +0000
commited5640d8b587fbcfed7dd7967f3de04b37a76f26 (patch)
tree7a5f7c6c9d02226d7471cb3cc8fbbf631b415303 /sw/source/core/table
parentInitial commit. (diff)
downloadlibreoffice-ed5640d8b587fbcfed7dd7967f3de04b37a76f26.tar.xz
libreoffice-ed5640d8b587fbcfed7dd7967f3de04b37a76f26.zip
Adding upstream version 4:7.4.7.upstream/4%7.4.7upstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'sw/source/core/table')
-rw-r--r--sw/source/core/table/swnewtable.cxx2512
-rw-r--r--sw/source/core/table/swtable.cxx2960
2 files changed, 5472 insertions, 0 deletions
diff --git a/sw/source/core/table/swnewtable.cxx b/sw/source/core/table/swnewtable.cxx
new file mode 100644
index 000000000..36607971f
--- /dev/null
+++ b/sw/source/core/table/swnewtable.cxx
@@ -0,0 +1,2512 @@
+/* -*- 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 <swtable.hxx>
+#include <tblsel.hxx>
+#include <tblrwcl.hxx>
+#include <ndtxt.hxx>
+#include <ndole.hxx>
+#include <node.hxx>
+#include <UndoTable.hxx>
+#include <pam.hxx>
+#include <frmfmt.hxx>
+#include <frmatr.hxx>
+#include <cellfrm.hxx>
+#include <fmtfsize.hxx>
+#include <doc.hxx>
+#include <IDocumentUndoRedo.hxx>
+#include <IDocumentContentOperations.hxx>
+#include <IDocumentFieldsAccess.hxx>
+#include <IDocumentLayoutAccess.hxx>
+#include <cstdlib>
+#include <vector>
+#include <set>
+#include <list>
+#include <memory>
+#include <editeng/boxitem.hxx>
+#include <editeng/protitem.hxx>
+#include <swtblfmt.hxx>
+#include <calbck.hxx>
+#include <sal/log.hxx>
+#include <osl/diagnose.h>
+
+#ifdef DBG_UTIL
+#define CHECK_TABLE(t) (t).CheckConsistency();
+#else
+#define CHECK_TABLE(t)
+#endif
+
+/** SwBoxSelection is a small helperclass (structure) to handle selections
+ of cells (boxes) between table functions
+
+ It contains an "array" of table boxes, a rectangulare selection of table boxes.
+ To be more specific, it contains a vector of box selections,
+ every box selection (SwSelBoxes) contains the selected boxes inside one row.
+ The member mnMergeWidth contains the width of the selected boxes
+*/
+
+class SwBoxSelection
+{
+public:
+ std::vector<SwSelBoxes> maBoxes;
+ tools::Long mnMergeWidth;
+ SwBoxSelection() : mnMergeWidth(0) {}
+ bool isEmpty() const { return maBoxes.empty(); }
+ void push_back(const SwSelBoxes& rNew) { maBoxes.push_back(rNew); }
+};
+
+/** NewMerge(..) removes the superfluous cells after cell merge
+
+SwTable::NewMerge(..) does some cleaning up,
+it simply deletes the superfluous cells ("cell span")
+and notifies the Undo about it.
+The main work has been done by SwTable::PrepareMerge(..) already.
+
+@param rBoxes
+the boxes to remove
+
+@param pUndo
+the undo object to notify, maybe empty
+
+@return true for compatibility reasons with OldMerge(..)
+*/
+
+bool SwTable::NewMerge( SwDoc* pDoc, const SwSelBoxes& rBoxes,
+ const SwSelBoxes& rMerged, SwUndoTableMerge* pUndo )
+{
+ if( pUndo )
+ pUndo->SetSelBoxes( rBoxes );
+ DeleteSel( pDoc, rBoxes, &rMerged, nullptr, true, true );
+
+ CHECK_TABLE( *this )
+ return true;
+}
+
+/** lcl_CheckMinMax helps evaluating (horizontal) min/max of boxes
+
+lcl_CheckMinMax(..) compares the left border and the right border
+of a given cell with the given range and sets it accordingly.
+
+@param rMin
+will be decremented if necessary to the left border of the cell
+
+@param rMax
+will be incremented if necessary to the right border of the cell
+
+@param rLine
+the row (table line) of the interesting box
+
+@param nCheck
+the index of the box in the table box array of the given row
+
+@param bSet
+if bSet is false, rMin and rMax will be manipulated if necessary
+if bSet is true, rMin and rMax will be set to the left and right border of the box
+
+*/
+
+static void lcl_CheckMinMax( tools::Long& rMin, tools::Long& rMax, const SwTableLine& rLine, size_t nCheck, bool bSet )
+{
+ ++nCheck;
+ if( rLine.GetTabBoxes().size() < nCheck )
+ { // robust
+ OSL_FAIL( "Box out of table line" );
+ nCheck = rLine.GetTabBoxes().size();
+ }
+
+ tools::Long nNew = 0; // will be the right border of the current box
+ tools::Long nWidth = 0; // the width of the current box
+ for( size_t nCurrBox = 0; nCurrBox < nCheck; ++nCurrBox )
+ {
+ SwTableBox* pBox = rLine.GetTabBoxes()[nCurrBox];
+ OSL_ENSURE( pBox, "Missing table box" );
+ nWidth = pBox->GetFrameFormat()->GetFrameSize().GetWidth();
+ nNew += nWidth;
+ }
+ // nNew is the right border of the wished box
+ if( bSet || nNew > rMax )
+ rMax = nNew;
+ nNew -= nWidth; // nNew becomes the left border of the wished box
+ if( bSet || nNew < rMin )
+ rMin = nNew;
+}
+
+/** lcl_Box2LeftBorder(..) delivers the left (logical) border of a table box
+
+The left logical border of a table box is the sum of the cell width before this
+box.
+
+@param rBox
+is the requested table box
+
+@return is the left logical border (long, even it cannot be negative)
+
+*/
+
+static tools::Long lcl_Box2LeftBorder( const SwTableBox& rBox )
+{
+ if( !rBox.GetUpper() )
+ return 0;
+ tools::Long nLeft = 0;
+ const SwTableLine &rLine = *rBox.GetUpper();
+ const size_t nCount = rLine.GetTabBoxes().size();
+ for( size_t nCurrBox = 0; nCurrBox < nCount; ++nCurrBox )
+ {
+ SwTableBox* pBox = rLine.GetTabBoxes()[nCurrBox];
+ OSL_ENSURE( pBox, "Missing table box" );
+ if( pBox == &rBox )
+ return nLeft;
+ nLeft += pBox->GetFrameFormat()->GetFrameSize().GetWidth();
+ }
+ OSL_FAIL( "Box not found in own upper?" );
+ return nLeft;
+}
+
+/** lcl_LeftBorder2Box delivers the box to a given left border
+
+It's used to find the master/follow table boxes in previous/next rows.
+Don't call this function to check if there is such a box,
+call it if you know there has to be such box.
+
+@param nLeft
+the left border (logical x-value) of the demanded box
+
+@param rLine
+the row (table line) to be scanned
+
+@return a pointer to the table box inside the given row with the wished left border
+
+*/
+
+static SwTableBox* lcl_LeftBorder2Box( tools::Long nLeft, const SwTableLine* pLine )
+{
+ if( !pLine )
+ return nullptr;
+ tools::Long nCurrLeft = 0;
+ const size_t nCount = pLine->GetTabBoxes().size();
+ for( size_t nCurrBox = 0; nCurrBox < nCount; ++nCurrBox )
+ {
+ SwTableBox* pBox = pLine->GetTabBoxes()[nCurrBox];
+ OSL_ENSURE( pBox, "Missing table box" );
+ if( pBox->GetFrameFormat()->GetFrameSize().GetWidth() )
+ {
+ if( nCurrLeft == nLeft )
+ return pBox;
+ // HACK: It appears that rounding errors may result in positions not matching
+ // exactly, so allow a little tolerance. This happens at least with merged cells
+ // in the doc from fdo#38414 .
+ if( std::abs( nCurrLeft - nLeft ) <= ( nLeft / 1000 ))
+ return pBox;
+ if( nCurrLeft >= nLeft )
+ {
+ SAL_WARN( "sw.core", "Possibly wrong box found" );
+ return pBox;
+ }
+ }
+ nCurrLeft += pBox->GetFrameFormat()->GetFrameSize().GetWidth();
+ }
+ OSL_FAIL( "Didn't find wished box" );
+ return nullptr;
+}
+
+/** lcl_ChangeRowSpan corrects row span after insertion/deletion of rows
+
+lcl_ChangeRowSpan(..) has to be called after an insertion or deletion of rows
+to adjust the row spans of previous rows accordingly.
+If rows are deleted, the previous rows with row spans into the deleted area
+have to be decremented by the number of _overlapped_ inserted rows.
+If rows are inserted, the previous rows with row span into the inserted area
+have to be incremented by the number of inserted rows.
+For those row spans which ends exactly above the inserted area it has to be
+decided by the parameter bSingle if they have to be expanded or not.
+
+@param rTable
+the table to manipulate (has to be a new model table)
+
+@param nDiff
+the number of rows which has been inserted (nDiff > 0) or deleted (nDiff < 0)
+
+@param nRowIdx
+the index of the first row which has to be checked
+
+@param bSingle
+true if the new inserted row should not extend row spans which ends in the row above
+this is for rows inserted by UI "insert row"
+false if all cells of an inserted row has to be overlapped by the previous row
+this is for rows inserted by "split row"
+false is also needed for deleted rows
+
+*/
+
+static void lcl_ChangeRowSpan( const SwTable& rTable, const tools::Long nDiff,
+ sal_uInt16 nRowIdx, const bool bSingle )
+{
+ if( !nDiff || nRowIdx >= rTable.GetTabLines().size() )
+ return;
+ OSL_ENSURE( !bSingle || nDiff > 0, "Don't set bSingle when deleting lines!" );
+ bool bGoOn;
+ // nDistance is the distance between the current row and the critical row,
+ // e.g. the deleted rows or the inserted rows.
+ // If the row span is lower than the distance there is nothing to do
+ // because the row span ends before the critical area.
+ // When the inserted rows should not be overlapped by row spans which ends
+ // exactly in the row above, the trick is to start with a distance of 1.
+ tools::Long nDistance = bSingle ? 1 : 0;
+ do
+ {
+ bGoOn = false; // will be set to true if we found a non-master cell
+ // which has to be manipulated => we have to check the previous row, too.
+ const SwTableLine* pLine = rTable.GetTabLines()[ nRowIdx ];
+ const size_t nBoxCount = pLine->GetTabBoxes().size();
+ for( size_t nCurrBox = 0; nCurrBox < nBoxCount; ++nCurrBox )
+ {
+ sal_Int32 nRowSpan = pLine->GetTabBoxes()[nCurrBox]->getRowSpan();
+ sal_Int32 nAbsSpan = nRowSpan > 0 ? nRowSpan : -nRowSpan;
+ // Check if the last overlapped cell is above or below
+ // the critical area
+ if( nAbsSpan > nDistance )
+ {
+ if( nDiff > 0 )
+ {
+ if( nRowSpan > 0 )
+ nRowSpan += nDiff; // increment row span of master cell
+ else
+ {
+ nRowSpan -= nDiff; // increment row span of non-master cell
+ bGoOn = true;
+ }
+ }
+ else
+ {
+ if( nRowSpan > 0 )
+ { // A master cell
+ // end of row span behind the deleted area ..
+ if( nRowSpan - nDistance > -nDiff )
+ nRowSpan += nDiff;
+ else // .. or inside the deleted area
+ nRowSpan = nDistance + 1;
+ }
+ else
+ { // Same for a non-master cell
+ if( nRowSpan + nDistance < nDiff )
+ nRowSpan -= nDiff;
+ else
+ nRowSpan = -nDistance - 1;
+ bGoOn = true; // We have to continue
+ }
+ }
+ pLine->GetTabBoxes()[ nCurrBox ]->setRowSpan( nRowSpan );
+ }
+ }
+ ++nDistance;
+ if( nRowIdx )
+ --nRowIdx;
+ else
+ bGoOn = false; //robust
+ } while( bGoOn );
+}
+
+/** CollectBoxSelection(..) create a rectangulare selection based on the given SwPaM
+ and prepares the selected cells for merging
+*/
+
+std::unique_ptr<SwBoxSelection> SwTable::CollectBoxSelection( const SwPaM& rPam ) const
+{
+ OSL_ENSURE( m_bNewModel, "Don't call me for old tables" );
+ if( m_aLines.empty() )
+ return nullptr;
+ const SwNode* pStartNd = rPam.Start()->nNode.GetNode().FindTableBoxStartNode();
+ const SwNode* pEndNd = rPam.End()->nNode.GetNode().FindTableBoxStartNode();
+ if( !pStartNd || !pEndNd || pStartNd == pEndNd )
+ return nullptr;
+
+ const size_t nLines = m_aLines.size();
+ size_t nTop = 0;
+ size_t nBottom = 0;
+ tools::Long nMin = 0, nMax = 0;
+ int nFound = 0;
+ for( size_t nRow = 0; nFound < 2 && nRow < nLines; ++nRow )
+ {
+ SwTableLine* pLine = m_aLines[nRow];
+ OSL_ENSURE( pLine, "Missing table line" );
+ const size_t nCols = pLine->GetTabBoxes().size();
+ for( size_t nCol = 0; nCol < nCols; ++nCol )
+ {
+ SwTableBox* pBox = pLine->GetTabBoxes()[nCol];
+ OSL_ENSURE( pBox, "Missing table box" );
+ if( nFound )
+ {
+ if( pBox->GetSttNd() == pEndNd )
+ {
+ nBottom = nRow;
+ lcl_CheckMinMax( nMin, nMax, *pLine, nCol, false );
+ ++nFound;
+ break;
+ }
+ }
+ else if( pBox->GetSttNd() == pStartNd )
+ {
+ nTop = nRow;
+ lcl_CheckMinMax( nMin, nMax, *pLine, nCol, true );
+ ++nFound;
+ }
+ }
+ }
+ if( nFound < 2 )
+ return nullptr;
+
+ bool bOkay = true;
+ tools::Long nMid = ( nMin + nMax ) / 2;
+
+ auto pRet(std::make_unique<SwBoxSelection>());
+ std::vector< std::pair< SwTableBox*, tools::Long > > aNewWidthVector;
+ size_t nCheckBottom = nBottom;
+ tools::Long nLeftSpan = 0;
+ tools::Long nRightSpan = 0;
+ tools::Long nLeftSpanCnt = 0;
+ tools::Long nRightSpanCnt = 0;
+ for( size_t nRow = nTop; nRow <= nBottom && bOkay && nRow < nLines; ++nRow )
+ {
+ SwTableLine* pLine = m_aLines[nRow];
+ OSL_ENSURE( pLine, "Missing table line" );
+ SwSelBoxes aBoxes;
+ tools::Long nRight = 0;
+ const size_t nCount = pLine->GetTabBoxes().size();
+ for( size_t nCurrBox = 0; nCurrBox < nCount; ++nCurrBox )
+ {
+ SwTableBox* pBox = pLine->GetTabBoxes()[nCurrBox];
+ OSL_ENSURE( pBox, "Missing table box" );
+ tools::Long nLeft = nRight;
+ nRight += pBox->GetFrameFormat()->GetFrameSize().GetWidth();
+ sal_Int32 nRowSpan = pBox->getRowSpan();
+ if( nRight <= nMin )
+ {
+ if( nRight == nMin && nLeftSpanCnt )
+ bOkay = false;
+ continue;
+ }
+ SwTableBox* pInnerBox = nullptr;
+ SwTableBox* pLeftBox = nullptr;
+ SwTableBox* pRightBox = nullptr;
+ tools::Long nDiff = 0;
+ tools::Long nDiff2 = 0;
+ if( nLeft < nMin )
+ {
+ if( nRight >= nMid || nRight + nLeft >= nMin + nMin )
+ {
+ if( nCurrBox )
+ {
+ aBoxes.insert(pBox);
+ pInnerBox = pBox;
+ pLeftBox = pLine->GetTabBoxes()[nCurrBox-1];
+ nDiff = nMin - nLeft;
+ if( nRight > nMax )
+ {
+ if( nCurrBox+1 < nCount )
+ {
+ pRightBox = pLine->GetTabBoxes()[nCurrBox+1];
+ nDiff2 = nRight - nMax;
+ }
+ else
+ bOkay = false;
+ }
+ else if( nRightSpanCnt && nRight == nMax )
+ bOkay = false;
+ }
+ else
+ bOkay = false;
+ }
+ else if( nCurrBox+1 < nCount )
+ {
+ pLeftBox = pBox;
+ pInnerBox = pLine->GetTabBoxes()[nCurrBox+1];
+ nDiff = nMin - nRight;
+ }
+ else
+ bOkay = false;
+ }
+ else if( nRight <= nMax )
+ {
+ aBoxes.insert(pBox);
+ if( nRow == nTop && nRowSpan < 0 )
+ {
+ bOkay = false;
+ break;
+ }
+ if( nRowSpan > 1 && nRow + nRowSpan - 1 > nBottom )
+ nBottom = nRow + nRowSpan - 1;
+ if( nRowSpan < -1 && nRow - nRowSpan - 1 > nBottom )
+ nBottom = nRow - nRowSpan - 1;
+ if( nRightSpanCnt && nRight == nMax )
+ bOkay = false;
+ }
+ else if( nLeft < nMax )
+ {
+ if( nLeft <= nMid || nRight + nLeft <= nMax )
+ {
+ if( nCurrBox+1 < nCount )
+ {
+ aBoxes.insert(pBox);
+ pInnerBox = pBox;
+ pRightBox = pLine->GetTabBoxes()[nCurrBox+1];
+ nDiff = nRight - nMax;
+ }
+ else
+ bOkay = false;
+ }
+ else if( nCurrBox )
+ {
+ pRightBox = pBox;
+ pInnerBox = pLine->GetTabBoxes()[nCurrBox-1];
+ nDiff = nLeft - nMax;
+ }
+ else
+ bOkay = false;
+ }
+ else
+ break;
+ if( pInnerBox )
+ {
+ if( nRow == nBottom )
+ {
+ tools::Long nTmpSpan = pInnerBox->getRowSpan();
+ if( nTmpSpan > 1 )
+ nBottom += nTmpSpan - 1;
+ else if( nTmpSpan < -1 )
+ nBottom -= nTmpSpan + 1;
+ }
+ SwTableBox* pOuterBox = pLeftBox;
+ do
+ {
+ if( pOuterBox )
+ {
+ tools::Long nOutSpan = pOuterBox->getRowSpan();
+ if( nOutSpan != 1 )
+ {
+ size_t nCheck = nRow;
+ if( nOutSpan < 0 )
+ {
+ const SwTableBox& rBox =
+ pOuterBox->FindStartOfRowSpan( *this );
+ nOutSpan = rBox.getRowSpan();
+ const SwTableLine* pTmpL = rBox.GetUpper();
+ nCheck = GetTabLines().GetPos( pTmpL );
+ if( nCheck < nTop )
+ bOkay = false;
+ if( pOuterBox == pLeftBox )
+ {
+ if( !nLeftSpanCnt || nMin - nDiff != nLeftSpan )
+ bOkay = false;
+ }
+ else
+ {
+ if( !nRightSpanCnt || nMax + nDiff != nRightSpan )
+ bOkay = false;
+ }
+ }
+ else
+ {
+ if( pOuterBox == pLeftBox )
+ {
+ if( nLeftSpanCnt )
+ bOkay = false;
+ nLeftSpan = nMin - nDiff;
+ nLeftSpanCnt = nOutSpan;
+ }
+ else
+ {
+ if( nRightSpanCnt )
+ bOkay = false;
+ nRightSpan = nMax + nDiff;
+ nRightSpanCnt = nOutSpan;
+ }
+ }
+ nCheck += nOutSpan - 1;
+ if( nCheck > nCheckBottom )
+ nCheckBottom = nCheck;
+ }
+ else if( ( nLeftSpanCnt && pLeftBox == pOuterBox ) ||
+ ( nRightSpanCnt && pRightBox == pOuterBox ) )
+ bOkay = false;
+ std::pair< SwTableBox*, long > aTmp;
+ aTmp.first = pInnerBox;
+ aTmp.second = -nDiff;
+ aNewWidthVector.push_back(aTmp);
+ aTmp.first = pOuterBox;
+ aTmp.second = nDiff;
+ aNewWidthVector.push_back(aTmp);
+ }
+ pOuterBox = pOuterBox == pRightBox ? nullptr : pRightBox;
+ if( nDiff2 )
+ nDiff = nDiff2;
+ } while( pOuterBox );
+ }
+ }
+ if( nLeftSpanCnt )
+ --nLeftSpanCnt;
+ if( nRightSpanCnt )
+ --nRightSpanCnt;
+ pRet->push_back(aBoxes);
+ }
+ if( nCheckBottom > nBottom )
+ bOkay = false;
+ if( bOkay )
+ {
+ pRet->mnMergeWidth = nMax - nMin;
+ for (auto const& newWidth : aNewWidthVector)
+ {
+ SwFrameFormat* pFormat = newWidth.first->ClaimFrameFormat();
+ tools::Long nNewWidth = pFormat->GetFrameSize().GetWidth() + newWidth.second;
+ pFormat->SetFormatAttr( SwFormatFrameSize( SwFrameSize::Variable, nNewWidth, 0 ) );
+ }
+ }
+ else
+ pRet.reset();
+
+ return pRet;
+}
+
+/** lcl_InvalidateCellFrame(..) invalidates all layout representations of a given cell
+ to initiate a reformatting
+*/
+
+static void lcl_InvalidateCellFrame( const SwTableBox& rBox )
+{
+ SwIterator<SwCellFrame,SwFormat> aIter( *rBox.GetFrameFormat() );
+ for( SwCellFrame* pCell = aIter.First(); pCell; pCell = aIter.Next() )
+ {
+ if( pCell->GetTabBox() == &rBox )
+ {
+ pCell->InvalidateSize();
+ SwFrame* pLower = pCell->GetLower();
+ if( pLower )
+ pLower->InvalidateSize_();
+ }
+ }
+}
+
+/** lcl_InsertPosition(..) evaluates the insert positions in every table line,
+ when a selection of cells is given and returns the average cell widths
+*/
+
+static tools::Long lcl_InsertPosition( SwTable &rTable, std::vector<sal_uInt16>& rInsPos,
+ const SwSelBoxes& rBoxes, bool bBehind )
+{
+ sal_Int32 nAddWidth = 0;
+ tools::Long nCount = 0;
+ for (size_t j = 0; j < rBoxes.size(); ++j)
+ {
+ SwTableBox *pBox = rBoxes[j];
+ SwTableLine* pLine = pBox->GetUpper();
+ tools::Long nWidth = rBoxes[j]->GetFrameFormat()->GetFrameSize().GetWidth();
+ nAddWidth += nWidth;
+ sal_uInt16 nCurrBox = pLine->GetBoxPos( pBox );
+ sal_uInt16 nCurrLine = rTable.GetTabLines().GetPos( pLine );
+ OSL_ENSURE( nCurrLine != USHRT_MAX, "Time to say Good-Bye.." );
+ if( rInsPos[ nCurrLine ] == USHRT_MAX )
+ {
+ rInsPos[ nCurrLine ] = nCurrBox;
+ ++nCount;
+ }
+ else if( ( rInsPos[ nCurrLine ] > nCurrBox ) == !bBehind )
+ rInsPos[ nCurrLine ] = nCurrBox;
+ }
+ if( nCount )
+ nAddWidth /= nCount;
+ return nAddWidth;
+}
+
+/** SwTable::NewInsertCol(..) insert new column(s) into a table
+
+@param pDoc
+the document
+
+@param rBoxes
+the selected boxes
+
+@param nCnt
+the number of columns to insert
+
+@param bBehind
+insertion behind (true) or before (false) the selected boxes
+
+@return true, if any insertion has been done successfully
+
+*/
+
+bool SwTable::NewInsertCol( SwDoc& rDoc, const SwSelBoxes& rBoxes,
+ sal_uInt16 nCnt, bool bBehind )
+{
+ if( m_aLines.empty() || !nCnt )
+ return false;
+
+ CHECK_TABLE( *this )
+ tools::Long nNewBoxWidth = 0;
+ std::vector< sal_uInt16 > aInsPos( m_aLines.size(), USHRT_MAX );
+ { // Calculation of the insert positions and the width of the new boxes
+ sal_uInt64 nTableWidth = 0;
+ for( size_t i = 0; i < m_aLines[0]->GetTabBoxes().size(); ++i )
+ nTableWidth += m_aLines[0]->GetTabBoxes()[i]->GetFrameFormat()->GetFrameSize().GetWidth();
+
+ // Fill the vector of insert positions and the (average) width to insert
+ sal_uInt64 nAddWidth = lcl_InsertPosition( *this, aInsPos, rBoxes, bBehind );
+
+ // Given is the (average) width of the selected boxes, if we would
+ // insert nCnt of columns the table would grow
+ // So we will shrink the table first, then insert the new boxes and
+ // get a table with the same width than before.
+ // But we will not shrink the table by the full already calculated value,
+ // we will reduce this value proportional to the old table width
+ nAddWidth *= nCnt; // we have to insert nCnt boxes per line
+ sal_uInt64 nResultingWidth = nAddWidth + nTableWidth;
+ if( !nResultingWidth )
+ return false;
+ nAddWidth = (nAddWidth * nTableWidth) / nResultingWidth;
+ nNewBoxWidth = tools::Long( nAddWidth / nCnt ); // Rounding
+ nAddWidth = nNewBoxWidth * nCnt; // Rounding
+ if( !nAddWidth || nAddWidth >= nTableWidth )
+ return false;
+ AdjustWidths( static_cast< tools::Long >(nTableWidth), static_cast< tools::Long >(nTableWidth - nAddWidth) );
+ }
+
+ FndBox_ aFndBox( nullptr, nullptr );
+ aFndBox.SetTableLines( rBoxes, *this );
+ aFndBox.DelFrames( *this );
+
+ SwTableNode* pTableNd = GetTableNode();
+ std::vector<SwTableBoxFormat*> aInsFormat( nCnt, nullptr );
+ size_t nLastLine = SAL_MAX_SIZE;
+ sal_Int32 nLastRowSpan = 1;
+
+ for( size_t i = 0; i < m_aLines.size(); ++i )
+ {
+ SwTableLine* pLine = m_aLines[ i ];
+ sal_uInt16 nInsPos = aInsPos[i];
+ assert(nInsPos != USHRT_MAX); // didn't find insert position
+ SwTableBox* pBox = pLine->GetTabBoxes()[ nInsPos ];
+ if( bBehind )
+ ++nInsPos;
+ SwTableBoxFormat* pBoxFrameFormat = static_cast<SwTableBoxFormat*>(pBox->GetFrameFormat());
+ ::InsTableBox( rDoc, pTableNd, pLine, pBoxFrameFormat, pBox, nInsPos, nCnt );
+ sal_Int32 nRowSpan = pBox->getRowSpan();
+ tools::Long nDiff = i - nLastLine;
+ bool bNewSpan = false;
+ if( nLastLine != SAL_MAX_SIZE && nDiff <= nLastRowSpan &&
+ nRowSpan != nDiff - nLastRowSpan )
+ {
+ bNewSpan = true;
+ while( nLastLine < i )
+ {
+ SwTableLine* pTmpLine = m_aLines[ nLastLine ];
+ sal_uInt16 nTmpPos = aInsPos[nLastLine];
+ if( bBehind )
+ ++nTmpPos;
+ for( sal_uInt16 j = 0; j < nCnt; ++j )
+ pTmpLine->GetTabBoxes()[nTmpPos+j]->setRowSpan( nDiff );
+ if( nDiff > 0 )
+ nDiff = -nDiff;
+ ++nDiff;
+ ++nLastLine;
+ }
+ }
+ if( nRowSpan > 0 )
+ bNewSpan = true;
+ if( bNewSpan )
+ {
+ nLastLine = i;
+ if( nRowSpan < 0 )
+ nLastRowSpan = -nRowSpan;
+ else
+ nLastRowSpan = nRowSpan;
+ }
+ const SvxBoxItem& aSelBoxItem = pBoxFrameFormat->GetBox();
+ std::unique_ptr<SvxBoxItem> pNoRightBorder;
+ if( aSelBoxItem.GetRight() )
+ {
+ pNoRightBorder.reset( new SvxBoxItem( aSelBoxItem ));
+ pNoRightBorder->SetLine( nullptr, SvxBoxItemLine::RIGHT );
+ }
+ for( sal_uInt16 j = 0; j < nCnt; ++j )
+ {
+ SwTableBox *pCurrBox = pLine->GetTabBoxes()[nInsPos+j];
+ if( bNewSpan )
+ {
+ pCurrBox->setRowSpan( nLastRowSpan );
+ SwFrameFormat* pFrameFormat = pCurrBox->ClaimFrameFormat();
+ SwFormatFrameSize aFrameSz( pFrameFormat->GetFrameSize() );
+ aFrameSz.SetWidth( nNewBoxWidth );
+ pFrameFormat->SetFormatAttr( aFrameSz );
+ if( pNoRightBorder && ( !bBehind || j+1 < nCnt ) )
+ pFrameFormat->SetFormatAttr( *pNoRightBorder );
+ aInsFormat[j] = static_cast<SwTableBoxFormat*>(pFrameFormat);
+ }
+ else
+ pCurrBox->ChgFrameFormat( aInsFormat[j] );
+ }
+ if( bBehind && pNoRightBorder )
+ {
+ SwFrameFormat* pFrameFormat = pBox->ClaimFrameFormat();
+ pFrameFormat->SetFormatAttr( *pNoRightBorder );
+ }
+ }
+
+ aFndBox.MakeFrames( *this );
+#if OSL_DEBUG_LEVEL > 0
+ {
+ const SwTableBoxes &rTabBoxes = m_aLines[0]->GetTabBoxes();
+ tools::Long nNewWidth = 0;
+ for( size_t i = 0; i < rTabBoxes.size(); ++i )
+ nNewWidth += rTabBoxes[i]->GetFrameFormat()->GetFrameSize().GetWidth();
+ OSL_ENSURE( nNewWidth > 0, "Very small" );
+ }
+#endif
+ CHECK_TABLE( *this )
+
+ return true;
+}
+
+/** SwTable::PrepareMerge(..) some preparation for the coming Merge(..)
+
+For the old table model, ::GetMergeSel(..) is called only,
+for the new table model, PrepareMerge does the main work.
+It modifies all cells to merge (width, border, rowspan etc.) and collects
+the cells which have to be deleted by Merge(..) afterwards.
+If there are superfluous rows, these cells are put into the deletion list as well.
+
+@param rPam
+the selection to merge
+
+@param rBoxes
+should be empty at the beginning, at the end it is filled with boxes to delete.
+
+@param ppMergeBox
+will be set to the master cell box
+
+@param pUndo
+the undo object to record all changes
+can be Null, e.g. when called by Redo(..)
+
+@return
+
+*/
+
+bool SwTable::PrepareMerge( const SwPaM& rPam, SwSelBoxes& rBoxes,
+ SwSelBoxes& rMerged, SwTableBox** ppMergeBox, SwUndoTableMerge* pUndo )
+{
+ if( !m_bNewModel )
+ {
+ ::GetMergeSel( rPam, rBoxes, ppMergeBox, pUndo );
+ return rBoxes.size() > 1;
+ }
+ CHECK_TABLE( *this )
+ // We have to assert a "rectangular" box selection before we start to merge
+ std::unique_ptr< SwBoxSelection > pSel( CollectBoxSelection( rPam ) );
+ if (!pSel || pSel->isEmpty())
+ return false;
+ // Now we should have a rectangle of boxes,
+ // i.e. contiguous cells in contiguous rows
+ bool bMerge = false; // will be set if any content is transferred from
+ // a "not already overlapped" cell into the new master cell.
+ const SwSelBoxes& rFirstBoxes = pSel->maBoxes[0];
+ if (rFirstBoxes.empty())
+ return false;
+ SwTableBox *pMergeBox = rFirstBoxes[0]; // the master cell box
+ if( !pMergeBox )
+ return false;
+ (*ppMergeBox) = pMergeBox;
+ // The new master box will get the left and the top border of the top-left
+ // box of the selection and because the new master cell _is_ the top-left
+ // box, the left and right border does not need to be changed.
+ // The right and bottom border instead has to be derived from the right-
+ // bottom box of the selection. If this is an overlapped cell,
+ // the appropriate master box.
+ SwTableBox* pLastBox = nullptr; // the right-bottom (master) cell
+ SwDoc* pDoc = GetFrameFormat()->GetDoc();
+ SwPosition aInsPos( *pMergeBox->GetSttNd()->EndOfSectionNode() );
+ SwPaM aChkPam( aInsPos );
+ // The number of lines in the selection rectangle: nLineCount
+ const size_t nLineCount = pSel->maBoxes.size();
+ // BTW: nLineCount is the rowspan of the new master cell
+ sal_Int32 nRowSpan = static_cast<tools::Long>(nLineCount);
+ // We will need the first and last line of the selection
+ // to check if there any superfluous row after merging
+ SwTableLine* pFirstLn = nullptr;
+ SwTableLine* pLastLn = nullptr;
+ // Iteration over the lines of the selection...
+ for( size_t nCurrLine = 0; nCurrLine < nLineCount; ++nCurrLine )
+ {
+ // The selected boxes in the current line
+ const SwSelBoxes& rLineBoxes = pSel->maBoxes[nCurrLine];
+ size_t nColCount = rLineBoxes.size();
+ // Iteration over the selected cell in the current row
+ for (size_t nCurrCol = 0; nCurrCol < nColCount; ++nCurrCol)
+ {
+ SwTableBox* pBox = rLineBoxes[nCurrCol];
+ rMerged.insert( pBox );
+ // Only the first selected cell in every row will be alive,
+ // the other will be deleted => put into rBoxes
+ if( nCurrCol )
+ rBoxes.insert( pBox );
+ else
+ {
+ if( nCurrLine == 1 )
+ pFirstLn = pBox->GetUpper(); // we need this line later on
+ if( nCurrLine + 1 == nLineCount )
+ pLastLn = pBox->GetUpper(); // and this one, too.
+ }
+ // A box has to be merged if it's not the master box itself,
+ // but an already overlapped cell must not be merged as well.
+ bool bDoMerge = pBox != pMergeBox && pBox->getRowSpan() > 0;
+ // The last box has to be in the last "column" of the selection
+ // and it has to be a master cell
+ if( nCurrCol+1 == nColCount && pBox->getRowSpan() > 0 )
+ pLastBox = pBox;
+ if( bDoMerge )
+ {
+ bMerge = true;
+ // If the cell to merge contains only one empty paragraph,
+ // we do not transfer this paragraph.
+ if( !IsEmptyBox( *pBox, aChkPam ) )
+ {
+ SwNodeIndex& rInsPosNd = aInsPos.nNode;
+ SwPaM aPam( aInsPos );
+ aPam.GetPoint()->nNode.Assign( *pBox->GetSttNd()->EndOfSectionNode(), -1 );
+ SwContentNode* pCNd = aPam.GetContentNode();
+ aPam.GetPoint()->nContent.Assign( pCNd, pCNd ? pCNd->Len() : 0 );
+ SwNodeIndex aSttNdIdx( *pBox->GetSttNd(), 1 );
+ bool const bUndo = pDoc->GetIDocumentUndoRedo().DoesUndo();
+ if( pUndo )
+ {
+ pDoc->GetIDocumentUndoRedo().DoUndo(false);
+ }
+ pDoc->getIDocumentContentOperations().AppendTextNode( *aPam.GetPoint() );
+ if( pUndo )
+ {
+ pDoc->GetIDocumentUndoRedo().DoUndo(bUndo);
+ }
+ SwNodeRange aRg( aSttNdIdx, aPam.GetPoint()->nNode );
+ if( pUndo )
+ pUndo->MoveBoxContent( *pDoc, aRg, rInsPosNd );
+ else
+ {
+ pDoc->getIDocumentContentOperations().MoveNodeRange( aRg, rInsPosNd,
+ SwMoveFlags::NO_DELFRMS );
+ }
+ }
+ }
+ // Only the cell of the first selected column will stay alive
+ // and got a new row span
+ if( !nCurrCol )
+ pBox->setRowSpan( nRowSpan );
+ }
+ if( nRowSpan > 0 ) // the master cell is done, from now on we set
+ nRowSpan = -nRowSpan; // negative row spans
+ ++nRowSpan; // ... -3, -2, -1
+ }
+ if( bMerge )
+ {
+ // A row containing overlapped cells is superfluous,
+ // these cells can be put into rBoxes for deletion
+ FindSuperfluousRows_( rBoxes, pFirstLn, pLastLn );
+ // pNewFormat will be set to the new master box and the overlapped cells
+ SwFrameFormat* pNewFormat = pMergeBox->ClaimFrameFormat();
+ pNewFormat->SetFormatAttr( SwFormatFrameSize( SwFrameSize::Variable, pSel->mnMergeWidth, 0 ) );
+ for( size_t nCurrLine = 0; nCurrLine < nLineCount; ++nCurrLine )
+ {
+ const SwSelBoxes& rLineBoxes = pSel->maBoxes[nCurrLine];
+ size_t nColCount = rLineBoxes.size();
+ for (size_t nCurrCol = 0; nCurrCol < nColCount; ++nCurrCol)
+ {
+ SwTableBox* pBox = rLineBoxes[nCurrCol];
+ if( nCurrCol )
+ {
+ // Even this box will be deleted soon,
+ // we have to correct the width to avoid side effects
+ SwFrameFormat* pFormat = pBox->ClaimFrameFormat();
+ pFormat->SetFormatAttr( SwFormatFrameSize( SwFrameSize::Variable, 0, 0 ) );
+ }
+ else
+ {
+ pBox->ChgFrameFormat( static_cast<SwTableBoxFormat*>(pNewFormat) );
+ // remove numbering from cells that will be disabled in the merge
+ if( nCurrLine )
+ {
+ SwPaM aPam( *pBox->GetSttNd(), 0 );
+ aPam.GetPoint()->nNode++;
+ SwTextNode* pNd = aPam.GetNode().GetTextNode();
+ while( pNd )
+ {
+ pNd->SetCountedInList( false );
+
+ aPam.GetPoint()->nNode++;
+ pNd = aPam.GetNode().GetTextNode();
+ }
+ }
+ }
+ }
+ }
+ if( pLastBox ) // Robust
+ {
+ // The new borders of the master cell...
+ SvxBoxItem aBox( pMergeBox->GetFrameFormat()->GetBox() );
+ bool bOld = aBox.GetRight() || aBox.GetBottom();
+ const SvxBoxItem& rBox = pLastBox->GetFrameFormat()->GetBox();
+ aBox.SetLine( rBox.GetRight(), SvxBoxItemLine::RIGHT );
+ aBox.SetLine( rBox.GetBottom(), SvxBoxItemLine::BOTTOM );
+ if( bOld || aBox.GetLeft() || aBox.GetTop() || aBox.GetRight() || aBox.GetBottom() )
+ (*ppMergeBox)->GetFrameFormat()->SetFormatAttr( aBox );
+ }
+
+ if( pUndo )
+ pUndo->AddNewBox( pMergeBox->GetSttIdx() );
+ }
+ return bMerge;
+}
+
+/** SwTable::FindSuperfluousRows_(..) is looking for superfluous rows, i.e. rows
+ containing overlapped cells only.
+*/
+
+void SwTable::FindSuperfluousRows_( SwSelBoxes& rBoxes,
+ SwTableLine* pFirstLn, SwTableLine* pLastLn )
+{
+ if( !pFirstLn || !pLastLn )
+ {
+ if( rBoxes.empty() )
+ return;
+ pFirstLn = rBoxes[0]->GetUpper();
+ pLastLn = rBoxes.back()->GetUpper();
+ }
+ sal_uInt16 nFirstLn = GetTabLines().GetPos( pFirstLn );
+ sal_uInt16 nLastLn = GetTabLines().GetPos( pLastLn );
+ for( sal_uInt16 nRow = nFirstLn; nRow <= nLastLn; ++nRow )
+ {
+ SwTableLine* pLine = m_aLines[nRow];
+ OSL_ENSURE( pLine, "Missing table line" );
+ const size_t nCols = pLine->GetTabBoxes().size();
+ bool bSuperfl = true;
+ for( size_t nCol = 0; nCol < nCols; ++nCol )
+ {
+ SwTableBox *pBox = pLine->GetTabBoxes()[nCol];
+ if( pBox->getRowSpan() > 0 &&
+ rBoxes.end() == rBoxes.find( pBox ) )
+ {
+ bSuperfl = false;
+ break;
+ }
+ }
+ if( bSuperfl )
+ {
+ for( size_t nCol = 0; nCol < nCols; ++nCol )
+ {
+ SwTableBox* pBox = pLine->GetTabBoxes()[nCol];
+ rBoxes.insert( pBox );
+ }
+ }
+ }
+}
+
+/** SwTableBox::FindStartOfRowSpan(..) returns the "master" cell, the cell which
+ overlaps the given cell, it maybe the cell itself.
+*/
+
+SwTableBox& SwTableBox::FindStartOfRowSpan( const SwTable& rTable, sal_uInt16 nMaxStep )
+{
+ if( getRowSpan() > 0 || !nMaxStep )
+ return *this;
+
+ tools::Long nLeftBorder = lcl_Box2LeftBorder( *this );
+ SwTableBox* pBox = this;
+ const SwTableLine* pMyUpper = GetUpper();
+ sal_uInt16 nLine = rTable.GetTabLines().GetPos( pMyUpper );
+ if( nLine && nLine < rTable.GetTabLines().size() )
+ {
+ SwTableBox* pNext;
+ do
+ {
+ pNext = lcl_LeftBorder2Box( nLeftBorder, rTable.GetTabLines()[--nLine] );
+ if( pNext )
+ pBox = pNext;
+ } while( nLine && --nMaxStep && pNext && pBox->getRowSpan() < 1 );
+ }
+
+ return *pBox;
+}
+
+/** SwTableBox::FindEndOfRowSpan(..) returns the last overlapped cell if there is
+ any. Otherwise the cell itself will returned.
+*/
+
+SwTableBox& SwTableBox::FindEndOfRowSpan( const SwTable& rTable, sal_uInt16 nMaxStep )
+{
+ tools::Long nAbsSpan = getRowSpan();
+ if( nAbsSpan < 0 )
+ nAbsSpan = -nAbsSpan;
+ if( nAbsSpan == 1 || !nMaxStep )
+ return *this;
+
+ if( nMaxStep > --nAbsSpan )
+ nMaxStep = o3tl::narrowing<sal_uInt16>(nAbsSpan);
+ const SwTableLine* pMyUpper = GetUpper();
+ sal_uInt16 nLine = rTable.GetTabLines().GetPos( pMyUpper );
+ nMaxStep = nLine + nMaxStep;
+ if( nMaxStep >= rTable.GetTabLines().size() )
+ nMaxStep = rTable.GetTabLines().size() - 1;
+ tools::Long nLeftBorder = lcl_Box2LeftBorder( *this );
+ SwTableBox* pBox =
+ lcl_LeftBorder2Box( nLeftBorder, rTable.GetTabLines()[ nMaxStep ] );
+ if ( !pBox )
+ pBox = this;
+
+ return *pBox;
+}
+
+/** lcl_getAllMergedBoxes(..) collects all overlapped boxes to a given (master) box
+*/
+
+static void lcl_getAllMergedBoxes( const SwTable& rTable, SwSelBoxes& rBoxes, SwTableBox& rBox )
+{
+ SwTableBox* pBox = &rBox;
+ OSL_ENSURE( pBox == &rBox.FindStartOfRowSpan( rTable ), "Not a master box" );
+ rBoxes.insert( pBox );
+ if( pBox->getRowSpan() == 1 )
+ return;
+ const SwTableLine* pMyUpper = pBox->GetUpper();
+ sal_uInt16 nLine = rTable.GetTabLines().GetPos( pMyUpper );
+ tools::Long nLeftBorder = lcl_Box2LeftBorder( *pBox );
+ sal_uInt16 nCount = rTable.GetTabLines().size();
+ while( ++nLine < nCount && pBox && pBox->getRowSpan() != -1 )
+ {
+ pBox = lcl_LeftBorder2Box( nLeftBorder, rTable.GetTabLines()[nLine] );
+ if( pBox )
+ rBoxes.insert( pBox );
+ }
+}
+
+/** lcl_UnMerge(..) manipulates the row span attribute of a given master cell
+ and its overlapped cells to split them into several pieces.
+*/
+
+static void lcl_UnMerge( const SwTable& rTable, SwTableBox& rBox, size_t nCnt,
+ bool bSameHeight )
+{
+ SwSelBoxes aBoxes;
+ lcl_getAllMergedBoxes( rTable, aBoxes, rBox );
+ size_t const nCount = aBoxes.size();
+ if( nCount < 2 )
+ return;
+ if( nCnt > nCount )
+ nCnt = nCount;
+ std::unique_ptr<size_t[]> const pSplitIdx(new size_t[nCnt]);
+ if( bSameHeight )
+ {
+ std::unique_ptr<SwTwips[]> const pHeights(new SwTwips[nCount]);
+ SwTwips nHeight = 0;
+ for (size_t i = 0; i < nCount; ++i)
+ {
+ SwTableLine* pLine = aBoxes[ i ]->GetUpper();
+ SwFrameFormat *pRowFormat = pLine->GetFrameFormat();
+ pHeights[ i ] = pRowFormat->GetFrameSize().GetHeight();
+ nHeight += pHeights[ i ];
+ }
+ SwTwips nSumH = 0;
+ size_t nIdx = 0;
+ for (size_t i = 1; i <= nCnt; ++i)
+ {
+ SwTwips nSplit = ( i * nHeight ) / nCnt;
+ while( nSumH < nSplit && nIdx < nCount )
+ nSumH += pHeights[ nIdx++ ];
+ pSplitIdx[ i - 1 ] = nIdx;
+ }
+ }
+ else
+ {
+ for (size_t i = 1; i <= nCnt; ++i)
+ {
+ pSplitIdx[ i - 1 ] = ( i * nCount ) / nCnt;
+ }
+ }
+ size_t nIdx = 0;
+ for (size_t i = 0; i < nCnt; ++i)
+ {
+ size_t nNextIdx = pSplitIdx[ i ];
+ aBoxes[ nIdx ]->setRowSpan( nNextIdx - nIdx );
+ lcl_InvalidateCellFrame( *aBoxes[ nIdx ] );
+ while( ++nIdx < nNextIdx )
+ aBoxes[ nIdx ]->setRowSpan( nIdx - nNextIdx );
+ }
+}
+
+/** lcl_FillSelBoxes(..) puts all boxes of a given line into the selection structure
+*/
+
+static void lcl_FillSelBoxes( SwSelBoxes &rBoxes, SwTableLine &rLine )
+{
+ const size_t nBoxCount = rLine.GetTabBoxes().size();
+ for( size_t i = 0; i < nBoxCount; ++i )
+ rBoxes.insert( rLine.GetTabBoxes()[i] );
+}
+
+/** SwTable::InsertSpannedRow(..) inserts "superfluous" rows, i.e. rows containing
+ overlapped cells only. This is a preparation for an upcoming split.
+*/
+
+void SwTable::InsertSpannedRow( SwDoc& rDoc, sal_uInt16 nRowIdx, sal_uInt16 nCnt )
+{
+ CHECK_TABLE( *this )
+ OSL_ENSURE( nCnt && nRowIdx < GetTabLines().size(), "Wrong call of InsertSpannedRow" );
+ SwSelBoxes aBoxes;
+ SwTableLine& rLine = *GetTabLines()[ nRowIdx ];
+ lcl_FillSelBoxes( aBoxes, rLine );
+ SwFormatFrameSize aFSz( rLine.GetFrameFormat()->GetFrameSize() );
+ if( SwFrameSize::Variable != aFSz.GetHeightSizeType() )
+ {
+ SwFrameFormat* pFrameFormat = rLine.ClaimFrameFormat();
+ tools::Long nNewHeight = aFSz.GetHeight() / ( nCnt + 1 );
+ if( !nNewHeight )
+ ++nNewHeight;
+ aFSz.SetHeight( nNewHeight );
+ pFrameFormat->SetFormatAttr( aFSz );
+ }
+ InsertRow_( &rDoc, aBoxes, nCnt, true );
+ const size_t nBoxCount = rLine.GetTabBoxes().size();
+ for( sal_uInt16 n = 0; n < nCnt; ++n )
+ {
+ SwTableLine *pNewLine = GetTabLines()[ nRowIdx + nCnt - n ];
+ for( size_t nCurrBox = 0; nCurrBox < nBoxCount; ++nCurrBox )
+ {
+ sal_Int32 nRowSpan = rLine.GetTabBoxes()[nCurrBox]->getRowSpan();
+ if( nRowSpan > 0 )
+ nRowSpan = - nRowSpan;
+ pNewLine->GetTabBoxes()[ nCurrBox ]->setRowSpan( nRowSpan - n );
+ }
+ }
+ lcl_ChangeRowSpan( *this, nCnt, nRowIdx, false );
+ CHECK_TABLE( *this )
+}
+
+typedef std::pair< sal_uInt16, sal_uInt16 > SwLineOffset;
+typedef std::vector< SwLineOffset > SwLineOffsetArray;
+
+/*
+* When a couple of table boxes has to be split,
+* lcl_SophisticatedFillLineIndices delivers the information where and how many
+* rows have to be inserted.
+* Input
+* rTable: the table to manipulate
+* rBoxes: an array of boxes to split
+* nCnt: how many parts are wanted
+* Output
+* rArr: a list of pairs ( line index, number of lines to insert )
+*/
+static void lcl_SophisticatedFillLineIndices( SwLineOffsetArray &rArr,
+ const SwTable& rTable, const SwSelBoxes& rBoxes, sal_uInt16 nCnt )
+{
+ std::list< SwLineOffset > aBoxes;
+ SwLineOffset aLnOfs( USHRT_MAX, USHRT_MAX );
+ for (size_t i = 0; i < rBoxes.size(); ++i)
+ { // Collect all end line indices and the row spans
+ const SwTableBox &rBox = rBoxes[ i ]->FindStartOfRowSpan( rTable );
+ OSL_ENSURE( rBox.getRowSpan() > 0, "Didn't I say 'StartOfRowSpan' ??" );
+ if( nCnt > rBox.getRowSpan() )
+ {
+ const SwTableLine *pLine = rBox.GetUpper();
+ const sal_uInt16 nEnd = sal_uInt16( rBox.getRowSpan() +
+ rTable.GetTabLines().GetPos( pLine ) );
+ // The next if statement is a small optimization
+ if( aLnOfs.first != nEnd || aLnOfs.second != rBox.getRowSpan() )
+ {
+ aLnOfs.first = nEnd; // ok, this is the line behind the box
+ aLnOfs.second = sal_uInt16( rBox.getRowSpan() ); // the row span
+ aBoxes.insert( aBoxes.end(), aLnOfs );
+ }
+ }
+ }
+ // As I said, I noted the line index _behind_ the last line of the boxes
+ // in the resulting array the index has to be _on_ the line
+ // nSum is to evaluate the wished value
+ sal_uInt16 nSum = 1;
+ while( !aBoxes.empty() )
+ {
+ // I. step:
+ // Looking for the "smallest" line end with the smallest row span
+ std::list< SwLineOffset >::iterator pCurr = aBoxes.begin();
+ aLnOfs = *pCurr; // the line end and row span of the first box
+ while( ++pCurr != aBoxes.end() )
+ {
+ if( aLnOfs.first > pCurr->first )
+ { // Found a smaller line end
+ aLnOfs.first = pCurr->first;
+ aLnOfs.second = pCurr->second; // row span
+ }
+ else if( aLnOfs.first == pCurr->first &&
+ aLnOfs.second < pCurr->second )
+ aLnOfs.second = pCurr->second; // Found a smaller row span
+ }
+ OSL_ENSURE( aLnOfs.second < nCnt, "Clean-up failed" );
+ aLnOfs.second = nCnt - aLnOfs.second; // the number of rows to insert
+ rArr.emplace_back( aLnOfs.first - nSum, aLnOfs.second );
+ // the correction has to be incremented because in the following
+ // loops the line ends were manipulated
+ nSum = nSum + aLnOfs.second;
+
+ pCurr = aBoxes.begin();
+ while( pCurr != aBoxes.end() )
+ {
+ if( pCurr->first == aLnOfs.first )
+ { // These boxes can be removed because the last insertion
+ // of rows will expand their row span above the needed value
+ pCurr = aBoxes.erase(pCurr);
+ }
+ else
+ {
+ bool bBefore = ( pCurr->first - pCurr->second < aLnOfs.first );
+ // Manipulation of the end line indices as if the rows are
+ // already inserted
+ pCurr->first = pCurr->first + aLnOfs.second;
+ if( bBefore )
+ { // If the insertion is inside the box,
+ // its row span has to be incremented
+ pCurr->second = pCurr->second + aLnOfs.second;
+ if( pCurr->second >= nCnt )
+ { // if the row span is bigger than the split factor
+ // this box is done
+ pCurr = aBoxes.erase(pCurr);
+ }
+ else
+ ++pCurr;
+ }
+ else
+ ++pCurr;
+ }
+ }
+ }
+}
+
+typedef std::set< SwTwips > SwSplitLines;
+
+/** lcl_CalculateSplitLineHeights(..) delivers all y-positions where table rows have
+ to be split to fulfill the requested "split same height"
+*/
+
+static sal_uInt16 lcl_CalculateSplitLineHeights( SwSplitLines &rCurr, SwSplitLines &rNew,
+ const SwTable& rTable, const SwSelBoxes& rBoxes, sal_uInt16 nCnt )
+{
+ if( nCnt < 2 )
+ return 0;
+ std::vector< SwLineOffset > aBoxes;
+ SwLineOffset aLnOfs( USHRT_MAX, USHRT_MAX );
+ sal_uInt16 nFirst = USHRT_MAX; // becomes the index of the first line
+ sal_uInt16 nLast = 0; // becomes the index of the last line of the splitting
+ for (size_t i = 0; i < rBoxes.size(); ++i)
+ { // Collect all pairs (start+end) of line indices to split
+ const SwTableBox &rBox = rBoxes[ i ]->FindStartOfRowSpan( rTable );
+ OSL_ENSURE( rBox.getRowSpan() > 0, "Didn't I say 'StartOfRowSpan' ??" );
+ const SwTableLine *pLine = rBox.GetUpper();
+ const sal_uInt16 nStart = rTable.GetTabLines().GetPos( pLine );
+ const sal_uInt16 nEnd = sal_uInt16( rBox.getRowSpan() + nStart - 1 );
+ // The next if statement is a small optimization
+ if( aLnOfs.first != nStart || aLnOfs.second != nEnd )
+ {
+ aLnOfs.first = nStart;
+ aLnOfs.second = nEnd;
+ aBoxes.push_back( aLnOfs );
+ if( nStart < nFirst )
+ nFirst = nStart;
+ if( nEnd > nLast )
+ nLast = nEnd;
+ }
+ }
+
+ if (nFirst == USHRT_MAX)
+ {
+ assert(aBoxes.empty());
+ return 0;
+ }
+
+ SwTwips nHeight = 0;
+ std::unique_ptr<SwTwips[]> pLines(new SwTwips[ nLast + 1 - nFirst ]);
+ for( sal_uInt16 i = nFirst; i <= nLast; ++i )
+ {
+ bool bLayoutAvailable = false;
+ nHeight += rTable.GetTabLines()[ i ]->GetTableLineHeight( bLayoutAvailable );
+ rCurr.insert( rCurr.end(), nHeight );
+ pLines[ i - nFirst ] = nHeight;
+ }
+ for( const auto& rSplit : aBoxes )
+ {
+ SwTwips nBase = rSplit.first <= nFirst ? 0 :
+ pLines[ rSplit.first - nFirst - 1 ];
+ SwTwips nDiff = pLines[ rSplit.second - nFirst ] - nBase;
+ for( sal_uInt16 i = 1; i < nCnt; ++i )
+ {
+ SwTwips nSplit = nBase + ( i * nDiff ) / nCnt;
+ rNew.insert( nSplit );
+ }
+ }
+ return nFirst;
+}
+
+/** lcl_LineIndex(..) delivers the line index of the line behind or above
+ the box selection.
+*/
+
+static sal_uInt16 lcl_LineIndex( const SwTable& rTable, const SwSelBoxes& rBoxes,
+ bool bBehind )
+{
+ sal_uInt16 nDirect = USHRT_MAX;
+ sal_uInt16 nSpan = USHRT_MAX;
+ for (size_t i = 0; i < rBoxes.size(); ++i)
+ {
+ SwTableBox *pBox = rBoxes[i];
+ const SwTableLine* pLine = rBoxes[i]->GetUpper();
+ sal_uInt16 nPos = rTable.GetTabLines().GetPos( pLine );
+ if( USHRT_MAX != nPos )
+ {
+ if( bBehind )
+ {
+ if( nPos > nDirect || nDirect == USHRT_MAX )
+ nDirect = nPos;
+ sal_Int32 nRowSpan = pBox->getRowSpan();
+ if( nRowSpan < 2 )
+ nSpan = 0;
+ else if( nSpan )
+ {
+ sal_uInt16 nEndOfRowSpan = o3tl::narrowing<sal_uInt16>(nPos + nRowSpan - 1);
+ if( nEndOfRowSpan > nSpan || nSpan == USHRT_MAX )
+ nSpan = nEndOfRowSpan;
+ }
+ }
+ else if( nPos < nDirect )
+ nDirect = nPos;
+ }
+ }
+ if( nSpan && nSpan < USHRT_MAX )
+ return nSpan;
+ return nDirect;
+}
+
+/** SwTable::NewSplitRow(..) splits all selected boxes horizontally.
+*/
+
+bool SwTable::NewSplitRow( SwDoc& rDoc, const SwSelBoxes& rBoxes, sal_uInt16 nCnt,
+ bool bSameHeight )
+{
+ CHECK_TABLE( *this )
+ ++nCnt;
+ FndBox_ aFndBox( nullptr, nullptr );
+ aFndBox.SetTableLines( rBoxes, *this );
+
+ if( bSameHeight && rDoc.getIDocumentLayoutAccess().GetCurrentViewShell() )
+ {
+ SwSplitLines aRowLines;
+ SwSplitLines aSplitLines;
+ sal_uInt16 nFirst = lcl_CalculateSplitLineHeights( aRowLines, aSplitLines,
+ *this, rBoxes, nCnt );
+ aFndBox.DelFrames( *this );
+ SwTwips nLast = 0;
+ SwSplitLines::iterator pSplit = aSplitLines.begin();
+ for( const auto& rCurr : aRowLines )
+ {
+ while( pSplit != aSplitLines.end() && *pSplit < rCurr )
+ {
+ InsertSpannedRow( rDoc, nFirst, 1 );
+ SwTableLine* pRow = GetTabLines()[ nFirst ];
+ SwFrameFormat* pRowFormat = pRow->ClaimFrameFormat();
+ SwFormatFrameSize aFSz( pRowFormat->GetFrameSize() );
+ aFSz.SetHeightSizeType( SwFrameSize::Minimum );
+ aFSz.SetHeight( *pSplit - nLast );
+ pRowFormat->SetFormatAttr( aFSz );
+ nLast = *pSplit;
+ ++pSplit;
+ ++nFirst;
+ }
+ if( pSplit != aSplitLines.end() && rCurr == *pSplit )
+ ++pSplit;
+ SwTableLine* pRow = GetTabLines()[ nFirst ];
+ SwFrameFormat* pRowFormat = pRow->ClaimFrameFormat();
+ SwFormatFrameSize aFSz( pRowFormat->GetFrameSize() );
+ aFSz.SetHeightSizeType( SwFrameSize::Minimum );
+ aFSz.SetHeight( rCurr - nLast );
+ pRowFormat->SetFormatAttr( aFSz );
+ nLast = rCurr;
+ ++nFirst;
+ }
+ }
+ else
+ {
+ aFndBox.DelFrames( *this );
+ bSameHeight = false;
+ }
+ if( !bSameHeight )
+ {
+ SwLineOffsetArray aLineOffs;
+ lcl_SophisticatedFillLineIndices( aLineOffs, *this, rBoxes, nCnt );
+ SwLineOffsetArray::reverse_iterator pCurr( aLineOffs.rbegin() );
+ while( pCurr != aLineOffs.rend() )
+ {
+ InsertSpannedRow( rDoc, pCurr->first, pCurr->second );
+ ++pCurr;
+ }
+ }
+
+ std::set<size_t> aIndices;
+ for (size_t i = 0; i < rBoxes.size(); ++i)
+ {
+ OSL_ENSURE( rBoxes[i]->getRowSpan() != 1, "Forgot to split?" );
+ if( rBoxes[i]->getRowSpan() > 1 )
+ aIndices.insert( i );
+ }
+
+ for( const auto& rCurrBox : aIndices )
+ lcl_UnMerge( *this, *rBoxes[rCurrBox], nCnt, bSameHeight );
+
+ CHECK_TABLE( *this )
+ // update the layout
+ aFndBox.MakeFrames( *this );
+
+ return true;
+}
+
+/** SwTable::InsertRow(..) inserts one or more rows before or behind the selected
+ boxes.
+*/
+
+bool SwTable::InsertRow( SwDoc* pDoc, const SwSelBoxes& rBoxes,
+ sal_uInt16 nCnt, bool bBehind )
+{
+ bool bRet = false;
+ if( IsNewModel() )
+ {
+ CHECK_TABLE( *this )
+ sal_uInt16 nRowIdx = lcl_LineIndex( *this, rBoxes, bBehind );
+ if( nRowIdx < USHRT_MAX )
+ {
+ FndBox_ aFndBox( nullptr, nullptr );
+ aFndBox.SetTableLines( rBoxes, *this );
+ aFndBox.DelFrames( *this );
+
+ bRet = true;
+ SwTableLine *pLine = GetTabLines()[ nRowIdx ];
+ SwSelBoxes aLineBoxes;
+ lcl_FillSelBoxes( aLineBoxes, *pLine );
+ InsertRow_( pDoc, aLineBoxes, nCnt, bBehind );
+ const size_t nBoxCount = pLine->GetTabBoxes().size();
+ sal_uInt16 nOfs = bBehind ? 0 : 1;
+ for( sal_uInt16 n = 0; n < nCnt; ++n )
+ {
+ SwTableLine *pNewLine = GetTabLines()[ nRowIdx+nCnt-n-nOfs];
+ for( size_t nCurrBox = 0; nCurrBox < nBoxCount; ++nCurrBox )
+ {
+ sal_Int32 nRowSpan = pLine->GetTabBoxes()[nCurrBox]->getRowSpan();
+ if( bBehind )
+ {
+ if( nRowSpan == 1 || nRowSpan == -1 )
+ nRowSpan = n + 1;
+ else if( nRowSpan > 1 )
+ {
+ nRowSpan = - nRowSpan;
+
+ // tdf#123102 disable numbering of the new hidden
+ // paragraph in merged cells to avoid of bad
+ // renumbering of next list elements
+ SwTableBox* pBox = pNewLine->GetTabBoxes()[nCurrBox];
+ SwNodeIndex aIdx( *pBox->GetSttNd(), +1 );
+ SwContentNode* pCNd = aIdx.GetNode().GetContentNode();
+ if( pCNd && pCNd->IsTextNode() && pCNd->GetTextNode()->GetNumRule() )
+ {
+ SwPosition aPos( *pCNd->GetTextNode() );
+ SwPaM aPam( aPos, aPos );
+ pDoc->DelNumRules( aPam );
+ }
+ }
+ }
+ else
+ {
+ if( nRowSpan > 0 )
+ nRowSpan = n + 1;
+ else
+ --nRowSpan;
+ }
+ pNewLine->GetTabBoxes()[ nCurrBox ]->setRowSpan( nRowSpan - n );
+ }
+ }
+ if( bBehind )
+ ++nRowIdx;
+ if( nRowIdx )
+ lcl_ChangeRowSpan( *this, nCnt, --nRowIdx, true );
+ // update the layout
+ aFndBox.MakeFrames( *this );
+ }
+ CHECK_TABLE( *this )
+ }
+ else
+ bRet = InsertRow_( pDoc, rBoxes, nCnt, bBehind );
+ return bRet;
+}
+
+/** SwTable::PrepareDelBoxes(..) adjusts the row span attributes for an upcoming
+ deletion of table cells and invalidates the layout of these cells.
+*/
+
+void SwTable::PrepareDelBoxes( const SwSelBoxes& rBoxes )
+{
+ if( !IsNewModel() )
+ return;
+
+ for (size_t i = 0; i < rBoxes.size(); ++i)
+ {
+ SwTableBox* pBox = rBoxes[i];
+ sal_Int32 nRowSpan = pBox->getRowSpan();
+ if( nRowSpan != 1 && pBox->GetFrameFormat()->GetFrameSize().GetWidth() )
+ {
+ tools::Long nLeft = lcl_Box2LeftBorder( *pBox );
+ SwTableLine *pLine = pBox->GetUpper();
+ sal_uInt16 nLinePos = GetTabLines().GetPos( pLine);
+ OSL_ENSURE( nLinePos < USHRT_MAX, "Box/table mismatch" );
+ if( nRowSpan > 1 )
+ {
+ if( ++nLinePos < GetTabLines().size() )
+ {
+ pLine = GetTabLines()[ nLinePos ];
+ pBox = lcl_LeftBorder2Box( nLeft, pLine );
+ OSL_ENSURE( pBox, "RowSpan irritation I" );
+ if( pBox )
+ pBox->setRowSpan( --nRowSpan );
+ }
+ }
+ else if( nLinePos > 0 )
+ {
+ do
+ {
+ pLine = GetTabLines()[ --nLinePos ];
+ pBox = lcl_LeftBorder2Box( nLeft, pLine );
+ OSL_ENSURE( pBox, "RowSpan irritation II" );
+ if( pBox )
+ {
+ nRowSpan = pBox->getRowSpan();
+ if( nRowSpan > 1 )
+ {
+ lcl_InvalidateCellFrame( *pBox );
+ --nRowSpan;
+ }
+ else
+ ++nRowSpan;
+ pBox->setRowSpan( nRowSpan );
+ }
+ else
+ nRowSpan = 1;
+ }
+ while( nRowSpan < 0 && nLinePos > 0 );
+ }
+ }
+ }
+}
+
+/** lcl_SearchSelBox(..) adds cells of a given table row to the selection structure
+ if it overlaps with the given x-position range
+*/
+
+static void lcl_SearchSelBox( const SwTable &rTable, SwSelBoxes& rBoxes, tools::Long nMin, tools::Long nMax,
+ SwTableLine& rLine, bool bChkProtected, bool bColumn )
+{
+ tools::Long nLeft = 0;
+ tools::Long nRight = 0;
+ tools::Long nMid = ( nMax + nMin )/ 2;
+ const size_t nCount = rLine.GetTabBoxes().size();
+ for( size_t nCurrBox = 0; nCurrBox < nCount; ++nCurrBox )
+ {
+ SwTableBox* pBox = rLine.GetTabBoxes()[nCurrBox];
+ OSL_ENSURE( pBox, "Missing table box" );
+ tools::Long nWidth = pBox->GetFrameFormat()->GetFrameSize().GetWidth();
+ nRight += nWidth;
+ if( nRight > nMin )
+ {
+ bool bAdd = false;
+ if( nRight <= nMax )
+ bAdd = nLeft >= nMin || nRight >= nMid ||
+ nRight - nMin > nMin - nLeft;
+ else
+ bAdd = nLeft <= nMid || nRight - nMax < nMax - nLeft;
+ sal_Int32 nRowSpan = pBox->getRowSpan();
+ if( bAdd &&
+ ( !bChkProtected ||
+ !pBox->GetFrameFormat()->GetProtect().IsContentProtected() ) )
+ {
+ size_t const nOldCnt = rBoxes.size();
+ rBoxes.insert( pBox );
+ if( bColumn && nRowSpan != 1 && nOldCnt < rBoxes.size() )
+ {
+ SwTableBox *pMasterBox = pBox->getRowSpan() > 0 ? pBox
+ : &pBox->FindStartOfRowSpan( rTable );
+ lcl_getAllMergedBoxes( rTable, rBoxes, *pMasterBox );
+ }
+ }
+ }
+ if( nRight >= nMax )
+ break;
+ nLeft = nRight;
+ }
+}
+
+/** void SwTable::CreateSelection(..) fills the selection structure with table cells
+ for a given SwPaM, ie. start and end position inside a table
+*/
+
+void SwTable::CreateSelection( const SwPaM& rPam, SwSelBoxes& rBoxes,
+ const SearchType eSearch, bool bChkProtected ) const
+{
+ OSL_ENSURE( m_bNewModel, "Don't call me for old tables" );
+ if( m_aLines.empty() )
+ return;
+ const SwNode* pStartNd = rPam.GetPoint()->nNode.GetNode().FindTableBoxStartNode();
+ const SwNode* pEndNd = rPam.GetMark()->nNode.GetNode().FindTableBoxStartNode();
+ if( !pStartNd || !pEndNd )
+ return;
+ CreateSelection( pStartNd, pEndNd, rBoxes, eSearch, bChkProtected );
+}
+
+/** void SwTable::CreateSelection(..) fills the selection structure with table cells
+ for given start and end nodes inside a table
+*/
+void SwTable::CreateSelection( const SwNode* pStartNd, const SwNode* pEndNd,
+ SwSelBoxes& rBoxes, const SearchType eSearch, bool bChkProtected ) const
+{
+ rBoxes.clear();
+ // Looking for start and end of the selection given by SwNode-pointer
+ const size_t nLines = m_aLines.size();
+ // nTop becomes the line number of the upper box
+ // nBottom becomes the line number of the lower box
+ size_t nTop = 0;
+ size_t nBottom = 0;
+ // nUpperMin becomes the left border value of the upper box
+ // nUpperMax becomes the right border of the upper box
+ // nLowerMin and nLowerMax the borders of the lower box
+ tools::Long nUpperMin = 0, nUpperMax = 0;
+ tools::Long nLowerMin = 0, nLowerMax = 0;
+ // nFound will incremented if a box is found
+ // 0 => no box found; 1 => the upper box has been found; 2 => both found
+ int nFound = 0;
+ for( size_t nRow = 0; nFound < 2 && nRow < nLines; ++nRow )
+ {
+ SwTableLine* pLine = m_aLines[nRow];
+ OSL_ENSURE( pLine, "Missing table line" );
+ const size_t nCols = pLine->GetTabBoxes().size();
+ for( size_t nCol = 0; nCol < nCols; ++nCol )
+ {
+ SwTableBox* pBox = pLine->GetTabBoxes()[nCol];
+ OSL_ENSURE( pBox, "Missing table box" );
+ if( pBox->GetSttNd() == pEndNd || pBox->GetSttNd() == pStartNd )
+ {
+ if( !bChkProtected ||
+ !pBox->GetFrameFormat()->GetProtect().IsContentProtected() )
+ rBoxes.insert( pBox );
+ if( nFound )
+ {
+ nBottom = nRow;
+ lcl_CheckMinMax( nLowerMin, nLowerMax, *pLine, nCol, true );
+ ++nFound;
+ break;
+ }
+ else
+ {
+ nTop = nRow;
+ lcl_CheckMinMax( nUpperMin, nUpperMax, *pLine, nCol, true );
+ ++nFound;
+ // If start and end node are identical, we're nearly done...
+ if( pEndNd == pStartNd )
+ {
+ nBottom = nTop;
+ nLowerMin = nUpperMin;
+ nLowerMax = nUpperMax;
+ ++nFound;
+ }
+ }
+ }
+ }
+ }
+ if( nFound < 2 )
+ return; // At least one node was not a part of the given table
+ if( eSearch == SEARCH_ROW )
+ {
+ // Selection of a row is quiet easy:
+ // every (unprotected) box between start and end line
+ // with a positive row span will be collected
+ for( size_t nRow = nTop; nRow <= nBottom; ++nRow )
+ {
+ SwTableLine* pLine = m_aLines[nRow];
+ OSL_ENSURE( pLine, "Missing table line" );
+ const size_t nCount = pLine->GetTabBoxes().size();
+ for( size_t nCurrBox = 0; nCurrBox < nCount; ++nCurrBox )
+ {
+ SwTableBox* pBox = pLine->GetTabBoxes()[nCurrBox];
+ OSL_ENSURE( pBox, "Missing table box" );
+ if( pBox->getRowSpan() > 0 && ( !bChkProtected ||
+ !pBox->GetFrameFormat()->GetProtect().IsContentProtected() ) )
+ rBoxes.insert( pBox );
+ }
+ }
+ return;
+ }
+ bool bCombine = nTop == nBottom;
+ if( !bCombine )
+ {
+ tools::Long nMinWidth = nUpperMax - nUpperMin;
+ tools::Long nTmp = nLowerMax - nLowerMin;
+ if( nMinWidth > nTmp )
+ nMinWidth = nTmp;
+ nTmp = std::min(nLowerMax, nUpperMax);
+ nTmp -= ( nLowerMin < nUpperMin ) ? nUpperMin : nLowerMin;
+ // If the overlapping between upper and lower box is less than half
+ // of the width (of the smaller cell), bCombine is set,
+ // e.g. if upper and lower cell are in different columns
+ bCombine = ( nTmp + nTmp < nMinWidth );
+ }
+ if( bCombine )
+ {
+ if( nUpperMin < nLowerMin )
+ nLowerMin = nUpperMin;
+ else
+ nUpperMin = nLowerMin;
+ if( nUpperMax > nLowerMax )
+ nLowerMax = nUpperMax;
+ else
+ nUpperMax = nLowerMax;
+ }
+ const bool bColumn = eSearch == SEARCH_COL;
+ if( bColumn )
+ {
+ for( size_t i = 0; i < nTop; ++i )
+ lcl_SearchSelBox( *this, rBoxes, nUpperMin, nUpperMax,
+ *m_aLines[i], bChkProtected, bColumn );
+ }
+
+ {
+ tools::Long nMin = std::min(nUpperMin, nLowerMin);
+ tools::Long nMax = nUpperMax < nLowerMax ? nLowerMax : nUpperMax;
+ for( size_t i = nTop; i <= nBottom; ++i )
+ lcl_SearchSelBox( *this, rBoxes, nMin, nMax, *m_aLines[i],
+ bChkProtected, bColumn );
+ }
+ if( bColumn )
+ {
+ for( size_t i = nBottom + 1; i < nLines; ++i )
+ lcl_SearchSelBox( *this, rBoxes, nLowerMin, nLowerMax, *m_aLines[i],
+ bChkProtected, true );
+ }
+}
+
+/** void SwTable::ExpandColumnSelection(..) adds cell to the give selection to
+ assure that at least one cell of every row is part of the selection.
+*/
+
+void SwTable::ExpandColumnSelection( SwSelBoxes& rBoxes, tools::Long &rMin, tools::Long &rMax ) const
+{
+ OSL_ENSURE( m_bNewModel, "Don't call me for old tables" );
+ rMin = 0;
+ rMax = 0;
+ if( m_aLines.empty() || rBoxes.empty() )
+ return;
+
+ const size_t nLineCnt = m_aLines.size();
+ const size_t nBoxCnt = rBoxes.size();
+ size_t nBox = 0;
+ for( size_t nRow = 0; nRow < nLineCnt && nBox < nBoxCnt; ++nRow )
+ {
+ SwTableLine* pLine = m_aLines[nRow];
+ OSL_ENSURE( pLine, "Missing table line" );
+ const size_t nCols = pLine->GetTabBoxes().size();
+ for( size_t nCol = 0; nCol < nCols; ++nCol )
+ {
+ SwTableBox* pBox = pLine->GetTabBoxes()[nCol];
+ OSL_ENSURE( pBox, "Missing table box" );
+ if( pBox == rBoxes[nBox] )
+ {
+ lcl_CheckMinMax( rMin, rMax, *pLine, nCol, nBox == 0 );
+ if( ++nBox >= nBoxCnt )
+ break;
+ }
+ }
+ }
+ for( size_t nRow = 0; nRow < nLineCnt; ++nRow )
+ {
+ SwTableLine* pLine = m_aLines[nRow];
+ const size_t nCols = pLine->GetTabBoxes().size();
+ tools::Long nRight = 0;
+ for( size_t nCurrBox = 0; nCurrBox < nCols; ++nCurrBox )
+ {
+ tools::Long nLeft = nRight;
+ SwTableBox* pBox = pLine->GetTabBoxes()[nCurrBox];
+ nRight += pBox->GetFrameFormat()->GetFrameSize().GetWidth();
+ if( nLeft >= rMin && nRight <= rMax )
+ rBoxes.insert( pBox );
+ }
+ }
+}
+
+/** SwTable::PrepareDeleteCol(..) adjusts the widths of the neighbour cells of
+ a cell selection for an upcoming (column) deletion
+*/
+void SwTable::PrepareDeleteCol( tools::Long nMin, tools::Long nMax )
+{
+ OSL_ENSURE( m_bNewModel, "Don't call me for old tables" );
+ if( m_aLines.empty() || nMax < nMin )
+ return;
+ tools::Long nMid = nMin ? ( nMin + nMax ) / 2 : 0;
+ const SwTwips nTabSize = GetFrameFormat()->GetFrameSize().GetWidth();
+ if( nTabSize == nMax )
+ nMid = nMax;
+ const size_t nLineCnt = m_aLines.size();
+ for( size_t nRow = 0; nRow < nLineCnt; ++nRow )
+ {
+ SwTableLine* pLine = m_aLines[nRow];
+ const size_t nCols = pLine->GetTabBoxes().size();
+ tools::Long nRight = 0;
+ for( size_t nCurrBox = 0; nCurrBox < nCols; ++nCurrBox )
+ {
+ tools::Long nLeft = nRight;
+ SwTableBox* pBox = pLine->GetTabBoxes()[nCurrBox];
+ nRight += pBox->GetFrameFormat()->GetFrameSize().GetWidth();
+ if( nRight < nMin )
+ continue;
+ if( nLeft > nMax )
+ break;
+ tools::Long nNewWidth = -1;
+ if( nLeft < nMin )
+ {
+ if( nRight <= nMax )
+ nNewWidth = nMid - nLeft;
+ }
+ else if( nRight > nMax )
+ nNewWidth = nRight - nMid;
+ else
+ nNewWidth = 0;
+ if( nNewWidth >= 0 )
+ {
+ SwFrameFormat* pFrameFormat = pBox->ClaimFrameFormat();
+ SwFormatFrameSize aFrameSz( pFrameFormat->GetFrameSize() );
+ aFrameSz.SetWidth( nNewWidth );
+ pFrameFormat->SetFormatAttr( aFrameSz );
+ }
+ }
+ }
+}
+
+/** SwTable::ExpandSelection(..) adds all boxes to the box selections which are
+ overlapped by it.
+*/
+
+void SwTable::ExpandSelection( SwSelBoxes& rBoxes ) const
+{
+ for (size_t i = 0; i < rBoxes.size(); ++i)
+ {
+ SwTableBox *pBox = rBoxes[i];
+ sal_Int32 nRowSpan = pBox->getRowSpan();
+ if( nRowSpan != 1 )
+ {
+ SwTableBox *pMasterBox = nRowSpan > 0 ? pBox
+ : &pBox->FindStartOfRowSpan( *this );
+ lcl_getAllMergedBoxes( *this, rBoxes, *pMasterBox );
+ }
+ }
+}
+
+/** SwTable::CheckRowSpan(..) looks for the next line without an overlapping to
+ the previous line.
+*/
+
+void SwTable::CheckRowSpan( SwTableLine* &rpLine, bool bUp ) const
+{
+ OSL_ENSURE( IsNewModel(), "Don't call me for old tables" );
+ sal_uInt16 nLineIdx = GetTabLines().GetPos( rpLine );
+ OSL_ENSURE( nLineIdx < GetTabLines().size(), "Start line out of range" );
+ bool bChange = true;
+ if( bUp )
+ {
+ while( bChange )
+ {
+ bChange = false;
+ rpLine = GetTabLines()[ nLineIdx ];
+ const size_t nCols = rpLine->GetTabBoxes().size();
+ for( size_t nCol = 0; !bChange && nCol < nCols; ++nCol )
+ {
+ SwTableBox* pBox = rpLine->GetTabBoxes()[nCol];
+ if( pBox->getRowSpan() > 1 || pBox->getRowSpan() < -1 )
+ bChange = true;
+ }
+ if( bChange )
+ {
+ if( nLineIdx )
+ --nLineIdx;
+ else
+ {
+ bChange = false;
+ rpLine = nullptr;
+ }
+ }
+ }
+ }
+ else
+ {
+ const size_t nMaxLine = GetTabLines().size();
+ while( bChange )
+ {
+ bChange = false;
+ rpLine = GetTabLines()[ nLineIdx ];
+ const size_t nCols = rpLine->GetTabBoxes().size();
+ for( size_t nCol = 0; !bChange && nCol < nCols; ++nCol )
+ {
+ SwTableBox* pBox = rpLine->GetTabBoxes()[nCol];
+ if( pBox->getRowSpan() < 0 )
+ bChange = true;
+ }
+ if( bChange )
+ {
+ ++nLineIdx;
+ if( nLineIdx >= nMaxLine )
+ {
+ bChange = false;
+ rpLine = nullptr;
+ }
+ }
+ }
+ }
+}
+
+// This structure corrects the row span attributes for a top line of a table
+// In a top line no negative row span is allowed, so these have to be corrected.
+// If there has been at least one correction, all values are stored
+// and can be used by undo of table split
+SwSaveRowSpan::SwSaveRowSpan( SwTableBoxes& rBoxes, sal_uInt16 nSplitLn )
+ : mnSplitLine( nSplitLn )
+{
+ bool bDontSave = true; // nothing changed, nothing to save
+ const size_t nColCount = rBoxes.size();
+ OSL_ENSURE( nColCount, "Empty Table Line" );
+ mnRowSpans.resize( nColCount );
+ for( size_t nCurrCol = 0; nCurrCol < nColCount; ++nCurrCol )
+ {
+ SwTableBox* pBox = rBoxes[nCurrCol];
+ OSL_ENSURE( pBox, "Missing Table Box" );
+ sal_Int32 nRowSp = pBox->getRowSpan();
+ mnRowSpans[ nCurrCol ] = nRowSp;
+ if( nRowSp < 0 )
+ {
+ bDontSave = false;
+ nRowSp = -nRowSp;
+ pBox->setRowSpan( nRowSp ); // correction needed
+ }
+ }
+ if( bDontSave )
+ mnRowSpans.clear();
+}
+
+// This function is called by undo of table split to restore the old row span
+// values at the split line
+void SwTable::RestoreRowSpan( const SwSaveRowSpan& rSave )
+{
+ if( !IsNewModel() ) // for new model only
+ return;
+ sal_uInt16 nLineCount = GetTabLines().size();
+ OSL_ENSURE( rSave.mnSplitLine < nLineCount, "Restore behind last line?" );
+ if( rSave.mnSplitLine >= nLineCount )
+ return;
+
+ SwTableLine* pLine = GetTabLines()[rSave.mnSplitLine];
+ const size_t nColCount = pLine->GetTabBoxes().size();
+ OSL_ENSURE( nColCount, "Empty Table Line" );
+ OSL_ENSURE( nColCount == rSave.mnRowSpans.size(), "Wrong row span store" );
+ if( nColCount != rSave.mnRowSpans.size() )
+ return;
+
+ for( size_t nCurrCol = 0; nCurrCol < nColCount; ++nCurrCol )
+ {
+ SwTableBox* pBox = pLine->GetTabBoxes()[nCurrCol];
+ OSL_ENSURE( pBox, "Missing Table Box" );
+ sal_Int32 nRowSp = pBox->getRowSpan();
+ if( nRowSp != rSave.mnRowSpans[ nCurrCol ] )
+ {
+ OSL_ENSURE( -nRowSp == rSave.mnRowSpans[ nCurrCol ], "Pardon me?!" );
+ OSL_ENSURE( rSave.mnRowSpans[ nCurrCol ] < 0, "Pardon me?!" );
+ pBox->setRowSpan( -nRowSp );
+
+ sal_uInt16 nLine = rSave.mnSplitLine;
+ if( nLine )
+ {
+ tools::Long nLeftBorder = lcl_Box2LeftBorder( *pBox );
+ SwTableBox* pNext;
+ do
+ {
+ pNext = lcl_LeftBorder2Box( nLeftBorder, GetTabLines()[--nLine] );
+ if( pNext )
+ {
+ pBox = pNext;
+ tools::Long nNewSpan = pBox->getRowSpan();
+ if( pBox->getRowSpan() < 1 )
+ nNewSpan -= nRowSp;
+ else
+ {
+ nNewSpan += nRowSp;
+ pNext = nullptr;
+ }
+ pBox->setRowSpan( nNewSpan );
+ }
+ } while( nLine && pNext );
+ }
+ }
+ }
+}
+
+std::unique_ptr<SwSaveRowSpan> SwTable::CleanUpTopRowSpan( sal_uInt16 nSplitLine )
+{
+ if( !IsNewModel() )
+ return nullptr;
+ std::unique_ptr<SwSaveRowSpan> pRet(new SwSaveRowSpan( GetTabLines()[0]->GetTabBoxes(), nSplitLine ));
+ if( pRet->mnRowSpans.empty() )
+ return nullptr;
+ return pRet;
+}
+
+void SwTable::CleanUpBottomRowSpan( sal_uInt16 nDelLines )
+{
+ if( !IsNewModel() )
+ return;
+ const size_t nLastLine = GetTabLines().size()-1;
+ SwTableLine* pLine = GetTabLines()[nLastLine];
+ const size_t nColCount = pLine->GetTabBoxes().size();
+ OSL_ENSURE( nColCount, "Empty Table Line" );
+ for( size_t nCurrCol = 0; nCurrCol < nColCount; ++nCurrCol )
+ {
+ SwTableBox* pBox = pLine->GetTabBoxes()[nCurrCol];
+ OSL_ENSURE( pBox, "Missing Table Box" );
+ sal_Int32 nRowSp = pBox->getRowSpan();
+ if( nRowSp < 0 )
+ nRowSp = -nRowSp;
+ if( nRowSp > 1 )
+ {
+ lcl_ChangeRowSpan( *this, -static_cast<tools::Long>(nDelLines),
+ o3tl::narrowing<sal_uInt16>(nLastLine), false );
+ break;
+ }
+ }
+}
+
+/**
+ This is kind of similar to InsertSpannedRow()/InsertRow() but that one would
+ recursively copy subtables, which would kind of defeat the purpose;
+ this function directly moves the subtable rows's cells into the newly
+ created rows. For the non-subtable boxes, covered-cells are created.
+
+ Outer row heights are adjusted to match the inner row heights, and the
+ last row's height is tweaked to ensure the sum of the heights is at least
+ the original outer row's minimal height.
+
+ Inner row backgrounds are copied to its cells, if they lack a background.
+
+ This currently can't handle more than 1 subtable in a row;
+ the inner rows of all subtables would need to be sorted by their height
+ to create the correct outer row structure, which is tricky and probably
+ requires a layout for the typical variable-height case.
+
+ */
+void SwTable::ConvertSubtableBox(sal_uInt16 const nRow, sal_uInt16 const nBox)
+{
+ SwDoc *const pDoc(GetFrameFormat()->GetDoc());
+ SwTableLine *const pSourceLine(GetTabLines()[nRow]);
+ SwTableBox *const pSubTableBox(pSourceLine->GetTabBoxes()[nBox]);
+ assert(!pSubTableBox->GetTabLines().empty());
+ // are relative (%) heights possible? apparently not
+ SwFormatFrameSize const outerSize(pSourceLine->GetFrameFormat()->GetFrameSize());
+ if (outerSize.GetHeightSizeType() != SwFrameSize::Variable)
+ { // tdf#145871 clear fixed size in first row
+ pSourceLine->ClaimFrameFormat();
+ pSourceLine->GetFrameFormat()->ResetFormatAttr(RES_FRM_SIZE);
+ }
+ tools::Long minHeights(0);
+ {
+ SwFrameFormat const& rSubLineFormat(*pSubTableBox->GetTabLines()[0]->GetFrameFormat());
+ SwFormatFrameSize const* pSize = rSubLineFormat.GetItemIfSet(RES_FRM_SIZE);
+ if (pSize)
+ { // for first row, apply height from inner row to outer row.
+ // in case the existing outer row height was larger than the entire
+ // subtable, the last inserted row needs to be tweaked (below)
+ pSourceLine->GetFrameFormat()->SetFormatAttr(*pSize);
+ if (pSize->GetHeightSizeType() != SwFrameSize::Variable)
+ {
+ minHeights += pSize->GetHeight();
+ }
+ }
+ }
+ for (size_t i = 1; i < pSubTableBox->GetTabLines().size(); ++i)
+ {
+ SwTableLine *const pSubLine(pSubTableBox->GetTabLines()[i]);
+ SwTableLine *const pNewLine = new SwTableLine(
+ static_cast<SwTableLineFormat*>(pSourceLine->GetFrameFormat()),
+ pSourceLine->GetTabBoxes().size() - 1 + pSubLine->GetTabBoxes().size(),
+ nullptr);
+ SwFrameFormat const& rSubLineFormat(*pSubLine->GetFrameFormat());
+ SwFormatFrameSize const* pSize = rSubLineFormat.GetItemIfSet(RES_FRM_SIZE);
+ if (pSize)
+ { // for rows 2..N, copy inner row height to outer row
+ pNewLine->ClaimFrameFormat();
+ pNewLine->GetFrameFormat()->SetFormatAttr(*pSize);
+ if (pSize->GetHeightSizeType() != SwFrameSize::Variable)
+ {
+ minHeights += pSize->GetHeight();
+ }
+ }
+ // ensure the sum of the lines is at least as high as the outer line was
+ if (i == pSubTableBox->GetTabLines().size() - 1
+ && outerSize.GetHeightSizeType() != SwFrameSize::Variable
+ && minHeights < outerSize.GetHeight())
+ {
+ SwFormatFrameSize lastSize(pNewLine->GetFrameFormat()->GetFrameSize());
+ lastSize.SetHeight(lastSize.GetHeight() + outerSize.GetHeight() - minHeights);
+ if (lastSize.GetHeightSizeType() == SwFrameSize::Variable)
+ {
+ lastSize.SetHeightSizeType(SwFrameSize::Minimum);
+ }
+ pNewLine->ClaimFrameFormat();
+ pNewLine->GetFrameFormat()->SetFormatAttr(lastSize);
+ }
+ SfxPoolItem const* pRowBrush(nullptr);
+ (void)rSubLineFormat.GetItemState(RES_BACKGROUND, true, &pRowBrush);
+ GetTabLines().insert(GetTabLines().begin() + nRow + i, pNewLine);
+ for (size_t j = 0; j < pSourceLine->GetTabBoxes().size(); ++j)
+ {
+ if (j == nBox)
+ {
+ for (size_t k = 0; k < pSubLine->GetTabBoxes().size(); ++k)
+ {
+ // move box k to new outer row
+ SwTableBox *const pSourceBox(pSubLine->GetTabBoxes()[k]);
+ assert(pSourceBox->getRowSpan() == 1);
+ // import filter (xmltbli.cxx) converts all box widths to absolute
+ assert(pSourceBox->GetFrameFormat()->GetFrameSize().GetWidthPercent() == 0);
+ ::InsTableBox(*pDoc, GetTableNode(), pNewLine,
+ static_cast<SwTableBoxFormat*>(pSourceBox->GetFrameFormat()),
+ pSourceBox, j+k, 1);
+ // insert dummy text node...
+ pDoc->GetNodes().MakeTextNode(
+ SwNodeIndex(*pSourceBox->GetSttNd(), +1),
+ pDoc->GetDfltTextFormatColl());
+ SwNodeRange content(*pSourceBox->GetSttNd(), SwNodeOffset(+2),
+ *pSourceBox->GetSttNd()->EndOfSectionNode());
+ SwTableBox *const pNewBox(pNewLine->GetTabBoxes()[j+k]);
+ SwNodeIndex insPos(*pNewBox->GetSttNd(), 1);
+ // MoveNodes would delete the box SwStartNode/SwEndNode
+ // without the dummy node
+#if 0
+ pDoc->GetNodes().MoveNodes(content, pDoc->GetNodes(), insPos, false);
+#else
+ pDoc->getIDocumentContentOperations().MoveNodeRange(content, insPos, SwMoveFlags::NO_DELFRMS|SwMoveFlags::REDLINES);
+#endif
+ // delete the empty node that was bundled in the new box
+ pDoc->GetNodes().Delete(insPos);
+ if (pRowBrush)
+ {
+ if (pNewBox->GetFrameFormat()->GetItemState(RES_BACKGROUND, true) != SfxItemState::SET)
+ { // set inner row background on inner cell
+ pNewBox->ClaimFrameFormat();
+ pNewBox->GetFrameFormat()->SetFormatAttr(*pRowBrush);
+ }
+ }
+ // assume that the borders can be left as they are, because
+ // lines don't have borders, only boxes do
+ }
+ }
+ else
+ {
+ // insert covered cell for box j
+ SwTableBox *const pSourceBox(pSourceLine->GetTabBoxes()[j]);
+ assert(pSourceBox->GetTabLines().empty()); // checked for that
+ sal_uInt16 const nInsPos(j < nBox ? j : j + pSubLine->GetTabBoxes().size() - 1);
+ ::InsTableBox(*pDoc, GetTableNode(), pNewLine,
+ static_cast<SwTableBoxFormat*>(pSourceBox->GetFrameFormat()),
+ pSourceBox, nInsPos, 1);
+ // adjust row span:
+ // N rows in subtable, N-1 rows inserted:
+ // -1 -> -N ; -(N-1) ... -1
+ // -2 -> -(N+1) ; -N .. -2
+ // 1 -> N ; -(N-1) .. -1
+ // 2 -> N+1 ; -N .. -2
+ sal_Int32 newSourceRowSpan(pSourceBox->getRowSpan());
+ sal_Int32 newBoxRowSpan;
+ if (newSourceRowSpan < 0)
+ {
+ newSourceRowSpan -= pSubTableBox->GetTabLines().size() - 1;
+ newBoxRowSpan = newSourceRowSpan + i;
+ }
+ else
+ {
+ newSourceRowSpan += pSubTableBox->GetTabLines().size() - 1;
+ newBoxRowSpan = -(newSourceRowSpan - sal::static_int_cast<tools::Long>(i));
+ }
+ pNewLine->GetTabBoxes()[nInsPos]->setRowSpan(newBoxRowSpan);
+ if (i == pSubTableBox->GetTabLines().size() - 1)
+ { // only last iteration
+ pSourceBox->setRowSpan(newSourceRowSpan);
+ }
+ }
+ }
+ }
+ // delete inner rows 2..N
+ while (1 < pSubTableBox->GetTabLines().size())
+ {
+ // careful: the last box deletes pSubLine!
+ SwTableLine *const pSubLine(pSubTableBox->GetTabLines()[1]);
+ for (size_t j = pSubLine->GetTabBoxes().size(); 0 < j; --j)
+ {
+ SwTableBox *const pBox(pSubLine->GetTabBoxes()[0]);
+ DeleteBox_(*this, pBox, nullptr, false, false, nullptr);
+ }
+ }
+ // fix row spans in lines preceding nRow
+ lcl_ChangeRowSpan(*this, pSubTableBox->GetTabLines().size() - 1, nRow - 1, false);
+ // note: the first line of the inner table remains; caller will call
+ // GCLines() to remove it
+}
+
+bool SwTable::CanConvertSubtables() const
+{
+ for (SwTableLine const*const pLine : GetTabLines())
+ {
+ bool haveSubtable(false);
+ for (SwTableBox const*const pBox : pLine->GetTabBoxes())
+ {
+ if (pBox->IsFormulaOrValueBox() == RES_BOXATR_FORMULA)
+ {
+ return false; // no table box formulas yet
+ }
+ if (!pBox->GetTabLines().empty())
+ {
+ if (haveSubtable)
+ { // can't handle 2 subtable in a row yet
+ return false;
+ }
+ haveSubtable = true;
+ bool haveNonFixedInnerLine(false);
+ for (SwTableLine const*const pInnerLine : pBox->GetTabLines())
+ {
+ // bitmap row background will look different
+ SwFrameFormat const& rRowFormat(*pInnerLine->GetFrameFormat());
+ std::unique_ptr<SvxBrushItem> pBrush(rRowFormat.makeBackgroundBrushItem());
+ assert(pBrush);
+ if (pBrush->GetGraphicObject() != nullptr)
+ {
+ /* TODO: all cells could override this?
+ for (SwTableBox & rInnerBox : rInnerLine.GetTabBoxes())
+ */
+ if (1 < pInnerLine->GetTabBoxes().size()) // except if only 1 cell?
+ {
+ return false;
+ }
+ }
+ if (SwFormatFrameSize const* pSize = rRowFormat.GetItemIfSet(RES_FRM_SIZE))
+ {
+ if (pSize->GetHeightSizeType() != SwFrameSize::Fixed)
+ {
+ haveNonFixedInnerLine = true;
+ }
+ }
+ else
+ {
+ haveNonFixedInnerLine = true; // default
+ }
+ for (SwTableBox const*const pInnerBox : pInnerLine->GetTabBoxes())
+ {
+ if (!pInnerBox->GetTabLines().empty())
+ {
+ return false; // nested subtable :(
+ }
+ }
+ }
+ if (haveNonFixedInnerLine)
+ {
+ if (SwFormatFrameSize const* pSize = pLine->GetFrameFormat()->GetItemIfSet(RES_FRM_SIZE))
+ {
+ if (pSize->GetHeightSizeType() != SwFrameSize::Variable)
+ {
+ // not possible to distribute fixed outer row height on rows without layout
+ return false;
+ }
+ }
+ }
+ }
+ }
+ }
+ // note: fields that refer to table cells may be *outside* the table,
+ // so the entire document needs to be imported before checking here
+ // (same for table box formulas and charts)
+ SwDoc *const pDoc(GetFrameFormat()->GetDoc());
+ SwFieldType const*const pTableFields(
+ pDoc->getIDocumentFieldsAccess().GetFieldType(SwFieldIds::Table, "", false));
+ std::vector<SwFormatField*> vFields;
+ pTableFields->GatherFields(vFields);
+ if (!vFields.empty())
+ {
+ return false; // no formulas in fields yet
+ }
+ if (pDoc->GetAttrPool().GetItemCount2(RES_BOXATR_FORMULA) != 0)
+ {
+ return false; // no table box formulas yet
+ }
+ OUString const tableName(GetFrameFormat()->GetName());
+ SwNodeIndex temp(*pDoc->GetNodes().GetEndOfAutotext().StartOfSectionNode(), +1);
+ while (SwStartNode const*const pStartNode = temp.GetNode().GetStartNode())
+ {
+ ++temp;
+ SwOLENode const*const pOLENode(temp.GetNode().GetOLENode());
+ if (pOLENode && tableName == pOLENode->GetChartTableName())
+ { // there are charts that refer to this table
+ // presumably such charts would need to be adapted somehow?
+ return false;
+ }
+ temp.Assign(*pStartNode->EndOfSectionNode(), +1);
+ }
+ return true;
+}
+
+void SwTable::ConvertSubtables()
+{
+ FndBox_ all(nullptr, nullptr);
+ all.DelFrames(*this); // tdf#151375 avoid UAF by frames on deleted cells
+ for (size_t i = 0; i < GetTabLines().size(); ++i)
+ {
+ SwTableLine *const pLine(GetTabLines()[i]);
+ for (size_t j = 0; j < pLine->GetTabBoxes().size(); ++j)
+ {
+ SwTableBox *const pBox(pLine->GetTabBoxes()[j]);
+ SwTableLines & rInnerLines(pBox->GetTabLines());
+ if (!rInnerLines.empty())
+ {
+ ConvertSubtableBox(i, j);
+ }
+ }
+ }
+ GCLines();
+ m_bNewModel = true;
+ all.MakeFrames(*this);
+#if 0
+ // note: outline nodes (and ordinary lists) are sorted by MoveNodes() itself
+ // (this could change order inside table of contents, but that's a
+ // really esoteric use-case)
+ // nodes were moved - sort marks, redlines, footnotes
+ SwDoc *const pDoc(GetFrameFormat()->GetDoc());
+ pDoc->getIDocumentMarkAccess()->assureSortedMarkContainers();
+ pDoc->getIDocumentRedlineAccess().GetRedlineTable().Resort();
+ pDoc->GetFootnoteIdxs().UpdateAllFootnote();
+#endif
+ // assume that there aren't any node indexes to the deleted box start/end nodes
+ CHECK_TABLE( *this )
+}
+
+#ifdef DBG_UTIL
+
+namespace {
+
+struct RowSpanCheck
+{
+ sal_Int32 nRowSpan;
+ SwTwips nLeft;
+ SwTwips nRight;
+};
+
+}
+
+void SwTable::CheckConsistency() const
+{
+ if( !IsNewModel() )
+ return;
+ const size_t nLineCount = GetTabLines().size();
+ const SwTwips nTabSize = GetFrameFormat()->GetFrameSize().GetWidth();
+ SwTwips nLineWidth = 0;
+ std::list< RowSpanCheck > aRowSpanCells;
+ std::list< RowSpanCheck >::iterator aIter = aRowSpanCells.end();
+ SwNodeIndex index(*GetTableNode());
+ ++index;
+ for( size_t nCurrLine = 0; nCurrLine < nLineCount; ++nCurrLine )
+ {
+ SwTwips nWidth = 0;
+ SwTableLine* pLine = GetTabLines()[nCurrLine];
+ SAL_WARN_IF( !pLine, "sw.core", "Missing Table Line" );
+ const size_t nColCount = pLine->GetTabBoxes().size();
+ SAL_WARN_IF( !nColCount, "sw.core", "Empty Table Line" );
+ for( size_t nCurrCol = 0; nCurrCol < nColCount; ++nCurrCol )
+ {
+ SwTableBox* pBox = pLine->GetTabBoxes()[nCurrCol];
+ assert(pBox);
+ SAL_WARN_IF(GetTableNode()->EndOfSectionIndex() <= index.GetIndex(), "sw.core", "Box not in table nodes");
+ SAL_WARN_IF(!index.GetNode().IsStartNode(), "sw.core", "No box start node");
+ index = *index.GetNode().EndOfSectionNode();
+ ++index;
+ SwTwips nNewWidth = pBox->GetFrameFormat()->GetFrameSize().GetWidth() + nWidth;
+ sal_Int32 nRowSp = pBox->getRowSpan();
+ if( nRowSp < 0 )
+ {
+ SAL_WARN_IF( aIter == aRowSpanCells.end(),
+ "sw.core", "Missing master box");
+ if (aIter != aRowSpanCells.end())
+ {
+ SAL_WARN_IF( aIter->nLeft != nWidth || aIter->nRight != nNewWidth,
+ "sw.core", "Wrong position/size of overlapped table box");
+ --(aIter->nRowSpan);
+ SAL_WARN_IF( aIter->nRowSpan != -nRowSp, "sw.core",
+ "Wrong row span value" );
+ if( nRowSp == -1 )
+ {
+ aIter = aRowSpanCells.erase(aIter);
+ }
+ else
+ ++aIter;
+ }
+ }
+ else if( nRowSp != 1 )
+ {
+ SAL_WARN_IF( !nRowSp, "sw.core", "Zero row span?!" );
+ RowSpanCheck aEntry;
+ aEntry.nLeft = nWidth;
+ aEntry.nRight = nNewWidth;
+ aEntry.nRowSpan = nRowSp;
+ aRowSpanCells.insert( aIter, aEntry );
+ }
+ nWidth = nNewWidth;
+ }
+ if( !nCurrLine )
+ nLineWidth = nWidth;
+ SAL_WARN_IF( nWidth != nLineWidth, "sw.core",
+ "Different Line Widths: first: " << nLineWidth
+ << " current [" << nCurrLine << "]: " << nWidth);
+ SAL_WARN_IF( std::abs(nWidth - nTabSize) > 1 /* how tolerant? */, "sw.core",
+ "Line width differs from table width: " << nTabSize
+ << " current [" << nCurrLine << "]: " << nWidth);
+ SAL_WARN_IF( nWidth < 0 || nWidth > USHRT_MAX, "sw.core",
+ "Width out of range [" << nCurrLine << "]: " << nWidth);
+ SAL_WARN_IF( aIter != aRowSpanCells.end(), "sw.core",
+ "Missing overlapped box" );
+ aIter = aRowSpanCells.begin();
+ }
+ bool bEmpty = aRowSpanCells.empty();
+ SAL_WARN_IF( !bEmpty, "sw.core", "Open row span detected" );
+ SAL_WARN_IF(GetTableNode()->EndOfSectionNode() != &index.GetNode(), "sw.core", "table end node not found");
+}
+
+#endif
+
+/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
diff --git a/sw/source/core/table/swtable.cxx b/sw/source/core/table/swtable.cxx
new file mode 100644
index 000000000..c2ff14822
--- /dev/null
+++ b/sw/source/core/table/swtable.cxx
@@ -0,0 +1,2960 @@
+/* -*- 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 <hintids.hxx>
+#include <hints.hxx>
+#include <editeng/lrspitem.hxx>
+#include <editeng/shaditem.hxx>
+#include <editeng/adjustitem.hxx>
+#include <editeng/colritem.hxx>
+#include <osl/diagnose.h>
+#include <sfx2/linkmgr.hxx>
+#include <fmtfsize.hxx>
+#include <fmtornt.hxx>
+#include <fmtpdsc.hxx>
+#include <fldbas.hxx>
+#include <fmtfld.hxx>
+#include <frmatr.hxx>
+#include <doc.hxx>
+#include <IDocumentLinksAdministration.hxx>
+#include <IDocumentRedlineAccess.hxx>
+#include <IDocumentFieldsAccess.hxx>
+#include <docary.hxx>
+#include <frame.hxx>
+#include <swtable.hxx>
+#include <ndtxt.hxx>
+#include <tabcol.hxx>
+#include <tabfrm.hxx>
+#include <cellfrm.hxx>
+#include <rowfrm.hxx>
+#include <swserv.hxx>
+#include <expfld.hxx>
+#include <mdiexp.hxx>
+#include <cellatr.hxx>
+#include <txatbase.hxx>
+#include <htmltbl.hxx>
+#include <swtblfmt.hxx>
+#include <ndindex.hxx>
+#include <tblrwcl.hxx>
+#include <shellres.hxx>
+#include <viewsh.hxx>
+#include <redline.hxx>
+#include <vector>
+#include <calbck.hxx>
+#include <o3tl/string_view.hxx>
+#include <svl/numformat.hxx>
+
+#ifdef DBG_UTIL
+#define CHECK_TABLE(t) (t).CheckConsistency();
+#else
+#define CHECK_TABLE(t)
+#endif
+
+using namespace com::sun::star;
+
+
+#define COLFUZZY 20
+
+static void ChgTextToNum( SwTableBox& rBox, const OUString& rText, const Color* pCol,
+ bool bChgAlign, SwNodeOffset nNdPos );
+
+sal_Int32 SwTableBox::getRowSpan() const
+{
+ return mnRowSpan;
+}
+
+void SwTableBox::setRowSpan( sal_Int32 nNewRowSpan )
+{
+ mnRowSpan = nNewRowSpan;
+}
+
+bool SwTableBox::getDummyFlag() const
+{
+ return mbDummyFlag;
+}
+
+void SwTableBox::setDummyFlag( bool bDummy )
+{
+ mbDummyFlag = bDummy;
+}
+
+//JP 15.09.98: Bug 55741 - Keep tabs (front and rear)
+static OUString& lcl_TabToBlankAtSttEnd( OUString& rText )
+{
+ sal_Unicode c;
+ sal_Int32 n;
+
+ for( n = 0; n < rText.getLength() && ' ' >= ( c = rText[n] ); ++n )
+ if( '\x9' == c )
+ rText = rText.replaceAt( n, 1, u" " );
+ for( n = rText.getLength(); n && ' ' >= ( c = rText[--n] ); )
+ if( '\x9' == c )
+ rText = rText.replaceAt( n, 1, u" " );
+ return rText;
+}
+
+static OUString& lcl_DelTabsAtSttEnd( OUString& rText )
+{
+ sal_Unicode c;
+ sal_Int32 n;
+ OUStringBuffer sBuff(rText);
+
+ for( n = 0; n < sBuff.getLength() && ' ' >= ( c = sBuff[ n ]); ++n )
+ {
+ if( '\x9' == c )
+ sBuff.remove( n--, 1 );
+ }
+ for( n = sBuff.getLength(); n && ' ' >= ( c = sBuff[ --n ]); )
+ {
+ if( '\x9' == c )
+ sBuff.remove( n, 1 );
+ }
+ rText = sBuff.makeStringAndClear();
+ return rText;
+}
+
+void InsTableBox( SwDoc& rDoc, SwTableNode* pTableNd,
+ SwTableLine* pLine, SwTableBoxFormat* pBoxFrameFormat,
+ SwTableBox* pBox,
+ sal_uInt16 nInsPos, sal_uInt16 nCnt )
+{
+ OSL_ENSURE( pBox->GetSttNd(), "Box with no start node" );
+ SwNodeIndex aIdx( *pBox->GetSttNd(), +1 );
+ SwContentNode* pCNd = aIdx.GetNode().GetContentNode();
+ if( !pCNd )
+ pCNd = rDoc.GetNodes().GoNext( &aIdx );
+ OSL_ENSURE( pCNd, "Box with no content node" );
+
+ if( pCNd->IsTextNode() )
+ {
+ if( pCNd->GetpSwAttrSet() )
+ {
+ SwAttrSet aAttrSet( *pCNd->GetpSwAttrSet() );
+ if(pCNd->GetSwAttrSet().HasItem(RES_PARATR_LIST_AUTOFMT))
+ {
+ SwFormatAutoFormat format = aAttrSet.Get(RES_PARATR_LIST_AUTOFMT);
+ const std::shared_ptr<SfxItemSet>& handle = format.GetStyleHandle();
+ aAttrSet.Put(*handle);
+ }
+ if( pBox->GetSaveNumFormatColor() )
+ {
+ if( pBox->GetSaveUserColor() )
+ aAttrSet.Put( SvxColorItem( *pBox->GetSaveUserColor(), RES_CHRATR_COLOR ));
+ else
+ aAttrSet.ClearItem( RES_CHRATR_COLOR );
+ }
+ rDoc.GetNodes().InsBoxen( pTableNd, pLine, pBoxFrameFormat,
+ static_cast<SwTextNode*>(pCNd)->GetTextColl(),
+ &aAttrSet, nInsPos, nCnt );
+ }
+ else
+ rDoc.GetNodes().InsBoxen( pTableNd, pLine, pBoxFrameFormat,
+ static_cast<SwTextNode*>(pCNd)->GetTextColl(),
+ pCNd->GetpSwAttrSet(), nInsPos, nCnt );
+ }
+ else
+ rDoc.GetNodes().InsBoxen( pTableNd, pLine, pBoxFrameFormat,
+ rDoc.GetDfltTextFormatColl(), nullptr,
+ nInsPos, nCnt );
+
+ sal_Int32 nRowSpan = pBox->getRowSpan();
+ if( nRowSpan != 1 )
+ {
+ SwTableBoxes& rTableBoxes = pLine->GetTabBoxes();
+ for( sal_uInt16 i = 0; i < nCnt; ++i )
+ {
+ pBox = rTableBoxes[ i + nInsPos ];
+ pBox->setRowSpan( nRowSpan );
+ }
+ }
+}
+
+SwTable::SwTable()
+ : SwClient( nullptr ),
+ m_pTableNode( nullptr ),
+ m_nGraphicsThatResize( 0 ),
+ m_nRowsToRepeat( 1 ),
+ m_bModifyLocked( false ),
+ m_bNewModel( true )
+{
+ // default value set in the options
+ m_eTableChgMode = GetTableChgDefaultMode();
+}
+
+SwTable::SwTable( const SwTable& rTable )
+ : SwClient( rTable.GetFrameFormat() ),
+ m_pTableNode( nullptr ),
+ m_eTableChgMode( rTable.m_eTableChgMode ),
+ m_nGraphicsThatResize( 0 ),
+ m_nRowsToRepeat( rTable.GetRowsToRepeat() ),
+ maTableStyleName(rTable.maTableStyleName),
+ m_bModifyLocked( false ),
+ m_bNewModel( rTable.m_bNewModel )
+{
+}
+
+void DelBoxNode( SwTableSortBoxes const & rSortCntBoxes )
+{
+ for (size_t n = 0; n < rSortCntBoxes.size(); ++n)
+ {
+ rSortCntBoxes[ n ]->m_pStartNode = nullptr;
+ }
+}
+
+SwTable::~SwTable()
+{
+ if( m_xRefObj.is() )
+ {
+ SwDoc* pDoc = GetFrameFormat()->GetDoc();
+ if( !pDoc->IsInDtor() ) // then remove from the list
+ pDoc->getIDocumentLinksAdministration().GetLinkManager().RemoveServer( m_xRefObj.get() );
+
+ m_xRefObj->Closed();
+ }
+
+ // the table can be deleted if it's the last client of the FrameFormat
+ SwTableFormat* pFormat = GetFrameFormat();
+ pFormat->Remove( this ); // remove
+
+ if( !pFormat->HasWriterListeners() )
+ pFormat->GetDoc()->DelTableFrameFormat( pFormat ); // and delete
+
+ // Delete the pointers from the SortArray of the boxes. The objects
+ // are preserved and are deleted by the lines/boxes arrays dtor.
+ // Note: unfortunately not enough, pointers to the StartNode of the
+ // section need deletion.
+ DelBoxNode(m_TabSortContentBoxes);
+ m_TabSortContentBoxes.clear();
+}
+
+namespace
+{
+
+template<class T>
+T lcl_MulDiv64(sal_uInt64 nA, sal_uInt64 nM, sal_uInt64 nD)
+{
+ assert(nD != 0);
+ return nD == 0 ? static_cast<T>(nA*nM) : static_cast<T>((nA*nM)/nD);
+}
+
+}
+
+static void FormatInArr( std::vector<SwFormat*>& rFormatArr, SwFormat* pBoxFormat )
+{
+ std::vector<SwFormat*>::const_iterator it = std::find( rFormatArr.begin(), rFormatArr.end(), pBoxFormat );
+ if ( it == rFormatArr.end() )
+ rFormatArr.push_back( pBoxFormat );
+}
+
+static void lcl_ModifyBoxes( SwTableBoxes &rBoxes, const tools::Long nOld,
+ const tools::Long nNew, std::vector<SwFormat*>& rFormatArr );
+
+static void lcl_ModifyLines( SwTableLines &rLines, const tools::Long nOld,
+ const tools::Long nNew, std::vector<SwFormat*>& rFormatArr, const bool bCheckSum )
+{
+ for ( size_t i = 0; i < rLines.size(); ++i )
+ ::lcl_ModifyBoxes( rLines[i]->GetTabBoxes(), nOld, nNew, rFormatArr );
+ if( bCheckSum )
+ {
+ for(SwFormat* pFormat : rFormatArr)
+ {
+ const SwTwips nBox = lcl_MulDiv64<SwTwips>(pFormat->GetFrameSize().GetWidth(), nNew, nOld);
+ SwFormatFrameSize aNewBox( SwFrameSize::Variable, nBox, 0 );
+ pFormat->LockModify();
+ pFormat->SetFormatAttr( aNewBox );
+ pFormat->UnlockModify();
+ }
+ }
+}
+
+static void lcl_ModifyBoxes( SwTableBoxes &rBoxes, const tools::Long nOld,
+ const tools::Long nNew, std::vector<SwFormat*>& rFormatArr )
+{
+ sal_uInt64 nSum = 0; // To avoid rounding errors we summarize all box widths
+ sal_uInt64 nOriginalSum = 0; // Sum of original widths
+ for ( size_t i = 0; i < rBoxes.size(); ++i )
+ {
+ SwTableBox &rBox = *rBoxes[i];
+ if ( !rBox.GetTabLines().empty() )
+ {
+ // For SubTables the rounding problem will not be solved :-(
+ ::lcl_ModifyLines( rBox.GetTabLines(), nOld, nNew, rFormatArr, false );
+ }
+ // Adjust the box
+ SwFrameFormat *pFormat = rBox.GetFrameFormat();
+ sal_uInt64 nBox = pFormat->GetFrameSize().GetWidth();
+ nOriginalSum += nBox;
+ nBox = lcl_MulDiv64<sal_uInt64>(nBox, nNew, nOld);
+ const sal_uInt64 nWishedSum = lcl_MulDiv64<sal_uInt64>(nOriginalSum, nNew, nOld) - nSum;
+ if( nWishedSum > 0 )
+ {
+ if( nBox == nWishedSum )
+ FormatInArr( rFormatArr, pFormat );
+ else
+ {
+ nBox = nWishedSum;
+ pFormat = rBox.ClaimFrameFormat();
+ SwFormatFrameSize aNewBox( SwFrameSize::Variable, static_cast< SwTwips >(nBox), 0 );
+ pFormat->LockModify();
+ pFormat->SetFormatAttr( aNewBox );
+ pFormat->UnlockModify();
+ }
+ }
+ else {
+ OSL_FAIL( "Rounding error" );
+ }
+ nSum += nBox;
+ }
+}
+
+void SwTable::SwClientNotify(const SwModify&, const SfxHint& rHint)
+{
+ if (rHint.GetId() != SfxHintId::SwLegacyModify)
+ return;
+ auto pLegacy = static_cast<const sw::LegacyModifyHint*>(&rHint);
+ // catch SSize changes, to adjust the lines/boxes
+ const sal_uInt16 nWhich = pLegacy->GetWhich();
+ const SwFormatFrameSize* pNewSize = nullptr, *pOldSize = nullptr;
+ switch(nWhich)
+ {
+ case RES_ATTRSET_CHG:
+ {
+ if (pLegacy->m_pOld && pLegacy->m_pNew
+ && (pNewSize = static_cast<const SwAttrSetChg*>(pLegacy->m_pNew)->GetChgSet()->GetItemIfSet(
+ RES_FRM_SIZE,
+ false)))
+ {
+ pOldSize = &static_cast<const SwAttrSetChg*>(pLegacy->m_pOld)->GetChgSet()->GetFrameSize();
+ }
+ }
+ break;
+ case RES_FRM_SIZE:
+ {
+ pOldSize = static_cast<const SwFormatFrameSize*>(pLegacy->m_pOld);
+ pNewSize = static_cast<const SwFormatFrameSize*>(pLegacy->m_pNew);
+ }
+ break;
+ default:
+ CheckRegistration(pLegacy->m_pOld);
+ }
+ if (pOldSize && pNewSize && !m_bModifyLocked)
+ AdjustWidths(pOldSize->GetWidth(), pNewSize->GetWidth());
+}
+
+void SwTable::AdjustWidths( const tools::Long nOld, const tools::Long nNew )
+{
+ std::vector<SwFormat*> aFormatArr;
+ aFormatArr.reserve( m_aLines[0]->GetTabBoxes().size() );
+ ::lcl_ModifyLines( m_aLines, nOld, nNew, aFormatArr, true );
+}
+
+static void lcl_RefreshHidden( SwTabCols &rToFill, size_t nPos )
+{
+ for ( size_t i = 0; i < rToFill.Count(); ++i )
+ {
+ if ( std::abs(static_cast<tools::Long>(nPos) - rToFill[i]) <= COLFUZZY )
+ {
+ rToFill.SetHidden( i, false );
+ break;
+ }
+ }
+}
+
+static void lcl_SortedTabColInsert( SwTabCols &rToFill, const SwTableBox *pBox,
+ const SwFrameFormat *pTabFormat, const bool bHidden,
+ const bool bRefreshHidden )
+{
+ const tools::Long nWish = pTabFormat->GetFrameSize().GetWidth();
+ OSL_ENSURE(nWish, "weird <= 0 width frmfrm");
+
+ // The value for the left edge of the box is calculated from the
+ // widths of the previous boxes.
+ tools::Long nPos = 0;
+ tools::Long nLeftMin = 0;
+ tools::Long nRightMax = 0;
+ if (nWish != 0) //fdo#33012 0 width frmfmt
+ {
+ SwTwips nSum = 0;
+ const SwTableBox *pCur = pBox;
+ const SwTableLine *pLine = pBox->GetUpper();
+ const tools::Long nAct = rToFill.GetRight() - rToFill.GetLeft(); // +1 why?
+
+ while ( pLine )
+ {
+ const SwTableBoxes &rBoxes = pLine->GetTabBoxes();
+ for ( size_t i = 0; i < rBoxes.size(); ++i )
+ {
+ const SwTwips nWidth = rBoxes[i]->GetFrameFormat()->GetFrameSize().GetWidth();
+ nSum += nWidth;
+ const tools::Long nTmp = lcl_MulDiv64<tools::Long>(nSum, nAct, nWish);
+
+ if (rBoxes[i] != pCur)
+ {
+ if ( pLine == pBox->GetUpper() || 0 == nLeftMin )
+ nLeftMin = nTmp - nPos;
+ nPos = nTmp;
+ }
+ else
+ {
+ nSum -= nWidth;
+ if ( 0 == nRightMax )
+ nRightMax = nTmp - nPos;
+ break;
+ }
+ }
+ pCur = pLine->GetUpper();
+ pLine = pCur ? pCur->GetUpper() : nullptr;
+ }
+ }
+
+ bool bInsert = !bRefreshHidden;
+ for ( size_t j = 0; bInsert && (j < rToFill.Count()); ++j )
+ {
+ tools::Long nCmp = rToFill[j];
+ if ( (nPos >= ((nCmp >= COLFUZZY) ? nCmp - COLFUZZY : nCmp)) &&
+ (nPos <= (nCmp + COLFUZZY)) )
+ {
+ bInsert = false; // Already has it.
+ }
+ else if ( nPos < nCmp )
+ {
+ bInsert = false;
+ rToFill.Insert( nPos, bHidden, j );
+ }
+ }
+ if ( bInsert )
+ rToFill.Insert( nPos, bHidden, rToFill.Count() );
+ else if ( bRefreshHidden )
+ ::lcl_RefreshHidden( rToFill, nPos );
+
+ if ( !bHidden || bRefreshHidden )
+ return;
+
+ // calculate minimum/maximum values for the existing entries:
+ nLeftMin = nPos - nLeftMin;
+ nRightMax = nPos + nRightMax;
+
+ // check if nPos is entry:
+ bool bFoundPos = false;
+ bool bFoundMax = false;
+ for ( size_t j = 0; !(bFoundPos && bFoundMax ) && j < rToFill.Count(); ++j )
+ {
+ SwTabColsEntry& rEntry = rToFill.GetEntry( j );
+ tools::Long nCmp = rToFill[j];
+
+ if ( (nPos >= ((nCmp >= COLFUZZY) ? nCmp - COLFUZZY : nCmp)) &&
+ (nPos <= (nCmp + COLFUZZY)) )
+ {
+ // check if nLeftMin is > old minimum for entry nPos:
+ const tools::Long nOldMin = rEntry.nMin;
+ if ( nLeftMin > nOldMin )
+ rEntry.nMin = nLeftMin;
+ // check if nRightMin is < old maximum for entry nPos:
+ const tools::Long nOldMax = rEntry.nMax;
+ if ( nRightMax < nOldMax )
+ rEntry.nMax = nRightMax;
+
+ bFoundPos = true;
+ }
+ else if ( (nRightMax >= ((nCmp >= COLFUZZY) ? nCmp - COLFUZZY : nCmp)) &&
+ (nRightMax <= (nCmp + COLFUZZY)) )
+ {
+ // check if nPos is > old minimum for entry nRightMax:
+ const tools::Long nOldMin = rEntry.nMin;
+ if ( nPos > nOldMin )
+ rEntry.nMin = nPos;
+
+ bFoundMax = true;
+ }
+ }
+}
+
+static void lcl_ProcessBoxGet( const SwTableBox *pBox, SwTabCols &rToFill,
+ const SwFrameFormat *pTabFormat, bool bRefreshHidden )
+{
+ if ( !pBox->GetTabLines().empty() )
+ {
+ const SwTableLines &rLines = pBox->GetTabLines();
+ for ( size_t i = 0; i < rLines.size(); ++i )
+ {
+ const SwTableBoxes &rBoxes = rLines[i]->GetTabBoxes();
+ for ( size_t j = 0; j < rBoxes.size(); ++j )
+ ::lcl_ProcessBoxGet( rBoxes[j], rToFill, pTabFormat, bRefreshHidden);
+ }
+ }
+ else
+ ::lcl_SortedTabColInsert( rToFill, pBox, pTabFormat, false, bRefreshHidden );
+}
+
+static void lcl_ProcessLineGet( const SwTableLine *pLine, SwTabCols &rToFill,
+ const SwFrameFormat *pTabFormat )
+{
+ for ( size_t i = 0; i < pLine->GetTabBoxes().size(); ++i )
+ {
+ const SwTableBox *pBox = pLine->GetTabBoxes()[i];
+ if ( pBox->GetSttNd() )
+ ::lcl_SortedTabColInsert( rToFill, pBox, pTabFormat, true, false );
+ else
+ for ( size_t j = 0; j < pBox->GetTabLines().size(); ++j )
+ ::lcl_ProcessLineGet( pBox->GetTabLines()[j], rToFill, pTabFormat );
+ }
+}
+
+void SwTable::GetTabCols( SwTabCols &rToFill, const SwTableBox *pStart,
+ bool bRefreshHidden, bool bCurRowOnly ) const
+{
+ // Optimization: if bHidden is set, we only update the Hidden Array.
+ if ( bRefreshHidden )
+ {
+ // remove corrections
+ for ( size_t i = 0; i < rToFill.Count(); ++i )
+ {
+ SwTabColsEntry& rEntry = rToFill.GetEntry( i );
+ rEntry.nPos -= rToFill.GetLeft();
+ rEntry.nMin -= rToFill.GetLeft();
+ rEntry.nMax -= rToFill.GetLeft();
+ }
+
+ // All are hidden, so add the visible ones.
+ for ( size_t i = 0; i < rToFill.Count(); ++i )
+ rToFill.SetHidden( i, true );
+ }
+ else
+ {
+ rToFill.Remove( 0, rToFill.Count() );
+ }
+
+ // Insertion cases:
+ // 1. All boxes which are inferior to Line which is superior to the Start,
+ // as well as their inferior boxes if present.
+ // 2. Starting from the Line, the superior box plus its neighbours; but no inferiors.
+ // 3. Apply 2. to the Line superior to the chain of boxes,
+ // until the Line's superior is not a box but the table.
+ // Only those boxes are inserted that don't contain further rows. The insertion
+ // function takes care to avoid duplicates. In order to achieve this, we work
+ // with some degree of fuzzyness (to avoid rounding errors).
+ // Only the left edge of the boxes are inserted.
+ // Finally, the first entry is removed again, because it's already
+ // covered by the border.
+ // 4. Scan the table again and insert _all_ boxes, this time as hidden.
+
+ const SwFrameFormat *pTabFormat = GetFrameFormat();
+
+ // 1.
+ const SwTableBoxes &rBoxes = pStart->GetUpper()->GetTabBoxes();
+
+ for ( size_t i = 0; i < rBoxes.size(); ++i )
+ ::lcl_ProcessBoxGet( rBoxes[i], rToFill, pTabFormat, bRefreshHidden );
+
+ // 2. and 3.
+ const SwTableLine *pLine = pStart->GetUpper()->GetUpper() ?
+ pStart->GetUpper()->GetUpper()->GetUpper() : nullptr;
+ while ( pLine )
+ {
+ const SwTableBoxes &rBoxes2 = pLine->GetTabBoxes();
+ for ( size_t k = 0; k < rBoxes2.size(); ++k )
+ ::lcl_SortedTabColInsert( rToFill, rBoxes2[k],
+ pTabFormat, false, bRefreshHidden );
+ pLine = pLine->GetUpper() ? pLine->GetUpper()->GetUpper() : nullptr;
+ }
+
+ if ( !bRefreshHidden )
+ {
+ // 4.
+ if ( !bCurRowOnly )
+ {
+ for ( size_t i = 0; i < m_aLines.size(); ++i )
+ ::lcl_ProcessLineGet( m_aLines[i], rToFill, pTabFormat );
+ }
+
+ rToFill.Remove( 0 );
+ }
+
+ // Now the coordinates are relative to the left table border - i.e.
+ // relative to SwTabCols.nLeft. However, they are expected
+ // relative to the left document border, i.e. SwTabCols.nLeftMin.
+ // So all values need to be extended by nLeft.
+ for ( size_t i = 0; i < rToFill.Count(); ++i )
+ {
+ SwTabColsEntry& rEntry = rToFill.GetEntry( i );
+ rEntry.nPos += rToFill.GetLeft();
+ rEntry.nMin += rToFill.GetLeft();
+ rEntry.nMax += rToFill.GetLeft();
+ }
+}
+
+// Structure for parameter passing
+struct Parm
+{
+ const SwTabCols &rNew;
+ const SwTabCols &rOld;
+ tools::Long nNewWish,
+ nOldWish;
+ std::deque<SwTableBox*> aBoxArr;
+ SwShareBoxFormats aShareFormats;
+
+ Parm( const SwTabCols &rN, const SwTabCols &rO )
+ : rNew( rN ), rOld( rO ), nNewWish(0), nOldWish(0)
+ {}
+};
+
+static void lcl_ProcessBoxSet( SwTableBox *pBox, Parm &rParm );
+
+static void lcl_ProcessLine( SwTableLine *pLine, Parm &rParm )
+{
+ SwTableBoxes &rBoxes = pLine->GetTabBoxes();
+ for ( size_t i = rBoxes.size(); i > 0; )
+ {
+ --i;
+ ::lcl_ProcessBoxSet( rBoxes[i], rParm );
+ }
+}
+
+static void lcl_ProcessBoxSet( SwTableBox *pBox, Parm &rParm )
+{
+ if ( !pBox->GetTabLines().empty() )
+ {
+ SwTableLines &rLines = pBox->GetTabLines();
+ for ( size_t i = rLines.size(); i > 0; )
+ {
+ --i;
+ lcl_ProcessLine( rLines[i], rParm );
+ }
+ }
+ else
+ {
+ // Search the old TabCols for the current position (calculate from
+ // left and right edge). Adjust the box if the values differ from
+ // the new TabCols. If the adjusted edge has no neighbour we also
+ // adjust all superior boxes.
+
+ const tools::Long nOldAct = rParm.rOld.GetRight() -
+ rParm.rOld.GetLeft(); // +1 why?
+
+ // The value for the left edge of the box is calculated from the
+ // widths of the previous boxes plus the left edge.
+ tools::Long nLeft = rParm.rOld.GetLeft();
+ const SwTableBox *pCur = pBox;
+ const SwTableLine *pLine = pBox->GetUpper();
+
+ while ( pLine )
+ {
+ const SwTableBoxes &rBoxes = pLine->GetTabBoxes();
+ for ( size_t i = 0; (i < rBoxes.size()) && (rBoxes[i] != pCur); ++i)
+ {
+ nLeft += lcl_MulDiv64<tools::Long>(
+ rBoxes[i]->GetFrameFormat()->GetFrameSize().GetWidth(),
+ nOldAct, rParm.nOldWish);
+ }
+ pCur = pLine->GetUpper();
+ pLine = pCur ? pCur->GetUpper() : nullptr;
+ }
+ tools::Long nLeftDiff = 0;
+ tools::Long nRightDiff = 0;
+ if ( nLeft != rParm.rOld.GetLeft() ) // There are still boxes before this.
+ {
+ // Right edge is left edge plus width.
+ const tools::Long nWidth = lcl_MulDiv64<tools::Long>(
+ pBox->GetFrameFormat()->GetFrameSize().GetWidth(),
+ nOldAct, rParm.nOldWish);
+ const tools::Long nRight = nLeft + nWidth;
+ size_t nLeftPos = 0;
+ size_t nRightPos = 0;
+ bool bFoundLeftPos = false;
+ bool bFoundRightPos = false;
+ for ( size_t i = 0; i < rParm.rOld.Count(); ++i )
+ {
+ if ( nLeft >= (rParm.rOld[i] - COLFUZZY) &&
+ nLeft <= (rParm.rOld[i] + COLFUZZY) )
+ {
+ nLeftPos = i;
+ bFoundLeftPos = true;
+ }
+ else if ( nRight >= (rParm.rOld[i] - COLFUZZY) &&
+ nRight <= (rParm.rOld[i] + COLFUZZY) )
+ {
+ nRightPos = i;
+ bFoundRightPos = true;
+ }
+ }
+ nLeftDiff = bFoundLeftPos ?
+ rParm.rOld[nLeftPos] - rParm.rNew[nLeftPos] : 0;
+ nRightDiff= bFoundRightPos ?
+ rParm.rNew[nRightPos] - rParm.rOld[nRightPos] : 0;
+ }
+ else // The first box.
+ {
+ nLeftDiff = rParm.rOld.GetLeft() - rParm.rNew.GetLeft();
+ if ( rParm.rOld.Count() )
+ {
+ // Calculate the difference to the edge touching the first box.
+ const tools::Long nWidth = lcl_MulDiv64<tools::Long>(
+ pBox->GetFrameFormat()->GetFrameSize().GetWidth(),
+ nOldAct, rParm.nOldWish);
+ const tools::Long nTmp = nWidth + rParm.rOld.GetLeft();
+ for ( size_t i = 0; i < rParm.rOld.Count(); ++i )
+ {
+ if ( nTmp >= (rParm.rOld[i] - COLFUZZY) &&
+ nTmp <= (rParm.rOld[i] + COLFUZZY) )
+ {
+ nRightDiff = rParm.rNew[i] - rParm.rOld[i];
+ break;
+ }
+ }
+ }
+ }
+
+ if( pBox->getRowSpan() == 1 )
+ {
+ const sal_uInt16 nPos = pBox->GetUpper()->GetBoxPos( pBox );
+ SwTableBoxes& rTableBoxes = pBox->GetUpper()->GetTabBoxes();
+ if( nPos && rTableBoxes[ nPos - 1 ]->getRowSpan() != 1 )
+ nLeftDiff = 0;
+ if( nPos + 1 < o3tl::narrowing<sal_uInt16>(rTableBoxes.size()) &&
+ rTableBoxes[ nPos + 1 ]->getRowSpan() != 1 )
+ nRightDiff = 0;
+ }
+ else
+ nLeftDiff = nRightDiff = 0;
+
+ if ( nLeftDiff || nRightDiff )
+ {
+ // The difference is the actual difference amount. For stretched
+ // tables, it does not make sense to adjust the attributes of the
+ // boxes by this amount. The difference amount needs to be converted
+ // accordingly.
+ tools::Long nTmp = rParm.rNew.GetRight() - rParm.rNew.GetLeft(); // +1 why?
+ nLeftDiff *= rParm.nNewWish;
+ nLeftDiff /= nTmp;
+ nRightDiff *= rParm.nNewWish;
+ nRightDiff /= nTmp;
+ tools::Long nDiff = nLeftDiff + nRightDiff;
+
+ // Adjust the box and all superiors by the difference amount.
+ while ( pBox )
+ {
+ SwFormatFrameSize aFormatFrameSize( pBox->GetFrameFormat()->GetFrameSize() );
+ aFormatFrameSize.SetWidth( aFormatFrameSize.GetWidth() + nDiff );
+ if ( aFormatFrameSize.GetWidth() < 0 )
+ aFormatFrameSize.SetWidth( -aFormatFrameSize.GetWidth() );
+ rParm.aShareFormats.SetSize( *pBox, aFormatFrameSize );
+
+ // The outer cells of the last row are responsible to adjust a surrounding cell.
+ // Last line check:
+ if ( pBox->GetUpper()->GetUpper() &&
+ pBox->GetUpper() != pBox->GetUpper()->GetUpper()->GetTabLines().back())
+ {
+ pBox = nullptr;
+ }
+ else
+ {
+ // Middle cell check:
+ if ( pBox != pBox->GetUpper()->GetTabBoxes().front() )
+ nDiff = nRightDiff;
+
+ if ( pBox != pBox->GetUpper()->GetTabBoxes().back() )
+ nDiff -= nRightDiff;
+
+ pBox = nDiff ? pBox->GetUpper()->GetUpper() : nullptr;
+ }
+ }
+ }
+ }
+}
+
+static void lcl_ProcessBoxPtr( SwTableBox *pBox, std::deque<SwTableBox*> &rBoxArr,
+ bool bBefore )
+{
+ if ( !pBox->GetTabLines().empty() )
+ {
+ const SwTableLines &rLines = pBox->GetTabLines();
+ for ( size_t i = 0; i < rLines.size(); ++i )
+ {
+ const SwTableBoxes &rBoxes = rLines[i]->GetTabBoxes();
+ for ( size_t j = 0; j < rBoxes.size(); ++j )
+ ::lcl_ProcessBoxPtr( rBoxes[j], rBoxArr, bBefore );
+ }
+ }
+ else if ( bBefore )
+ rBoxArr.push_front( pBox );
+ else
+ rBoxArr.push_back( pBox );
+}
+
+static void lcl_AdjustBox( SwTableBox *pBox, const tools::Long nDiff, Parm &rParm );
+
+static void lcl_AdjustLines( SwTableLines &rLines, const tools::Long nDiff, Parm &rParm )
+{
+ for ( size_t i = 0; i < rLines.size(); ++i )
+ {
+ SwTableBox *pBox = rLines[i]->GetTabBoxes()
+ [rLines[i]->GetTabBoxes().size()-1];
+ lcl_AdjustBox( pBox, nDiff, rParm );
+ }
+}
+
+static void lcl_AdjustBox( SwTableBox *pBox, const tools::Long nDiff, Parm &rParm )
+{
+ if ( !pBox->GetTabLines().empty() )
+ ::lcl_AdjustLines( pBox->GetTabLines(), nDiff, rParm );
+
+ // Adjust the size of the box.
+ SwFormatFrameSize aFormatFrameSize( pBox->GetFrameFormat()->GetFrameSize() );
+ aFormatFrameSize.SetWidth( aFormatFrameSize.GetWidth() + nDiff );
+
+ rParm.aShareFormats.SetSize( *pBox, aFormatFrameSize );
+}
+
+void SwTable::SetTabCols( const SwTabCols &rNew, const SwTabCols &rOld,
+ const SwTableBox *pStart, bool bCurRowOnly )
+{
+ CHECK_TABLE( *this )
+
+ SetHTMLTableLayout(std::shared_ptr<SwHTMLTableLayout>()); // delete HTML-Layout
+
+ // FME: Made rOld const. The caller is responsible for passing correct
+ // values of rOld. Therefore we do not have to call GetTabCols anymore:
+ //GetTabCols( rOld, pStart );
+
+ Parm aParm( rNew, rOld );
+
+ OSL_ENSURE( rOld.Count() == rNew.Count(), "Number of columns changed.");
+
+ // Convert the edges. We need to adjust the size of the table and some boxes.
+ // For the size adjustment, we must not make use of the Modify, since that'd
+ // adjust all boxes, which we really don't want.
+ SwFrameFormat *pFormat = GetFrameFormat();
+ aParm.nOldWish = aParm.nNewWish = pFormat->GetFrameSize().GetWidth();
+ if ( (rOld.GetLeft() != rNew.GetLeft()) ||
+ (rOld.GetRight()!= rNew.GetRight()) )
+ {
+ LockModify();
+ {
+ SvxLRSpaceItem aLR( pFormat->GetLRSpace() );
+ SvxShadowItem aSh( pFormat->GetShadow() );
+
+ SwTwips nShRight = aSh.CalcShadowSpace( SvxShadowItemSide::RIGHT );
+ SwTwips nShLeft = aSh.CalcShadowSpace( SvxShadowItemSide::LEFT );
+
+ aLR.SetLeft ( rNew.GetLeft() - nShLeft );
+ aLR.SetRight( rNew.GetRightMax() - rNew.GetRight() - nShRight );
+ pFormat->SetFormatAttr( aLR );
+
+ // The alignment of the table needs to be adjusted accordingly.
+ // This is done by preserving the exact positions that have been
+ // set by the user.
+ SwFormatHoriOrient aOri( pFormat->GetHoriOrient() );
+ if( text::HoriOrientation::NONE != aOri.GetHoriOrient() &&
+ text::HoriOrientation::CENTER != aOri.GetHoriOrient() )
+ {
+ const bool bLeftDist = rNew.GetLeft() != nShLeft;
+ const bool bRightDist = rNew.GetRight() + nShRight != rNew.GetRightMax();
+ if(!bLeftDist && !bRightDist)
+ aOri.SetHoriOrient( text::HoriOrientation::FULL );
+ else if(!bRightDist && rNew.GetLeft() > nShLeft )
+ aOri.SetHoriOrient( text::HoriOrientation::RIGHT );
+ else if(!bLeftDist && rNew.GetRight() + nShRight < rNew.GetRightMax())
+ aOri.SetHoriOrient( text::HoriOrientation::LEFT );
+ else
+ {
+ // if an automatic table hasn't (really) changed size, then leave it as auto.
+ const tools::Long nOldWidth = rOld.GetRight() - rOld.GetLeft();
+ const tools::Long nNewWidth = rNew.GetRight() - rNew.GetLeft();
+ if (aOri.GetHoriOrient() != text::HoriOrientation::FULL
+ || std::abs(nOldWidth - nNewWidth) > COLFUZZY)
+ {
+ aOri.SetHoriOrient(text::HoriOrientation::LEFT_AND_WIDTH);
+ }
+ }
+ }
+ pFormat->SetFormatAttr( aOri );
+ }
+ const tools::Long nAct = rOld.GetRight() - rOld.GetLeft(); // +1 why?
+ tools::Long nTabDiff = 0;
+
+ if ( rOld.GetLeft() != rNew.GetLeft() )
+ {
+ nTabDiff = rOld.GetLeft() - rNew.GetLeft();
+ nTabDiff *= aParm.nOldWish;
+ nTabDiff /= nAct;
+ }
+ if ( rOld.GetRight() != rNew.GetRight() )
+ {
+ tools::Long nDiff = rNew.GetRight() - rOld.GetRight();
+ nDiff *= aParm.nOldWish;
+ nDiff /= nAct;
+ nTabDiff += nDiff;
+ if( !IsNewModel() )
+ ::lcl_AdjustLines( GetTabLines(), nDiff, aParm );
+ }
+
+ // Adjust the size of the table, watch out for stretched tables.
+ if ( nTabDiff )
+ {
+ aParm.nNewWish += nTabDiff;
+ if ( aParm.nNewWish < 0 )
+ aParm.nNewWish = USHRT_MAX; // Oops! Have to roll back.
+ SwFormatFrameSize aSz( pFormat->GetFrameSize() );
+ if ( aSz.GetWidth() != aParm.nNewWish )
+ {
+ aSz.SetWidth( aParm.nNewWish );
+ aSz.SetWidthPercent( 0 );
+ pFormat->SetFormatAttr( aSz );
+ }
+ }
+ UnlockModify();
+ }
+
+ if( IsNewModel() )
+ NewSetTabCols( aParm, rNew, rOld, pStart, bCurRowOnly );
+ else
+ {
+ if ( bCurRowOnly )
+ {
+ // To adjust the current row, we need to process all its boxes,
+ // similar to the filling of the TabCols (see GetTabCols()).
+ // Unfortunately we again have to take care to adjust the boxes
+ // from back to front, respectively from outer to inner.
+ // The best way to achieve this is probably to track the boxes
+ // in a PtrArray.
+ const SwTableBoxes &rBoxes = pStart->GetUpper()->GetTabBoxes();
+ for ( size_t i = 0; i < rBoxes.size(); ++i )
+ ::lcl_ProcessBoxPtr( rBoxes[i], aParm.aBoxArr, false );
+
+ const SwTableLine *pLine = pStart->GetUpper()->GetUpper() ?
+ pStart->GetUpper()->GetUpper()->GetUpper() : nullptr;
+ const SwTableBox *pExcl = pStart->GetUpper()->GetUpper();
+ while ( pLine )
+ {
+ const SwTableBoxes &rBoxes2 = pLine->GetTabBoxes();
+ bool bBefore = true;
+ for ( size_t i = 0; i < rBoxes2.size(); ++i )
+ {
+ if ( rBoxes2[i] != pExcl )
+ ::lcl_ProcessBoxPtr( rBoxes2[i], aParm.aBoxArr, bBefore );
+ else
+ bBefore = false;
+ }
+ pExcl = pLine->GetUpper();
+ pLine = pLine->GetUpper() ? pLine->GetUpper()->GetUpper() : nullptr;
+ }
+ // After we've inserted a bunch of boxes (hopefully all and in
+ // correct order), we just need to process them in reverse order.
+ for ( int j = aParm.aBoxArr.size()-1; j >= 0; --j )
+ {
+ SwTableBox *pBox = aParm.aBoxArr[j];
+ ::lcl_ProcessBoxSet( pBox, aParm );
+ }
+ }
+ else
+ {
+ // Adjusting the entire table is 'easy'. All boxes without lines are
+ // adjusted, as are their superiors. Of course we need to process
+ // in reverse order to prevent fooling ourselves!
+ SwTableLines &rLines = GetTabLines();
+ for ( size_t i = rLines.size(); i > 0; )
+ {
+ --i;
+ ::lcl_ProcessLine( rLines[i], aParm );
+ }
+ }
+ }
+
+#ifdef DBG_UTIL
+ {
+ // do some checking for correct table widths
+ SwTwips nSize = GetFrameFormat()->GetFrameSize().GetWidth();
+ for (size_t n = 0; n < m_aLines.size(); ++n)
+ {
+ CheckBoxWidth( *m_aLines[ n ], nSize );
+ }
+ }
+#endif
+}
+
+typedef std::pair<sal_uInt16, sal_uInt16> ColChange;
+typedef std::list< ColChange > ChangeList;
+
+static void lcl_AdjustWidthsInLine( SwTableLine* pLine, ChangeList& rOldNew,
+ Parm& rParm, sal_uInt16 nColFuzzy )
+{
+ ChangeList::iterator pCurr = rOldNew.begin();
+ if( pCurr == rOldNew.end() )
+ return;
+ const size_t nCount = pLine->GetTabBoxes().size();
+ SwTwips nBorder = 0;
+ SwTwips nRest = 0;
+ for( size_t i = 0; i < nCount; ++i )
+ {
+ SwTableBox* pBox = pLine->GetTabBoxes()[i];
+ SwTwips nWidth = pBox->GetFrameFormat()->GetFrameSize().GetWidth();
+ SwTwips nNewWidth = nWidth - nRest;
+ nRest = 0;
+ nBorder += nWidth;
+ if( pCurr != rOldNew.end() && nBorder + nColFuzzy >= pCurr->first )
+ {
+ nBorder -= nColFuzzy;
+ while( pCurr != rOldNew.end() && nBorder > pCurr->first )
+ ++pCurr;
+ if( pCurr != rOldNew.end() )
+ {
+ nBorder += nColFuzzy;
+ if( nBorder + nColFuzzy >= pCurr->first )
+ {
+ if( pCurr->second == pCurr->first )
+ nRest = 0;
+ else
+ nRest = pCurr->second - nBorder;
+ nNewWidth += nRest;
+ ++pCurr;
+ }
+ }
+ }
+ if( nNewWidth != nWidth )
+ {
+ if( nNewWidth < 0 )
+ {
+ nRest += 1 - nNewWidth;
+ nNewWidth = 1;
+ }
+ SwFormatFrameSize aFormatFrameSize( pBox->GetFrameFormat()->GetFrameSize() );
+ aFormatFrameSize.SetWidth( nNewWidth );
+ rParm.aShareFormats.SetSize( *pBox, aFormatFrameSize );
+ }
+ }
+}
+
+static void lcl_CalcNewWidths( std::vector<sal_uInt16> &rSpanPos, ChangeList& rChanges,
+ SwTableLine* pLine, tools::Long nWish, tools::Long nWidth, bool bTop )
+{
+ if( rChanges.empty() )
+ {
+ rSpanPos.clear();
+ return;
+ }
+ if( rSpanPos.empty() )
+ {
+ rChanges.clear();
+ return;
+ }
+ std::vector<sal_uInt16> aNewSpanPos;
+ ChangeList::iterator pCurr = rChanges.begin();
+ ChangeList aNewChanges { *pCurr }; // Nullposition
+ std::vector<sal_uInt16>::iterator pSpan = rSpanPos.begin();
+ sal_uInt16 nCurr = 0;
+ SwTwips nOrgSum = 0;
+ bool bRowSpan = false;
+ sal_uInt16 nRowSpanCount = 0;
+ const size_t nCount = pLine->GetTabBoxes().size();
+ for( size_t nCurrBox = 0; nCurrBox < nCount; ++nCurrBox )
+ {
+ SwTableBox* pBox = pLine->GetTabBoxes()[nCurrBox];
+ SwTwips nCurrWidth = pBox->GetFrameFormat()->GetFrameSize().GetWidth();
+ const sal_Int32 nRowSpan = pBox->getRowSpan();
+ const bool bCurrRowSpan = bTop ? nRowSpan < 0 :
+ ( nRowSpan > 1 || nRowSpan < -1 );
+ if( bRowSpan || bCurrRowSpan )
+ aNewSpanPos.push_back( nRowSpanCount );
+ bRowSpan = bCurrRowSpan;
+ nOrgSum += nCurrWidth;
+ const sal_uInt16 nPos = lcl_MulDiv64<sal_uInt16>(
+ lcl_MulDiv64<sal_uInt64>(nOrgSum, nWidth, nWish),
+ nWish, nWidth);
+ while( pCurr != rChanges.end() && pCurr->first < nPos )
+ {
+ ++nCurr;
+ ++pCurr;
+ }
+ bool bNew = true;
+ if( pCurr != rChanges.end() && pCurr->first <= nPos &&
+ pCurr->first != pCurr->second )
+ {
+ pSpan = std::find_if(pSpan, rSpanPos.end(),
+ [nCurr](const sal_uInt16 nSpan) { return nSpan >= nCurr; });
+ if( pSpan != rSpanPos.end() && *pSpan == nCurr )
+ {
+ aNewChanges.push_back( *pCurr );
+ ++nRowSpanCount;
+ bNew = false;
+ }
+ }
+ if( bNew )
+ {
+ ColChange aTmp( nPos, nPos );
+ aNewChanges.push_back( aTmp );
+ ++nRowSpanCount;
+ }
+ }
+
+ pCurr = aNewChanges.begin();
+ ChangeList::iterator pLast = pCurr;
+ ChangeList::iterator pLeftMove = pCurr;
+ while( pCurr != aNewChanges.end() )
+ {
+ if( pLeftMove == pCurr )
+ {
+ while( ++pLeftMove != aNewChanges.end() && pLeftMove->first <= pLeftMove->second )
+ ;
+ }
+ if( pCurr->second == pCurr->first )
+ {
+ if( pLeftMove != aNewChanges.end() && pCurr->second > pLeftMove->second )
+ {
+ if( pLeftMove->first == pLast->first )
+ pCurr->second = pLeftMove->second;
+ else
+ {
+ pCurr->second = lcl_MulDiv64<sal_uInt16>(
+ pCurr->first - pLast->first,
+ pLeftMove->second - pLast->second,
+ pLeftMove->first - pLast->first) + pLast->second;
+ }
+ }
+ pLast = pCurr;
+ ++pCurr;
+ }
+ else if( pCurr->second > pCurr->first )
+ {
+ pLast = pCurr;
+ ++pCurr;
+ ChangeList::iterator pNext = pCurr;
+ while( pNext != pLeftMove && pNext->second == pNext->first &&
+ pNext->second < pLast->second )
+ ++pNext;
+ while( pCurr != pNext )
+ {
+ if( pNext == aNewChanges.end() || pNext->first == pLast->first )
+ pCurr->second = pLast->second;
+ else
+ {
+ pCurr->second = lcl_MulDiv64<sal_uInt16>(
+ pCurr->first - pLast->first,
+ pNext->second - pLast->second,
+ pNext->first - pLast->first) + pLast->second;
+ }
+ ++pCurr;
+ }
+ pLast = pCurr;
+ }
+ else
+ {
+ pLast = pCurr;
+ ++pCurr;
+ }
+ }
+
+ rChanges.swap(aNewChanges);
+ rSpanPos.swap(aNewSpanPos);
+}
+
+void SwTable::NewSetTabCols( Parm &rParm, const SwTabCols &rNew,
+ const SwTabCols &rOld, const SwTableBox *pStart, bool bCurRowOnly )
+{
+#if OSL_DEBUG_LEVEL > 1
+ static int nCallCount = 0;
+ ++nCallCount;
+#endif
+ // First step: evaluate which lines have been moved/which widths changed
+ ChangeList aOldNew;
+ const tools::Long nNewWidth = rParm.rNew.GetRight() - rParm.rNew.GetLeft();
+ const tools::Long nOldWidth = rParm.rOld.GetRight() - rParm.rOld.GetLeft();
+ if( nNewWidth < 1 || nOldWidth < 1 )
+ return;
+ for( size_t i = 0; i <= rOld.Count(); ++i )
+ {
+ tools::Long nNewPos;
+ tools::Long nOldPos;
+ if( i == rOld.Count() )
+ {
+ nOldPos = rParm.rOld.GetRight() - rParm.rOld.GetLeft();
+ nNewPos = rParm.rNew.GetRight() - rParm.rNew.GetLeft();
+ }
+ else
+ {
+ nOldPos = rOld[i] - rParm.rOld.GetLeft();
+ nNewPos = rNew[i] - rParm.rNew.GetLeft();
+ }
+ nNewPos = lcl_MulDiv64<tools::Long>(nNewPos, rParm.nNewWish, nNewWidth);
+ nOldPos = lcl_MulDiv64<tools::Long>(nOldPos, rParm.nOldWish, nOldWidth);
+ if( nOldPos != nNewPos && nNewPos > 0 && nOldPos > 0 )
+ {
+ ColChange aChg( o3tl::narrowing<sal_uInt16>(nOldPos), o3tl::narrowing<sal_uInt16>(nNewPos) );
+ aOldNew.push_back( aChg );
+ }
+ }
+ // Finished first step
+ int nCount = aOldNew.size();
+ if( !nCount )
+ return; // no change, nothing to do
+ SwTableLines &rLines = GetTabLines();
+ if( bCurRowOnly )
+ {
+ const SwTableLine* pCurrLine = pStart->GetUpper();
+ sal_uInt16 nCurr = rLines.GetPos( pCurrLine );
+ if( nCurr >= USHRT_MAX )
+ return;
+
+ ColChange aChg( 0, 0 );
+ aOldNew.push_front( aChg );
+ std::vector<sal_uInt16> aRowSpanPos;
+ if( nCurr )
+ {
+ ChangeList aCopy;
+ sal_uInt16 nPos = 0;
+ for( const auto& rCop : aOldNew )
+ {
+ aCopy.push_back( rCop );
+ aRowSpanPos.push_back( nPos++ );
+ }
+ lcl_CalcNewWidths( aRowSpanPos, aCopy, rLines[nCurr],
+ rParm.nOldWish, nOldWidth, true );
+ bool bGoOn = !aRowSpanPos.empty();
+ sal_uInt16 j = nCurr;
+ while( bGoOn )
+ {
+ lcl_CalcNewWidths( aRowSpanPos, aCopy, rLines[--j],
+ rParm.nOldWish, nOldWidth, true );
+ lcl_AdjustWidthsInLine( rLines[j], aCopy, rParm, 0 );
+ bGoOn = !aRowSpanPos.empty() && j > 0;
+ }
+ aRowSpanPos.clear();
+ }
+ if( nCurr+1 < o3tl::narrowing<sal_uInt16>(rLines.size()) )
+ {
+ ChangeList aCopy;
+ sal_uInt16 nPos = 0;
+ for( const auto& rCop : aOldNew )
+ {
+ aCopy.push_back( rCop );
+ aRowSpanPos.push_back( nPos++ );
+ }
+ lcl_CalcNewWidths( aRowSpanPos, aCopy, rLines[nCurr],
+ rParm.nOldWish, nOldWidth, false );
+ bool bGoOn = !aRowSpanPos.empty();
+ sal_uInt16 j = nCurr;
+ while( bGoOn )
+ {
+ lcl_CalcNewWidths( aRowSpanPos, aCopy, rLines[++j],
+ rParm.nOldWish, nOldWidth, false );
+ lcl_AdjustWidthsInLine( rLines[j], aCopy, rParm, 0 );
+ bGoOn = !aRowSpanPos.empty() && j+1 < o3tl::narrowing<sal_uInt16>(rLines.size());
+ }
+ }
+ ::lcl_AdjustWidthsInLine( rLines[nCurr], aOldNew, rParm, COLFUZZY );
+ }
+ else
+ {
+ for( size_t i = 0; i < rLines.size(); ++i )
+ ::lcl_AdjustWidthsInLine( rLines[i], aOldNew, rParm, COLFUZZY );
+ }
+ CHECK_TABLE( *this )
+}
+
+// return the pointer of the box specified.
+static bool lcl_IsValidRowName( std::u16string_view rStr )
+{
+ bool bIsValid = true;
+ size_t nLen = rStr.size();
+ for( size_t i = 0; i < nLen && bIsValid; ++i )
+ {
+ const sal_Unicode cChar = rStr[i];
+ if (cChar < '0' || cChar > '9')
+ bIsValid = false;
+ }
+ return bIsValid;
+}
+
+// #i80314#
+// add 3rd parameter and its handling
+sal_uInt16 SwTable::GetBoxNum( OUString& rStr, bool bFirstPart,
+ const bool bPerformValidCheck )
+{
+ sal_uInt16 nRet = 0;
+ if( bFirstPart ) // true == column; false == row
+ {
+ sal_Int32 nPos = 0;
+ // the first one uses letters for addressing!
+ bool bFirst = true;
+ sal_uInt32 num = 0;
+ bool overflow = false;
+ while (nPos<rStr.getLength())
+ {
+ sal_Unicode cChar = rStr[nPos];
+ if ((cChar<'A' || cChar>'Z') && (cChar<'a' || cChar>'z'))
+ break;
+ cChar -= 'A';
+ if( cChar >= 26 )
+ cChar -= 'a' - '[';
+ if( bFirst )
+ bFirst = false;
+ else
+ ++num;
+ num = num * 52 + cChar;
+ if (num > SAL_MAX_UINT16) {
+ overflow = true;
+ }
+ ++nPos;
+ }
+ nRet = overflow ? SAL_MAX_UINT16 : num;
+ rStr = rStr.copy( nPos ); // Remove char from String
+ }
+ else
+ {
+ const sal_Int32 nPos = rStr.indexOf( "." );
+ if ( nPos<0 )
+ {
+ nRet = 0;
+ if ( !bPerformValidCheck || lcl_IsValidRowName( rStr ) )
+ {
+ nRet = o3tl::narrowing<sal_uInt16>(rStr.toInt32());
+ }
+ rStr.clear();
+ }
+ else
+ {
+ nRet = 0;
+ const std::u16string_view aText( rStr.subView( 0, nPos ) );
+ if ( !bPerformValidCheck || lcl_IsValidRowName( aText ) )
+ {
+ nRet = o3tl::narrowing<sal_uInt16>(o3tl::toInt32(aText));
+ }
+ rStr = rStr.copy( nPos+1 );
+ }
+ }
+ return nRet;
+}
+
+// #i80314#
+// add 2nd parameter and its handling
+const SwTableBox* SwTable::GetTableBox( const OUString& rName,
+ const bool bPerformValidCheck ) const
+{
+ const SwTableBox* pBox = nullptr;
+ const SwTableLine* pLine;
+ const SwTableLines* pLines;
+
+ sal_uInt16 nLine, nBox;
+ OUString aNm( rName );
+ while( !aNm.isEmpty() )
+ {
+ nBox = SwTable::GetBoxNum( aNm, nullptr == pBox, bPerformValidCheck );
+ // first box ?
+ if( !pBox )
+ pLines = &GetTabLines();
+ else
+ {
+ pLines = &pBox->GetTabLines();
+ if( nBox )
+ --nBox;
+ }
+
+ nLine = SwTable::GetBoxNum( aNm, false, bPerformValidCheck );
+
+ // determine line
+ if( !nLine || nLine > pLines->size() )
+ return nullptr;
+ pLine = (*pLines)[ nLine-1 ];
+
+ // determine box
+ const SwTableBoxes* pBoxes = &pLine->GetTabBoxes();
+ if( nBox >= pBoxes->size() )
+ return nullptr;
+ pBox = (*pBoxes)[ nBox ];
+ }
+
+ // check if the box found has any contents
+ if( pBox && !pBox->GetSttNd() )
+ {
+ OSL_FAIL( "Box without content, looking for the next one!" );
+ // "drop this" until the first box
+ while( !pBox->GetTabLines().empty() )
+ pBox = pBox->GetTabLines().front()->GetTabBoxes().front();
+ }
+ return pBox;
+}
+
+SwTableBox* SwTable::GetTableBox( SwNodeOffset nSttIdx )
+{
+ // For optimizations, don't always process the entire SortArray.
+ // Converting text to table, tries certain conditions
+ // to ask for a table box of a table that is not yet having a format
+ if(!GetFrameFormat())
+ return nullptr;
+ SwTableBox* pRet = nullptr;
+ SwNodes& rNds = GetFrameFormat()->GetDoc()->GetNodes();
+ SwNodeOffset nIndex = nSttIdx + 1;
+ SwContentNode* pCNd = nullptr;
+ SwTableNode* pTableNd = nullptr;
+
+ while ( nIndex < rNds.Count() )
+ {
+ pTableNd = rNds[ nIndex ]->GetTableNode();
+ if ( pTableNd )
+ break;
+
+ pCNd = rNds[ nIndex ]->GetContentNode();
+ if ( pCNd )
+ break;
+
+ ++nIndex;
+ }
+
+ if ( pCNd || pTableNd )
+ {
+ sw::BroadcastingModify* pModify = pCNd;
+ // #144862# Better handling of table in table
+ if ( pTableNd && pTableNd->GetTable().GetFrameFormat() )
+ pModify = pTableNd->GetTable().GetFrameFormat();
+
+ SwFrame* pFrame = pModify ? SwIterator<SwFrame,sw::BroadcastingModify>(*pModify).First() : nullptr;
+ while ( pFrame && !pFrame->IsCellFrame() )
+ pFrame = pFrame->GetUpper();
+ if ( pFrame )
+ pRet = const_cast<SwTableBox*>(static_cast<SwCellFrame*>(pFrame)->GetTabBox());
+ }
+
+ // In case the layout doesn't exist yet or anything else goes wrong.
+ if ( !pRet )
+ {
+ for (size_t n = m_TabSortContentBoxes.size(); n; )
+ {
+ if (m_TabSortContentBoxes[ --n ]->GetSttIdx() == nSttIdx)
+ {
+ return m_TabSortContentBoxes[ n ];
+ }
+ }
+ }
+ return pRet;
+}
+
+bool SwTable::IsTableComplex() const
+{
+ // Returns true for complex tables, i.e. tables that contain nestings,
+ // like containing boxes not part of the first line, e.g. results of
+ // splits/merges which lead to more complex structures.
+ for (size_t n = 0; n < m_TabSortContentBoxes.size(); ++n)
+ {
+ if (m_TabSortContentBoxes[ n ]->GetUpper()->GetUpper())
+ {
+ return true;
+ }
+ }
+ return false;
+}
+
+SwTableLine::SwTableLine( SwTableLineFormat *pFormat, sal_uInt16 nBoxes,
+ SwTableBox *pUp )
+ : SwClient( pFormat )
+ , m_pUpper( pUp )
+ , m_eRedlineType( RedlineType::None )
+{
+ m_aBoxes.reserve( nBoxes );
+}
+
+SwTableLine::~SwTableLine()
+{
+ for (size_t i = 0; i < m_aBoxes.size(); ++i)
+ {
+ delete m_aBoxes[i];
+ }
+ // the TabelleLine can be deleted if it's the last client of the FrameFormat
+ sw::BroadcastingModify* pMod = GetFrameFormat();
+ pMod->Remove( this ); // remove,
+ if( !pMod->HasWriterListeners() )
+ delete pMod; // and delete
+}
+
+SwFrameFormat* SwTableLine::ClaimFrameFormat()
+{
+ // This method makes sure that this object is an exclusive SwTableLine client
+ // of an SwTableLineFormat object
+ // If other SwTableLine objects currently listen to the same SwTableLineFormat as
+ // this one, something needs to be done
+ SwTableLineFormat *pRet = static_cast<SwTableLineFormat*>(GetFrameFormat());
+ SwIterator<SwTableLine,SwFormat> aIter( *pRet );
+ for( SwTableLine* pLast = aIter.First(); pLast; pLast = aIter.Next() )
+ {
+ if ( pLast != this )
+ {
+ // found another SwTableLine that is a client of the current Format
+ // create a new Format as a copy and use it for this object
+ SwTableLineFormat *pNewFormat = pRet->GetDoc()->MakeTableLineFormat();
+ *pNewFormat = *pRet;
+
+ // register SwRowFrames that know me as clients at the new Format
+ SwIterator<SwRowFrame,SwFormat> aFrameIter( *pRet );
+ for( SwRowFrame* pFrame = aFrameIter.First(); pFrame; pFrame = aFrameIter.Next() )
+ if( pFrame->GetTabLine() == this )
+ pFrame->RegisterToFormat( *pNewFormat );
+
+ // register myself
+ pNewFormat->Add( this );
+ pRet = pNewFormat;
+ break;
+ }
+ }
+
+ return pRet;
+}
+
+void SwTableLine::ChgFrameFormat(SwTableLineFormat* pNewFormat)
+{
+ auto pOld = GetFrameFormat();
+ pOld->CallSwClientNotify(sw::TableLineFormatChanged(*pNewFormat, *this));
+ // Now, re-register self.
+ pNewFormat->Add(this);
+ if(!pOld->HasWriterListeners())
+ delete pOld;
+}
+
+SwTwips SwTableLine::GetTableLineHeight( bool& bLayoutAvailable ) const
+{
+ SwTwips nRet = 0;
+ bLayoutAvailable = false;
+ SwIterator<SwRowFrame,SwFormat> aIter( *GetFrameFormat() );
+ // A row could appear several times in headers/footers so only one chain of master/follow tables
+ // will be accepted...
+ const SwTabFrame* pChain = nullptr; // My chain
+ for( SwRowFrame* pLast = aIter.First(); pLast; pLast = aIter.Next() )
+ {
+ if( pLast->GetTabLine() == this )
+ {
+ const SwTabFrame* pTab = pLast->FindTabFrame();
+ bLayoutAvailable = ( pTab && pTab->IsVertical() ) ?
+ ( 0 < pTab->getFrameArea().Height() ) :
+ ( 0 < pTab->getFrameArea().Width() );
+
+ // The first one defines the chain, if a chain is defined, only members of the chain
+ // will be added.
+ if (pTab && (!pChain || pChain->IsAnFollow( pTab ) || pTab->IsAnFollow(pChain)))
+ {
+ pChain = pTab; // defines my chain (even it is already)
+ if( pTab->IsVertical() )
+ nRet += pLast->getFrameArea().Width();
+ else
+ nRet += pLast->getFrameArea().Height();
+ // Optimization, if there are no master/follows in my chain, nothing more to add
+ if( !pTab->HasFollow() && !pTab->IsFollow() )
+ break;
+ // This is not an optimization, this is necessary to avoid double additions of
+ // repeating rows
+ if( pTab->IsInHeadline(*pLast) )
+ break;
+ }
+ }
+ }
+ return nRet;
+}
+
+bool SwTableLine::IsEmpty() const
+{
+ for (size_t i = 0; i < m_aBoxes.size(); ++i)
+ {
+ if ( !m_aBoxes[i]->IsEmpty() )
+ return false;
+ }
+ return true;
+}
+
+bool SwTable::HasDeletedRow() const
+{
+ const SwRedlineTable& aRedlineTable = GetFrameFormat()->GetDoc()->getIDocumentRedlineAccess().GetRedlineTable();
+ if ( aRedlineTable.empty() )
+ return false;
+
+ SwRedlineTable::size_type nRedlinePos = 0;
+ for (size_t i = 0; i < m_aLines.size(); ++i)
+ {
+ if ( m_aLines[i]->IsDeleted(nRedlinePos) )
+ return true;
+ }
+ return false;
+}
+
+bool SwTable::IsDeleted() const
+{
+ const SwRedlineTable& aRedlineTable = GetFrameFormat()->GetDoc()->getIDocumentRedlineAccess().GetRedlineTable();
+ if ( aRedlineTable.empty() )
+ return false;
+
+ SwRedlineTable::size_type nRedlinePos = 0;
+ for (size_t i = 0; i < m_aLines.size(); ++i)
+ {
+ if ( !m_aLines[i]->IsDeleted(nRedlinePos) )
+ return false;
+ }
+ return true;
+}
+
+// TODO Set HasTextChangesOnly=true, if needed based on the redlines in the cells.
+// At tracked row deletion, return with the newest deletion of the row or
+// at tracked row insertion, return with the oldest insertion in the row, which
+// contain the change data of the row change.
+// If the return value is SwRedlineTable::npos, there is no tracked row change.
+SwRedlineTable::size_type SwTableLine::UpdateTextChangesOnly(
+ SwRedlineTable::size_type& rRedlinePos, bool bUpdateProperty ) const
+{
+ SwRedlineTable::size_type nRet = SwRedlineTable::npos;
+ const SwRedlineTable& aRedlineTable = GetFrameFormat()->GetDoc()->getIDocumentRedlineAccess().GetRedlineTable();
+
+ // check table row property "HasTextChangesOnly", if it's defined and its
+ // value is false, and all text content is in delete redlines, the row is deleted
+ const SvxPrintItem *pHasTextChangesOnlyProp =
+ GetFrameFormat()->GetAttrSet().GetItem<SvxPrintItem>(RES_PRINT);
+ if ( pHasTextChangesOnlyProp && !pHasTextChangesOnlyProp->GetValue() )
+ {
+ const SwTableBoxes & rBoxes = GetTabBoxes();
+ size_t nBoxes = rBoxes.size();
+ bool bInsertion = false;
+ bool bPlainTextInLine = false;
+ SwRedlineTable::size_type nOldestRedline = SwRedlineTable::npos;
+ SwRedlineTable::size_type nNewestRedline = SwRedlineTable::npos;
+ for (size_t nBoxIndex = 0; nBoxIndex < nBoxes && rRedlinePos < aRedlineTable.size(); ++nBoxIndex)
+ {
+ auto pBox = rBoxes[nBoxIndex];
+ if ( pBox->IsEmpty() )
+ {
+ // no text content, check the next cells
+ continue;
+ }
+
+ bool bHasRedlineInBox = false;
+ SwPosition aCellStart( SwNodeIndex( *pBox->GetSttNd(), 0 ) );
+ SwPosition aCellEnd( SwNodeIndex( *pBox->GetSttNd()->EndOfSectionNode(), -1 ) );
+ SwNodeIndex pEndNodeIndex(aCellEnd.nNode.GetNode());
+ SwRangeRedline* pPreviousDeleteRedline = nullptr;
+ for( ; rRedlinePos < aRedlineTable.size(); ++rRedlinePos )
+ {
+ const SwRangeRedline* pRedline = aRedlineTable[ rRedlinePos ];
+
+ if ( pRedline->Start()->nNode > pEndNodeIndex )
+ {
+ // no more redlines in the actual cell,
+ // check the next ones
+ break;
+ }
+
+ // redline in the cell
+ if ( aCellStart <= *pRedline->Start() )
+ {
+ if ( !bHasRedlineInBox )
+ {
+ bHasRedlineInBox = true;
+ // plain text before the first redline in the text
+ if ( pRedline->Start()->nContent.GetIndex() > 0 )
+ bPlainTextInLine = true;
+ }
+
+ RedlineType nType = pRedline->GetType();
+
+ // first insert redline
+ if ( !bInsertion )
+ {
+ if ( RedlineType::Insert == nType )
+ {
+ bInsertion = true;
+ }
+ else
+ {
+ // plain text between the delete redlines
+ if ( pPreviousDeleteRedline &&
+ *pPreviousDeleteRedline->End() < *pRedline->Start() )
+ {
+ bPlainTextInLine = true;
+ }
+ pPreviousDeleteRedline = const_cast<SwRangeRedline*>(pRedline);
+ }
+ }
+
+ // search newest and oldest redlines
+ if ( nNewestRedline == SwRedlineTable::npos ||
+ aRedlineTable[nNewestRedline]->GetRedlineData().GetTimeStamp() <
+ pRedline->GetRedlineData().GetTimeStamp() )
+ {
+ nNewestRedline = rRedlinePos;
+ }
+ if ( nOldestRedline == SwRedlineTable::npos ||
+ aRedlineTable[nOldestRedline]->GetRedlineData().GetTimeStamp() >
+ pRedline->GetRedlineData().GetTimeStamp() )
+ {
+ nOldestRedline = rRedlinePos;
+ }
+ }
+ }
+
+ // there is text content outside of redlines: not a deletion
+ if ( !bInsertion && ( !bHasRedlineInBox || ( pPreviousDeleteRedline &&
+ ( pPreviousDeleteRedline->End()->nNode < aCellEnd.nNode ||
+ pPreviousDeleteRedline->End()->nContent.GetIndex() <
+ aCellEnd.nNode.GetNode().GetContentNode()->Len() ) ) ) )
+ {
+ bPlainTextInLine = true;
+ // not deleted cell content: the row is not empty
+ // maybe insertion of a row, try to search it
+ bInsertion = true;
+ }
+ }
+
+ // choose return redline, if it exists or remove changed row attribute
+ if ( bInsertion && SwRedlineTable::npos != nOldestRedline &&
+ RedlineType::Insert == aRedlineTable[ nOldestRedline ]->GetType() )
+ {
+ // there is an insert redline, which is the oldest redline in the row
+ nRet = nOldestRedline;
+ }
+ else if ( !bInsertion && !bPlainTextInLine && SwRedlineTable::npos != nNewestRedline &&
+ RedlineType::Delete == aRedlineTable[ nNewestRedline ]->GetType() )
+ {
+ // there is a delete redline, which is the newest redline in the row,
+ // and no text outside of redlines, and no insert redline in the row,
+ // i.e. whole text content is deleted
+ nRet = nNewestRedline;
+ }
+ else
+ {
+ // no longer tracked row insertion or deletion
+ nRet = SwRedlineTable::npos;
+ // set TextChangesOnly = true to remove the tracked deletion
+ // FIXME Undo is not supported here (this is only a fallback,
+ // because using SetRowNotTracked() is not recommended here)
+ if ( bUpdateProperty )
+ {
+ SvxPrintItem aUnsetTracking(RES_PRINT, true);
+ SwFrameFormat *pFormat = const_cast<SwTableLine*>(this)->ClaimFrameFormat();
+ pFormat->LockModify();
+ pFormat->SetFormatAttr( aUnsetTracking );
+ pFormat->UnlockModify();
+ }
+ }
+ }
+
+ // cache the result
+ const_cast<SwTableLine*>(this)->SetRedlineType( SwRedlineTable::npos == nRet
+ ? RedlineType::None
+ : aRedlineTable[ nRet ]->GetType());
+
+ return nRet;
+}
+
+bool SwTableLine::IsDeleted(SwRedlineTable::size_type& rRedlinePos) const
+{
+ SwRedlineTable::size_type nPos = UpdateTextChangesOnly(rRedlinePos);
+ if ( nPos != SwRedlineTable::npos )
+ {
+ const SwRedlineTable& aRedlineTable =
+ GetFrameFormat()->GetDoc()->getIDocumentRedlineAccess().GetRedlineTable();
+ if ( RedlineType::Delete == aRedlineTable[nPos]->GetType() )
+ return true;
+ }
+ return false;
+}
+
+RedlineType SwTableLine::GetRedlineType() const
+{
+ const SwRedlineTable& aRedlineTable = GetFrameFormat()->GetDoc()->getIDocumentRedlineAccess().GetRedlineTable();
+ if ( aRedlineTable.empty() )
+ return RedlineType::None;
+
+ // check table row property "HasTextChangesOnly", if it's defined and its value is
+ // false, return with the cached redline type, if it exists, otherwise calculate it
+ const SvxPrintItem *pHasTextChangesOnlyProp =
+ GetFrameFormat()->GetAttrSet().GetItem<SvxPrintItem>(RES_PRINT);
+ if ( pHasTextChangesOnlyProp && !pHasTextChangesOnlyProp->GetValue() )
+ {
+ if ( RedlineType::None != m_eRedlineType )
+ return m_eRedlineType;
+
+ SwRedlineTable::size_type nPos = 0;
+ nPos = UpdateTextChangesOnly(nPos);
+ if ( nPos != SwRedlineTable::npos )
+ return aRedlineTable[nPos]->GetType();
+ }
+ else if ( RedlineType::None != m_eRedlineType )
+ // empty the cache
+ const_cast<SwTableLine*>(this)->SetRedlineType( RedlineType::None );
+
+ return RedlineType::None;
+}
+
+SwTableBox::SwTableBox( SwTableBoxFormat* pFormat, sal_uInt16 nLines, SwTableLine *pUp )
+ : SwClient(nullptr)
+ , m_aLines()
+ , m_pStartNode(nullptr)
+ , m_pUpper(pUp)
+ , mnRowSpan(1)
+ , mbDummyFlag(false)
+ , mbDirectFormatting(false)
+{
+ m_aLines.reserve( nLines );
+ CheckBoxFormat( pFormat )->Add( this );
+}
+
+SwTableBox::SwTableBox( SwTableBoxFormat* pFormat, const SwNodeIndex &rIdx,
+ SwTableLine *pUp )
+ : SwClient(nullptr)
+ , m_aLines()
+ , m_pUpper(pUp)
+ , mnRowSpan(1)
+ , mbDummyFlag(false)
+ , mbDirectFormatting(false)
+{
+ CheckBoxFormat( pFormat )->Add( this );
+
+ m_pStartNode = rIdx.GetNode().GetStartNode();
+
+ // insert into the table
+ const SwTableNode* pTableNd = m_pStartNode->FindTableNode();
+ assert(pTableNd && "In which table is that box?");
+ SwTableSortBoxes& rSrtArr = const_cast<SwTableSortBoxes&>(pTableNd->GetTable().
+ GetTabSortBoxes());
+ SwTableBox* p = this; // error: &this
+ rSrtArr.insert( p ); // insert
+}
+
+SwTableBox::SwTableBox( SwTableBoxFormat* pFormat, const SwStartNode& rSttNd, SwTableLine *pUp )
+ : SwClient(nullptr)
+ , m_aLines()
+ , m_pStartNode(&rSttNd)
+ , m_pUpper(pUp)
+ , mnRowSpan(1)
+ , mbDummyFlag(false)
+ , mbDirectFormatting(false)
+{
+ CheckBoxFormat( pFormat )->Add( this );
+
+ // insert into the table
+ const SwTableNode* pTableNd = m_pStartNode->FindTableNode();
+ OSL_ENSURE( pTableNd, "In which table is the box?" );
+ SwTableSortBoxes& rSrtArr = const_cast<SwTableSortBoxes&>(pTableNd->GetTable().
+ GetTabSortBoxes());
+ SwTableBox* p = this; // error: &this
+ rSrtArr.insert( p ); // insert
+}
+
+void SwTableBox::RemoveFromTable()
+{
+ if (m_pStartNode) // box containing contents?
+ {
+ // remove from table
+ const SwTableNode* pTableNd = m_pStartNode->FindTableNode();
+ assert(pTableNd && "In which table is that box?");
+ SwTableSortBoxes& rSrtArr = const_cast<SwTableSortBoxes&>(pTableNd->GetTable().
+ GetTabSortBoxes());
+ SwTableBox *p = this; // error: &this
+ rSrtArr.erase( p ); // remove
+ m_pStartNode = nullptr; // clear it so this is only run once
+ }
+}
+
+SwTableBox::~SwTableBox()
+{
+ if (!GetFrameFormat()->GetDoc()->IsInDtor())
+ {
+ RemoveFromTable();
+ }
+
+ // the TabelleBox can be deleted if it's the last client of the FrameFormat
+ sw::BroadcastingModify* pMod = GetFrameFormat();
+ pMod->Remove( this ); // remove,
+ if( !pMod->HasWriterListeners() )
+ delete pMod; // and delete
+}
+
+SwTableBoxFormat* SwTableBox::CheckBoxFormat( SwTableBoxFormat* pFormat )
+{
+ // We might need to create a new format here, because the box must be
+ // added to the format solely if pFormat has a value or form.
+ if( SfxItemState::SET == pFormat->GetItemState( RES_BOXATR_VALUE, false ) ||
+ SfxItemState::SET == pFormat->GetItemState( RES_BOXATR_FORMULA, false ) )
+ {
+ SwTableBox* pOther = SwIterator<SwTableBox,SwFormat>( *pFormat ).First();
+ if( pOther )
+ {
+ SwTableBoxFormat* pNewFormat = pFormat->GetDoc()->MakeTableBoxFormat();
+ pNewFormat->LockModify();
+ *pNewFormat = *pFormat;
+
+ // Remove values and formulas
+ pNewFormat->ResetFormatAttr( RES_BOXATR_FORMULA, RES_BOXATR_VALUE );
+ pNewFormat->UnlockModify();
+
+ pFormat = pNewFormat;
+ }
+ }
+ return pFormat;
+}
+
+SwFrameFormat* SwTableBox::ClaimFrameFormat()
+{
+ // This method makes sure that this object is an exclusive SwTableBox client
+ // of an SwTableBoxFormat object
+ // If other SwTableBox objects currently listen to the same SwTableBoxFormat as
+ // this one, something needs to be done
+ SwTableBoxFormat *pRet = static_cast<SwTableBoxFormat*>(GetFrameFormat());
+ SwIterator<SwTableBox,SwFormat> aIter( *pRet );
+ for( SwTableBox* pLast = aIter.First(); pLast; pLast = aIter.Next() )
+ {
+ if ( pLast != this )
+ {
+ // Found another SwTableBox object
+ // create a new Format as a copy and assign me to it
+ // don't copy values and formulas
+ SwTableBoxFormat* pNewFormat = pRet->GetDoc()->MakeTableBoxFormat();
+ pNewFormat->LockModify();
+ *pNewFormat = *pRet;
+ pNewFormat->ResetFormatAttr( RES_BOXATR_FORMULA, RES_BOXATR_VALUE );
+ pNewFormat->UnlockModify();
+
+ // re-register SwCellFrame objects that know me
+ SwIterator<SwCellFrame,SwFormat> aFrameIter( *pRet );
+ for( SwCellFrame* pCell = aFrameIter.First(); pCell; pCell = aFrameIter.Next() )
+ if( pCell->GetTabBox() == this )
+ pCell->RegisterToFormat( *pNewFormat );
+
+ // re-register myself
+ pNewFormat->Add( this );
+ pRet = pNewFormat;
+ break;
+ }
+ }
+ return pRet;
+}
+
+void SwTableBox::ChgFrameFormat(SwTableBoxFormat* pNewFormat, bool bNeedToReregister)
+{
+ SwFrameFormat* pOld = GetFrameFormat();
+ // tdf#84635 We set bNeedToReregister=false to avoid a quadratic slowdown on loading large tables,
+ // and since we are creating the table for the first time, no re-registration is necessary.
+ // First, re-register the Frames.
+ if(bNeedToReregister)
+ pOld->CallSwClientNotify(sw::TableBoxFormatChanged(*pNewFormat, *this));
+ // Now, re-register self.
+ pNewFormat->Add(this);
+ if(!pOld->HasWriterListeners())
+ delete pOld;
+}
+
+// Return the name of this box. This is determined dynamically
+// resulting from the position in the lines/boxes/tables.
+void sw_GetTableBoxColStr( sal_uInt16 nCol, OUString& rNm )
+{
+ const sal_uInt16 coDiff = 52; // 'A'-'Z' 'a' - 'z'
+
+ do {
+ const sal_uInt16 nCalc = nCol % coDiff;
+ if( nCalc >= 26 )
+ rNm = OUStringChar( sal_Unicode('a' - 26 + nCalc) ) + rNm;
+ else
+ rNm = OUStringChar( sal_Unicode('A' + nCalc) ) + rNm;
+
+ nCol = nCol - nCalc;
+ if( 0 == nCol )
+ break;
+ nCol /= coDiff;
+ --nCol;
+ } while( true );
+}
+
+Point SwTableBox::GetCoordinates() const
+{
+ if( !m_pStartNode ) // box without content?
+ {
+ // search for the next first box?
+ return Point( 0, 0 );
+ }
+
+ const SwTable& rTable = m_pStartNode->FindTableNode()->GetTable();
+ sal_uInt16 nX, nY;
+ const SwTableBox* pBox = this;
+ do {
+ const SwTableLine* pLine = pBox->GetUpper();
+ // at the first level?
+ const SwTableLines* pLines = pLine->GetUpper()
+ ? &pLine->GetUpper()->GetTabLines() : &rTable.GetTabLines();
+
+ nY = pLines->GetPos( pLine ) + 1 ;
+ nX = pBox->GetUpper()->GetBoxPos( pBox ) + 1;
+ pBox = pLine->GetUpper();
+ } while( pBox );
+ return Point( nX, nY );
+}
+
+OUString SwTableBox::GetName() const
+{
+ if( !m_pStartNode ) // box without content?
+ {
+ // search for the next first box?
+ return OUString();
+ }
+
+ const SwTable& rTable = m_pStartNode->FindTableNode()->GetTable();
+ sal_uInt16 nPos;
+ OUString sNm, sTmp;
+ const SwTableBox* pBox = this;
+ do {
+ const SwTableLine* pLine = pBox->GetUpper();
+ // at the first level?
+ const SwTableLines* pLines = pLine->GetUpper()
+ ? &pLine->GetUpper()->GetTabLines() : &rTable.GetTabLines();
+
+ nPos = pLines->GetPos( pLine ) + 1;
+ sTmp = OUString::number( nPos );
+ if( !sNm.isEmpty() )
+ sNm = sTmp + "." + sNm;
+ else
+ sNm = sTmp;
+
+ nPos = pBox->GetUpper()->GetBoxPos( pBox );
+ sTmp = OUString::number(nPos + 1);
+ pBox = pLine->GetUpper();
+ if( nullptr != pBox )
+ sNm = sTmp + "." + sNm;
+ else
+ sw_GetTableBoxColStr( nPos, sNm );
+
+ } while( pBox );
+ return sNm;
+}
+
+bool SwTableBox::IsInHeadline( const SwTable* pTable ) const
+{
+ if( !GetUpper() ) // should only happen upon merge.
+ return false;
+
+ if( !pTable )
+ pTable = &m_pStartNode->FindTableNode()->GetTable();
+
+ const SwTableLine* pLine = GetUpper();
+ while( pLine->GetUpper() )
+ pLine = pLine->GetUpper()->GetUpper();
+
+ // Headerline?
+ return pTable->GetTabLines()[ 0 ] == pLine;
+}
+
+SwNodeOffset SwTableBox::GetSttIdx() const
+{
+ return m_pStartNode ? m_pStartNode->GetIndex() : SwNodeOffset(0);
+}
+
+bool SwTableBox::IsEmpty() const
+{
+ const SwStartNode *pSttNd = GetSttNd();
+ if( pSttNd &&
+ pSttNd->GetIndex() + 2 == pSttNd->EndOfSectionIndex() )
+ {
+ const SwContentNode *pCNd =
+ pSttNd->GetNodes()[pSttNd->GetIndex()+1]->GetContentNode();
+ if( pCNd && !pCNd->Len() )
+ return true;
+ }
+
+ return false;
+}
+
+ // retrieve information from the client
+bool SwTable::GetInfo( SfxPoolItem& rInfo ) const
+{
+ switch( rInfo.Which() )
+ {
+ case RES_AUTOFMT_DOCNODE:
+ {
+ const SwTableNode* pNode = GetTableNode();
+ if (pNode && &pNode->GetNodes() == static_cast<SwAutoFormatGetDocNode&>(rInfo).pNodes)
+ {
+ if (!m_TabSortContentBoxes.empty())
+ {
+ SwNodeIndex aIdx( *m_TabSortContentBoxes[0]->GetSttNd() );
+ GetFrameFormat()->GetDoc()->GetNodes().GoNext( &aIdx );
+ }
+ return false;
+ }
+ break;
+ }
+ case RES_FINDNEARESTNODE:
+ if( GetFrameFormat() &&
+ GetFrameFormat()->GetFormatAttr( RES_PAGEDESC ).GetPageDesc() &&
+ !m_TabSortContentBoxes.empty() &&
+ m_TabSortContentBoxes[0]->GetSttNd()->GetNodes().IsDocNodes() )
+ static_cast<SwFindNearestNode&>(rInfo).CheckNode( *
+ m_TabSortContentBoxes[0]->GetSttNd()->FindTableNode() );
+ break;
+
+ case RES_CONTENT_VISIBLE:
+ static_cast<SwPtrMsgPoolItem&>(rInfo).pObject = SwIterator<SwFrame,SwFormat>( *GetFrameFormat() ).First();
+ return false;
+ }
+ return true;
+}
+
+SwTable * SwTable::FindTable( SwFrameFormat const*const pFormat )
+{
+ return pFormat
+ ? SwIterator<SwTable,SwFormat>(*pFormat).First()
+ : nullptr;
+}
+
+SwTableNode* SwTable::GetTableNode() const
+{
+ return !GetTabSortBoxes().empty() ?
+ const_cast<SwTableNode*>(GetTabSortBoxes()[ 0 ]->GetSttNd()->FindTableNode()) :
+ m_pTableNode;
+}
+
+void SwTable::SetRefObject( SwServerObject* pObj )
+{
+ if( m_xRefObj.is() )
+ m_xRefObj->Closed();
+
+ m_xRefObj = pObj;
+}
+
+void SwTable::SetHTMLTableLayout(std::shared_ptr<SwHTMLTableLayout> const& r)
+{
+ m_xHTMLLayout = r;
+}
+
+static void ChgTextToNum( SwTableBox& rBox, const OUString& rText, const Color* pCol,
+ bool bChgAlign )
+{
+ SwNodeOffset nNdPos = rBox.IsValidNumTextNd();
+ ChgTextToNum( rBox,rText,pCol,bChgAlign,nNdPos);
+}
+void ChgTextToNum( SwTableBox& rBox, const OUString& rText, const Color* pCol,
+ bool bChgAlign, SwNodeOffset nNdPos )
+{
+
+ if( NODE_OFFSET_MAX == nNdPos )
+ return;
+
+ SwDoc* pDoc = rBox.GetFrameFormat()->GetDoc();
+ SwTextNode* pTNd = pDoc->GetNodes()[ nNdPos ]->GetTextNode();
+
+ // assign adjustment
+ if( bChgAlign )
+ {
+ const SfxPoolItem* pItem;
+ pItem = &pTNd->SwContentNode::GetAttr( RES_PARATR_ADJUST );
+ SvxAdjust eAdjust = static_cast<const SvxAdjustItem*>(pItem)->GetAdjust();
+ if( SvxAdjust::Left == eAdjust || SvxAdjust::Block == eAdjust )
+ {
+ SvxAdjustItem aAdjust( *static_cast<const SvxAdjustItem*>(pItem) );
+ aAdjust.SetAdjust( SvxAdjust::Right );
+ pTNd->SetAttr( aAdjust );
+ }
+ }
+
+ // assign color or save "user color"
+ const SvxColorItem* pColorItem = nullptr;
+ if( pTNd->GetpSwAttrSet() )
+ pColorItem = pTNd->GetpSwAttrSet()->GetItemIfSet( RES_CHRATR_COLOR, false );
+
+ const std::optional<Color>& pOldNumFormatColor = rBox.GetSaveNumFormatColor();
+ std::optional<Color> pNewUserColor;
+ if (pColorItem)
+ pNewUserColor = pColorItem->GetValue();
+
+ if( ( pNewUserColor && pOldNumFormatColor &&
+ *pNewUserColor == *pOldNumFormatColor ) ||
+ ( !pNewUserColor && !pOldNumFormatColor ))
+ {
+ // Keep the user color, set updated values, delete old NumFormatColor if needed
+ if( pCol )
+ // if needed, set the color
+ pTNd->SetAttr( SvxColorItem( *pCol, RES_CHRATR_COLOR ));
+ else if( pColorItem )
+ {
+ pNewUserColor = rBox.GetSaveUserColor();
+ if( pNewUserColor )
+ pTNd->SetAttr( SvxColorItem( *pNewUserColor, RES_CHRATR_COLOR ));
+ else
+ pTNd->ResetAttr( RES_CHRATR_COLOR );
+ }
+ }
+ else
+ {
+ // Save user color, set NumFormat color if needed, but never reset the color
+ rBox.SetSaveUserColor( pNewUserColor ? *pNewUserColor : std::optional<Color>() );
+
+ if( pCol )
+ // if needed, set the color
+ pTNd->SetAttr( SvxColorItem( *pCol, RES_CHRATR_COLOR ));
+
+ }
+ rBox.SetSaveNumFormatColor( pCol ? *pCol : std::optional<Color>() );
+
+ if( pTNd->GetText() != rText )
+ {
+ // Exchange text. Bugfix to keep Tabs (front and back!) and annotations (inword comment anchors)
+ const OUString& rOrig = pTNd->GetText();
+ sal_Int32 n;
+
+ for( n = 0; n < rOrig.getLength() && ('\x9' == rOrig[n] || CH_TXTATR_INWORD == rOrig[n]); ++n )
+ ;
+ for( ; n < rOrig.getLength() && '\x01' == rOrig[n]; ++n )
+ ;
+ SwIndex aIdx( pTNd, n );
+ for( n = rOrig.getLength(); n && ('\x9' == rOrig[--n] || CH_TXTATR_INWORD == rOrig[n]); )
+ ;
+ sal_Int32 nEndPos = n;
+ n -= aIdx.GetIndex() - 1;
+
+ // Reset DontExpand-Flags before exchange, to retrigger expansion
+ {
+ SwIndex aResetIdx( aIdx, n );
+ pTNd->DontExpandFormat( aResetIdx, false, false );
+ }
+
+ if( !pDoc->getIDocumentRedlineAccess().IsIgnoreRedline() && !pDoc->getIDocumentRedlineAccess().GetRedlineTable().empty() )
+ {
+ SwPaM aTemp(*pTNd, 0, *pTNd, rOrig.getLength());
+ pDoc->getIDocumentRedlineAccess().DeleteRedline(aTemp, true, RedlineType::Any);
+ }
+
+ // preserve comments inside of the number by deleting number portions starting from the back
+ sal_Int32 nCommentPos = pTNd->GetText().lastIndexOf( CH_TXTATR_INWORD, nEndPos );
+ while( nCommentPos > aIdx.GetIndex() )
+ {
+ pTNd->EraseText( SwIndex(pTNd, nCommentPos+1), nEndPos - nCommentPos, SwInsertFlags::EMPTYEXPAND );
+ // find the next non-sequential comment anchor
+ do
+ {
+ nEndPos = nCommentPos;
+ n = nEndPos - aIdx.GetIndex();
+ nCommentPos = pTNd->GetText().lastIndexOf( CH_TXTATR_INWORD, nEndPos );
+ --nEndPos;
+ }
+ while( nCommentPos > aIdx.GetIndex() && nCommentPos == nEndPos );
+ }
+
+ pTNd->EraseText( aIdx, n, SwInsertFlags::EMPTYEXPAND );
+ pTNd->InsertText( rText, aIdx, SwInsertFlags::EMPTYEXPAND );
+
+ if( pDoc->getIDocumentRedlineAccess().IsRedlineOn() )
+ {
+ SwPaM aTemp(*pTNd, 0, *pTNd, rText.getLength());
+ pDoc->getIDocumentRedlineAccess().AppendRedline(new SwRangeRedline(RedlineType::Insert, aTemp), true);
+ }
+ }
+
+ // assign vertical orientation
+ const SwFormatVertOrient* pVertOrientItem;
+ if( bChgAlign &&
+ ( !(pVertOrientItem = rBox.GetFrameFormat()->GetItemIfSet( RES_VERT_ORIENT )) ||
+ text::VertOrientation::TOP == pVertOrientItem->GetVertOrient() ))
+ {
+ rBox.GetFrameFormat()->SetFormatAttr( SwFormatVertOrient( 0, text::VertOrientation::BOTTOM ));
+ }
+
+}
+
+static void ChgNumToText( SwTableBox& rBox, sal_uLong nFormat )
+{
+ SwNodeOffset nNdPos = rBox.IsValidNumTextNd( false );
+ if( NODE_OFFSET_MAX == nNdPos )
+ return;
+
+ SwDoc* pDoc = rBox.GetFrameFormat()->GetDoc();
+ SwTextNode* pTNd = pDoc->GetNodes()[ nNdPos ]->GetTextNode();
+ bool bChgAlign = pDoc->IsInsTableAlignNum();
+
+ const Color * pCol = nullptr;
+ if( getSwDefaultTextFormat() != nFormat )
+ {
+ // special text format:
+ OUString sTmp;
+ const OUString sText( pTNd->GetText() );
+ pDoc->GetNumberFormatter()->GetOutputString( sText, nFormat, sTmp, &pCol );
+ if( sText != sTmp )
+ {
+ // exchange text
+ SwIndex aIdx( pTNd, sText.getLength() );
+ // Reset DontExpand-Flags before exchange, to retrigger expansion
+ pTNd->DontExpandFormat( aIdx, false, false );
+ aIdx = 0;
+ pTNd->EraseText( aIdx, SAL_MAX_INT32, SwInsertFlags::EMPTYEXPAND );
+ pTNd->InsertText( sTmp, aIdx, SwInsertFlags::EMPTYEXPAND );
+ }
+ }
+
+ const SfxItemSet* pAttrSet = pTNd->GetpSwAttrSet();
+
+ // assign adjustment
+ const SvxAdjustItem* pAdjustItem;
+ if( bChgAlign && pAttrSet &&
+ (pAdjustItem = pAttrSet->GetItemIfSet( RES_PARATR_ADJUST, false )) &&
+ SvxAdjust::Right == pAdjustItem->GetAdjust() )
+ {
+ pTNd->SetAttr( SvxAdjustItem( SvxAdjust::Left, RES_PARATR_ADJUST ) );
+ }
+
+ // assign color or save "user color"
+ const SvxColorItem* pColorItem = nullptr;
+ if( pAttrSet )
+ pColorItem = pAttrSet->GetItemIfSet( RES_CHRATR_COLOR, false );
+
+ const std::optional<Color>& pOldNumFormatColor = rBox.GetSaveNumFormatColor();
+ std::optional<Color> pNewUserColor;
+ if (pColorItem)
+ pNewUserColor = pColorItem->GetValue();
+
+ if( ( pNewUserColor && pOldNumFormatColor &&
+ *pNewUserColor == *pOldNumFormatColor ) ||
+ ( !pNewUserColor && !pOldNumFormatColor ))
+ {
+ // Keep the user color, set updated values, delete old NumFormatColor if needed
+ if( pCol )
+ // if needed, set the color
+ pTNd->SetAttr( SvxColorItem( *pCol, RES_CHRATR_COLOR ));
+ else if( pColorItem )
+ {
+ pNewUserColor = rBox.GetSaveUserColor();
+ if( pNewUserColor )
+ pTNd->SetAttr( SvxColorItem( *pNewUserColor, RES_CHRATR_COLOR ));
+ else
+ pTNd->ResetAttr( RES_CHRATR_COLOR );
+ }
+ }
+ else
+ {
+ // Save user color, set NumFormat color if needed, but never reset the color
+ rBox.SetSaveUserColor( pNewUserColor );
+
+ if( pCol )
+ // if needed, set the color
+ pTNd->SetAttr( SvxColorItem( *pCol, RES_CHRATR_COLOR ));
+
+ }
+ rBox.SetSaveNumFormatColor( pCol ? *pCol : std::optional<Color>() );
+
+ // assign vertical orientation
+ const SwFormatVertOrient* pVertOrientItem;
+ if( bChgAlign &&
+ (pVertOrientItem = rBox.GetFrameFormat()->GetItemIfSet( RES_VERT_ORIENT, false )) &&
+ text::VertOrientation::BOTTOM == pVertOrientItem->GetVertOrient() )
+ {
+ rBox.GetFrameFormat()->SetFormatAttr( SwFormatVertOrient( 0, text::VertOrientation::TOP ));
+ }
+
+}
+void SwTableBoxFormat::BoxAttributeChanged(SwTableBox& rBox, const SwTableBoxNumFormat* pNewFormat, const SwTableBoxFormula* pNewFormula, const SwTableBoxValue* pNewValue, sal_uLong nOldFormat)
+{
+ sal_uLong nNewFormat;
+ if(pNewFormat)
+ {
+ nNewFormat = pNewFormat->GetValue();
+ // new formatting
+ // is it newer or has the current been removed?
+ if( SfxItemState::SET != GetItemState(RES_BOXATR_VALUE, false))
+ pNewFormat = nullptr;
+ }
+ else
+ {
+ // fetch the current Item
+ pNewFormat = GetItemIfSet(RES_BOXATR_FORMAT, false);
+ nOldFormat = GetTableBoxNumFormat().GetValue();
+ nNewFormat = pNewFormat ? pNewFormat->GetValue() : nOldFormat;
+ }
+
+ // is it newer or has the current been removed?
+ if(pNewValue)
+ {
+ if(GetDoc()->GetNumberFormatter()->IsTextFormat(nNewFormat))
+ nOldFormat = 0;
+ else
+ {
+ if(SfxItemState::SET == GetItemState(RES_BOXATR_VALUE, false))
+ nOldFormat = getSwDefaultTextFormat();
+ else
+ nNewFormat = getSwDefaultTextFormat();
+ }
+ }
+
+ // Logic:
+ // Value change: -> "simulate" a format change!
+ // Format change:
+ // Text -> !Text or format change:
+ // - align right for horizontal alignment, if LEFT or JUSTIFIED
+ // - align bottom for vertical alignment, if TOP is set, or default
+ // - replace text (color? negative numbers RED?)
+ // !Text -> Text:
+ // - align left for horizontal alignment, if RIGHT
+ // - align top for vertical alignment, if BOTTOM is set
+ SvNumberFormatter* pNumFormatr = GetDoc()->GetNumberFormatter();
+ bool bNewIsTextFormat = pNumFormatr->IsTextFormat(nNewFormat);
+
+ if((!bNewIsTextFormat && nOldFormat != nNewFormat) || pNewFormula)
+ {
+ bool bIsNumFormat = false;
+ OUString aOrigText;
+ bool bChgText = true;
+ double fVal = 0;
+ if(!pNewValue)
+ pNewValue = GetItemIfSet(RES_BOXATR_VALUE, false);
+ if(!pNewValue)
+ {
+ // so far, no value has been set, so try to evaluate the content
+ SwNodeOffset nNdPos = rBox.IsValidNumTextNd();
+ if(NODE_OFFSET_MAX != nNdPos)
+ {
+ sal_uInt32 nTmpFormatIdx = nNewFormat;
+ OUString aText(GetDoc()->GetNodes()[nNdPos] ->GetTextNode()->GetRedlineText());
+ aOrigText = aText;
+ if(aText.isEmpty())
+ bChgText = false;
+ else
+ {
+ // Keep Tabs
+ lcl_TabToBlankAtSttEnd(aText);
+
+ // JP 22.04.98: Bug 49659 -
+ // Special casing for percent
+ if(SvNumFormatType::PERCENT == pNumFormatr->GetType(nNewFormat))
+ {
+ sal_uInt32 nTmpFormat = 0;
+ if(GetDoc()->IsNumberFormat(aText, nTmpFormat, fVal))
+ {
+ if(SvNumFormatType::NUMBER == pNumFormatr->GetType( nTmpFormat))
+ aText += "%";
+
+ bIsNumFormat = GetDoc()->IsNumberFormat(aText, nTmpFormatIdx, fVal);
+ }
+ }
+ else
+ bIsNumFormat = GetDoc()->IsNumberFormat(aText, nTmpFormatIdx, fVal);
+
+ if(bIsNumFormat)
+ {
+ // directly assign value - without Modify
+ bool bIsLockMod = IsModifyLocked();
+ LockModify();
+ SetFormatAttr(SwTableBoxValue(fVal));
+ if(!bIsLockMod)
+ UnlockModify();
+ }
+ }
+ }
+ }
+ else
+ {
+ fVal = pNewValue->GetValue();
+ bIsNumFormat = true;
+ }
+
+ // format contents with the new value assigned and write to paragraph
+ const Color* pCol = nullptr;
+ OUString sNewText;
+ bool bChangeFormat = true;
+ if(DBL_MAX == fVal)
+ {
+ sNewText = SwViewShell::GetShellRes()->aCalc_Error;
+ }
+ else
+ {
+ if(bIsNumFormat)
+ pNumFormatr->GetOutputString(fVal, nNewFormat, sNewText, &pCol);
+ else
+ {
+ // Original text could not be parsed as
+ // number/date/time/..., so keep the text.
+#if 0
+ // Actually the text should be formatted
+ // according to the format, which may include
+ // additional text from the format, for example
+ // in {0;-0;"BAD: "@}. But other places when
+ // entering a new value or changing text or
+ // changing to a different format of type Text
+ // don't do this (yet?).
+ pNumFormatr->GetOutputString(aOrigText, nNewFormat, sNewText, &pCol);
+#else
+ sNewText = aOrigText;
+#endif
+ // Remove the newly assigned numbering format as well if text actually exists.
+ // Exception: assume user-defined formats are always intentional.
+ if (bChgText && pNumFormatr->IsTextFormat(nOldFormat)
+ && !pNumFormatr->IsUserDefined(nNewFormat))
+ {
+ rBox.GetFrameFormat()->ResetFormatAttr(RES_BOXATR_FORMAT);
+ bChangeFormat = false;
+ }
+ }
+
+ if(!bChgText)
+ sNewText.clear();
+ }
+
+ // across all boxes
+ if (bChangeFormat)
+ ChgTextToNum(rBox, sNewText, pCol, GetDoc()->IsInsTableAlignNum());
+
+ }
+ else if(bNewIsTextFormat && nOldFormat != nNewFormat)
+ ChgNumToText(rBox, nNewFormat);
+}
+
+SwTableBox* SwTableBoxFormat::SwTableBoxFormat::GetTableBox()
+{
+ SwIterator<SwTableBox,SwFormat> aIter(*this);
+ auto pBox = aIter.First();
+ SAL_INFO_IF(!pBox, "sw.core", "no box found at format");
+ SAL_WARN_IF(pBox && aIter.Next(), "sw.core", "more than one box found at format");
+ return pBox;
+}
+
+// for detection of modifications (mainly TableBoxAttribute)
+void SwTableBoxFormat::SwClientNotify(const SwModify& rMod, const SfxHint& rHint)
+{
+ if(rHint.GetId() != SfxHintId::SwLegacyModify)
+ return;
+ auto pLegacy = static_cast<const sw::LegacyModifyHint*>(&rHint);
+ if(IsModifyLocked() || !GetDoc() || GetDoc()->IsInDtor())
+ {
+ SwFrameFormat::SwClientNotify(rMod, rHint);
+ return;
+ }
+ const SwTableBoxNumFormat* pNewFormat = nullptr;
+ const SwTableBoxFormula* pNewFormula = nullptr;
+ const SwTableBoxValue* pNewVal = nullptr;
+ sal_uLong nOldFormat = getSwDefaultTextFormat();
+
+ switch(pLegacy->m_pNew ? pLegacy->m_pNew->Which() : 0)
+ {
+ case RES_ATTRSET_CHG:
+ {
+ const SfxItemSet& rSet = *static_cast<const SwAttrSetChg*>(pLegacy->m_pNew)->GetChgSet();
+ pNewFormat = rSet.GetItemIfSet( RES_BOXATR_FORMAT, false);
+ if(pNewFormat)
+ nOldFormat = static_cast<const SwAttrSetChg*>(pLegacy->m_pOld)->GetChgSet()->Get(RES_BOXATR_FORMAT).GetValue();
+ pNewFormula = rSet.GetItemIfSet(RES_BOXATR_FORMULA, false);
+ pNewVal = rSet.GetItemIfSet(RES_BOXATR_VALUE, false);
+ break;
+ }
+ case RES_BOXATR_FORMAT:
+ pNewFormat = static_cast<const SwTableBoxNumFormat*>(pLegacy->m_pNew);
+ nOldFormat = static_cast<const SwTableBoxNumFormat*>(pLegacy->m_pOld)->GetValue();
+ break;
+ case RES_BOXATR_FORMULA:
+ pNewFormula = static_cast<const SwTableBoxFormula*>(pLegacy->m_pNew);
+ break;
+ case RES_BOXATR_VALUE:
+ pNewVal = static_cast<const SwTableBoxValue*>(pLegacy->m_pNew);
+ break;
+ }
+
+ // something changed and some BoxAttribut remained in the set!
+ if( pNewFormat || pNewFormula || pNewVal )
+ {
+ GetDoc()->getIDocumentFieldsAccess().SetFieldsDirty(true, nullptr, SwNodeOffset(0));
+
+ if(SfxItemState::SET == GetItemState(RES_BOXATR_FORMAT, false) ||
+ SfxItemState::SET == GetItemState(RES_BOXATR_VALUE, false) ||
+ SfxItemState::SET == GetItemState(RES_BOXATR_FORMULA, false) )
+ {
+ if(auto pBox = GetTableBox())
+ BoxAttributeChanged(*pBox, pNewFormat, pNewFormula, pNewVal, nOldFormat);
+ }
+ }
+ // call base class
+ SwFrameFormat::SwClientNotify(rMod, rHint);
+}
+
+bool SwTableBoxFormat::supportsFullDrawingLayerFillAttributeSet() const
+{
+ return false;
+}
+
+bool SwTableFormat::supportsFullDrawingLayerFillAttributeSet() const
+{
+ return false;
+}
+
+bool SwTableLineFormat::supportsFullDrawingLayerFillAttributeSet() const
+{
+ return false;
+}
+
+bool SwTableBox::HasNumContent( double& rNum, sal_uInt32& rFormatIndex,
+ bool& rIsEmptyTextNd ) const
+{
+ bool bRet = false;
+ SwNodeOffset nNdPos = IsValidNumTextNd();
+ if( NODE_OFFSET_MAX != nNdPos )
+ {
+ OUString aText( m_pStartNode->GetNodes()[ nNdPos ]->GetTextNode()->GetRedlineText() );
+ // Keep Tabs
+ lcl_TabToBlankAtSttEnd( aText );
+ rIsEmptyTextNd = aText.isEmpty();
+ SvNumberFormatter* pNumFormatr = GetFrameFormat()->GetDoc()->GetNumberFormatter();
+
+ if( const SwTableBoxNumFormat* pItem = GetFrameFormat()->GetItemIfSet( RES_BOXATR_FORMAT, false) )
+ {
+ rFormatIndex = pItem->GetValue();
+ // Special casing for percent
+ if( !rIsEmptyTextNd && SvNumFormatType::PERCENT == pNumFormatr->GetType( rFormatIndex ))
+ {
+ sal_uInt32 nTmpFormat = 0;
+ if( GetFrameFormat()->GetDoc()->IsNumberFormat( aText, nTmpFormat, rNum ) &&
+ SvNumFormatType::NUMBER == pNumFormatr->GetType( nTmpFormat ))
+ aText += "%";
+ }
+ }
+ else
+ rFormatIndex = 0;
+
+ bRet = GetFrameFormat()->GetDoc()->IsNumberFormat( aText, rFormatIndex, rNum );
+ }
+ else
+ rIsEmptyTextNd = false;
+ return bRet;
+}
+
+bool SwTableBox::IsNumberChanged() const
+{
+ bool bRet = true;
+
+ if( SfxItemState::SET == GetFrameFormat()->GetItemState( RES_BOXATR_FORMULA, false ))
+ {
+ const SwTableBoxNumFormat *pNumFormat = GetFrameFormat()->GetItemIfSet( RES_BOXATR_FORMAT, false );
+ const SwTableBoxValue *pValue = GetFrameFormat()->GetItemIfSet( RES_BOXATR_VALUE, false );
+
+ SwNodeOffset nNdPos;
+ if( pNumFormat && pValue && NODE_OFFSET_MAX != ( nNdPos = IsValidNumTextNd() ) )
+ {
+ OUString sNewText, sOldText( m_pStartNode->GetNodes()[ nNdPos ]->
+ GetTextNode()->GetRedlineText() );
+ lcl_DelTabsAtSttEnd( sOldText );
+
+ const Color* pCol = nullptr;
+ GetFrameFormat()->GetDoc()->GetNumberFormatter()->GetOutputString(
+ pValue->GetValue(), pNumFormat->GetValue(), sNewText, &pCol );
+
+ bRet = sNewText != sOldText ||
+ !( ( !pCol && !GetSaveNumFormatColor() ) ||
+ ( pCol && GetSaveNumFormatColor() &&
+ *pCol == *GetSaveNumFormatColor() ));
+ }
+ }
+ return bRet;
+}
+
+SwNodeOffset SwTableBox::IsValidNumTextNd( bool bCheckAttr ) const
+{
+ SwNodeOffset nPos = NODE_OFFSET_MAX;
+ if( m_pStartNode )
+ {
+ SwNodeIndex aIdx( *m_pStartNode );
+ SwNodeOffset nIndex = aIdx.GetIndex();
+ const SwNodeOffset nIndexEnd = m_pStartNode->GetNodes()[ nIndex ]->EndOfSectionIndex();
+ const SwTextNode *pTextNode = nullptr;
+ while( ++nIndex < nIndexEnd )
+ {
+ const SwNode* pNode = m_pStartNode->GetNodes()[nIndex];
+ if( pNode->IsTableNode() )
+ {
+ pTextNode = nullptr;
+ break;
+ }
+ if( pNode->IsTextNode() )
+ {
+ if( pTextNode )
+ {
+ pTextNode = nullptr;
+ break;
+ }
+ else
+ {
+ pTextNode = pNode->GetTextNode();
+ nPos = nIndex;
+ }
+ }
+ }
+ if( pTextNode )
+ {
+ if( bCheckAttr )
+ {
+ const SwpHints* pHts = pTextNode->GetpSwpHints();
+ // do some tests if there's only text in the node!
+ // Flys/fields/...
+ if( pHts )
+ {
+ sal_Int32 nNextSetField = 0;
+ for( size_t n = 0; n < pHts->Count(); ++n )
+ {
+ const SwTextAttr* pAttr = pHts->Get(n);
+ if( RES_TXTATR_NOEND_BEGIN <= pAttr->Which() )
+ {
+ if ( (pAttr->GetStart() == nNextSetField)
+ && (pAttr->Which() == RES_TXTATR_FIELD))
+ {
+ // #i104949# hideous hack for report builder:
+ // it inserts hidden variable-set fields at
+ // the beginning of para in cell, but they
+ // should not turn cell into text cell
+ const SwField* pField = pAttr->GetFormatField().GetField();
+ if (pField &&
+ (pField->GetTypeId() == SwFieldTypesEnum::Set) &&
+ (0 != (static_cast<SwSetExpField const*>
+ (pField)->GetSubType() &
+ nsSwExtendedSubType::SUB_INVISIBLE)))
+ {
+ nNextSetField = pAttr->GetStart() + 1;
+ continue;
+ }
+ }
+ else if( RES_TXTATR_ANNOTATION == pAttr->Which() ||
+ RES_TXTATR_FTN == pAttr->Which() )
+ {
+ continue;
+ }
+ nPos = NODE_OFFSET_MAX;
+ break;
+ }
+ }
+ }
+ }
+ }
+ else
+ nPos = NODE_OFFSET_MAX;
+ }
+ return nPos;
+}
+
+// is this a Formula box or one with numeric content (AutoSum)
+sal_uInt16 SwTableBox::IsFormulaOrValueBox() const
+{
+ sal_uInt16 nWhich = 0;
+ const SwTextNode* pTNd;
+ SwFrameFormat* pFormat = GetFrameFormat();
+ if( SfxItemState::SET == pFormat->GetItemState( RES_BOXATR_FORMULA, false ))
+ nWhich = RES_BOXATR_FORMULA;
+ else if( SfxItemState::SET == pFormat->GetItemState( RES_BOXATR_VALUE, false ) &&
+ !pFormat->GetDoc()->GetNumberFormatter()->IsTextFormat(
+ pFormat->GetTableBoxNumFormat().GetValue() ))
+ nWhich = RES_BOXATR_VALUE;
+ else if( m_pStartNode && m_pStartNode->GetIndex() + 2 == m_pStartNode->EndOfSectionIndex()
+ && nullptr != ( pTNd = m_pStartNode->GetNodes()[ m_pStartNode->GetIndex() + 1 ]
+ ->GetTextNode() ) && pTNd->GetText().isEmpty())
+ nWhich = USHRT_MAX;
+
+ return nWhich;
+}
+
+void SwTableBox::ActualiseValueBox()
+{
+ SwFrameFormat* pFormat = GetFrameFormat();
+ const SwTableBoxNumFormat *pFormatItem = pFormat->GetItemIfSet( RES_BOXATR_FORMAT, true );
+ if (!pFormatItem)
+ return;
+ const SwTableBoxValue *pValItem = pFormat->GetItemIfSet( RES_BOXATR_VALUE );
+ if (!pValItem)
+ return;
+
+ const sal_uLong nFormatId = pFormatItem->GetValue();
+ SwNodeOffset nNdPos = NODE_OFFSET_MAX;
+ SvNumberFormatter* pNumFormatr = pFormat->GetDoc()->GetNumberFormatter();
+
+ if( !pNumFormatr->IsTextFormat( nFormatId ) &&
+ NODE_OFFSET_MAX != (nNdPos = IsValidNumTextNd()) )
+ {
+ double fVal = pValItem->GetValue();
+ const Color* pCol = nullptr;
+ OUString sNewText;
+ pNumFormatr->GetOutputString( fVal, nFormatId, sNewText, &pCol );
+
+ const OUString& rText = m_pStartNode->GetNodes()[ nNdPos ]->GetTextNode()->GetText();
+ if( rText != sNewText )
+ ChgTextToNum( *this, sNewText, pCol, false ,nNdPos);
+ }
+}
+
+struct SwTableCellInfo::Impl
+{
+ const SwTable * m_pTable;
+ const SwCellFrame * m_pCellFrame;
+ const SwTabFrame * m_pTabFrame;
+ typedef o3tl::sorted_vector<const SwTableBox *> TableBoxes_t;
+ TableBoxes_t m_HandledTableBoxes;
+
+public:
+ Impl()
+ : m_pTable(nullptr), m_pCellFrame(nullptr), m_pTabFrame(nullptr)
+ {
+ }
+
+ void setTable(const SwTable * pTable)
+ {
+ m_pTable = pTable;
+ SwFrameFormat * pFrameFormat = m_pTable->GetFrameFormat();
+ m_pTabFrame = SwIterator<SwTabFrame,SwFormat>(*pFrameFormat).First();
+ if (m_pTabFrame && m_pTabFrame->IsFollow())
+ m_pTabFrame = m_pTabFrame->FindMaster(true);
+ }
+
+ const SwCellFrame * getCellFrame() const { return m_pCellFrame; }
+
+ const SwFrame * getNextFrameInTable(const SwFrame * pFrame);
+ const SwCellFrame * getNextCellFrame(const SwFrame * pFrame);
+ const SwCellFrame * getNextTableBoxsCellFrame(const SwFrame * pFrame);
+ bool getNext();
+};
+
+const SwFrame * SwTableCellInfo::Impl::getNextFrameInTable(const SwFrame * pFrame)
+{
+ const SwFrame * pResult = nullptr;
+
+ if (((! pFrame->IsTabFrame()) || pFrame == m_pTabFrame) && pFrame->GetLower())
+ pResult = pFrame->GetLower();
+ else if (pFrame->GetNext())
+ pResult = pFrame->GetNext();
+ else
+ {
+ while (pFrame->GetUpper() != nullptr)
+ {
+ pFrame = pFrame->GetUpper();
+
+ if (pFrame->IsTabFrame())
+ {
+ m_pTabFrame = static_cast<const SwTabFrame *>(pFrame)->GetFollow();
+ pResult = m_pTabFrame;
+ break;
+ }
+ else if (pFrame->GetNext())
+ {
+ pResult = pFrame->GetNext();
+ break;
+ }
+ }
+ }
+
+ return pResult;
+}
+
+const SwCellFrame * SwTableCellInfo::Impl::getNextCellFrame(const SwFrame * pFrame)
+{
+ const SwCellFrame * pResult = nullptr;
+
+ while ((pFrame = getNextFrameInTable(pFrame)) != nullptr)
+ {
+ if (pFrame->IsCellFrame())
+ {
+ pResult = static_cast<const SwCellFrame *>(pFrame);
+ break;
+ }
+ }
+
+ return pResult;
+}
+
+const SwCellFrame * SwTableCellInfo::Impl::getNextTableBoxsCellFrame(const SwFrame * pFrame)
+{
+ const SwCellFrame * pResult = nullptr;
+
+ while ((pFrame = getNextCellFrame(pFrame)) != nullptr)
+ {
+ const SwCellFrame * pCellFrame = static_cast<const SwCellFrame *>(pFrame);
+ const SwTableBox * pTabBox = pCellFrame->GetTabBox();
+ auto aIt = m_HandledTableBoxes.insert(pTabBox);
+ if (aIt.second)
+ {
+ pResult = pCellFrame;
+ break;
+ }
+ }
+
+ return pResult;
+}
+
+const SwCellFrame * SwTableCellInfo::getCellFrame() const
+{
+ return m_pImpl->getCellFrame();
+}
+
+bool SwTableCellInfo::Impl::getNext()
+{
+ if (m_pCellFrame == nullptr)
+ {
+ if (m_pTabFrame != nullptr)
+ m_pCellFrame = Impl::getNextTableBoxsCellFrame(m_pTabFrame);
+ }
+ else
+ m_pCellFrame = Impl::getNextTableBoxsCellFrame(m_pCellFrame);
+
+ return m_pCellFrame != nullptr;
+}
+
+SwTableCellInfo::SwTableCellInfo(const SwTable * pTable)
+ : m_pImpl(std::make_unique<Impl>())
+{
+ m_pImpl->setTable(pTable);
+}
+
+SwTableCellInfo::~SwTableCellInfo()
+{
+}
+
+bool SwTableCellInfo::getNext()
+{
+ return m_pImpl->getNext();
+}
+
+SwRect SwTableCellInfo::getRect() const
+{
+ SwRect aRet;
+
+ if (getCellFrame() != nullptr)
+ aRet = getCellFrame()->getFrameArea();
+
+ return aRet;
+}
+
+const SwTableBox * SwTableCellInfo::getTableBox() const
+{
+ const SwTableBox * pRet = nullptr;
+
+ if (getCellFrame() != nullptr)
+ pRet = getCellFrame()->GetTabBox();
+
+ return pRet;
+}
+
+void SwTable::RegisterToFormat( SwFormat& rFormat )
+{
+ rFormat.Add( this );
+}
+
+bool SwTable::HasLayout() const
+{
+ const SwFrameFormat* pFrameFormat = GetFrameFormat();
+ //a table in a clipboard document doesn't have any layout information
+ return pFrameFormat && SwIterator<SwTabFrame,SwFormat>(*pFrameFormat).First();
+}
+
+void SwTableBox::RegisterToFormat( SwFormat& rFormat )
+{
+ rFormat.Add( this );
+}
+
+// free's any remaining child objects
+SwTableLines::~SwTableLines()
+{
+ for ( const_iterator it = begin(); it != end(); ++it )
+ delete *it;
+}
+
+/* vim:set shiftwidth=4 softtabstop=4 expandtab: */