Reduce potential confusion with the global tools namespace. Will hopefully make it possible to remove the annoying initial :: when referring to the global tools namespace. Unless we have even more tools subnamespaces somewhere. Thorsten said it was OK. Change-Id: Id088dfe8f4244cb79df9aa988995b31a1758c996 Reviewed-on: https://gerrit.libreoffice.org/42644 Tested-by: Jenkins <ci@libreoffice.org> Reviewed-by: Tor Lillqvist <tml@collabora.com>
1307 lines
46 KiB
C++
1307 lines
46 KiB
C++
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
|
|
/*
|
|
* This file is part of the LibreOffice project.
|
|
*
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
*
|
|
* This file incorporates work covered by the following license notice:
|
|
*
|
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
|
* contributor license agreements. See the NOTICE file distributed
|
|
* with this work for additional information regarding copyright
|
|
* ownership. The ASF licenses this file to you under the Apache
|
|
* License, Version 2.0 (the "License"); you may not use this file
|
|
* except in compliance with the License. You may obtain a copy of
|
|
* the License at http://www.apache.org/licenses/LICENSE-2.0 .
|
|
*/
|
|
|
|
#include <svx/framelinkarray.hxx>
|
|
|
|
#include <math.h>
|
|
#include <vector>
|
|
#include <algorithm>
|
|
#include <vcl/outdev.hxx>
|
|
#include <drawinglayer/primitive2d/borderlineprimitive2d.hxx>
|
|
#include <basegfx/matrix/b2dhommatrixtools.hxx>
|
|
|
|
namespace svx {
|
|
namespace frame {
|
|
|
|
class Cell
|
|
{
|
|
private:
|
|
Style maLeft;
|
|
Style maRight;
|
|
Style maTop;
|
|
Style maBottom;
|
|
Style maTLBR;
|
|
Style maBLTR;
|
|
|
|
public:
|
|
long mnAddLeft;
|
|
long mnAddRight;
|
|
long mnAddTop;
|
|
long mnAddBottom;
|
|
|
|
SvxRotateMode meRotMode;
|
|
double mfOrientation;
|
|
basegfx::B2DHomMatrix maCoordinateSystem;
|
|
size_t maCellIndex;
|
|
|
|
bool mbMergeOrig;
|
|
bool mbOverlapX;
|
|
bool mbOverlapY;
|
|
|
|
public:
|
|
explicit Cell();
|
|
|
|
void SetStyleLeft(const Style& rStyle) { maLeft = rStyle; maLeft.SetUsingCell(this); }
|
|
void SetStyleRight(const Style& rStyle) { maRight = rStyle; maRight.SetUsingCell(this); }
|
|
void SetStyleTop(const Style& rStyle) { maTop = rStyle; maTop.SetUsingCell(this); }
|
|
void SetStyleBottom(const Style& rStyle) { maBottom = rStyle; maBottom.SetUsingCell(this); }
|
|
void SetStyleTLBR(const Style& rStyle) { maTLBR = rStyle; maTLBR.SetUsingCell(this); }
|
|
void SetStyleBLTR(const Style& rStyle) { maBLTR = rStyle; maBLTR.SetUsingCell(this); }
|
|
|
|
const Style& GetStyleLeft() const { return maLeft; }
|
|
const Style& GetStyleRight() const { return maRight; }
|
|
const Style& GetStyleTop() const { return maTop; }
|
|
const Style& GetStyleBottom() const { return maBottom; }
|
|
const Style& GetStyleTLBR() const { return maTLBR; }
|
|
const Style& GetStyleBLTR() const { return maBLTR; }
|
|
|
|
bool IsMerged() const { return mbMergeOrig || mbOverlapX || mbOverlapY; }
|
|
bool IsRotated() const { return mfOrientation != 0.0; }
|
|
|
|
void MirrorSelfX();
|
|
|
|
basegfx::B2DHomMatrix CreateCoordinateSystem(const Array& rArray) const;
|
|
size_t GetCellIndex(const Array& rArray) const;
|
|
};
|
|
|
|
typedef std::vector< long > LongVec;
|
|
typedef std::vector< Cell > CellVec;
|
|
|
|
size_t Cell::GetCellIndex(const Array& rArray) const
|
|
{
|
|
if(static_cast<size_t>(-1) == maCellIndex)
|
|
{
|
|
rArray.AddCellIndices();
|
|
}
|
|
|
|
return maCellIndex;
|
|
}
|
|
|
|
basegfx::B2DHomMatrix Cell::CreateCoordinateSystem(const Array& rArray) const
|
|
{
|
|
if(!maCoordinateSystem.isIdentity())
|
|
{
|
|
return maCoordinateSystem;
|
|
}
|
|
|
|
const size_t nCellIndex(GetCellIndex(rArray));
|
|
|
|
if(static_cast<size_t>(-1) != nCellIndex)
|
|
{
|
|
const basegfx::B2DRange aRange(rArray.GetCellRange(nCellIndex));
|
|
|
|
if(!aRange.isEmpty())
|
|
{
|
|
basegfx::B2DPoint aOrigin(aRange.getMinimum());
|
|
basegfx::B2DVector aX(aRange.getWidth(), 0.0);
|
|
basegfx::B2DVector aY(0.0, aRange.getHeight());
|
|
|
|
if (IsRotated() && SvxRotateMode::SVX_ROTATE_MODE_STANDARD != meRotMode)
|
|
{
|
|
// when rotated, adapt values. Get Skew (cos/sin == 1/tan)
|
|
const double fSkew(aY.getY() * (cos(mfOrientation) / sin(mfOrientation)));
|
|
|
|
switch (meRotMode)
|
|
{
|
|
case SvxRotateMode::SVX_ROTATE_MODE_TOP:
|
|
// shear Y-Axis
|
|
aY.setX(-fSkew);
|
|
break;
|
|
case SvxRotateMode::SVX_ROTATE_MODE_CENTER:
|
|
// shear origin half, Y full
|
|
aOrigin.setX(aOrigin.getX() + (fSkew * 0.5));
|
|
aY.setX(-fSkew);
|
|
break;
|
|
case SvxRotateMode::SVX_ROTATE_MODE_BOTTOM:
|
|
// shear origin full, Y full
|
|
aOrigin.setX(aOrigin.getX() + fSkew);
|
|
aY.setX(-fSkew);
|
|
break;
|
|
default: // SvxRotateMode::SVX_ROTATE_MODE_STANDARD, already excluded above
|
|
break;
|
|
}
|
|
}
|
|
|
|
// use column vectors as coordinate axes, homogen column for translation
|
|
const_cast<Cell*>(this)->maCoordinateSystem = basegfx::utils::createCoordinateSystemTransform(aOrigin, aX, aY);
|
|
}
|
|
}
|
|
|
|
return maCoordinateSystem;
|
|
}
|
|
|
|
Cell::Cell() :
|
|
mnAddLeft( 0 ),
|
|
mnAddRight( 0 ),
|
|
mnAddTop( 0 ),
|
|
mnAddBottom( 0 ),
|
|
meRotMode(SvxRotateMode::SVX_ROTATE_MODE_STANDARD ),
|
|
mfOrientation( 0.0 ),
|
|
maCoordinateSystem(),
|
|
maCellIndex(static_cast<size_t>(-1)),
|
|
mbMergeOrig( false ),
|
|
mbOverlapX( false ),
|
|
mbOverlapY( false )
|
|
{
|
|
}
|
|
|
|
void Cell::MirrorSelfX()
|
|
{
|
|
std::swap( maLeft, maRight );
|
|
std::swap( mnAddLeft, mnAddRight );
|
|
maLeft.MirrorSelf();
|
|
maRight.MirrorSelf();
|
|
mfOrientation = -mfOrientation;
|
|
maCoordinateSystem.identity();
|
|
maCellIndex = static_cast<size_t>(-1);
|
|
}
|
|
|
|
|
|
void lclRecalcCoordVec( LongVec& rCoords, const LongVec& rSizes )
|
|
{
|
|
DBG_ASSERT( rCoords.size() == rSizes.size() + 1, "lclRecalcCoordVec - inconsistent vectors" );
|
|
LongVec::iterator aCIt = rCoords.begin();
|
|
LongVec::const_iterator aSIt = rSizes.begin(), aSEnd = rSizes.end();
|
|
for( ; aSIt != aSEnd; ++aCIt, ++aSIt )
|
|
*(aCIt + 1) = *aCIt + *aSIt;
|
|
}
|
|
|
|
void lclSetMergedRange( CellVec& rCells, size_t nWidth, size_t nFirstCol, size_t nFirstRow, size_t nLastCol, size_t nLastRow )
|
|
{
|
|
for( size_t nCol = nFirstCol; nCol <= nLastCol; ++nCol )
|
|
{
|
|
for( size_t nRow = nFirstRow; nRow <= nLastRow; ++nRow )
|
|
{
|
|
Cell& rCell = rCells[ nRow * nWidth + nCol ];
|
|
rCell.mbMergeOrig = false;
|
|
rCell.mbOverlapX = nCol > nFirstCol;
|
|
rCell.mbOverlapY = nRow > nFirstRow;
|
|
}
|
|
}
|
|
rCells[ nFirstRow * nWidth + nFirstCol ].mbMergeOrig = true;
|
|
}
|
|
|
|
|
|
static const Style OBJ_STYLE_NONE;
|
|
static const Cell OBJ_CELL_NONE;
|
|
|
|
struct ArrayImpl
|
|
{
|
|
CellVec maCells;
|
|
LongVec maWidths;
|
|
LongVec maHeights;
|
|
mutable LongVec maXCoords;
|
|
mutable LongVec maYCoords;
|
|
size_t mnWidth;
|
|
size_t mnHeight;
|
|
size_t mnFirstClipCol;
|
|
size_t mnFirstClipRow;
|
|
size_t mnLastClipCol;
|
|
size_t mnLastClipRow;
|
|
mutable bool mbXCoordsDirty;
|
|
mutable bool mbYCoordsDirty;
|
|
bool mbMayHaveCellRotation;
|
|
|
|
explicit ArrayImpl( size_t nWidth, size_t nHeight );
|
|
|
|
bool IsValidPos( size_t nCol, size_t nRow ) const
|
|
{ return (nCol < mnWidth) && (nRow < mnHeight); }
|
|
size_t GetIndex( size_t nCol, size_t nRow ) const
|
|
{ return nRow * mnWidth + nCol; }
|
|
|
|
const Cell& GetCell( size_t nCol, size_t nRow ) const;
|
|
Cell& GetCellAcc( size_t nCol, size_t nRow );
|
|
|
|
size_t GetMergedFirstCol( size_t nCol, size_t nRow ) const;
|
|
size_t GetMergedFirstRow( size_t nCol, size_t nRow ) const;
|
|
size_t GetMergedLastCol( size_t nCol, size_t nRow ) const;
|
|
size_t GetMergedLastRow( size_t nCol, size_t nRow ) const;
|
|
|
|
const Cell& GetMergedOriginCell( size_t nCol, size_t nRow ) const;
|
|
|
|
bool IsMergedOverlappedLeft( size_t nCol, size_t nRow ) const;
|
|
bool IsMergedOverlappedRight( size_t nCol, size_t nRow ) const;
|
|
bool IsMergedOverlappedTop( size_t nCol, size_t nRow ) const;
|
|
bool IsMergedOverlappedBottom( size_t nCol, size_t nRow ) const;
|
|
|
|
bool IsInClipRange( size_t nCol, size_t nRow ) const;
|
|
bool IsColInClipRange( size_t nCol ) const;
|
|
bool IsRowInClipRange( size_t nRow ) const;
|
|
|
|
size_t GetMirrorCol( size_t nCol ) const { return mnWidth - nCol - 1; }
|
|
|
|
long GetColPosition( size_t nCol ) const;
|
|
long GetRowPosition( size_t nRow ) const;
|
|
|
|
bool HasCellRotation() const;
|
|
};
|
|
|
|
ArrayImpl::ArrayImpl( size_t nWidth, size_t nHeight ) :
|
|
mnWidth( nWidth ),
|
|
mnHeight( nHeight ),
|
|
mnFirstClipCol( 0 ),
|
|
mnFirstClipRow( 0 ),
|
|
mnLastClipCol( nWidth - 1 ),
|
|
mnLastClipRow( nHeight - 1 ),
|
|
mbXCoordsDirty( false ),
|
|
mbYCoordsDirty( false ),
|
|
mbMayHaveCellRotation( false )
|
|
{
|
|
// default-construct all vectors
|
|
maCells.resize( mnWidth * mnHeight );
|
|
maWidths.resize( mnWidth, 0L );
|
|
maHeights.resize( mnHeight, 0L );
|
|
maXCoords.resize( mnWidth + 1, 0L );
|
|
maYCoords.resize( mnHeight + 1, 0L );
|
|
}
|
|
|
|
const Cell& ArrayImpl::GetCell( size_t nCol, size_t nRow ) const
|
|
{
|
|
return IsValidPos( nCol, nRow ) ? maCells[ GetIndex( nCol, nRow ) ] : OBJ_CELL_NONE;
|
|
}
|
|
|
|
Cell& ArrayImpl::GetCellAcc( size_t nCol, size_t nRow )
|
|
{
|
|
static Cell aDummy;
|
|
return IsValidPos( nCol, nRow ) ? maCells[ GetIndex( nCol, nRow ) ] : aDummy;
|
|
}
|
|
|
|
size_t ArrayImpl::GetMergedFirstCol( size_t nCol, size_t nRow ) const
|
|
{
|
|
size_t nFirstCol = nCol;
|
|
while( (nFirstCol > 0) && GetCell( nFirstCol, nRow ).mbOverlapX ) --nFirstCol;
|
|
return nFirstCol;
|
|
}
|
|
|
|
size_t ArrayImpl::GetMergedFirstRow( size_t nCol, size_t nRow ) const
|
|
{
|
|
size_t nFirstRow = nRow;
|
|
while( (nFirstRow > 0) && GetCell( nCol, nFirstRow ).mbOverlapY ) --nFirstRow;
|
|
return nFirstRow;
|
|
}
|
|
|
|
size_t ArrayImpl::GetMergedLastCol( size_t nCol, size_t nRow ) const
|
|
{
|
|
size_t nLastCol = nCol + 1;
|
|
while( (nLastCol < mnWidth) && GetCell( nLastCol, nRow ).mbOverlapX ) ++nLastCol;
|
|
return nLastCol - 1;
|
|
}
|
|
|
|
size_t ArrayImpl::GetMergedLastRow( size_t nCol, size_t nRow ) const
|
|
{
|
|
size_t nLastRow = nRow + 1;
|
|
while( (nLastRow < mnHeight) && GetCell( nCol, nLastRow ).mbOverlapY ) ++nLastRow;
|
|
return nLastRow - 1;
|
|
}
|
|
|
|
const Cell& ArrayImpl::GetMergedOriginCell( size_t nCol, size_t nRow ) const
|
|
{
|
|
return GetCell( GetMergedFirstCol( nCol, nRow ), GetMergedFirstRow( nCol, nRow ) );
|
|
}
|
|
|
|
bool ArrayImpl::IsMergedOverlappedLeft( size_t nCol, size_t nRow ) const
|
|
{
|
|
const Cell& rCell = GetCell( nCol, nRow );
|
|
return rCell.mbOverlapX || (rCell.mnAddLeft > 0);
|
|
}
|
|
|
|
bool ArrayImpl::IsMergedOverlappedRight( size_t nCol, size_t nRow ) const
|
|
{
|
|
return GetCell( nCol + 1, nRow ).mbOverlapX || (GetCell( nCol, nRow ).mnAddRight > 0);
|
|
}
|
|
|
|
bool ArrayImpl::IsMergedOverlappedTop( size_t nCol, size_t nRow ) const
|
|
{
|
|
const Cell& rCell = GetCell( nCol, nRow );
|
|
return rCell.mbOverlapY || (rCell.mnAddTop > 0);
|
|
}
|
|
|
|
bool ArrayImpl::IsMergedOverlappedBottom( size_t nCol, size_t nRow ) const
|
|
{
|
|
return GetCell( nCol, nRow + 1 ).mbOverlapY || (GetCell( nCol, nRow ).mnAddBottom > 0);
|
|
}
|
|
|
|
bool ArrayImpl::IsColInClipRange( size_t nCol ) const
|
|
{
|
|
return (mnFirstClipCol <= nCol) && (nCol <= mnLastClipCol);
|
|
}
|
|
|
|
bool ArrayImpl::IsRowInClipRange( size_t nRow ) const
|
|
{
|
|
return (mnFirstClipRow <= nRow) && (nRow <= mnLastClipRow);
|
|
}
|
|
|
|
bool ArrayImpl::IsInClipRange( size_t nCol, size_t nRow ) const
|
|
{
|
|
return IsColInClipRange( nCol ) && IsRowInClipRange( nRow );
|
|
}
|
|
|
|
long ArrayImpl::GetColPosition( size_t nCol ) const
|
|
{
|
|
if( mbXCoordsDirty )
|
|
{
|
|
lclRecalcCoordVec( maXCoords, maWidths );
|
|
mbXCoordsDirty = false;
|
|
}
|
|
return maXCoords[ nCol ];
|
|
}
|
|
|
|
long ArrayImpl::GetRowPosition( size_t nRow ) const
|
|
{
|
|
if( mbYCoordsDirty )
|
|
{
|
|
lclRecalcCoordVec( maYCoords, maHeights );
|
|
mbYCoordsDirty = false;
|
|
}
|
|
return maYCoords[ nRow ];
|
|
}
|
|
|
|
bool ArrayImpl::HasCellRotation() const
|
|
{
|
|
// check cell array
|
|
for (const auto& aCell : maCells)
|
|
{
|
|
if (aCell.IsRotated())
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
class MergedCellIterator
|
|
{
|
|
public:
|
|
explicit MergedCellIterator( const Array& rArray, size_t nCol, size_t nRow );
|
|
|
|
bool Is() const { return (mnCol <= mnLastCol) && (mnRow <= mnLastRow); }
|
|
size_t Col() const { return mnCol; }
|
|
size_t Row() const { return mnRow; }
|
|
|
|
MergedCellIterator& operator++();
|
|
|
|
private:
|
|
size_t mnFirstCol;
|
|
size_t mnFirstRow;
|
|
size_t mnLastCol;
|
|
size_t mnLastRow;
|
|
size_t mnCol;
|
|
size_t mnRow;
|
|
};
|
|
|
|
|
|
MergedCellIterator::MergedCellIterator( const Array& rArray, size_t nCol, size_t nRow )
|
|
{
|
|
DBG_ASSERT( rArray.IsMerged( nCol, nRow ), "svx::frame::MergedCellIterator::MergedCellIterator - not in merged range" );
|
|
rArray.GetMergedRange( mnFirstCol, mnFirstRow, mnLastCol, mnLastRow, nCol, nRow );
|
|
mnCol = mnFirstCol;
|
|
mnRow = mnFirstRow;
|
|
}
|
|
|
|
MergedCellIterator& MergedCellIterator::operator++()
|
|
{
|
|
DBG_ASSERT( Is(), "svx::frame::MergedCellIterator::operator++() - already invalid" );
|
|
if( ++mnCol > mnLastCol )
|
|
{
|
|
mnCol = mnFirstCol;
|
|
++mnRow;
|
|
}
|
|
return *this;
|
|
}
|
|
|
|
|
|
#define DBG_FRAME_CHECK( cond, funcname, error ) DBG_ASSERT( cond, "svx::frame::Array::" funcname " - " error )
|
|
#define DBG_FRAME_CHECK_COL( col, funcname ) DBG_FRAME_CHECK( (col) < GetColCount(), funcname, "invalid column index" )
|
|
#define DBG_FRAME_CHECK_ROW( row, funcname ) DBG_FRAME_CHECK( (row) < GetRowCount(), funcname, "invalid row index" )
|
|
#define DBG_FRAME_CHECK_COLROW( col, row, funcname ) DBG_FRAME_CHECK( ((col) < GetColCount()) && ((row) < GetRowCount()), funcname, "invalid cell index" )
|
|
#define DBG_FRAME_CHECK_COL_1( col, funcname ) DBG_FRAME_CHECK( (col) <= GetColCount(), funcname, "invalid column index" )
|
|
#define DBG_FRAME_CHECK_ROW_1( row, funcname ) DBG_FRAME_CHECK( (row) <= GetRowCount(), funcname, "invalid row index" )
|
|
|
|
|
|
#define CELL( col, row ) mxImpl->GetCell( col, row )
|
|
#define CELLACC( col, row ) mxImpl->GetCellAcc( col, row )
|
|
#define ORIGCELL( col, row ) mxImpl->GetMergedOriginCell( col, row )
|
|
|
|
|
|
Array::Array()
|
|
{
|
|
Initialize( 0, 0 );
|
|
}
|
|
|
|
Array::~Array()
|
|
{
|
|
}
|
|
|
|
// array size and column/row indexes
|
|
void Array::Initialize( size_t nWidth, size_t nHeight )
|
|
{
|
|
mxImpl.reset( new ArrayImpl( nWidth, nHeight ) );
|
|
}
|
|
|
|
size_t Array::GetColCount() const
|
|
{
|
|
return mxImpl->mnWidth;
|
|
}
|
|
|
|
size_t Array::GetRowCount() const
|
|
{
|
|
return mxImpl->mnHeight;
|
|
}
|
|
|
|
size_t Array::GetCellCount() const
|
|
{
|
|
return mxImpl->maCells.size();
|
|
}
|
|
|
|
size_t Array::GetCellIndex( size_t nCol, size_t nRow, bool bRTL ) const
|
|
{
|
|
DBG_FRAME_CHECK_COLROW( nCol, nRow, "GetCellIndex" );
|
|
if (bRTL)
|
|
nCol = mxImpl->GetMirrorCol(nCol);
|
|
return mxImpl->GetIndex( nCol, nRow );
|
|
}
|
|
|
|
// cell border styles
|
|
void Array::SetCellStyleLeft( size_t nCol, size_t nRow, const Style& rStyle )
|
|
{
|
|
DBG_FRAME_CHECK_COLROW( nCol, nRow, "SetCellStyleLeft" );
|
|
CELLACC( nCol, nRow ).SetStyleLeft(rStyle);
|
|
}
|
|
|
|
void Array::SetCellStyleRight( size_t nCol, size_t nRow, const Style& rStyle )
|
|
{
|
|
DBG_FRAME_CHECK_COLROW( nCol, nRow, "SetCellStyleRight" );
|
|
CELLACC( nCol, nRow ).SetStyleRight(rStyle);
|
|
}
|
|
|
|
void Array::SetCellStyleTop( size_t nCol, size_t nRow, const Style& rStyle )
|
|
{
|
|
DBG_FRAME_CHECK_COLROW( nCol, nRow, "SetCellStyleTop" );
|
|
CELLACC( nCol, nRow ).SetStyleTop(rStyle);
|
|
}
|
|
|
|
void Array::SetCellStyleBottom( size_t nCol, size_t nRow, const Style& rStyle )
|
|
{
|
|
DBG_FRAME_CHECK_COLROW( nCol, nRow, "SetCellStyleBottom" );
|
|
CELLACC( nCol, nRow ).SetStyleBottom(rStyle);
|
|
}
|
|
|
|
void Array::SetCellStyleTLBR( size_t nCol, size_t nRow, const Style& rStyle )
|
|
{
|
|
DBG_FRAME_CHECK_COLROW( nCol, nRow, "SetCellStyleTLBR" );
|
|
CELLACC( nCol, nRow ).SetStyleTLBR(rStyle);
|
|
}
|
|
|
|
void Array::SetCellStyleBLTR( size_t nCol, size_t nRow, const Style& rStyle )
|
|
{
|
|
DBG_FRAME_CHECK_COLROW( nCol, nRow, "SetCellStyleBLTR" );
|
|
CELLACC( nCol, nRow ).SetStyleBLTR(rStyle);
|
|
}
|
|
|
|
void Array::SetCellStyleDiag( size_t nCol, size_t nRow, const Style& rTLBR, const Style& rBLTR )
|
|
{
|
|
DBG_FRAME_CHECK_COLROW( nCol, nRow, "SetCellStyleDiag" );
|
|
Cell& rCell = CELLACC( nCol, nRow );
|
|
rCell.SetStyleTLBR(rTLBR);
|
|
rCell.SetStyleBLTR(rBLTR);
|
|
}
|
|
|
|
void Array::SetColumnStyleLeft( size_t nCol, const Style& rStyle )
|
|
{
|
|
DBG_FRAME_CHECK_COL( nCol, "SetColumnStyleLeft" );
|
|
for( size_t nRow = 0; nRow < mxImpl->mnHeight; ++nRow )
|
|
SetCellStyleLeft( nCol, nRow, rStyle );
|
|
}
|
|
|
|
void Array::SetColumnStyleRight( size_t nCol, const Style& rStyle )
|
|
{
|
|
DBG_FRAME_CHECK_COL( nCol, "SetColumnStyleRight" );
|
|
for( size_t nRow = 0; nRow < mxImpl->mnHeight; ++nRow )
|
|
SetCellStyleRight( nCol, nRow, rStyle );
|
|
}
|
|
|
|
void Array::SetRowStyleTop( size_t nRow, const Style& rStyle )
|
|
{
|
|
DBG_FRAME_CHECK_ROW( nRow, "SetRowStyleTop" );
|
|
for( size_t nCol = 0; nCol < mxImpl->mnWidth; ++nCol )
|
|
SetCellStyleTop( nCol, nRow, rStyle );
|
|
}
|
|
|
|
void Array::SetRowStyleBottom( size_t nRow, const Style& rStyle )
|
|
{
|
|
DBG_FRAME_CHECK_ROW( nRow, "SetRowStyleBottom" );
|
|
for( size_t nCol = 0; nCol < mxImpl->mnWidth; ++nCol )
|
|
SetCellStyleBottom( nCol, nRow, rStyle );
|
|
}
|
|
|
|
void Array::SetCellRotation(size_t nCol, size_t nRow, SvxRotateMode eRotMode, double fOrientation)
|
|
{
|
|
DBG_FRAME_CHECK_COLROW(nCol, nRow, "SetCellRotation");
|
|
Cell& rTarget = CELLACC(nCol, nRow);
|
|
rTarget.meRotMode = eRotMode;
|
|
rTarget.mfOrientation = fOrientation;
|
|
|
|
if (!mxImpl->mbMayHaveCellRotation)
|
|
{
|
|
// activate once when a cell gets actually rotated to allow fast
|
|
// answering HasCellRotation() calls
|
|
mxImpl->mbMayHaveCellRotation = rTarget.IsRotated();
|
|
}
|
|
}
|
|
|
|
bool Array::HasCellRotation() const
|
|
{
|
|
if (!mxImpl->mbMayHaveCellRotation)
|
|
{
|
|
// never set, no need to check
|
|
return false;
|
|
}
|
|
|
|
return mxImpl->HasCellRotation();
|
|
}
|
|
|
|
const Style& Array::GetCellStyleLeft( size_t nCol, size_t nRow ) const
|
|
{
|
|
// outside clipping rows or overlapped in merged cells: invisible
|
|
if( !mxImpl->IsRowInClipRange( nRow ) || mxImpl->IsMergedOverlappedLeft( nCol, nRow ) )
|
|
return OBJ_STYLE_NONE;
|
|
// left clipping border: always own left style
|
|
if( nCol == mxImpl->mnFirstClipCol )
|
|
return ORIGCELL( nCol, nRow ).GetStyleLeft();
|
|
// right clipping border: always right style of left neighbor cell
|
|
if( nCol == mxImpl->mnLastClipCol + 1 )
|
|
return ORIGCELL( nCol - 1, nRow ).GetStyleRight();
|
|
// outside clipping columns: invisible
|
|
if( !mxImpl->IsColInClipRange( nCol ) )
|
|
return OBJ_STYLE_NONE;
|
|
// inside clipping range: maximum of own left style and right style of left neighbor cell
|
|
return std::max( ORIGCELL( nCol, nRow ).GetStyleLeft(), ORIGCELL( nCol - 1, nRow ).GetStyleRight() );
|
|
}
|
|
|
|
const Style& Array::GetCellStyleRight( size_t nCol, size_t nRow ) const
|
|
{
|
|
// outside clipping rows or overlapped in merged cells: invisible
|
|
if( !mxImpl->IsRowInClipRange( nRow ) || mxImpl->IsMergedOverlappedRight( nCol, nRow ) )
|
|
return OBJ_STYLE_NONE;
|
|
// left clipping border: always left style of right neighbor cell
|
|
if( nCol + 1 == mxImpl->mnFirstClipCol )
|
|
return ORIGCELL( nCol + 1, nRow ).GetStyleLeft();
|
|
// right clipping border: always own right style
|
|
if( nCol == mxImpl->mnLastClipCol )
|
|
return ORIGCELL( nCol, nRow ).GetStyleRight();
|
|
// outside clipping columns: invisible
|
|
if( !mxImpl->IsColInClipRange( nCol ) )
|
|
return OBJ_STYLE_NONE;
|
|
// inside clipping range: maximum of own right style and left style of right neighbor cell
|
|
return std::max( ORIGCELL( nCol, nRow ).GetStyleRight(), ORIGCELL( nCol + 1, nRow ).GetStyleLeft() );
|
|
}
|
|
|
|
const Style& Array::GetCellStyleTop( size_t nCol, size_t nRow ) const
|
|
{
|
|
// outside clipping columns or overlapped in merged cells: invisible
|
|
if( !mxImpl->IsColInClipRange( nCol ) || mxImpl->IsMergedOverlappedTop( nCol, nRow ) )
|
|
return OBJ_STYLE_NONE;
|
|
// top clipping border: always own top style
|
|
if( nRow == mxImpl->mnFirstClipRow )
|
|
return ORIGCELL( nCol, nRow ).GetStyleTop();
|
|
// bottom clipping border: always bottom style of top neighbor cell
|
|
if( nRow == mxImpl->mnLastClipRow + 1 )
|
|
return ORIGCELL( nCol, nRow - 1 ).GetStyleBottom();
|
|
// outside clipping rows: invisible
|
|
if( !mxImpl->IsRowInClipRange( nRow ) )
|
|
return OBJ_STYLE_NONE;
|
|
// inside clipping range: maximum of own top style and bottom style of top neighbor cell
|
|
return std::max( ORIGCELL( nCol, nRow ).GetStyleTop(), ORIGCELL( nCol, nRow - 1 ).GetStyleBottom() );
|
|
}
|
|
|
|
const Style& Array::GetCellStyleBottom( size_t nCol, size_t nRow ) const
|
|
{
|
|
// outside clipping columns or overlapped in merged cells: invisible
|
|
if( !mxImpl->IsColInClipRange( nCol ) || mxImpl->IsMergedOverlappedBottom( nCol, nRow ) )
|
|
return OBJ_STYLE_NONE;
|
|
// top clipping border: always top style of bottom neighbor cell
|
|
if( nRow + 1 == mxImpl->mnFirstClipRow )
|
|
return ORIGCELL( nCol, nRow + 1 ).GetStyleTop();
|
|
// bottom clipping border: always own bottom style
|
|
if( nRow == mxImpl->mnLastClipRow )
|
|
return ORIGCELL( nCol, nRow ).GetStyleBottom();
|
|
// outside clipping rows: invisible
|
|
if( !mxImpl->IsRowInClipRange( nRow ) )
|
|
return OBJ_STYLE_NONE;
|
|
// inside clipping range: maximum of own bottom style and top style of bottom neighbor cell
|
|
return std::max( ORIGCELL( nCol, nRow ).GetStyleBottom(), ORIGCELL( nCol, nRow + 1 ).GetStyleTop() );
|
|
}
|
|
|
|
const Style& Array::GetCellStyleTLBR( size_t nCol, size_t nRow ) const
|
|
{
|
|
return CELL( nCol, nRow ).GetStyleTLBR();
|
|
}
|
|
|
|
const Style& Array::GetCellStyleBLTR( size_t nCol, size_t nRow ) const
|
|
{
|
|
return CELL( nCol, nRow ).GetStyleBLTR();
|
|
}
|
|
|
|
const Style& Array::GetCellStyleTL( size_t nCol, size_t nRow ) const
|
|
{
|
|
// not in clipping range: always invisible
|
|
if( !mxImpl->IsInClipRange( nCol, nRow ) )
|
|
return OBJ_STYLE_NONE;
|
|
// return style only for top-left cell
|
|
size_t nFirstCol = mxImpl->GetMergedFirstCol( nCol, nRow );
|
|
size_t nFirstRow = mxImpl->GetMergedFirstRow( nCol, nRow );
|
|
return ((nCol == nFirstCol) && (nRow == nFirstRow)) ?
|
|
CELL( nFirstCol, nFirstRow ).GetStyleTLBR() : OBJ_STYLE_NONE;
|
|
}
|
|
|
|
const Style& Array::GetCellStyleBR( size_t nCol, size_t nRow ) const
|
|
{
|
|
// not in clipping range: always invisible
|
|
if( !mxImpl->IsInClipRange( nCol, nRow ) )
|
|
return OBJ_STYLE_NONE;
|
|
// return style only for bottom-right cell
|
|
size_t nLastCol = mxImpl->GetMergedLastCol( nCol, nRow );
|
|
size_t nLastRow = mxImpl->GetMergedLastRow( nCol, nRow );
|
|
return ((nCol == nLastCol) && (nRow == nLastRow)) ?
|
|
CELL( mxImpl->GetMergedFirstCol( nCol, nRow ), mxImpl->GetMergedFirstRow( nCol, nRow ) ).GetStyleTLBR() : OBJ_STYLE_NONE;
|
|
}
|
|
|
|
const Style& Array::GetCellStyleBL( size_t nCol, size_t nRow ) const
|
|
{
|
|
// not in clipping range: always invisible
|
|
if( !mxImpl->IsInClipRange( nCol, nRow ) )
|
|
return OBJ_STYLE_NONE;
|
|
// return style only for bottom-left cell
|
|
size_t nFirstCol = mxImpl->GetMergedFirstCol( nCol, nRow );
|
|
size_t nLastRow = mxImpl->GetMergedLastRow( nCol, nRow );
|
|
return ((nCol == nFirstCol) && (nRow == nLastRow)) ?
|
|
CELL( nFirstCol, mxImpl->GetMergedFirstRow( nCol, nRow ) ).GetStyleBLTR() : OBJ_STYLE_NONE;
|
|
}
|
|
|
|
const Style& Array::GetCellStyleTR( size_t nCol, size_t nRow ) const
|
|
{
|
|
// not in clipping range: always invisible
|
|
if( !mxImpl->IsInClipRange( nCol, nRow ) )
|
|
return OBJ_STYLE_NONE;
|
|
// return style only for top-right cell
|
|
size_t nFirstRow = mxImpl->GetMergedFirstRow( nCol, nRow );
|
|
size_t nLastCol = mxImpl->GetMergedLastCol( nCol, nRow );
|
|
return ((nCol == nLastCol) && (nRow == nFirstRow)) ?
|
|
CELL( mxImpl->GetMergedFirstCol( nCol, nRow ), nFirstRow ).GetStyleBLTR() : OBJ_STYLE_NONE;
|
|
}
|
|
|
|
// cell merging
|
|
void Array::SetMergedRange( size_t nFirstCol, size_t nFirstRow, size_t nLastCol, size_t nLastRow )
|
|
{
|
|
DBG_FRAME_CHECK_COLROW( nFirstCol, nFirstRow, "SetMergedRange" );
|
|
DBG_FRAME_CHECK_COLROW( nLastCol, nLastRow, "SetMergedRange" );
|
|
#if OSL_DEBUG_LEVEL >= 2
|
|
{
|
|
bool bFound = false;
|
|
for( size_t nCurrCol = nFirstCol; !bFound && (nCurrCol <= nLastCol); ++nCurrCol )
|
|
for( size_t nCurrRow = nFirstRow; !bFound && (nCurrRow <= nLastRow); ++nCurrRow )
|
|
bFound = CELL( nCurrCol, nCurrRow ).IsMerged();
|
|
DBG_FRAME_CHECK( !bFound, "SetMergedRange", "overlapping merged ranges" );
|
|
}
|
|
#endif
|
|
if( mxImpl->IsValidPos( nFirstCol, nFirstRow ) && mxImpl->IsValidPos( nLastCol, nLastRow ) )
|
|
lclSetMergedRange( mxImpl->maCells, mxImpl->mnWidth, nFirstCol, nFirstRow, nLastCol, nLastRow );
|
|
}
|
|
|
|
void Array::SetAddMergedLeftSize( size_t nCol, size_t nRow, long nAddSize )
|
|
{
|
|
DBG_FRAME_CHECK_COLROW( nCol, nRow, "SetAddMergedLeftSize" );
|
|
DBG_FRAME_CHECK( mxImpl->GetMergedFirstCol( nCol, nRow ) == 0, "SetAddMergedLeftSize", "additional border inside array" );
|
|
for( MergedCellIterator aIt( *this, nCol, nRow ); aIt.Is(); ++aIt )
|
|
CELLACC( aIt.Col(), aIt.Row() ).mnAddLeft = nAddSize;
|
|
}
|
|
|
|
void Array::SetAddMergedRightSize( size_t nCol, size_t nRow, long nAddSize )
|
|
{
|
|
DBG_FRAME_CHECK_COLROW( nCol, nRow, "SetAddMergedRightSize" );
|
|
DBG_FRAME_CHECK( mxImpl->GetMergedLastCol( nCol, nRow ) + 1 == mxImpl->mnWidth, "SetAddMergedRightSize", "additional border inside array" );
|
|
for( MergedCellIterator aIt( *this, nCol, nRow ); aIt.Is(); ++aIt )
|
|
CELLACC( aIt.Col(), aIt.Row() ).mnAddRight = nAddSize;
|
|
}
|
|
|
|
void Array::SetAddMergedTopSize( size_t nCol, size_t nRow, long nAddSize )
|
|
{
|
|
DBG_FRAME_CHECK_COLROW( nCol, nRow, "SetAddMergedTopSize" );
|
|
DBG_FRAME_CHECK( mxImpl->GetMergedFirstRow( nCol, nRow ) == 0, "SetAddMergedTopSize", "additional border inside array" );
|
|
for( MergedCellIterator aIt( *this, nCol, nRow ); aIt.Is(); ++aIt )
|
|
CELLACC( aIt.Col(), aIt.Row() ).mnAddTop = nAddSize;
|
|
}
|
|
|
|
void Array::SetAddMergedBottomSize( size_t nCol, size_t nRow, long nAddSize )
|
|
{
|
|
DBG_FRAME_CHECK_COLROW( nCol, nRow, "SetAddMergedBottomSize" );
|
|
DBG_FRAME_CHECK( mxImpl->GetMergedLastRow( nCol, nRow ) + 1 == mxImpl->mnHeight, "SetAddMergedBottomSize", "additional border inside array" );
|
|
for( MergedCellIterator aIt( *this, nCol, nRow ); aIt.Is(); ++aIt )
|
|
CELLACC( aIt.Col(), aIt.Row() ).mnAddBottom = nAddSize;
|
|
}
|
|
|
|
bool Array::IsMerged( size_t nCol, size_t nRow ) const
|
|
{
|
|
DBG_FRAME_CHECK_COLROW( nCol, nRow, "IsMerged" );
|
|
return CELL( nCol, nRow ).IsMerged();
|
|
}
|
|
|
|
void Array::GetMergedOrigin( size_t& rnFirstCol, size_t& rnFirstRow, size_t nCol, size_t nRow ) const
|
|
{
|
|
DBG_FRAME_CHECK_COLROW( nCol, nRow, "GetMergedOrigin" );
|
|
rnFirstCol = mxImpl->GetMergedFirstCol( nCol, nRow );
|
|
rnFirstRow = mxImpl->GetMergedFirstRow( nCol, nRow );
|
|
}
|
|
|
|
void Array::GetMergedRange( size_t& rnFirstCol, size_t& rnFirstRow,
|
|
size_t& rnLastCol, size_t& rnLastRow, size_t nCol, size_t nRow ) const
|
|
{
|
|
GetMergedOrigin( rnFirstCol, rnFirstRow, nCol, nRow );
|
|
rnLastCol = mxImpl->GetMergedLastCol( nCol, nRow );
|
|
rnLastRow = mxImpl->GetMergedLastRow( nCol, nRow );
|
|
}
|
|
|
|
// clipping
|
|
void Array::SetClipRange( size_t nFirstCol, size_t nFirstRow, size_t nLastCol, size_t nLastRow )
|
|
{
|
|
DBG_FRAME_CHECK_COLROW( nFirstCol, nFirstRow, "SetClipRange" );
|
|
DBG_FRAME_CHECK_COLROW( nLastCol, nLastRow, "SetClipRange" );
|
|
mxImpl->mnFirstClipCol = nFirstCol;
|
|
mxImpl->mnFirstClipRow = nFirstRow;
|
|
mxImpl->mnLastClipCol = nLastCol;
|
|
mxImpl->mnLastClipRow = nLastRow;
|
|
}
|
|
|
|
// cell coordinates
|
|
void Array::SetXOffset( long nXOffset )
|
|
{
|
|
mxImpl->maXCoords[ 0 ] = nXOffset;
|
|
mxImpl->mbXCoordsDirty = true;
|
|
}
|
|
|
|
void Array::SetYOffset( long nYOffset )
|
|
{
|
|
mxImpl->maYCoords[ 0 ] = nYOffset;
|
|
mxImpl->mbYCoordsDirty = true;
|
|
}
|
|
|
|
void Array::SetColWidth( size_t nCol, long nWidth )
|
|
{
|
|
DBG_FRAME_CHECK_COL( nCol, "SetColWidth" );
|
|
mxImpl->maWidths[ nCol ] = nWidth;
|
|
mxImpl->mbXCoordsDirty = true;
|
|
}
|
|
|
|
void Array::SetRowHeight( size_t nRow, long nHeight )
|
|
{
|
|
DBG_FRAME_CHECK_ROW( nRow, "SetRowHeight" );
|
|
mxImpl->maHeights[ nRow ] = nHeight;
|
|
mxImpl->mbYCoordsDirty = true;
|
|
}
|
|
|
|
void Array::SetAllColWidths( long nWidth )
|
|
{
|
|
std::fill( mxImpl->maWidths.begin(), mxImpl->maWidths.end(), nWidth );
|
|
mxImpl->mbXCoordsDirty = true;
|
|
}
|
|
|
|
void Array::SetAllRowHeights( long nHeight )
|
|
{
|
|
std::fill( mxImpl->maHeights.begin(), mxImpl->maHeights.end(), nHeight );
|
|
mxImpl->mbYCoordsDirty = true;
|
|
}
|
|
|
|
long Array::GetColPosition( size_t nCol ) const
|
|
{
|
|
DBG_FRAME_CHECK_COL_1( nCol, "GetColPosition" );
|
|
return mxImpl->GetColPosition( nCol );
|
|
}
|
|
|
|
long Array::GetRowPosition( size_t nRow ) const
|
|
{
|
|
DBG_FRAME_CHECK_ROW_1( nRow, "GetRowPosition" );
|
|
return mxImpl->GetRowPosition( nRow );
|
|
}
|
|
|
|
long Array::GetColWidth( size_t nFirstCol, size_t nLastCol ) const
|
|
{
|
|
DBG_FRAME_CHECK_COL( nFirstCol, "GetColWidth" );
|
|
DBG_FRAME_CHECK_COL( nLastCol, "GetColWidth" );
|
|
return GetColPosition( nLastCol + 1 ) - GetColPosition( nFirstCol );
|
|
}
|
|
|
|
long Array::GetRowHeight( size_t nFirstRow, size_t nLastRow ) const
|
|
{
|
|
DBG_FRAME_CHECK_ROW( nFirstRow, "GetRowHeight" );
|
|
DBG_FRAME_CHECK_ROW( nLastRow, "GetRowHeight" );
|
|
return GetRowPosition( nLastRow + 1 ) - GetRowPosition( nFirstRow );
|
|
}
|
|
|
|
long Array::GetWidth() const
|
|
{
|
|
return GetColPosition( mxImpl->mnWidth ) - GetColPosition( 0 );
|
|
}
|
|
|
|
long Array::GetHeight() const
|
|
{
|
|
return GetRowPosition( mxImpl->mnHeight ) - GetRowPosition( 0 );
|
|
}
|
|
|
|
basegfx::B2DRange Array::GetCellRange( size_t nCellIndex ) const
|
|
{
|
|
return GetCellRange(nCellIndex % GetColCount(), nCellIndex / GetColCount());
|
|
}
|
|
|
|
basegfx::B2DRange Array::GetCellRange( size_t nCol, size_t nRow ) const
|
|
{
|
|
size_t nFirstCol = mxImpl->GetMergedFirstCol( nCol, nRow );
|
|
size_t nFirstRow = mxImpl->GetMergedFirstRow( nCol, nRow );
|
|
size_t nLastCol = mxImpl->GetMergedLastCol( nCol, nRow );
|
|
size_t nLastRow = mxImpl->GetMergedLastRow( nCol, nRow );
|
|
const Point aPoint( GetColPosition( nFirstCol ), GetRowPosition( nFirstRow ) );
|
|
const Size aSize( GetColWidth( nFirstCol, nLastCol ) + 1, GetRowHeight( nFirstRow, nLastRow ) + 1 );
|
|
tools::Rectangle aRect(aPoint, aSize);
|
|
|
|
// adjust rectangle for partly visible merged cells
|
|
const Cell& rCell = CELL( nCol, nRow );
|
|
if( rCell.IsMerged() )
|
|
{
|
|
aRect.Left() -= rCell.mnAddLeft;
|
|
aRect.Right() += rCell.mnAddRight;
|
|
aRect.Top() -= rCell.mnAddTop;
|
|
aRect.Bottom() += rCell.mnAddBottom;
|
|
}
|
|
return basegfx::B2DRange(aRect.Left(), aRect.Top(), aRect.Right(), aRect.Bottom());
|
|
}
|
|
|
|
// mirroring
|
|
void Array::MirrorSelfX()
|
|
{
|
|
CellVec aNewCells;
|
|
aNewCells.reserve( GetCellCount() );
|
|
|
|
size_t nCol, nRow;
|
|
for( nRow = 0; nRow < mxImpl->mnHeight; ++nRow )
|
|
{
|
|
for( nCol = 0; nCol < mxImpl->mnWidth; ++nCol )
|
|
{
|
|
aNewCells.push_back( CELL( mxImpl->GetMirrorCol( nCol ), nRow ) );
|
|
aNewCells.back().MirrorSelfX();
|
|
}
|
|
}
|
|
for( nRow = 0; nRow < mxImpl->mnHeight; ++nRow )
|
|
{
|
|
for( nCol = 0; nCol < mxImpl->mnWidth; ++nCol )
|
|
{
|
|
if( CELL( nCol, nRow ).mbMergeOrig )
|
|
{
|
|
size_t nLastCol = mxImpl->GetMergedLastCol( nCol, nRow );
|
|
size_t nLastRow = mxImpl->GetMergedLastRow( nCol, nRow );
|
|
lclSetMergedRange( aNewCells, mxImpl->mnWidth,
|
|
mxImpl->GetMirrorCol( nLastCol ), nRow,
|
|
mxImpl->GetMirrorCol( nCol ), nLastRow );
|
|
}
|
|
}
|
|
}
|
|
mxImpl->maCells.swap( aNewCells );
|
|
|
|
std::reverse( mxImpl->maWidths.begin(), mxImpl->maWidths.end() );
|
|
mxImpl->mbXCoordsDirty = true;
|
|
}
|
|
|
|
// drawing
|
|
void HelperCreateHorizontalEntry(
|
|
const Array& rArray, const Style& rStyle, size_t col, size_t row,
|
|
const basegfx::B2DPoint& rOrigin, const basegfx::B2DVector& rX, const basegfx::B2DVector& rY,
|
|
drawinglayer::primitive2d::Primitive2DContainer& rSequence,
|
|
bool bUpper, const Color* pForceColor)
|
|
{
|
|
// get involved styles at start
|
|
const Style& rStartFromTR(rArray.GetCellStyleBL( col, row - 1 ));
|
|
const Style& rStartLFromT(rArray.GetCellStyleLeft( col, row - 1 ));
|
|
const Style& rStartLFromL(rArray.GetCellStyleTop( col - 1, row ));
|
|
const Style& rStartLFromB(rArray.GetCellStyleLeft( col, row ));
|
|
const Style& rStartFromBR(rArray.GetCellStyleTL( col, row ));
|
|
StyleVectorTable aStart;
|
|
|
|
aStart.add(rStartFromTR, rX, rX - rY, false);
|
|
aStart.add(rStartLFromT, rX, -rY, true);
|
|
aStart.add(rStartLFromL, rX, -rX, true);
|
|
aStart.add(rStartLFromB, rX, rY, false);
|
|
aStart.add(rStartFromBR, rX, rX + rY, false);
|
|
aStart.sort();
|
|
|
|
// get involved styles at end
|
|
const Style& rEndFromTL(rArray.GetCellStyleBR( col, row - 1 ));
|
|
const Style& rEndRFromT(rArray.GetCellStyleRight( col, row - 1 ));
|
|
const Style& rEndRFromR(rArray.GetCellStyleTop( col + 1, row ));
|
|
const Style& rEndRFromB(rArray.GetCellStyleRight( col, row ));
|
|
const Style& rEndFromBL(rArray.GetCellStyleTR( col, row ));
|
|
StyleVectorTable aEnd;
|
|
const basegfx::B2DVector aAxis(-rX);
|
|
|
|
aEnd.add(rEndFromTL, aAxis, aAxis - rY, true);
|
|
aEnd.add(rEndRFromT, aAxis, -rY, true);
|
|
aEnd.add(rEndRFromR, aAxis, rX, false);
|
|
aEnd.add(rEndRFromB, aAxis, rY, false);
|
|
aEnd.add(rEndFromBL, aAxis, rY - rX, true);
|
|
aEnd.sort();
|
|
|
|
CreateBorderPrimitives(
|
|
rSequence,
|
|
bUpper ? rOrigin : basegfx::B2DPoint(rOrigin + rY),
|
|
rX,
|
|
rStyle,
|
|
aStart,
|
|
aEnd,
|
|
pForceColor
|
|
);
|
|
}
|
|
|
|
void HelperCreateVerticalEntry(
|
|
const Array& rArray, const Style& rStyle, size_t col, size_t row,
|
|
const basegfx::B2DPoint& rOrigin, const basegfx::B2DVector& rX, const basegfx::B2DVector& rY,
|
|
drawinglayer::primitive2d::Primitive2DContainer& rSequence,
|
|
bool bLeft, const Color* pForceColor)
|
|
{
|
|
// get involved styles at start
|
|
const Style& rStartFromBL(rArray.GetCellStyleTR( col - 1, row ));
|
|
const Style& rStartTFromL(rArray.GetCellStyleTop( col - 1, row ));
|
|
const Style& rStartTFromT(rArray.GetCellStyleLeft( col, row - 1 ));
|
|
const Style& rStartTFromR(rArray.GetCellStyleTop( col, row ));
|
|
const Style& rStartFromBR(rArray.GetCellStyleTL( col, row ));
|
|
StyleVectorTable aStart;
|
|
|
|
aStart.add(rStartFromBR, rY, rX + rY, false);
|
|
aStart.add(rStartTFromR, rY, rX, false);
|
|
aStart.add(rStartTFromT, rY, -rY, true);
|
|
aStart.add(rStartTFromL, rY, -rX, true);
|
|
aStart.add(rStartFromBL, rY, rY - rX, true);
|
|
aStart.sort();
|
|
|
|
// get involved styles at end
|
|
const Style& rEndFromTL(rArray.GetCellStyleBR( col - 1, row ));
|
|
const Style& rEndBFromL(rArray.GetCellStyleBottom( col - 1, row ));
|
|
const Style& rEndBFromB(rArray.GetCellStyleLeft( col, row + 1 ));
|
|
const Style& rEndBFromR(rArray.GetCellStyleBottom( col, row ));
|
|
const Style& rEndFromTR(rArray.GetCellStyleBL( col, row ));
|
|
StyleVectorTable aEnd;
|
|
const basegfx::B2DVector aAxis(-rY);
|
|
|
|
aEnd.add(rEndFromTR, aAxis, rX - rY, false);
|
|
aEnd.add(rEndBFromR, aAxis, rX, false);
|
|
aEnd.add(rEndBFromB, aAxis, rY, false);
|
|
aEnd.add(rEndBFromL, aAxis, -rX, true);
|
|
aEnd.add(rEndFromTL, aAxis, aAxis - rX, true);
|
|
aEnd.sort();
|
|
|
|
CreateBorderPrimitives(
|
|
rSequence,
|
|
bLeft ? rOrigin : basegfx::B2DPoint(rOrigin + rX),
|
|
rY,
|
|
rStyle,
|
|
aStart,
|
|
aEnd,
|
|
pForceColor
|
|
);
|
|
}
|
|
|
|
void HelperCreateEntry(const Array& rArray, const Style& rStyle, drawinglayer::primitive2d::Primitive2DContainer& rSequence, const Color* pForceColor)
|
|
{
|
|
const Cell* pCell = rStyle.GetUsingCell();
|
|
|
|
if(nullptr != pCell)
|
|
{
|
|
const size_t nCellIndex(pCell->GetCellIndex(rArray));
|
|
|
|
if(static_cast<size_t>(-1) != nCellIndex)
|
|
{
|
|
size_t col(nCellIndex % rArray.GetColCount());
|
|
size_t row(nCellIndex / rArray.GetColCount());
|
|
const bool bL(&rStyle == &pCell->GetStyleLeft());
|
|
const bool bR(&rStyle == &pCell->GetStyleRight());
|
|
const bool bT(&rStyle == &pCell->GetStyleTop());
|
|
const bool bB(&rStyle == &pCell->GetStyleBottom());
|
|
|
|
if(bL || bR || bT || bB)
|
|
{
|
|
const basegfx::B2DHomMatrix aCoordinateSystem(pCell->CreateCoordinateSystem(rArray));
|
|
const basegfx::B2DVector aX(basegfx::utils::getColumn(aCoordinateSystem, 0));
|
|
const basegfx::B2DVector aY(basegfx::utils::getColumn(aCoordinateSystem, 1));
|
|
const basegfx::B2DPoint aOrigin(basegfx::utils::getColumn(aCoordinateSystem, 2));
|
|
|
|
if(bL || bR)
|
|
{
|
|
// left/right
|
|
HelperCreateVerticalEntry(rArray, rStyle, bL ? col : col + 1, row, aOrigin, aX, aY, rSequence, bL, pForceColor);
|
|
}
|
|
else if(bT || bB)
|
|
{
|
|
// top/bottom
|
|
HelperCreateHorizontalEntry(rArray, rStyle, col, bT ? row : row + 1, aOrigin, aX, aY, rSequence, bT, pForceColor);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void HelperMergeInB2DPrimitiveArray(
|
|
const drawinglayer::primitive2d::Primitive2DContainer& rSource,
|
|
drawinglayer::primitive2d::Primitive2DContainer& rTarget)
|
|
{
|
|
if(rSource.size() > 1)
|
|
{
|
|
drawinglayer::primitive2d::Primitive2DReference aCandidate;
|
|
|
|
for(const auto& a : rSource)
|
|
{
|
|
if(aCandidate.is())
|
|
{
|
|
const drawinglayer::primitive2d::Primitive2DReference aMerge(
|
|
drawinglayer::primitive2d::tryMergeBorderLinePrimitive2D(aCandidate, a));
|
|
|
|
if(aMerge.is())
|
|
{
|
|
aCandidate = aMerge;
|
|
}
|
|
else
|
|
{
|
|
rTarget.append(aCandidate);
|
|
aCandidate = a;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
aCandidate = a;
|
|
}
|
|
}
|
|
|
|
if(aCandidate.is())
|
|
{
|
|
rTarget.append(aCandidate);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
rTarget.append(rSource);
|
|
}
|
|
}
|
|
|
|
drawinglayer::primitive2d::Primitive2DContainer Array::CreateB2DPrimitiveRange(
|
|
size_t nFirstCol, size_t nFirstRow, size_t nLastCol, size_t nLastRow,
|
|
const Color* pForceColor ) const
|
|
{
|
|
DBG_FRAME_CHECK_COLROW( nFirstCol, nFirstRow, "CreateB2DPrimitiveRange" );
|
|
DBG_FRAME_CHECK_COLROW( nLastCol, nLastRow, "CreateB2DPrimitiveRange" );
|
|
|
|
// various primitive sequences to collect the different border types
|
|
drawinglayer::primitive2d::Primitive2DContainer aHorizontalSequence;
|
|
std::vector< drawinglayer::primitive2d::Primitive2DContainer > aVerticalSequences(nLastCol - nFirstCol + 1);
|
|
drawinglayer::primitive2d::Primitive2DContainer aCrossSequence;
|
|
|
|
for (size_t nRow = nFirstRow; nRow <= nLastRow; ++nRow)
|
|
{
|
|
for (size_t nCol = nFirstCol; nCol <= nLastCol; ++nCol)
|
|
{
|
|
const Cell& rCell = CELL(nCol, nRow);
|
|
const basegfx::B2DHomMatrix aCoordinateSystem(rCell.CreateCoordinateSystem(*this));
|
|
const basegfx::B2DVector aX(basegfx::utils::getColumn(aCoordinateSystem, 0));
|
|
const basegfx::B2DVector aY(basegfx::utils::getColumn(aCoordinateSystem, 1));
|
|
|
|
if(!aX.equalZero() && !aY.equalZero())
|
|
{
|
|
size_t _nFirstCol = mxImpl->GetMergedFirstCol(nCol, nRow);
|
|
size_t _nFirstRow = mxImpl->GetMergedFirstRow(nCol, nRow);
|
|
size_t _nLastCol = mxImpl->GetMergedLastCol(nCol, nRow);
|
|
size_t _nLastRow = mxImpl->GetMergedLastRow(nCol, nRow);
|
|
const basegfx::B2DPoint aOrigin(basegfx::utils::getColumn(aCoordinateSystem, 2));
|
|
|
|
const bool bOverlapX(rCell.mbOverlapX);
|
|
const bool bOverlapY(rCell.mbOverlapY);
|
|
const bool bFirstCol(nCol == nFirstCol);
|
|
const bool bLastCol(nCol == nLastCol);
|
|
const bool bFirstRow(nRow == nFirstRow);
|
|
const bool bLastRow(nRow == nLastRow);
|
|
|
|
if (!bOverlapX || bFirstRow)
|
|
{
|
|
const Style& rTop = GetCellStyleTop(_nFirstCol, _nFirstRow);
|
|
|
|
if(rTop.IsUsed())
|
|
{
|
|
HelperCreateEntry(*this, rTop, aHorizontalSequence, pForceColor);
|
|
}
|
|
}
|
|
|
|
if (bLastRow)
|
|
{
|
|
const Style& rBottom = GetCellStyleBottom(_nFirstCol, _nFirstRow);
|
|
|
|
if(rBottom.IsUsed())
|
|
{
|
|
HelperCreateEntry(*this, rBottom, aHorizontalSequence, pForceColor);
|
|
}
|
|
}
|
|
|
|
if (!bOverlapY || bFirstCol)
|
|
{
|
|
const Style& rLeft(GetCellStyleLeft(_nFirstCol, _nFirstRow));
|
|
|
|
if(rLeft.IsUsed())
|
|
{
|
|
HelperCreateEntry(*this, rLeft, aVerticalSequences[nCol - nFirstCol], pForceColor);
|
|
}
|
|
}
|
|
|
|
if (bLastCol)
|
|
{
|
|
const Style& rRight(GetCellStyleRight(_nFirstCol, _nFirstRow));
|
|
|
|
if(rRight.IsUsed())
|
|
{
|
|
HelperCreateEntry(*this, rRight, aVerticalSequences[nCol - nFirstCol], pForceColor);
|
|
}
|
|
}
|
|
|
|
if ((!bOverlapX && !bOverlapY) || (bFirstCol && bFirstRow) || (!bOverlapY && bFirstCol) || (!bOverlapX && bFirstRow))
|
|
{
|
|
const Style& rTLBR = GetCellStyleTLBR(_nFirstCol, _nFirstRow);
|
|
if(rTLBR.IsUsed())
|
|
{
|
|
/// top-left and bottom-right Style Tables
|
|
/// Fill top-left Style Table
|
|
const Style& rTLFromRight(GetCellStyleTop(_nFirstCol, _nFirstRow));
|
|
const Style& rTLFromBottom(GetCellStyleLeft(_nFirstCol, _nFirstRow));
|
|
StyleVectorTable aStart;
|
|
const basegfx::B2DVector aAxisA(aX + aY);
|
|
|
|
aStart.add(rTLFromRight, aAxisA, aX, false);
|
|
aStart.add(rTLFromBottom, aAxisA, aY, false);
|
|
aStart.sort();
|
|
|
|
/// Fill bottom-right Style Table
|
|
const Style& rBRFromBottom(GetCellStyleRight(_nLastCol, _nLastRow));
|
|
const Style& rBRFromLeft(GetCellStyleBottom(_nLastCol, _nLastRow));
|
|
StyleVectorTable aEnd;
|
|
const basegfx::B2DVector aAxisB(-aX -aY);
|
|
|
|
aEnd.add(rBRFromBottom, aAxisB, -aY, true);
|
|
aEnd.add(rBRFromLeft, aAxisB, -aX, true);
|
|
aEnd.sort();
|
|
|
|
CreateBorderPrimitives(
|
|
aCrossSequence,
|
|
aOrigin,
|
|
aX + aY,
|
|
rTLBR,
|
|
aStart,
|
|
aEnd,
|
|
pForceColor
|
|
);
|
|
}
|
|
|
|
const Style& rBLTR = GetCellStyleBLTR(_nFirstCol, _nFirstRow);
|
|
if(rBLTR.IsUsed())
|
|
{
|
|
/// bottom-left and top-right Style Tables
|
|
/// Fill bottom-left Style Table
|
|
const Style& rBLFromTop(GetCellStyleLeft(_nFirstCol, _nLastRow));
|
|
const Style& rBLFromBottom(GetCellStyleBottom(_nFirstCol, _nLastRow));
|
|
StyleVectorTable aStart;
|
|
const basegfx::B2DVector aAxisA(aX - aY);
|
|
|
|
aStart.add(rBLFromTop, aAxisA, -aY, true);
|
|
aStart.add(rBLFromBottom, aAxisA, aX, false);
|
|
aStart.sort();
|
|
|
|
/// Fill top-right Style Table
|
|
const Style& rTRFromLeft(GetCellStyleTop(_nLastCol, _nFirstRow));
|
|
const Style& rTRFromBottom(GetCellStyleRight(_nLastCol, _nFirstRow));
|
|
StyleVectorTable aEnd;
|
|
const basegfx::B2DVector aAxisB(aY - aX);
|
|
|
|
aEnd.add(rTRFromLeft, aAxisB, -aX, true);
|
|
aEnd.add(rTRFromBottom, aAxisB, aY, false);
|
|
aEnd.sort();
|
|
|
|
CreateBorderPrimitives(
|
|
aCrossSequence,
|
|
aOrigin + aY,
|
|
aX - aY,
|
|
rBLTR,
|
|
aStart,
|
|
aEnd,
|
|
pForceColor
|
|
);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// to stay compatible, create order as it was formally. Also try to
|
|
// merge primitives as far as possible
|
|
HelperMergeInB2DPrimitiveArray(aHorizontalSequence, aCrossSequence);
|
|
for(const auto& aVert : aVerticalSequences)
|
|
{
|
|
HelperMergeInB2DPrimitiveArray(aVert, aCrossSequence);
|
|
}
|
|
|
|
return aCrossSequence;
|
|
}
|
|
|
|
drawinglayer::primitive2d::Primitive2DContainer Array::CreateB2DPrimitiveArray() const
|
|
{
|
|
drawinglayer::primitive2d::Primitive2DContainer aPrimitives;
|
|
|
|
if (mxImpl->mnWidth && mxImpl->mnHeight)
|
|
{
|
|
aPrimitives = CreateB2DPrimitiveRange(0, 0, mxImpl->mnWidth - 1, mxImpl->mnHeight - 1, nullptr);
|
|
}
|
|
|
|
return aPrimitives;
|
|
}
|
|
|
|
void Array::AddCellIndices() const
|
|
{
|
|
for (size_t a(0); a < mxImpl->maCells.size(); a++)
|
|
{
|
|
const_cast<Array*>(this)->mxImpl->maCells[a].maCellIndex = a;
|
|
}
|
|
}
|
|
|
|
#undef ORIGCELL
|
|
#undef CELLACC
|
|
#undef CELL
|
|
|
|
|
|
#undef DBG_FRAME_CHECK_ROW_1
|
|
#undef DBG_FRAME_CHECK_COL_1
|
|
#undef DBG_FRAME_CHECK_COLROW
|
|
#undef DBG_FRAME_CHECK_ROW
|
|
#undef DBG_FRAME_CHECK_COL
|
|
#undef DBG_FRAME_CHECK
|
|
|
|
|
|
}
|
|
}
|
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|