summaryrefslogtreecommitdiff
path: root/sw/source/core/layout/flowfrm.cxx
diff options
context:
space:
mode:
Diffstat (limited to 'sw/source/core/layout/flowfrm.cxx')
-rw-r--r--sw/source/core/layout/flowfrm.cxx2736
1 files changed, 2736 insertions, 0 deletions
diff --git a/sw/source/core/layout/flowfrm.cxx b/sw/source/core/layout/flowfrm.cxx
new file mode 100644
index 000000000000..d0015b12114d
--- /dev/null
+++ b/sw/source/core/layout/flowfrm.cxx
@@ -0,0 +1,2736 @@
+/*************************************************************************
+ *
+ * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
+ *
+ * Copyright 2000, 2010 Oracle and/or its affiliates.
+ *
+ * OpenOffice.org - a multi-platform office productivity suite
+ *
+ * This file is part of OpenOffice.org.
+ *
+ * OpenOffice.org is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License version 3
+ * only, as published by the Free Software Foundation.
+ *
+ * OpenOffice.org is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU Lesser General Public License version 3 for more details
+ * (a copy is included in the LICENSE file that accompanied this code).
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * version 3 along with OpenOffice.org. If not, see
+ * <http://www.openoffice.org/license.html>
+ * for a copy of the LGPLv3 License.
+ *
+ ************************************************************************/
+
+// MARKER(update_precomp.py): autogen include statement, do not remove
+#include "precompiled_sw.hxx"
+#include "pam.hxx"
+#include "swtable.hxx"
+#include "frame.hxx"
+#include "pagefrm.hxx"
+#include "flyfrm.hxx"
+#include "viewsh.hxx"
+#include "doc.hxx"
+#include "viewimp.hxx"
+#include "dflyobj.hxx"
+#include "frmtool.hxx"
+#include "dcontact.hxx"
+#include <editeng/brkitem.hxx>
+#include <editeng/keepitem.hxx>
+#include <fmtsrnd.hxx>
+#include <fmtanchr.hxx>
+#include <fmtpdsc.hxx>
+#include <editeng/ulspitem.hxx>
+#include <tgrditem.hxx>
+#include <txtftn.hxx>
+#include <fmtftn.hxx>
+#include <editeng/pgrditem.hxx>
+#include <paratr.hxx>
+
+#include "ftnfrm.hxx"
+#include "txtfrm.hxx"
+#include "tabfrm.hxx"
+#include "pagedesc.hxx"
+#include "layact.hxx"
+#include "fmtornt.hxx"
+#include "flyfrms.hxx"
+#include "sectfrm.hxx"
+#include "section.hxx"
+#include "dbg_lay.hxx"
+#include "lineinfo.hxx"
+// OD 2004-03-02 #106629#
+#include <fmtclbl.hxx>
+// --> OD 2004-06-23 #i28701#
+#include <sortedobjs.hxx>
+#include <layouter.hxx>
+// <--
+// --> OD 2004-10-15 #i26945#
+#include <fmtfollowtextflow.hxx>
+// <--
+
+BOOL SwFlowFrm::bMoveBwdJump = FALSE;
+
+
+/*************************************************************************
+|*
+|* SwFlowFrm::SwFlowFrm()
+|*
+|* Ersterstellung MA 26. Apr. 95
+|* Letzte Aenderung MA 26. Apr. 95
+|*
+|*************************************************************************/
+
+
+SwFlowFrm::SwFlowFrm( SwFrm &rFrm ) :
+ rThis( rFrm ),
+ pFollow( 0 )
+{
+ bLockJoin = bIsFollow = bCntntLock = bOwnFtnNum =
+ bFtnLock = bFlyLock = FALSE;
+}
+
+
+/*************************************************************************
+|*
+|* SwFlowFrm::IsFollowLocked()
+|* return TRUE if any follow has the JoinLocked flag
+|*
+|*************************************************************************/
+
+sal_Bool SwFlowFrm::HasLockedFollow() const
+{
+ const SwFlowFrm* pFrm = GetFollow();
+ while( pFrm )
+ {
+ if( pFrm->IsJoinLocked() )
+ return sal_True;
+ pFrm = pFrm->GetFollow();
+ }
+ return sal_False;
+}
+
+/*************************************************************************
+|*
+|* SwFlowFrm::IsKeepFwdMoveAllowed()
+|*
+|* Ersterstellung MA 20. Jul. 94
+|* Letzte Aenderung MA 16. May. 95
+|*
+|*************************************************************************/
+
+
+BOOL SwFlowFrm::IsKeepFwdMoveAllowed()
+{
+ //Wenn der Vorgaenger das KeepAttribut traegt und auch dessen
+ //Vorgaenger usw. bis zum ersten der Kette und fuer diesen das
+ //IsFwdMoveAllowed ein FALSE liefert, so ist das Moven eben nicht erlaubt.
+ SwFrm *pFrm = &rThis;
+ if ( !pFrm->IsInFtn() )
+ do
+ { if ( pFrm->GetAttrSet()->GetKeep().GetValue() )
+ pFrm = pFrm->GetIndPrev();
+ else
+ return TRUE;
+ } while ( pFrm );
+
+ //Siehe IsFwdMoveAllowed()
+ BOOL bRet = FALSE;
+ if ( pFrm && pFrm->GetIndPrev() )
+ bRet = TRUE;
+ return bRet;
+}
+
+/*************************************************************************
+|*
+|* SwFlowFrm::CheckKeep()
+|*
+|* Beschreibung
+|* Ersterstellung MA 20. Jun. 95
+|* Letzte Aenderung MA 09. Apr. 97
+|*
+|*************************************************************************/
+
+
+void SwFlowFrm::CheckKeep()
+{
+ //Den 'letzten' Vorgaenger mit KeepAttribut anstossen, denn
+ //die ganze Truppe koennte zuruckrutschen.
+ SwFrm *pPre = rThis.GetIndPrev();
+ if( pPre->IsSctFrm() )
+ {
+ SwFrm *pLast = ((SwSectionFrm*)pPre)->FindLastCntnt();
+ if( pLast && pLast->FindSctFrm() == pPre )
+ pPre = pLast;
+ else
+ return;
+ }
+ SwFrm* pTmp;
+ BOOL bKeep;
+ while ( TRUE == (bKeep = pPre->GetAttrSet()->GetKeep().GetValue()) &&
+ 0 != ( pTmp = pPre->GetIndPrev() ) )
+ {
+ if( pTmp->IsSctFrm() )
+ {
+ SwFrm *pLast = ((SwSectionFrm*)pTmp)->FindLastCntnt();
+ if( pLast && pLast->FindSctFrm() == pTmp )
+ pTmp = pLast;
+ else
+ break;
+ }
+ pPre = pTmp;
+ }
+ if ( bKeep )
+ pPre->InvalidatePos();
+}
+
+/*************************************************************************
+|*
+|* SwFlowFrm::IsKeep()
+|*
+|* Ersterstellung MA 09. Apr. 97
+|* Letzte Aenderung MA 09. Apr. 97
+|*
+|*************************************************************************/
+
+BOOL SwFlowFrm::IsKeep( const SwAttrSet& rAttrs, bool bCheckIfLastRowShouldKeep ) const
+{
+ // 1. The keep attribute is ignored inside footnotes
+ // 2. For compatibility reasons, the keep attribute is
+ // ignored for frames inside table cells
+ // 3. If bBreakCheck is set to true, this function only checks
+ // if there are any break after attributes set at rAttrs
+ // or break before attributes set for the next content (or next table)
+ BOOL bKeep = bCheckIfLastRowShouldKeep ||
+ ( !rThis.IsInFtn() &&
+ ( !rThis.IsInTab() || rThis.IsTabFrm() ) &&
+ rAttrs.GetKeep().GetValue() );
+
+ ASSERT( !bCheckIfLastRowShouldKeep || rThis.IsTabFrm(),
+ "IsKeep with bCheckIfLastRowShouldKeep should only be used for tabfrms" )
+
+ // Ignore keep attribute if there are break situations:
+ if ( bKeep )
+ {
+ switch ( rAttrs.GetBreak().GetBreak() )
+ {
+ case SVX_BREAK_COLUMN_AFTER:
+ case SVX_BREAK_COLUMN_BOTH:
+ case SVX_BREAK_PAGE_AFTER:
+ case SVX_BREAK_PAGE_BOTH:
+ {
+ bKeep = FALSE;
+ }
+ default: break;
+ }
+ if ( bKeep )
+ {
+ SwFrm *pNxt;
+ if( 0 != (pNxt = rThis.FindNextCnt()) &&
+ (!pFollow || pNxt != pFollow->GetFrm()))
+ {
+ // --> FME 2006-05-15 #135914#
+ // The last row of a table only keeps with the next content
+ // it they are in the same section:
+ if ( bCheckIfLastRowShouldKeep )
+ {
+ const SwSection* pThisSection = 0;
+ const SwSection* pNextSection = 0;
+ const SwSectionFrm* pThisSectionFrm = rThis.FindSctFrm();
+ const SwSectionFrm* pNextSectionFrm = pNxt->FindSctFrm();
+
+ if ( pThisSectionFrm )
+ pThisSection = pThisSectionFrm->GetSection();
+
+ if ( pNextSectionFrm )
+ pNextSection = pNextSectionFrm->GetSection();
+
+ if ( pThisSection != pNextSection )
+ bKeep = FALSE;
+ }
+ // <--
+
+ if ( bKeep )
+ {
+ const SwAttrSet* pSet = NULL;
+
+ if ( pNxt->IsInTab() )
+ {
+ SwTabFrm* pTab = pNxt->FindTabFrm();
+ if ( ! rThis.IsInTab() || rThis.FindTabFrm() != pTab )
+ pSet = &pTab->GetFmt()->GetAttrSet();
+ }
+
+ if ( ! pSet )
+ pSet = pNxt->GetAttrSet();
+
+ ASSERT( pSet, "No AttrSet to check keep attribute" )
+
+ if ( pSet->GetPageDesc().GetPageDesc() )
+ bKeep = FALSE;
+ else switch ( pSet->GetBreak().GetBreak() )
+ {
+ case SVX_BREAK_COLUMN_BEFORE:
+ case SVX_BREAK_COLUMN_BOTH:
+ case SVX_BREAK_PAGE_BEFORE:
+ case SVX_BREAK_PAGE_BOTH:
+ bKeep = FALSE;
+ default: break;
+ }
+ }
+ }
+ }
+ }
+ return bKeep;
+}
+
+/*************************************************************************
+|*
+|* SwFlowFrm::BwdMoveNecessary()
+|*
+|* Ersterstellung MA 20. Jul. 94
+|* Letzte Aenderung MA 02. May. 96
+|*
+|*************************************************************************/
+
+
+BYTE SwFlowFrm::BwdMoveNecessary( const SwPageFrm *pPage, const SwRect &rRect )
+{
+ // Der return-Wert entscheidet mit,
+ // ob auf Zurueckgeflossen werden muss, (3)
+ // ob das gute alte WouldFit gerufen werden kann (0, 1)
+ // oder ob ein Umhaengen und eine Probeformatierung sinnvoll ist (2)
+ // dabei bedeutet Bit 1, dass Objekte an mir selbst verankert sind
+ // und Bit 2, dass ich anderen Objekten ausweichen muss.
+
+ //Wenn ein SurroundObj, dass einen Umfluss wuenscht mit dem Rect ueberlappt
+ //ist der Fluss notwendig (weil die Verhaeltnisse nicht geschaetzt werden
+ //koennen), es kann allerdings ggf. eine TestFormatierung stattfinden.
+ //Wenn das SurroundObj ein Fly ist und ich selbst ein Lower bin oder der Fly
+ //Lower von mir ist, so spielt er keine Rolle.
+ //Wenn das SurroundObj in einem zeichengebunden Fly verankert ist, und ich
+ //selbst nicht Lower dieses Zeichengebundenen Flys bin, so spielt der Fly
+ //keine Rolle.
+ //#32639# Wenn das Objekt bei mir verankert ist kann ich es
+ //vernachlaessigen, weil es hoechstwahrscheinlich (!?) mitfliesst,
+ //eine TestFormatierung ist dann allerdings nicht erlaubt!
+ BYTE nRet = 0;
+ SwFlowFrm *pTmp = this;
+ do
+ { // Wenn an uns oder einem Follow Objekte haengen, so
+ // kann keine ProbeFormatierung stattfinden, da absatzgebundene
+ // nicht richtig beruecksichtigt wuerden und zeichengebundene sollten
+ // gar nicht zur Probe formatiert werden.
+ if( pTmp->GetFrm()->GetDrawObjs() )
+ nRet = 1;
+ pTmp = pTmp->GetFollow();
+ } while ( !nRet && pTmp );
+ if ( pPage->GetSortedObjs() )
+ {
+ // --> OD 2004-07-01 #i28701# - new type <SwSortedObjs>
+ const SwSortedObjs &rObjs = *pPage->GetSortedObjs();
+ ULONG nIndex = ULONG_MAX;
+ for ( USHORT i = 0; nRet < 3 && i < rObjs.Count(); ++i )
+ {
+ // --> OD 2004-07-01 #i28701# - consider changed type of
+ // <SwSortedObjs> entries.
+ SwAnchoredObject* pObj = rObjs[i];
+ const SwFrmFmt& rFmt = pObj->GetFrmFmt();
+ const SwRect aRect( pObj->GetObjRect() );
+ if ( aRect.IsOver( rRect ) &&
+ rFmt.GetSurround().GetSurround() != SURROUND_THROUGHT )
+ {
+ if( rThis.IsLayoutFrm() && //Fly Lower von This?
+ Is_Lower_Of( &rThis, pObj->GetDrawObj() ) )
+ continue;
+ if( pObj->ISA(SwFlyFrm) )
+ {
+ const SwFlyFrm *pFly = static_cast<const SwFlyFrm*>(pObj);
+ if ( pFly->IsAnLower( &rThis ) )//This Lower vom Fly?
+ continue;
+ }
+
+ const SwFrm* pAnchor = pObj->GetAnchorFrm();
+ if ( pAnchor == &rThis )
+ {
+ nRet |= 1;
+ continue;
+ }
+
+ //Nicht wenn das Objekt im Textfluss hinter mir verankert ist,
+ //denn dann weiche ich ihm nicht aus.
+ if ( ::IsFrmInSameKontext( pAnchor, &rThis ) )
+ {
+ if ( rFmt.GetAnchor().GetAnchorId() == FLY_AT_PARA )
+ {
+ // Den Index des anderen erhalten wir immer ueber das Ankerattr.
+ ULONG nTmpIndex = rFmt.GetAnchor().GetCntntAnchor()->nNode.GetIndex();
+ // Jetzt wird noch ueberprueft, ob der aktuelle Absatz vor dem
+ // Anker des verdraengenden Objekts im Text steht, dann wird
+ // nicht ausgewichen.
+ // Der Index wird moeglichst ueber einen SwFmtAnchor ermittelt,
+ // da sonst recht teuer.
+ if( ULONG_MAX == nIndex )
+ {
+ const SwNode *pNode;
+ if ( rThis.IsCntntFrm() )
+ pNode = ((SwCntntFrm&)rThis).GetNode();
+ else if( rThis.IsSctFrm() )
+ pNode = ((SwSectionFmt*)((SwSectionFrm&)rThis).
+ GetFmt())->GetSectionNode();
+ else
+ {
+ ASSERT( rThis.IsTabFrm(), "new FowFrm?" );
+ pNode = ((SwTabFrm&)rThis).GetTable()->
+ GetTabSortBoxes()[0]->GetSttNd()->FindTableNode();
+ }
+ nIndex = pNode->GetIndex();
+ }
+ if( nIndex < nTmpIndex )
+ continue;
+ }
+ }
+ else
+ continue;
+
+ nRet |= 2;
+ }
+ }
+ }
+ return nRet;
+}
+
+/*************************************************************************
+|*
+|* SwFlowFrm::CutTree(), PasteTree(), MoveSubTree()
+|*
+|* Beschreibung Eine Spezialisierte Form des Cut() und Paste(), die
+|* eine ganze Kette umhaengt (naehmlich this und folgende). Dabei werden
+|* nur minimale Operationen und Benachrichtigungen ausgefuehrt.
+|* Ersterstellung MA 18. Mar. 93
+|* Letzte Aenderung MA 18. May. 95
+|*
+|*************************************************************************/
+
+
+SwLayoutFrm *SwFlowFrm::CutTree( SwFrm *pStart )
+{
+ //Der Start und alle Nachbarn werden ausgeschnitten, sie werden aneinander-
+ //gereiht und ein Henkel auf den ersten wird zurueckgeliefert.
+ //Zurueckbleibende werden geeignet invalidiert.
+
+ SwLayoutFrm *pLay = pStart->GetUpper();
+ if ( pLay->IsInFtn() )
+ pLay = pLay->FindFtnFrm();
+
+ // --> OD 2006-05-08 #i58846#
+ // <pPrepare( PREP_QUOVADIS )> only for frames in footnotes
+ if( pStart->IsInFtn() )
+ {
+ SwFrm* pTmp = pStart->GetIndPrev();
+ if( pTmp )
+ pTmp->Prepare( PREP_QUOVADIS );
+ }
+ // <--
+
+ //Nur fix auschneiden und zwar so, dass klare Verhaeltnisse bei den
+ //Verlassenen herrschen. Die Pointer der ausgeschnittenen Kette zeigen
+ //noch wer weiss wo hin.
+ if ( pStart == pStart->GetUpper()->Lower() )
+ pStart->GetUpper()->pLower = 0;
+ if ( pStart->GetPrev() )
+ {
+ pStart->GetPrev()->pNext = 0;
+ pStart->pPrev = 0;
+ }
+
+ if ( pLay->IsFtnFrm() )
+ {
+ if ( !pLay->Lower() && !pLay->IsColLocked() &&
+ !((SwFtnFrm*)pLay)->IsBackMoveLocked() )
+ {
+ pLay->Cut();
+ delete pLay;
+ }
+ else
+ {
+ BOOL bUnlock = !((SwFtnFrm*)pLay)->IsBackMoveLocked();
+ ((SwFtnFrm*)pLay)->LockBackMove();
+ pLay->InvalidateSize();
+ pLay->Calc();
+ SwCntntFrm *pCnt = pLay->ContainsCntnt();
+ while ( pCnt && pLay->IsAnLower( pCnt ) )
+ {
+ //Kann sein, dass der CntFrm gelockt ist, wir wollen hier nicht
+ //in eine endlose Seitenwanderung hineinlaufen und rufen das
+ //Calc garnicht erst!
+ ASSERT( pCnt->IsTxtFrm(), "Die Graphic ist gelandet." );
+ if ( ((SwTxtFrm*)pCnt)->IsLocked() ||
+ ((SwTxtFrm*)pCnt)->GetFollow() == pStart )
+ break;
+ pCnt->Calc();
+ pCnt = pCnt->GetNextCntntFrm();
+ }
+ if( bUnlock )
+ ((SwFtnFrm*)pLay)->UnlockBackMove();
+ }
+ pLay = 0;
+ }
+ return pLay;
+}
+
+
+
+BOOL SwFlowFrm::PasteTree( SwFrm *pStart, SwLayoutFrm *pParent, SwFrm *pSibling,
+ SwFrm *pOldParent )
+{
+ //returnt TRUE wenn in der Kette ein LayoutFrm steht.
+ BOOL bRet = FALSE;
+
+ //Die mit pStart beginnende Kette wird vor den Sibling unter den Parent
+ //gehaengt. Fuer geeignete Invalidierung wird ebenfalls gesorgt.
+
+ //Ich bekomme eine fertige Kette. Der Anfang der Kette muss verpointert
+ //werden, dann alle Upper fuer die Kette und schliesslich dass Ende.
+ //Unterwegs werden alle geeignet invalidiert.
+ if ( pSibling )
+ {
+ if ( 0 != (pStart->pPrev = pSibling->GetPrev()) )
+ pStart->GetPrev()->pNext = pStart;
+ else
+ pParent->pLower = pStart;
+ pSibling->_InvalidatePos();
+ pSibling->_InvalidatePrt();
+ }
+ else
+ {
+ if ( 0 == (pStart->pPrev = pParent->Lower()) )
+ pParent->pLower = pStart;
+ else
+ //Modified for #i100782#,04/03/2009
+ //If the pParent has more than 1 child nodes, former design will
+ //ignore them directly without any collection work. It will make some
+ //dangling pointers. This lead the crash...
+ //The new design will find the last child of pParent in loop way, and
+ //add the pStart after the last child.
+ // pParent->Lower()->pNext = pStart;
+ {
+ SwFrm* pTemp = pParent->pLower;
+ while (pTemp)
+ {
+ if (pTemp->pNext)
+ pTemp = pTemp->pNext;
+ else
+ {
+ pStart->pPrev = pTemp;
+ pTemp->pNext = pStart;
+ break;
+ }
+ }
+ }
+ //End modification for #i100782#,04/03/2009
+
+ // #i27145#
+ if ( pParent->IsSctFrm() )
+ {
+ // We have no sibling because pParent is a section frame and
+ // has just been created to contain some content. The printing
+ // area of the frame behind pParent has to be invalidated, so
+ // that the correct distance between pParent and the next frame
+ // can be calculated.
+ pParent->InvalidateNextPrtArea();
+ }
+ }
+ SwFrm *pFloat = pStart;
+ SwFrm *pLst = 0;
+ SWRECTFN( pParent )
+ SwTwips nGrowVal = 0;
+ do
+ { pFloat->pUpper = pParent;
+ pFloat->_InvalidateAll();
+ pFloat->CheckDirChange();
+
+ //Ich bin Freund des TxtFrm und darf deshalb so einiges. Das mit
+ //dem CacheIdx scheint etwas riskant!
+ if ( pFloat->IsTxtFrm() )
+ {
+ if ( ((SwTxtFrm*)pFloat)->GetCacheIdx() != USHRT_MAX )
+ ((SwTxtFrm*)pFloat)->Init(); //Ich bin sein Freund.
+ }
+ else
+ bRet = TRUE;
+
+ nGrowVal += (pFloat->Frm().*fnRect->fnGetHeight)();
+ if ( pFloat->GetNext() )
+ pFloat = pFloat->GetNext();
+ else
+ {
+ pLst = pFloat;
+ pFloat = 0;
+ }
+ } while ( pFloat );
+
+ if ( pSibling )
+ {
+ pLst->pNext = pSibling;
+ pSibling->pPrev = pLst;
+ if( pSibling->IsInFtn() )
+ {
+ if( pSibling->IsSctFrm() )
+ pSibling = ((SwSectionFrm*)pSibling)->ContainsAny();
+ if( pSibling )
+ pSibling->Prepare( PREP_ERGOSUM );
+ }
+ }
+ if ( nGrowVal )
+ {
+ if ( pOldParent && pOldParent->IsBodyFrm() ) //Fuer variable Seitenhoehe beim Browsen
+ pOldParent->Shrink( nGrowVal );
+ pParent->Grow( nGrowVal );
+ }
+
+ if ( pParent->IsFtnFrm() )
+ ((SwFtnFrm*)pParent)->InvalidateNxtFtnCnts( pParent->FindPageFrm() );
+ return bRet;
+}
+
+
+
+void SwFlowFrm::MoveSubTree( SwLayoutFrm* pParent, SwFrm* pSibling )
+{
+ ASSERT( pParent, "Kein Parent uebergeben." );
+ ASSERT( rThis.GetUpper(), "Wo kommen wir denn her?" );
+
+ //Sparsamer benachrichtigen wenn eine Action laeuft.
+ ViewShell *pSh = rThis.GetShell();
+ const SwViewImp *pImp = pSh ? pSh->Imp() : 0;
+ const BOOL bComplete = pImp && pImp->IsAction() && pImp->GetLayAction().IsComplete();
+
+ if ( !bComplete )
+ {
+ SwFrm *pPre = rThis.GetIndPrev();
+ if ( pPre )
+ {
+ pPre->SetRetouche();
+ // --> OD 2004-11-23 #115759# - follow-up of #i26250#
+ // invalidate printing area of previous frame, if it's in a table
+ if ( pPre->GetUpper()->IsInTab() )
+ {
+ pPre->_InvalidatePrt();
+ }
+ // <--
+ pPre->InvalidatePage();
+ }
+ else
+ { rThis.GetUpper()->SetCompletePaint();
+ rThis.GetUpper()->InvalidatePage();
+ }
+ }
+
+ SwPageFrm *pOldPage = rThis.FindPageFrm();
+
+ SwLayoutFrm *pOldParent = CutTree( &rThis );
+ const BOOL bInvaLay = PasteTree( &rThis, pParent, pSibling, pOldParent );
+
+ // Wenn durch das Cut&Paste ein leerer SectionFrm entstanden ist, sollte
+ // dieser automatisch verschwinden.
+ SwSectionFrm *pSct;
+ // --> OD 2006-01-04 #126020# - adjust check for empty section
+ // --> OD 2006-02-01 #130797# - correct fix #126020#
+ if ( pOldParent && !pOldParent->Lower() &&
+ ( pOldParent->IsInSct() &&
+ !(pSct = pOldParent->FindSctFrm())->ContainsCntnt() &&
+ !pSct->ContainsAny( true ) ) )
+ // <--
+ {
+ pSct->DelEmpty( FALSE );
+ }
+
+ // In einem spaltigen Bereich rufen wir lieber kein Calc "von unten"
+ if( !rThis.IsInSct() &&
+ ( !rThis.IsInTab() || ( rThis.IsTabFrm() && !rThis.GetUpper()->IsInTab() ) ) )
+ rThis.GetUpper()->Calc();
+ else if( rThis.GetUpper()->IsSctFrm() )
+ {
+ SwSectionFrm* pTmpSct = (SwSectionFrm*)rThis.GetUpper();
+ BOOL bOld = pTmpSct->IsCntntLocked();
+ pTmpSct->SetCntntLock( TRUE );
+ pTmpSct->Calc();
+ if( !bOld )
+ pTmpSct->SetCntntLock( FALSE );
+ }
+ SwPageFrm *pPage = rThis.FindPageFrm();
+
+ if ( pOldPage != pPage )
+ {
+ rThis.InvalidatePage( pPage );
+ if ( rThis.IsLayoutFrm() )
+ {
+ SwCntntFrm *pCnt = ((SwLayoutFrm*)&rThis)->ContainsCntnt();
+ if ( pCnt )
+ pCnt->InvalidatePage( pPage );
+ }
+ else if ( pSh && pSh->GetDoc()->GetLineNumberInfo().IsRestartEachPage()
+ && pPage->FindFirstBodyCntnt() == &rThis )
+ {
+ rThis._InvalidateLineNum();
+ }
+ }
+ if ( bInvaLay || (pSibling && pSibling->IsLayoutFrm()) )
+ rThis.GetUpper()->InvalidatePage( pPage );
+}
+
+/*************************************************************************
+|*
+|* SwFlowFrm::IsAnFollow()
+|*
+|* Ersterstellung MA 26. Apr. 95
+|* Letzte Aenderung MA 26. Apr. 95
+|*
+|*************************************************************************/
+
+
+BOOL SwFlowFrm::IsAnFollow( const SwFlowFrm *pAssumed ) const
+{
+ const SwFlowFrm *pFoll = this;
+ do
+ { if ( pAssumed == pFoll )
+ return TRUE;
+ pFoll = pFoll->GetFollow();
+ } while ( pFoll );
+ return FALSE;
+}
+
+
+/*************************************************************************
+|*
+|* SwFlowFrm::FindMaster()
+|*
+|* Ersterstellung MA 26. Apr. 95
+|* Letzte Aenderung MA 26. Apr. 95
+|*
+|*************************************************************************/
+
+SwTxtFrm* SwCntntFrm::FindMaster() const
+{
+ ASSERT( IsFollow(), "SwCntntFrm::FindMaster(): !IsFollow" );
+
+ const SwCntntFrm* pCnt = GetPrevCntntFrm();
+
+ while ( pCnt )
+ {
+ if ( pCnt->HasFollow() && pCnt->GetFollow() == this )
+ {
+ ASSERT( pCnt->IsTxtFrm(), "NoTxtFrm with follow found" )
+ return (SwTxtFrm*)pCnt;
+ }
+ pCnt = pCnt->GetPrevCntntFrm();
+ }
+
+ ASSERT( FALSE, "Follow ist lost in Space." );
+ return 0;
+}
+
+SwSectionFrm* SwSectionFrm::FindMaster() const
+{
+ ASSERT( IsFollow(), "SwSectionFrm::FindMaster(): !IsFollow" );
+
+ SwClientIter aIter( *pSection->GetFmt() );
+ SwClient *pLast = aIter.GoStart();
+
+ while ( pLast )
+ {
+ if ( pLast->ISA( SwFrm ) )
+ {
+ ASSERT( ((SwFrm*)pLast)->IsSctFrm(),
+ "Non-section frame registered in section format" )
+ SwSectionFrm* pSect = (SwSectionFrm*)pLast;
+ if( pSect->GetFollow() == this )
+ return pSect;
+ }
+ pLast = aIter++;
+ }
+
+ ASSERT( FALSE, "Follow ist lost in Space." );
+ return 0;
+}
+
+SwTabFrm* SwTabFrm::FindMaster( bool bFirstMaster ) const
+{
+ ASSERT( IsFollow(), "SwTabFrm::FindMaster(): !IsFollow" );
+
+ SwClientIter aIter( *GetTable()->GetFrmFmt() );
+ SwClient* pLast = aIter.GoStart();
+
+ while ( pLast )
+ {
+ if ( pLast->ISA( SwFrm ) )
+ {
+ ASSERT( ((SwFrm*)pLast)->IsTabFrm(),
+ "Non-table frame registered in table format" )
+ SwTabFrm* pTab = (SwTabFrm*)pLast;
+
+ if ( bFirstMaster )
+ {
+ //
+ // Optimization. This makes code like this obsolete:
+ // while ( pTab->IsFollow() )
+ // pTab = pTab->FindMaster();
+ //
+ if ( !pTab->IsFollow() )
+ {
+ SwTabFrm* pNxt = pTab;
+ while ( pNxt )
+ {
+ if ( pNxt->GetFollow() == this )
+ return pTab;
+ pNxt = pNxt->GetFollow();
+ }
+ }
+ }
+ else
+ {
+ if ( pTab->GetFollow() == this )
+ return pTab;
+ }
+ }
+ pLast = aIter++;
+ }
+
+ ASSERT( FALSE, "Follow ist lost in Space." );
+ return 0;
+}
+
+/*************************************************************************
+|*
+|* SwFrm::GetLeaf()
+|*
+|* Beschreibung Liefert das naechste/vorhergehende LayoutBlatt,
+|* das _nicht_ unterhalb von this liegt (oder gar this selbst ist).
+|* Ausserdem muss dieses LayoutBlatt im gleichen Textfluss wie
+|* pAnch Ausgangsfrm liegen (Body, Ftn)
+|* Ersterstellung MA 25. Nov. 92
+|* Letzte Aenderung MA 25. Apr. 95
+|*
+|*************************************************************************/
+
+
+const SwLayoutFrm *SwFrm::GetLeaf( MakePageType eMakePage, BOOL bFwd,
+ const SwFrm *pAnch ) const
+{
+ //Ohne Fluss kein genuss...
+ if ( !(IsInDocBody() || IsInFtn() || IsInFly()) )
+ return 0;
+
+ const SwFrm *pLeaf = this;
+ BOOL bFound = FALSE;
+
+ do
+ { pLeaf = ((SwFrm*)pLeaf)->GetLeaf( eMakePage, bFwd );
+
+ if ( pLeaf &&
+ (!IsLayoutFrm() || !((SwLayoutFrm*)this)->IsAnLower( pLeaf )))
+ {
+ if ( pAnch->IsInDocBody() == pLeaf->IsInDocBody() &&
+ pAnch->IsInFtn() == pLeaf->IsInFtn() )
+ {
+ bFound = TRUE;
+ }
+ }
+ } while ( !bFound && pLeaf );
+
+ return (const SwLayoutFrm*)pLeaf;
+}
+
+/*************************************************************************
+|*
+|* SwFrm::GetLeaf()
+|*
+|* Beschreibung Ruft Get[Next|Prev]Leaf
+|*
+|* Ersterstellung MA 20. Mar. 93
+|* Letzte Aenderung MA 25. Apr. 95
+|*
+|*************************************************************************/
+
+
+SwLayoutFrm *SwFrm::GetLeaf( MakePageType eMakePage, BOOL bFwd )
+{
+ if ( IsInFtn() )
+ return bFwd ? GetNextFtnLeaf( eMakePage ) : GetPrevFtnLeaf( eMakePage );
+
+ // --> OD 2005-08-16 #i53323#
+ // A frame could be inside a table AND inside a section.
+ // Thus, it has to be determined, which is the first parent.
+ bool bInTab( IsInTab() );
+ bool bInSct( IsInSct() );
+ if ( bInTab && bInSct )
+ {
+ const SwFrm* pUpperFrm( GetUpper() );
+ while ( pUpperFrm )
+ {
+ if ( pUpperFrm->IsTabFrm() )
+ {
+ // the table is the first.
+ bInSct = false;
+ break;
+ }
+ else if ( pUpperFrm->IsSctFrm() )
+ {
+ // the section is the first.
+ bInTab = false;
+ break;
+ }
+
+ pUpperFrm = pUpperFrm->GetUpper();
+ }
+ }
+
+ if ( bInTab && ( !IsTabFrm() || GetUpper()->IsCellFrm() ) ) // TABLE IN TABLE
+ return bFwd ? GetNextCellLeaf( eMakePage ) : GetPrevCellLeaf( eMakePage );
+
+ if ( bInSct )
+ return bFwd ? GetNextSctLeaf( eMakePage ) : GetPrevSctLeaf( eMakePage );
+ // <--
+
+ return bFwd ? GetNextLeaf( eMakePage ) : GetPrevLeaf( eMakePage );
+}
+
+
+
+BOOL SwFrm::WrongPageDesc( SwPageFrm* pNew )
+{
+ //Jetzt wirds leider etwas kompliziert:
+ //Ich bringe ich evtl. selbst
+ //einen Pagedesc mit; der der Folgeseite muss dann damit
+ //uebereinstimmen.
+ //Anderfalls muss ich mir etwas genauer ansehen wo der
+ //Folgepagedesc herkam.
+ //Wenn die Folgeseite selbst schon sagt, dass ihr
+ //Pagedesc nicht stimmt so kann ich das Teil bedenkenlos
+ //auswechseln.
+ //Wenn die Seite meint, dass ihr Pagedesc stimmt, so heisst
+ //das leider noch immer nicht, dass ich damit etwas anfangen
+ //kann: Wenn der erste BodyCntnt einen PageDesc oder einen
+ //PageBreak wuenscht, so muss ich ebenfalls eine neue
+ //Seite einfuegen; es sein denn die gewuenschte Seite ist
+ //die richtige.
+ //Wenn ich einen neue Seite eingefuegt habe, so fangen die
+ //Probleme leider erst an, denn wahrscheinlich wird die dann
+ //folgende Seite verkehrt gewesen und ausgewechselt worden
+ //sein. Das hat zur Folge, dass ich zwar eine neue (und
+ //jetzt richtige) Seite habe, die Bedingungen zum auswechseln
+ //aber leider noch immer stimmen.
+ //Ausweg: Vorlaeufiger Versuch, nur einmal eine neue Seite
+ //einsetzen (Leerseiten werden noetigenfalls bereits von
+ //InsertPage() eingefuegt.
+ const SwFmtPageDesc &rFmtDesc = GetAttrSet()->GetPageDesc();
+
+ //Mein Pagedesc zaehlt nicht, wenn ich ein Follow bin!
+ SwPageDesc *pDesc = 0;
+ USHORT nTmp = 0;
+ SwFlowFrm *pFlow = SwFlowFrm::CastFlowFrm( this );
+ if ( !pFlow || !pFlow->IsFollow() )
+ {
+ pDesc = (SwPageDesc*)rFmtDesc.GetPageDesc();
+ if( pDesc )
+ {
+ if( !pDesc->GetRightFmt() )
+ nTmp = 2;
+ else if( !pDesc->GetLeftFmt() )
+ nTmp = 1;
+ else if( rFmtDesc.GetNumOffset() )
+ nTmp = rFmtDesc.GetNumOffset();
+ }
+ }
+
+ //Bringt der Cntnt einen Pagedesc mit oder muss zaehlt die
+ //virtuelle Seitennummer des neuen Layoutleafs?
+ // Bei Follows zaehlt der PageDesc nicht
+ const BOOL bOdd = nTmp ? ( nTmp % 2 ? TRUE : FALSE )
+ : pNew->OnRightPage();
+ if ( !pDesc )
+ pDesc = pNew->FindPageDesc();
+ const SwFlowFrm *pNewFlow = pNew->FindFirstBodyCntnt();
+ // Haben wir uns selbst gefunden?
+ if( pNewFlow == pFlow )
+ pNewFlow = NULL;
+ if ( pNewFlow && pNewFlow->GetFrm()->IsInTab() )
+ pNewFlow = pNewFlow->GetFrm()->FindTabFrm();
+ const SwPageDesc *pNewDesc= ( pNewFlow && !pNewFlow->IsFollow() )
+ ? pNewFlow->GetFrm()->GetAttrSet()->GetPageDesc().GetPageDesc():0;
+
+ return ( pNew->GetPageDesc() != pDesc || // own desc ?
+ pNew->GetFmt() != (bOdd ? pDesc->GetRightFmt() : pDesc->GetLeftFmt()) ||
+ ( pNewDesc && pNewDesc == pDesc ) );
+}
+
+
+/*************************************************************************
+|*
+|* SwFrm::GetNextLeaf()
+|*
+|* Beschreibung Liefert das naechste LayoutBlatt in den das
+|* Frame gemoved werden kann.
+|*
+|* Ersterstellung MA 16. Nov. 92
+|* Letzte Aenderung MA 05. Dec. 96
+|*
+|*************************************************************************/
+
+SwLayoutFrm *SwFrm::GetNextLeaf( MakePageType eMakePage )
+{
+ ASSERT( !IsInFtn(), "GetNextLeaf(), don't call me for Ftn." );
+ ASSERT( !IsInSct(), "GetNextLeaf(), don't call me for Sections." );
+
+ const BOOL bBody = IsInDocBody(); //Wenn ich aus dem DocBody komme
+ //Will ich auch im Body landen.
+
+ // Bei Flys macht es keinen Sinn, Seiten einzufuegen, wir wollen lediglich
+ // die Verkettung absuchen.
+ if( IsInFly() )
+ eMakePage = MAKEPAGE_NONE;
+
+ //Bei Tabellen gleich den grossen Sprung wagen, ein einfaches GetNext...
+ //wuerde die erste Zellen und in der Folge alle weiteren Zellen nacheinander
+ //abklappern....
+ SwLayoutFrm *pLayLeaf = 0;
+ if ( IsTabFrm() )
+ {
+ SwCntntFrm* pTmp = ((SwTabFrm*)this)->FindLastCntnt();
+ if ( pTmp )
+ pLayLeaf = pTmp->GetUpper();
+ }
+ if ( !pLayLeaf )
+ pLayLeaf = GetNextLayoutLeaf();
+
+ SwLayoutFrm *pOldLayLeaf = 0; //Damit bei neu erzeugten Seiten
+ //nicht wieder vom Anfang gesucht
+ //wird.
+ BOOL bNewPg = FALSE; //nur einmal eine neue Seite einfuegen.
+
+ while ( TRUE )
+ {
+ if ( pLayLeaf )
+ {
+ //Es gibt noch einen weiteren LayoutFrm, mal sehen,
+ //ob er bereit ist mich aufzunehmen.
+ //Dazu braucht er nur von der gleichen Art wie mein Ausgangspunkt
+ //sein (DocBody bzw. Footnote.)
+ if ( pLayLeaf->FindPageFrm()->IsFtnPage() )
+ { //Wenn ich bei den Endnotenseiten angelangt bin hat sichs.
+ pLayLeaf = 0;
+ continue;
+ }
+ if ( (bBody && !pLayLeaf->IsInDocBody()) || pLayLeaf->IsInTab()
+ || pLayLeaf->IsInSct() )
+ {
+ //Er will mich nicht; neuer Versuch, neues Glueck
+ pOldLayLeaf = pLayLeaf;
+ pLayLeaf = pLayLeaf->GetNextLayoutLeaf();
+ continue;
+ }
+ //Er will mich, also ist er der gesuchte und ich bin fertig.
+ //Bei einem Seitenwechsel kann es allerdings noch sein, dass
+ //Der Seitentyp nicht der gewuenschte ist, in diesem Fall muessen
+ //wir eine Seite des richtigen Typs einfuegen.
+
+ if( !IsFlowFrm() && ( eMakePage == MAKEPAGE_NONE ||
+ eMakePage==MAKEPAGE_APPEND || eMakePage==MAKEPAGE_NOSECTION ) )
+ return pLayLeaf;
+
+ SwPageFrm *pNew = pLayLeaf->FindPageFrm();
+ // #111704# The pagedesc check does not make sense for frames in fly frames
+ if ( pNew != FindPageFrm() && !bNewPg && !IsInFly() &&
+ // --> FME 2005-05-10 #i46683#
+ // Do not consider page descriptions in browse mode (since
+ // MoveBwd ignored them)
+ !pNew->GetFmt()->getIDocumentSettingAccess()->get(IDocumentSettingAccess::BROWSE_MODE) )
+ // <--
+ {
+ if( WrongPageDesc( pNew ) )
+ {
+ SwFtnContFrm *pCont = pNew->FindFtnCont();
+ if( pCont )
+ {
+ // Falls die Referenz der ersten Fussnote dieser Seite
+ // vor der Seite liegt, fuegen wir lieber keine neue Seite
+ // ein (Bug #55620#)
+ SwFtnFrm *pFtn = (SwFtnFrm*)pCont->Lower();
+ if( pFtn && pFtn->GetRef() )
+ {
+ const USHORT nRefNum = pNew->GetPhyPageNum();
+ if( pFtn->GetRef()->GetPhyPageNum() < nRefNum )
+ break;
+ }
+ }
+ //Erwischt, die folgende Seite ist verkehrt, also
+ //muss eine neue eingefuegt werden.
+ if ( eMakePage == MAKEPAGE_INSERT )
+ {
+ bNewPg = TRUE;
+
+ SwPageFrm *pPg = pOldLayLeaf ?
+ pOldLayLeaf->FindPageFrm() : 0;
+ if ( pPg && pPg->IsEmptyPage() )
+ //Nicht hinter, sondern vor der EmptyPage einfuegen.
+ pPg = (SwPageFrm*)pPg->GetPrev();
+
+ if ( !pPg || pPg == pNew )
+ pPg = FindPageFrm();
+
+ InsertPage( pPg, FALSE );
+ pLayLeaf = GetNextLayoutLeaf();
+ pOldLayLeaf = 0;
+ continue;
+ }
+ else
+ pLayLeaf = 0;
+ }
+ }
+ break;
+ }
+ else
+ {
+ //Es gibt keinen passenden weiteren LayoutFrm, also muss eine
+ //neue Seite her.
+ if ( eMakePage == MAKEPAGE_APPEND || eMakePage == MAKEPAGE_INSERT )
+ {
+ InsertPage(
+ pOldLayLeaf ? pOldLayLeaf->FindPageFrm() : FindPageFrm(),
+ FALSE );
+
+ //und nochmal das ganze
+ pLayLeaf = pOldLayLeaf ? pOldLayLeaf : GetNextLayoutLeaf();
+ }
+ else
+ break;
+ }
+ }
+ return pLayLeaf;
+}
+
+/*************************************************************************
+|*
+|* SwFrm::GetPrevLeaf()
+|*
+|* Beschreibung Liefert das vorhergehende LayoutBlatt in das der
+|* Frame gemoved werden kann.
+|* Ersterstellung MA 16. Nov. 92
+|* Letzte Aenderung MA 25. Apr. 95
+|*
+|*************************************************************************/
+
+
+SwLayoutFrm *SwFrm::GetPrevLeaf( MakePageType )
+{
+ ASSERT( !IsInFtn(), "GetPrevLeaf(), don't call me for Ftn." );
+
+ const BOOL bBody = IsInDocBody(); //Wenn ich aus dem DocBody komme
+ //will ich auch im Body landen.
+ const BOOL bFly = IsInFly();
+
+ SwLayoutFrm *pLayLeaf = GetPrevLayoutLeaf();
+ SwLayoutFrm *pPrevLeaf = 0;
+
+ while ( pLayLeaf )
+ {
+ if ( pLayLeaf->IsInTab() || //In Tabellen geht's niemals hinein.
+ pLayLeaf->IsInSct() ) //In Bereiche natuerlich auch nicht!
+ pLayLeaf = pLayLeaf->GetPrevLayoutLeaf();
+ else if ( bBody && pLayLeaf->IsInDocBody() )
+ {
+ if ( pLayLeaf->Lower() )
+ break;
+ pPrevLeaf = pLayLeaf;
+ pLayLeaf = pLayLeaf->GetPrevLayoutLeaf();
+ if ( pLayLeaf )
+ SwFlowFrm::SetMoveBwdJump( TRUE );
+ }
+ else if ( bFly )
+ break; //Cntnts in Flys sollte jedes Layout-Blatt recht sein.
+ else
+ pLayLeaf = pLayLeaf->GetPrevLayoutLeaf();
+ }
+ return pLayLeaf ? pLayLeaf : pPrevLeaf;
+}
+
+/*************************************************************************
+|*
+|* SwFlowFrm::IsPrevObjMove()
+|*
+|* Ersterstellung MA 20. Feb. 96
+|* Letzte Aenderung MA 22. Feb. 96
+|*
+|*************************************************************************/
+
+
+BOOL SwFlowFrm::IsPrevObjMove() const
+{
+ //TRUE der FlowFrm soll auf einen Rahmen des Vorgaengers Ruecksicht nehmen
+ // und fuer diesen ggf. Umbrechen.
+
+ //!!!!!!!!!!!Hack!!!!!!!!!!!
+ if ( rThis.GetUpper()->GetFmt()->getIDocumentSettingAccess()->get(IDocumentSettingAccess::BROWSE_MODE) )
+ return FALSE;
+
+ SwFrm *pPre = rThis.FindPrev();
+
+ if ( pPre && pPre->GetDrawObjs() )
+ {
+ ASSERT( SwFlowFrm::CastFlowFrm( pPre ), "new flowfrm?" );
+ if( SwFlowFrm::CastFlowFrm( pPre )->IsAnFollow( this ) )
+ return FALSE;
+ SwLayoutFrm* pPreUp = pPre->GetUpper();
+ // Wenn der Upper ein SectionFrm oder die Spalte eines SectionFrms ist,
+ // duerfen wir aus diesem durchaus heraushaengen,
+ // es muss stattdessen der Upper des SectionFrms beruecksichtigt werden.
+ if( pPreUp->IsInSct() )
+ {
+ if( pPreUp->IsSctFrm() )
+ pPreUp = pPreUp->GetUpper();
+ else if( pPreUp->IsColBodyFrm() &&
+ pPreUp->GetUpper()->GetUpper()->IsSctFrm() )
+ pPreUp = pPreUp->GetUpper()->GetUpper()->GetUpper();
+ }
+ // --> OD 2004-10-15 #i26945# - re-factoring:
+ // use <GetVertPosOrientFrm()> to determine, if object has followed the
+ // text flow to the next layout frame
+ for ( USHORT i = 0; i < pPre->GetDrawObjs()->Count(); ++i )
+ {
+ // --> OD 2004-07-01 #i28701# - consider changed type of
+ // <SwSortedObjs> entries.
+ const SwAnchoredObject* pObj = (*pPre->GetDrawObjs())[i];
+ // OD 2004-01-20 #110582# - do not consider hidden objects
+ // --> OD 2004-10-15 #i26945# - do not consider object, which
+ // doesn't follow the text flow.
+ if ( pObj->GetFrmFmt().GetDoc()->IsVisibleLayerId(
+ pObj->GetDrawObj()->GetLayer() ) &&
+ pObj->GetFrmFmt().GetFollowTextFlow().GetValue() )
+ // <--
+ {
+ const SwLayoutFrm* pVertPosOrientFrm = pObj->GetVertPosOrientFrm();
+ if ( pVertPosOrientFrm &&
+ pPreUp != pVertPosOrientFrm &&
+ !pPreUp->IsAnLower( pVertPosOrientFrm ) )
+ {
+ return TRUE;
+ }
+ }
+ }
+ // <--
+ }
+ return FALSE;
+}
+
+/*************************************************************************
+|*
+|* BOOL SwFlowFrm::IsPageBreak()
+|*
+|* Beschreibung Wenn vor dem Frm ein harter Seitenumbruch steht UND
+|* es einen Vorgaenger auf der gleichen Seite gibt, wird TRUE
+|* zurueckgeliefert (es muss ein PageBreak erzeugt werden) FALSE sonst.
+|* Wenn in bAct TRUE uebergeben wird, gibt die Funktion dann TRUE
+|* zurueck, wenn ein PageBreak besteht.
+|* Fuer Follows wird der harte Seitenumbruch natuerlich nicht
+|* ausgewertet.
+|* Der Seitenumbruch steht im eigenen FrmFmt (BEFORE) oder im FrmFmt
+|* des Vorgaengers (AFTER). Wenn es keinen Vorgaenger auf der Seite
+|* gibt ist jede weitere Ueberlegung ueberfluessig.
+|* Ein Seitenumbruch (oder der Bedarf) liegt auch dann vor, wenn
+|* im FrmFmt ein PageDesc angegeben wird.
+|* Die Implementierung arbeitet zuaechst nur auf CntntFrms!
+|* -->Fuer LayoutFrms ist die Definition des Vorgaengers unklar.
+|* Ersterstellung MA ??
+|* Letzte Aenderung MA 21. Mar. 95
+|*
+|*************************************************************************/
+
+
+BOOL SwFlowFrm::IsPageBreak( BOOL bAct ) const
+{
+ if ( !IsFollow() && rThis.IsInDocBody() &&
+ ( !rThis.IsInTab() || ( rThis.IsTabFrm() && !rThis.GetUpper()->IsInTab() ) ) && // i66968
+ !rThis.GetUpper()->GetFmt()->getIDocumentSettingAccess()->get(IDocumentSettingAccess::BROWSE_MODE) )
+ {
+ const SwAttrSet *pSet = rThis.GetAttrSet();
+
+ //Vorgaenger ermitteln
+ const SwFrm *pPrev = rThis.FindPrev();
+ while ( pPrev && ( !pPrev->IsInDocBody() ||
+ ( pPrev->IsTxtFrm() && ((SwTxtFrm*)pPrev)->IsHiddenNow() ) ) )
+ pPrev = pPrev->FindPrev();
+
+ if ( pPrev )
+ {
+ ASSERT( pPrev->IsInDocBody(), "IsPageBreak: Not in DocBody?" );
+ if ( bAct )
+ { if ( rThis.FindPageFrm() == pPrev->FindPageFrm() )
+ return FALSE;
+ }
+ else
+ { if ( rThis.FindPageFrm() != pPrev->FindPageFrm() )
+ return FALSE;
+ }
+
+ const SvxBreak eBreak = pSet->GetBreak().GetBreak();
+ if ( eBreak == SVX_BREAK_PAGE_BEFORE || eBreak == SVX_BREAK_PAGE_BOTH )
+ return TRUE;
+ else
+ {
+ const SvxBreak &ePrB = pPrev->GetAttrSet()->GetBreak().GetBreak();
+ if ( ePrB == SVX_BREAK_PAGE_AFTER ||
+ ePrB == SVX_BREAK_PAGE_BOTH ||
+ pSet->GetPageDesc().GetPageDesc() )
+ return TRUE;
+ }
+ }
+ }
+ return FALSE;
+}
+
+/*************************************************************************
+|*
+|* BOOL SwFlowFrm::IsColBreak()
+|*
+|* Beschreibung Wenn vor dem Frm ein harter Spaltenumbruch steht UND
+|* es einen Vorgaenger in der gleichen Spalte gibt, wird TRUE
+|* zurueckgeliefert (es muss ein PageBreak erzeugt werden) FALSE sonst.
+|* Wenn in bAct TRUE uebergeben wird, gibt die Funktion dann TRUE
+|* zurueck, wenn ein ColBreak besteht.
+|* Fuer Follows wird der harte Spaltenumbruch natuerlich nicht
+|* ausgewertet.
+|* Der Spaltenumbruch steht im eigenen FrmFmt (BEFORE) oder im FrmFmt
+|* des Vorgaengers (AFTER). Wenn es keinen Vorgaenger in der Spalte
+|* gibt ist jede weitere Ueberlegung ueberfluessig.
+|* Die Implementierung arbeitet zuaechst nur auf CntntFrms!
+|* -->Fuer LayoutFrms ist die Definition des Vorgaengers unklar.
+|* Ersterstellung MA 11. Jun. 93
+|* Letzte Aenderung MA 21. Mar. 95
+|*
+|*************************************************************************/
+
+BOOL SwFlowFrm::IsColBreak( BOOL bAct ) const
+{
+ if ( !IsFollow() && (rThis.IsMoveable() || bAct) )
+ {
+ const SwFrm *pCol = rThis.FindColFrm();
+ if ( pCol )
+ {
+ //Vorgaenger ermitteln
+ const SwFrm *pPrev = rThis.FindPrev();
+ while( pPrev && ( ( !pPrev->IsInDocBody() && !rThis.IsInFly() ) ||
+ ( pPrev->IsTxtFrm() && ((SwTxtFrm*)pPrev)->IsHiddenNow() ) ) )
+ pPrev = pPrev->FindPrev();
+
+ if ( pPrev )
+ {
+ if ( bAct )
+ { if ( pCol == pPrev->FindColFrm() )
+ return FALSE;
+ }
+ else
+ { if ( pCol != pPrev->FindColFrm() )
+ return FALSE;
+ }
+
+ const SvxBreak eBreak = rThis.GetAttrSet()->GetBreak().GetBreak();
+ if ( eBreak == SVX_BREAK_COLUMN_BEFORE ||
+ eBreak == SVX_BREAK_COLUMN_BOTH )
+ return TRUE;
+ else
+ {
+ const SvxBreak &ePrB = pPrev->GetAttrSet()->GetBreak().GetBreak();
+ if ( ePrB == SVX_BREAK_COLUMN_AFTER ||
+ ePrB == SVX_BREAK_COLUMN_BOTH )
+ return TRUE;
+ }
+ }
+ }
+ }
+ return FALSE;
+}
+
+BOOL SwFlowFrm::HasParaSpaceAtPages( BOOL bSct ) const
+{
+ if( rThis.IsInSct() )
+ {
+ const SwFrm* pTmp = rThis.GetUpper();
+ while( pTmp )
+ {
+ if( pTmp->IsCellFrm() || pTmp->IsFlyFrm() ||
+ pTmp->IsFooterFrm() || pTmp->IsHeaderFrm() ||
+ ( pTmp->IsFtnFrm() && !((SwFtnFrm*)pTmp)->GetMaster() ) )
+ return TRUE;
+ if( pTmp->IsPageFrm() )
+ return ( pTmp->GetPrev() && !IsPageBreak(TRUE) ) ? FALSE : TRUE;
+ if( pTmp->IsColumnFrm() && pTmp->GetPrev() )
+ return IsColBreak( TRUE );
+ if( pTmp->IsSctFrm() && ( !bSct || pTmp->GetPrev() ) )
+ return FALSE;
+ pTmp = pTmp->GetUpper();
+ }
+ ASSERT( FALSE, "HasParaSpaceAtPages: Where's my page?" );
+ return FALSE;
+ }
+ if( !rThis.IsInDocBody() || ( rThis.IsInTab() && !rThis.IsTabFrm()) ||
+ IsPageBreak( TRUE ) || ( rThis.FindColFrm() && IsColBreak( TRUE ) ) )
+ return TRUE;
+ const SwFrm* pTmp = rThis.FindColFrm();
+ if( pTmp )
+ {
+ if( pTmp->GetPrev() )
+ return FALSE;
+ }
+ else
+ pTmp = &rThis;
+ pTmp = pTmp->FindPageFrm();
+ return pTmp && !pTmp->GetPrev();
+}
+
+/** helper method to determine previous frame for calculation of the
+ upper space
+
+ OD 2004-03-10 #i11860#
+
+ @author OD
+*/
+const SwFrm* SwFlowFrm::_GetPrevFrmForUpperSpaceCalc( const SwFrm* _pProposedPrevFrm ) const
+{
+ const SwFrm* pPrevFrm = _pProposedPrevFrm
+ ? _pProposedPrevFrm
+ : rThis.GetPrev();
+
+ // Skip hidden paragraphs and empty sections
+ while ( pPrevFrm &&
+ ( ( pPrevFrm->IsTxtFrm() &&
+ static_cast<const SwTxtFrm*>(pPrevFrm)->IsHiddenNow() ) ||
+ ( pPrevFrm->IsSctFrm() &&
+ !static_cast<const SwSectionFrm*>(pPrevFrm)->GetSection() ) ) )
+ {
+ pPrevFrm = pPrevFrm->GetPrev();
+ }
+
+ // Special case: no direct previous frame is found but frame is in footnote
+ // Search for a previous frame in previous footnote,
+ // if frame isn't in a section, which is also in the footnote
+ if ( !pPrevFrm && rThis.IsInFtn() &&
+ ( rThis.IsSctFrm() ||
+ !rThis.IsInSct() || !rThis.FindSctFrm()->IsInFtn() ) )
+ {
+ const SwFtnFrm* pPrevFtnFrm =
+ static_cast<const SwFtnFrm*>(rThis.FindFtnFrm()->GetPrev());
+ if ( pPrevFtnFrm )
+ {
+ pPrevFrm = pPrevFtnFrm->GetLastLower();
+
+ // Skip hidden paragraphs and empty sections
+ while ( pPrevFrm &&
+ ( ( pPrevFrm->IsTxtFrm() &&
+ static_cast<const SwTxtFrm*>(pPrevFrm)->IsHiddenNow() ) ||
+ ( pPrevFrm->IsSctFrm() &&
+ !static_cast<const SwSectionFrm*>(pPrevFrm)->GetSection() ) ) )
+ {
+ pPrevFrm = pPrevFrm->GetPrev();
+ }
+ }
+ }
+ // Special case: found previous frame is a section
+ // Search for the last content in the section
+ if( pPrevFrm && pPrevFrm->IsSctFrm() )
+ {
+ const SwSectionFrm* pPrevSectFrm =
+ static_cast<const SwSectionFrm*>(pPrevFrm);
+ pPrevFrm = pPrevSectFrm->FindLastCntnt();
+ // If the last content is in a table _inside_ the section,
+ // take the table herself.
+ // OD 2004-02-18 #106629# - correction:
+ // Check directly, if table is inside table, instead of indirectly
+ // by checking, if section isn't inside a table
+ if ( pPrevFrm && pPrevFrm->IsInTab() )
+ {
+ const SwTabFrm* pTableFrm = pPrevFrm->FindTabFrm();
+ if ( pPrevSectFrm->IsAnLower( pTableFrm ) )
+ {
+ pPrevFrm = pTableFrm;
+ }
+ }
+ // OD 2004-02-18 #106629# correction: skip hidden text frames
+ while ( pPrevFrm &&
+ pPrevFrm->IsTxtFrm() &&
+ static_cast<const SwTxtFrm*>(pPrevFrm)->IsHiddenNow() )
+ {
+ pPrevFrm = pPrevFrm->GetPrev();
+ }
+ }
+
+ return pPrevFrm;
+}
+
+// OD 2004-03-12 #i11860# - add 3rd parameter <_bConsiderGrid>
+SwTwips SwFlowFrm::CalcUpperSpace( const SwBorderAttrs *pAttrs,
+ const SwFrm* pPr,
+ const bool _bConsiderGrid ) const
+{
+ // OD 2004-03-10 #i11860# - use new method <GetPrevFrmForUpperSpaceCalc(..)>
+ const SwFrm* pPrevFrm = _GetPrevFrmForUpperSpaceCalc( pPr );
+
+ SwBorderAttrAccess *pAccess;
+ SwFrm* pOwn;
+ if( !pAttrs )
+ {
+ if( rThis.IsSctFrm() )
+ {
+ SwSectionFrm* pFoll = &((SwSectionFrm&)rThis);
+ do
+ pOwn = pFoll->ContainsAny();
+ while( !pOwn && 0 != ( pFoll = pFoll->GetFollow() ) );
+ if( !pOwn )
+ return 0;
+ }
+ else
+ pOwn = &rThis;
+ pAccess= new SwBorderAttrAccess( SwFrm::GetCache(), pOwn );
+ pAttrs = pAccess->Get();
+ }
+ else
+ {
+ pAccess = NULL;
+ pOwn = &rThis;
+ }
+ SwTwips nUpper = 0;
+ // OD 06.01.2004 #i11859#
+ {
+ const IDocumentSettingAccess* pIDSA = rThis.GetUpper()->GetFmt()->getIDocumentSettingAccess();
+ const bool bUseFormerLineSpacing = pIDSA->get(IDocumentSettingAccess::OLD_LINE_SPACING);
+ if( pPrevFrm )
+ {
+ // OD 2004-03-10 #i11860# - use new method to determine needed spacing
+ // values of found previous frame and use these values.
+ SwTwips nPrevLowerSpace = 0;
+ SwTwips nPrevLineSpacing = 0;
+ // --> OD 2009-08-28 #i102458#
+ bool bPrevLineSpacingPorportional = false;
+ GetSpacingValuesOfFrm( (*pPrevFrm),
+ nPrevLowerSpace, nPrevLineSpacing,
+ bPrevLineSpacingPorportional );
+ // <--
+ if( pIDSA->get(IDocumentSettingAccess::PARA_SPACE_MAX) )
+ {
+ nUpper = nPrevLowerSpace + pAttrs->GetULSpace().GetUpper();
+ SwTwips nAdd = nPrevLineSpacing;
+ // OD 07.01.2004 #i11859# - consideration of the line spacing
+ // for the upper spacing of a text frame
+ if ( bUseFormerLineSpacing )
+ {
+ // former consideration
+ if ( pOwn->IsTxtFrm() )
+ {
+ nAdd = Max( nAdd, static_cast<SwTxtFrm&>(rThis).GetLineSpace() );
+ }
+ nUpper += nAdd;
+ }
+ else
+ {
+ // new consideration:
+ // Only the proportional line spacing of the previous
+ // text frame is considered for the upper spacing and
+ // the line spacing values are add up instead of
+ // building its maximum.
+ if ( pOwn->IsTxtFrm() )
+ {
+ // --> OD 2009-08-28 #i102458#
+ // Correction:
+ // A proportional line spacing of the previous text frame
+ // is added up to a own leading line spacing.
+ // Otherwise, the maximum of the leading line spacing
+ // of the previous text frame and the own leading line
+ // spacing is built.
+ if ( bPrevLineSpacingPorportional )
+ {
+ nAdd += static_cast<SwTxtFrm&>(rThis).GetLineSpace( true );
+ }
+ else
+ {
+ nAdd = Max( nAdd, static_cast<SwTxtFrm&>(rThis).GetLineSpace( true ) );
+ }
+ // <--
+ }
+ nUpper += nAdd;
+ }
+ }
+ else
+ {
+ nUpper = Max( static_cast<long>(nPrevLowerSpace),
+ static_cast<long>(pAttrs->GetULSpace().GetUpper()) );
+ // OD 07.01.2004 #i11859# - consideration of the line spacing
+ // for the upper spacing of a text frame
+ if ( bUseFormerLineSpacing )
+ {
+ // former consideration
+ if ( pOwn->IsTxtFrm() )
+ nUpper = Max( nUpper, ((SwTxtFrm*)pOwn)->GetLineSpace() );
+ if ( nPrevLineSpacing != 0 )
+ {
+ nUpper = Max( nUpper, nPrevLineSpacing );
+ }
+ }
+ else
+ {
+ // new consideration:
+ // Only the proportional line spacing of the previous
+ // text frame is considered for the upper spacing and
+ // the line spacing values are add up and added to
+ // the paragraph spacing instead of building the
+ // maximum of the line spacings and the paragraph spacing.
+ SwTwips nAdd = nPrevLineSpacing;
+ if ( pOwn->IsTxtFrm() )
+ {
+ // --> OD 2009-08-28 #i102458#
+ // Correction:
+ // A proportional line spacing of the previous text frame
+ // is added up to a own leading line spacing.
+ // Otherwise, the maximum of the leading line spacing
+ // of the previous text frame and the own leading line
+ // spacing is built.
+ if ( bPrevLineSpacingPorportional )
+ {
+ nAdd += static_cast<SwTxtFrm&>(rThis).GetLineSpace( true );
+ }
+ else
+ {
+ nAdd = Max( nAdd, static_cast<SwTxtFrm&>(rThis).GetLineSpace( true ) );
+ }
+ // <--
+ }
+ nUpper += nAdd;
+ }
+ }
+ }
+ else if ( pIDSA->get(IDocumentSettingAccess::PARA_SPACE_MAX_AT_PAGES) &&
+ CastFlowFrm( pOwn )->HasParaSpaceAtPages( rThis.IsSctFrm() ) )
+ {
+ nUpper = pAttrs->GetULSpace().GetUpper();
+ }
+ }
+
+ // OD 2004-02-26 #i25029# - pass previous frame <pPrevFrm>
+ // to method <GetTopLine(..)>, if parameter <pPr> is set.
+ // Note: parameter <pPr> is set, if method is called from <SwTxtFrm::WouldFit(..)>
+ nUpper += pAttrs->GetTopLine( rThis, (pPr ? pPrevFrm : 0L) );
+
+ // OD 2004-03-12 #i11860# - consider value of new parameter <_bConsiderGrid>
+ // and use new method <GetUpperSpaceAmountConsideredForPageGrid(..)>
+
+ //consider grid in square page mode
+ if ( _bConsiderGrid && rThis.GetUpper()->GetFmt()->GetDoc()->IsSquaredPageMode() )
+ {
+ nUpper += _GetUpperSpaceAmountConsideredForPageGrid( nUpper );
+ }
+
+ delete pAccess;
+ return nUpper;
+}
+
+/** method to detemine the upper space amount, which is considered for
+ the page grid
+
+ OD 2004-03-12 #i11860#
+ Precondition: Position of frame is valid.
+
+ @author OD
+*/
+SwTwips SwFlowFrm::_GetUpperSpaceAmountConsideredForPageGrid(
+ const SwTwips _nUpperSpaceWithoutGrid ) const
+{
+ SwTwips nUpperSpaceAmountConsideredForPageGrid = 0;
+
+ if ( rThis.IsInDocBody() && rThis.GetAttrSet()->GetParaGrid().GetValue() )
+ {
+ const SwPageFrm* pPageFrm = rThis.FindPageFrm();
+ GETGRID( pPageFrm )
+ if( pGrid )
+ {
+ const SwFrm* pBodyFrm = pPageFrm->FindBodyCont();
+ if ( pBodyFrm )
+ {
+ const long nGridLineHeight =
+ pGrid->GetBaseHeight() + pGrid->GetRubyHeight();
+
+ SWRECTFN( (&rThis) )
+ const SwTwips nBodyPrtTop = (pBodyFrm->*fnRect->fnGetPrtTop)();
+ const SwTwips nProposedPrtTop =
+ (*fnRect->fnYInc)( (rThis.Frm().*fnRect->fnGetTop)(),
+ _nUpperSpaceWithoutGrid );
+
+ const SwTwips nSpaceAbovePrtTop =
+ (*fnRect->fnYDiff)( nProposedPrtTop, nBodyPrtTop );
+ const SwTwips nSpaceOfCompleteLinesAbove =
+ nGridLineHeight * ( nSpaceAbovePrtTop / nGridLineHeight );
+ SwTwips nNewPrtTop =
+ (*fnRect->fnYInc)( nBodyPrtTop, nSpaceOfCompleteLinesAbove );
+ if ( (*fnRect->fnYDiff)( nProposedPrtTop, nNewPrtTop ) > 0 )
+ {
+ nNewPrtTop = (*fnRect->fnYInc)( nNewPrtTop, nGridLineHeight );
+ }
+
+ const SwTwips nNewUpperSpace =
+ (*fnRect->fnYDiff)( nNewPrtTop,
+ (rThis.Frm().*fnRect->fnGetTop)() );
+
+ nUpperSpaceAmountConsideredForPageGrid =
+ nNewUpperSpace - _nUpperSpaceWithoutGrid;
+
+ ASSERT( nUpperSpaceAmountConsideredForPageGrid >= 0,
+ "<SwFlowFrm::GetUpperSpaceAmountConsideredForPageGrid(..)> - negative space considered for page grid!" );
+ }
+ }
+ }
+ return nUpperSpaceAmountConsideredForPageGrid;
+}
+
+/** method to determent the upper space amount, which is considered for
+ the previous frame
+
+ OD 2004-03-11 #i11860#
+
+ @author OD
+*/
+SwTwips SwFlowFrm::_GetUpperSpaceAmountConsideredForPrevFrm() const
+{
+ SwTwips nUpperSpaceAmountOfPrevFrm = 0;
+
+ const SwFrm* pPrevFrm = _GetPrevFrmForUpperSpaceCalc();
+ if ( pPrevFrm )
+ {
+ SwTwips nPrevLowerSpace = 0;
+ SwTwips nPrevLineSpacing = 0;
+ // --> OD 2009-08-28 #i102458#
+ bool bDummy = false;
+ GetSpacingValuesOfFrm( (*pPrevFrm), nPrevLowerSpace, nPrevLineSpacing, bDummy );
+ // <--
+ if ( nPrevLowerSpace > 0 || nPrevLineSpacing > 0 )
+ {
+ const IDocumentSettingAccess* pIDSA = rThis.GetUpper()->GetFmt()->getIDocumentSettingAccess();
+ if ( pIDSA->get(IDocumentSettingAccess::PARA_SPACE_MAX) ||
+ !pIDSA->get(IDocumentSettingAccess::OLD_LINE_SPACING) )
+ {
+ nUpperSpaceAmountOfPrevFrm = nPrevLowerSpace + nPrevLineSpacing;
+ }
+ else
+ {
+ nUpperSpaceAmountOfPrevFrm = Max( nPrevLowerSpace, nPrevLineSpacing );
+ }
+ }
+ }
+
+ return nUpperSpaceAmountOfPrevFrm;
+}
+
+/** method to determine the upper space amount, which is considered for
+ the previous frame and the page grid, if option 'Use former object
+ positioning' is OFF
+
+ OD 2004-03-18 #i11860#
+
+ @author OD
+*/
+SwTwips SwFlowFrm::GetUpperSpaceAmountConsideredForPrevFrmAndPageGrid() const
+{
+ SwTwips nUpperSpaceAmountConsideredForPrevFrmAndPageGrid = 0;
+
+ if ( !rThis.GetUpper()->GetFmt()->getIDocumentSettingAccess()->get(IDocumentSettingAccess::USE_FORMER_OBJECT_POS) )
+ {
+ nUpperSpaceAmountConsideredForPrevFrmAndPageGrid =
+ _GetUpperSpaceAmountConsideredForPrevFrm() +
+ _GetUpperSpaceAmountConsideredForPageGrid( CalcUpperSpace( 0L, 0L, false ) );
+ }
+
+ return nUpperSpaceAmountConsideredForPrevFrmAndPageGrid;
+}
+
+/** calculation of lower space
+
+ OD 2004-03-02 #106629#
+
+ @author OD
+*/
+SwTwips SwFlowFrm::CalcLowerSpace( const SwBorderAttrs* _pAttrs ) const
+{
+ SwTwips nLowerSpace = 0;
+
+ SwBorderAttrAccess* pAttrAccess = 0L;
+ if ( !_pAttrs )
+ {
+ pAttrAccess = new SwBorderAttrAccess( SwFrm::GetCache(), &rThis );
+ _pAttrs = pAttrAccess->Get();
+ }
+
+ sal_Bool bCommonBorder = sal_True;
+ if ( rThis.IsInSct() && rThis.GetUpper()->IsColBodyFrm() )
+ {
+ const SwSectionFrm* pSectFrm = rThis.FindSctFrm();
+ bCommonBorder = pSectFrm->GetFmt()->GetBalancedColumns().GetValue();
+ }
+ nLowerSpace = bCommonBorder ?
+ _pAttrs->GetBottomLine( rThis ) :
+ _pAttrs->CalcBottomLine();
+
+ // --> OD 2004-07-16 #i26250#
+ // - correct consideration of table frames
+ // - use new method <CalcAddLowerSpaceAsLastInTableCell(..)>
+ if ( ( ( rThis.IsTabFrm() && rThis.GetUpper()->IsInTab() ) ||
+ // --> OD 2004-11-16 #115759# - no lower spacing, if frame has a follow
+ ( rThis.IsInTab() && !GetFollow() ) ) &&
+ // <--
+ !rThis.GetIndNext() )
+ {
+ nLowerSpace += CalcAddLowerSpaceAsLastInTableCell( _pAttrs );
+ }
+ // <--
+
+ delete pAttrAccess;
+
+ return nLowerSpace;
+}
+
+/** calculation of the additional space to be considered, if flow frame
+ is the last inside a table cell
+
+ OD 2004-07-16 #i26250#
+
+ @author OD
+*/
+SwTwips SwFlowFrm::CalcAddLowerSpaceAsLastInTableCell(
+ const SwBorderAttrs* _pAttrs ) const
+{
+ SwTwips nAdditionalLowerSpace = 0;
+
+ if ( rThis.GetUpper()->GetFmt()->getIDocumentSettingAccess()->get(IDocumentSettingAccess::ADD_PARA_SPACING_TO_TABLE_CELLS) )
+ {
+ const SwFrm* pFrm = &rThis;
+ if ( pFrm->IsSctFrm() )
+ {
+ const SwSectionFrm* pSectFrm = static_cast<const SwSectionFrm*>(pFrm);
+ pFrm = pSectFrm->FindLastCntnt();
+ if ( pFrm && pFrm->IsInTab() )
+ {
+ const SwTabFrm* pTableFrm = pFrm->FindTabFrm();
+ if ( pSectFrm->IsAnLower( pTableFrm ) )
+ {
+ pFrm = pTableFrm;
+ }
+ }
+ }
+
+ SwBorderAttrAccess* pAttrAccess = 0L;
+ if ( !_pAttrs || pFrm != &rThis )
+ {
+ pAttrAccess = new SwBorderAttrAccess( SwFrm::GetCache(), pFrm );
+ _pAttrs = pAttrAccess->Get();
+ }
+
+ nAdditionalLowerSpace += _pAttrs->GetULSpace().GetLower();
+
+ delete pAttrAccess;
+ }
+
+ return nAdditionalLowerSpace;
+}
+
+/*************************************************************************
+|*
+|* BOOL SwFlowFrm::CheckMoveFwd()
+|*
+|* Beschreibung Moved den Frm vorwaerts wenn es durch die aktuellen
+|* Bedingungen und Attribute notwendig erscheint.
+|* Ersterstellung MA 05. Dec. 96
+|* Letzte Aenderung MA 09. Mar. 98
+|*
+|*************************************************************************/
+
+
+BOOL SwFlowFrm::CheckMoveFwd( BOOL &rbMakePage, BOOL bKeep, BOOL )
+{
+ const SwFrm* pNxt = rThis.GetIndNext();
+
+ if ( bKeep && //!bMovedBwd &&
+ ( !pNxt || ( pNxt->IsTxtFrm() && ((SwTxtFrm*)pNxt)->IsEmptyMaster() ) ) &&
+ ( 0 != (pNxt = rThis.FindNext()) ) && IsKeepFwdMoveAllowed() )
+ {
+ if( pNxt->IsSctFrm() )
+ { // Nicht auf leere SectionFrms hereinfallen
+ const SwFrm* pTmp = NULL;
+ while( pNxt && pNxt->IsSctFrm() &&
+ ( !((SwSectionFrm*)pNxt)->GetSection() ||
+ 0 == ( pTmp = ((SwSectionFrm*)pNxt)->ContainsAny() ) ) )
+ {
+ pNxt = pNxt->FindNext();
+ pTmp = NULL;
+ }
+ if( pTmp )
+ pNxt = pTmp; // the content of the next notempty sectionfrm
+ }
+ if( pNxt && pNxt->GetValidPosFlag() )
+ {
+ BOOL bMove = FALSE;
+ const SwSectionFrm *pSct = rThis.FindSctFrm();
+ if( pSct && !pSct->GetValidSizeFlag() )
+ {
+ const SwSectionFrm* pNxtSct = pNxt->FindSctFrm();
+ if( pNxtSct && pSct->IsAnFollow( pNxtSct ) )
+ bMove = TRUE;
+ }
+ else
+ bMove = TRUE;
+ if( bMove )
+ {
+ //Keep together with the following frame
+ MoveFwd( rbMakePage, FALSE );
+ return TRUE;
+ }
+ }
+ }
+
+ BOOL bMovedFwd = FALSE;
+
+ if ( rThis.GetIndPrev() )
+ {
+ if ( IsPrevObjMove() ) //Auf Objekte des Prev Ruecksicht nehmen?
+ {
+ bMovedFwd = TRUE;
+ if ( !MoveFwd( rbMakePage, FALSE ) )
+ rbMakePage = FALSE;
+ }
+ else
+ {
+ if ( IsPageBreak( FALSE ) )
+ {
+ while ( MoveFwd( rbMakePage, TRUE ) )
+ /* do nothing */;
+ rbMakePage = FALSE;
+ bMovedFwd = TRUE;
+ }
+ else if ( IsColBreak ( FALSE ) )
+ {
+ const SwPageFrm *pPage = rThis.FindPageFrm();
+ SwFrm *pCol = rThis.FindColFrm();
+ do
+ { MoveFwd( rbMakePage, FALSE );
+ SwFrm *pTmp = rThis.FindColFrm();
+ if( pTmp != pCol )
+ {
+ bMovedFwd = TRUE;
+ pCol = pTmp;
+ }
+ else
+ break;
+ } while ( IsColBreak( FALSE ) );
+ if ( pPage != rThis.FindPageFrm() )
+ rbMakePage = FALSE;
+ }
+ }
+ }
+ return bMovedFwd;
+}
+
+/*************************************************************************
+|*
+|* BOOL SwFlowFrm::MoveFwd()
+|*
+|* Beschreibung Returnwert sagt, ob der Frm die Seite gewechselt hat.
+|* Ersterstellung MA 05. Dec. 96
+|* Letzte Aenderung MA 05. Dec. 96
+|*
+|*************************************************************************/
+
+
+BOOL SwFlowFrm::MoveFwd( BOOL bMakePage, BOOL bPageBreak, BOOL bMoveAlways )
+{
+//!!!!MoveFtnCntFwd muss ggf. mitgepflegt werden.
+ SwFtnBossFrm *pOldBoss = rThis.FindFtnBossFrm();
+ if ( rThis.IsInFtn() )
+ return ((SwCntntFrm&)rThis).MoveFtnCntFwd( bMakePage, pOldBoss );
+
+ if( !IsFwdMoveAllowed() && !bMoveAlways )
+ {
+ BOOL bNoFwd = TRUE;
+ if( rThis.IsInSct() )
+ {
+ SwFtnBossFrm* pBoss = rThis.FindFtnBossFrm();
+ bNoFwd = !pBoss->IsInSct() || ( !pBoss->Lower()->GetNext() &&
+ !pBoss->GetPrev() );
+ }
+
+ // Allow the MoveFwd even if we do not have an IndPrev in these cases:
+ if ( rThis.IsInTab() &&
+ ( !rThis.IsTabFrm() ||
+ ( rThis.GetUpper()->IsInTab() &&
+ rThis.GetUpper()->FindTabFrm()->IsFwdMoveAllowed() ) ) &&
+ 0 != const_cast<SwFrm&>(rThis).GetNextCellLeaf( MAKEPAGE_NONE ) )
+/*
+ &&
+ // NEW TABLES
+ // Have a look at our main competitor: We don't move inside row span cells:
+ ( !rThis.GetUpper()->IsCellFrm() || !rThis.GetUpper()->IsLeaveUpperAllowed() ) )*/
+ {
+ bNoFwd = FALSE;
+ }
+
+ if( bNoFwd )
+ {
+ //Fuer PageBreak ist das Moven erlaubt, wenn der Frm nicht
+ //bereits der erste der Seite ist.
+ if ( !bPageBreak )
+ return FALSE;
+
+ const SwFrm *pCol = rThis.FindColFrm();
+ if ( !pCol || !pCol->GetPrev() )
+ return FALSE;
+ }
+ }
+
+ BOOL bSamePage = TRUE;
+ SwLayoutFrm *pNewUpper =
+ rThis.GetLeaf( bMakePage ? MAKEPAGE_INSERT : MAKEPAGE_NONE, TRUE );
+
+ if ( pNewUpper )
+ {
+ PROTOCOL_ENTER( &rThis, PROT_MOVE_FWD, 0, 0 );
+ SwPageFrm *pOldPage = pOldBoss->FindPageFrm();
+ //Wir moven uns und alle direkten Nachfolger vor den ersten
+ //CntntFrm unterhalb des neuen Uppers.
+
+ // Wenn unser NewUpper in einem SectionFrm liegt, muessen wir
+ // verhindern, dass sich dieser im Calc selbst zerstoert
+ SwSectionFrm* pSect = pNewUpper->FindSctFrm();
+ BOOL bUnlock = FALSE;
+ if( pSect )
+ {
+ // Wenn wir nur innerhalb unseres SectionFrms die Spalte wechseln,
+ // rufen wir lieber kein Calc, sonst wird noch der SectionFrm
+ // formatiert, der wiederum uns ruft etc.
+ if( pSect != rThis.FindSctFrm() )
+ {
+ bUnlock = !pSect->IsColLocked();
+ pSect->ColLock();
+ pNewUpper->Calc();
+ if( bUnlock )
+ pSect->ColUnlock();
+ }
+ }
+ // Do not calculate split cell frames.
+ else if ( !pNewUpper->IsCellFrm() || ((SwLayoutFrm*)pNewUpper)->Lower() )
+ pNewUpper->Calc();
+
+ SwFtnBossFrm *pNewBoss = pNewUpper->FindFtnBossFrm();
+ BOOL bBossChg = pNewBoss != pOldBoss;
+ pNewBoss = pNewBoss->FindFtnBossFrm( TRUE );
+ pOldBoss = pOldBoss->FindFtnBossFrm( TRUE );
+ SwPageFrm* pNewPage = pOldPage;
+
+ // First, we move the footnotes.
+ BOOL bFtnMoved = FALSE;
+
+ // --> FME 2004-07-15 #i26831#
+ // If pSect has just been created, the printing area of pSect has
+ // been calculated based on the first content of its follow.
+ // In this case we prefer to call a SimpleFormat for this new
+ // section after we inserted the contents. Otherwise the section
+ // frame will invalidate its lowers, if its printing area changes
+ // in SwSectionFrm::Format, which can cause loops.
+ const bool bForceSimpleFormat = pSect && pSect->HasFollow() &&
+ !pSect->ContainsAny();
+ // <--
+
+ if ( pNewBoss != pOldBoss )
+ {
+ pNewPage = pNewBoss->FindPageFrm();
+ bSamePage = pNewPage == pOldPage;
+ //Damit die Fussnoten nicht auf dumme Gedanken kommen
+ //setzen wir hier die Deadline.
+ SWRECTFN( pOldBoss )
+ SwSaveFtnHeight aHeight( pOldBoss,
+ (pOldBoss->Frm().*fnRect->fnGetBottom)() );
+ SwCntntFrm* pStart = rThis.IsCntntFrm() ?
+ (SwCntntFrm*)&rThis : ((SwLayoutFrm&)rThis).ContainsCntnt();
+ ASSERT( pStart || ( rThis.IsTabFrm() && !((SwTabFrm&)rThis).Lower() ),
+ "MoveFwd: Missing Content" );
+ SwLayoutFrm* pBody = pStart ? ( pStart->IsTxtFrm() ?
+ (SwLayoutFrm*)((SwTxtFrm*)pStart)->FindBodyFrm() : 0 ) : 0;
+ if( pBody )
+ bFtnMoved = pBody->MoveLowerFtns( pStart, pOldBoss, pNewBoss,
+ FALSE);
+ }
+ // Bei SectionFrms ist es moeglich, dass wir selbst durch pNewUpper->Calc()
+ // bewegt wurden, z. B. in den pNewUpper.
+ // MoveSubTree bzw. PasteTree ist auf so etwas nicht vorbereitet.
+ if( pNewUpper != rThis.GetUpper() )
+ {
+ // --> FME 2004-04-19 #i27145#
+ SwSectionFrm* pOldSct = 0;
+ if ( rThis.GetUpper()->IsSctFrm() )
+ {
+ pOldSct = static_cast<SwSectionFrm*>(rThis.GetUpper());
+ }
+ // <--
+
+ MoveSubTree( pNewUpper, pNewUpper->Lower() );
+
+ // --> FME 2004-04-19 #i27145#
+ if ( pOldSct && pOldSct->GetSection() )
+ {
+ // Prevent loops by setting the new height at
+ // the section frame if footnotes have been moved.
+ // Otherwise the call of SwLayNotify::~SwLayNotify() for
+ // the (invalid) section frame will invalidate the first
+ // lower of its follow, because it grows due to the removed
+ // footnotes.
+ // Note: If pOldSct has become empty during MoveSubTree, it
+ // has already been scheduled for removal. No SimpleFormat
+ // for these.
+ pOldSct->SimpleFormat();
+ }
+ // <--
+
+ // --> FME 2004-07-15 #i26831#
+ if ( bForceSimpleFormat )
+ {
+ pSect->SimpleFormat();
+ }
+ // <--
+
+ if ( bFtnMoved && !bSamePage )
+ {
+ pOldPage->UpdateFtnNum();
+ pNewPage->UpdateFtnNum();
+ }
+
+ if( bBossChg )
+ {
+ rThis.Prepare( PREP_BOSS_CHGD, 0, FALSE );
+ if( !bSamePage )
+ {
+ ViewShell *pSh = rThis.GetShell();
+ if ( pSh && !pSh->Imp()->IsUpdateExpFlds() )
+ pSh->GetDoc()->SetNewFldLst(true); //Wird von CalcLayout() hinterher erledigt!
+
+ pNewPage->InvalidateSpelling();
+ pNewPage->InvalidateSmartTags(); // SMARTTAGS
+ pNewPage->InvalidateAutoCompleteWords();
+ pNewPage->InvalidateWordCount();
+ }
+ }
+ }
+ // OD 30.10.2002 #97265# - no <CheckPageDesc(..)> in online layout
+ if ( !pNewPage->GetFmt()->getIDocumentSettingAccess()->get(IDocumentSettingAccess::BROWSE_MODE) )
+ {
+ // --> OD 2009-12-31 #i106452#
+ // check page description not only in situation with sections.
+ if ( !bSamePage &&
+ ( rThis.GetAttrSet()->GetPageDesc().GetPageDesc() ||
+ pOldPage->GetPageDesc()->GetFollow() != pNewPage->GetPageDesc() ) )
+ {
+ SwFrm::CheckPageDescs( pNewPage, FALSE );
+ }
+ // <--
+ }
+ }
+ return bSamePage;
+}
+
+
+/*************************************************************************
+|*
+|* BOOL SwFlowFrm::MoveBwd()
+|*
+|* Beschreibung Returnwert sagt, ob der Frm die Seite wechseln soll.
+|* Sollte von abgeleiteten Klassen gerufen werden.
+|* Das moven selbst muessen die abgeleiteten uebernehmen.
+|* Ersterstellung MA 05. Dec. 96
+|* Letzte Aenderung MA 05. Dec. 96
+|*
+|*************************************************************************/
+
+BOOL SwFlowFrm::MoveBwd( BOOL &rbReformat )
+{
+ SwFlowFrm::SetMoveBwdJump( FALSE );
+
+ SwFtnFrm* pFtn = rThis.FindFtnFrm();
+ if ( pFtn && pFtn->IsBackMoveLocked() )
+ return FALSE;
+
+ // --> OD 2004-11-29 #115759# - text frames, which are directly inside
+ // tables aren't allowed to move backward.
+ if ( rThis.IsTxtFrm() && rThis.IsInTab() )
+ {
+ const SwLayoutFrm* pUpperFrm = rThis.GetUpper();
+ while ( pUpperFrm )
+ {
+ if ( pUpperFrm->IsTabFrm() )
+ {
+ return FALSE;
+ }
+ else if ( pUpperFrm->IsColumnFrm() && pUpperFrm->IsInSct() )
+ {
+ break;
+ }
+ pUpperFrm = pUpperFrm->GetUpper();
+ }
+ }
+ // <--
+
+ SwFtnBossFrm * pOldBoss = rThis.FindFtnBossFrm();
+ SwPageFrm * const pOldPage = pOldBoss->FindPageFrm();
+ SwLayoutFrm *pNewUpper = 0;
+ BOOL bCheckPageDescs = FALSE;
+ bool bCheckPageDescOfNextPage = false;
+
+ if ( pFtn )
+ {
+ //Wenn die Fussnote bereits auf der gleichen Seite/Spalte wie die Referenz
+ //steht, ist nix mit zurueckfliessen. Die breaks brauche fuer die
+ //Fussnoten nicht geprueft zu werden.
+
+ // --> FME 2004-11-15 #i37084# FindLastCntnt does not necessarily
+ // have to have a result != 0
+ SwFrm* pRef = 0;
+ const bool bEndnote = pFtn->GetAttr()->GetFtn().IsEndNote();
+ if( bEndnote && pFtn->IsInSct() )
+ {
+ SwSectionFrm* pSect = pFtn->FindSctFrm();
+ if( pSect->IsEndnAtEnd() )
+ pRef = pSect->FindLastCntnt( FINDMODE_LASTCNT );
+ }
+ if( !pRef )
+ pRef = pFtn->GetRef();
+ // <--
+
+ ASSERT( pRef, "MoveBwd: Endnote for an empty section?" );
+
+ if( !bEndnote )
+ pOldBoss = pOldBoss->FindFtnBossFrm( TRUE );
+ SwFtnBossFrm *pRefBoss = pRef->FindFtnBossFrm( !bEndnote );
+ if ( pOldBoss != pRefBoss &&
+ // OD 08.11.2002 #104840# - use <SwLayoutFrm::IsBefore(..)>
+ ( !bEndnote ||
+ pRefBoss->IsBefore( pOldBoss ) )
+ )
+ pNewUpper = rThis.GetLeaf( MAKEPAGE_FTN, FALSE );
+ }
+ else if ( IsPageBreak( TRUE ) ) //PageBreak zu beachten?
+ {
+ //Wenn auf der vorhergehenden Seite kein Frm im Body steht,
+ //so ist das Zurueckfliessen trotz Pagebreak sinnvoll
+ //(sonst: leere Seite).
+ //Natuerlich muessen Leereseiten geflissentlich uebersehen werden!
+ const SwFrm *pFlow = &rThis;
+ do
+ {
+ pFlow = pFlow->FindPrev();
+ } while ( pFlow &&
+ ( pFlow->FindPageFrm() == pOldPage ||
+ !pFlow->IsInDocBody() ) );
+ if ( pFlow )
+ {
+ long nDiff = pOldPage->GetPhyPageNum() - pFlow->GetPhyPageNum();
+ if ( nDiff > 1 )
+ {
+ if ( ((SwPageFrm*)pOldPage->GetPrev())->IsEmptyPage() )
+ nDiff -= 1;
+ if ( nDiff > 1 )
+ {
+ pNewUpper = rThis.GetLeaf( MAKEPAGE_NONE, FALSE );
+ // --> OD 2006-05-08 #i53139#
+ // Now <pNewUpper> is a previous layout frame, which contains
+ // content. But the new upper layout frame has to be the next one.
+ // Thus, hack for issue i14206 no longer needed, but fix for issue 114442
+ // --> OD 2006-05-17 #136024# - correct fix for i53139:
+ // Check for wrong page description before using next new upper.
+ // --> OD 2006-06-06 #i66051# - further correction of fix for i53139
+ // Check for correct type of new next upper layout frame
+ // --> OD 2006-06-08 #136538# - another correction of fix for i53139
+ // Assumption, that in all cases <pNewUpper> is a previous
+ // layout frame, which contains content, is wrong.
+ // --> OD 2006-07-05 #136538# - another correction of fix for i53139
+ // Beside type check, check also, if proposed new next upper
+ // frame is inside the same frame types.
+ // --> OD 2007-01-10 #i73194# - and yet another correction
+ // of fix for i53139:
+ // Assure that the new next upper layout frame doesn't
+ // equal the current one.
+ // E.g.: content is on page 3, on page 2 is only a 'ghost'
+ // section and on page 1 is normal content. Method <FindPrev(..)>
+ // will find the last content of page 1, but <GetLeaf(..)>
+ // returns new upper on page 2.
+ if ( pNewUpper->Lower() )
+ {
+ SwLayoutFrm* pNewNextUpper = pNewUpper->GetLeaf( MAKEPAGE_NONE, TRUE );
+ if ( pNewNextUpper &&
+ pNewNextUpper != rThis.GetUpper() &&
+ pNewNextUpper->GetType() == pNewUpper->GetType() &&
+ pNewNextUpper->IsInDocBody() == pNewUpper->IsInDocBody() &&
+ pNewNextUpper->IsInFtn() == pNewUpper->IsInFtn() &&
+ pNewNextUpper->IsInTab() == pNewUpper->IsInTab() &&
+ pNewNextUpper->IsInSct() == pNewUpper->IsInSct() &&
+ !rThis.WrongPageDesc( pNewNextUpper->FindPageFrm() ) )
+ {
+ pNewUpper = pNewNextUpper;
+ bCheckPageDescOfNextPage = true;
+ }
+ }
+ // <--
+
+ bCheckPageDescs = TRUE;
+ }
+ }
+ }
+ }
+ else if ( IsColBreak( TRUE ) )
+ {
+ //Wenn in der vorhergehenden Spalte kein CntntFrm steht, so ist
+ //das Zurueckfliessen trotz ColumnBreak sinnvoll
+ //(sonst: leere Spalte).
+ if( rThis.IsInSct() )
+ {
+ pNewUpper = rThis.GetLeaf( MAKEPAGE_NONE, FALSE );
+ if( pNewUpper && !SwFlowFrm::IsMoveBwdJump() &&
+ ( pNewUpper->ContainsCntnt() ||
+ ( ( !pNewUpper->IsColBodyFrm() ||
+ !pNewUpper->GetUpper()->GetPrev() ) &&
+ !pNewUpper->FindSctFrm()->GetPrev() ) ) )
+ {
+ pNewUpper = 0;
+ }
+ // --> OD 2006-05-08 #i53139#
+ // --> OD 2006-09-11 #i69409# - check <pNewUpper>
+ // --> OD 2006-11-02 #i71065# - check <SwFlowFrm::IsMoveBwdJump()>
+ else if ( pNewUpper && !SwFlowFrm::IsMoveBwdJump() )
+ // <--
+ {
+ // Now <pNewUpper> is a previous layout frame, which
+ // contains content. But the new upper layout frame
+ // has to be the next one.
+ // --> OD 2006-05-17 #136024# - correct fix for i53139
+ // Check for wrong page description before using next new upper.
+ // --> OD 2006-06-06 #i66051# - further correction of fix for i53139
+ // Check for correct type of new next upper layout frame
+ // --> OD 2006-07-05 #136538# - another correction of fix for i53139
+ // Beside type check, check also, if proposed new next upper
+ // frame is inside the same frame types.
+ SwLayoutFrm* pNewNextUpper = pNewUpper->GetLeaf( MAKEPAGE_NOSECTION, TRUE );
+ if ( pNewNextUpper &&
+ pNewNextUpper->GetType() == pNewUpper->GetType() &&
+ pNewNextUpper->IsInDocBody() == pNewUpper->IsInDocBody() &&
+ pNewNextUpper->IsInFtn() == pNewUpper->IsInFtn() &&
+ pNewNextUpper->IsInTab() == pNewUpper->IsInTab() &&
+ pNewNextUpper->IsInSct() == pNewUpper->IsInSct() &&
+ !rThis.WrongPageDesc( pNewNextUpper->FindPageFrm() ) )
+ {
+ pNewUpper = pNewNextUpper;
+ }
+ }
+ // <--
+ }
+ else
+ {
+ const SwFrm *pCol = rThis.FindColFrm();
+ BOOL bGoOn = TRUE;
+ BOOL bJump = FALSE;
+ do
+ {
+ if ( pCol->GetPrev() )
+ pCol = pCol->GetPrev();
+ else
+ {
+ bGoOn = FALSE;
+ pCol = rThis.GetLeaf( MAKEPAGE_NONE, FALSE );
+ }
+ if ( pCol )
+ {
+ // ColumnFrms jetzt mit BodyFrm
+ SwLayoutFrm* pColBody = pCol->IsColumnFrm() ?
+ (SwLayoutFrm*)((SwLayoutFrm*)pCol)->Lower() :
+ (SwLayoutFrm*)pCol;
+ if ( pColBody->ContainsCntnt() )
+ {
+ bGoOn = FALSE; // Hier gibt's Inhalt, wir akzeptieren diese
+ // nur, wenn GetLeaf() das MoveBwdJump-Flag gesetzt hat.
+ if( SwFlowFrm::IsMoveBwdJump() )
+ {
+ pNewUpper = pColBody;
+ // --> OD 2006-05-08 #i53139#
+ // Now <pNewUpper> is a previous layout frame, which
+ // contains content. But the new upper layout frame
+ // has to be the next one.
+ // --> OD 2006-05-17 #136024# - correct fix for i53139
+ // Check for wrong page description before using next new upper.
+ // --> OD 2006-06-06 #i66051# - further correction of fix for i53139
+ // Check for correct type of new next upper layout frame
+ // --> OD 2006-07-05 #136538# - another correction of fix for i53139
+ // Beside type check, check also, if proposed new next upper
+ // frame is inside the same frame types.
+ // --> OD 2006-11-02 #i71065#
+ // Check that the proposed new next upper layout
+ // frame isn't the current one.
+ SwLayoutFrm* pNewNextUpper = pNewUpper->GetLeaf( MAKEPAGE_NONE, TRUE );
+ if ( pNewNextUpper &&
+ pNewNextUpper != rThis.GetUpper() &&
+ pNewNextUpper->GetType() == pNewUpper->GetType() &&
+ pNewNextUpper->IsInDocBody() == pNewUpper->IsInDocBody() &&
+ pNewNextUpper->IsInFtn() == pNewUpper->IsInFtn() &&
+ pNewNextUpper->IsInTab() == pNewUpper->IsInTab() &&
+ pNewNextUpper->IsInSct() == pNewUpper->IsInSct() &&
+ !rThis.WrongPageDesc( pNewNextUpper->FindPageFrm() ) )
+ {
+ pNewUpper = pNewNextUpper;
+ }
+ // <--
+ }
+ }
+ else
+ {
+ if( pNewUpper ) // Wir hatten schon eine leere Spalte, haben
+ bJump = TRUE; // also eine uebersprungen
+ pNewUpper = pColBody; // Diese leere Spalte kommt in Frage,
+ // trotzdem weitersuchen
+ }
+ }
+ } while( bGoOn );
+ if( bJump )
+ SwFlowFrm::SetMoveBwdJump( TRUE );
+ }
+ }
+ else //Keine Breaks also kann ich zurueckfliessen
+ pNewUpper = rThis.GetLeaf( MAKEPAGE_NONE, FALSE );
+
+ // --> OD 2004-06-23 #i27801# - no move backward of 'master' text frame,
+ // if - due to its object positioning - it isn't allowed to be on the new page frame
+ // --> OD 2005-03-07 #i44049# - add another condition for not moving backward:
+ // If one of its objects has restarted the layout process, moving backward
+ // isn't sensible either.
+ // --> OD 2005-04-19 #i47697# - refine condition made for issue i44049:
+ // - allow move backward as long as the anchored object is only temporarily
+ // positions considering its wrapping style.
+ if ( pNewUpper &&
+ rThis.IsTxtFrm() && !IsFollow() )
+ {
+ sal_uInt32 nToPageNum( 0L );
+ const bool bMoveFwdByObjPos = SwLayouter::FrmMovedFwdByObjPos(
+ *(pOldPage->GetFmt()->GetDoc()),
+ static_cast<SwTxtFrm&>(rThis),
+ nToPageNum );
+ if ( bMoveFwdByObjPos &&
+ pNewUpper->FindPageFrm()->GetPhyPageNum() < nToPageNum )
+ {
+ pNewUpper = 0;
+ }
+ // --> OD 2005-03-07 #i44049# - check, if one of its anchored objects
+ // has restarted the layout process.
+ else if ( rThis.GetDrawObjs() )
+ {
+ sal_uInt32 i = 0;
+ for ( ; i < rThis.GetDrawObjs()->Count(); ++i )
+ {
+ SwAnchoredObject* pAnchoredObj = (*rThis.GetDrawObjs())[i];
+ // --> OD 2005-04-19 #i47697# - refine condition - see above
+ if ( pAnchoredObj->RestartLayoutProcess() &&
+ !pAnchoredObj->IsTmpConsiderWrapInfluence() )
+ // <--
+ {
+ pNewUpper = 0;
+ break;
+ }
+ }
+ }
+ // <--
+ }
+ // <--
+
+ //Fuer Follows ist das zurueckfliessen nur dann erlaubt wenn in der
+ //neuen Umgebung kein Nachbar existiert (denn dieses waere der Master).
+ //(6677)Wenn allerdings leere Blaetter uebersprungen wurden wird doch gemoved.
+ if ( pNewUpper && IsFollow() && pNewUpper->Lower() )
+ {
+ // --> OD 2007-09-05 #i79774#, #b6596954#
+ // neglect empty sections in proposed new upper frame
+ bool bProposedNewUpperContainsOnlyEmptySections( true );
+ {
+ const SwFrm* pLower( pNewUpper->Lower() );
+ while ( pLower )
+ {
+ if ( pLower->IsSctFrm() &&
+ !dynamic_cast<const SwSectionFrm*>(pLower)->GetSection() )
+ {
+ pLower = pLower->GetNext();
+ continue;
+ }
+ else
+ {
+ bProposedNewUpperContainsOnlyEmptySections = false;
+ break;
+ }
+ }
+ }
+ if ( !bProposedNewUpperContainsOnlyEmptySections )
+ {
+ if ( SwFlowFrm::IsMoveBwdJump() )
+ {
+ //Nicht hinter den Master sondern in das naechstfolgende leere
+ //Blatt moven.
+ SwFrm *pFrm = pNewUpper->Lower();
+ while ( pFrm->GetNext() )
+ pFrm = pFrm->GetNext();
+ pNewUpper = pFrm->GetLeaf( MAKEPAGE_INSERT, TRUE );
+ if( pNewUpper == rThis.GetUpper() ) //Landen wir wieder an der gleichen Stelle?
+ pNewUpper = NULL; //dann eruebrigt sich das Moven
+ }
+ else
+ pNewUpper = 0;
+ }
+ // <--
+ }
+ if ( pNewUpper && !ShouldBwdMoved( pNewUpper, TRUE, rbReformat ) )
+ {
+ if( !pNewUpper->Lower() )
+ {
+ if( pNewUpper->IsFtnContFrm() )
+ {
+ pNewUpper->Cut();
+ delete pNewUpper;
+ }
+ else
+ {
+ SwSectionFrm* pSectFrm = pNewUpper->FindSctFrm();
+ // --> OD 2006-01-04 #126020# - adjust check for empty section
+ // --> OD 2006-02-01 #130797# - correct fix #126020#
+ if ( pSectFrm && !pSectFrm->IsColLocked() &&
+ !pSectFrm->ContainsCntnt() && !pSectFrm->ContainsAny( true ) )
+ // <--
+ {
+ pSectFrm->DelEmpty( TRUE );
+ delete pSectFrm;
+ rThis.bValidPos = TRUE;
+ }
+ }
+ }
+ pNewUpper = 0;
+ }
+
+ // OD 2004-05-26 #i21478# - don't move backward, if flow frame wants to
+ // keep with next frame and next frame is locked.
+ // --> OD 2004-12-08 #i38232# - If next frame is a table, do *not* check,
+ // if it's locked.
+ if ( pNewUpper && !IsFollow() &&
+ rThis.GetAttrSet()->GetKeep().GetValue() && rThis.GetIndNext() )
+ {
+ SwFrm* pIndNext = rThis.GetIndNext();
+ // --> OD 2004-12-08 #i38232#
+ if ( !pIndNext->IsTabFrm() )
+ {
+ // get first content of section, while empty sections are skipped
+ while ( pIndNext && pIndNext->IsSctFrm() )
+ {
+ if( static_cast<SwSectionFrm*>(pIndNext)->GetSection() )
+ {
+ SwFrm* pTmp = static_cast<SwSectionFrm*>(pIndNext)->ContainsAny();
+ if ( pTmp )
+ {
+ pIndNext = pTmp;
+ break;
+ }
+ }
+ pIndNext = pIndNext->GetIndNext();
+ }
+ ASSERT( !pIndNext || pIndNext->ISA(SwTxtFrm),
+ "<SwFlowFrm::MovedBwd(..)> - incorrect next found." );
+ if ( pIndNext && pIndNext->IsFlowFrm() &&
+ SwFlowFrm::CastFlowFrm(pIndNext)->IsJoinLocked() )
+ {
+ pNewUpper = 0L;
+ }
+ }
+ // <--
+ }
+
+ // --> OD 2006-05-10 #i65250#
+ // layout loop control for flowing content again and again moving
+ // backward under the same layout condition.
+ if ( pNewUpper && !IsFollow() &&
+ pNewUpper != rThis.GetUpper() &&
+ SwLayouter::MoveBwdSuppressed( *(pOldPage->GetFmt()->GetDoc()),
+ *this, *pNewUpper ) )
+ {
+ SwLayoutFrm* pNextNewUpper = pNewUpper->GetLeaf(
+ ( !rThis.IsSctFrm() && rThis.IsInSct() )
+ ? MAKEPAGE_NOSECTION
+ : MAKEPAGE_NONE,
+ TRUE );
+ // --> OD 2007-01-10 #i73194# - make code robust
+ ASSERT( pNextNewUpper, "<SwFlowFrm::MoveBwd(..)> - missing next new upper" );
+ if ( pNextNewUpper &&
+ ( pNextNewUpper == rThis.GetUpper() ||
+ pNextNewUpper->GetType() != rThis.GetUpper()->GetType() ) )
+ // <--
+ {
+ pNewUpper = 0L;
+#if OSL_DEBUG_LEVEL > 1
+ ASSERT( false,
+ "<SwFlowFrm::MoveBwd(..)> - layout loop control for layout action <Move Backward> applied!" );
+#endif
+ }
+ }
+ // <--
+
+ ASSERT( pNewUpper != rThis.GetUpper(),
+ "<SwFlowFrm::MoveBwd(..)> - moving backward to the current upper frame!? -> Please inform OD." );
+ if ( pNewUpper )
+ {
+ PROTOCOL_ENTER( &rThis, PROT_MOVE_BWD, 0, 0 );
+ if ( pNewUpper->IsFtnContFrm() )
+ {
+ //Kann sein, dass ich einen Container bekam.
+ SwFtnFrm *pOld = rThis.FindFtnFrm();
+ SwFtnFrm *pNew = new SwFtnFrm( pOld->GetFmt(),
+ pOld->GetRef(), pOld->GetAttr() );
+ if ( pOld->GetMaster() )
+ {
+ pNew->SetMaster( pOld->GetMaster() );
+ pOld->GetMaster()->SetFollow( pNew );
+ }
+ pNew->SetFollow( pOld );
+ pOld->SetMaster( pNew );
+ pNew->Paste( pNewUpper );
+ pNewUpper = pNew;
+ }
+ if( pNewUpper->IsFtnFrm() && rThis.IsInSct() )
+ {
+ SwSectionFrm* pSct = rThis.FindSctFrm();
+ //Wenn wir in einem Bereich in einer Fussnote stecken, muss im
+ //neuen Upper ggf. ein SwSectionFrm angelegt werden
+ if( pSct->IsInFtn() )
+ {
+ SwFrm* pTmp = pNewUpper->Lower();
+ if( pTmp )
+ {
+ while( pTmp->GetNext() )
+ pTmp = pTmp->GetNext();
+ if( !pTmp->IsSctFrm() ||
+ ((SwSectionFrm*)pTmp)->GetFollow() != pSct )
+ pTmp = NULL;
+ }
+ if( pTmp )
+ pNewUpper = (SwSectionFrm*)pTmp;
+ else
+ {
+ pSct = new SwSectionFrm( *pSct, TRUE );
+ pSct->Paste( pNewUpper );
+ pSct->Init();
+ pNewUpper = pSct;
+ pSct->SimpleFormat();
+ }
+ }
+ }
+ BOOL bUnlock = FALSE;
+ BOOL bFollow = FALSE;
+ //Section locken, sonst kann sie bei Fluss des einzigen Cntnt etwa
+ //von zweiter in die erste Spalte zerstoert werden.
+ SwSectionFrm* pSect = pNewUpper->FindSctFrm();
+ if( pSect )
+ {
+ bUnlock = !pSect->IsColLocked();
+ pSect->ColLock();
+ bFollow = pSect->HasFollow();
+ }
+ pNewUpper->Calc();
+ rThis.Cut();
+ // --> OD 2005-02-23 #b6229852#
+ // optimization: format section, if its size is invalidated and if it's
+ // the new parent of moved backward frame.
+ bool bFormatSect( false );
+ // <--
+ if( bUnlock )
+ {
+ pSect->ColUnlock();
+ if( pSect->HasFollow() != bFollow )
+ {
+ pSect->InvalidateSize();
+ // --> OD 2005-02-23 #b6229852# - optimization
+ if ( pSect == pNewUpper )
+ bFormatSect = true;
+ // <--
+ }
+ }
+
+ rThis.Paste( pNewUpper );
+ // --> OD 2005-02-23 #b6229852# - optimization
+ if ( bFormatSect )
+ pSect->Calc();
+ // <--
+
+ SwPageFrm *pNewPage = rThis.FindPageFrm();
+ if( pNewPage != pOldPage )
+ {
+ rThis.Prepare( PREP_BOSS_CHGD, (const void*)pOldPage, FALSE );
+ ViewShell *pSh = rThis.GetShell();
+ if ( pSh && !pSh->Imp()->IsUpdateExpFlds() )
+ pSh->GetDoc()->SetNewFldLst(true); //Wird von CalcLayout() hinterher eledigt!
+
+ pNewPage->InvalidateSpelling();
+ pNewPage->InvalidateSmartTags(); // SMARTTAGS
+ pNewPage->InvalidateAutoCompleteWords();
+ pNewPage->InvalidateWordCount();
+
+ // OD 30.10.2002 #97265# - no <CheckPageDesc(..)> in online layout
+ if ( !pNewPage->GetFmt()->getIDocumentSettingAccess()->get(IDocumentSettingAccess::BROWSE_MODE) )
+ {
+ if ( bCheckPageDescs && pNewPage->GetNext() )
+ {
+ SwPageFrm* pStartPage = bCheckPageDescOfNextPage ?
+ pNewPage :
+ (SwPageFrm*)pNewPage->GetNext();
+ SwFrm::CheckPageDescs( pStartPage, FALSE);
+ }
+ else if ( rThis.GetAttrSet()->GetPageDesc().GetPageDesc() )
+ {
+ //Erste Seite wird etwa durch Ausblenden eines Bereiches leer
+ SwFrm::CheckPageDescs( (SwPageFrm*)pNewPage, FALSE);
+ }
+ }
+ }
+ }
+ return pNewUpper != 0;
+}
+
+/*************************************************************************
+|*
+|* SwFlowFrm::CastFlowFrm
+|*
+|* Ersterstellung MA 03. May. 95
+|* Letzte Aenderung AMA 02. Dec. 97
+|*
+|*************************************************************************/
+
+SwFlowFrm *SwFlowFrm::CastFlowFrm( SwFrm *pFrm )
+{
+ if ( pFrm->IsCntntFrm() )
+ return (SwCntntFrm*)pFrm;
+ if ( pFrm->IsTabFrm() )
+ return (SwTabFrm*)pFrm;
+ if ( pFrm->IsSctFrm() )
+ return (SwSectionFrm*)pFrm;
+ return 0;
+}
+
+const SwFlowFrm *SwFlowFrm::CastFlowFrm( const SwFrm *pFrm )
+{
+ if ( pFrm->IsCntntFrm() )
+ return (SwCntntFrm*)pFrm;
+ if ( pFrm->IsTabFrm() )
+ return (SwTabFrm*)pFrm;
+ if ( pFrm->IsSctFrm() )
+ return (SwSectionFrm*)pFrm;
+ return 0;
+}