/************************************************************** * * 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 * * Unless required by applicable law or agreed to in writing, * software distributed under the License is distributed on an * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY * KIND, either express or implied. See the License for the * specific language governing permissions and limitations * under the License. * *************************************************************/ // MARKER(update_precomp.py): autogen include statement, do not remove #include "precompiled_sw.hxx" #include <hintids.hxx> #include <hints.hxx> #include <tools/pstm.hxx> #include <vcl/outdev.hxx> #include <svl/itemiter.hxx> #include <editeng/brshitem.hxx> #include <editeng/keepitem.hxx> #include <editeng/brkitem.hxx> #include <fmtornt.hxx> #include <pagefrm.hxx> #include <section.hxx> #include <rootfrm.hxx> #include <cntfrm.hxx> #include <dcontact.hxx> #include <anchoreddrawobject.hxx> #include <fmtanchr.hxx> #include <viewsh.hxx> #include <viewimp.hxx> #include "viewopt.hxx" #include <doc.hxx> #include <fesh.hxx> #include <docsh.hxx> #include <flyfrm.hxx> #include <frmtool.hxx> #include <ftninfo.hxx> #include <dflyobj.hxx> #include <fmtclbl.hxx> #include <fmtfordr.hxx> #include <fmtfsize.hxx> #include <fmtpdsc.hxx> #include <txtftn.hxx> #include <fmtftn.hxx> #include <fmtsrnd.hxx> #include <ftnfrm.hxx> #include <tabfrm.hxx> #include <htmltbl.hxx> #include <flyfrms.hxx> #include <sectfrm.hxx> #include <fmtclds.hxx> #include <txtfrm.hxx> #include <ndtxt.hxx> #include <bodyfrm.hxx> #include <cellfrm.hxx> #include <dbg_lay.hxx> #include <editeng/frmdiritem.hxx> // OD 2004-05-24 #i28701# #include <sortedobjs.hxx> using namespace ::com::sun::star; /************************************************************************* |* |* SwFrm::SwFrm() |* |* Ersterstellung AK 12-Feb-1991 |* Letzte Aenderung MA 05. Apr. 94 |* |*************************************************************************/ SwFrm::SwFrm( SwModify *pMod, SwFrm* pSib ) : SwClient( pMod ), //Solution:Add a member to identify if the acc table should dispose bIfAccTableShouldDisposing( sal_False ), // --> OD 2006-05-10 #i65250# mnFrmId( SwFrm::mnLastFrmId++ ), // <-- mpRoot( pSib ? pSib->getRootFrm() : 0 ), pUpper( 0 ), pNext( 0 ), pPrev( 0 ), pDrawObjs( 0 ) , bInfBody( sal_False ) , bInfTab ( sal_False ) , bInfFly ( sal_False ) , bInfFtn ( sal_False ) , bInfSct ( sal_False ) { #ifdef DBG_UTIL bFlag01 = bFlag02 = bFlag03 = bFlag04 = bFlag05 = 0; #endif ASSERT( pMod, "Kein Frameformat uebergeben." ); bInvalidR2L = bInvalidVert = 1; //Badaa: 2008-04-18 * Support for Classical Mongolian Script (SCMS) joint with Jiayanmin bDerivedR2L = bDerivedVert = bRightToLeft = bVertical = bReverse = bVertLR = 0; bValidPos = bValidPrtArea = bValidSize = bValidLineNum = bRetouche = bFixSize = bColLocked = sal_False; bCompletePaint = bInfInvalid = sal_True; } bool SwFrm::KnowsFormat( const SwFmt& rFmt ) const { return GetRegisteredIn() == &rFmt; } void SwFrm::RegisterToFormat( SwFmt& rFmt ) { rFmt.Add( this ); } void SwFrm::CheckDir( sal_uInt16 nDir, sal_Bool bVert, sal_Bool bOnlyBiDi, sal_Bool bBrowse ) { if( FRMDIR_ENVIRONMENT == nDir || ( bVert && bOnlyBiDi ) ) { bDerivedVert = 1; if( FRMDIR_ENVIRONMENT == nDir ) bDerivedR2L = 1; SetDirFlags( bVert ); } else if( bVert ) { bInvalidVert = 0; if( FRMDIR_HORI_LEFT_TOP == nDir || FRMDIR_HORI_RIGHT_TOP == nDir || bBrowse ) //Badaa: 2008-04-18 * Support for Classical Mongolian Script (SCMS) joint with Jiayanmin { bVertical = 0; bVertLR = 0; } else { bVertical = 1; if(FRMDIR_VERT_TOP_RIGHT == nDir) bVertLR = 0; else if(FRMDIR_VERT_TOP_LEFT==nDir) bVertLR = 1; } } else { bInvalidR2L = 0; if( FRMDIR_HORI_RIGHT_TOP == nDir ) bRightToLeft = 1; else bRightToLeft = 0; } } void SwFrm::CheckDirection( sal_Bool bVert ) { if( bVert ) { if( !IsHeaderFrm() && !IsFooterFrm() ) { bDerivedVert = 1; SetDirFlags( bVert ); } } else { bDerivedR2L = 1; SetDirFlags( bVert ); } } void SwSectionFrm::CheckDirection( sal_Bool bVert ) { const SwFrmFmt* pFmt = GetFmt(); if( pFmt ) { const ViewShell *pSh = getRootFrm()->GetCurrShell(); const sal_Bool bBrowseMode = pSh && pSh->GetViewOptions()->getBrowseMode(); CheckDir(((SvxFrameDirectionItem&)pFmt->GetFmtAttr(RES_FRAMEDIR)).GetValue(), bVert, sal_True, bBrowseMode ); } else SwFrm::CheckDirection( bVert ); } void SwFlyFrm::CheckDirection( sal_Bool bVert ) { const SwFrmFmt* pFmt = GetFmt(); if( pFmt ) { const ViewShell *pSh = getRootFrm()->GetCurrShell(); const sal_Bool bBrowseMode = pSh && pSh->GetViewOptions()->getBrowseMode(); CheckDir(((SvxFrameDirectionItem&)pFmt->GetFmtAttr(RES_FRAMEDIR)).GetValue(), bVert, sal_False, bBrowseMode ); } else SwFrm::CheckDirection( bVert ); } void SwTabFrm::CheckDirection( sal_Bool bVert ) { const SwFrmFmt* pFmt = GetFmt(); if( pFmt ) { const ViewShell *pSh = getRootFrm()->GetCurrShell(); const sal_Bool bBrowseMode = pSh && pSh->GetViewOptions()->getBrowseMode(); CheckDir(((SvxFrameDirectionItem&)pFmt->GetFmtAttr(RES_FRAMEDIR)).GetValue(), bVert, sal_True, bBrowseMode ); } else SwFrm::CheckDirection( bVert ); } void SwCellFrm::CheckDirection( sal_Bool bVert ) { const SwFrmFmt* pFmt = GetFmt(); const SfxPoolItem* pItem; // --> FME 2006-03-30 #b6402837# Check if the item is set, before actually // using it. Otherwise the dynamic pool default is used, which may be set // to LTR in case of OOo 1.0 documents. // <-- if( pFmt && SFX_ITEM_SET == pFmt->GetItemState( RES_FRAMEDIR, sal_True, &pItem ) ) { const SvxFrameDirectionItem* pFrmDirItem = static_cast<const SvxFrameDirectionItem*>(pItem); const ViewShell *pSh = getRootFrm()->GetCurrShell(); const sal_Bool bBrowseMode = pSh && pSh->GetViewOptions()->getBrowseMode(); CheckDir( pFrmDirItem->GetValue(), bVert, sal_False, bBrowseMode ); } else SwFrm::CheckDirection( bVert ); } void SwTxtFrm::CheckDirection( sal_Bool bVert ) { const ViewShell *pSh = getRootFrm()->GetCurrShell(); const sal_Bool bBrowseMode = pSh && pSh->GetViewOptions()->getBrowseMode(); CheckDir( GetTxtNode()->GetSwAttrSet().GetFrmDir().GetValue(), bVert, sal_True, bBrowseMode ); } /*************************************************************************/ void SwFrm::Modify( const SfxPoolItem* pOld, const SfxPoolItem * pNew ) { sal_uInt8 nInvFlags = 0; if( pNew && RES_ATTRSET_CHG == pNew->Which() ) { SfxItemIter aNIter( *((SwAttrSetChg*)pNew)->GetChgSet() ); SfxItemIter aOIter( *((SwAttrSetChg*)pOld)->GetChgSet() ); while( sal_True ) { _UpdateAttrFrm( (SfxPoolItem*)aOIter.GetCurItem(), (SfxPoolItem*)aNIter.GetCurItem(), nInvFlags ); if( aNIter.IsAtEnd() ) break; aNIter.NextItem(); aOIter.NextItem(); } } else _UpdateAttrFrm( pOld, pNew, nInvFlags ); if ( nInvFlags != 0 ) { SwPageFrm *pPage = FindPageFrm(); InvalidatePage( pPage ); if ( nInvFlags & 0x01 ) { _InvalidatePrt(); if( !GetPrev() && IsTabFrm() && IsInSct() ) FindSctFrm()->_InvalidatePrt(); } if ( nInvFlags & 0x02 ) _InvalidateSize(); if ( nInvFlags & 0x04 ) _InvalidatePos(); if ( nInvFlags & 0x08 ) SetCompletePaint(); SwFrm *pNxt; if ( nInvFlags & 0x30 && 0 != (pNxt = GetNext()) ) { pNxt->InvalidatePage( pPage ); if ( nInvFlags & 0x10 ) pNxt->_InvalidatePos(); if ( nInvFlags & 0x20 ) pNxt->SetCompletePaint(); } } } void SwFrm::_UpdateAttrFrm( const SfxPoolItem *pOld, const SfxPoolItem *pNew, sal_uInt8 &rInvFlags ) { sal_uInt16 nWhich = pOld ? pOld->Which() : pNew ? pNew->Which() : 0; switch( nWhich ) { case RES_BOX: case RES_SHADOW: Prepare( PREP_FIXSIZE_CHG ); // hier kein break ! case RES_LR_SPACE: case RES_UL_SPACE: rInvFlags |= 0x0B; break; case RES_HEADER_FOOTER_EAT_SPACING: rInvFlags |= 0x03; break; case RES_BACKGROUND: rInvFlags |= 0x28; break; case RES_KEEP: rInvFlags |= 0x04; break; case RES_FRM_SIZE: ReinitializeFrmSizeAttrFlags(); rInvFlags |= 0x13; break; case RES_FMT_CHG: rInvFlags |= 0x0F; break; case RES_ROW_SPLIT: { if ( IsRowFrm() ) { sal_Bool bInFollowFlowRow = 0 != IsInFollowFlowRow(); if ( bInFollowFlowRow || 0 != IsInSplitTableRow() ) { SwTabFrm* pTab = FindTabFrm(); if ( bInFollowFlowRow ) pTab = pTab->FindMaster(); pTab->SetRemoveFollowFlowLinePending( sal_True ); } } break; } case RES_COL: ASSERT( sal_False, "Spalten fuer neuen FrmTyp?" ); break; default: /* do Nothing */; } } /************************************************************************* |* |* SwFrm::Prepare() |* Ersterstellung MA 13. Apr. 93 |* Letzte Aenderung MA 26. Jun. 96 |* |*************************************************************************/ void SwFrm::Prepare( const PrepareHint, const void *, sal_Bool ) { /* Do nothing */ } /************************************************************************* |* |* SwFrm::InvalidatePage() |* Beschreibung: Invalidiert die Seite, in der der Frm gerade steht. |* Je nachdem ob es ein Layout, Cntnt oder FlyFrm ist wird die Seite |* entsprechend Invalidiert. |* Ersterstellung MA 22. Jul. 92 |* Letzte Aenderung MA 14. Oct. 94 |* |*************************************************************************/ void SwFrm::InvalidatePage( const SwPageFrm *pPage ) const { if ( !pPage ) { pPage = FindPageFrm(); // --> OD 2004-07-02 #i28701# - for at-character and as-character // anchored Writer fly frames additionally invalidate also page frame // its 'anchor character' is on. if ( pPage && pPage->GetUpper() && IsFlyFrm() ) { const SwFlyFrm* pFlyFrm = static_cast<const SwFlyFrm*>(this); if ( pFlyFrm->IsAutoPos() || pFlyFrm->IsFlyInCntFrm() ) { // --> OD 2004-09-23 #i33751#, #i34060# - method <GetPageFrmOfAnchor()> // is replaced by method <FindPageFrmOfAnchor()>. It's return value // have to be checked. SwPageFrm* pPageFrmOfAnchor = const_cast<SwFlyFrm*>(pFlyFrm)->FindPageFrmOfAnchor(); if ( pPageFrmOfAnchor && pPageFrmOfAnchor != pPage ) // <-- { InvalidatePage( pPageFrmOfAnchor ); } } } // <-- } if ( pPage && pPage->GetUpper() ) { if ( pPage->GetFmt()->GetDoc()->IsInDtor() ) return; SwRootFrm *pRoot = (SwRootFrm*)pPage->GetUpper(); const SwFlyFrm *pFly = FindFlyFrm(); if ( IsCntntFrm() ) { if ( pRoot->IsTurboAllowed() ) { // JP 21.09.95: wenn sich der ContentFrame 2 mal eintragen // will, kann es doch eine TurboAction bleiben. // ODER???? if ( !pRoot->GetTurbo() || this == pRoot->GetTurbo() ) pRoot->SetTurbo( (const SwCntntFrm*)this ); else { pRoot->DisallowTurbo(); //Die Seite des Turbo koennte eine andere als die meinige //sein, deshalb muss sie invalidiert werden. const SwFrm *pTmp = pRoot->GetTurbo(); pRoot->ResetTurbo(); pTmp->InvalidatePage(); } } if ( !pRoot->GetTurbo() ) { if ( pFly ) { if( !pFly->IsLocked() ) { if ( pFly->IsFlyInCntFrm() ) { pPage->InvalidateFlyInCnt(); ((SwFlyInCntFrm*)pFly)->InvalidateCntnt(); pFly->GetAnchorFrm()->InvalidatePage(); } else pPage->InvalidateFlyCntnt(); } } else pPage->InvalidateCntnt(); } } else { pRoot->DisallowTurbo(); if ( pFly ) { if ( !pFly->IsLocked() ) { if ( pFly->IsFlyInCntFrm() ) { pPage->InvalidateFlyInCnt(); ((SwFlyInCntFrm*)pFly)->InvalidateLayout(); pFly->GetAnchorFrm()->InvalidatePage(); } else pPage->InvalidateFlyLayout(); } } else pPage->InvalidateLayout(); if ( pRoot->GetTurbo() ) { const SwFrm *pTmp = pRoot->GetTurbo(); pRoot->ResetTurbo(); pTmp->InvalidatePage(); } } pRoot->SetIdleFlags(); const SwTxtFrm *pTxtFrm = dynamic_cast< const SwTxtFrm * >(this); if (pTxtFrm) { const SwTxtNode *pTxtNode = pTxtFrm->GetTxtNode(); if (pTxtNode && pTxtNode->IsGrammarCheckDirty()) pRoot->SetNeedGrammarCheck( sal_True ); } } } /************************************************************************* |* |* SwFrm::ChgSize() |* |* Ersterstellung AK 15-Feb-1991 |* Letzte Aenderung MA 18. Nov. 98 |* |*************************************************************************/ Size SwFrm::ChgSize( const Size& aNewSize ) { bFixSize = sal_True; const Size aOldSize( Frm().SSize() ); if ( aNewSize == aOldSize ) return aOldSize; if ( GetUpper() ) { SWRECTFN2( this ) SwRect aNew( Point(0,0), aNewSize ); (aFrm.*fnRect->fnSetWidth)( (aNew.*fnRect->fnGetWidth)() ); long nNew = (aNew.*fnRect->fnGetHeight)(); long nDiff = nNew - (aFrm.*fnRect->fnGetHeight)(); if( nDiff ) { if ( GetUpper()->IsFtnBossFrm() && HasFixSize() && NA_GROW_SHRINK != ((SwFtnBossFrm*)GetUpper())->NeighbourhoodAdjustment( this ) ) { (aFrm.*fnRect->fnSetHeight)( nNew ); SwTwips nReal = ((SwLayoutFrm*)this)->AdjustNeighbourhood(nDiff); if ( nReal != nDiff ) (aFrm.*fnRect->fnSetHeight)( nNew - nDiff + nReal ); } else { // OD 24.10.2002 #97265# - grow/shrink not for neighbour frames // NOTE: neighbour frames are cell and column frames. if ( !bNeighb ) { if ( nDiff > 0 ) Grow( nDiff ); else Shrink( -nDiff ); if ( GetUpper() && (aFrm.*fnRect->fnGetHeight)() != nNew ) GetUpper()->_InvalidateSize(); } // Auch wenn das Grow/Shrink noch nicht die gewuenschte Breite eingestellt hat, // wie z.B. beim Aufruf durch ChgColumns, um die Spaltenbreiten einzustellen, // wird die Breite jetzt gesetzt. (aFrm.*fnRect->fnSetHeight)( nNew ); } } } else aFrm.SSize( aNewSize ); if ( Frm().SSize() != aOldSize ) { SwPageFrm *pPage = FindPageFrm(); if ( GetNext() ) { GetNext()->_InvalidatePos(); GetNext()->InvalidatePage( pPage ); } if( IsLayoutFrm() ) { if( IsRightToLeft() ) _InvalidatePos(); if( ((SwLayoutFrm*)this)->Lower() ) ((SwLayoutFrm*)this)->Lower()->_InvalidateSize(); } _InvalidatePrt(); _InvalidateSize(); InvalidatePage( pPage ); } return aFrm.SSize(); } /************************************************************************* |* |* SwFrm::InsertBefore() |* |* Beschreibung SwFrm wird in eine bestehende Struktur eingefuegt |* Eingefuegt wird unterhalb des Parent und entweder |* vor pBehind oder am Ende der Kette wenn pBehind |* leer ist. |* Letzte Aenderung MA 06. Aug. 99 |* |*************************************************************************/ void SwFrm::InsertBefore( SwLayoutFrm* pParent, SwFrm* pBehind ) { ASSERT( pParent, "Kein Parent fuer Insert." ); ASSERT( (!pBehind || (pBehind && pParent == pBehind->GetUpper())), "Framebaum inkonsistent." ); pUpper = pParent; pNext = pBehind; if( pBehind ) { //Einfuegen vor pBehind. if( 0 != (pPrev = pBehind->pPrev) ) pPrev->pNext = this; else pUpper->pLower = this; pBehind->pPrev = this; } else { //Einfuegen am Ende, oder als ersten Node im Unterbaum pPrev = pUpper->Lower(); if ( pPrev ) { while( pPrev->pNext ) pPrev = pPrev->pNext; pPrev->pNext = this; } else pUpper->pLower = this; } } /************************************************************************* |* |* SwFrm::InsertBehind() |* |* Beschreibung SwFrm wird in eine bestehende Struktur eingefuegt |* Eingefuegt wird unterhalb des Parent und entweder |* hinter pBefore oder am Anfang der Kette wenn pBefore |* leer ist. |* Letzte Aenderung MA 06. Aug. 99 |* |*************************************************************************/ void SwFrm::InsertBehind( SwLayoutFrm *pParent, SwFrm *pBefore ) { ASSERT( pParent, "Kein Parent fuer Insert." ); ASSERT( (!pBefore || (pBefore && pParent == pBefore->GetUpper())), "Framebaum inkonsistent." ); pUpper = pParent; pPrev = pBefore; if ( pBefore ) { //Einfuegen hinter pBefore if ( 0 != (pNext = pBefore->pNext) ) pNext->pPrev = this; pBefore->pNext = this; } else { //Einfuegen am Anfang der Kette pNext = pParent->Lower(); if ( pParent->Lower() ) pParent->Lower()->pPrev = this; pParent->pLower = this; } } /************************************************************************* |* |* SwFrm::InsertGroup() |* |* Beschreibung Eine Kette von SwFrms wird in eine bestehende Struktur |* eingefuegt |* Letzte Aenderung AMA 9. Dec. 97 |* |* Bisher wird dies genutzt, um einen SectionFrame, der ggf. schon Geschwister |* mit sich bringt, in eine bestehende Struktur einzufuegen. |* |* Wenn man den dritten Parameter als NULL uebergibt, entspricht |* diese Methode dem SwFrm::InsertBefore(..), nur eben mit Geschwistern. |* |* Wenn man einen dritten Parameter uebergibt, passiert folgendes: |* this wird pNext von pParent, |* pSct wird pNext vom Letzten der this-Kette, |* pBehind wird vom pParent an den pSct umgehaengt. |* Dies dient dazu: ein SectionFrm (this) wird nicht als |* Kind an einen anderen SectionFrm (pParent) gehaengt, sondern pParent |* wird in zwei Geschwister aufgespalten (pParent+pSct) und this dazwischen |* eingebaut. |* |*************************************************************************/ void SwFrm::InsertGroupBefore( SwFrm* pParent, SwFrm* pBehind, SwFrm* pSct ) { ASSERT( pParent, "Kein Parent fuer Insert." ); ASSERT( (!pBehind || (pBehind && ( pParent == pBehind->GetUpper()) || ( pParent->IsSctFrm() && pBehind->GetUpper()->IsColBodyFrm() ) ) ), "Framebaum inkonsistent." ); if( pSct ) { pUpper = pParent->GetUpper(); SwFrm *pLast = this; while( pLast->GetNext() ) { pLast = pLast->GetNext(); pLast->pUpper = GetUpper(); } if( pBehind ) { pLast->pNext = pSct; pSct->pPrev = pLast; pSct->pNext = pParent->GetNext(); } else { pLast->pNext = pParent->GetNext(); if( pLast->GetNext() ) pLast->GetNext()->pPrev = pLast; } pParent->pNext = this; pPrev = pParent; if( pSct->GetNext() ) pSct->GetNext()->pPrev = pSct; while( pLast->GetNext() ) { pLast = pLast->GetNext(); pLast->pUpper = GetUpper(); } if( pBehind ) { //Einfuegen vor pBehind. if( pBehind->GetPrev() ) pBehind->GetPrev()->pNext = NULL; else pBehind->GetUpper()->pLower = NULL; pBehind->pPrev = NULL; SwLayoutFrm* pTmp = (SwLayoutFrm*)pSct; if( pTmp->Lower() ) { ASSERT( pTmp->Lower()->IsColumnFrm(), "InsertGrp: Used SectionFrm" ); pTmp = (SwLayoutFrm*)((SwLayoutFrm*)pTmp->Lower())->Lower(); ASSERT( pTmp, "InsertGrp: Missing ColBody" ); } pBehind->pUpper = pTmp; pBehind->GetUpper()->pLower = pBehind; pLast = pBehind->GetNext(); while ( pLast ) { pLast->pUpper = pBehind->GetUpper(); pLast = pLast->GetNext(); }; } else { ASSERT( pSct->IsSctFrm(), "InsertGroup: For SectionFrms only" ); delete ((SwSectionFrm*)pSct); } } else { pUpper = (SwLayoutFrm*)pParent; SwFrm *pLast = this; while( pLast->GetNext() ) { pLast = pLast->GetNext(); pLast->pUpper = GetUpper(); } pLast->pNext = pBehind; if( pBehind ) { //Einfuegen vor pBehind. if( 0 != (pPrev = pBehind->pPrev) ) pPrev->pNext = this; else pUpper->pLower = this; pBehind->pPrev = pLast; } else { //Einfuegen am Ende, oder des ersten Nodes im Unterbaum pPrev = pUpper->Lower(); if ( pPrev ) { while( pPrev->pNext ) pPrev = pPrev->pNext; pPrev->pNext = this; } else pUpper->pLower = this; } } } /************************************************************************* |* |* SwFrm::Remove() |* |* Ersterstellung AK 01-Mar-1991 |* Letzte Aenderung MA 07. Dec. 95 |* |*************************************************************************/ void SwFrm::Remove() { ASSERT( pUpper, "Removen ohne Upper?" ); if( pPrev ) // einer aus der Mitte wird removed pPrev->pNext = pNext; else { // der erste in einer Folge wird removed ASSERT( pUpper->pLower == this, "Layout inkonsistent." ); pUpper->pLower = pNext; } if( pNext ) pNext->pPrev = pPrev; // Verbindung kappen. pNext = pPrev = 0; pUpper = 0; } /************************************************************************* |* |* SwCntntFrm::Paste() |* |* Ersterstellung MA 23. Feb. 94 |* Letzte Aenderung MA 09. Sep. 98 |* |*************************************************************************/ void SwCntntFrm::Paste( SwFrm* pParent, SwFrm* pSibling) { ASSERT( pParent, "Kein Parent fuer Paste." ); ASSERT( pParent->IsLayoutFrm(), "Parent ist CntntFrm." ); ASSERT( pParent != this, "Bin selbst der Parent." ); ASSERT( pSibling != this, "Bin mein eigener Nachbar." ); ASSERT( !GetPrev() && !GetNext() && !GetUpper(), "Bin noch irgendwo angemeldet." ); ASSERT( !pSibling || pSibling->IsFlowFrm(), "<SwCntntFrm::Paste(..)> - sibling not of expected type." ) //In den Baum einhaengen. InsertBefore( (SwLayoutFrm*)pParent, pSibling ); SwPageFrm *pPage = FindPageFrm(); _InvalidateAll(); InvalidatePage( pPage ); if( pPage ) { pPage->InvalidateSpelling(); pPage->InvalidateSmartTags(); // SMARTTAGS pPage->InvalidateAutoCompleteWords(); pPage->InvalidateWordCount(); } if ( GetNext() ) { SwFrm* pNxt = GetNext(); pNxt->_InvalidatePrt(); pNxt->_InvalidatePos(); pNxt->InvalidatePage( pPage ); if( pNxt->IsSctFrm() ) pNxt = ((SwSectionFrm*)pNxt)->ContainsCntnt(); if( pNxt && pNxt->IsTxtFrm() && pNxt->IsInFtn() ) pNxt->Prepare( PREP_FTN, 0, sal_False ); } if ( Frm().Height() ) pParent->Grow( Frm().Height() ); if ( Frm().Width() != pParent->Prt().Width() ) Prepare( PREP_FIXSIZE_CHG ); if ( GetPrev() ) { if ( IsFollow() ) //Ich bin jetzt direkter Nachfolger meines Masters geworden ((SwCntntFrm*)GetPrev())->Prepare( PREP_FOLLOW_FOLLOWS ); else { if ( GetPrev()->Frm().Height() != GetPrev()->Prt().Height() + GetPrev()->Prt().Top() ) //Umrandung zu beruecksichtigen? GetPrev()->_InvalidatePrt(); // OD 18.02.2003 #104989# - force complete paint of previous frame, // if frame is inserted at the end of a section frame, in order to // get subsidiary lines repainted for the section. if ( pParent->IsSctFrm() && !GetNext() ) { // force complete paint of previous frame, if new inserted frame // in the section is the last one. GetPrev()->SetCompletePaint(); } GetPrev()->InvalidatePage( pPage ); } } if ( IsInFtn() ) { SwFrm* pFrm = GetIndPrev(); if( pFrm && pFrm->IsSctFrm() ) pFrm = ((SwSectionFrm*)pFrm)->ContainsAny(); if( pFrm ) pFrm->Prepare( PREP_QUOVADIS, 0, sal_False ); if( !GetNext() ) { pFrm = FindFtnFrm()->GetNext(); if( pFrm && 0 != (pFrm=((SwLayoutFrm*)pFrm)->ContainsAny()) ) pFrm->_InvalidatePrt(); } } _InvalidateLineNum(); SwFrm *pNxt = FindNextCnt(); if ( pNxt ) { while ( pNxt && pNxt->IsInTab() ) { if( 0 != (pNxt = pNxt->FindTabFrm()) ) pNxt = pNxt->FindNextCnt(); } if ( pNxt ) { pNxt->_InvalidateLineNum(); if ( pNxt != GetNext() ) pNxt->InvalidatePage(); } } } /************************************************************************* |* |* SwCntntFrm::Cut() |* |* Ersterstellung AK 14-Feb-1991 |* Letzte Aenderung MA 09. Sep. 98 |* |*************************************************************************/ void SwCntntFrm::Cut() { ASSERT( GetUpper(), "Cut ohne Upper()." ); SwPageFrm *pPage = FindPageFrm(); InvalidatePage( pPage ); SwFrm *pFrm = GetIndPrev(); if( pFrm ) { if( pFrm->IsSctFrm() ) pFrm = ((SwSectionFrm*)pFrm)->ContainsAny(); if ( pFrm && pFrm->IsCntntFrm() ) { pFrm->_InvalidatePrt(); if( IsInFtn() ) pFrm->Prepare( PREP_QUOVADIS, 0, sal_False ); } // --> OD 2004-07-15 #i26250# - invalidate printing area of previous // table frame. else if ( pFrm && pFrm->IsTabFrm() ) { pFrm->InvalidatePrt(); } // <-- } SwFrm *pNxt = FindNextCnt(); if ( pNxt ) { while ( pNxt && pNxt->IsInTab() ) { if( 0 != (pNxt = pNxt->FindTabFrm()) ) pNxt = pNxt->FindNextCnt(); } if ( pNxt ) { pNxt->_InvalidateLineNum(); if ( pNxt != GetNext() ) pNxt->InvalidatePage(); } } if( 0 != (pFrm = GetIndNext()) ) { //Der alte Nachfolger hat evtl. einen Abstand zum Vorgaenger //berechnet, der ist jetzt, wo er der erste wird obsolet bzw. anders. pFrm->_InvalidatePrt(); pFrm->_InvalidatePos(); pFrm->InvalidatePage( pPage ); if( pFrm->IsSctFrm() ) { pFrm = ((SwSectionFrm*)pFrm)->ContainsAny(); if( pFrm ) { pFrm->_InvalidatePrt(); pFrm->_InvalidatePos(); pFrm->InvalidatePage( pPage ); } } if( pFrm && IsInFtn() ) pFrm->Prepare( PREP_ERGOSUM, 0, sal_False ); if( IsInSct() && !GetPrev() ) { SwSectionFrm* pSct = FindSctFrm(); if( !pSct->IsFollow() ) { pSct->_InvalidatePrt(); pSct->InvalidatePage( pPage ); } } } else { InvalidateNextPos(); //Einer muss die Retusche uebernehmen: Vorgaenger oder Upper if ( 0 != (pFrm = GetPrev()) ) { pFrm->SetRetouche(); pFrm->Prepare( PREP_WIDOWS_ORPHANS ); pFrm->_InvalidatePos(); pFrm->InvalidatePage( pPage ); } //Wenn ich der einzige CntntFrm in meinem Upper bin (war), so muss //er die Retouche uebernehmen. //Ausserdem kann eine Leerseite entstanden sein. else { SwRootFrm *pRoot = getRootFrm(); if ( pRoot ) { pRoot->SetSuperfluous(); GetUpper()->SetCompletePaint(); GetUpper()->InvalidatePage( pPage ); } if( IsInSct() ) { SwSectionFrm* pSct = FindSctFrm(); if( !pSct->IsFollow() ) { pSct->_InvalidatePrt(); pSct->InvalidatePage( pPage ); } } // --> FME 2005-08-03 #i52253# The master table should take care // of removing the follow flow line. if ( IsInTab() ) { SwTabFrm* pThisTab = FindTabFrm(); SwTabFrm* pMasterTab = pThisTab && pThisTab->IsFollow() ? pThisTab->FindMaster() : 0; if ( pMasterTab ) { pMasterTab->_InvalidatePos(); pMasterTab->SetRemoveFollowFlowLinePending( sal_True ); } } // <-- } } //Erst removen, dann Upper Shrinken. SwLayoutFrm *pUp = GetUpper(); Remove(); if ( pUp ) { SwSectionFrm *pSct = 0; if ( !pUp->Lower() && ( ( pUp->IsFtnFrm() && !pUp->IsColLocked() ) || ( pUp->IsInSct() && // --> FME 2004-06-03 #i29438# // We have to consider the case that the section may be "empty" // except from a temporary empty table frame. // This can happen due to the new cell split feature. !pUp->IsCellFrm() && // <-- // --> OD 2006-01-04 #126020# - adjust check for empty section // --> OD 2006-02-01 #130797# - correct fix #126020# !(pSct = pUp->FindSctFrm())->ContainsCntnt() && !pSct->ContainsAny( true ) ) ) ) // <-- { if ( pUp->GetUpper() ) { // --> OD 2006-09-25 #b6448963# // prevent delete of <ColLocked> footnote frame // if( pUp->IsFtnFrm() ) if ( pUp->IsFtnFrm() && !pUp->IsColLocked()) // <-- { if( pUp->GetNext() && !pUp->GetPrev() ) { SwFrm* pTmp = ((SwLayoutFrm*)pUp->GetNext())->ContainsAny(); if( pTmp ) pTmp->_InvalidatePrt(); } pUp->Cut(); delete pUp; } else { // --> OD 2006-09-25 #b6448963# // if ( pSct->IsColLocked() || !pSct->IsInFtn() ) if ( pSct->IsColLocked() || !pSct->IsInFtn() || ( pUp->IsFtnFrm() && pUp->IsColLocked() ) ) // <-- { pSct->DelEmpty( sal_False ); // Wenn ein gelockter Bereich nicht geloescht werden darf, // so ist zumindest seine Groesse durch das Entfernen seines // letzten Contents ungueltig geworden. pSct->_InvalidateSize(); } else { pSct->DelEmpty( sal_True ); delete pSct; } } } } else { SWRECTFN( this ) long nFrmHeight = (Frm().*fnRect->fnGetHeight)(); if( nFrmHeight ) pUp->Shrink( nFrmHeight ); } } } /************************************************************************* |* |* SwLayoutFrm::Paste() |* |* Ersterstellung MA 23. Feb. 94 |* Letzte Aenderung MA 23. Feb. 94 |* |*************************************************************************/ void SwLayoutFrm::Paste( SwFrm* pParent, SwFrm* pSibling) { ASSERT( pParent, "Kein Parent fuer Paste." ); ASSERT( pParent->IsLayoutFrm(), "Parent ist CntntFrm." ); ASSERT( pParent != this, "Bin selbst der Parent." ); ASSERT( pSibling != this, "Bin mein eigener Nachbar." ); ASSERT( !GetPrev() && !GetNext() && !GetUpper(), "Bin noch irgendwo angemeldet." ); //In den Baum einhaengen. InsertBefore( (SwLayoutFrm*)pParent, pSibling ); // OD 24.10.2002 #103517# - correct setting of variable <fnRect> // <fnRect> is used for the following: // (1) To invalidate the frame's size, if its size, which has to be the // same as its upper/parent, differs from its upper's/parent's. // (2) To adjust/grow the frame's upper/parent, if it has a dimension in its // size, which is not determined by its upper/parent. // Which size is which depends on the frame type and the layout direction // (vertical or horizontal). // There are the following cases: // (A) Header and footer frames both in vertical and in horizontal layout // have to size the width to the upper/parent. A dimension in the height // has to cause a adjustment/grow of the upper/parent. // --> <fnRect> = fnRectHori // (B) Cell and column frames in vertical layout, the width has to be the // same as upper/parent and a dimension in height causes adjustment/grow // of the upper/parent. // --> <fnRect> = fnRectHori // in horizontal layout the other way around // --> <fnRect> = fnRectVert // (C) Other frames in vertical layout, the height has to be the // same as upper/parent and a dimension in width causes adjustment/grow // of the upper/parent. // --> <fnRect> = fnRectVert // in horizontal layout the other way around // --> <fnRect> = fnRectHori //SwRectFn fnRect = IsVertical() ? fnRectHori : fnRectVert; SwRectFn fnRect; if ( IsHeaderFrm() || IsFooterFrm() ) fnRect = fnRectHori; else if ( IsCellFrm() || IsColumnFrm() ) //Badaa: 2008-04-18 * Support for Classical Mongolian Script (SCMS) joint with Jiayanmin fnRect = GetUpper()->IsVertical() ? fnRectHori : ( GetUpper()->IsVertLR() ? fnRectVertL2R : fnRectVert ); else //Badaa: 2008-04-18 * Support for Classical Mongolian Script (SCMS) joint with Jiayanmin fnRect = GetUpper()->IsVertical() ? ( GetUpper()->IsVertLR() ? fnRectVertL2R : fnRectVert ) : fnRectHori; if( (Frm().*fnRect->fnGetWidth)() != (pParent->Prt().*fnRect->fnGetWidth)()) _InvalidateSize(); _InvalidatePos(); const SwPageFrm *pPage = FindPageFrm(); InvalidatePage( pPage ); SwFrm *pFrm; if( !IsColumnFrm() ) { if( 0 != ( pFrm = GetIndNext() ) ) { pFrm->_InvalidatePos(); if( IsInFtn() ) { if( pFrm->IsSctFrm() ) pFrm = ((SwSectionFrm*)pFrm)->ContainsAny(); if( pFrm ) pFrm->Prepare( PREP_ERGOSUM, 0, sal_False ); } } if ( IsInFtn() && 0 != ( pFrm = GetIndPrev() ) ) { if( pFrm->IsSctFrm() ) pFrm = ((SwSectionFrm*)pFrm)->ContainsAny(); if( pFrm ) pFrm->Prepare( PREP_QUOVADIS, 0, sal_False ); } } if( (Frm().*fnRect->fnGetHeight)() ) { // AdjustNeighbourhood wird jetzt auch in Spalten aufgerufen, // die sich nicht in Rahmen befinden sal_uInt8 nAdjust = GetUpper()->IsFtnBossFrm() ? ((SwFtnBossFrm*)GetUpper())->NeighbourhoodAdjustment( this ) : NA_GROW_SHRINK; SwTwips nGrow = (Frm().*fnRect->fnGetHeight)(); if( NA_ONLY_ADJUST == nAdjust ) AdjustNeighbourhood( nGrow ); else { SwTwips nReal = 0; if( NA_ADJUST_GROW == nAdjust ) nReal = AdjustNeighbourhood( nGrow ); if( nReal < nGrow ) nReal += pParent->Grow( nGrow - nReal ); if( NA_GROW_ADJUST == nAdjust && nReal < nGrow ) AdjustNeighbourhood( nGrow - nReal ); } } } /************************************************************************* |* |* SwLayoutFrm::Cut() |* |* Ersterstellung MA 23. Feb. 94 |* Letzte Aenderung MA 23. Feb. 94 |* |*************************************************************************/ void SwLayoutFrm::Cut() { if ( GetNext() ) GetNext()->_InvalidatePos(); SWRECTFN( this ) SwTwips nShrink = (Frm().*fnRect->fnGetHeight)(); //Erst removen, dann Upper Shrinken. SwLayoutFrm *pUp = GetUpper(); // AdjustNeighbourhood wird jetzt auch in Spalten aufgerufen, // die sich nicht in Rahmen befinden // Remove must not be called before a AdjustNeighbourhood, but it has to // be called before the upper-shrink-call, if the upper-shrink takes care // of his content if ( pUp && nShrink ) { if( pUp->IsFtnBossFrm() ) { sal_uInt8 nAdjust= ((SwFtnBossFrm*)pUp)->NeighbourhoodAdjustment( this ); if( NA_ONLY_ADJUST == nAdjust ) AdjustNeighbourhood( -nShrink ); else { SwTwips nReal = 0; if( NA_ADJUST_GROW == nAdjust ) nReal = -AdjustNeighbourhood( -nShrink ); if( nReal < nShrink ) { SwTwips nOldHeight = (Frm().*fnRect->fnGetHeight)(); (Frm().*fnRect->fnSetHeight)( 0 ); nReal += pUp->Shrink( nShrink - nReal ); (Frm().*fnRect->fnSetHeight)( nOldHeight ); } if( NA_GROW_ADJUST == nAdjust && nReal < nShrink ) AdjustNeighbourhood( nReal - nShrink ); } Remove(); } else { Remove(); pUp->Shrink( nShrink ); } } else Remove(); if( pUp && !pUp->Lower() ) { pUp->SetCompletePaint(); pUp->InvalidatePage(); } } /************************************************************************* |* |* SwFrm::Grow() |* |* Ersterstellung AK 19-Feb-1991 |* Letzte Aenderung MA 05. May. 94 |* |*************************************************************************/ SwTwips SwFrm::Grow( SwTwips nDist, sal_Bool bTst, sal_Bool bInfo ) { ASSERT( nDist >= 0, "Negatives Wachstum?" ); PROTOCOL_ENTER( this, bTst ? PROT_GROW_TST : PROT_GROW, 0, &nDist ) if ( nDist ) { SWRECTFN( this ) SwTwips nPrtHeight = (Prt().*fnRect->fnGetHeight)(); if( nPrtHeight > 0 && nDist > (LONG_MAX - nPrtHeight) ) nDist = LONG_MAX - nPrtHeight; if ( IsFlyFrm() ) return ((SwFlyFrm*)this)->_Grow( nDist, bTst ); else if( IsSctFrm() ) return ((SwSectionFrm*)this)->_Grow( nDist, bTst ); else { const SwCellFrm* pThisCell = dynamic_cast<const SwCellFrm*>(this); if ( pThisCell ) { const SwTabFrm* pTab = FindTabFrm(); // NEW TABLES if ( ( 0 != pTab->IsVertical() ) != ( 0 != IsVertical() ) || pThisCell->GetLayoutRowSpan() < 1 ) return 0; } const SwTwips nReal = GrowFrm( nDist, bTst, bInfo ); if( !bTst ) { nPrtHeight = (Prt().*fnRect->fnGetHeight)(); (Prt().*fnRect->fnSetHeight)( nPrtHeight + ( IsCntntFrm() ? nDist : nReal ) ); } return nReal; } } return 0L; } /************************************************************************* |* |* SwFrm::Shrink() |* |* Ersterstellung AK 14-Feb-1991 |* Letzte Aenderung MA 05. May. 94 |* |*************************************************************************/ SwTwips SwFrm::Shrink( SwTwips nDist, sal_Bool bTst, sal_Bool bInfo ) { ASSERT( nDist >= 0, "Negative Verkleinerung?" ); PROTOCOL_ENTER( this, bTst ? PROT_SHRINK_TST : PROT_SHRINK, 0, &nDist ) if ( nDist ) { if ( IsFlyFrm() ) return ((SwFlyFrm*)this)->_Shrink( nDist, bTst ); else if( IsSctFrm() ) return ((SwSectionFrm*)this)->_Shrink( nDist, bTst ); else { const SwCellFrm* pThisCell = dynamic_cast<const SwCellFrm*>(this); if ( pThisCell ) { const SwTabFrm* pTab = FindTabFrm(); // NEW TABLES if ( ( 0 != pTab->IsVertical() ) != ( 0 != IsVertical() ) || pThisCell->GetLayoutRowSpan() < 1 ) return 0; } SWRECTFN( this ) SwTwips nReal = (Frm().*fnRect->fnGetHeight)(); ShrinkFrm( nDist, bTst, bInfo ); nReal -= (Frm().*fnRect->fnGetHeight)(); if( !bTst ) { const SwTwips nPrtHeight = (Prt().*fnRect->fnGetHeight)(); (Prt().*fnRect->fnSetHeight)( nPrtHeight - ( IsCntntFrm() ? nDist : nReal ) ); } return nReal; } } return 0L; } /************************************************************************* |* |* SwFrm::AdjustNeighbourhood() |* |* Beschreibung Wenn sich die Groesse eines Frm's direkt unterhalb |* eines Fussnotenbosses (Seite/Spalte) veraendert hat, so muss dieser |* "Normalisiert" werden. |* Es gibt dort immer einen Frame, der den "maximal moeglichen" Raum |* einnimmt (der Frame, der den Body.Text enhaelt) und keinen oder |* mehrere Frames die den Platz einnehmen den sie halt brauchen |* (Kopf-/Fussbereich, Fussnoten). |* Hat sich einer der Frames veraendert, so muss der Body-Text-Frame |* entsprechen wachsen oder schrumpfen; unabhaegig davon, dass er fix ist. |* !! Ist es moeglich dies allgemeiner zu loesen, also nicht auf die |* Seite beschraenkt und nicht auf einen Speziellen Frame, der den |* maximalen Platz einnimmt (gesteuert ueber Attribut FrmSize)? Probleme: |* Was ist wenn mehrere Frames nebeneinander stehen, die den maximalen |* Platz einnehmen? |* Wie wird der Maximale Platz berechnet? |* Wie klein duerfen diese Frames werden? |* |* Es wird auf jeden Fall nur so viel Platz genehmigt, dass ein |* Minimalwert fuer die Hoehe des Bodys nicht unterschritten wird. |* |* Parameter: nDiff ist der Betrag, um den Platz geschaffen werden muss |* |* Ersterstellung MA 07. May. 92 |* Letzte Aenderung AMA 02. Nov. 98 |* |*************************************************************************/ SwTwips SwFrm::AdjustNeighbourhood( SwTwips nDiff, sal_Bool bTst ) { PROTOCOL_ENTER( this, PROT_ADJUSTN, 0, &nDiff ); if ( !nDiff || !GetUpper()->IsFtnBossFrm() ) // nur innerhalb von Seiten/Spalten return 0L; const ViewShell *pSh = getRootFrm()->GetCurrShell(); const sal_Bool bBrowse = pSh && pSh->GetViewOptions()->getBrowseMode(); //Der (Page)Body veraendert sich nur im BrowseMode, aber nicht wenn er //Spalten enthaelt. if ( IsPageBodyFrm() && (!bBrowse || (((SwLayoutFrm*)this)->Lower() && ((SwLayoutFrm*)this)->Lower()->IsColumnFrm())) ) return 0L; //In der BrowseView kann der PageFrm selbst ersteinmal einiges von den //Wuenschen abfangen. long nBrowseAdd = 0; if ( bBrowse && GetUpper()->IsPageFrm() ) // nur (Page)BodyFrms { ViewShell *pViewShell = getRootFrm()->GetCurrShell(); SwLayoutFrm *pUp = GetUpper(); long nChg; const long nUpPrtBottom = pUp->Frm().Height() - pUp->Prt().Height() - pUp->Prt().Top(); SwRect aInva( pUp->Frm() ); if ( pViewShell ) { aInva.Pos().X() = pViewShell->VisArea().Left(); aInva.Width( pViewShell->VisArea().Width() ); } if ( nDiff > 0 ) { nChg = BROWSE_HEIGHT - pUp->Frm().Height(); nChg = Min( nDiff, nChg ); if ( !IsBodyFrm() ) { SetCompletePaint(); if ( !pViewShell || pViewShell->VisArea().Height() >= pUp->Frm().Height() ) { //Ersteinmal den Body verkleinern. Der waechst dann schon //wieder. SwFrm *pBody = ((SwFtnBossFrm*)pUp)->FindBodyCont(); const long nTmp = nChg - pBody->Prt().Height(); if ( !bTst ) { pBody->Frm().Height(Max( 0L, pBody->Frm().Height() - nChg )); pBody->_InvalidatePrt(); pBody->_InvalidateSize(); if ( pBody->GetNext() ) pBody->GetNext()->_InvalidatePos(); if ( !IsHeaderFrm() ) pBody->SetCompletePaint(); } nChg = nTmp <= 0 ? 0 : nTmp; } } const long nTmp = nUpPrtBottom + 20; aInva.Top( aInva.Bottom() - nTmp ); aInva.Height( nChg + nTmp ); } else { //Die Seite kann bis auf 0 schrumpfen. Die erste Seite bleibt //mindestens so gross wie die VisArea. nChg = nDiff; long nInvaAdd = 0; if ( pViewShell && !pUp->GetPrev() && pUp->Frm().Height() + nDiff < pViewShell->VisArea().Height() ) { //Das heisst aber wiederum trotzdem, das wir geeignet invalidieren //muessen. nChg = pViewShell->VisArea().Height() - pUp->Frm().Height(); nInvaAdd = -(nDiff - nChg); } //Invalidieren inklusive unterem Rand. long nBorder = nUpPrtBottom + 20; nBorder -= nChg; aInva.Top( aInva.Bottom() - (nBorder+nInvaAdd) ); if ( !IsBodyFrm() ) { SetCompletePaint(); if ( !IsHeaderFrm() ) ((SwFtnBossFrm*)pUp)->FindBodyCont()->SetCompletePaint(); } //Wegen der Rahmen die Seite invalidieren. Dadurch wird die Seite //wieder entsprechend gross wenn ein Rahmen nicht passt. Das //funktioniert anderfalls nur zufaellig fuer absatzgebundene Rahmen //(NotifyFlys). pUp->InvalidateSize(); } if ( !bTst ) { //Unabhaengig von nChg if ( pViewShell && aInva.HasArea() && pUp->GetUpper() ) pViewShell->InvalidateWindows( aInva ); } if ( !bTst && nChg ) { const SwRect aOldRect( pUp->Frm() ); pUp->Frm().SSize().Height() += nChg; pUp->Prt().SSize().Height() += nChg; if ( pViewShell ) pViewShell->Imp()->SetFirstVisPageInvalid(); if ( GetNext() ) GetNext()->_InvalidatePos(); //Ggf. noch ein Repaint ausloesen. const SvxGraphicPosition ePos = pUp->GetFmt()->GetBackground().GetGraphicPos(); if ( ePos != GPOS_NONE && ePos != GPOS_TILED ) pViewShell->InvalidateWindows( pUp->Frm() ); if ( pUp->GetUpper() ) { if ( pUp->GetNext() ) pUp->GetNext()->InvalidatePos(); //Mies aber wahr: im Notify am ViewImp wird evtl. ein Calc //auf die Seite und deren Lower gerufen. Die Werte sollten //unverandert bleiben, weil der Aufrufer bereits fuer die //Anpassung von Frm und Prt sorgen wird. const long nOldFrmHeight = Frm().Height(); const long nOldPrtHeight = Prt().Height(); const sal_Bool bOldComplete = IsCompletePaint(); if ( IsBodyFrm() ) Prt().SSize().Height() = nOldFrmHeight; // PAGES01 if ( pUp->GetUpper() ) static_cast<SwRootFrm*>(pUp->GetUpper())->CheckViewLayout( 0, 0 ); //((SwPageFrm*)pUp)->AdjustRootSize( CHG_CHGPAGE, &aOldRect ); Frm().SSize().Height() = nOldFrmHeight; Prt().SSize().Height() = nOldPrtHeight; bCompletePaint = bOldComplete; } if ( !IsBodyFrm() ) pUp->_InvalidateSize(); InvalidatePage( (SwPageFrm*)pUp ); } nDiff -= nChg; if ( !nDiff ) return nChg; else nBrowseAdd = nChg; } const SwFtnBossFrm *pBoss = (SwFtnBossFrm*)GetUpper(); SwTwips nReal = 0, nAdd = 0; SwFrm *pFrm = 0; SWRECTFN( this ) if( IsBodyFrm() ) { if( IsInSct() ) { SwSectionFrm *pSect = FindSctFrm(); if( nDiff > 0 && pSect->IsEndnAtEnd() && GetNext() && GetNext()->IsFtnContFrm() ) { SwFtnContFrm* pCont = (SwFtnContFrm*)GetNext(); SwTwips nMinH = 0; SwFtnFrm* pFtn = (SwFtnFrm*)pCont->Lower(); sal_Bool bFtn = sal_False; while( pFtn ) { if( !pFtn->GetAttr()->GetFtn().IsEndNote() ) { nMinH += (pFtn->Frm().*fnRect->fnGetHeight)(); bFtn = sal_True; } pFtn = (SwFtnFrm*)pFtn->GetNext(); } if( bFtn ) nMinH += (pCont->Prt().*fnRect->fnGetTop)(); nReal = (pCont->Frm().*fnRect->fnGetHeight)() - nMinH; if( nReal > nDiff ) nReal = nDiff; if( nReal > 0 ) pFrm = GetNext(); else nReal = 0; } if( !bTst && !pSect->IsColLocked() ) pSect->InvalidateSize(); } if( !pFrm ) return nBrowseAdd; } else { const sal_Bool bFtnPage = pBoss->IsPageFrm() && ((SwPageFrm*)pBoss)->IsFtnPage(); if ( bFtnPage && !IsFtnContFrm() ) pFrm = (SwFrm*)pBoss->FindFtnCont(); if ( !pFrm ) pFrm = (SwFrm*)pBoss->FindBodyCont(); if ( !pFrm ) return 0; //Wenn ich keinen finde eruebrigt sich alles weitere. nReal = (pFrm->Frm().*fnRect->fnGetHeight)(); if( nReal > nDiff ) nReal = nDiff; if( !bFtnPage ) { //Minimalgrenze beachten! if( nReal ) { const SwTwips nMax = pBoss->GetVarSpace(); if ( nReal > nMax ) nReal = nMax; } if( !IsFtnContFrm() && nDiff > nReal && pFrm->GetNext() && pFrm->GetNext()->IsFtnContFrm() && ( pFrm->GetNext()->IsVertical() == IsVertical() ) ) { //Wenn der Body nicht genuegend her gibt, kann ich noch mal //schauen ob es eine Fussnote gibt, falls ja kann dieser //entsprechend viel gemopst werden. const SwTwips nAddMax = (pFrm->GetNext()->Frm().*fnRect-> fnGetHeight)(); nAdd = nDiff - nReal; if ( nAdd > nAddMax ) nAdd = nAddMax; if ( !bTst ) { (pFrm->GetNext()->Frm().*fnRect->fnSetHeight)(nAddMax-nAdd); //Badaa: 2008-04-18 * Support for Classical Mongolian Script (SCMS) joint with Jiayanmin if( bVert && !bVertL2R && !bRev ) pFrm->GetNext()->Frm().Pos().X() += nAdd; pFrm->GetNext()->InvalidatePrt(); if ( pFrm->GetNext()->GetNext() ) pFrm->GetNext()->GetNext()->_InvalidatePos(); } } } } if ( !bTst && nReal ) { SwTwips nTmp = (pFrm->Frm().*fnRect->fnGetHeight)(); (pFrm->Frm().*fnRect->fnSetHeight)( nTmp - nReal ); //Badaa: 2008-04-18 * Support for Classical Mongolian Script (SCMS) joint with Jiayanmin if( bVert && !bVertL2R && !bRev ) pFrm->Frm().Pos().X() += nReal; pFrm->InvalidatePrt(); if ( pFrm->GetNext() ) pFrm->GetNext()->_InvalidatePos(); if( nReal < 0 && pFrm->IsInSct() ) { SwLayoutFrm* pUp = pFrm->GetUpper(); if( pUp && 0 != ( pUp = pUp->GetUpper() ) && pUp->IsSctFrm() && !pUp->IsColLocked() ) pUp->InvalidateSize(); } if( ( IsHeaderFrm() || IsFooterFrm() ) && pBoss->GetDrawObjs() ) { const SwSortedObjs &rObjs = *pBoss->GetDrawObjs(); ASSERT( pBoss->IsPageFrm(), "Header/Footer out of page?" ); for ( sal_uInt16 i = 0; i < rObjs.Count(); ++i ) { SwAnchoredObject* pAnchoredObj = rObjs[i]; if ( pAnchoredObj->ISA(SwFlyFrm) ) { SwFlyFrm* pFly = static_cast<SwFlyFrm*>(pAnchoredObj); ASSERT( !pFly->IsFlyInCntFrm(), "FlyInCnt at Page?" ); const SwFmtVertOrient &rVert = pFly->GetFmt()->GetVertOrient(); // Wann muss invalidiert werden? // Wenn ein Rahmen am SeitenTextBereich ausgerichtet ist, // muss bei Aenderung des Headers ein TOP, MIDDLE oder NONE, // bei Aenderung des Footers ein BOTTOM oder MIDDLE // ausgerichteter Rahmen seine Position neu berechnen. if( ( rVert.GetRelationOrient() == text::RelOrientation::PRINT_AREA || rVert.GetRelationOrient() == text::RelOrientation::PAGE_PRINT_AREA ) && ((IsHeaderFrm() && rVert.GetVertOrient()!=text::VertOrientation::BOTTOM) || (IsFooterFrm() && rVert.GetVertOrient()!=text::VertOrientation::NONE && rVert.GetVertOrient() != text::VertOrientation::TOP)) ) { pFly->_InvalidatePos(); pFly->_Invalidate(); } } } } } return (nBrowseAdd + nReal + nAdd); } /************************************************************************* |* |* SwFrm::ImplInvalidateSize(), ImplInvalidatePrt(), ImplInvalidatePos(), |* ImplInvalidateLineNum() |* |* Ersterstellung MA 15. Oct. 92 |* Letzte Aenderung MA 24. Mar. 94 |* |*************************************************************************/ /** method to perform additional actions on an invalidation OD 2004-05-19 #i28701# @author OD */ void SwFrm::_ActionOnInvalidation( const InvalidationType ) { // default behaviour is to perform no additional action } /** method to determine, if an invalidation is allowed. OD 2004-05-19 #i28701# @author OD */ bool SwFrm::_InvalidationAllowed( const InvalidationType ) const { // default behaviour is to allow invalidation return true; } void SwFrm::ImplInvalidateSize() { if ( _InvalidationAllowed( INVALID_SIZE ) ) { bValidSize = sal_False; if ( IsFlyFrm() ) ((SwFlyFrm*)this)->_Invalidate(); else InvalidatePage(); // OD 2004-05-19 #i28701# _ActionOnInvalidation( INVALID_SIZE ); } } void SwFrm::ImplInvalidatePrt() { if ( _InvalidationAllowed( INVALID_PRTAREA ) ) { bValidPrtArea = sal_False; if ( IsFlyFrm() ) ((SwFlyFrm*)this)->_Invalidate(); else InvalidatePage(); // OD 2004-05-19 #i28701# _ActionOnInvalidation( INVALID_PRTAREA ); } } void SwFrm::ImplInvalidatePos() { if ( _InvalidationAllowed( INVALID_POS ) ) { bValidPos = sal_False; if ( IsFlyFrm() ) { ((SwFlyFrm*)this)->_Invalidate(); } else { InvalidatePage(); } // OD 2004-05-19 #i28701# _ActionOnInvalidation( INVALID_POS ); } } void SwFrm::ImplInvalidateLineNum() { if ( _InvalidationAllowed( INVALID_LINENUM ) ) { bValidLineNum = sal_False; ASSERT( IsTxtFrm(), "line numbers are implemented for text only" ); InvalidatePage(); // OD 2004-05-19 #i28701# _ActionOnInvalidation( INVALID_LINENUM ); } } /************************************************************************* |* |* SwFrm::ReinitializeFrmSizeAttrFlags |* |* Ersterstellung MA 15. Oct. 96 |* Letzte Aenderung MA 15. Oct. 96 |* |*************************************************************************/ void SwFrm::ReinitializeFrmSizeAttrFlags() { const SwFmtFrmSize &rFmtSize = GetAttrSet()->GetFrmSize(); if ( ATT_VAR_SIZE == rFmtSize.GetHeightSizeType() || ATT_MIN_SIZE == rFmtSize.GetHeightSizeType()) { bFixSize = sal_False; if ( GetType() & (FRM_HEADER | FRM_FOOTER | FRM_ROW) ) { SwFrm *pFrm = ((SwLayoutFrm*)this)->Lower(); while ( pFrm ) { pFrm->_InvalidateSize(); pFrm->_InvalidatePrt(); pFrm = pFrm->GetNext(); } SwCntntFrm *pCnt = ((SwLayoutFrm*)this)->ContainsCntnt(); // --> OD 2004-12-20 #i36991# - be save. // E.g., a row can contain *no* content. if ( pCnt ) { pCnt->InvalidatePage(); do { pCnt->Prepare( PREP_ADJUST_FRM ); pCnt->_InvalidateSize(); pCnt = pCnt->GetNextCntntFrm(); } while ( ((SwLayoutFrm*)this)->IsAnLower( pCnt ) ); } // <-- } } else if ( rFmtSize.GetHeightSizeType() == ATT_FIX_SIZE ) { if( IsVertical() ) ChgSize( Size( rFmtSize.GetWidth(), Frm().Height())); else ChgSize( Size( Frm().Width(), rFmtSize.GetHeight())); } } /************************************************************************* |* SwFrm::ValidateThisAndAllLowers() * * FME 2007-08-30 #i81146# new loop control |*************************************************************************/ void SwFrm::ValidateThisAndAllLowers( const sal_uInt16 nStage ) { // Stage 0: Only validate frames. Do not process any objects. // Stage 1: Only validate fly frames and all of their contents. // Stage 2: Validate all. const bool bOnlyObject = 1 == nStage; const bool bIncludeObjects = 1 <= nStage; if ( !bOnlyObject || ISA(SwFlyFrm) ) { bValidSize = sal_True; bValidPrtArea = sal_True; bValidPos = sal_True; } if ( bIncludeObjects ) { const SwSortedObjs* pObjs = GetDrawObjs(); if ( pObjs ) { const sal_uInt32 nCnt = pObjs->Count(); for ( sal_uInt32 i = 0; i < nCnt; ++i ) { SwAnchoredObject* pAnchObj = (*pObjs)[i]; if ( pAnchObj->ISA(SwFlyFrm) ) static_cast<SwFlyFrm*>(pAnchObj)->ValidateThisAndAllLowers( 2 ); else if ( pAnchObj->ISA(SwAnchoredDrawObject) ) static_cast<SwAnchoredDrawObject*>(pAnchObj)->ValidateThis(); } } } if ( IsLayoutFrm() ) { SwFrm* pLower = static_cast<SwLayoutFrm*>(this)->Lower(); while ( pLower ) { pLower->ValidateThisAndAllLowers( nStage ); pLower = pLower->GetNext(); } } } /************************************************************************* |* |* SwCntntFrm::GrowFrm() |* |* Ersterstellung MA 30. Jul. 92 |* Letzte Aenderung MA 25. Mar. 99 |* |*************************************************************************/ SwTwips SwCntntFrm::GrowFrm( SwTwips nDist, sal_Bool bTst, sal_Bool bInfo ) { SWRECTFN( this ) SwTwips nFrmHeight = (Frm().*fnRect->fnGetHeight)(); if( nFrmHeight > 0 && nDist > (LONG_MAX - nFrmHeight ) ) nDist = LONG_MAX - nFrmHeight; const ViewShell *pSh = getRootFrm()->GetCurrShell(); const sal_Bool bBrowse = pSh && pSh->GetViewOptions()->getBrowseMode(); const sal_uInt16 nTmpType = bBrowse ? 0x2084: 0x2004; //Row+Cell, Browse mit Body if( !(GetUpper()->GetType() & nTmpType) && GetUpper()->HasFixSize() ) { if ( !bTst ) { (Frm().*fnRect->fnSetHeight)( nFrmHeight + nDist ); //Badaa: 2008-04-18 * Support for Classical Mongolian Script (SCMS) joint with Jiayanmin if( IsVertical() && !IsVertLR() && !IsReverse() ) Frm().Pos().X() -= nDist; if ( GetNext() ) { GetNext()->InvalidatePos(); } // --> OD 2004-07-05 #i28701# - Due to the new object positioning the // frame on the next page/column can flow backward (e.g. it was moved forward // due to the positioning of its objects ). Thus, invalivate this next frame, // if document compatibility option 'Consider wrapping style influence on // object positioning' is ON. else if ( GetUpper()->GetFmt()->getIDocumentSettingAccess()->get(IDocumentSettingAccess::CONSIDER_WRAP_ON_OBJECT_POSITION) ) { InvalidateNextPos(); } // <-- } return 0; } SwTwips nReal = (GetUpper()->Prt().*fnRect->fnGetHeight)(); SwFrm *pFrm = GetUpper()->Lower(); while( pFrm && nReal > 0 ) { nReal -= (pFrm->Frm().*fnRect->fnGetHeight)(); pFrm = pFrm->GetNext(); } if ( !bTst ) { //Cntnts werden immer auf den gewuenschten Wert gebracht. long nOld = (Frm().*fnRect->fnGetHeight)(); (Frm().*fnRect->fnSetHeight)( nOld + nDist ); //Badaa: 2008-04-18 * Support for Classical Mongolian Script (SCMS) joint with Jiayanmin if( IsVertical()&& !IsVertLR() && !IsReverse() ) Frm().Pos().X() -= nDist; if ( nOld && IsInTab() ) { SwTabFrm *pTab = FindTabFrm(); if ( pTab->GetTable()->GetHTMLTableLayout() && !pTab->IsJoinLocked() && !pTab->GetFmt()->GetDoc()->GetDocShell()->IsReadOnly() ) { pTab->InvalidatePos(); pTab->SetResizeHTMLTable(); } } } //Upper nur growen wenn notwendig. if ( nReal < nDist ) { if( GetUpper() ) { if( bTst || !GetUpper()->IsFooterFrm() ) nReal = GetUpper()->Grow( nDist - (nReal > 0 ? nReal : 0), bTst, bInfo ); else { nReal = 0; GetUpper()->InvalidateSize(); } } else nReal = 0; } else nReal = nDist; // --> OD 2004-07-05 #i28701# - Due to the new object positioning the // frame on the next page/column can flow backward (e.g. it was moved forward // due to the positioning of its objects ). Thus, invalivate this next frame, // if document compatibility option 'Consider wrapping style influence on // object positioning' is ON. if ( !bTst ) { if ( GetNext() ) { GetNext()->InvalidatePos(); } else if ( GetUpper()->GetFmt()->getIDocumentSettingAccess()->get(IDocumentSettingAccess::CONSIDER_WRAP_ON_OBJECT_POSITION) ) { InvalidateNextPos(); } } // <-- return nReal; } /************************************************************************* |* |* SwCntntFrm::ShrinkFrm() |* |* Ersterstellung MA 30. Jul. 92 |* Letzte Aenderung MA 05. May. 94 |* |*************************************************************************/ SwTwips SwCntntFrm::ShrinkFrm( SwTwips nDist, sal_Bool bTst, sal_Bool bInfo ) { SWRECTFN( this ) ASSERT( nDist >= 0, "nDist < 0" ); ASSERT( nDist <= (Frm().*fnRect->fnGetHeight)(), "nDist > als aktuelle Grosse." ); if ( !bTst ) { SwTwips nRstHeight; if( GetUpper() ) nRstHeight = (Frm().*fnRect->fnBottomDist) ( (GetUpper()->*fnRect->fnGetPrtBottom)() ); else nRstHeight = 0; if( nRstHeight < 0 ) { SwTwips nNextHeight = 0; if( GetUpper()->IsSctFrm() && nDist > LONG_MAX/2 ) { SwFrm *pNxt = GetNext(); while( pNxt ) { nNextHeight += (pNxt->Frm().*fnRect->fnGetHeight)(); pNxt = pNxt->GetNext(); } } nRstHeight = nDist + nRstHeight - nNextHeight; } else nRstHeight = nDist; (Frm().*fnRect->fnSetHeight)( (Frm().*fnRect->fnGetHeight)() - nDist ); //Badaa: 2008-04-18 * Support for Classical Mongolian Script (SCMS) joint with Jiayanmin if( IsVertical() && !IsVertLR() ) Frm().Pos().X() += nDist; nDist = nRstHeight; if ( IsInTab() ) { SwTabFrm *pTab = FindTabFrm(); if ( pTab->GetTable()->GetHTMLTableLayout() && !pTab->IsJoinLocked() && !pTab->GetFmt()->GetDoc()->GetDocShell()->IsReadOnly() ) { pTab->InvalidatePos(); pTab->SetResizeHTMLTable(); } } } SwTwips nReal; if( GetUpper() && nDist > 0 ) { if( bTst || !GetUpper()->IsFooterFrm() ) nReal = GetUpper()->Shrink( nDist, bTst, bInfo ); else { nReal = 0; // #108745# Sorry, dear old footer friend, I'm not gonna invalidate you, // if there are any objects anchored inside your content, which // overlap with the shrinking frame. // This may lead to a footer frame that is too big, but this is better // than looping. // #109722# : The fix for #108745# was too strict. bool bInvalidate = true; const SwRect aRect( Frm() ); const SwPageFrm* pPage = FindPageFrm(); const SwSortedObjs* pSorted = pPage ? pPage->GetSortedObjs() : 0; if( pSorted ) { for ( sal_uInt16 i = 0; i < pSorted->Count(); ++i ) { const SwAnchoredObject* pAnchoredObj = (*pSorted)[i]; const SwRect aBound( pAnchoredObj->GetObjRectWithSpaces() ); if( aBound.Left() > aRect.Right() ) continue; if( aBound.IsOver( aRect ) ) { const SwFrmFmt& rFmt = pAnchoredObj->GetFrmFmt(); if( SURROUND_THROUGHT != rFmt.GetSurround().GetSurround() ) { const SwFrm* pAnchor = pAnchoredObj->GetAnchorFrm(); if ( pAnchor && pAnchor->FindFooterOrHeader() == GetUpper() ) { bInvalidate = false; break; } } } } } if ( bInvalidate ) GetUpper()->InvalidateSize(); } } else nReal = 0; if ( !bTst ) { //Die Position des naechsten Frm's veraendert sich auf jeden Fall. InvalidateNextPos(); //Wenn ich keinen Nachfolger habe, so muss ich mich eben selbst um //die Retusche kuemmern. if ( !GetNext() ) SetRetouche(); } return nReal; } /************************************************************************* |* |* SwCntntFrm::Modify() |* |* Beschreibung |* Ersterstellung AK 05-Mar-1991 |* Letzte Aenderung MA 13. Oct. 95 |* |*************************************************************************/ void SwCntntFrm::Modify( const SfxPoolItem* pOld, const SfxPoolItem * pNew ) { sal_uInt8 nInvFlags = 0; if( pNew && RES_ATTRSET_CHG == pNew->Which() ) { SfxItemIter aNIter( *((SwAttrSetChg*)pNew)->GetChgSet() ); SfxItemIter aOIter( *((SwAttrSetChg*)pOld)->GetChgSet() ); SwAttrSetChg aOldSet( *(SwAttrSetChg*)pOld ); SwAttrSetChg aNewSet( *(SwAttrSetChg*)pNew ); while( sal_True ) { _UpdateAttr( (SfxPoolItem*)aOIter.GetCurItem(), (SfxPoolItem*)aNIter.GetCurItem(), nInvFlags, &aOldSet, &aNewSet ); if( aNIter.IsAtEnd() ) break; aNIter.NextItem(); aOIter.NextItem(); } if ( aOldSet.Count() || aNewSet.Count() ) SwFrm::Modify( &aOldSet, &aNewSet ); } else _UpdateAttr( pOld, pNew, nInvFlags ); if ( nInvFlags != 0 ) { SwPageFrm *pPage = FindPageFrm(); InvalidatePage( pPage ); if ( nInvFlags & 0x01 ) SetCompletePaint(); if ( nInvFlags & 0x02 ) _InvalidatePos(); if ( nInvFlags & 0x04 ) _InvalidateSize(); if ( nInvFlags & 0x88 ) { if( IsInSct() && !GetPrev() ) { SwSectionFrm *pSect = FindSctFrm(); if( pSect->ContainsAny() == this ) { pSect->_InvalidatePrt(); pSect->InvalidatePage( pPage ); } } _InvalidatePrt(); } SwFrm* pNextFrm = GetIndNext(); if ( pNextFrm && nInvFlags & 0x10) { pNextFrm->_InvalidatePrt(); pNextFrm->InvalidatePage( pPage ); } if ( pNextFrm && nInvFlags & 0x80 ) { pNextFrm->SetCompletePaint(); } if ( nInvFlags & 0x20 ) { SwFrm* pPrevFrm = GetPrev(); if ( pPrevFrm ) { pPrevFrm->_InvalidatePrt(); pPrevFrm->InvalidatePage( pPage ); } } if ( nInvFlags & 0x40 ) InvalidateNextPos(); } } void SwCntntFrm::_UpdateAttr( const SfxPoolItem* pOld, const SfxPoolItem* pNew, sal_uInt8 &rInvFlags, SwAttrSetChg *pOldSet, SwAttrSetChg *pNewSet ) { sal_Bool bClear = sal_True; sal_uInt16 nWhich = pOld ? pOld->Which() : pNew ? pNew->Which() : 0; switch ( nWhich ) { case RES_FMT_CHG: rInvFlags = 0xFF; /* kein break hier */ case RES_PAGEDESC: //Attributaenderung (an/aus) if ( IsInDocBody() && !IsInTab() ) { rInvFlags |= 0x02; SwPageFrm *pPage = FindPageFrm(); if ( !GetPrev() ) CheckPageDescs( pPage ); if ( pPage && GetAttrSet()->GetPageDesc().GetNumOffset() ) ((SwRootFrm*)pPage->GetUpper())->SetVirtPageNum( sal_True ); SwDocPosUpdate aMsgHnt( pPage->Frm().Top() ); pPage->GetFmt()->GetDoc()->UpdatePageFlds( &aMsgHnt ); } break; case RES_UL_SPACE: { // OD 2004-02-18 #106629# - correction // Invalidation of the printing area of next frame, not only // for footnote content. if ( !GetIndNext() ) { SwFrm* pNxt = FindNext(); if ( pNxt ) { SwPageFrm* pPg = pNxt->FindPageFrm(); pNxt->InvalidatePage( pPg ); pNxt->_InvalidatePrt(); if( pNxt->IsSctFrm() ) { SwFrm* pCnt = ((SwSectionFrm*)pNxt)->ContainsAny(); if( pCnt ) { pCnt->_InvalidatePrt(); pCnt->InvalidatePage( pPg ); } } pNxt->SetCompletePaint(); } } // OD 2004-03-17 #i11860# if ( GetIndNext() && !GetUpper()->GetFmt()->getIDocumentSettingAccess()->get(IDocumentSettingAccess::USE_FORMER_OBJECT_POS) ) { // OD 2004-07-01 #i28701# - use new method <InvalidateObjs(..)> GetIndNext()->InvalidateObjs( true ); } Prepare( PREP_UL_SPACE ); //TxtFrm muss Zeilenabst. korrigieren. rInvFlags |= 0x80; /* kein Break hier */ } case RES_LR_SPACE: case RES_BOX: case RES_SHADOW: Prepare( PREP_FIXSIZE_CHG ); SwFrm::Modify( pOld, pNew ); rInvFlags |= 0x30; break; case RES_BREAK: { rInvFlags |= 0x42; const IDocumentSettingAccess* pIDSA = GetUpper()->GetFmt()->getIDocumentSettingAccess(); if( pIDSA->get(IDocumentSettingAccess::PARA_SPACE_MAX) || pIDSA->get(IDocumentSettingAccess::PARA_SPACE_MAX_AT_PAGES) ) { rInvFlags |= 0x1; SwFrm* pNxt = FindNext(); if( pNxt ) { SwPageFrm* pPg = pNxt->FindPageFrm(); pNxt->InvalidatePage( pPg ); pNxt->_InvalidatePrt(); if( pNxt->IsSctFrm() ) { SwFrm* pCnt = ((SwSectionFrm*)pNxt)->ContainsAny(); if( pCnt ) { pCnt->_InvalidatePrt(); pCnt->InvalidatePage( pPg ); } } pNxt->SetCompletePaint(); } } } break; // OD 2004-02-26 #i25029# case RES_PARATR_CONNECT_BORDER: { rInvFlags |= 0x01; if ( IsTxtFrm() ) { InvalidateNextPrtArea(); } if ( !GetIndNext() && IsInTab() && IsInSplitTableRow() ) { FindTabFrm()->InvalidateSize(); } } break; case RES_PARATR_TABSTOP: case RES_CHRATR_PROPORTIONALFONTSIZE: case RES_CHRATR_SHADOWED: case RES_CHRATR_AUTOKERN: case RES_CHRATR_UNDERLINE: case RES_CHRATR_OVERLINE: case RES_CHRATR_KERNING: case RES_CHRATR_FONT: case RES_CHRATR_FONTSIZE: case RES_CHRATR_ESCAPEMENT: case RES_CHRATR_CONTOUR: case RES_PARATR_NUMRULE: rInvFlags |= 0x01; break; case RES_FRM_SIZE: rInvFlags |= 0x01; /* no break here */ default: bClear = sal_False; } if ( bClear ) { if ( pOldSet || pNewSet ) { if ( pOldSet ) pOldSet->ClearItem( nWhich ); if ( pNewSet ) pNewSet->ClearItem( nWhich ); } else SwFrm::Modify( pOld, pNew ); } } /************************************************************************* |* |* SwLayoutFrm::SwLayoutFrm() |* |* Ersterstellung AK 14-Feb-1991 |* Letzte Aenderung MA 12. May. 95 |* |*************************************************************************/ SwLayoutFrm::SwLayoutFrm( SwFrmFmt* pFmt, SwFrm* pSib ): SwFrm( pFmt, pSib ), pLower( 0 ) { const SwFmtFrmSize &rFmtSize = pFmt->GetFrmSize(); if ( rFmtSize.GetHeightSizeType() == ATT_FIX_SIZE ) bFixSize = sal_True; } // --> OD 2004-06-29 #i28701# TYPEINIT1(SwLayoutFrm,SwFrm); // <-- /*-----------------10.06.99 09:42------------------- * SwLayoutFrm::InnerHeight() * --------------------------------------------------*/ SwTwips SwLayoutFrm::InnerHeight() const { if( !Lower() ) return 0; SwTwips nRet = 0; const SwFrm* pCnt = Lower(); SWRECTFN( this ) if( pCnt->IsColumnFrm() || pCnt->IsCellFrm() ) { do { SwTwips nTmp = ((SwLayoutFrm*)pCnt)->InnerHeight(); if( pCnt->GetValidPrtAreaFlag() ) nTmp += (pCnt->Frm().*fnRect->fnGetHeight)() - (pCnt->Prt().*fnRect->fnGetHeight)(); if( nRet < nTmp ) nRet = nTmp; pCnt = pCnt->GetNext(); } while ( pCnt ); } else { do { nRet += (pCnt->Frm().*fnRect->fnGetHeight)(); if( pCnt->IsCntntFrm() && ((SwTxtFrm*)pCnt)->IsUndersized() ) nRet += ((SwTxtFrm*)pCnt)->GetParHeight() - (pCnt->Prt().*fnRect->fnGetHeight)(); if( pCnt->IsLayoutFrm() && !pCnt->IsTabFrm() ) nRet += ((SwLayoutFrm*)pCnt)->InnerHeight() - (pCnt->Prt().*fnRect->fnGetHeight)(); pCnt = pCnt->GetNext(); } while( pCnt ); } return nRet; } /************************************************************************* |* |* SwLayoutFrm::GrowFrm() |* |* Ersterstellung MA 30. Jul. 92 |* Letzte Aenderung MA 23. Sep. 96 |* |*************************************************************************/ SwTwips SwLayoutFrm::GrowFrm( SwTwips nDist, sal_Bool bTst, sal_Bool bInfo ) { const ViewShell *pSh = getRootFrm()->GetCurrShell(); const sal_Bool bBrowse = pSh && pSh->GetViewOptions()->getBrowseMode(); const sal_uInt16 nTmpType = bBrowse ? 0x2084: 0x2004; //Row+Cell, Browse mit Body if( !(GetType() & nTmpType) && HasFixSize() ) return 0; SWRECTFN( this ) const SwTwips nFrmHeight = (Frm().*fnRect->fnGetHeight)(); const SwTwips nFrmPos = Frm().Pos().X(); if ( nFrmHeight > 0 && nDist > (LONG_MAX - nFrmHeight) ) nDist = LONG_MAX - nFrmHeight; SwTwips nMin = 0; if ( GetUpper() && !IsCellFrm() ) { SwFrm *pFrm = GetUpper()->Lower(); while( pFrm ) { nMin += (pFrm->Frm().*fnRect->fnGetHeight)(); pFrm = pFrm->GetNext(); } nMin = (GetUpper()->Prt().*fnRect->fnGetHeight)() - nMin; if ( nMin < 0 ) nMin = 0; } SwRect aOldFrm( Frm() ); sal_Bool bMoveAccFrm = sal_False; sal_Bool bChgPos = IsVertical() && !IsReverse(); if ( !bTst ) { (Frm().*fnRect->fnSetHeight)( nFrmHeight + nDist ); //Badaa: 2008-04-18 * Support for Classical Mongolian Script (SCMS) joint with Jiayanmin if( bChgPos && !IsVertLR() ) Frm().Pos().X() -= nDist; bMoveAccFrm = sal_True; } SwTwips nReal = nDist - nMin; if ( nReal > 0 ) { if ( GetUpper() ) { // AdjustNeighbourhood jetzt auch in Spalten (aber nicht in Rahmen) sal_uInt8 nAdjust = GetUpper()->IsFtnBossFrm() ? ((SwFtnBossFrm*)GetUpper())->NeighbourhoodAdjustment( this ) : NA_GROW_SHRINK; if( NA_ONLY_ADJUST == nAdjust ) nReal = AdjustNeighbourhood( nReal, bTst ); else { if( NA_ADJUST_GROW == nAdjust ) nReal += AdjustNeighbourhood( nReal, bTst ); SwTwips nGrow = 0; if( 0 < nReal ) { SwFrm* pToGrow = GetUpper(); // NEW TABLES // A cell with a row span of > 1 is allowed to grow the // line containing the end of the row span if it is // located in the same table frame: const SwCellFrm* pThisCell = dynamic_cast<const SwCellFrm*>(this); if ( pThisCell && pThisCell->GetLayoutRowSpan() > 1 ) { SwCellFrm& rEndCell = const_cast<SwCellFrm&>(pThisCell->FindStartEndOfRowSpanCell( false, true )); if ( -1 == rEndCell.GetTabBox()->getRowSpan() ) pToGrow = rEndCell.GetUpper(); else pToGrow = 0; } nGrow = pToGrow ? pToGrow->Grow( nReal, bTst, bInfo ) : 0; } if( NA_GROW_ADJUST == nAdjust && nGrow < nReal ) nReal += AdjustNeighbourhood( nReal - nGrow, bTst ); if ( IsFtnFrm() && (nGrow != nReal) && GetNext() ) { //Fussnoten koennen ihre Nachfolger verdraengen. SwTwips nSpace = bTst ? 0 : -nDist; const SwFrm *pFrm = GetUpper()->Lower(); do { nSpace += (pFrm->Frm().*fnRect->fnGetHeight)(); pFrm = pFrm->GetNext(); } while ( pFrm != GetNext() ); nSpace = (GetUpper()->Prt().*fnRect->fnGetHeight)() -nSpace; if ( nSpace < 0 ) nSpace = 0; nSpace += nGrow; if ( nReal > nSpace ) nReal = nSpace; if ( nReal && !bTst ) ((SwFtnFrm*)this)->InvalidateNxtFtnCnts( FindPageFrm() ); } else nReal = nGrow; } } else nReal = 0; nReal += nMin; } else nReal = nDist; if ( !bTst ) { if( nReal != nDist && // NEW TABLES ( !IsCellFrm() || static_cast<SwCellFrm*>(this)->GetLayoutRowSpan() > 1 ) ) { (Frm().*fnRect->fnSetHeight)( nFrmHeight + nReal ); //Badaa: 2008-04-18 * Support for Classical Mongolian Script (SCMS) joint with Jiayanmin if( bChgPos && !IsVertLR() ) Frm().Pos().X() = nFrmPos - nReal; bMoveAccFrm = sal_True; } if ( nReal ) { SwPageFrm *pPage = FindPageFrm(); if ( GetNext() ) { GetNext()->_InvalidatePos(); if ( GetNext()->IsCntntFrm() ) GetNext()->InvalidatePage( pPage ); } if ( !IsPageBodyFrm() ) { _InvalidateAll(); InvalidatePage( pPage ); } if ( !(GetType() & 0x1823) ) //Tab, Row, FtnCont, Root, Page NotifyLowerObjs(); if( IsCellFrm() ) InvaPercentLowers( nReal ); const SvxGraphicPosition ePos = GetFmt()->GetBackground().GetGraphicPos(); if ( GPOS_NONE != ePos && GPOS_TILED != ePos ) SetCompletePaint(); } } if( bMoveAccFrm && IsAccessibleFrm() ) { SwRootFrm *pRootFrm = getRootFrm(); if( pRootFrm && pRootFrm->IsAnyShellAccessible() && pRootFrm->GetCurrShell() ) { pRootFrm->GetCurrShell()->Imp()->MoveAccessibleFrm( this, aOldFrm ); } } return nReal; } /************************************************************************* |* |* SwLayoutFrm::ShrinkFrm() |* |* Ersterstellung MA 30. Jul. 92 |* Letzte Aenderung MA 25. Mar. 99 |* |*************************************************************************/ SwTwips SwLayoutFrm::ShrinkFrm( SwTwips nDist, sal_Bool bTst, sal_Bool bInfo ) { const ViewShell *pSh = getRootFrm()->GetCurrShell(); const sal_Bool bBrowse = pSh && pSh->GetViewOptions()->getBrowseMode(); const sal_uInt16 nTmpType = bBrowse ? 0x2084: 0x2004; //Row+Cell, Browse mit Body if( !(GetType() & nTmpType) && HasFixSize() ) return 0; ASSERT( nDist >= 0, "nDist < 0" ); SWRECTFN( this ) SwTwips nFrmHeight = (Frm().*fnRect->fnGetHeight)(); if ( nDist > nFrmHeight ) nDist = nFrmHeight; SwTwips nMin = 0; sal_Bool bChgPos = IsVertical() && !IsReverse(); if ( Lower() ) { if( !Lower()->IsNeighbourFrm() ) { const SwFrm *pFrm = Lower(); const long nTmp = (Prt().*fnRect->fnGetHeight)(); while( pFrm && nMin < nTmp ) { nMin += (pFrm->Frm().*fnRect->fnGetHeight)(); pFrm = pFrm->GetNext(); } } } SwTwips nReal = nDist; SwTwips nMinDiff = (Prt().*fnRect->fnGetHeight)() - nMin; if( nReal > nMinDiff ) nReal = nMinDiff; if( nReal <= 0 ) return nDist; SwRect aOldFrm( Frm() ); sal_Bool bMoveAccFrm = sal_False; SwTwips nRealDist = nReal; if ( !bTst ) { (Frm().*fnRect->fnSetHeight)( nFrmHeight - nReal ); //Badaa: 2008-04-18 * Support for Classical Mongolian Script (SCMS) joint with Jiayanmin if( bChgPos && !IsVertLR() ) Frm().Pos().X() += nReal; bMoveAccFrm = sal_True; } sal_uInt8 nAdjust = GetUpper() && GetUpper()->IsFtnBossFrm() ? ((SwFtnBossFrm*)GetUpper())->NeighbourhoodAdjustment( this ) : NA_GROW_SHRINK; // AdjustNeighbourhood auch in Spalten (aber nicht in Rahmen) if( NA_ONLY_ADJUST == nAdjust ) { if ( IsPageBodyFrm() && !bBrowse ) nReal = nDist; else { nReal = AdjustNeighbourhood( -nReal, bTst ); nReal *= -1; if ( !bTst && IsBodyFrm() && nReal < nRealDist ) { (Frm().*fnRect->fnSetHeight)( (Frm().*fnRect->fnGetHeight)() + nRealDist - nReal ); //Badaa: 2008-04-18 * Support for Classical Mongolian Script (SCMS) joint with Jiayanmin if( bChgPos && !IsVertLR() ) Frm().Pos().X() += nRealDist - nReal; ASSERT( !IsAccessibleFrm(), "bMoveAccFrm has to be set!" ); } } } else if( IsColumnFrm() || IsColBodyFrm() ) { SwTwips nTmp = GetUpper()->Shrink( nReal, bTst, bInfo ); if ( nTmp != nReal ) { (Frm().*fnRect->fnSetHeight)( (Frm().*fnRect->fnGetHeight)() + nReal - nTmp ); //Badaa: 2008-04-18 * Support for Classical Mongolian Script (SCMS) joint with Jiayanmin if( bChgPos && !IsVertLR() ) Frm().Pos().X() += nTmp - nReal; ASSERT( !IsAccessibleFrm(), "bMoveAccFrm has to be set!" ); nReal = nTmp; } } else { SwTwips nShrink = nReal; SwFrm* pToShrink = GetUpper(); const SwCellFrm* pThisCell = dynamic_cast<const SwCellFrm*>(this); // NEW TABLES if ( pThisCell && pThisCell->GetLayoutRowSpan() > 1 ) { SwCellFrm& rEndCell = const_cast<SwCellFrm&>(pThisCell->FindStartEndOfRowSpanCell( false, true )); pToShrink = rEndCell.GetUpper(); } nReal = pToShrink ? pToShrink->Shrink( nShrink, bTst, bInfo ) : 0; if( ( NA_GROW_ADJUST == nAdjust || NA_ADJUST_GROW == nAdjust ) && nReal < nShrink ) AdjustNeighbourhood( nReal - nShrink ); } if( bMoveAccFrm && IsAccessibleFrm() ) { SwRootFrm *pRootFrm = getRootFrm(); if( pRootFrm && pRootFrm->IsAnyShellAccessible() && pRootFrm->GetCurrShell() ) { pRootFrm->GetCurrShell()->Imp()->MoveAccessibleFrm( this, aOldFrm ); } } if ( !bTst && (IsCellFrm() || IsColumnFrm() ? nReal : nRealDist) ) { SwPageFrm *pPage = FindPageFrm(); if ( GetNext() ) { GetNext()->_InvalidatePos(); if ( GetNext()->IsCntntFrm() ) GetNext()->InvalidatePage( pPage ); if ( IsTabFrm() ) ((SwTabFrm*)this)->SetComplete(); } else { if ( IsRetoucheFrm() ) SetRetouche(); if ( IsTabFrm() ) { if( IsTabFrm() ) ((SwTabFrm*)this)->SetComplete(); if ( Lower() ) //Kann auch im Join stehen und leer sein! InvalidateNextPos(); } } if ( !IsBodyFrm() ) { _InvalidateAll(); InvalidatePage( pPage ); const SvxGraphicPosition ePos = GetFmt()->GetBackground().GetGraphicPos(); if ( GPOS_NONE != ePos && GPOS_TILED != ePos ) SetCompletePaint(); } if ( !(GetType() & 0x1823) ) //Tab, Row, FtnCont, Root, Page NotifyLowerObjs(); if( IsCellFrm() ) InvaPercentLowers( nReal ); SwCntntFrm *pCnt; if( IsFtnFrm() && !((SwFtnFrm*)this)->GetAttr()->GetFtn().IsEndNote() && ( GetFmt()->GetDoc()->GetFtnInfo().ePos != FTNPOS_CHAPTER || ( IsInSct() && FindSctFrm()->IsFtnAtEnd() ) ) && 0 != (pCnt = ((SwFtnFrm*)this)->GetRefFromAttr() ) ) { if ( pCnt->IsFollow() ) { // Wenn wir sowieso schon in einer anderen Spalte/Seite sitzen // als der Frame mit der Referenz, dann brauchen wir nicht // auch noch seinen Master zu invalidieren. SwFrm *pTmp = pCnt->FindFtnBossFrm(sal_True) == FindFtnBossFrm(sal_True) ? pCnt->FindMaster()->GetFrm() : pCnt; pTmp->Prepare( PREP_ADJUST_FRM ); pTmp->InvalidateSize(); } else pCnt->InvalidatePos(); } } return nReal; } /************************************************************************* |* |* SwLayoutFrm::ChgLowersProp() |* |* Beschreibung Aendert die Grosse der direkt untergeordneten Frm's |* die eine Fixe Groesse haben, proportional zur Groessenaenderung der |* PrtArea des Frm's. |* Die Variablen Frm's werden auch proportional angepasst; sie werden |* sich schon wieder zurechtwachsen/-schrumpfen. |* Ersterstellung MA 11.03.92 |* Letzte Aenderung AMA 2. Nov. 98 |* |*************************************************************************/ void SwLayoutFrm::ChgLowersProp( const Size& rOldSize ) { // no change of lower properties for root frame or if no lower exists. if ( IsRootFrm() || !Lower() ) return; // declare and init <SwFrm* pLowerFrm> with first lower SwFrm *pLowerFrm = Lower(); // declare and init const booleans <bHeightChgd> and <bWidthChg> const bool bHeightChgd = rOldSize.Height() != Prt().Height(); const bool bWidthChgd = rOldSize.Width() != Prt().Width(); // declare and init variables <bVert>, <bRev> and <fnRect> SWRECTFN( this ) // This shortcut basically tries to handle only lower frames that // are affected by the size change. Otherwise much more lower frames // are invalidated. if ( !( bVert ? bHeightChgd : bWidthChgd ) && ! Lower()->IsColumnFrm() && ( ( IsBodyFrm() && IsInDocBody() && ( !IsInSct() || !FindSctFrm()->IsColLocked() ) ) || // --> FME 2004-07-21 #i10826# Section frames without columns should not // invalidate all lowers! IsSctFrm() ) ) // <-- { // Determine page frame the body frame resp. the section frame belongs to. SwPageFrm *pPage = FindPageFrm(); // Determine last lower by traveling through them using <GetNext()>. // During travel check each section frame, if it will be sized to // maximum. If Yes, invalidate size of section frame and set // corresponding flags at the page. do { if( pLowerFrm->IsSctFrm() &&((SwSectionFrm*)pLowerFrm)->_ToMaximize() ) { pLowerFrm->_InvalidateSize(); pLowerFrm->InvalidatePage( pPage ); } if( pLowerFrm->GetNext() ) pLowerFrm = pLowerFrm->GetNext(); else break; } while( sal_True ); // If found last lower is a section frame containing no section // (section frame isn't valid and will be deleted in the future), // travel backwards. while( pLowerFrm->IsSctFrm() && !((SwSectionFrm*)pLowerFrm)->GetSection() && pLowerFrm->GetPrev() ) pLowerFrm = pLowerFrm->GetPrev(); // If found last lower is a section frame, set <pLowerFrm> to its last // content, if the section frame is valid and is not sized to maximum. // Otherwise set <pLowerFrm> to NULL - In this case body frame only // contains invalid section frames. if( pLowerFrm->IsSctFrm() ) pLowerFrm = ((SwSectionFrm*)pLowerFrm)->GetSection() && !((SwSectionFrm*)pLowerFrm)->ToMaximize( sal_False ) ? ((SwSectionFrm*)pLowerFrm)->FindLastCntnt() : NULL; // continue with found last lower, probably the last content of a section if ( pLowerFrm ) { // If <pLowerFrm> is in a table frame, set <pLowerFrm> to this table // frame and continue. if ( pLowerFrm->IsInTab() ) { // OD 28.10.2002 #97265# - safeguard for setting <pLowerFrm> to // its table frame - check, if the table frame is also a lower // of the body frame, in order to assure that <pLowerFrm> is not // set to a frame, which is an *upper* of the body frame. SwFrm* pTableFrm = pLowerFrm->FindTabFrm(); if ( IsAnLower( pTableFrm ) ) { pLowerFrm = pTableFrm; } } // Check, if variable size of body frame resp. section frame has grown // OD 28.10.2002 #97265# - correct check, if variable size has grown. SwTwips nOldHeight = bVert ? rOldSize.Width() : rOldSize.Height(); if( nOldHeight < (Prt().*fnRect->fnGetHeight)() ) { // If variable size of body|section frame has grown, only found // last lower and the position of the its next have to be invalidated. pLowerFrm->_InvalidateAll(); pLowerFrm->InvalidatePage( pPage ); if( !pLowerFrm->IsFlowFrm() || !SwFlowFrm::CastFlowFrm( pLowerFrm )->HasFollow() ) pLowerFrm->InvalidateNextPos( sal_True ); if ( pLowerFrm->IsTxtFrm() ) ((SwCntntFrm*)pLowerFrm)->Prepare( PREP_ADJUST_FRM ); } else { // variable size of body|section frame has shrinked. Thus, // invalidate all lowers not matching the new body|section size // and the dedicated new last lower. if( bVert ) { SwTwips nBot = Frm().Left() + Prt().Left(); while ( pLowerFrm->GetPrev() && pLowerFrm->Frm().Left() < nBot ) { pLowerFrm->_InvalidateAll(); pLowerFrm->InvalidatePage( pPage ); pLowerFrm = pLowerFrm->GetPrev(); } } else { SwTwips nBot = Frm().Top() + Prt().Bottom(); while ( pLowerFrm->GetPrev() && pLowerFrm->Frm().Top() > nBot ) { pLowerFrm->_InvalidateAll(); pLowerFrm->InvalidatePage( pPage ); pLowerFrm = pLowerFrm->GetPrev(); } } if ( pLowerFrm ) { pLowerFrm->_InvalidateSize(); pLowerFrm->InvalidatePage( pPage ); if ( pLowerFrm->IsTxtFrm() ) ((SwCntntFrm*)pLowerFrm)->Prepare( PREP_ADJUST_FRM ); } } // --> OD 2005-01-31 #i41694# - improvement by removing duplicates if ( pLowerFrm ) { if ( pLowerFrm->IsInSct() ) { // --> OD 2005-01-31 #i41694# - follow-up of issue #i10826#: // No invalidation of section frame, if it's the this. SwFrm* pSectFrm = pLowerFrm->FindSctFrm(); if( pSectFrm != this && IsAnLower( pSectFrm ) ) { pSectFrm->_InvalidateSize(); pSectFrm->InvalidatePage( pPage ); } // <-- } } // <-- } return; } // end of { special case } // Invalidate page for content only once. bool bInvaPageForCntnt = true; // Declare booleans <bFixChgd> and <bVarChgd>, indicating for text frame // adjustment, if fixed/variable size has changed. bool bFixChgd, bVarChgd; if( bVert == pLowerFrm->IsNeighbourFrm() ) { bFixChgd = bWidthChgd; bVarChgd = bHeightChgd; } else { bFixChgd = bHeightChgd; bVarChgd = bWidthChgd; } // Declare const unsigned short <nFixWidth> and init it this frame types // which has fixed width in vertical respectively horizontal layout. // In vertical layout these are neighbour frames (cell and column frames), // header frames and footer frames. // In horizontal layout these are all frames, which aren't neighbour frames. const sal_uInt16 nFixWidth = bVert ? (FRM_NEIGHBOUR | FRM_HEADFOOT) : ~FRM_NEIGHBOUR; // Declare const unsigned short <nFixHeight> and init it this frame types // which has fixed height in vertical respectively horizontal layout. // In vertical layout these are all frames, which aren't neighbour frames, // header frames, footer frames, body frames or foot note container frames. // In horizontal layout these are neighbour frames. const sal_uInt16 nFixHeight= bVert ? ~(FRM_NEIGHBOUR | FRM_HEADFOOT | FRM_BODYFTNC) : FRM_NEIGHBOUR; // Travel through all lowers using <GetNext()> while ( pLowerFrm ) { if ( pLowerFrm->IsTxtFrm() ) { // Text frames will only be invalidated - prepare invalidation if ( bFixChgd ) static_cast<SwCntntFrm*>(pLowerFrm)->Prepare( PREP_FIXSIZE_CHG ); if ( bVarChgd ) static_cast<SwCntntFrm*>(pLowerFrm)->Prepare( PREP_ADJUST_FRM ); } else { // If lower isn't a table, row, cell or section frame, adjust its // frame size. const sal_uInt16 nLowerType = pLowerFrm->GetType(); if ( !(nLowerType & (FRM_TAB|FRM_ROW|FRM_CELL|FRM_SECTION)) ) { if ( bWidthChgd ) { if( nLowerType & nFixWidth ) { // Considering previous conditions: // In vertical layout set width of column, header and // footer frames to its upper width. // In horizontal layout set width of header, footer, // foot note container, foot note, body and no-text // frames to its upper width. pLowerFrm->Frm().Width( Prt().Width() ); } else if( rOldSize.Width() && !pLowerFrm->IsFtnFrm() ) { // Adjust frame width proportional, if lower isn't a // foot note frame and condition <nLowerType & nFixWidth> // isn't true. // Considering previous conditions: // In vertical layout these are foot note container, // body and no-text frames. // In horizontal layout these are column and no-text frames. // OD 24.10.2002 #97265# - <double> calculation // Perform <double> calculation of new width, if // one of the coefficients is greater than 50000 SwTwips nNewWidth; if ( (pLowerFrm->Frm().Width() > 50000) || (Prt().Width() > 50000) ) { double nNewWidthTmp = ( double(pLowerFrm->Frm().Width()) * double(Prt().Width()) ) / double(rOldSize.Width()); nNewWidth = SwTwips(nNewWidthTmp); } else { nNewWidth = (pLowerFrm->Frm().Width() * Prt().Width()) / rOldSize.Width(); } pLowerFrm->Frm().Width( nNewWidth ); } } if ( bHeightChgd ) { if( nLowerType & nFixHeight ) { // Considering previous conditions: // In vertical layout set height of foot note and // no-text frames to its upper height. // In horizontal layout set height of column frames // to its upper height. pLowerFrm->Frm().Height( Prt().Height() ); } // OD 01.10.2002 #102211# // add conditions <!pLowerFrm->IsHeaderFrm()> and // <!pLowerFrm->IsFooterFrm()> in order to avoid that // the <Grow> of header or footer are overwritten. // NOTE: Height of header/footer frame is determined by contents. else if ( rOldSize.Height() && !pLowerFrm->IsFtnFrm() && !pLowerFrm->IsHeaderFrm() && !pLowerFrm->IsFooterFrm() ) { // Adjust frame height proportional, if lower isn't a // foot note, a header or a footer frame and // condition <nLowerType & nFixHeight> isn't true. // Considering previous conditions: // In vertical layout these are column, foot note container, // body and no-text frames. // In horizontal layout these are column, foot note // container, body and no-text frames. // OD 29.10.2002 #97265# - special case for page lowers // The page lowers that have to be adjusted on page height // change are the body frame and the foot note container // frame. // In vertical layout the height of both is directly // adjusted to the page height change. // In horizontal layout the height of the body frame is // directly adjsuted to the page height change and the // foot note frame height isn't touched, because its // determined by its content. // OD 31.03.2003 #108446# - apply special case for page // lowers - see description above - also for section columns. if ( IsPageFrm() || ( IsColumnFrm() && IsInSct() ) ) { ASSERT( pLowerFrm->IsBodyFrm() || pLowerFrm->IsFtnContFrm(), "ChgLowersProp - only for body or foot note container" ); if ( pLowerFrm->IsBodyFrm() || pLowerFrm->IsFtnContFrm() ) { if ( IsVertical() || pLowerFrm->IsBodyFrm() ) { SwTwips nNewHeight = pLowerFrm->Frm().Height() + ( Prt().Height() - rOldSize.Height() ); if ( nNewHeight < 0) { // OD 01.04.2003 #108446# - adjust assertion condition and text ASSERT( !( IsPageFrm() && (pLowerFrm->Frm().Height()>0) && (pLowerFrm->IsValid()) ), "ChgLowersProg - negative height for lower."); nNewHeight = 0; } pLowerFrm->Frm().Height( nNewHeight ); } } } else { SwTwips nNewHeight; // OD 24.10.2002 #97265# - <double> calculation // Perform <double> calculation of new height, if // one of the coefficients is greater than 50000 if ( (pLowerFrm->Frm().Height() > 50000) || (Prt().Height() > 50000) ) { double nNewHeightTmp = ( double(pLowerFrm->Frm().Height()) * double(Prt().Height()) ) / double(rOldSize.Height()); nNewHeight = SwTwips(nNewHeightTmp); } else { nNewHeight = ( pLowerFrm->Frm().Height() * Prt().Height() ) / rOldSize.Height(); } if( !pLowerFrm->GetNext() ) { SwTwips nSum = Prt().Height(); SwFrm* pTmp = Lower(); while( pTmp->GetNext() ) { if( !pTmp->IsFtnContFrm() || !pTmp->IsVertical() ) nSum -= pTmp->Frm().Height(); pTmp = pTmp->GetNext(); } if( nSum - nNewHeight == 1 && nSum == pLowerFrm->Frm().Height() ) nNewHeight = nSum; } pLowerFrm->Frm().Height( nNewHeight ); } } } } } // end of else { NOT text frame } pLowerFrm->_InvalidateAll(); if ( bInvaPageForCntnt && pLowerFrm->IsCntntFrm() ) { pLowerFrm->InvalidatePage(); bInvaPageForCntnt = false; } if ( !pLowerFrm->GetNext() && pLowerFrm->IsRetoucheFrm() ) { //Wenn ein Wachstum stattgefunden hat, und die untergeordneten //zur Retouche faehig sind (derzeit Tab, Section und Cntnt), so //trigger ich sie an. if ( rOldSize.Height() < Prt().SSize().Height() || rOldSize.Width() < Prt().SSize().Width() ) pLowerFrm->SetRetouche(); } pLowerFrm = pLowerFrm->GetNext(); } // Finally adjust the columns if width is set to auto // Possible optimisation: execute this code earlier in this function and // return??? if ( ( (bVert && bHeightChgd) || (! bVert && bWidthChgd) ) && Lower()->IsColumnFrm() ) { // get column attribute const SwFmtCol* pColAttr = NULL; if ( IsPageBodyFrm() ) { ASSERT( GetUpper()->IsPageFrm(), "Upper is not page frame" ) pColAttr = &GetUpper()->GetFmt()->GetCol(); } else { ASSERT( IsFlyFrm() || IsSctFrm(), "Columns not in fly or section" ) pColAttr = &GetFmt()->GetCol(); } if ( pColAttr->IsOrtho() && pColAttr->GetNumCols() > 1 ) AdjustColumns( pColAttr, sal_False ); } } /************************************************************************* |* |* SwLayoutFrm::Format() |* |* Beschreibung: "Formatiert" den Frame; Frm und PrtArea. |* Die Fixsize wird hier nicht eingestellt. |* Ersterstellung MA 28. Jul. 92 |* Letzte Aenderung MA 21. Mar. 95 |* |*************************************************************************/ void SwLayoutFrm::Format( const SwBorderAttrs *pAttrs ) { ASSERT( pAttrs, "LayoutFrm::Format, pAttrs ist 0." ); if ( bValidPrtArea && bValidSize ) return; const sal_uInt16 nLeft = (sal_uInt16)pAttrs->CalcLeft( this ); const sal_uInt16 nUpper = pAttrs->CalcTop(); const sal_uInt16 nRight = (sal_uInt16)((SwBorderAttrs*)pAttrs)->CalcRight( this ); const sal_uInt16 nLower = pAttrs->CalcBottom(); sal_Bool bVert = IsVertical() && !IsPageFrm(); //Badaa: 2008-04-18 * Support for Classical Mongolian Script (SCMS) joint with Jiayanmin SwRectFn fnRect = bVert ? ( IsVertLR() ? fnRectVertL2R : fnRectVert ) : fnRectHori; if ( !bValidPrtArea ) { bValidPrtArea = sal_True; (this->*fnRect->fnSetXMargins)( nLeft, nRight ); (this->*fnRect->fnSetYMargins)( nUpper, nLower ); } if ( !bValidSize ) { if ( !HasFixSize() ) { const SwTwips nBorder = nUpper + nLower; const SwFmtFrmSize &rSz = GetFmt()->GetFrmSize(); SwTwips nMinHeight = rSz.GetHeightSizeType() == ATT_MIN_SIZE ? rSz.GetHeight() : 0; do { bValidSize = sal_True; //Die Groesse in der VarSize wird durch den Inhalt plus den //Raendern bestimmt. SwTwips nRemaining = 0; SwFrm *pFrm = Lower(); while ( pFrm ) { nRemaining += (pFrm->Frm().*fnRect->fnGetHeight)(); if( pFrm->IsTxtFrm() && ((SwTxtFrm*)pFrm)->IsUndersized() ) // Dieser TxtFrm waere gern ein bisschen groesser nRemaining += ((SwTxtFrm*)pFrm)->GetParHeight() - (pFrm->Prt().*fnRect->fnGetHeight)(); else if( pFrm->IsSctFrm() && ((SwSectionFrm*)pFrm)->IsUndersized() ) nRemaining += ((SwSectionFrm*)pFrm)->Undersize(); pFrm = pFrm->GetNext(); } nRemaining += nBorder; nRemaining = Max( nRemaining, nMinHeight ); const SwTwips nDiff = nRemaining-(Frm().*fnRect->fnGetHeight)(); const long nOldLeft = (Frm().*fnRect->fnGetLeft)(); const long nOldTop = (Frm().*fnRect->fnGetTop)(); if ( nDiff ) { if ( nDiff > 0 ) Grow( nDiff ); else Shrink( -nDiff ); //Schnell auf dem kurzen Dienstweg die Position updaten. MakePos(); } //Unterkante des Uppers nicht ueberschreiten. if ( GetUpper() && (Frm().*fnRect->fnGetHeight)() ) { const SwTwips nLimit = (GetUpper()->*fnRect->fnGetPrtBottom)(); if( (this->*fnRect->fnSetLimit)( nLimit ) && nOldLeft == (Frm().*fnRect->fnGetLeft)() && nOldTop == (Frm().*fnRect->fnGetTop)() ) bValidSize = bValidPrtArea = sal_True; } } while ( !bValidSize ); } else if ( GetType() & 0x0018 ) { do { if ( Frm().Height() != pAttrs->GetSize().Height() ) ChgSize( Size( Frm().Width(), pAttrs->GetSize().Height())); bValidSize = sal_True; MakePos(); } while ( !bValidSize ); } else bValidSize = sal_True; } } /************************************************************************* |* |* SwLayoutFrm::InvalidatePercentLowers() |* |* Ersterstellung MA 13. Jun. 96 |* Letzte Aenderung MA 13. Jun. 96 |* |*************************************************************************/ static void InvaPercentFlys( SwFrm *pFrm, SwTwips nDiff ) { ASSERT( pFrm->GetDrawObjs(), "Can't find any Objects" ); for ( sal_uInt16 i = 0; i < pFrm->GetDrawObjs()->Count(); ++i ) { SwAnchoredObject* pAnchoredObj = (*pFrm->GetDrawObjs())[i]; if ( pAnchoredObj->ISA(SwFlyFrm) ) { SwFlyFrm *pFly = static_cast<SwFlyFrm*>(pAnchoredObj); const SwFmtFrmSize &rSz = pFly->GetFmt()->GetFrmSize(); if ( rSz.GetWidthPercent() || rSz.GetHeightPercent() ) { sal_Bool bNotify = sal_True; // If we've a fly with more than 90% relative height... if( rSz.GetHeightPercent() > 90 && pFly->GetAnchorFrm() && rSz.GetHeightPercent() != 0xFF && nDiff ) { const SwFrm *pRel = pFly->IsFlyLayFrm() ? pFly->GetAnchorFrm(): pFly->GetAnchorFrm()->GetUpper(); // ... and we have already more than 90% height and we // not allow the text to go through... // then a notifycation could cause an endless loop, e.g. // 100% height and no text wrap inside a cell of a table. if( pFly->Frm().Height()*10 > ( nDiff + pRel->Prt().Height() )*9 && pFly->GetFmt()->GetSurround().GetSurround() != SURROUND_THROUGHT ) bNotify = sal_False; } if( bNotify ) pFly->InvalidateSize(); } } } } void SwLayoutFrm::InvaPercentLowers( SwTwips nDiff ) { if ( GetDrawObjs() ) ::InvaPercentFlys( this, nDiff ); SwFrm *pFrm = ContainsCntnt(); if ( pFrm ) do { if ( pFrm->IsInTab() && !IsTabFrm() ) { SwFrm *pTmp = pFrm->FindTabFrm(); ASSERT( pTmp, "Where's my TabFrm?" ); if( IsAnLower( pTmp ) ) pFrm = pTmp; } if ( pFrm->IsTabFrm() ) { const SwFmtFrmSize &rSz = ((SwLayoutFrm*)pFrm)->GetFmt()->GetFrmSize(); if ( rSz.GetWidthPercent() || rSz.GetHeightPercent() ) pFrm->InvalidatePrt(); } else if ( pFrm->GetDrawObjs() ) ::InvaPercentFlys( pFrm, nDiff ); pFrm = pFrm->FindNextCnt(); } while ( pFrm && IsAnLower( pFrm ) ) ; } /************************************************************************* |* |* SwLayoutFrm::CalcRel() |* |* Ersterstellung MA 13. Jun. 96 |* Letzte Aenderung MA 10. Oct. 96 |* |*************************************************************************/ long SwLayoutFrm::CalcRel( const SwFmtFrmSize &rSz, sal_Bool ) const { long nRet = rSz.GetWidth(), nPercent = rSz.GetWidthPercent(); if ( nPercent ) { const SwFrm *pRel = GetUpper(); long nRel = LONG_MAX; const ViewShell *pSh = getRootFrm()->GetCurrShell(); const sal_Bool bBrowseMode = pSh && pSh->GetViewOptions()->getBrowseMode(); if( pRel->IsPageBodyFrm() && pSh && bBrowseMode && pSh->VisArea().Width() ) { nRel = pSh->GetBrowseWidth(); long nDiff = nRel - pRel->Prt().Width(); if ( nDiff > 0 ) nRel -= nDiff; } nRel = Min( nRel, pRel->Prt().Width() ); nRet = nRel * nPercent / 100; } return nRet; } /************************************************************************* |* Local helpers for SwLayoutFrm::FormatWidthCols() |*************************************************************************/ long MA_FASTCALL lcl_CalcMinColDiff( SwLayoutFrm *pLayFrm ) { long nDiff = 0, nFirstDiff = 0; SwLayoutFrm *pCol = (SwLayoutFrm*)pLayFrm->Lower(); ASSERT( pCol, "Where's the columnframe?" ); SwFrm *pFrm = pCol->Lower(); do { if( pFrm && pFrm->IsBodyFrm() ) pFrm = ((SwBodyFrm*)pFrm)->Lower(); if ( pFrm && pFrm->IsTxtFrm() ) { const long nTmp = ((SwTxtFrm*)pFrm)->FirstLineHeight(); if ( nTmp != USHRT_MAX ) { if ( pCol == pLayFrm->Lower() ) nFirstDiff = nTmp; else nDiff = nDiff ? Min( nDiff, nTmp ) : nTmp; } } //Leere Spalten ueberspringen! pCol = (SwLayoutFrm*)pCol->GetNext(); while ( pCol && 0 == (pFrm = pCol->Lower()) ) pCol = (SwLayoutFrm*)pCol->GetNext(); } while ( pFrm && pCol ); return nDiff ? nDiff : nFirstDiff ? nFirstDiff : 240; } sal_Bool lcl_IsFlyHeightClipped( SwLayoutFrm *pLay ) { SwFrm *pFrm = pLay->ContainsCntnt(); while ( pFrm ) { if ( pFrm->IsInTab() ) pFrm = pFrm->FindTabFrm(); if ( pFrm->GetDrawObjs() ) { sal_uInt32 nCnt = pFrm->GetDrawObjs()->Count(); for ( sal_uInt16 i = 0; i < nCnt; ++i ) { SwAnchoredObject* pAnchoredObj = (*pFrm->GetDrawObjs())[i]; if ( pAnchoredObj->ISA(SwFlyFrm) ) { SwFlyFrm* pFly = static_cast<SwFlyFrm*>(pAnchoredObj); if ( pFly->IsHeightClipped() && ( !pFly->IsFlyFreeFrm() || pFly->GetPageFrm() ) ) return sal_True; } } } pFrm = pFrm->FindNextCnt(); } return sal_False; } /************************************************************************* |* SwLayoutFrm::FormatWidthCols() |*************************************************************************/ void SwLayoutFrm::FormatWidthCols( const SwBorderAttrs &rAttrs, const SwTwips nBorder, const SwTwips nMinHeight ) { //Wenn Spalten im Spiel sind, so wird die Groesse an der //letzten Spalte ausgerichtet. //1. Inhalt formatieren. //2. Hoehe der letzten Spalte ermitteln, wenn diese zu // zu gross ist muss der Fly wachsen. // Der Betrag um den der Fly waechst ist aber nicht etwa // der Betrag des Ueberhangs, denn wir muessen davon // ausgehen, dass etwas Masse zurueckfliesst und so // zusaetzlicher Platz geschaffen wird. // Im Ersten Ansatz ist der Betrag um den gewachsen wird // der Ueberhang geteilt durch die Spaltenanzahl oder // der Ueberhang selbst wenn er kleiner als die Spalten- // anzahl ist. //3. Weiter mit 1. bis zur Stabilitaet. const SwFmtCol &rCol = rAttrs.GetAttrSet().GetCol(); const sal_uInt16 nNumCols = rCol.GetNumCols(); sal_Bool bEnd = sal_False; sal_Bool bBackLock = sal_False; ViewShell *pSh = getRootFrm()->GetCurrShell(); SwViewImp *pImp = pSh ? pSh->Imp() : 0; { // Zugrunde liegender Algorithmus // Es wird versucht, eine optimale Hoehe fuer die Spalten zu finden. // nMinimum beginnt mit der uebergebenen Mindesthoehe und wird dann als // Maximum der Hoehen gepflegt, bei denen noch Spalteninhalt aus einer // Spalte herausragt. // nMaximum beginnt bei LONG_MAX und wird als Minimum der Hoehen gepflegt, // bei denen der Inhalt gepasst hat. // Bei spaltigen Bereichen beginnt nMaximum bei dem maximalen Wert, den // die Umgebung vorgibt, dies kann natuerlich ein Wert sein, bei dem noch // Inhalt heraushaengt. // Es werden die Spalten formatiert, wenn Inhalt heraushaengt, wird nMinimum // ggf. angepasst, dann wird gewachsen, mindestens um nMinDiff, aber nicht ueber // ein groesseres nMaximum hinaus. Wenn kein Inhalt heraushaengt, sondern // noch Luft in einer Spalte ist, schrumpfen wir entsprechend, mindestens um // nMinDiff, aber nicht unter das nMinimum. // Abgebrochen wird, wenn kein Inhalt mehr heraushaengt und das Minimum sich auf // weniger als ein MinDiff dem Maximum angenaehert hat oder das von der // Umgebung vorgegebene Maximum erreicht ist und trotzdem Inhalt heraus- // haengt. // Kritik an der Implementation // 1. Es kann theoretisch Situationen geben, in denen der Inhalt in einer geringeren // Hoehe passt und in einer groesseren Hoehe nicht passt. Damit der Code robust // gegen solche Verhaeltnisse ist, sind ein paar Abfragen bezgl. Minimum und Maximum // drin, die wahrscheinlich niemals zuschlagen koennen. // 2. Es wird fuer das Schrumpfen das gleiche nMinDiff benutzt wie fuer das Wachstum, // das nMinDiff ist allerdings mehr oder weniger die kleinste erste Zeilenhoehe und // als Mindestwert fuer das Schrumpfen nicht unbedingt optimal. long nMinimum = nMinHeight; long nMaximum; sal_Bool bNoBalance = sal_False; SWRECTFN( this ) if( IsSctFrm() ) { nMaximum = (Frm().*fnRect->fnGetHeight)() - nBorder + (Frm().*fnRect->fnBottomDist)( (GetUpper()->*fnRect->fnGetPrtBottom)() ); nMaximum += GetUpper()->Grow( LONG_MAX, sal_True ); if( nMaximum < nMinimum ) { if( nMaximum < 0 ) nMinimum = nMaximum = 0; else nMinimum = nMaximum; } if( nMaximum > BROWSE_HEIGHT ) nMaximum = BROWSE_HEIGHT; bNoBalance = ((SwSectionFrm*)this)->GetSection()->GetFmt()-> GetBalancedColumns().GetValue(); SwFrm* pAny = ContainsAny(); if( bNoBalance || ( !(Frm().*fnRect->fnGetHeight)() && pAny ) ) { long nTop = (this->*fnRect->fnGetTopMargin)(); // --> OD 2004-11-01 #i23129# - correction: enlarge section // to the calculated maximum height. (Frm().*fnRect->fnAddBottom)( nMaximum - (Frm().*fnRect->fnGetHeight)() ); // <-- if( nTop > nMaximum ) nTop = nMaximum; (this->*fnRect->fnSetYMargins)( nTop, 0 ); } if( !pAny && !((SwSectionFrm*)this)->IsFtnLock() ) { SwFtnContFrm* pFtnCont = ((SwSectionFrm*)this)->ContainsFtnCont(); if( pFtnCont ) { SwFrm* pFtnAny = pFtnCont->ContainsAny(); if( pFtnAny && pFtnAny->IsValid() ) { bBackLock = sal_True; ((SwSectionFrm*)this)->SetFtnLock( sal_True ); } } } } else nMaximum = LONG_MAX; // --> OD 2004-08-25 #i3317# - reset temporarly consideration // of wrapping style influence SwPageFrm* pPageFrm = FindPageFrm(); SwSortedObjs* pObjs = pPageFrm ? pPageFrm->GetSortedObjs() : 0L; if ( pObjs ) { sal_uInt32 i = 0; for ( i = 0; i < pObjs->Count(); ++i ) { SwAnchoredObject* pAnchoredObj = (*pObjs)[i]; if ( IsAnLower( pAnchoredObj->GetAnchorFrm() ) ) { pAnchoredObj->SetTmpConsiderWrapInfluence( false ); } } } // <-- do { //Kann eine Weile dauern, deshalb hier auf Waitcrsr pruefen. if ( pImp ) pImp->CheckWaitCrsr(); bValidSize = sal_True; //Erstmal die Spalten formatieren, das entlastet den //Stack ein wenig. //Bei der Gelegenheit stellen wir auch gleich mal die //Breiten und Hoehen der Spalten ein (so sie denn falsch sind). SwLayoutFrm *pCol = (SwLayoutFrm*)Lower(); // --> FME 2004-07-19 #i27399# // Simply setting the column width based on the values returned by // CalcColWidth does not work for automatic column width. AdjustColumns( &rCol, sal_False ); // <-- for ( sal_uInt16 i = 0; i < nNumCols; ++i ) { pCol->Calc(); // ColumnFrms besitzen jetzt einen BodyFrm, der auch kalkuliert werden will pCol->Lower()->Calc(); if( pCol->Lower()->GetNext() ) pCol->Lower()->GetNext()->Calc(); // SwFtnCont pCol = (SwLayoutFrm*)pCol->GetNext(); } ::CalcCntnt( this ); pCol = (SwLayoutFrm*)Lower(); ASSERT( pCol && pCol->GetNext(), ":-( Spalten auf Urlaub?"); // bMinDiff wird gesetzt, wenn es keine leere Spalte gibt sal_Bool bMinDiff = sal_True; // OD 28.03.2003 #108446# - check for all column content and all columns while ( bMinDiff && pCol ) { bMinDiff = 0 != pCol->ContainsCntnt(); pCol = (SwLayoutFrm*)pCol->GetNext(); } pCol = (SwLayoutFrm*)Lower(); // OD 28.03.2003 #108446# - initialize local variable SwFrm *pLow = NULL; SwTwips nDiff = 0; SwTwips nMaxFree = 0; SwTwips nAllFree = LONG_MAX; // bFoundLower wird gesetzt, wenn es mind. eine nichtleere Spalte gibt sal_Bool bFoundLower = sal_False; while( pCol ) { SwLayoutFrm* pLay = (SwLayoutFrm*)pCol->Lower(); SwTwips nInnerHeight = (pLay->Frm().*fnRect->fnGetHeight)() - (pLay->Prt().*fnRect->fnGetHeight)(); if( pLay->Lower() ) { bFoundLower = sal_True; nInnerHeight += pLay->InnerHeight(); } else if( nInnerHeight < 0 ) nInnerHeight = 0; if( pLay->GetNext() ) { bFoundLower = sal_True; pLay = (SwLayoutFrm*)pLay->GetNext(); ASSERT( pLay->IsFtnContFrm(),"FtnContainer exspected" ); nInnerHeight += pLay->InnerHeight(); nInnerHeight += (pLay->Frm().*fnRect->fnGetHeight)() - (pLay->Prt().*fnRect->fnGetHeight)(); } nInnerHeight -= (pCol->Prt().*fnRect->fnGetHeight)(); if( nInnerHeight > nDiff ) { nDiff = nInnerHeight; nAllFree = 0; } else { if( nMaxFree < -nInnerHeight ) nMaxFree = -nInnerHeight; if( nAllFree > -nInnerHeight ) nAllFree = -nInnerHeight; } pCol = (SwLayoutFrm*)pCol->GetNext(); } if ( bFoundLower || ( IsSctFrm() && ((SwSectionFrm*)this)->HasFollow() ) ) { SwTwips nMinDiff = ::lcl_CalcMinColDiff( this ); // Hier wird entschieden, ob wir wachsen muessen, naemlich wenn // ein Spalteninhalt (nDiff) oder ein Fly herausragt. // Bei spaltigen Bereichen wird beruecksichtigt, dass mit dem // Besitz eines nichtleeren Follows die Groesse festgelegt ist. if ( nDiff || ::lcl_IsFlyHeightClipped( this ) || ( IsSctFrm() && ((SwSectionFrm*)this)->CalcMinDiff( nMinDiff ) ) ) { long nPrtHeight = (Prt().*fnRect->fnGetHeight)(); // Das Minimum darf nicht kleiner sein als unsere PrtHeight, // solange noch etwas herausragt. if( nMinimum < nPrtHeight ) nMinimum = nPrtHeight; // Es muss sichergestellt sein, dass das Maximum nicht kleiner // als die PrtHeight ist, wenn noch etwas herausragt if( nMaximum < nPrtHeight ) nMaximum = nPrtHeight; // Robust, aber kann das ueberhaupt eintreten? if( !nDiff ) // wenn nur Flys herausragen, wachsen wir um nMinDiff nDiff = nMinDiff; // Wenn wir um mehr als nMinDiff wachsen wollen, wird dies auf die // Spalten verteilt if ( Abs(nDiff - nMinDiff) > nNumCols && nDiff > (long)nNumCols ) nDiff /= nNumCols; if ( bMinDiff ) { // Wenn es keinen leeren Spalten gibt, wollen wir mind. um nMinDiff // wachsen. Sonderfall: Wenn wir kleiner als die minimale Frmhoehe // sind und die PrtHeight kleiner als nMinDiff ist, wachsen wir so, // dass die PrtHeight hinterher genau nMinDiff ist. long nFrmHeight = (Frm().*fnRect->fnGetHeight)(); if ( nFrmHeight > nMinHeight || nPrtHeight >= nMinDiff ) nDiff = Max( nDiff, nMinDiff ); else if( nDiff < nMinDiff ) nDiff = nMinDiff - nPrtHeight + 1; } // nMaximum ist eine Groesse, in der der Inhalt gepasst hat, // oder der von der Umgebung vorgegebene Wert, deshalb // brauchen wir nicht ueber diesen Wrt hinauswachsen. if( nDiff + nPrtHeight > nMaximum ) nDiff = nMaximum - nPrtHeight; } else if( nMaximum > nMinimum ) // Wir passen, haben wir auch noch Spielraum? { long nPrtHeight = (Prt().*fnRect->fnGetHeight)(); if ( nMaximum < nPrtHeight ) nDiff = nMaximum - nPrtHeight; // wir sind ueber eine funktionierende // Hoehe hinausgewachsen und schrumpfen wieder auf diese zurueck, // aber kann das ueberhaupt eintreten? else { // Wir haben ein neues Maximum, eine Groesse, fuer die der Inhalt passt. nMaximum = nPrtHeight; // Wenn der Freiraum in den Spalten groesser ist als nMinDiff und wir // nicht dadurch wieder unter das Minimum rutschen, wollen wir ein wenig // Luft herauslassen. if ( !bNoBalance && // --> OD 2004-11-04 #i23129# - <nMinDiff> can be // big, because of an object at the beginning of // a column. Thus, decrease optimization here. //nMaxFree >= nMinDiff && nMaxFree > 0 && // <-- ( !nAllFree || nMinimum < nPrtHeight - nMinDiff ) ) { nMaxFree /= nNumCols; // auf die Spalten verteilen nDiff = nMaxFree < nMinDiff ? -nMinDiff : -nMaxFree; // mind. nMinDiff if( nPrtHeight + nDiff <= nMinimum ) // Unter das Minimum? nDiff = ( nMinimum - nMaximum ) / 2; // dann lieber die Mitte } else if( nAllFree ) { nDiff = -nAllFree; if( nPrtHeight + nDiff <= nMinimum ) // Less than minimum? nDiff = ( nMinimum - nMaximum ) / 2; // Take the center } } } if( nDiff ) // jetzt wird geschrumpft oder gewachsen.. { Size aOldSz( Prt().SSize() ); long nTop = (this->*fnRect->fnGetTopMargin)(); nDiff = (Prt().*fnRect->fnGetHeight)() + nDiff + nBorder - (Frm().*fnRect->fnGetHeight)(); (Frm().*fnRect->fnAddBottom)( nDiff ); // --> OD 2006-08-16 #i68520# if ( dynamic_cast<SwFlyFrm*>(this) ) { dynamic_cast<SwFlyFrm*>(this)->InvalidateObjRectWithSpaces(); } // <-- (this->*fnRect->fnSetYMargins)( nTop, nBorder - nTop ); ChgLowersProp( aOldSz ); NotifyLowerObjs(); // --> OD 2004-08-25 #i3317# - reset temporarly consideration // of wrapping style influence SwPageFrm* pTmpPageFrm = FindPageFrm(); SwSortedObjs* pTmpObjs = pTmpPageFrm ? pTmpPageFrm->GetSortedObjs() : 0L; if ( pTmpObjs ) { sal_uInt32 i = 0; for ( i = 0; i < pTmpObjs->Count(); ++i ) { SwAnchoredObject* pAnchoredObj = (*pTmpObjs)[i]; if ( IsAnLower( pAnchoredObj->GetAnchorFrm() ) ) { pAnchoredObj->SetTmpConsiderWrapInfluence( false ); } } } // <-- //Es muss geeignet invalidiert werden, damit //sich die Frms huebsch ausbalancieren //- Der jeweils erste ab der zweiten Spalte bekommt // ein InvalidatePos(); pCol = (SwLayoutFrm*)Lower()->GetNext(); while ( pCol ) { pLow = pCol->Lower(); if ( pLow ) pLow->_InvalidatePos(); pCol = (SwLayoutFrm*)pCol->GetNext(); } if( IsSctFrm() && ((SwSectionFrm*)this)->HasFollow() ) { // Wenn wir einen Follow erzeugt haben, muessen wir // seinem Inhalt die Chance geben, im CalcCntnt // zurueckzufliessen SwCntntFrm* pTmpCntnt = ((SwSectionFrm*)this)->GetFollow()->ContainsCntnt(); if( pTmpCntnt ) pTmpCntnt->_InvalidatePos(); } } else bEnd = sal_True; } else bEnd = sal_True; } while ( !bEnd || !bValidSize ); } // OD 01.04.2003 #108446# - Don't collect endnotes for sections. Thus, set // 2nd parameter to <true>. ::CalcCntnt( this, true ); if( IsSctFrm() ) { // OD 14.03.2003 #i11760# - adjust 2nd parameter - sal_True --> true ::CalcCntnt( this, true ); if( bBackLock ) ((SwSectionFrm*)this)->SetFtnLock( sal_False ); } } /************************************************************************* |* |* SwRootFrm::InvalidateAllCntnt() |* |* Ersterstellung MA 13. Feb. 98 |* Letzte Aenderung MA 12. Aug. 00 |* |*************************************************************************/ SwCntntFrm* lcl_InvalidateSection( SwFrm *pCnt, sal_uInt8 nInv ) { SwSectionFrm* pSect = pCnt->FindSctFrm(); // Wenn unser CntntFrm in einer Tabelle oder Fussnote steht, sind nur // Bereiche gemeint, die ebenfalls innerhalb liegen. // Ausnahme: Wenn direkt eine Tabelle uebergeben wird. if( ( ( pCnt->IsInTab() && !pSect->IsInTab() ) || ( pCnt->IsInFtn() && !pSect->IsInFtn() ) ) && !pCnt->IsTabFrm() ) return NULL; if( nInv & INV_SIZE ) pSect->_InvalidateSize(); if( nInv & INV_POS ) pSect->_InvalidatePos(); if( nInv & INV_PRTAREA ) pSect->_InvalidatePrt(); SwFlowFrm *pFoll = pSect->GetFollow(); // Temporary separation from follow pSect->SetFollow( NULL ); SwCntntFrm* pRet = pSect->FindLastCntnt(); pSect->SetFollow( pFoll ); return pRet; } SwCntntFrm* lcl_InvalidateTable( SwTabFrm *pTable, sal_uInt8 nInv ) { if( ( nInv & INV_SECTION ) && pTable->IsInSct() ) lcl_InvalidateSection( pTable, nInv ); if( nInv & INV_SIZE ) pTable->_InvalidateSize(); if( nInv & INV_POS ) pTable->_InvalidatePos(); if( nInv & INV_PRTAREA ) pTable->_InvalidatePrt(); return pTable->FindLastCntnt(); } void lcl_InvalidateAllCntnt( SwCntntFrm *pCnt, sal_uInt8 nInv ); void lcl_InvalidateCntnt( SwCntntFrm *pCnt, sal_uInt8 nInv ) { SwCntntFrm *pLastTabCnt = NULL; SwCntntFrm *pLastSctCnt = NULL; while ( pCnt ) { if( nInv & INV_SECTION ) { if( pCnt->IsInSct() ) { // Siehe oben bei Tabellen if( !pLastSctCnt ) pLastSctCnt = lcl_InvalidateSection( pCnt, nInv ); if( pLastSctCnt == pCnt ) pLastSctCnt = NULL; } #ifdef DBG_UTIL else ASSERT( !pLastSctCnt, "Where's the last SctCntnt?" ); #endif } if( nInv & INV_TABLE ) { if( pCnt->IsInTab() ) { // Um nicht fuer jeden CntntFrm einer Tabelle das FindTabFrm() zu rufen // und wieder die gleiche Tabelle zu invalidieren, merken wir uns den letzten // CntntFrm der Tabelle und reagieren erst wieder auf IsInTab(), wenn wir // an diesem vorbei sind. // Beim Eintritt in die Tabelle wird der LastSctCnt auf Null gesetzt, // damit Bereiche im Innern der Tabelle richtig invalidiert werden. // Sollte die Tabelle selbst in einem Bereich stehen, so wird an // diesem die Invalidierung bis zu dreimal durchgefuehrt, das ist vertretbar. if( !pLastTabCnt ) { pLastTabCnt = lcl_InvalidateTable( pCnt->FindTabFrm(), nInv ); pLastSctCnt = NULL; } if( pLastTabCnt == pCnt ) { pLastTabCnt = NULL; pLastSctCnt = NULL; } } #ifdef DBG_UTIL else ASSERT( !pLastTabCnt, "Where's the last TabCntnt?" ); #endif } if( nInv & INV_SIZE ) pCnt->Prepare( PREP_CLEAR, 0, sal_False ); if( nInv & INV_POS ) pCnt->_InvalidatePos(); if( nInv & INV_PRTAREA ) pCnt->_InvalidatePrt(); if ( nInv & INV_LINENUM ) pCnt->InvalidateLineNum(); if ( pCnt->GetDrawObjs() ) lcl_InvalidateAllCntnt( pCnt, nInv ); pCnt = pCnt->GetNextCntntFrm(); } } void lcl_InvalidateAllCntnt( SwCntntFrm *pCnt, sal_uInt8 nInv ) { SwSortedObjs &rObjs = *pCnt->GetDrawObjs(); for ( sal_uInt16 i = 0; i < rObjs.Count(); ++i ) { SwAnchoredObject* pAnchoredObj = rObjs[i]; if ( pAnchoredObj->ISA(SwFlyFrm) ) { SwFlyFrm *pFly = static_cast<SwFlyFrm*>(pAnchoredObj); if ( pFly->IsFlyInCntFrm() ) { ::lcl_InvalidateCntnt( pFly->ContainsCntnt(), nInv ); if( nInv & INV_DIRECTION ) pFly->CheckDirChange(); } } } } void SwRootFrm::InvalidateAllCntnt( sal_uInt8 nInv ) { // Erst werden alle Seitengebundenen FlyFrms abgearbeitet. SwPageFrm *pPage = (SwPageFrm*)Lower(); while( pPage ) { pPage->InvalidateFlyLayout(); pPage->InvalidateFlyCntnt(); pPage->InvalidateFlyInCnt(); pPage->InvalidateLayout(); pPage->InvalidateCntnt(); pPage->InvalidatePage( pPage ); //Damit ggf. auch der Turbo verschwindet if ( pPage->GetSortedObjs() ) { const SwSortedObjs &rObjs = *pPage->GetSortedObjs(); for ( sal_uInt16 i = 0; i < rObjs.Count(); ++i ) { SwAnchoredObject* pAnchoredObj = rObjs[i]; if ( pAnchoredObj->ISA(SwFlyFrm) ) { SwFlyFrm* pFly = static_cast<SwFlyFrm*>(pAnchoredObj); ::lcl_InvalidateCntnt( pFly->ContainsCntnt(), nInv ); if ( nInv & INV_DIRECTION ) pFly->CheckDirChange(); } } } if( nInv & INV_DIRECTION ) pPage->CheckDirChange(); pPage = (SwPageFrm*)(pPage->GetNext()); } //Hier den gesamten Dokumentinhalt und die zeichengebundenen Flys. ::lcl_InvalidateCntnt( ContainsCntnt(), nInv ); if( nInv & INV_PRTAREA ) { ViewShell *pSh = getRootFrm()->GetCurrShell(); if( pSh ) pSh->InvalidateWindows( Frm() ); } } /** method to invalidate/re-calculate the position of all floating screen objects (Writer fly frames and drawing objects), which are anchored to paragraph or to character. OD 2004-03-16 #i11860# @author OD */ void SwRootFrm::InvalidateAllObjPos() { const SwPageFrm* pPageFrm = static_cast<const SwPageFrm*>(Lower()); while( pPageFrm ) { pPageFrm->InvalidateFlyLayout(); if ( pPageFrm->GetSortedObjs() ) { const SwSortedObjs& rObjs = *(pPageFrm->GetSortedObjs()); for ( sal_uInt8 i = 0; i < rObjs.Count(); ++i ) { SwAnchoredObject* pAnchoredObj = rObjs[i]; const SwFmtAnchor& rAnch = pAnchoredObj->GetFrmFmt().GetAnchor(); if ((rAnch.GetAnchorId() != FLY_AT_PARA) && (rAnch.GetAnchorId() != FLY_AT_CHAR)) { // only to paragraph and to character anchored objects are considered. continue; } // --> OD 2004-07-07 #i28701# - special invalidation for anchored // objects, whose wrapping style influence has to be considered. if ( pAnchoredObj->ConsiderObjWrapInfluenceOnObjPos() ) pAnchoredObj->InvalidateObjPosForConsiderWrapInfluence( true ); else pAnchoredObj->InvalidateObjPos(); // <-- } } pPageFrm = static_cast<const SwPageFrm*>(pPageFrm->GetNext()); } }