xref: /aoo4110/main/sw/source/core/text/itrtxt.cxx (revision b1cdbd2c)
1*b1cdbd2cSJim Jagielski /**************************************************************
2*b1cdbd2cSJim Jagielski  *
3*b1cdbd2cSJim Jagielski  * Licensed to the Apache Software Foundation (ASF) under one
4*b1cdbd2cSJim Jagielski  * or more contributor license agreements.  See the NOTICE file
5*b1cdbd2cSJim Jagielski  * distributed with this work for additional information
6*b1cdbd2cSJim Jagielski  * regarding copyright ownership.  The ASF licenses this file
7*b1cdbd2cSJim Jagielski  * to you under the Apache License, Version 2.0 (the
8*b1cdbd2cSJim Jagielski  * "License"); you may not use this file except in compliance
9*b1cdbd2cSJim Jagielski  * with the License.  You may obtain a copy of the License at
10*b1cdbd2cSJim Jagielski  *
11*b1cdbd2cSJim Jagielski  *   http://www.apache.org/licenses/LICENSE-2.0
12*b1cdbd2cSJim Jagielski  *
13*b1cdbd2cSJim Jagielski  * Unless required by applicable law or agreed to in writing,
14*b1cdbd2cSJim Jagielski  * software distributed under the License is distributed on an
15*b1cdbd2cSJim Jagielski  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16*b1cdbd2cSJim Jagielski  * KIND, either express or implied.  See the License for the
17*b1cdbd2cSJim Jagielski  * specific language governing permissions and limitations
18*b1cdbd2cSJim Jagielski  * under the License.
19*b1cdbd2cSJim Jagielski  *
20*b1cdbd2cSJim Jagielski  *************************************************************/
21*b1cdbd2cSJim Jagielski 
22*b1cdbd2cSJim Jagielski 
23*b1cdbd2cSJim Jagielski 
24*b1cdbd2cSJim Jagielski // MARKER(update_precomp.py): autogen include statement, do not remove
25*b1cdbd2cSJim Jagielski #include "precompiled_sw.hxx"
26*b1cdbd2cSJim Jagielski 
27*b1cdbd2cSJim Jagielski 
28*b1cdbd2cSJim Jagielski #include "ndtxt.hxx"
29*b1cdbd2cSJim Jagielski #include "flyfrm.hxx"
30*b1cdbd2cSJim Jagielski #include "paratr.hxx"
31*b1cdbd2cSJim Jagielski #include "errhdl.hxx"
32*b1cdbd2cSJim Jagielski #include <vcl/outdev.hxx>
33*b1cdbd2cSJim Jagielski #include <editeng/paravertalignitem.hxx>
34*b1cdbd2cSJim Jagielski 
35*b1cdbd2cSJim Jagielski #include "pormulti.hxx"
36*b1cdbd2cSJim Jagielski #include <pagefrm.hxx>
37*b1cdbd2cSJim Jagielski #include <pagedesc.hxx> // SwPageDesc
38*b1cdbd2cSJim Jagielski #include <tgrditem.hxx>
39*b1cdbd2cSJim Jagielski #include <porfld.hxx>
40*b1cdbd2cSJim Jagielski 
41*b1cdbd2cSJim Jagielski #include "txtcfg.hxx"
42*b1cdbd2cSJim Jagielski #include "itrtxt.hxx"
43*b1cdbd2cSJim Jagielski #include "txtfrm.hxx"
44*b1cdbd2cSJim Jagielski #include "porfly.hxx"
45*b1cdbd2cSJim Jagielski 
46*b1cdbd2cSJim Jagielski #if OSL_DEBUG_LEVEL > 1
47*b1cdbd2cSJim Jagielski # include "txtfrm.hxx"      // GetFrmID,
48*b1cdbd2cSJim Jagielski #endif
49*b1cdbd2cSJim Jagielski 
50*b1cdbd2cSJim Jagielski /*************************************************************************
51*b1cdbd2cSJim Jagielski  *						SwTxtIter::CtorInitTxtIter()
52*b1cdbd2cSJim Jagielski  *************************************************************************/
53*b1cdbd2cSJim Jagielski 
CtorInitTxtIter(SwTxtFrm * pNewFrm,SwTxtInfo * pNewInf)54*b1cdbd2cSJim Jagielski void SwTxtIter::CtorInitTxtIter( SwTxtFrm *pNewFrm, SwTxtInfo *pNewInf )
55*b1cdbd2cSJim Jagielski {
56*b1cdbd2cSJim Jagielski #ifdef DBGTXT
57*b1cdbd2cSJim Jagielski 	// nStopAt laesst sich vom CV bearbeiten.
58*b1cdbd2cSJim Jagielski 	static MSHORT nStopAt = 0;
59*b1cdbd2cSJim Jagielski 	if( nStopAt == pNewFrm->GetFrmId() )
60*b1cdbd2cSJim Jagielski 	{
61*b1cdbd2cSJim Jagielski 		int i = pNewFrm->GetFrmId();
62*b1cdbd2cSJim Jagielski 	}
63*b1cdbd2cSJim Jagielski #endif
64*b1cdbd2cSJim Jagielski 
65*b1cdbd2cSJim Jagielski 	SwTxtNode *pNode = pNewFrm->GetTxtNode();
66*b1cdbd2cSJim Jagielski 
67*b1cdbd2cSJim Jagielski 	ASSERT( pNewFrm->GetPara(), "No paragraph" );
68*b1cdbd2cSJim Jagielski 
69*b1cdbd2cSJim Jagielski     CtorInitAttrIter( *pNode, pNewFrm->GetPara()->GetScriptInfo(), pNewFrm );
70*b1cdbd2cSJim Jagielski 
71*b1cdbd2cSJim Jagielski 	pFrm = pNewFrm;
72*b1cdbd2cSJim Jagielski     pInf = pNewInf;
73*b1cdbd2cSJim Jagielski     // --> OD 2008-01-17 #newlistlevelattrs#
74*b1cdbd2cSJim Jagielski     aLineInf.CtorInitLineInfo( pNode->GetSwAttrSet(), *pNode );
75*b1cdbd2cSJim Jagielski     // <--
76*b1cdbd2cSJim Jagielski     nFrameStart = pFrm->Frm().Pos().Y() + pFrm->Prt().Pos().Y();
77*b1cdbd2cSJim Jagielski     SwTxtIter::Init();
78*b1cdbd2cSJim Jagielski 	if( pNode->GetSwAttrSet().GetRegister().GetValue() )
79*b1cdbd2cSJim Jagielski 		bRegisterOn = pFrm->FillRegister( nRegStart, nRegDiff );
80*b1cdbd2cSJim Jagielski 	else
81*b1cdbd2cSJim Jagielski 		bRegisterOn = sal_False;
82*b1cdbd2cSJim Jagielski }
83*b1cdbd2cSJim Jagielski 
84*b1cdbd2cSJim Jagielski /*************************************************************************
85*b1cdbd2cSJim Jagielski  *                      SwTxtIter::Init()
86*b1cdbd2cSJim Jagielski  *************************************************************************/
87*b1cdbd2cSJim Jagielski 
Init()88*b1cdbd2cSJim Jagielski void SwTxtIter::Init()
89*b1cdbd2cSJim Jagielski {
90*b1cdbd2cSJim Jagielski 	pCurr = pInf->GetParaPortion();
91*b1cdbd2cSJim Jagielski 	nStart = pInf->GetTxtStart();
92*b1cdbd2cSJim Jagielski     nY = nFrameStart;
93*b1cdbd2cSJim Jagielski     bPrev = sal_True;
94*b1cdbd2cSJim Jagielski 	pPrev = 0;
95*b1cdbd2cSJim Jagielski 	nLineNr = 1;
96*b1cdbd2cSJim Jagielski }
97*b1cdbd2cSJim Jagielski 
98*b1cdbd2cSJim Jagielski /*************************************************************************
99*b1cdbd2cSJim Jagielski  *				   SwTxtIter::_GetHeightAndAscent()
100*b1cdbd2cSJim Jagielski  *************************************************************************/
101*b1cdbd2cSJim Jagielski 
CalcAscentAndHeight(KSHORT & rAscent,KSHORT & rHeight) const102*b1cdbd2cSJim Jagielski void SwTxtIter::CalcAscentAndHeight( KSHORT &rAscent, KSHORT &rHeight ) const
103*b1cdbd2cSJim Jagielski {
104*b1cdbd2cSJim Jagielski 	rHeight = GetLineHeight();
105*b1cdbd2cSJim Jagielski     rAscent = pCurr->GetAscent() + rHeight - pCurr->Height();
106*b1cdbd2cSJim Jagielski }
107*b1cdbd2cSJim Jagielski 
108*b1cdbd2cSJim Jagielski /*************************************************************************
109*b1cdbd2cSJim Jagielski  *					  SwTxtIter::_GetPrev()
110*b1cdbd2cSJim Jagielski  *************************************************************************/
111*b1cdbd2cSJim Jagielski 
_GetPrev()112*b1cdbd2cSJim Jagielski SwLineLayout *SwTxtIter::_GetPrev()
113*b1cdbd2cSJim Jagielski {
114*b1cdbd2cSJim Jagielski 	pPrev = 0;
115*b1cdbd2cSJim Jagielski 	bPrev = sal_True;
116*b1cdbd2cSJim Jagielski 	SwLineLayout *pLay = pInf->GetParaPortion();
117*b1cdbd2cSJim Jagielski 	if( pCurr == pLay )
118*b1cdbd2cSJim Jagielski 		return 0;
119*b1cdbd2cSJim Jagielski 	while( pLay->GetNext() != pCurr )
120*b1cdbd2cSJim Jagielski 		pLay = pLay->GetNext();
121*b1cdbd2cSJim Jagielski 	return pPrev = pLay;
122*b1cdbd2cSJim Jagielski }
123*b1cdbd2cSJim Jagielski 
124*b1cdbd2cSJim Jagielski /*************************************************************************
125*b1cdbd2cSJim Jagielski  *                    SwTxtIter::GetPrev()
126*b1cdbd2cSJim Jagielski  *************************************************************************/
127*b1cdbd2cSJim Jagielski 
GetPrev()128*b1cdbd2cSJim Jagielski const SwLineLayout *SwTxtIter::GetPrev()
129*b1cdbd2cSJim Jagielski {
130*b1cdbd2cSJim Jagielski 	if(! bPrev)
131*b1cdbd2cSJim Jagielski 		_GetPrev();
132*b1cdbd2cSJim Jagielski 	return pPrev;
133*b1cdbd2cSJim Jagielski }
134*b1cdbd2cSJim Jagielski 
135*b1cdbd2cSJim Jagielski /*************************************************************************
136*b1cdbd2cSJim Jagielski  *                    SwTxtIter::Prev()
137*b1cdbd2cSJim Jagielski  *************************************************************************/
138*b1cdbd2cSJim Jagielski 
Prev()139*b1cdbd2cSJim Jagielski const SwLineLayout *SwTxtIter::Prev()
140*b1cdbd2cSJim Jagielski {
141*b1cdbd2cSJim Jagielski 	if( !bPrev )
142*b1cdbd2cSJim Jagielski 		_GetPrev();
143*b1cdbd2cSJim Jagielski 	if( pPrev )
144*b1cdbd2cSJim Jagielski 	{
145*b1cdbd2cSJim Jagielski 		bPrev = sal_False;
146*b1cdbd2cSJim Jagielski 		pCurr = pPrev;
147*b1cdbd2cSJim Jagielski 		nStart = nStart - pCurr->GetLen();
148*b1cdbd2cSJim Jagielski 		nY = nY - GetLineHeight();
149*b1cdbd2cSJim Jagielski 		if( !pCurr->IsDummy() && !(--nLineNr) )
150*b1cdbd2cSJim Jagielski 			++nLineNr;
151*b1cdbd2cSJim Jagielski 		return pCurr;
152*b1cdbd2cSJim Jagielski 	}
153*b1cdbd2cSJim Jagielski 	else
154*b1cdbd2cSJim Jagielski 		return 0;
155*b1cdbd2cSJim Jagielski }
156*b1cdbd2cSJim Jagielski 
157*b1cdbd2cSJim Jagielski /*************************************************************************
158*b1cdbd2cSJim Jagielski  *                      SwTxtIter::Next()
159*b1cdbd2cSJim Jagielski  *************************************************************************/
160*b1cdbd2cSJim Jagielski 
Next()161*b1cdbd2cSJim Jagielski const SwLineLayout *SwTxtIter::Next()
162*b1cdbd2cSJim Jagielski {
163*b1cdbd2cSJim Jagielski 	if(pCurr->GetNext())
164*b1cdbd2cSJim Jagielski 	{
165*b1cdbd2cSJim Jagielski 		pPrev = pCurr;
166*b1cdbd2cSJim Jagielski 		bPrev = sal_True;
167*b1cdbd2cSJim Jagielski 		nStart = nStart + pCurr->GetLen();
168*b1cdbd2cSJim Jagielski 		nY += GetLineHeight();
169*b1cdbd2cSJim Jagielski 		if( pCurr->GetLen() || ( nLineNr>1 && !pCurr->IsDummy() ) )
170*b1cdbd2cSJim Jagielski 			++nLineNr;
171*b1cdbd2cSJim Jagielski 		return pCurr = pCurr->GetNext();
172*b1cdbd2cSJim Jagielski 	}
173*b1cdbd2cSJim Jagielski 	else
174*b1cdbd2cSJim Jagielski 		return 0;
175*b1cdbd2cSJim Jagielski }
176*b1cdbd2cSJim Jagielski 
177*b1cdbd2cSJim Jagielski /*************************************************************************
178*b1cdbd2cSJim Jagielski  *                      SwTxtIter::NextLine()
179*b1cdbd2cSJim Jagielski  *************************************************************************/
180*b1cdbd2cSJim Jagielski 
NextLine()181*b1cdbd2cSJim Jagielski const SwLineLayout *SwTxtIter::NextLine()
182*b1cdbd2cSJim Jagielski {
183*b1cdbd2cSJim Jagielski 	const SwLineLayout *pNext = Next();
184*b1cdbd2cSJim Jagielski 	while( pNext && pNext->IsDummy() && pNext->GetNext() )
185*b1cdbd2cSJim Jagielski 	{
186*b1cdbd2cSJim Jagielski 		DBG_LOOP;
187*b1cdbd2cSJim Jagielski 		pNext = Next();
188*b1cdbd2cSJim Jagielski 	}
189*b1cdbd2cSJim Jagielski 	return pNext;
190*b1cdbd2cSJim Jagielski }
191*b1cdbd2cSJim Jagielski 
192*b1cdbd2cSJim Jagielski /*************************************************************************
193*b1cdbd2cSJim Jagielski  *						SwTxtIter::GetNextLine()
194*b1cdbd2cSJim Jagielski  *************************************************************************/
195*b1cdbd2cSJim Jagielski 
GetNextLine() const196*b1cdbd2cSJim Jagielski const SwLineLayout *SwTxtIter::GetNextLine() const
197*b1cdbd2cSJim Jagielski {
198*b1cdbd2cSJim Jagielski 	const SwLineLayout *pNext = pCurr->GetNext();
199*b1cdbd2cSJim Jagielski 	while( pNext && pNext->IsDummy() && pNext->GetNext() )
200*b1cdbd2cSJim Jagielski 	{
201*b1cdbd2cSJim Jagielski 		DBG_LOOP;
202*b1cdbd2cSJim Jagielski 		pNext = pNext->GetNext();
203*b1cdbd2cSJim Jagielski 	}
204*b1cdbd2cSJim Jagielski 	return (SwLineLayout*)pNext;
205*b1cdbd2cSJim Jagielski }
206*b1cdbd2cSJim Jagielski 
207*b1cdbd2cSJim Jagielski /*************************************************************************
208*b1cdbd2cSJim Jagielski  *						SwTxtIter::GetPrevLine()
209*b1cdbd2cSJim Jagielski  *************************************************************************/
210*b1cdbd2cSJim Jagielski 
GetPrevLine()211*b1cdbd2cSJim Jagielski const SwLineLayout *SwTxtIter::GetPrevLine()
212*b1cdbd2cSJim Jagielski {
213*b1cdbd2cSJim Jagielski 	const SwLineLayout *pRoot = pInf->GetParaPortion();
214*b1cdbd2cSJim Jagielski 	if( pRoot == pCurr )
215*b1cdbd2cSJim Jagielski 		return 0;
216*b1cdbd2cSJim Jagielski 	const SwLineLayout *pLay = pRoot;
217*b1cdbd2cSJim Jagielski 
218*b1cdbd2cSJim Jagielski 	while( pLay->GetNext() != pCurr )
219*b1cdbd2cSJim Jagielski 		pLay = pLay->GetNext();
220*b1cdbd2cSJim Jagielski 
221*b1cdbd2cSJim Jagielski 	if( pLay->IsDummy() )
222*b1cdbd2cSJim Jagielski 	{
223*b1cdbd2cSJim Jagielski 		const SwLineLayout *pTmp = pRoot;
224*b1cdbd2cSJim Jagielski 		pLay = pRoot->IsDummy() ? 0 : pRoot;
225*b1cdbd2cSJim Jagielski 		while( pTmp->GetNext() != pCurr )
226*b1cdbd2cSJim Jagielski 		{
227*b1cdbd2cSJim Jagielski 			if( !pTmp->IsDummy() )
228*b1cdbd2cSJim Jagielski 				pLay = pTmp;
229*b1cdbd2cSJim Jagielski 			pTmp = pTmp->GetNext();
230*b1cdbd2cSJim Jagielski 		}
231*b1cdbd2cSJim Jagielski 	}
232*b1cdbd2cSJim Jagielski 
233*b1cdbd2cSJim Jagielski 	// Wenn sich nichts getan hat, dann gibt es nur noch Dummys
234*b1cdbd2cSJim Jagielski 	return (SwLineLayout*)pLay;
235*b1cdbd2cSJim Jagielski }
236*b1cdbd2cSJim Jagielski 
237*b1cdbd2cSJim Jagielski /*************************************************************************
238*b1cdbd2cSJim Jagielski  *                      SwTxtIter::PrevLine()
239*b1cdbd2cSJim Jagielski  *************************************************************************/
240*b1cdbd2cSJim Jagielski 
PrevLine()241*b1cdbd2cSJim Jagielski const SwLineLayout *SwTxtIter::PrevLine()
242*b1cdbd2cSJim Jagielski {
243*b1cdbd2cSJim Jagielski     const SwLineLayout *pMyPrev = Prev();
244*b1cdbd2cSJim Jagielski     if( !pMyPrev )
245*b1cdbd2cSJim Jagielski 		return 0;
246*b1cdbd2cSJim Jagielski 
247*b1cdbd2cSJim Jagielski     const SwLineLayout *pLast = pMyPrev;
248*b1cdbd2cSJim Jagielski     while( pMyPrev && pMyPrev->IsDummy() )
249*b1cdbd2cSJim Jagielski 	{
250*b1cdbd2cSJim Jagielski 		DBG_LOOP;
251*b1cdbd2cSJim Jagielski         pLast = pMyPrev;
252*b1cdbd2cSJim Jagielski         pMyPrev = Prev();
253*b1cdbd2cSJim Jagielski 	}
254*b1cdbd2cSJim Jagielski     return (SwLineLayout*)(pMyPrev ? pMyPrev : pLast);
255*b1cdbd2cSJim Jagielski }
256*b1cdbd2cSJim Jagielski 
257*b1cdbd2cSJim Jagielski /*************************************************************************
258*b1cdbd2cSJim Jagielski  *                      SwTxtIter::Bottom()
259*b1cdbd2cSJim Jagielski  *************************************************************************/
260*b1cdbd2cSJim Jagielski 
Bottom()261*b1cdbd2cSJim Jagielski void SwTxtIter::Bottom()
262*b1cdbd2cSJim Jagielski {
263*b1cdbd2cSJim Jagielski 	while( Next() )
264*b1cdbd2cSJim Jagielski 	{
265*b1cdbd2cSJim Jagielski 		DBG_LOOP;
266*b1cdbd2cSJim Jagielski 	}
267*b1cdbd2cSJim Jagielski }
268*b1cdbd2cSJim Jagielski 
269*b1cdbd2cSJim Jagielski /*************************************************************************
270*b1cdbd2cSJim Jagielski  *                      SwTxtIter::CharToLine()
271*b1cdbd2cSJim Jagielski  *************************************************************************/
272*b1cdbd2cSJim Jagielski 
CharToLine(const xub_StrLen nChar)273*b1cdbd2cSJim Jagielski void SwTxtIter::CharToLine(const xub_StrLen nChar)
274*b1cdbd2cSJim Jagielski {
275*b1cdbd2cSJim Jagielski 	while( nStart + pCurr->GetLen() <= nChar && Next() )
276*b1cdbd2cSJim Jagielski 		;
277*b1cdbd2cSJim Jagielski 	while( nStart > nChar && Prev() )
278*b1cdbd2cSJim Jagielski 		;
279*b1cdbd2cSJim Jagielski }
280*b1cdbd2cSJim Jagielski 
281*b1cdbd2cSJim Jagielski /*************************************************************************
282*b1cdbd2cSJim Jagielski  *                      SwTxtIter::CharCrsrToLine()
283*b1cdbd2cSJim Jagielski  *************************************************************************/
284*b1cdbd2cSJim Jagielski 
285*b1cdbd2cSJim Jagielski // 1170: beruecksichtigt Mehrdeutigkeiten:
CharCrsrToLine(const xub_StrLen nPosition)286*b1cdbd2cSJim Jagielski const SwLineLayout *SwTxtCursor::CharCrsrToLine( const xub_StrLen nPosition )
287*b1cdbd2cSJim Jagielski {
288*b1cdbd2cSJim Jagielski     CharToLine( nPosition );
289*b1cdbd2cSJim Jagielski     if( nPosition != nStart )
290*b1cdbd2cSJim Jagielski 		bRightMargin = sal_False;
291*b1cdbd2cSJim Jagielski     sal_Bool bPrevious = bRightMargin && pCurr->GetLen() && GetPrev() &&
292*b1cdbd2cSJim Jagielski 		GetPrev()->GetLen();
293*b1cdbd2cSJim Jagielski     if( bPrevious && nPosition && CH_BREAK == GetInfo().GetChar( nPosition-1 ) )
294*b1cdbd2cSJim Jagielski         bPrevious = sal_False;
295*b1cdbd2cSJim Jagielski     return bPrevious ? PrevLine() : pCurr;
296*b1cdbd2cSJim Jagielski }
297*b1cdbd2cSJim Jagielski 
298*b1cdbd2cSJim Jagielski /*************************************************************************
299*b1cdbd2cSJim Jagielski  *                      SwTxtCrsr::AdjustBaseLine()
300*b1cdbd2cSJim Jagielski  *************************************************************************/
301*b1cdbd2cSJim Jagielski 
AdjustBaseLine(const SwLineLayout & rLine,const SwLinePortion * pPor,sal_uInt16 nPorHeight,sal_uInt16 nPorAscent,const sal_Bool bAutoToCentered) const302*b1cdbd2cSJim Jagielski sal_uInt16 SwTxtCursor::AdjustBaseLine( const SwLineLayout& rLine,
303*b1cdbd2cSJim Jagielski                                     const SwLinePortion* pPor,
304*b1cdbd2cSJim Jagielski                                     sal_uInt16 nPorHeight, sal_uInt16 nPorAscent,
305*b1cdbd2cSJim Jagielski                                     const sal_Bool bAutoToCentered ) const
306*b1cdbd2cSJim Jagielski {
307*b1cdbd2cSJim Jagielski     if ( pPor )
308*b1cdbd2cSJim Jagielski     {
309*b1cdbd2cSJim Jagielski         nPorHeight = pPor->Height();
310*b1cdbd2cSJim Jagielski         nPorAscent = pPor->GetAscent();
311*b1cdbd2cSJim Jagielski     }
312*b1cdbd2cSJim Jagielski 
313*b1cdbd2cSJim Jagielski     sal_uInt16 nOfst = rLine.GetRealHeight() - rLine.Height();
314*b1cdbd2cSJim Jagielski 
315*b1cdbd2cSJim Jagielski     GETGRID( pFrm->FindPageFrm() )
316*b1cdbd2cSJim Jagielski     const sal_Bool bHasGrid = pGrid && GetInfo().SnapToGrid();
317*b1cdbd2cSJim Jagielski 
318*b1cdbd2cSJim Jagielski     if ( bHasGrid )
319*b1cdbd2cSJim Jagielski     {
320*b1cdbd2cSJim Jagielski         const sal_uInt16 nRubyHeight = pGrid->GetRubyHeight();
321*b1cdbd2cSJim Jagielski         const sal_Bool bRubyTop = ! pGrid->GetRubyTextBelow();
322*b1cdbd2cSJim Jagielski 
323*b1cdbd2cSJim Jagielski         if ( GetInfo().IsMulti() )
324*b1cdbd2cSJim Jagielski             // we are inside the GetCharRect recursion for multi portions
325*b1cdbd2cSJim Jagielski             // we center the portion in its surrounding line
326*b1cdbd2cSJim Jagielski             nOfst = ( pCurr->Height() - nPorHeight ) / 2 + nPorAscent;
327*b1cdbd2cSJim Jagielski         else
328*b1cdbd2cSJim Jagielski         {
329*b1cdbd2cSJim Jagielski             // We have to take care for ruby portions.
330*b1cdbd2cSJim Jagielski             // The ruby portion is NOT centered
331*b1cdbd2cSJim Jagielski             nOfst = nOfst + nPorAscent;
332*b1cdbd2cSJim Jagielski 
333*b1cdbd2cSJim Jagielski             if ( ! pPor || ! pPor->IsMultiPortion() ||
334*b1cdbd2cSJim Jagielski                  ! ((SwMultiPortion*)pPor)->IsRuby() )
335*b1cdbd2cSJim Jagielski             {
336*b1cdbd2cSJim Jagielski                 // Portions which are bigger than on grid distance are
337*b1cdbd2cSJim Jagielski                 // centered inside the whole line.
338*b1cdbd2cSJim Jagielski 
339*b1cdbd2cSJim Jagielski                 //for text refactor
340*b1cdbd2cSJim Jagielski                 const sal_uInt16 nLineNetto =  rLine.Height() - nRubyHeight;
341*b1cdbd2cSJim Jagielski                 //const sal_uInt16 nLineNetto = ( nPorHeight > nGridWidth ) ?
342*b1cdbd2cSJim Jagielski                  //                           rLine.Height() - nRubyHeight :
343*b1cdbd2cSJim Jagielski                  //                           nGridWidth;
344*b1cdbd2cSJim Jagielski                 nOfst += ( nLineNetto - nPorHeight ) / 2;
345*b1cdbd2cSJim Jagielski                 if ( bRubyTop )
346*b1cdbd2cSJim Jagielski                     nOfst = nOfst + nRubyHeight;
347*b1cdbd2cSJim Jagielski             }
348*b1cdbd2cSJim Jagielski         }
349*b1cdbd2cSJim Jagielski     }
350*b1cdbd2cSJim Jagielski     else
351*b1cdbd2cSJim Jagielski     {
352*b1cdbd2cSJim Jagielski         switch ( GetLineInfo().GetVertAlign() ) {
353*b1cdbd2cSJim Jagielski             case SvxParaVertAlignItem::TOP :
354*b1cdbd2cSJim Jagielski                 nOfst = nOfst + nPorAscent;
355*b1cdbd2cSJim Jagielski                 break;
356*b1cdbd2cSJim Jagielski             case SvxParaVertAlignItem::CENTER :
357*b1cdbd2cSJim Jagielski                 ASSERT( rLine.Height() >= nPorHeight, "Portion height > Line height");
358*b1cdbd2cSJim Jagielski                 nOfst += ( rLine.Height() - nPorHeight ) / 2 + nPorAscent;
359*b1cdbd2cSJim Jagielski                 break;
360*b1cdbd2cSJim Jagielski             case SvxParaVertAlignItem::BOTTOM :
361*b1cdbd2cSJim Jagielski                 nOfst += rLine.Height() - nPorHeight + nPorAscent;
362*b1cdbd2cSJim Jagielski                 break;
363*b1cdbd2cSJim Jagielski             case SvxParaVertAlignItem::AUTOMATIC :
364*b1cdbd2cSJim Jagielski                 if ( bAutoToCentered || GetInfo().GetTxtFrm()->IsVertical() )
365*b1cdbd2cSJim Jagielski                 {
366*b1cdbd2cSJim Jagielski                     //Badaa: 2008-04-18 * Support for Classical Mongolian Script (SCMS) joint with Jiayanmin
367*b1cdbd2cSJim Jagielski                     if( GetInfo().GetTxtFrm()->IsVertLR() )
368*b1cdbd2cSJim Jagielski                     		nOfst += rLine.Height() - ( rLine.Height() - nPorHeight ) / 2 - nPorAscent;
369*b1cdbd2cSJim Jagielski                     else
370*b1cdbd2cSJim Jagielski                     		nOfst += ( rLine.Height() - nPorHeight ) / 2 + nPorAscent;
371*b1cdbd2cSJim Jagielski                     break;
372*b1cdbd2cSJim Jagielski                 }
373*b1cdbd2cSJim Jagielski             case SvxParaVertAlignItem::BASELINE :
374*b1cdbd2cSJim Jagielski                 // base line
375*b1cdbd2cSJim Jagielski                 nOfst = nOfst + rLine.GetAscent();
376*b1cdbd2cSJim Jagielski                 break;
377*b1cdbd2cSJim Jagielski         }
378*b1cdbd2cSJim Jagielski     }
379*b1cdbd2cSJim Jagielski 
380*b1cdbd2cSJim Jagielski     return nOfst;
381*b1cdbd2cSJim Jagielski }
382*b1cdbd2cSJim Jagielski 
383*b1cdbd2cSJim Jagielski /*************************************************************************
384*b1cdbd2cSJim Jagielski  *                      SwTxtIter::TwipsToLine()
385*b1cdbd2cSJim Jagielski  *************************************************************************/
386*b1cdbd2cSJim Jagielski 
TwipsToLine(const SwTwips y)387*b1cdbd2cSJim Jagielski const SwLineLayout *SwTxtIter::TwipsToLine( const SwTwips y)
388*b1cdbd2cSJim Jagielski {
389*b1cdbd2cSJim Jagielski 	while( nY + GetLineHeight() <= y && Next() )
390*b1cdbd2cSJim Jagielski 		;
391*b1cdbd2cSJim Jagielski 	while( nY > y && Prev() )
392*b1cdbd2cSJim Jagielski 		;
393*b1cdbd2cSJim Jagielski 	return pCurr;
394*b1cdbd2cSJim Jagielski }
395*b1cdbd2cSJim Jagielski 
396*b1cdbd2cSJim Jagielski //
397*b1cdbd2cSJim Jagielski // Local helper function to check, if pCurr needs a field rest portion:
398*b1cdbd2cSJim Jagielski //
lcl_NeedsFieldRest(const SwLineLayout * pCurr)399*b1cdbd2cSJim Jagielski sal_Bool lcl_NeedsFieldRest( const SwLineLayout* pCurr )
400*b1cdbd2cSJim Jagielski {
401*b1cdbd2cSJim Jagielski 	const SwLinePortion *pPor = pCurr->GetPortion();
402*b1cdbd2cSJim Jagielski 	sal_Bool bRet = sal_False;
403*b1cdbd2cSJim Jagielski 	while( pPor && !bRet )
404*b1cdbd2cSJim Jagielski 	{
405*b1cdbd2cSJim Jagielski 		bRet = pPor->InFldGrp() && ((SwFldPortion*)pPor)->HasFollow();
406*b1cdbd2cSJim Jagielski 		if( !pPor->GetPortion() || !pPor->GetPortion()->InFldGrp() )
407*b1cdbd2cSJim Jagielski 			break;
408*b1cdbd2cSJim Jagielski 		pPor = pPor->GetPortion();
409*b1cdbd2cSJim Jagielski 	}
410*b1cdbd2cSJim Jagielski 	return bRet;
411*b1cdbd2cSJim Jagielski }
412*b1cdbd2cSJim Jagielski 
413*b1cdbd2cSJim Jagielski /*************************************************************************
414*b1cdbd2cSJim Jagielski  *						SwTxtIter::TruncLines()
415*b1cdbd2cSJim Jagielski  *************************************************************************/
416*b1cdbd2cSJim Jagielski 
TruncLines(sal_Bool bNoteFollow)417*b1cdbd2cSJim Jagielski void SwTxtIter::TruncLines( sal_Bool bNoteFollow )
418*b1cdbd2cSJim Jagielski {
419*b1cdbd2cSJim Jagielski 	SwLineLayout *pDel = pCurr->GetNext();
420*b1cdbd2cSJim Jagielski     const xub_StrLen nEnd = nStart + pCurr->GetLen();
421*b1cdbd2cSJim Jagielski 
422*b1cdbd2cSJim Jagielski 	if( pDel )
423*b1cdbd2cSJim Jagielski 	{
424*b1cdbd2cSJim Jagielski 		pCurr->SetNext( 0 );
425*b1cdbd2cSJim Jagielski 		if( GetHints() && bNoteFollow )
426*b1cdbd2cSJim Jagielski         {
427*b1cdbd2cSJim Jagielski 			GetInfo().GetParaPortion()->SetFollowField( pDel->IsRest() ||
428*b1cdbd2cSJim Jagielski                                                         lcl_NeedsFieldRest( pCurr ) );
429*b1cdbd2cSJim Jagielski 
430*b1cdbd2cSJim Jagielski             // bug 88534: wrong positioning of flys
431*b1cdbd2cSJim Jagielski             SwTxtFrm* pFollow = GetTxtFrm()->GetFollow();
432*b1cdbd2cSJim Jagielski             if ( pFollow && ! pFollow->IsLocked() &&
433*b1cdbd2cSJim Jagielski                  nEnd == pFollow->GetOfst() )
434*b1cdbd2cSJim Jagielski             {
435*b1cdbd2cSJim Jagielski                 xub_StrLen nRangeEnd = nEnd;
436*b1cdbd2cSJim Jagielski                 SwLineLayout* pLine = pDel;
437*b1cdbd2cSJim Jagielski 
438*b1cdbd2cSJim Jagielski                 // determine range to be searched for flys anchored as characters
439*b1cdbd2cSJim Jagielski                 while ( pLine )
440*b1cdbd2cSJim Jagielski                 {
441*b1cdbd2cSJim Jagielski                     nRangeEnd = nRangeEnd + pLine->GetLen();
442*b1cdbd2cSJim Jagielski                     pLine = pLine->GetNext();
443*b1cdbd2cSJim Jagielski                 }
444*b1cdbd2cSJim Jagielski 
445*b1cdbd2cSJim Jagielski                 SwpHints* pTmpHints = GetTxtFrm()->GetTxtNode()->GetpSwpHints();
446*b1cdbd2cSJim Jagielski 
447*b1cdbd2cSJim Jagielski                 // examine hints in range nEnd - (nEnd + nRangeChar)
448*b1cdbd2cSJim Jagielski                 for( sal_uInt16 i = 0; i < pTmpHints->Count(); i++ )
449*b1cdbd2cSJim Jagielski                 {
450*b1cdbd2cSJim Jagielski                     const SwTxtAttr* pHt = pTmpHints->GetTextHint( i );
451*b1cdbd2cSJim Jagielski                     if( RES_TXTATR_FLYCNT == pHt->Which() )
452*b1cdbd2cSJim Jagielski                     {
453*b1cdbd2cSJim Jagielski                         // check, if hint is in our range
454*b1cdbd2cSJim Jagielski                         const sal_uInt16 nTmpPos = *pHt->GetStart();
455*b1cdbd2cSJim Jagielski                         if ( nEnd <= nTmpPos && nTmpPos < nRangeEnd )
456*b1cdbd2cSJim Jagielski                             pFollow->_InvalidateRange(
457*b1cdbd2cSJim Jagielski                                 SwCharRange( nTmpPos, nTmpPos ), 0 );
458*b1cdbd2cSJim Jagielski                     }
459*b1cdbd2cSJim Jagielski                 }
460*b1cdbd2cSJim Jagielski             }
461*b1cdbd2cSJim Jagielski         }
462*b1cdbd2cSJim Jagielski 		delete pDel;
463*b1cdbd2cSJim Jagielski 	}
464*b1cdbd2cSJim Jagielski     if( pCurr->IsDummy() &&
465*b1cdbd2cSJim Jagielski         !pCurr->GetLen() &&
466*b1cdbd2cSJim Jagielski          nStart < GetTxtFrm()->GetTxt().Len() )
467*b1cdbd2cSJim Jagielski         pCurr->SetRealHeight( 1 );
468*b1cdbd2cSJim Jagielski 	if( GetHints() )
469*b1cdbd2cSJim Jagielski         pFrm->RemoveFtn( nEnd );
470*b1cdbd2cSJim Jagielski }
471*b1cdbd2cSJim Jagielski 
472*b1cdbd2cSJim Jagielski /*************************************************************************
473*b1cdbd2cSJim Jagielski  *						SwTxtIter::CntHyphens()
474*b1cdbd2cSJim Jagielski  *************************************************************************/
475*b1cdbd2cSJim Jagielski 
CntHyphens(sal_uInt8 & nEndCnt,sal_uInt8 & nMidCnt) const476*b1cdbd2cSJim Jagielski void SwTxtIter::CntHyphens( sal_uInt8 &nEndCnt, sal_uInt8 &nMidCnt) const
477*b1cdbd2cSJim Jagielski {
478*b1cdbd2cSJim Jagielski 	nEndCnt = 0;
479*b1cdbd2cSJim Jagielski 	nMidCnt = 0;
480*b1cdbd2cSJim Jagielski 	if ( bPrev && pPrev && !pPrev->IsEndHyph() && !pPrev->IsMidHyph() )
481*b1cdbd2cSJim Jagielski 		 return;
482*b1cdbd2cSJim Jagielski 	SwLineLayout *pLay = pInf->GetParaPortion();
483*b1cdbd2cSJim Jagielski 	if( pCurr == pLay )
484*b1cdbd2cSJim Jagielski 		return;
485*b1cdbd2cSJim Jagielski 	while( pLay != pCurr )
486*b1cdbd2cSJim Jagielski 	{
487*b1cdbd2cSJim Jagielski 		DBG_LOOP;
488*b1cdbd2cSJim Jagielski 		if ( pLay->IsEndHyph() )
489*b1cdbd2cSJim Jagielski 			nEndCnt++;
490*b1cdbd2cSJim Jagielski 		else
491*b1cdbd2cSJim Jagielski 			nEndCnt = 0;
492*b1cdbd2cSJim Jagielski 		if ( pLay->IsMidHyph() )
493*b1cdbd2cSJim Jagielski 			nMidCnt++;
494*b1cdbd2cSJim Jagielski 		else
495*b1cdbd2cSJim Jagielski 			nMidCnt = 0;
496*b1cdbd2cSJim Jagielski 		pLay = pLay->GetNext();
497*b1cdbd2cSJim Jagielski 	}
498*b1cdbd2cSJim Jagielski }
499*b1cdbd2cSJim Jagielski 
500*b1cdbd2cSJim Jagielski /*************************************************************************
501*b1cdbd2cSJim Jagielski  *                          SwHookOut
502*b1cdbd2cSJim Jagielski  *
503*b1cdbd2cSJim Jagielski  * Change current output device to formatting device, this has to be done before
504*b1cdbd2cSJim Jagielski  * formatting.
505*b1cdbd2cSJim Jagielski  *************************************************************************/
506*b1cdbd2cSJim Jagielski 
SwHookOut(SwTxtSizeInfo & rInfo)507*b1cdbd2cSJim Jagielski SwHookOut::SwHookOut( SwTxtSizeInfo& rInfo ) :
508*b1cdbd2cSJim Jagielski      pInf( &rInfo ),
509*b1cdbd2cSJim Jagielski      pOut( rInfo.GetOut() ),
510*b1cdbd2cSJim Jagielski      bOnWin( rInfo.OnWin() )
511*b1cdbd2cSJim Jagielski {
512*b1cdbd2cSJim Jagielski     ASSERT( rInfo.GetRefDev(), "No reference device for text formatting" )
513*b1cdbd2cSJim Jagielski 
514*b1cdbd2cSJim Jagielski     // set new values
515*b1cdbd2cSJim Jagielski     rInfo.SetOut( rInfo.GetRefDev() );
516*b1cdbd2cSJim Jagielski     rInfo.SetOnWin( sal_False );
517*b1cdbd2cSJim Jagielski }
518*b1cdbd2cSJim Jagielski 
~SwHookOut()519*b1cdbd2cSJim Jagielski SwHookOut::~SwHookOut()
520*b1cdbd2cSJim Jagielski {
521*b1cdbd2cSJim Jagielski     pInf->SetOut( pOut );
522*b1cdbd2cSJim Jagielski     pInf->SetOnWin( bOnWin );
523*b1cdbd2cSJim Jagielski }
524