548 lines
17 KiB
C++
548 lines
17 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 .
|
|
*/
|
|
|
|
#ifndef _FRMTOOL_HXX
|
|
#define _FRMTOOL_HXX
|
|
|
|
#include "swtypes.hxx"
|
|
#include "layfrm.hxx"
|
|
#include "frmatr.hxx"
|
|
#include "swcache.hxx"
|
|
// #i96772#
|
|
#include <editeng/lrspitem.hxx>
|
|
|
|
class SwPageFrm;
|
|
class SwFlyFrm;
|
|
class SwCntntFrm;
|
|
class SwRootFrm;
|
|
class SwDoc;
|
|
class SwAttrSet;
|
|
class SdrObject;
|
|
class BigInt;
|
|
class SvxBrushItem;
|
|
class XFillStyleItem;
|
|
class XFillGradientItem;
|
|
class SdrMarkList;
|
|
class SwNodeIndex;
|
|
class OutputDevice;
|
|
class SwPageDesc;
|
|
|
|
#define FAR_AWAY LONG_MAX - 20000 // initial position of a Fly
|
|
#define BROWSE_HEIGHT 56700L * 10L // 10 Meters
|
|
#define GRFNUM_NO 0
|
|
#define GRFNUM_YES 1
|
|
#define GRFNUM_REPLACE 2
|
|
|
|
// draw background with brush or graphics
|
|
// The 6th parameter indicates that the method should consider background
|
|
// transparency, saved in the color of the brush item.
|
|
void DrawGraphic( const SvxBrushItem *, const XFillStyleItem*, const XFillGradientItem*, OutputDevice *,
|
|
const SwRect &rOrg, const SwRect &rOut, const sal_uInt8 nGrfNum = GRFNUM_NO,
|
|
const sal_Bool bConsiderBackgroundTransparency = sal_False );
|
|
|
|
// method to align rectangle.
|
|
// Created declaration here to avoid <extern> declarations
|
|
void SwAlignRect( SwRect &rRect, const ViewShell *pSh );
|
|
|
|
// method to align graphic rectangle
|
|
// Created declaration here to avoid <extern> declarations
|
|
void SwAlignGrfRect( SwRect *pGrfRect, const OutputDevice &rOut );
|
|
|
|
// get Fly, if no List is given use the current shell
|
|
// Implementation in feshview.cxx
|
|
SwFlyFrm *GetFlyFromMarked( const SdrMarkList *pLst, ViewShell *pSh );
|
|
|
|
sal_uLong SqRt( BigInt nX );
|
|
|
|
SwFrm *SaveCntnt( SwLayoutFrm *pLay, SwFrm *pStart );
|
|
void RestoreCntnt( SwFrm *pSav, SwLayoutFrm *pParent, SwFrm *pSibling, bool bGrow );
|
|
|
|
// Get CntntNodes, create CntntFrms, and add them to LayFrm.
|
|
void _InsertCnt( SwLayoutFrm *pLay, SwDoc *pDoc, sal_uLong nIndex,
|
|
sal_Bool bPages = sal_False, sal_uLong nEndIndex = 0,
|
|
SwFrm *pPrv = 0 );
|
|
|
|
// Creation of frames for a specific section (uses _InsertCnt)
|
|
void MakeFrms( SwDoc *pDoc, const SwNodeIndex &rSttIdx,
|
|
const SwNodeIndex &rEndIdx );
|
|
|
|
// prevent creation of Flys in _InsertCnt, e.g. for table headlines
|
|
extern bool bDontCreateObjects;
|
|
|
|
// for FlyCnts, see SwFlyAtCntFrm::MakeAll()
|
|
extern bool bSetCompletePaintOnInvalidate;
|
|
|
|
// for table settings via keyboard
|
|
long CalcRowRstHeight( SwLayoutFrm *pRow );
|
|
long CalcHeightWidthFlys( const SwFrm *pFrm );
|
|
|
|
SwPageFrm *InsertNewPage( SwPageDesc &rDesc, SwFrm *pUpper,
|
|
bool bOdd, bool bFirst, bool bInsertEmpty, sal_Bool bFtn,
|
|
SwFrm *pSibling );
|
|
|
|
// connect Flys with page
|
|
void RegistFlys( SwPageFrm*, const SwLayoutFrm* );
|
|
|
|
// notification of Fly's background if needed
|
|
void Notify( SwFlyFrm *pFly, SwPageFrm *pOld, const SwRect &rOld,
|
|
const SwRect* pOldRect = 0 );
|
|
|
|
void Notify_Background( const SdrObject* pObj,
|
|
SwPageFrm* pPage,
|
|
const SwRect& rRect,
|
|
const PrepareHint eHint,
|
|
const sal_Bool bInva );
|
|
|
|
const SwFrm* GetVirtualUpper( const SwFrm* pFrm, const Point& rPos );
|
|
|
|
bool Is_Lower_Of( const SwFrm *pCurrFrm, const SdrObject* pObj );
|
|
|
|
// FIXME: EasyHack (refactoring): rename method and parameter name in all files
|
|
const SwFrm *FindKontext( const SwFrm *pFrm, sal_uInt16 nAdditionalKontextTyp );
|
|
|
|
bool IsFrmInSameKontext( const SwFrm *pInnerFrm, const SwFrm *pFrm );
|
|
|
|
const SwFrm * FindPage( const SwRect &rRect, const SwFrm *pPage );
|
|
|
|
// used by SwCntntNode::GetFrm and SwFlyFrm::GetFrm
|
|
SwFrm* GetFrmOfModify( const SwRootFrm* pLayout,
|
|
SwModify const&,
|
|
sal_uInt16 const nFrmType,
|
|
const Point* = 0,
|
|
const SwPosition *pPos = 0,
|
|
const sal_Bool bCalcFrm = sal_False );
|
|
|
|
// Should extra data (reline stroke, line numbers) be painted?
|
|
bool IsExtraData( const SwDoc *pDoc );
|
|
|
|
// #i11760# - method declaration <CalcCntnt(..)>
|
|
void CalcCntnt( SwLayoutFrm *pLay,
|
|
bool bNoColl = false,
|
|
bool bNoCalcFollow = false );
|
|
|
|
// Notify classes memorize the current sizes in their constructor and do
|
|
// the necessary notifications in their destructor if needed
|
|
class SwFrmNotify
|
|
{
|
|
protected:
|
|
SwFrm *pFrm;
|
|
const SwRect aFrm;
|
|
const SwRect aPrt;
|
|
SwTwips mnFlyAnchorOfst;
|
|
SwTwips mnFlyAnchorOfstNoWrap;
|
|
sal_Bool bHadFollow;
|
|
sal_Bool bInvaKeep;
|
|
sal_Bool bValidSize;
|
|
// #i49383#
|
|
bool mbFrmDeleted;
|
|
|
|
public:
|
|
SwFrmNotify( SwFrm *pFrm );
|
|
~SwFrmNotify();
|
|
|
|
const SwRect &Frm() const { return aFrm; }
|
|
const SwRect &Prt() const { return aPrt; }
|
|
void SetInvaKeep() { bInvaKeep = sal_True; }
|
|
// #i49383#
|
|
void FrmDeleted()
|
|
{
|
|
mbFrmDeleted = true;
|
|
}
|
|
};
|
|
|
|
class SwLayNotify : public SwFrmNotify
|
|
{
|
|
sal_Bool bLowersComplete;
|
|
|
|
SwLayoutFrm *GetLay() { return (SwLayoutFrm*)pFrm; }
|
|
|
|
public:
|
|
SwLayNotify( SwLayoutFrm *pLayFrm );
|
|
~SwLayNotify();
|
|
|
|
void SetLowersComplete( sal_Bool b ) { bLowersComplete = b; }
|
|
sal_Bool IsLowersComplete() { return bLowersComplete; }
|
|
};
|
|
|
|
class SwFlyNotify : public SwLayNotify
|
|
{
|
|
SwPageFrm *pOldPage;
|
|
const SwRect aFrmAndSpace;
|
|
SwFlyFrm *GetFly() { return (SwFlyFrm*)pFrm; }
|
|
|
|
public:
|
|
SwFlyNotify( SwFlyFrm *pFlyFrm );
|
|
~SwFlyNotify();
|
|
|
|
SwPageFrm *GetOldPage() const { return pOldPage; }
|
|
};
|
|
|
|
class SwCntntNotify : public SwFrmNotify
|
|
{
|
|
private:
|
|
// #i11859#
|
|
bool mbChkHeightOfLastLine;
|
|
SwTwips mnHeightOfLastLine;
|
|
|
|
// #i25029#
|
|
bool mbInvalidatePrevPrtArea;
|
|
bool mbBordersJoinedWithPrev;
|
|
|
|
SwCntntFrm *GetCnt() { return (SwCntntFrm*)pFrm; }
|
|
|
|
public:
|
|
SwCntntNotify( SwCntntFrm *pCntFrm );
|
|
~SwCntntNotify();
|
|
|
|
// #i25029#
|
|
void SetInvalidatePrevPrtArea()
|
|
{
|
|
mbInvalidatePrevPrtArea = true;
|
|
}
|
|
|
|
void SetBordersJoinedWithPrev()
|
|
{
|
|
mbBordersJoinedWithPrev = true;
|
|
}
|
|
};
|
|
|
|
// SwBorderAttrs encapsulates the calculation for margin attributes including
|
|
// border. The whole class is cached.
|
|
//
|
|
// WARNING! If more attributes should be cached also adjust the method
|
|
// Modify::Modify!
|
|
class SwBorderAttrs : public SwCacheObj
|
|
{
|
|
const SwAttrSet &rAttrSet;
|
|
const SvxULSpaceItem &rUL;
|
|
// #i96772#
|
|
SvxLRSpaceItem rLR;
|
|
const SvxBoxItem &rBox;
|
|
const SvxShadowItem &rShadow;
|
|
const Size aFrmSize;
|
|
|
|
// Is it a frame that can have a margin without a border?
|
|
sal_Bool bBorderDist : 1;
|
|
|
|
// the following bool values set the cached values to INVALID - until they
|
|
// are calculated for the first time
|
|
sal_Bool bTopLine : 1;
|
|
sal_Bool bBottomLine : 1;
|
|
sal_Bool bLeftLine : 1;
|
|
sal_Bool bRightLine : 1;
|
|
sal_Bool bTop : 1;
|
|
sal_Bool bBottom : 1;
|
|
sal_Bool bLine : 1;
|
|
|
|
sal_Bool bIsLine : 1; // border on at least one side?
|
|
|
|
sal_Bool bCacheGetLine : 1; // cache GetTopLine(), GetBottomLine()?
|
|
sal_Bool bCachedGetTopLine : 1; // is GetTopLine() cached?
|
|
sal_Bool bCachedGetBottomLine : 1; // is GetBottomLine() cached?
|
|
// Booleans indicate that <bJoinedWithPrev> and <bJoinedWithNext> are
|
|
// cached and valid.
|
|
// Caching depends on value of <bCacheGetLine>.
|
|
mutable sal_Bool bCachedJoinedWithPrev : 1;
|
|
mutable sal_Bool bCachedJoinedWithNext : 1;
|
|
// Booleans indicate that borders are joined with previous/next frame.
|
|
sal_Bool bJoinedWithPrev :1;
|
|
sal_Bool bJoinedWithNext :1;
|
|
|
|
// The cached values (un-defined until calculated for the first time)
|
|
sal_uInt16 nTopLine,
|
|
nBottomLine,
|
|
nLeftLine,
|
|
nRightLine,
|
|
nTop,
|
|
nBottom,
|
|
nGetTopLine,
|
|
nGetBottomLine;
|
|
|
|
// only calculate lines and shadow
|
|
void _CalcTopLine();
|
|
void _CalcBottomLine();
|
|
void _CalcLeftLine();
|
|
void _CalcRightLine();
|
|
|
|
// lines + shadow + margin
|
|
void _CalcTop();
|
|
void _CalcBottom();
|
|
|
|
void _IsLine();
|
|
|
|
// #i25029# - If <_pPrevFrm> is set, its value is taken for testing, if
|
|
// borders/shadow have to be joined with previous frame.
|
|
void _GetTopLine ( const SwFrm& _rFrm,
|
|
const SwFrm* _pPrevFrm = 0L );
|
|
void _GetBottomLine( const SwFrm& _rFrm );
|
|
|
|
// calculate cached values <bJoinedWithPrev> and <bJoinedWithNext>
|
|
// #i25029# - If <_pPrevFrm> is set, its value is taken for testing, if
|
|
// borders/shadow have to be joined with previous frame.
|
|
void _CalcJoinedWithPrev( const SwFrm& _rFrm,
|
|
const SwFrm* _pPrevFrm = 0L );
|
|
void _CalcJoinedWithNext( const SwFrm& _rFrm );
|
|
|
|
// internal helper method for _CalcJoinedWithPrev and _CalcJoinedWithNext
|
|
sal_Bool _JoinWithCmp( const SwFrm& _rCallerFrm,
|
|
const SwFrm& _rCmpFrm ) const;
|
|
|
|
// Are the left and right line and the LRSpace equal?
|
|
sal_Bool CmpLeftRight( const SwBorderAttrs &rCmpAttrs,
|
|
const SwFrm *pCaller,
|
|
const SwFrm *pCmp ) const;
|
|
|
|
public:
|
|
DECL_FIXEDMEMPOOL_NEWDEL(SwBorderAttrs)
|
|
|
|
SwBorderAttrs( const SwModify *pOwner, const SwFrm *pConstructor );
|
|
~SwBorderAttrs();
|
|
|
|
inline const SwAttrSet &GetAttrSet() const { return rAttrSet; }
|
|
inline const SvxULSpaceItem &GetULSpace() const { return rUL; }
|
|
inline const SvxLRSpaceItem &GetLRSpace() const { return rLR; }
|
|
inline const SvxBoxItem &GetBox() const { return rBox; }
|
|
inline const SvxShadowItem &GetShadow() const { return rShadow; }
|
|
|
|
inline sal_uInt16 CalcTopLine() const;
|
|
inline sal_uInt16 CalcBottomLine() const;
|
|
inline sal_uInt16 CalcLeftLine() const;
|
|
inline sal_uInt16 CalcRightLine() const;
|
|
inline sal_uInt16 CalcTop() const;
|
|
inline sal_uInt16 CalcBottom() const;
|
|
long CalcLeft( const SwFrm *pCaller ) const;
|
|
long CalcRight( const SwFrm *pCaller ) const;
|
|
|
|
inline sal_Bool IsLine() const;
|
|
|
|
inline const Size &GetSize() const { return aFrmSize; }
|
|
|
|
inline sal_Bool IsBorderDist() const { return bBorderDist; }
|
|
|
|
// Should upper (or lower) border be evaluated for this frame?
|
|
// #i25029# - If <_pPrevFrm> is set, its value is taken for testing, if
|
|
// borders/shadow have to be joined with previous frame.
|
|
inline sal_uInt16 GetTopLine ( const SwFrm& _rFrm,
|
|
const SwFrm* _pPrevFrm = 0L ) const;
|
|
inline sal_uInt16 GetBottomLine( const SwFrm& _rFrm ) const;
|
|
inline void SetGetCacheLine( sal_Bool bNew ) const;
|
|
|
|
// Accessors for cached values <bJoinedWithPrev> and <bJoinedWithPrev>
|
|
// #i25029# - If <_pPrevFrm> is set, its value is taken for testing, if
|
|
// borders/shadow have to be joined with previous frame.
|
|
sal_Bool JoinedWithPrev( const SwFrm& _rFrm,
|
|
const SwFrm* _pPrevFrm = 0L ) const;
|
|
sal_Bool JoinedWithNext( const SwFrm& _rFrm ) const;
|
|
};
|
|
|
|
class SwBorderAttrAccess : public SwCacheAccess
|
|
{
|
|
const SwFrm *pConstructor; //opt: for passing on to SwBorderAttrs
|
|
|
|
protected:
|
|
virtual SwCacheObj *NewObj();
|
|
|
|
public:
|
|
SwBorderAttrAccess( SwCache &rCache, const SwFrm *pOwner );
|
|
|
|
SwBorderAttrs *Get();
|
|
};
|
|
|
|
// Iterator for draw objects of a page. The objects will be iterated sorted by
|
|
// their Z-order. Iterating is not cheap since for each operation the _whole_
|
|
// SortArray needs to be traversed.
|
|
class SwOrderIter
|
|
{
|
|
const SwPageFrm *pPage;
|
|
const SdrObject *pCurrent;
|
|
const sal_Bool bFlysOnly;
|
|
|
|
public:
|
|
SwOrderIter( const SwPageFrm *pPage, sal_Bool bFlysOnly = sal_True );
|
|
|
|
void Current( const SdrObject *pNew ) { pCurrent = pNew; }
|
|
const SdrObject *Current() const { return pCurrent; }
|
|
const SdrObject *operator()() const { return pCurrent; }
|
|
const SdrObject *Top();
|
|
const SdrObject *Bottom();
|
|
const SdrObject *Next();
|
|
const SdrObject *Prev();
|
|
};
|
|
|
|
class StackHack
|
|
{
|
|
static sal_uInt8 nCnt;
|
|
static sal_Bool bLocked;
|
|
|
|
public:
|
|
StackHack()
|
|
{
|
|
if ( ++StackHack::nCnt > 50 )
|
|
StackHack::bLocked = sal_True;
|
|
}
|
|
~StackHack()
|
|
{
|
|
if ( --StackHack::nCnt < 5 )
|
|
StackHack::bLocked = sal_False;
|
|
}
|
|
|
|
static sal_Bool IsLocked() { return StackHack::bLocked; }
|
|
static sal_uInt8 Count() { return StackHack::nCnt; }
|
|
};
|
|
|
|
// Should upper (or lower) border be evaluated for this frame?
|
|
// #i25029# - If <_pPrevFrm> is set, its value is taken for testing, if
|
|
// borders/shadow have to be joined with previous frame.
|
|
inline sal_uInt16 SwBorderAttrs::GetTopLine ( const SwFrm& _rFrm,
|
|
const SwFrm* _pPrevFrm ) const
|
|
{
|
|
if ( !bCachedGetTopLine || _pPrevFrm )
|
|
{
|
|
const_cast<SwBorderAttrs*>(this)->_GetTopLine( _rFrm, _pPrevFrm );
|
|
}
|
|
return nGetTopLine;
|
|
}
|
|
inline sal_uInt16 SwBorderAttrs::GetBottomLine( const SwFrm& _rFrm ) const
|
|
{
|
|
if ( !bCachedGetBottomLine )
|
|
const_cast<SwBorderAttrs*>(this)->_GetBottomLine( _rFrm );
|
|
return nGetBottomLine;
|
|
}
|
|
inline void SwBorderAttrs::SetGetCacheLine( sal_Bool bNew ) const
|
|
{
|
|
((SwBorderAttrs*)this)->bCacheGetLine = bNew;
|
|
((SwBorderAttrs*)this)->bCachedGetBottomLine =
|
|
((SwBorderAttrs*)this)->bCachedGetTopLine = sal_False;
|
|
// invalidate cache for values <bJoinedWithPrev> and <bJoinedWithNext>
|
|
bCachedJoinedWithPrev = sal_False;
|
|
bCachedJoinedWithNext = sal_False;
|
|
}
|
|
|
|
inline sal_uInt16 SwBorderAttrs::CalcTopLine() const
|
|
{
|
|
if ( bTopLine )
|
|
((SwBorderAttrs*)this)->_CalcTopLine();
|
|
return nTopLine;
|
|
}
|
|
inline sal_uInt16 SwBorderAttrs::CalcBottomLine() const
|
|
{
|
|
if ( bBottomLine )
|
|
((SwBorderAttrs*)this)->_CalcBottomLine();
|
|
return nBottomLine;
|
|
}
|
|
inline sal_uInt16 SwBorderAttrs::CalcLeftLine() const
|
|
{
|
|
if ( bLeftLine )
|
|
((SwBorderAttrs*)this)->_CalcLeftLine();
|
|
return nLeftLine;
|
|
}
|
|
inline sal_uInt16 SwBorderAttrs::CalcRightLine() const
|
|
{
|
|
if ( bRightLine )
|
|
((SwBorderAttrs*)this)->_CalcRightLine();
|
|
return nRightLine;
|
|
}
|
|
inline sal_uInt16 SwBorderAttrs::CalcTop() const
|
|
{
|
|
if ( bTop )
|
|
((SwBorderAttrs*)this)->_CalcTop();
|
|
return nTop;
|
|
}
|
|
inline sal_uInt16 SwBorderAttrs::CalcBottom() const
|
|
{
|
|
if ( bBottom )
|
|
((SwBorderAttrs*)this)->_CalcBottom();
|
|
return nBottom;
|
|
}
|
|
inline sal_Bool SwBorderAttrs::IsLine() const
|
|
{
|
|
if ( bLine )
|
|
((SwBorderAttrs*)this)->_IsLine();
|
|
return bIsLine;
|
|
}
|
|
|
|
/** method to determine the spacing values of a frame
|
|
|
|
#i28701#
|
|
Values only provided for flow frames (table, section or text frames)
|
|
Note: line spacing value is only determined for text frames
|
|
#i102458#
|
|
Add output parameter <obIsLineSpacingProportional>
|
|
|
|
@param rFrm
|
|
input parameter - frame, for which the spacing values are determined.
|
|
|
|
@param onPrevLowerSpacing
|
|
output parameter - lower spacing of the frame in SwTwips
|
|
|
|
@param onPrevLineSpacing
|
|
output parameter - line spacing of the frame in SwTwips
|
|
|
|
@param obIsLineSpacingProportional
|
|
*/
|
|
void GetSpacingValuesOfFrm( const SwFrm& rFrm,
|
|
SwTwips& onLowerSpacing,
|
|
SwTwips& onLineSpacing,
|
|
bool& obIsLineSpacingProportional );
|
|
|
|
/** method to get the content of the table cell
|
|
|
|
Content from any nested tables will be omitted.
|
|
Note: line spacing value is only determined for text frames
|
|
|
|
@param rCell_
|
|
input parameter - the cell which should be searched for content.
|
|
|
|
return
|
|
pointer to the found content frame or 0
|
|
*/
|
|
|
|
const SwCntntFrm* GetCellCntnt( const SwLayoutFrm& rCell_ );
|
|
|
|
|
|
/** helper class to check if a frame has been deleted during an operation
|
|
* WARNING! This should only be used as a last and desperate means to make the
|
|
* code robust.
|
|
*/
|
|
|
|
class SwDeletionChecker
|
|
{
|
|
private:
|
|
const SwFrm* mpFrm;
|
|
const SwModify* mpRegIn;
|
|
|
|
public:
|
|
SwDeletionChecker( const SwFrm* pFrm )
|
|
: mpFrm( pFrm ),
|
|
mpRegIn( pFrm ? const_cast<SwFrm*>(pFrm)->GetRegisteredIn() : 0 )
|
|
{
|
|
}
|
|
|
|
/**
|
|
* return
|
|
* true if mpFrm != 0 and mpFrm is not client of pRegIn
|
|
* false otherwise
|
|
*/
|
|
bool HasBeenDeleted();
|
|
};
|
|
|
|
#endif
|
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|