/************************************************************** * * 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 "pam.hxx" #include "swtable.hxx" #include "frame.hxx" #include "rootfrm.hxx" #include "pagefrm.hxx" #include "flyfrm.hxx" #include "viewsh.hxx" #include "doc.hxx" #include "viewimp.hxx" #include "viewopt.hxx" #include "dflyobj.hxx" #include "frmtool.hxx" #include "dcontact.hxx" #include #include #include #include #include #include #include #include #include #include #include #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" #include #include #include #include #include sal_Bool SwFlowFrm::bMoveBwdJump = sal_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 = sal_False; } /************************************************************************* |* |* SwFlowFrm::IsFollowLocked() |* return sal_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 |* |*************************************************************************/ sal_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 sal_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 sal_True; } while ( pFrm ); //Siehe IsFwdMoveAllowed() sal_Bool bRet = sal_False; if ( pFrm && pFrm->GetIndPrev() ) bRet = sal_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; sal_Bool bKeep; while ( sal_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 |* |*************************************************************************/ sal_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) sal_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 = sal_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 = sal_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 = sal_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 = sal_False; default: break; } } } } } return bKeep; } /************************************************************************* |* |* SwFlowFrm::BwdMoveNecessary() |* |* Ersterstellung MA 20. Jul. 94 |* Letzte Aenderung MA 02. May. 96 |* |*************************************************************************/ sal_uInt8 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! sal_uInt8 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 const SwSortedObjs &rObjs = *pPage->GetSortedObjs(); sal_uLong nIndex = ULONG_MAX; for ( sal_uInt16 i = 0; nRet < 3 && i < rObjs.Count(); ++i ) { // --> OD 2004-07-01 #i28701# - consider changed type of // 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(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. sal_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# // 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 { sal_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; } sal_Bool SwFlowFrm::PasteTree( SwFrm *pStart, SwLayoutFrm *pParent, SwFrm *pSibling, SwFrm *pOldParent ) { //returnt sal_True wenn in der Kette ein LayoutFrm steht. sal_Bool bRet = sal_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 = sal_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.getRootFrm()->GetCurrShell(); const SwViewImp *pImp = pSh ? pSh->Imp() : 0; const sal_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 sal_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( sal_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(); sal_Bool bOld = pTmpSct->IsCntntLocked(); pTmpSct->SetCntntLock( sal_True ); pTmpSct->Calc(); if( !bOld ) pTmpSct->SetCntntLock( sal_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 |* |*************************************************************************/ sal_Bool SwFlowFrm::IsAnFollow( const SwFlowFrm *pAssumed ) const { const SwFlowFrm *pFoll = this; do { if ( pAssumed == pFoll ) return sal_True; pFoll = pFoll->GetFollow(); } while ( pFoll ); return sal_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( sal_False, "Follow ist lost in Space." ); return 0; } SwSectionFrm* SwSectionFrm::FindMaster() const { ASSERT( IsFollow(), "SwSectionFrm::FindMaster(): !IsFollow" ); SwIterator aIter( *pSection->GetFmt() ); SwSectionFrm* pSect = aIter.First(); while ( pSect ) { if( pSect->GetFollow() == this ) return pSect; pSect = aIter.Next(); } ASSERT( sal_False, "Follow ist lost in Space." ); return 0; } SwTabFrm* SwTabFrm::FindMaster( bool bFirstMaster ) const { ASSERT( IsFollow(), "SwTabFrm::FindMaster(): !IsFollow" ); SwIterator aIter( *GetTable()->GetFrmFmt() ); SwTabFrm* pTab = aIter.First(); while ( pTab ) { 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; } pTab = aIter.Next(); } ASSERT( sal_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, sal_Bool bFwd, const SwFrm *pAnch ) const { //Ohne Fluss kein genuss... if ( !(IsInDocBody() || IsInFtn() || IsInFly()) ) return 0; const SwFrm *pLeaf = this; sal_Bool bFound = sal_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 = sal_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, sal_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 ); } sal_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; sal_uInt16 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 sal_Bool bOdd = nTmp ? ( nTmp % 2 ? sal_True : sal_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 sal_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. sal_Bool bNewPg = sal_False; //nur einmal eine neue Seite einfuegen. while ( sal_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(); const ViewShell *pSh = getRootFrm()->GetCurrShell(); // #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) !(pSh && pSh->GetViewOptions()->getBrowseMode() ) ) // <-- { 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 sal_uInt16 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 = sal_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, sal_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(), sal_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 sal_Bool bBody = IsInDocBody(); //Wenn ich aus dem DocBody komme //will ich auch im Body landen. const sal_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( sal_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 |* |*************************************************************************/ sal_Bool SwFlowFrm::IsPrevObjMove() const { //sal_True der FlowFrm soll auf einen Rahmen des Vorgaengers Ruecksicht nehmen // und fuer diesen ggf. Umbrechen. //!!!!!!!!!!!Hack!!!!!!!!!!! const ViewShell *pSh = rThis.getRootFrm()->GetCurrShell(); if( pSh && pSh->GetViewOptions()->getBrowseMode() ) return sal_False; SwFrm *pPre = rThis.FindPrev(); if ( pPre && pPre->GetDrawObjs() ) { ASSERT( SwFlowFrm::CastFlowFrm( pPre ), "new flowfrm?" ); if( SwFlowFrm::CastFlowFrm( pPre )->IsAnFollow( this ) ) return sal_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 to determine, if object has followed the // text flow to the next layout frame for ( sal_uInt16 i = 0; i < pPre->GetDrawObjs()->Count(); ++i ) { // --> OD 2004-07-01 #i28701# - consider changed type of // 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 sal_True; } } } // <-- } return sal_False; } /************************************************************************* |* |* sal_Bool SwFlowFrm::IsPageBreak() |* |* Beschreibung Wenn vor dem Frm ein harter Seitenumbruch steht UND |* es einen Vorgaenger auf der gleichen Seite gibt, wird sal_True |* zurueckgeliefert (es muss ein PageBreak erzeugt werden) sal_False sonst. |* Wenn in bAct sal_True uebergeben wird, gibt die Funktion dann sal_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 |* |*************************************************************************/ sal_Bool SwFlowFrm::IsPageBreak( sal_Bool bAct ) const { if ( !IsFollow() && rThis.IsInDocBody() && ( !rThis.IsInTab() || ( rThis.IsTabFrm() && !rThis.GetUpper()->IsInTab() ) ) ) // i66968 { const ViewShell *pSh = rThis.getRootFrm()->GetCurrShell(); if( pSh && pSh->GetViewOptions()->getBrowseMode() ) return sal_False; 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 sal_False; } else { if ( rThis.FindPageFrm() != pPrev->FindPageFrm() ) return sal_False; } const SvxBreak eBreak = pSet->GetBreak().GetBreak(); if ( eBreak == SVX_BREAK_PAGE_BEFORE || eBreak == SVX_BREAK_PAGE_BOTH ) return sal_True; else { const SvxBreak &ePrB = pPrev->GetAttrSet()->GetBreak().GetBreak(); if ( ePrB == SVX_BREAK_PAGE_AFTER || ePrB == SVX_BREAK_PAGE_BOTH || pSet->GetPageDesc().GetPageDesc() ) return sal_True; } } } return sal_False; } /************************************************************************* |* |* sal_Bool SwFlowFrm::IsColBreak() |* |* Beschreibung Wenn vor dem Frm ein harter Spaltenumbruch steht UND |* es einen Vorgaenger in der gleichen Spalte gibt, wird sal_True |* zurueckgeliefert (es muss ein PageBreak erzeugt werden) sal_False sonst. |* Wenn in bAct sal_True uebergeben wird, gibt die Funktion dann sal_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 |* |*************************************************************************/ sal_Bool SwFlowFrm::IsColBreak( sal_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 sal_False; } else { if ( pCol != pPrev->FindColFrm() ) return sal_False; } const SvxBreak eBreak = rThis.GetAttrSet()->GetBreak().GetBreak(); if ( eBreak == SVX_BREAK_COLUMN_BEFORE || eBreak == SVX_BREAK_COLUMN_BOTH ) return sal_True; else { const SvxBreak &ePrB = pPrev->GetAttrSet()->GetBreak().GetBreak(); if ( ePrB == SVX_BREAK_COLUMN_AFTER || ePrB == SVX_BREAK_COLUMN_BOTH ) return sal_True; } } } } return sal_False; } sal_Bool SwFlowFrm::HasParaSpaceAtPages( sal_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 sal_True; if( pTmp->IsPageFrm() ) return ( pTmp->GetPrev() && !IsPageBreak(sal_True) ) ? sal_False : sal_True; if( pTmp->IsColumnFrm() && pTmp->GetPrev() ) return IsColBreak( sal_True ); if( pTmp->IsSctFrm() && ( !bSct || pTmp->GetPrev() ) ) return sal_False; pTmp = pTmp->GetUpper(); } ASSERT( sal_False, "HasParaSpaceAtPages: Where's my page?" ); return sal_False; } if( !rThis.IsInDocBody() || ( rThis.IsInTab() && !rThis.IsTabFrm()) || IsPageBreak( sal_True ) || ( rThis.FindColFrm() && IsColBreak( sal_True ) ) ) return sal_True; const SwFrm* pTmp = rThis.FindColFrm(); if( pTmp ) { if( pTmp->GetPrev() ) return sal_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(pPrevFrm)->IsHiddenNow() ) || ( pPrevFrm->IsSctFrm() && !static_cast(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(rThis.FindFtnFrm()->GetPrev()); if ( pPrevFtnFrm ) { pPrevFrm = pPrevFtnFrm->GetLastLower(); // Skip hidden paragraphs and empty sections while ( pPrevFrm && ( ( pPrevFrm->IsTxtFrm() && static_cast(pPrevFrm)->IsHiddenNow() ) || ( pPrevFrm->IsSctFrm() && !static_cast(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(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(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 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(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(rThis).GetLineSpace( true ); } else { nAdd = Max( nAdd, static_cast(rThis).GetLineSpace( true ) ); } // <-- } nUpper += nAdd; } } else { nUpper = Max( static_cast(nPrevLowerSpace), static_cast(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(rThis).GetLineSpace( true ); } else { nAdd = Max( nAdd, static_cast(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 // to method , if parameter is set. // Note: parameter is set, if method is called from nUpper += pAttrs->GetTopLine( rThis, (pPr ? pPrevFrm : 0L) ); // OD 2004-03-12 #i11860# - consider value of new parameter <_bConsiderGrid> // and use new method //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, " - 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 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(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; } /************************************************************************* |* |* sal_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 |* |*************************************************************************/ sal_Bool SwFlowFrm::CheckMoveFwd( sal_Bool &rbMakePage, sal_Bool bKeep, sal_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() ) { sal_Bool bMove = sal_False; const SwSectionFrm *pSct = rThis.FindSctFrm(); if( pSct && !pSct->GetValidSizeFlag() ) { const SwSectionFrm* pNxtSct = pNxt->FindSctFrm(); if( pNxtSct && pSct->IsAnFollow( pNxtSct ) ) bMove = sal_True; } else bMove = sal_True; if( bMove ) { //Keep together with the following frame MoveFwd( rbMakePage, sal_False ); return sal_True; } } } sal_Bool bMovedFwd = sal_False; if ( rThis.GetIndPrev() ) { if ( IsPrevObjMove() ) //Auf Objekte des Prev Ruecksicht nehmen? { bMovedFwd = sal_True; if ( !MoveFwd( rbMakePage, sal_False ) ) rbMakePage = sal_False; } else { if ( IsPageBreak( sal_False ) ) { while ( MoveFwd( rbMakePage, sal_True ) ) /* do nothing */; rbMakePage = sal_False; bMovedFwd = sal_True; } else if ( IsColBreak ( sal_False ) ) { const SwPageFrm *pPage = rThis.FindPageFrm(); SwFrm *pCol = rThis.FindColFrm(); do { MoveFwd( rbMakePage, sal_False ); SwFrm *pTmp = rThis.FindColFrm(); if( pTmp != pCol ) { bMovedFwd = sal_True; pCol = pTmp; } else break; } while ( IsColBreak( sal_False ) ); if ( pPage != rThis.FindPageFrm() ) rbMakePage = sal_False; } } } return bMovedFwd; } /************************************************************************* |* |* sal_Bool SwFlowFrm::MoveFwd() |* |* Beschreibung Returnwert sagt, ob der Frm die Seite gewechselt hat. |* Ersterstellung MA 05. Dec. 96 |* Letzte Aenderung MA 05. Dec. 96 |* |*************************************************************************/ sal_Bool SwFlowFrm::MoveFwd( sal_Bool bMakePage, sal_Bool bPageBreak, sal_Bool bMoveAlways ) { //!!!!MoveFtnCntFwd muss ggf. mitgepflegt werden. SwFtnBossFrm *pOldBoss = rThis.FindFtnBossFrm(); if ( rThis.IsInFtn() ) return ((SwCntntFrm&)rThis).MoveFtnCntFwd( bMakePage, pOldBoss ); if( !IsFwdMoveAllowed() && !bMoveAlways ) { sal_Bool bNoFwd = sal_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(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 = sal_False; } if( bNoFwd ) { //Fuer PageBreak ist das Moven erlaubt, wenn der Frm nicht //bereits der erste der Seite ist. if ( !bPageBreak ) return sal_False; const SwFrm *pCol = rThis.FindColFrm(); if ( !pCol || !pCol->GetPrev() ) return sal_False; } } sal_Bool bSamePage = sal_True; SwLayoutFrm *pNewUpper = rThis.GetLeaf( bMakePage ? MAKEPAGE_INSERT : MAKEPAGE_NONE, sal_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(); sal_Bool bUnlock = sal_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(); sal_Bool bBossChg = pNewBoss != pOldBoss; pNewBoss = pNewBoss->FindFtnBossFrm( sal_True ); pOldBoss = pOldBoss->FindFtnBossFrm( sal_True ); SwPageFrm* pNewPage = pOldPage; // First, we move the footnotes. sal_Bool bFtnMoved = sal_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, sal_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(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, sal_False ); if( !bSamePage ) { ViewShell *pSh = rThis.getRootFrm()->GetCurrShell(); 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 in online layout const ViewShell *pSh = rThis.getRootFrm()->GetCurrShell(); if ( !( pSh && pSh->GetViewOptions()->getBrowseMode() ) ) { // --> 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, sal_False ); } // <-- } } return bSamePage; } /************************************************************************* |* |* sal_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 |* |*************************************************************************/ sal_Bool SwFlowFrm::MoveBwd( sal_Bool &rbReformat ) { SwFlowFrm::SetMoveBwdJump( sal_False ); SwFtnFrm* pFtn = rThis.FindFtnFrm(); if ( pFtn && pFtn->IsBackMoveLocked() ) return sal_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 sal_False; } else if ( pUpperFrm->IsColumnFrm() && pUpperFrm->IsInSct() ) { break; } pUpperFrm = pUpperFrm->GetUpper(); } } // <-- SwFtnBossFrm * pOldBoss = rThis.FindFtnBossFrm(); SwPageFrm * const pOldPage = pOldBoss->FindPageFrm(); SwLayoutFrm *pNewUpper = 0; sal_Bool bCheckPageDescs = sal_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( sal_True ); SwFtnBossFrm *pRefBoss = pRef->FindFtnBossFrm( !bEndnote ); if ( pOldBoss != pRefBoss && // OD 08.11.2002 #104840# - use ( !bEndnote || pRefBoss->IsBefore( pOldBoss ) ) ) pNewUpper = rThis.GetLeaf( MAKEPAGE_FTN, sal_False ); } else if ( IsPageBreak( sal_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, sal_False ); // --> OD 2006-05-08 #i53139# // Now 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 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 // will find the last content of page 1, but // returns new upper on page 2. if ( pNewUpper->Lower() ) { SwLayoutFrm* pNewNextUpper = pNewUpper->GetLeaf( MAKEPAGE_NONE, sal_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 = sal_True; } } } } else if ( IsColBreak( sal_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, sal_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 // --> OD 2006-11-02 #i71065# - check else if ( pNewUpper && !SwFlowFrm::IsMoveBwdJump() ) // <-- { // Now 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, sal_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(); sal_Bool bGoOn = sal_True; sal_Bool bJump = sal_False; do { if ( pCol->GetPrev() ) pCol = pCol->GetPrev(); else { bGoOn = sal_False; pCol = rThis.GetLeaf( MAKEPAGE_NONE, sal_False ); } if ( pCol ) { // ColumnFrms jetzt mit BodyFrm SwLayoutFrm* pColBody = pCol->IsColumnFrm() ? (SwLayoutFrm*)((SwLayoutFrm*)pCol)->Lower() : (SwLayoutFrm*)pCol; if ( pColBody->ContainsCntnt() ) { bGoOn = sal_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 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, sal_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 = sal_True; // also eine uebersprungen pNewUpper = pColBody; // Diese leere Spalte kommt in Frage, // trotzdem weitersuchen } } } while( bGoOn ); if( bJump ) SwFlowFrm::SetMoveBwdJump( sal_True ); } } else //Keine Breaks also kann ich zurueckfliessen pNewUpper = rThis.GetLeaf( MAKEPAGE_NONE, sal_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(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(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, sal_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, sal_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( sal_True ); delete pSectFrm; rThis.bValidPos = sal_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(pIndNext)->GetSection() ) { SwFrm* pTmp = static_cast(pIndNext)->ContainsAny(); if ( pTmp ) { pIndNext = pTmp; break; } } pIndNext = pIndNext->GetIndNext(); } ASSERT( !pIndNext || pIndNext->ISA(SwTxtFrm), " - 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, sal_True ); // --> OD 2007-01-10 #i73194# - make code robust ASSERT( pNextNewUpper, " - missing next new upper" ); if ( pNextNewUpper && ( pNextNewUpper == rThis.GetUpper() || pNextNewUpper->GetType() != rThis.GetUpper()->GetType() ) ) // <-- { pNewUpper = 0L; #if OSL_DEBUG_LEVEL > 1 ASSERT( false, " - layout loop control for layout action applied!" ); #endif } } // <-- ASSERT( pNewUpper != rThis.GetUpper(), " - 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, 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, sal_True ); pSct->Paste( pNewUpper ); pSct->Init(); pNewUpper = pSct; pSct->SimpleFormat(); } } } sal_Bool bUnlock = sal_False; sal_Bool bFollow = sal_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, sal_False ); ViewShell *pSh = rThis.getRootFrm()->GetCurrShell(); 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 in online layout if ( !( pSh && pSh->GetViewOptions()->getBrowseMode() ) ) { if ( bCheckPageDescs && pNewPage->GetNext() ) { SwPageFrm* pStartPage = bCheckPageDescOfNextPage ? pNewPage : (SwPageFrm*)pNewPage->GetNext(); SwFrm::CheckPageDescs( pStartPage, sal_False); } else if ( rThis.GetAttrSet()->GetPageDesc().GetPageDesc() ) { //Erste Seite wird etwa durch Ausblenden eines Bereiches leer SwFrm::CheckPageDescs( (SwPageFrm*)pNewPage, sal_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; }