Files
libreoffice/sw/source/core/inc/frmtool.hxx
Miklos Vajna a1bef744a2 SwXTextFrame: use XFillGradientItem instead of extending SvxBrushItem
Change-Id: I8863cbe1d01fceac0f4b4152264c1464c90e02b8
2013-01-28 13:49:41 +01:00

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: */