xref: /trunk/main/sw/source/core/text/widorp.cxx (revision cf6516809c57e1bb0a940545cca99cdad54d4ce2)
1*efeef26fSAndrew Rist /**************************************************************
2cdf0e10cSrcweir  *
3*efeef26fSAndrew Rist  * Licensed to the Apache Software Foundation (ASF) under one
4*efeef26fSAndrew Rist  * or more contributor license agreements.  See the NOTICE file
5*efeef26fSAndrew Rist  * distributed with this work for additional information
6*efeef26fSAndrew Rist  * regarding copyright ownership.  The ASF licenses this file
7*efeef26fSAndrew Rist  * to you under the Apache License, Version 2.0 (the
8*efeef26fSAndrew Rist  * "License"); you may not use this file except in compliance
9*efeef26fSAndrew Rist  * with the License.  You may obtain a copy of the License at
10cdf0e10cSrcweir  *
11*efeef26fSAndrew Rist  *   http://www.apache.org/licenses/LICENSE-2.0
12cdf0e10cSrcweir  *
13*efeef26fSAndrew Rist  * Unless required by applicable law or agreed to in writing,
14*efeef26fSAndrew Rist  * software distributed under the License is distributed on an
15*efeef26fSAndrew Rist  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16*efeef26fSAndrew Rist  * KIND, either express or implied.  See the License for the
17*efeef26fSAndrew Rist  * specific language governing permissions and limitations
18*efeef26fSAndrew Rist  * under the License.
19cdf0e10cSrcweir  *
20*efeef26fSAndrew Rist  *************************************************************/
21*efeef26fSAndrew Rist 
22*efeef26fSAndrew Rist 
23cdf0e10cSrcweir 
24cdf0e10cSrcweir // MARKER(update_precomp.py): autogen include statement, do not remove
25cdf0e10cSrcweir #include "precompiled_sw.hxx"
26cdf0e10cSrcweir 
27cdf0e10cSrcweir 
28cdf0e10cSrcweir #include "hintids.hxx"
29cdf0e10cSrcweir 
30cdf0e10cSrcweir #include "layfrm.hxx"
31cdf0e10cSrcweir #include "ftnboss.hxx"
32cdf0e10cSrcweir #include "ndtxt.hxx"
33cdf0e10cSrcweir #include "paratr.hxx"
34cdf0e10cSrcweir #include <editeng/orphitem.hxx>
35cdf0e10cSrcweir #include <editeng/widwitem.hxx>
36cdf0e10cSrcweir #include <editeng/keepitem.hxx>
37cdf0e10cSrcweir #include <editeng/spltitem.hxx>
38cdf0e10cSrcweir #include <frmatr.hxx>
39cdf0e10cSrcweir #include <txtftn.hxx>
40cdf0e10cSrcweir #include <fmtftn.hxx>
41cdf0e10cSrcweir #include <rowfrm.hxx>
42cdf0e10cSrcweir 
43cdf0e10cSrcweir #include "txtcfg.hxx"
44cdf0e10cSrcweir #include "widorp.hxx"
45cdf0e10cSrcweir #include "txtfrm.hxx"
46cdf0e10cSrcweir #include "itrtxt.hxx"
47cdf0e10cSrcweir #include "sectfrm.hxx"  //SwSectionFrm
48cdf0e10cSrcweir #include "ftnfrm.hxx"
49cdf0e10cSrcweir 
50cdf0e10cSrcweir #undef WIDOWTWIPS
51cdf0e10cSrcweir 
52cdf0e10cSrcweir 
53cdf0e10cSrcweir /*************************************************************************
54cdf0e10cSrcweir  *                  inline IsNastyFollow()
55cdf0e10cSrcweir  *************************************************************************/
56cdf0e10cSrcweir // Ein Follow, der auf der selben Seite steht, wie sein Master ist nasty.
IsNastyFollow(const SwTxtFrm * pFrm)57cdf0e10cSrcweir inline sal_Bool IsNastyFollow( const SwTxtFrm *pFrm )
58cdf0e10cSrcweir {
59cdf0e10cSrcweir     ASSERT( !pFrm->IsFollow() || !pFrm->GetPrev() ||
60cdf0e10cSrcweir             ((const SwTxtFrm*)pFrm->GetPrev())->GetFollow() == pFrm,
61cdf0e10cSrcweir             "IsNastyFollow: Was ist denn hier los?" );
62cdf0e10cSrcweir     return  pFrm->IsFollow() && pFrm->GetPrev();
63cdf0e10cSrcweir }
64cdf0e10cSrcweir 
65cdf0e10cSrcweir /*************************************************************************
66cdf0e10cSrcweir  *                  SwTxtFrmBreak::SwTxtFrmBreak()
67cdf0e10cSrcweir  *************************************************************************/
68cdf0e10cSrcweir 
SwTxtFrmBreak(SwTxtFrm * pNewFrm,const SwTwips nRst)69cdf0e10cSrcweir SwTxtFrmBreak::SwTxtFrmBreak( SwTxtFrm *pNewFrm, const SwTwips nRst )
70cdf0e10cSrcweir     : nRstHeight(nRst), pFrm(pNewFrm)
71cdf0e10cSrcweir {
72cdf0e10cSrcweir     SWAP_IF_SWAPPED( pFrm )
73cdf0e10cSrcweir     SWRECTFN( pFrm )
74cdf0e10cSrcweir     nOrigin = (pFrm->*fnRect->fnGetPrtTop)();
75cdf0e10cSrcweir     SwSectionFrm* pSct;
76cdf0e10cSrcweir     bKeep = !pFrm->IsMoveable() || IsNastyFollow( pFrm ) ||
77cdf0e10cSrcweir             ( pFrm->IsInSct() && (pSct=pFrm->FindSctFrm())->Lower()->IsColumnFrm()
78cdf0e10cSrcweir               && !pSct->MoveAllowed( pFrm ) ) ||
79cdf0e10cSrcweir             !pFrm->GetTxtNode()->GetSwAttrSet().GetSplit().GetValue() ||
80cdf0e10cSrcweir             pFrm->GetTxtNode()->GetSwAttrSet().GetKeep().GetValue();
81cdf0e10cSrcweir     bBreak = sal_False;
82cdf0e10cSrcweir 
83cdf0e10cSrcweir     if( !nRstHeight && !pFrm->IsFollow() && pFrm->IsInFtn() && pFrm->HasPara() )
84cdf0e10cSrcweir     {
85cdf0e10cSrcweir         nRstHeight = pFrm->GetFtnFrmHeight();
86cdf0e10cSrcweir         nRstHeight += (pFrm->Prt().*fnRect->fnGetHeight)() -
87cdf0e10cSrcweir                       (pFrm->Frm().*fnRect->fnGetHeight)();
88cdf0e10cSrcweir         if( nRstHeight < 0 )
89cdf0e10cSrcweir             nRstHeight = 0;
90cdf0e10cSrcweir     }
91cdf0e10cSrcweir 
92cdf0e10cSrcweir     UNDO_SWAP( pFrm )
93cdf0e10cSrcweir }
94cdf0e10cSrcweir 
95cdf0e10cSrcweir /* BP 18.6.93: Widows.
96cdf0e10cSrcweir  * Im Gegensatz zur ersten Implementierung werden die Widows nicht
97cdf0e10cSrcweir  * mehr vorausschauend berechnet, sondern erst beim Formatieren des
98cdf0e10cSrcweir  * gesplitteten Follows festgestellt. Im Master faellt die Widows-
99cdf0e10cSrcweir  * Berechnung also generell weg (nWidows wird manipuliert).
100cdf0e10cSrcweir  * Wenn der Follow feststellt, dass die Widowsregel zutrifft,
101cdf0e10cSrcweir  * verschickt er an seinen Vorgaenger ein Prepare.
102cdf0e10cSrcweir  * Ein besonderes Problem ergibt sich, wenn die Widows zuschlagen,
103cdf0e10cSrcweir  * aber im Master noch ein paar Zeilen zur Verfuegung stehen.
104cdf0e10cSrcweir  *
105cdf0e10cSrcweir  */
106cdf0e10cSrcweir 
107cdf0e10cSrcweir /*************************************************************************
108cdf0e10cSrcweir  *                  SwTxtFrmBreak::IsInside()
109cdf0e10cSrcweir  *************************************************************************/
110cdf0e10cSrcweir 
111cdf0e10cSrcweir /* BP(22.07.92): Berechnung von Witwen und Waisen.
112cdf0e10cSrcweir  * Die Methode liefert sal_True zurueck, wenn eine dieser Regelung zutrifft.
113cdf0e10cSrcweir  *
114cdf0e10cSrcweir  * Eine Schwierigkeit gibt es im Zusammenhang mit Widows und
115cdf0e10cSrcweir  * unterschiedlichen Formaten zwischen Master- und Folgeframes:
116cdf0e10cSrcweir  * Beispiel: Wenn die erste Spalte 3cm und die zweite 4cm breit ist
117cdf0e10cSrcweir  * und Widows auf sagen wir 3 gesetzt ist, so ist erst bei der Formatierung
118cdf0e10cSrcweir  * des Follows entscheidbar, ob die Widowsbedingung einhaltbar ist oder
119cdf0e10cSrcweir  * nicht. Leider ist davon abhaengig, ob der Absatz als Ganzes auf die
120cdf0e10cSrcweir  * naechste Seite rutscht.
121cdf0e10cSrcweir  */
122cdf0e10cSrcweir 
IsInside(SwTxtMargin & rLine) const123cdf0e10cSrcweir sal_Bool SwTxtFrmBreak::IsInside( SwTxtMargin &rLine ) const
124cdf0e10cSrcweir {
125cdf0e10cSrcweir     sal_Bool bFit = sal_False;
126cdf0e10cSrcweir 
127cdf0e10cSrcweir     SWAP_IF_SWAPPED( pFrm )
128cdf0e10cSrcweir     SWRECTFN( pFrm )
129cdf0e10cSrcweir     // nOrigin is an absolut value, rLine referes to the swapped situation.
130cdf0e10cSrcweir 
131cdf0e10cSrcweir     SwTwips nTmpY;
132cdf0e10cSrcweir     if ( pFrm->IsVertical() )
133cdf0e10cSrcweir         nTmpY = pFrm->SwitchHorizontalToVertical( rLine.Y() + rLine.GetLineHeight() );
134cdf0e10cSrcweir     else
135cdf0e10cSrcweir         nTmpY = rLine.Y() + rLine.GetLineHeight();
136cdf0e10cSrcweir 
137cdf0e10cSrcweir     SwTwips nLineHeight = (*fnRect->fnYDiff)( nTmpY , nOrigin );
138cdf0e10cSrcweir 
139cdf0e10cSrcweir     // 7455 und 6114: Raum fuer die Umrandung unten einkalkulieren.
140cdf0e10cSrcweir     nLineHeight += (pFrm->*fnRect->fnGetBottomMargin)();
141cdf0e10cSrcweir 
142cdf0e10cSrcweir     if( nRstHeight )
143cdf0e10cSrcweir         bFit = nRstHeight >= nLineHeight;
144cdf0e10cSrcweir     else
145cdf0e10cSrcweir     {
146cdf0e10cSrcweir         // Der Frm besitzt eine Hoehe, mit der er auf die Seite passt.
147cdf0e10cSrcweir         SwTwips nHeight =
148cdf0e10cSrcweir             (*fnRect->fnYDiff)( (pFrm->GetUpper()->*fnRect->fnGetPrtBottom)(), nOrigin );
149cdf0e10cSrcweir         // Wenn sich alles innerhalb des bestehenden Frames abspielt,
150cdf0e10cSrcweir         // ist das Ergebnis sal_True;
151cdf0e10cSrcweir         bFit = nHeight >= nLineHeight;
152cdf0e10cSrcweir 
153cdf0e10cSrcweir         // --> OD #i103292#
154cdf0e10cSrcweir         if ( !bFit )
155cdf0e10cSrcweir         {
156cdf0e10cSrcweir             if ( rLine.GetNext() &&
157cdf0e10cSrcweir                  pFrm->IsInTab() && !pFrm->GetFollow() && !pFrm->GetIndNext() )
158cdf0e10cSrcweir             {
159cdf0e10cSrcweir                 // add additional space taken as lower space as last content in a table
160cdf0e10cSrcweir                 // for all text lines except the last one.
161cdf0e10cSrcweir                 nHeight += pFrm->CalcAddLowerSpaceAsLastInTableCell();
162cdf0e10cSrcweir                 bFit = nHeight >= nLineHeight;
163cdf0e10cSrcweir             }
164cdf0e10cSrcweir         }
165cdf0e10cSrcweir         // <--
166cdf0e10cSrcweir         if( !bFit )
167cdf0e10cSrcweir         {
168cdf0e10cSrcweir             // Die LineHeight sprengt die aktuelle Frm-Hoehe.
169cdf0e10cSrcweir             // Nun rufen wir ein Probe-Grow, um zu ermitteln, ob der
170cdf0e10cSrcweir             // Frame um den gewuenschten Bereich wachsen wuerde.
171cdf0e10cSrcweir             nHeight += pFrm->GrowTst( LONG_MAX );
172cdf0e10cSrcweir 
173cdf0e10cSrcweir             // Das Grow() returnt die Hoehe, um die der Upper des TxtFrm
174cdf0e10cSrcweir             // den TxtFrm wachsen lassen wuerde.
175cdf0e10cSrcweir             // Der TxtFrm selbst darf wachsen wie er will.
176cdf0e10cSrcweir             bFit = nHeight >= nLineHeight;
177cdf0e10cSrcweir         }
178cdf0e10cSrcweir     }
179cdf0e10cSrcweir 
180cdf0e10cSrcweir     UNDO_SWAP( pFrm );
181cdf0e10cSrcweir 
182cdf0e10cSrcweir     return bFit;
183cdf0e10cSrcweir }
184cdf0e10cSrcweir 
185cdf0e10cSrcweir /*************************************************************************
186cdf0e10cSrcweir  *                  SwTxtFrmBreak::IsBreakNow()
187cdf0e10cSrcweir  *************************************************************************/
188cdf0e10cSrcweir 
IsBreakNow(SwTxtMargin & rLine)189cdf0e10cSrcweir sal_Bool SwTxtFrmBreak::IsBreakNow( SwTxtMargin &rLine )
190cdf0e10cSrcweir {
191cdf0e10cSrcweir     SWAP_IF_SWAPPED( pFrm )
192cdf0e10cSrcweir 
193cdf0e10cSrcweir     // bKeep ist staerker als IsBreakNow()
194cdf0e10cSrcweir     // Ist noch genug Platz ?
195cdf0e10cSrcweir     if( bKeep || IsInside( rLine ) )
196cdf0e10cSrcweir         bBreak = sal_False;
197cdf0e10cSrcweir     else
198cdf0e10cSrcweir     {
199cdf0e10cSrcweir         /* Diese Klasse geht davon aus, dass der SwTxtMargin von Top nach Bottom
200cdf0e10cSrcweir          * durchgearbeitet wird. Aus Performancegruenden wird in folgenden
201cdf0e10cSrcweir          * Faellen der Laden fuer das weitere Aufspalten dicht gemacht:
202cdf0e10cSrcweir          * Wenn eine einzige Zeile nicht mehr passt.
203cdf0e10cSrcweir          * Sonderfall: bei DummyPortions ist LineNr == 1, obwohl wir splitten
204cdf0e10cSrcweir          * wollen.
205cdf0e10cSrcweir          */
206cdf0e10cSrcweir         // 6010: DropLines mit einbeziehen
207cdf0e10cSrcweir 
208cdf0e10cSrcweir         sal_Bool bFirstLine = 1 == rLine.GetLineNr() && !rLine.GetPrev();
209cdf0e10cSrcweir         bBreak = sal_True;
210cdf0e10cSrcweir         if( ( bFirstLine && pFrm->GetIndPrev() )
211cdf0e10cSrcweir             || ( rLine.GetLineNr() <= rLine.GetDropLines() ) )
212cdf0e10cSrcweir         {
213cdf0e10cSrcweir             bKeep = sal_True;
214cdf0e10cSrcweir             bBreak = sal_False;
215cdf0e10cSrcweir         }
216cdf0e10cSrcweir         else if(bFirstLine && pFrm->IsInFtn() && !pFrm->FindFtnFrm()->GetPrev())
217cdf0e10cSrcweir         {
218cdf0e10cSrcweir             SwLayoutFrm* pTmp = pFrm->FindFtnBossFrm()->FindBodyCont();
219cdf0e10cSrcweir             if( !pTmp || !pTmp->Lower() )
220cdf0e10cSrcweir                 bBreak = sal_False;
221cdf0e10cSrcweir         }
222cdf0e10cSrcweir     }
223cdf0e10cSrcweir 
224cdf0e10cSrcweir     UNDO_SWAP( pFrm )
225cdf0e10cSrcweir 
226cdf0e10cSrcweir     return bBreak;
227cdf0e10cSrcweir }
228cdf0e10cSrcweir 
229cdf0e10cSrcweir // OD 2004-02-27 #106629# - no longer inline
SetRstHeight(const SwTxtMargin & rLine)230cdf0e10cSrcweir void SwTxtFrmBreak::SetRstHeight( const SwTxtMargin &rLine )
231cdf0e10cSrcweir {
232cdf0e10cSrcweir     // OD, FME 2004-02-27 #106629# - consider bottom margin
233cdf0e10cSrcweir     SWRECTFN( pFrm )
234cdf0e10cSrcweir 
235cdf0e10cSrcweir     nRstHeight = (pFrm->*fnRect->fnGetBottomMargin)();
236cdf0e10cSrcweir 
237cdf0e10cSrcweir     if ( bVert )
238cdf0e10cSrcweir     //Badaa: 2008-04-18 * Support for Classical Mongolian Script (SCMS) joint with Jiayanmin
239cdf0e10cSrcweir     {
240cdf0e10cSrcweir         if ( pFrm->IsVertLR() )
241cdf0e10cSrcweir             nRstHeight = (*fnRect->fnYDiff)( pFrm->SwitchHorizontalToVertical( rLine.Y() ) , nOrigin );
242cdf0e10cSrcweir         else
243cdf0e10cSrcweir             nRstHeight += nOrigin - pFrm->SwitchHorizontalToVertical( rLine.Y() );
244cdf0e10cSrcweir     }
245cdf0e10cSrcweir     else
246cdf0e10cSrcweir         nRstHeight += rLine.Y() - nOrigin;
247cdf0e10cSrcweir }
248cdf0e10cSrcweir 
249cdf0e10cSrcweir /*************************************************************************
250cdf0e10cSrcweir  *                  WidowsAndOrphans::WidowsAndOrphans()
251cdf0e10cSrcweir  *************************************************************************/
252cdf0e10cSrcweir 
WidowsAndOrphans(SwTxtFrm * pNewFrm,const SwTwips nRst,sal_Bool bChkKeep)253cdf0e10cSrcweir WidowsAndOrphans::WidowsAndOrphans( SwTxtFrm *pNewFrm, const SwTwips nRst,
254cdf0e10cSrcweir     sal_Bool bChkKeep   )
255cdf0e10cSrcweir     : SwTxtFrmBreak( pNewFrm, nRst ), nWidLines( 0 ), nOrphLines( 0 )
256cdf0e10cSrcweir {
257cdf0e10cSrcweir     SWAP_IF_SWAPPED( pFrm )
258cdf0e10cSrcweir 
259cdf0e10cSrcweir     if( bKeep )
260cdf0e10cSrcweir     {
261cdf0e10cSrcweir         // 5652: bei Absaetzen, die zusammengehalten werden sollen und
262cdf0e10cSrcweir         // groesser sind als die Seite wird bKeep aufgehoben.
263cdf0e10cSrcweir         if( bChkKeep && !pFrm->GetPrev() && !pFrm->IsInFtn() &&
264cdf0e10cSrcweir             pFrm->IsMoveable() &&
265cdf0e10cSrcweir             ( !pFrm->IsInSct() || pFrm->FindSctFrm()->MoveAllowed(pFrm) ) )
266cdf0e10cSrcweir             bKeep = sal_False;
267cdf0e10cSrcweir         //Auch bei gesetztem Keep muessen Orphans beachtet werden,
268cdf0e10cSrcweir         //z.B. bei verketteten Rahmen erhaelt ein Follow im letzten Rahmen ein Keep,
269cdf0e10cSrcweir         //da er nicht (vorwaerts) Moveable ist,
270cdf0e10cSrcweir         //er darf aber trotzdem vom Master Zeilen anfordern wg. der Orphanregel.
271cdf0e10cSrcweir         if( pFrm->IsFollow() )
272cdf0e10cSrcweir             nWidLines = pFrm->GetTxtNode()->GetSwAttrSet().GetWidows().GetValue();
273cdf0e10cSrcweir     }
274cdf0e10cSrcweir     else
275cdf0e10cSrcweir     {
276cdf0e10cSrcweir         const SwAttrSet& rSet = pFrm->GetTxtNode()->GetSwAttrSet();
277cdf0e10cSrcweir         const SvxOrphansItem  &rOrph = rSet.GetOrphans();
278cdf0e10cSrcweir         if ( rOrph.GetValue() > 1 )
279cdf0e10cSrcweir             nOrphLines = rOrph.GetValue();
280cdf0e10cSrcweir         if ( pFrm->IsFollow() )
281cdf0e10cSrcweir             nWidLines = rSet.GetWidows().GetValue();
282cdf0e10cSrcweir 
283cdf0e10cSrcweir     }
284cdf0e10cSrcweir 
285cdf0e10cSrcweir     if ( bKeep || nWidLines || nOrphLines )
286cdf0e10cSrcweir     {
287cdf0e10cSrcweir         bool bResetFlags = false;
288cdf0e10cSrcweir 
289cdf0e10cSrcweir         if ( pFrm->IsInTab() )
290cdf0e10cSrcweir         {
291cdf0e10cSrcweir             // For compatibility reasons, we disable Keep/Widows/Orphans
292cdf0e10cSrcweir             // inside splittable row frames:
293cdf0e10cSrcweir             if ( pFrm->GetNextCellLeaf( MAKEPAGE_NONE ) || pFrm->IsInFollowFlowRow() )
294cdf0e10cSrcweir             {
295cdf0e10cSrcweir                 const SwFrm* pTmpFrm = pFrm->GetUpper();
296cdf0e10cSrcweir                 while ( !pTmpFrm->IsRowFrm() )
297cdf0e10cSrcweir                     pTmpFrm = pTmpFrm->GetUpper();
298cdf0e10cSrcweir                 if ( static_cast<const SwRowFrm*>(pTmpFrm)->IsRowSplitAllowed() )
299cdf0e10cSrcweir                     bResetFlags = true;
300cdf0e10cSrcweir             }
301cdf0e10cSrcweir         }
302cdf0e10cSrcweir 
303cdf0e10cSrcweir         if( pFrm->IsInFtn() && !pFrm->GetIndPrev() )
304cdf0e10cSrcweir         {
305cdf0e10cSrcweir             // Innerhalb von Fussnoten gibt es gute Gruende, das Keep-Attribut und
306cdf0e10cSrcweir             // die Widows/Orphans abzuschalten.
307cdf0e10cSrcweir             SwFtnFrm *pFtn = pFrm->FindFtnFrm();
308cdf0e10cSrcweir             sal_Bool bFt = !pFtn->GetAttr()->GetFtn().IsEndNote();
309cdf0e10cSrcweir             if( !pFtn->GetPrev() &&
310cdf0e10cSrcweir                 pFtn->FindFtnBossFrm( bFt ) != pFtn->GetRef()->FindFtnBossFrm( bFt )
311cdf0e10cSrcweir                 && ( !pFrm->IsInSct() || pFrm->FindSctFrm()->MoveAllowed(pFrm) ) )
312cdf0e10cSrcweir             {
313cdf0e10cSrcweir                 bResetFlags = true;
314cdf0e10cSrcweir             }
315cdf0e10cSrcweir         }
316cdf0e10cSrcweir 
317cdf0e10cSrcweir         if ( bResetFlags )
318cdf0e10cSrcweir         {
319cdf0e10cSrcweir             bKeep = sal_False;
320cdf0e10cSrcweir             nOrphLines = 0;
321cdf0e10cSrcweir             nWidLines = 0;
322cdf0e10cSrcweir         }
323cdf0e10cSrcweir     }
324cdf0e10cSrcweir 
325cdf0e10cSrcweir     UNDO_SWAP( pFrm )
326cdf0e10cSrcweir }
327cdf0e10cSrcweir 
328cdf0e10cSrcweir /*************************************************************************
329cdf0e10cSrcweir  *                  WidowsAndOrphans::FindBreak()
330cdf0e10cSrcweir  *************************************************************************/
331cdf0e10cSrcweir 
332cdf0e10cSrcweir /* Die Find*-Methoden suchen nicht nur, sondern stellen den SwTxtMargin auf
333cdf0e10cSrcweir  * die Zeile ein, wo der Absatz gebrochen werden soll und kuerzen ihn dort.
334cdf0e10cSrcweir  * FindBreak()
335cdf0e10cSrcweir  */
336cdf0e10cSrcweir 
FindBreak(SwTxtFrm * pFrame,SwTxtMargin & rLine,sal_Bool bHasToFit)337cdf0e10cSrcweir sal_Bool WidowsAndOrphans::FindBreak( SwTxtFrm *pFrame, SwTxtMargin &rLine,
338cdf0e10cSrcweir     sal_Bool bHasToFit )
339cdf0e10cSrcweir {
340cdf0e10cSrcweir     // OD 2004-02-25 #i16128# - Why member <pFrm> _*and*_ parameter <pFrame>??
341cdf0e10cSrcweir     // Thus, assertion on situation, that these are different to figure out why.
342cdf0e10cSrcweir     ASSERT( pFrm == pFrame, "<WidowsAndOrphans::FindBreak> - pFrm != pFrame" );
343cdf0e10cSrcweir 
344cdf0e10cSrcweir     SWAP_IF_SWAPPED( pFrm )
345cdf0e10cSrcweir 
346cdf0e10cSrcweir     sal_Bool bRet = sal_True;
347cdf0e10cSrcweir     MSHORT nOldOrphans = nOrphLines;
348cdf0e10cSrcweir     if( bHasToFit )
349cdf0e10cSrcweir         nOrphLines = 0;
350cdf0e10cSrcweir     rLine.Bottom();
351cdf0e10cSrcweir     // OD 2004-02-25 #i16128# - method renamed
352cdf0e10cSrcweir     if( !IsBreakNowWidAndOrp( rLine ) )
353cdf0e10cSrcweir         bRet = sal_False;
354cdf0e10cSrcweir     if( !FindWidows( pFrame, rLine ) )
355cdf0e10cSrcweir     {
356cdf0e10cSrcweir         sal_Bool bBack = sal_False;
357cdf0e10cSrcweir         // OD 2004-02-25 #i16128# - method renamed
358cdf0e10cSrcweir         while( IsBreakNowWidAndOrp( rLine ) )
359cdf0e10cSrcweir         {
360cdf0e10cSrcweir             if( rLine.PrevLine() )
361cdf0e10cSrcweir                 bBack = sal_True;
362cdf0e10cSrcweir             else
363cdf0e10cSrcweir                 break;
364cdf0e10cSrcweir         }
365cdf0e10cSrcweir         // Eigentlich werden bei HasToFit Schusterjungen (Orphans) nicht
366cdf0e10cSrcweir         // beruecksichtigt, wenn allerdings Dummy-Lines im Spiel sind und
367cdf0e10cSrcweir         // die Orphansregel verletzt wird, machen wir mal eine Ausnahme:
368cdf0e10cSrcweir         // Wir lassen einfach eine Dummyline zurueck und wandern mit dem Text
369cdf0e10cSrcweir         // komplett auf die naechste Seite/Spalte.
370cdf0e10cSrcweir         if( rLine.GetLineNr() <= nOldOrphans &&
371cdf0e10cSrcweir             rLine.GetInfo().GetParaPortion()->IsDummy() &&
372cdf0e10cSrcweir             ( ( bHasToFit && bRet ) || IsBreakNow( rLine ) ) )
373cdf0e10cSrcweir             rLine.Top();
374cdf0e10cSrcweir 
375cdf0e10cSrcweir         rLine.TruncLines( sal_True );
376cdf0e10cSrcweir         bRet = bBack;
377cdf0e10cSrcweir     }
378cdf0e10cSrcweir     nOrphLines = nOldOrphans;
379cdf0e10cSrcweir 
380cdf0e10cSrcweir     UNDO_SWAP( pFrm )
381cdf0e10cSrcweir 
382cdf0e10cSrcweir     return bRet;
383cdf0e10cSrcweir }
384cdf0e10cSrcweir 
385cdf0e10cSrcweir /*************************************************************************
386cdf0e10cSrcweir  *                  WidowsAndOrphans::FindWidows()
387cdf0e10cSrcweir  *************************************************************************/
388cdf0e10cSrcweir 
389cdf0e10cSrcweir /*  FindWidows positioniert den SwTxtMargin des Masters auf die umzubrechende
390cdf0e10cSrcweir  *  Zeile, indem der Follow formatiert und untersucht wird.
391cdf0e10cSrcweir  *  Liefert sal_True zurueck, wenn die Widows-Regelung in Kraft tritt,
392cdf0e10cSrcweir  *  d.h. der Absatz _zusammengehalten_ werden soll !
393cdf0e10cSrcweir  */
394cdf0e10cSrcweir 
FindWidows(SwTxtFrm * pFrame,SwTxtMargin & rLine)395cdf0e10cSrcweir sal_Bool WidowsAndOrphans::FindWidows( SwTxtFrm *pFrame, SwTxtMargin &rLine )
396cdf0e10cSrcweir {
397cdf0e10cSrcweir     ASSERT( ! pFrame->IsVertical() || ! pFrame->IsSwapped(),
398cdf0e10cSrcweir             "WidowsAndOrphans::FindWidows with swapped frame" )
399cdf0e10cSrcweir 
400cdf0e10cSrcweir     if( !nWidLines || !pFrame->IsFollow() )
401cdf0e10cSrcweir         return sal_False;
402cdf0e10cSrcweir 
403cdf0e10cSrcweir     rLine.Bottom();
404cdf0e10cSrcweir 
405cdf0e10cSrcweir     // Wir koennen noch was abzwacken
406cdf0e10cSrcweir     SwTxtFrm *pMaster = pFrame->FindMaster();
407cdf0e10cSrcweir     ASSERT(pMaster, "+WidowsAndOrphans::FindWidows: Widows in a master?");
408cdf0e10cSrcweir     if( !pMaster )
409cdf0e10cSrcweir         return sal_False;
410cdf0e10cSrcweir 
411cdf0e10cSrcweir     // 5156: Wenn die erste Zeile des Follows nicht passt, wird der Master
412cdf0e10cSrcweir     // wohl voll mit Dummies sein. In diesem Fall waere ein PREP_WIDOWS fatal.
413cdf0e10cSrcweir     if( pMaster->GetOfst() == pFrame->GetOfst() )
414cdf0e10cSrcweir         return sal_False;
415cdf0e10cSrcweir 
416cdf0e10cSrcweir     // Resthoehe des Masters
417cdf0e10cSrcweir     SWRECTFN( pFrame )
418cdf0e10cSrcweir 
419cdf0e10cSrcweir     const SwTwips nDocPrtTop = (pFrame->*fnRect->fnGetPrtTop)();
420cdf0e10cSrcweir     SwTwips nOldHeight;
421cdf0e10cSrcweir     SwTwips nTmpY = rLine.Y() + rLine.GetLineHeight();
422cdf0e10cSrcweir 
423cdf0e10cSrcweir     if ( bVert )
424cdf0e10cSrcweir     {
425cdf0e10cSrcweir         nTmpY = pFrame->SwitchHorizontalToVertical( nTmpY );
426cdf0e10cSrcweir         nOldHeight = -(pFrame->Prt().*fnRect->fnGetHeight)();
427cdf0e10cSrcweir     }
428cdf0e10cSrcweir     else
429cdf0e10cSrcweir         nOldHeight = (pFrame->Prt().*fnRect->fnGetHeight)();
430cdf0e10cSrcweir 
431cdf0e10cSrcweir     const SwTwips nChg = (*fnRect->fnYDiff)( nTmpY, nDocPrtTop + nOldHeight );
432cdf0e10cSrcweir 
433cdf0e10cSrcweir     // Unterhalb der Widows-Schwelle...
434cdf0e10cSrcweir     if( rLine.GetLineNr() >= nWidLines )
435cdf0e10cSrcweir     {
436cdf0e10cSrcweir         // 8575: Follow to Master I
437cdf0e10cSrcweir         // Wenn der Follow *waechst*, so besteht fuer den Master die Chance,
438cdf0e10cSrcweir         // Zeilen entgegenzunehmen, die er vor Kurzem gezwungen war an den
439cdf0e10cSrcweir         // Follow abzugeben: Prepare(Need); diese Abfrage unterhalb von nChg!
440cdf0e10cSrcweir         // (0W, 2O, 2M, 2F) + 1F = 3M, 2F
441cdf0e10cSrcweir         if( rLine.GetLineNr() > nWidLines && pFrame->IsJustWidow() )
442cdf0e10cSrcweir         {
443cdf0e10cSrcweir             // Wenn der Master gelockt ist, so hat er vermutlich gerade erst
444cdf0e10cSrcweir             // eine Zeile an uns abgegeben, diese geben nicht zurueck, nur
445cdf0e10cSrcweir             // weil bei uns daraus mehrere geworden sind (z.B. durch Rahmen).
446cdf0e10cSrcweir             if( !pMaster->IsLocked() && pMaster->GetUpper() )
447cdf0e10cSrcweir             {
448cdf0e10cSrcweir                 const SwTwips nTmpRstHeight = (pMaster->Frm().*fnRect->fnBottomDist)
449cdf0e10cSrcweir                             ( (pMaster->GetUpper()->*fnRect->fnGetPrtBottom)() );
450cdf0e10cSrcweir                 if ( nTmpRstHeight >=
451cdf0e10cSrcweir                      SwTwips(rLine.GetInfo().GetParaPortion()->Height() ) )
452cdf0e10cSrcweir                 {
453cdf0e10cSrcweir                     pMaster->Prepare( PREP_ADJUST_FRM );
454cdf0e10cSrcweir                     pMaster->_InvalidateSize();
455cdf0e10cSrcweir                     pMaster->InvalidatePage();
456cdf0e10cSrcweir                 }
457cdf0e10cSrcweir             }
458cdf0e10cSrcweir 
459cdf0e10cSrcweir             pFrame->SetJustWidow( sal_False );
460cdf0e10cSrcweir         }
461cdf0e10cSrcweir         return sal_False;
462cdf0e10cSrcweir     }
463cdf0e10cSrcweir 
464cdf0e10cSrcweir     // 8575: Follow to Master II
465cdf0e10cSrcweir     // Wenn der Follow *schrumpft*, so besteht fuer den Master die Chance,
466cdf0e10cSrcweir     // den kompletten Orphan zu inhalieren.
467cdf0e10cSrcweir     // (0W, 2O, 2M, 1F) - 1F = 3M, 0F     -> PREP_ADJUST_FRM
468cdf0e10cSrcweir     // (0W, 2O, 3M, 2F) - 1F = 2M, 2F     -> PREP_WIDOWS
469cdf0e10cSrcweir 
470cdf0e10cSrcweir     if( 0 > nChg && !pMaster->IsLocked() && pMaster->GetUpper() )
471cdf0e10cSrcweir     {
472cdf0e10cSrcweir         SwTwips nTmpRstHeight = (pMaster->Frm().*fnRect->fnBottomDist)
473cdf0e10cSrcweir                              ( (pMaster->GetUpper()->*fnRect->fnGetPrtBottom)() );
474cdf0e10cSrcweir         if( nTmpRstHeight >= SwTwips(rLine.GetInfo().GetParaPortion()->Height() ) )
475cdf0e10cSrcweir         {
476cdf0e10cSrcweir             pMaster->Prepare( PREP_ADJUST_FRM );
477cdf0e10cSrcweir             pMaster->_InvalidateSize();
478cdf0e10cSrcweir             pMaster->InvalidatePage();
479cdf0e10cSrcweir             pFrame->SetJustWidow( sal_False );
480cdf0e10cSrcweir             return sal_False;
481cdf0e10cSrcweir         }
482cdf0e10cSrcweir     }
483cdf0e10cSrcweir 
484cdf0e10cSrcweir     // Master to Follow
485cdf0e10cSrcweir     // Wenn der Follow nach seiner Formatierung weniger Zeilen enthaelt
486cdf0e10cSrcweir     // als Widows, so besteht noch die Chance, einige Zeilen des Masters
487cdf0e10cSrcweir     // abzuzwacken. Wenn dadurch die Orphans-Regel des Masters in Kraft
488cdf0e10cSrcweir     // tritt muss im CalcPrep() des Master-Frame der Frame so vergroessert
489cdf0e10cSrcweir     // werden, dass er nicht mehr auf seine urspruengliche Seite passt.
490cdf0e10cSrcweir     // Wenn er noch ein paar Zeilen entbehren kann, dann muss im CalcPrep()
491cdf0e10cSrcweir     // ein Shrink() erfolgen, der Follow mit dem Widows rutscht dann auf
492cdf0e10cSrcweir     // die Seite des Masters, haelt sich aber zusammen, so dass er (endlich)
493cdf0e10cSrcweir     // auf die naechste Seite rutscht. - So die Theorie!
494cdf0e10cSrcweir 
495cdf0e10cSrcweir 
496cdf0e10cSrcweir     // Wir fordern nur noch ein Zeile zur Zeit an, weil eine Zeile des Masters
497cdf0e10cSrcweir     // bei uns durchaus mehrere Zeilen ergeben koennten.
498cdf0e10cSrcweir     // Dafuer behaelt CalcFollow solange die Kontrolle, bis der Follow alle
499cdf0e10cSrcweir     // notwendigen Zeilen bekommen hat.
500cdf0e10cSrcweir     MSHORT nNeed = 1; // frueher: nWidLines - rLine.GetLineNr();
501cdf0e10cSrcweir 
502cdf0e10cSrcweir     // Special case: Master cannot give lines to follow
503cdf0e10cSrcweir     // --> FME 2008-09-16 #i91421#
504cdf0e10cSrcweir     if ( !pMaster->GetIndPrev() )
505cdf0e10cSrcweir     {
506cdf0e10cSrcweir         sal_uLong nLines = pMaster->GetThisLines();
507cdf0e10cSrcweir         if(nLines == 0 && pMaster->HasPara())
508cdf0e10cSrcweir         {
509cdf0e10cSrcweir             const SwParaPortion *pMasterPara = pMaster->GetPara();
510cdf0e10cSrcweir             if(pMasterPara && pMasterPara->GetNext())
511cdf0e10cSrcweir                 nLines = 2;
512cdf0e10cSrcweir         }
513cdf0e10cSrcweir         if( nLines <= nNeed )
514cdf0e10cSrcweir             return sal_False;
515cdf0e10cSrcweir     }
516cdf0e10cSrcweir 
517cdf0e10cSrcweir     pMaster->Prepare( PREP_WIDOWS, (void*)&nNeed );
518cdf0e10cSrcweir     return sal_True;
519cdf0e10cSrcweir }
520cdf0e10cSrcweir 
521cdf0e10cSrcweir /*************************************************************************
522cdf0e10cSrcweir  *                  WidowsAndOrphans::WouldFit()
523cdf0e10cSrcweir  *************************************************************************/
524cdf0e10cSrcweir 
WouldFit(SwTxtMargin & rLine,SwTwips & rMaxHeight,sal_Bool bTst)525cdf0e10cSrcweir sal_Bool WidowsAndOrphans::WouldFit( SwTxtMargin &rLine, SwTwips &rMaxHeight, sal_Bool bTst )
526cdf0e10cSrcweir {
527cdf0e10cSrcweir     // Here it does not matter, if pFrm is swapped or not.
528cdf0e10cSrcweir     // IsInside() takes care for itself
529cdf0e10cSrcweir 
530cdf0e10cSrcweir     // Wir erwarten, dass rLine auf der letzten Zeile steht!!
531cdf0e10cSrcweir     ASSERT( !rLine.GetNext(), "WouldFit: aLine::Bottom missed!" );
532cdf0e10cSrcweir     MSHORT nLineCnt = rLine.GetLineNr();
533cdf0e10cSrcweir 
534cdf0e10cSrcweir     // Erstmal die Orphansregel und den Initialenwunsch erfuellen ...
535cdf0e10cSrcweir     const MSHORT nMinLines = Max( GetOrphansLines(), rLine.GetDropLines() );
536cdf0e10cSrcweir     if ( nLineCnt < nMinLines )
537cdf0e10cSrcweir         return sal_False;
538cdf0e10cSrcweir 
539cdf0e10cSrcweir     rLine.Top();
540cdf0e10cSrcweir     SwTwips nLineSum = rLine.GetLineHeight();
541cdf0e10cSrcweir 
542cdf0e10cSrcweir     while( nMinLines > rLine.GetLineNr() )
543cdf0e10cSrcweir     {
544cdf0e10cSrcweir         DBG_LOOP;
545cdf0e10cSrcweir         if( !rLine.NextLine() )
546cdf0e10cSrcweir             return sal_False;
547cdf0e10cSrcweir         nLineSum += rLine.GetLineHeight();
548cdf0e10cSrcweir     }
549cdf0e10cSrcweir 
550cdf0e10cSrcweir     // Wenn wir jetzt schon nicht mehr passen ...
551cdf0e10cSrcweir     if( !IsInside( rLine ) )
552cdf0e10cSrcweir         return sal_False;
553cdf0e10cSrcweir 
554cdf0e10cSrcweir     // Jetzt noch die Widows-Regel ueberpruefen
555cdf0e10cSrcweir     if( !nWidLines && !pFrm->IsFollow() )
556cdf0e10cSrcweir     {
557cdf0e10cSrcweir         // I.A. brauchen Widows nur ueberprueft werden, wenn wir ein Follow
558cdf0e10cSrcweir         // sind. Bei WouldFit muss aber auch fuer den Master die Regel ueber-
559cdf0e10cSrcweir         // prueft werden, weil wir ja gerade erst die Trennstelle ermitteln.
560cdf0e10cSrcweir         // Im Ctor von WidowsAndOrphans wurde nWidLines aber nur fuer Follows
561cdf0e10cSrcweir         // aus dem AttrSet ermittelt, deshalb holen wir es hier nach:
562cdf0e10cSrcweir         const SwAttrSet& rSet = pFrm->GetTxtNode()->GetSwAttrSet();
563cdf0e10cSrcweir         nWidLines = rSet.GetWidows().GetValue();
564cdf0e10cSrcweir     }
565cdf0e10cSrcweir 
566cdf0e10cSrcweir     // Sind nach Orphans/Initialen noch genug Zeilen fuer die Widows uebrig?
567cdf0e10cSrcweir     // #111937#: If we are currently doing a test formatting, we may not
568cdf0e10cSrcweir     // consider the widows rule for two reasons:
569cdf0e10cSrcweir     // 1. The columns may have different widths.
570cdf0e10cSrcweir     //    Widow lines would have wrong width.
571cdf0e10cSrcweir     // 2. Test formatting is only done up to the given space.
572cdf0e10cSrcweir     //    we do not have any lines for widows at all.
573cdf0e10cSrcweir     if( bTst || nLineCnt - nMinLines >= GetWidowsLines() )
574cdf0e10cSrcweir     {
575cdf0e10cSrcweir         if( rMaxHeight >= nLineSum )
576cdf0e10cSrcweir         {
577cdf0e10cSrcweir             rMaxHeight -= nLineSum;
578cdf0e10cSrcweir             return sal_True;
579cdf0e10cSrcweir         }
580cdf0e10cSrcweir     }
581cdf0e10cSrcweir     return sal_False;
582cdf0e10cSrcweir }
583