xref: /aoo41x/main/sw/source/core/txtnode/swfont.cxx (revision 69a74367)
1 /**************************************************************
2  *
3  * Licensed to the Apache Software Foundation (ASF) under one
4  * or more contributor license agreements.  See the NOTICE file
5  * distributed with this work for additional information
6  * regarding copyright ownership.  The ASF licenses this file
7  * to you under the Apache License, Version 2.0 (the
8  * "License"); you may not use this file except in compliance
9  * with the License.  You may obtain a copy of the License at
10  *
11  *   http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing,
14  * software distributed under the License is distributed on an
15  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16  * KIND, either express or implied.  See the License for the
17  * specific language governing permissions and limitations
18  * under the License.
19  *
20  *************************************************************/
21 
22 
23 
24 // MARKER(update_precomp.py): autogen include statement, do not remove
25 #include "precompiled_sw.hxx"
26 
27 
28 #include <hintids.hxx>
29 
30 #ifndef _COM_SUN_STAR_I18N_SCRIPTTYPE_HDL_
31 #include <com/sun/star/i18n/ScriptType.hdl>
32 #endif
33 #ifndef _OUTDEV_HXX //autogen
34 #include <vcl/outdev.hxx>
35 #endif
36 #include <unotools/localedatawrapper.hxx>
37 #include <editeng/unolingu.hxx>
38 #include <editeng/brshitem.hxx>
39 #include <editeng/wrlmitem.hxx>
40 #include <editeng/blnkitem.hxx>
41 #include <editeng/nhypitem.hxx>
42 #include <editeng/kernitem.hxx>
43 #include <editeng/cmapitem.hxx>
44 #include <editeng/langitem.hxx>
45 #include <editeng/escpitem.hxx>
46 #include <editeng/akrnitem.hxx>
47 #include <editeng/shdditem.hxx>
48 #include <editeng/charreliefitem.hxx>
49 #ifndef _SVX_CNTRITEM_HXX //autogen
50 #include <editeng/cntritem.hxx>
51 #endif
52 #include <editeng/colritem.hxx>
53 #include <editeng/cscoitem.hxx>
54 #include <editeng/crsditem.hxx>
55 #include <editeng/udlnitem.hxx>
56 #include <editeng/wghtitem.hxx>
57 #include <editeng/postitem.hxx>
58 #include <editeng/fhgtitem.hxx>
59 #include <editeng/fontitem.hxx>
60 #ifndef _SVX_EMPHITEM_HXX //autogen
61 #include <editeng/emphitem.hxx>
62 #endif
63 #include <editeng/charscaleitem.hxx>
64 #include <editeng/charrotateitem.hxx>
65 #include <editeng/twolinesitem.hxx>
66 #include <editeng/charhiddenitem.hxx>
67 #include <IDocumentSettingAccess.hxx>
68 #include <vcl/window.hxx>
69 #include <charatr.hxx>
70 #include <viewsh.hxx>		// Bildschirmabgleich
71 #include <swfont.hxx>
72 #include <fntcache.hxx>     // FontCache
73 #include <txtfrm.hxx>       // SwTxtFrm
74 #include <scriptinfo.hxx>
75 
76 #if defined(WNT) || defined(PM2)
77 #define FNT_LEADING_HACK
78 #endif
79 
80 #if defined(WNT)
81 #define FNT_ATM_HACK
82 #endif
83 
84 #ifdef DBG_UTIL
85 // globale Variable
86 SvStatistics aSvStat;
87 #endif
88 
89 using namespace ::com::sun::star;
90 
91 /************************************************************************
92  * Hintergrundbrush setzen, z.B. bei Zeichenvorlagen
93  ***********************************************************************/
94 
SetBackColor(Color * pNewColor)95 void SwFont::SetBackColor( Color* pNewColor )
96 {
97 	delete pBackColor;
98 	pBackColor = pNewColor;
99 	bFntChg = sal_True;
100 	aSub[SW_LATIN].pMagic = aSub[SW_CJK].pMagic = aSub[SW_CTL].pMagic = 0;
101 }
102 
103 // maps directions for vertical layout
MapDirection(sal_uInt16 nDir,const sal_Bool bVertFormat)104 sal_uInt16 MapDirection( sal_uInt16 nDir, const sal_Bool bVertFormat )
105 {
106     if ( bVertFormat )
107     {
108         switch ( nDir )
109         {
110         case 0 :
111             nDir = 2700;
112             break;
113         case 900 :
114             nDir = 0;
115             break;
116         case 2700 :
117             nDir = 1800;
118             break;
119 #if OSL_DEBUG_LEVEL > 1
120         default :
121             ASSERT( sal_False, "Unsupported direction" );
122             break;
123 #endif
124         }
125     }
126     return nDir;
127 }
128 
129 // maps the absolute direction set at the font to its logical conterpart
130 // in the rotated environment
UnMapDirection(sal_uInt16 nDir,const sal_Bool bVertFormat)131 sal_uInt16 UnMapDirection( sal_uInt16 nDir, const sal_Bool bVertFormat )
132 {
133     if ( bVertFormat )
134     {
135         switch ( nDir )
136         {
137         case 0 :
138             nDir = 900;
139             break;
140         case 1800 :
141             nDir = 2700;
142             break;
143         case 2700 :
144             nDir = 0;
145             break;
146 #if OSL_DEBUG_LEVEL > 1
147         default :
148             ASSERT( sal_False, "Unsupported direction" );
149             break;
150 #endif
151         }
152     }
153     return nDir;
154 }
155 
GetOrientation(const sal_Bool bVertFormat) const156 sal_uInt16 SwFont::GetOrientation( const sal_Bool bVertFormat ) const
157 {
158     return UnMapDirection( aSub[nActual].GetOrientation(), bVertFormat );
159 }
160 
SetVertical(sal_uInt16 nDir,const sal_Bool bVertFormat)161 void SwFont::SetVertical( sal_uInt16 nDir, const sal_Bool bVertFormat )
162 {
163     // map direction if frame has vertical layout
164     nDir = MapDirection( nDir, bVertFormat );
165 
166     if( nDir != aSub[0].GetOrientation() )
167 	{
168 		bFntChg = sal_True;
169 		aSub[0].SetVertical( nDir, bVertFormat );
170         aSub[1].SetVertical( nDir, bVertFormat || nDir > 1000 );
171 		aSub[2].SetVertical( nDir, bVertFormat );
172 	}
173 }
174 
175 /*************************************************************************
176  Escapement:
177 	frEsc:  Fraction, Grad des Escapements
178 	Esc = resultierendes Escapement
179 	A1 = Original-Ascent			(nOrgAscent)
180 	A2 = verkleinerter Ascent		(nEscAscent)
181 	Ax = resultierender Ascent		(GetAscent())
182 	H1 = Original-Hoehe 			(nOrgHeight)
183 	H2 = verkleinerter Hoehe		(nEscHeight)
184 	Hx = resultierender Hoehe		(GetHeight())
185 	Bx = resultierende Baseline fuer die Textausgabe (CalcPos())
186 		 (Vorsicht: Y - A1!)
187 
188 	Escapement:
189 		Esc = H1 * frEsc;
190 
191 	Hochstellung:
192 		Ax = A2 + Esc;
193 		Hx = H2 + Esc;
194 		Bx = A1 - Esc;
195 
196 	Tiefstellung:
197 		Ax = A1;
198 		Hx = A1 + Esc + (H2 - A2);
199 		Bx = A1 + Esc;
200 
201 *************************************************************************/
202 
203 /*************************************************************************
204  *					SwSubFont::CalcEscAscent( const sal_uInt16 nOldAscent )
205  *************************************************************************/
206 
207 // nEsc ist der Prozentwert
CalcEscAscent(const sal_uInt16 nOldAscent) const208 sal_uInt16 SwSubFont::CalcEscAscent( const sal_uInt16 nOldAscent ) const
209 {
210 	if( DFLT_ESC_AUTO_SUPER != GetEscapement() &&
211 		DFLT_ESC_AUTO_SUB != GetEscapement() )
212 	{
213 		const long nAscent = nOldAscent +
214 							 ( (long) nOrgHeight * GetEscapement() ) / 100L;
215 		if ( nAscent>0 )
216 			return ( Max( sal_uInt16 (nAscent), nOrgAscent ));
217 	}
218 	return nOrgAscent;
219 }
220 
221 /*************************************************************************
222  *                      SwFont::SetDiffFnt()
223  *************************************************************************/
224 
SetDiffFnt(const SfxItemSet * pAttrSet,const IDocumentSettingAccess * pIDocumentSettingAccess)225 void SwFont::SetDiffFnt( const SfxItemSet *pAttrSet,
226                          const IDocumentSettingAccess *pIDocumentSettingAccess )
227 {
228 	delete pBackColor;
229 	pBackColor = NULL;
230 
231     if( pAttrSet )
232 	{
233 		const SfxPoolItem* pItem;
234 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_FONT,
235 			sal_True, &pItem ))
236 		{
237 			const SvxFontItem *pFont = (const SvxFontItem *)pItem;
238 			aSub[SW_LATIN].SetFamily( pFont->GetFamily() );
239 			aSub[SW_LATIN].Font::SetName( pFont->GetFamilyName() );
240 			aSub[SW_LATIN].Font::SetStyleName( pFont->GetStyleName() );
241 			aSub[SW_LATIN].Font::SetPitch( pFont->GetPitch() );
242 			aSub[SW_LATIN].Font::SetCharSet( pFont->GetCharSet() );
243 		}
244 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_FONTSIZE,
245 			sal_True, &pItem ))
246 		{
247 			const SvxFontHeightItem *pHeight = (const SvxFontHeightItem *)pItem;
248 			aSub[SW_LATIN].SvxFont::SetPropr( 100 );
249 			aSub[SW_LATIN].aSize = aSub[SW_LATIN].Font::GetSize();
250 			Size aTmpSize = aSub[SW_LATIN].aSize;
251 			aTmpSize.Height() = pHeight->GetHeight();
252 			aSub[SW_LATIN].SetSize( aTmpSize );
253 		}
254 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_POSTURE,
255 			sal_True, &pItem ))
256 			aSub[SW_LATIN].Font::SetItalic( ((SvxPostureItem*)pItem)->GetPosture() );
257 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_WEIGHT,
258 			sal_True, &pItem ))
259 			aSub[SW_LATIN].Font::SetWeight( ((SvxWeightItem*)pItem)->GetWeight() );
260 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_LANGUAGE,
261 			sal_True, &pItem ))
262 			aSub[SW_LATIN].SetLanguage( ((SvxLanguageItem*)pItem)->GetLanguage() );
263 
264 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_CJK_FONT,
265 			sal_True, &pItem ))
266 		{
267 			const SvxFontItem *pFont = (const SvxFontItem *)pItem;
268 			aSub[SW_CJK].SetFamily( pFont->GetFamily() );
269 			aSub[SW_CJK].Font::SetName( pFont->GetFamilyName() );
270 			aSub[SW_CJK].Font::SetStyleName( pFont->GetStyleName() );
271 			aSub[SW_CJK].Font::SetPitch( pFont->GetPitch() );
272 			aSub[SW_CJK].Font::SetCharSet( pFont->GetCharSet() );
273 		}
274 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_CJK_FONTSIZE,
275 			sal_True, &pItem ))
276 		{
277 			const SvxFontHeightItem *pHeight = (const SvxFontHeightItem *)pItem;
278 			aSub[SW_CJK].SvxFont::SetPropr( 100 );
279 			aSub[SW_CJK].aSize = aSub[SW_CJK].Font::GetSize();
280 			Size aTmpSize = aSub[SW_CJK].aSize;
281 			aTmpSize.Height() = pHeight->GetHeight();
282 			aSub[SW_CJK].SetSize( aTmpSize );
283 		}
284 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_CJK_POSTURE,
285 			sal_True, &pItem ))
286 			aSub[SW_CJK].Font::SetItalic( ((SvxPostureItem*)pItem)->GetPosture() );
287 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_CJK_WEIGHT,
288 			sal_True, &pItem ))
289 			aSub[SW_CJK].Font::SetWeight( ((SvxWeightItem*)pItem)->GetWeight() );
290 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_CJK_LANGUAGE,
291 			sal_True, &pItem ))
292 		{
293 			LanguageType eNewLang = ((SvxLanguageItem*)pItem)->GetLanguage();
294 			aSub[SW_CJK].SetLanguage( eNewLang );
295 			aSub[SW_LATIN].SetCJKContextLanguage( eNewLang );
296 			aSub[SW_CJK].SetCJKContextLanguage( eNewLang );
297 			aSub[SW_CTL].SetCJKContextLanguage( eNewLang );
298 		}
299 
300 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_CTL_FONT,
301 			sal_True, &pItem ))
302 		{
303 			const SvxFontItem *pFont = (const SvxFontItem *)pItem;
304 			aSub[SW_CTL].SetFamily( pFont->GetFamily() );
305 			aSub[SW_CTL].Font::SetName( pFont->GetFamilyName() );
306 			aSub[SW_CTL].Font::SetStyleName( pFont->GetStyleName() );
307 			aSub[SW_CTL].Font::SetPitch( pFont->GetPitch() );
308 			aSub[SW_CTL].Font::SetCharSet( pFont->GetCharSet() );
309 		}
310 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_CTL_FONTSIZE,
311 			sal_True, &pItem ))
312 		{
313 			const SvxFontHeightItem *pHeight = (const SvxFontHeightItem *)pItem;
314 			aSub[SW_CTL].SvxFont::SetPropr( 100 );
315 			aSub[SW_CTL].aSize = aSub[SW_CTL].Font::GetSize();
316 			Size aTmpSize = aSub[SW_CTL].aSize;
317 			aTmpSize.Height() = pHeight->GetHeight();
318 			aSub[SW_CTL].SetSize( aTmpSize );
319 		}
320 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_CTL_POSTURE,
321 			sal_True, &pItem ))
322 			aSub[SW_CTL].Font::SetItalic( ((SvxPostureItem*)pItem)->GetPosture() );
323 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_CTL_WEIGHT,
324 			sal_True, &pItem ))
325 			aSub[SW_CTL].Font::SetWeight( ((SvxWeightItem*)pItem)->GetWeight() );
326 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_CTL_LANGUAGE,
327 			sal_True, &pItem ))
328 			aSub[SW_CTL].SetLanguage( ((SvxLanguageItem*)pItem)->GetLanguage() );
329 
330 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_UNDERLINE,
331 			sal_True, &pItem ))
332 		{
333 			SetUnderline( ((SvxUnderlineItem*)pItem)->GetLineStyle() );
334 			SetUnderColor( ((SvxUnderlineItem*)pItem)->GetColor() );
335 		}
336 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_OVERLINE,
337 			sal_True, &pItem ))
338         {
339 			SetOverline( ((SvxOverlineItem*)pItem)->GetLineStyle() );
340 			SetOverColor( ((SvxOverlineItem*)pItem)->GetColor() );
341         }
342 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_CROSSEDOUT,
343 			sal_True, &pItem ))
344 			SetStrikeout( ((SvxCrossedOutItem*)pItem)->GetStrikeout() );
345 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_COLOR,
346 			sal_True, &pItem ))
347 			SetColor( ((SvxColorItem*)pItem)->GetValue() );
348 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_EMPHASIS_MARK,
349 			sal_True, &pItem ))
350 			SetEmphasisMark( ((SvxEmphasisMarkItem*)pItem)->GetEmphasisMark() );
351 
352 		SetTransparent( sal_True );
353 		SetAlign( ALIGN_BASELINE );
354 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_CONTOUR,
355 			sal_True, &pItem ))
356 			SetOutline( ((SvxContourItem*)pItem)->GetValue() );
357 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_SHADOWED,
358 			sal_True, &pItem ))
359 			SetShadow( ((SvxShadowedItem*)pItem)->GetValue() );
360 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_RELIEF,
361 			sal_True, &pItem ))
362 			SetRelief( (FontRelief)((SvxCharReliefItem*)pItem)->GetValue() );
363 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_SHADOWED,
364 			sal_True, &pItem ))
365 			SetPropWidth(((SvxShadowedItem*)pItem)->GetValue() ? 50 : 100 );
366 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_AUTOKERN,
367 			sal_True, &pItem ))
368 		{
369 			if( ((SvxAutoKernItem*)pItem)->GetValue() )
370             {
371                 SetAutoKern( ( !pIDocumentSettingAccess ||
372                                !pIDocumentSettingAccess->get(IDocumentSettingAccess::KERN_ASIAN_PUNCTUATION) ) ?
373                                 KERNING_FONTSPECIFIC :
374                                 KERNING_ASIAN );
375             }
376 			else
377     			SetAutoKern( 0 );
378 		}
379 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_WORDLINEMODE,
380 			sal_True, &pItem ))
381 			SetWordLineMode( ((SvxWordLineModeItem*)pItem)->GetValue() );
382 
383 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_ESCAPEMENT,
384 			sal_True, &pItem ))
385 		{
386 			const SvxEscapementItem *pEsc = (const SvxEscapementItem *)pItem;
387 			SetEscapement( pEsc->GetEsc() );
388 			if( aSub[SW_LATIN].IsEsc() )
389 				SetProportion( pEsc->GetProp() );
390 		}
391 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_CASEMAP,
392 			sal_True, &pItem ))
393 			SetCaseMap( ((SvxCaseMapItem*)pItem)->GetCaseMap() );
394 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_KERNING,
395 			sal_True, &pItem ))
396 			SetFixKerning( ((SvxKerningItem*)pItem)->GetValue() );
397 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_NOHYPHEN,
398 			sal_True, &pItem ))
399 			SetNoHyph( ((SvxNoHyphenItem*)pItem)->GetValue() );
400 		if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_BLINK,
401 			sal_True, &pItem ))
402 			SetBlink( ((SvxBlinkItem*)pItem)->GetValue() );
403         if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_ROTATE,
404             sal_True, &pItem ))
405             SetVertical( ((SvxCharRotateItem*)pItem)->GetValue() );
406         if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_BACKGROUND,
407             sal_True, &pItem ))
408             pBackColor = new Color( ((SvxBrushItem*)pItem)->GetColor() );
409         else
410             pBackColor = NULL;
411         const SfxPoolItem* pTwoLinesItem = 0;
412         if( SFX_ITEM_SET ==
413                 pAttrSet->GetItemState( RES_CHRATR_TWO_LINES, sal_True, &pTwoLinesItem ))
414             if ( ((SvxTwoLinesItem*)pTwoLinesItem)->GetValue() )
415                 SetVertical( 0 );
416 	}
417 	else
418 	{
419 		Invalidate();
420 		bNoHyph = sal_False;
421 		bBlink = sal_False;
422 	}
423 	bPaintBlank = sal_False;
424 	bPaintWrong = sal_False;
425 	ASSERT( aSub[SW_LATIN].IsTransparent(), "SwFont: Transparent revolution" );
426 }
427 
428 /*************************************************************************
429  *						class SwFont
430  *************************************************************************/
431 
SwFont(const SwFont & rFont)432 SwFont::SwFont( const SwFont &rFont )
433 {
434 	aSub[SW_LATIN] = rFont.aSub[SW_LATIN];
435 	aSub[SW_CJK] = rFont.aSub[SW_CJK];
436 	aSub[SW_CTL] = rFont.aSub[SW_CTL];
437     nActual = rFont.nActual;
438 	pBackColor = rFont.pBackColor ? new Color( *rFont.pBackColor ) : NULL;
439 	aUnderColor = rFont.GetUnderColor();
440 	aOverColor  = rFont.GetOverColor();
441     nToxCnt = 0;
442     nRefCnt = 0;
443     m_nMetaCount = 0;
444     m_nInputFieldCount = 0;
445 	bFntChg = rFont.bFntChg;
446 	bOrgChg = rFont.bOrgChg;
447 	bPaintBlank = rFont.bPaintBlank;
448 	bPaintWrong = sal_False;
449 	bURL = rFont.bURL;
450 	bGreyWave = rFont.bGreyWave;
451 	bNoColReplace = rFont.bNoColReplace;
452 	bNoHyph = rFont.bNoHyph;
453 	bBlink = rFont.bBlink;
454 }
455 
SwFont(const SwAttrSet * pAttrSet,const IDocumentSettingAccess * pIDocumentSettingAccess)456 SwFont::SwFont( const SwAttrSet* pAttrSet,
457                 const IDocumentSettingAccess* pIDocumentSettingAccess )
458 {
459     nActual = SW_LATIN;
460     nToxCnt = 0;
461     nRefCnt = 0;
462     m_nMetaCount = 0;
463     m_nInputFieldCount = 0;
464     bPaintBlank = sal_False;
465     bPaintWrong = sal_False;
466     bURL = sal_False;
467     bGreyWave = sal_False;
468     bNoColReplace = sal_False;
469     bNoHyph = pAttrSet->GetNoHyphenHere().GetValue();
470     bBlink = pAttrSet->GetBlink().GetValue();
471     bOrgChg = sal_True;
472     {
473         const SvxFontItem& rFont = pAttrSet->GetFont();
474         aSub[SW_LATIN].SetFamily( rFont.GetFamily() );
475         aSub[SW_LATIN].SetName( rFont.GetFamilyName() );
476         aSub[SW_LATIN].SetStyleName( rFont.GetStyleName() );
477         aSub[SW_LATIN].SetPitch( rFont.GetPitch() );
478         aSub[SW_LATIN].SetCharSet( rFont.GetCharSet() );
479         aSub[SW_LATIN].SvxFont::SetPropr( 100 );   // 100% der FontSize
480         Size aTmpSize = aSub[SW_LATIN].aSize;
481         aTmpSize.Height() = pAttrSet->GetSize().GetHeight();
482         aSub[SW_LATIN].SetSize( aTmpSize );
483         aSub[SW_LATIN].SetItalic( pAttrSet->GetPosture().GetPosture() );
484         aSub[SW_LATIN].SetWeight( pAttrSet->GetWeight().GetWeight() );
485         aSub[SW_LATIN].SetLanguage( pAttrSet->GetLanguage().GetLanguage() );
486     }
487 
488     {
489         const SvxFontItem& rFont = pAttrSet->GetCJKFont();
490         aSub[SW_CJK].SetFamily( rFont.GetFamily() );
491         aSub[SW_CJK].SetName( rFont.GetFamilyName() );
492         aSub[SW_CJK].SetStyleName( rFont.GetStyleName() );
493         aSub[SW_CJK].SetPitch( rFont.GetPitch() );
494         aSub[SW_CJK].SetCharSet( rFont.GetCharSet() );
495         aSub[SW_CJK].SvxFont::SetPropr( 100 );   // 100% der FontSize
496         Size aTmpSize = aSub[SW_CJK].aSize;
497         aTmpSize.Height() = pAttrSet->GetCJKSize().GetHeight();
498         aSub[SW_CJK].SetSize( aTmpSize );
499         aSub[SW_CJK].SetItalic( pAttrSet->GetCJKPosture().GetPosture() );
500         aSub[SW_CJK].SetWeight( pAttrSet->GetCJKWeight().GetWeight() );
501         LanguageType eNewLang = pAttrSet->GetCJKLanguage().GetLanguage();
502         aSub[SW_CJK].SetLanguage( eNewLang );
503         aSub[SW_LATIN].SetCJKContextLanguage( eNewLang );
504         aSub[SW_CJK].SetCJKContextLanguage( eNewLang );
505         aSub[SW_CTL].SetCJKContextLanguage( eNewLang );
506     }
507 
508     {
509         const SvxFontItem& rFont = pAttrSet->GetCTLFont();
510         aSub[SW_CTL].SetFamily( rFont.GetFamily() );
511         aSub[SW_CTL].SetName( rFont.GetFamilyName() );
512         aSub[SW_CTL].SetStyleName( rFont.GetStyleName() );
513         aSub[SW_CTL].SetPitch( rFont.GetPitch() );
514         aSub[SW_CTL].SetCharSet( rFont.GetCharSet() );
515         aSub[SW_CTL].SvxFont::SetPropr( 100 );   // 100% der FontSize
516         Size aTmpSize = aSub[SW_CTL].aSize;
517         aTmpSize.Height() = pAttrSet->GetCTLSize().GetHeight();
518         aSub[SW_CTL].SetSize( aTmpSize );
519         aSub[SW_CTL].SetItalic( pAttrSet->GetCTLPosture().GetPosture() );
520         aSub[SW_CTL].SetWeight( pAttrSet->GetCTLWeight().GetWeight() );
521         aSub[SW_CTL].SetLanguage( pAttrSet->GetCTLLanguage().GetLanguage() );
522     }
523 
524     const FontUnderline eUnderline = pAttrSet->GetUnderline().GetLineStyle();
525     if ( pAttrSet->GetCharHidden().GetValue() )
526         SetUnderline( UNDERLINE_DOTTED );
527     else
528         SetUnderline( eUnderline );
529     SetUnderColor( pAttrSet->GetUnderline().GetColor() );
530     SetOverline( pAttrSet->GetOverline().GetLineStyle() );
531     SetOverColor( pAttrSet->GetOverline().GetColor() );
532     SetEmphasisMark( pAttrSet->GetEmphasisMark().GetEmphasisMark() );
533     SetStrikeout( pAttrSet->GetCrossedOut().GetStrikeout() );
534     SetColor( pAttrSet->GetColor().GetValue() );
535     SetTransparent( sal_True );
536     SetAlign( ALIGN_BASELINE );
537     SetOutline( pAttrSet->GetContour().GetValue() );
538     SetShadow( pAttrSet->GetShadowed().GetValue() );
539     SetPropWidth( pAttrSet->GetCharScaleW().GetValue() );
540     SetRelief( (FontRelief)pAttrSet->GetCharRelief().GetValue() );
541 	if( pAttrSet->GetAutoKern().GetValue() )
542     {
543         SetAutoKern( ( !pIDocumentSettingAccess ||
544                        !pIDocumentSettingAccess->get(IDocumentSettingAccess::KERN_ASIAN_PUNCTUATION) ) ?
545                         KERNING_FONTSPECIFIC :
546                         KERNING_ASIAN );
547     }
548 	else
549     	SetAutoKern( 0 );
550     SetWordLineMode( pAttrSet->GetWordLineMode().GetValue() );
551     const SvxEscapementItem &rEsc = pAttrSet->GetEscapement();
552     SetEscapement( rEsc.GetEsc() );
553     if( aSub[SW_LATIN].IsEsc() )
554         SetProportion( rEsc.GetProp() );
555     SetCaseMap( pAttrSet->GetCaseMap().GetCaseMap() );
556     SetFixKerning( pAttrSet->GetKerning().GetValue() );
557     const SfxPoolItem* pItem;
558     if( SFX_ITEM_SET == pAttrSet->GetItemState( RES_CHRATR_BACKGROUND,
559         sal_True, &pItem ))
560         pBackColor = new Color( ((SvxBrushItem*)pItem)->GetColor() );
561     else
562         pBackColor = NULL;
563     const SvxTwoLinesItem& rTwoLinesItem = pAttrSet->Get2Lines();
564     if ( ! rTwoLinesItem.GetValue() )
565         SetVertical( pAttrSet->GetCharRotate().GetValue() );
566     else
567         SetVertical( 0 );
568 }
569 
operator =(const SwSubFont & rFont)570 SwSubFont& SwSubFont::operator=( const SwSubFont &rFont )
571 {
572 	SvxFont::operator=( rFont );
573 	pMagic = rFont.pMagic;
574 	nFntIndex = rFont.nFntIndex;
575 	nOrgHeight = rFont.nOrgHeight;
576 	nOrgAscent = rFont.nOrgAscent;
577 	nPropWidth = rFont.nPropWidth;
578 	aSize = rFont.aSize;
579 	return *this;
580 }
581 
operator =(const SwFont & rFont)582 SwFont& SwFont::operator=( const SwFont &rFont )
583 {
584 	aSub[SW_LATIN] = rFont.aSub[SW_LATIN];
585 	aSub[SW_CJK] = rFont.aSub[SW_CJK];
586 	aSub[SW_CTL] = rFont.aSub[SW_CTL];
587     nActual = rFont.nActual;
588 	delete pBackColor;
589 	pBackColor = rFont.pBackColor ? new Color( *rFont.pBackColor ) : NULL;
590 	aUnderColor = rFont.GetUnderColor();
591 	aOverColor  = rFont.GetOverColor();
592     nToxCnt = 0;
593     nRefCnt = 0;
594     m_nMetaCount = 0;
595     m_nInputFieldCount = 0;
596 	bFntChg = rFont.bFntChg;
597 	bOrgChg = rFont.bOrgChg;
598 	bPaintBlank = rFont.bPaintBlank;
599 	bPaintWrong = sal_False;
600 	bURL = rFont.bURL;
601 	bGreyWave = rFont.bGreyWave;
602 	bNoColReplace = rFont.bNoColReplace;
603 	bNoHyph = rFont.bNoHyph;
604 	bBlink = rFont.bBlink;
605 	return *this;
606 }
607 
608 /*************************************************************************
609  *						SwFont::GoMagic()
610  *************************************************************************/
611 
GoMagic(ViewShell * pSh,sal_uInt8 nWhich)612 void SwFont::GoMagic( ViewShell *pSh, sal_uInt8 nWhich )
613 {
614 	SwFntAccess aFntAccess( aSub[nWhich].pMagic, aSub[nWhich].nFntIndex,
615 							&aSub[nWhich], pSh, sal_True );
616 }
617 
618 /*************************************************************************
619  *						SwSubFont::IsSymbol()
620  *************************************************************************/
621 
IsSymbol(ViewShell * pSh)622 sal_Bool SwSubFont::IsSymbol( ViewShell *pSh )
623 {
624 	SwFntAccess aFntAccess( pMagic, nFntIndex, this, pSh, sal_False );
625 	return aFntAccess.Get()->IsSymbol();
626 }
627 
628 /*************************************************************************
629  *						SwSubFont::ChgFnt()
630  *************************************************************************/
631 
ChgFnt(ViewShell * pSh,OutputDevice & rOut)632 sal_Bool SwSubFont::ChgFnt( ViewShell *pSh, OutputDevice& rOut )
633 {
634     if ( pLastFont )
635 		pLastFont->Unlock();
636 	SwFntAccess aFntAccess( pMagic, nFntIndex, this, pSh, sal_True );
637 	SV_STAT( nChangeFont );
638 
639 	pLastFont = aFntAccess.Get();
640 
641     pLastFont->SetDevFont( pSh, rOut );
642 
643 	pLastFont->Lock();
644 	return UNDERLINE_NONE != GetUnderline() ||
645            UNDERLINE_NONE != GetOverline()  ||
646            STRIKEOUT_NONE != GetStrikeout();
647 }
648 
649 /*************************************************************************
650  *					  SwFont::ChgPhysFnt()
651  *************************************************************************/
652 
ChgPhysFnt(ViewShell * pSh,OutputDevice & rOut)653 void SwFont::ChgPhysFnt( ViewShell *pSh, OutputDevice& rOut )
654 {
655 	if( bOrgChg && aSub[nActual].IsEsc() )
656 	{
657 		const sal_uInt8 nOldProp = aSub[nActual].GetPropr();
658         SetProportion( 100 );
659         ChgFnt( pSh, rOut );
660 		SwFntAccess aFntAccess( aSub[nActual].pMagic, aSub[nActual].nFntIndex,
661 								&aSub[nActual], pSh );
662         aSub[nActual].nOrgHeight = aFntAccess.Get()->GetFontHeight( pSh, rOut );
663         aSub[nActual].nOrgAscent = aFntAccess.Get()->GetFontAscent( pSh, rOut );
664 		SetProportion( nOldProp );
665 		bOrgChg = sal_False;
666 	}
667 
668 	if( bFntChg )
669 	{
670         ChgFnt( pSh, rOut );
671 		bFntChg = bOrgChg;
672 	}
673     if( rOut.GetTextLineColor() != aUnderColor )
674         rOut.SetTextLineColor( aUnderColor );
675     if( rOut.GetOverlineColor() != aOverColor )
676         rOut.SetOverlineColor( aOverColor );
677 }
678 
679 /*************************************************************************
680  *						SwFont::CalcEscHeight()
681  *         Height = MaxAscent + MaxDescent
682  *      MaxAscent = Max (T1_ascent, T2_ascent + (Esc * T1_height) );
683  *     MaxDescent = Max (T1_height-T1_ascent,
684  * 						 T2_height-T2_ascent - (Esc * T1_height)
685  *************************************************************************/
686 
CalcEscHeight(const sal_uInt16 nOldHeight,const sal_uInt16 nOldAscent) const687 sal_uInt16 SwSubFont::CalcEscHeight( const sal_uInt16 nOldHeight,
688 							  const sal_uInt16 nOldAscent  ) const
689 {
690 	if( DFLT_ESC_AUTO_SUPER != GetEscapement() &&
691 		DFLT_ESC_AUTO_SUB != GetEscapement() )
692 	{
693 		long nDescent = nOldHeight - nOldAscent -
694 							 ( (long) nOrgHeight * GetEscapement() ) / 100L;
695 		const sal_uInt16 nDesc = ( nDescent>0 ) ? Max ( sal_uInt16(nDescent),
696 				   sal_uInt16(nOrgHeight - nOrgAscent) ) : nOrgHeight - nOrgAscent;
697 		return ( nDesc + CalcEscAscent( nOldAscent ) );
698 	}
699 	return nOrgHeight;
700 }
701 
_CheckKerning()702 short SwSubFont::_CheckKerning( )
703 {
704 	short nKernx = - short( Font::GetSize().Height() / 6 );
705 
706 	if ( nKernx < GetFixKerning() )
707 		return GetFixKerning();
708 	return nKernx;
709 }
710 
711 /*************************************************************************
712  *                    SwSubFont::GetAscent()
713  *************************************************************************/
714 
GetAscent(ViewShell * pSh,const OutputDevice & rOut)715 sal_uInt16 SwSubFont::GetAscent( ViewShell *pSh, const OutputDevice& rOut )
716 {
717 	sal_uInt16 nAscent;
718 	SwFntAccess aFntAccess( pMagic, nFntIndex, this, pSh );
719     nAscent = aFntAccess.Get()->GetFontAscent( pSh, rOut );
720 	if( GetEscapement() )
721 		nAscent = CalcEscAscent( nAscent );
722 	return nAscent;
723 }
724 
725 /*************************************************************************
726  *					  SwSubFont::GetHeight()
727  *************************************************************************/
728 
GetHeight(ViewShell * pSh,const OutputDevice & rOut)729 sal_uInt16 SwSubFont::GetHeight( ViewShell *pSh, const OutputDevice& rOut )
730 {
731 	SV_STAT( nGetTextSize );
732 	SwFntAccess aFntAccess( pMagic, nFntIndex, this, pSh );
733     const sal_uInt16 nHeight = aFntAccess.Get()->GetFontHeight( pSh, rOut );
734 	if ( GetEscapement() )
735 	{
736         const sal_uInt16 nAscent = aFntAccess.Get()->GetFontAscent( pSh, rOut );
737 		return CalcEscHeight( nHeight, nAscent ); // + nLeading;
738 	}
739 	return nHeight; // + nLeading;
740 }
741 
742 /*************************************************************************
743  *					  SwSubFont::_GetTxtSize()
744  *************************************************************************/
_GetTxtSize(SwDrawTextInfo & rInf)745 Size SwSubFont::_GetTxtSize( SwDrawTextInfo& rInf )
746 {
747     // Robust: Eigentlich sollte der Font bereits eingestellt sein, aber
748     // sicher ist sicher ...
749     if ( !pLastFont || pLastFont->GetOwner()!=pMagic ||
750          !IsSameInstance( rInf.GetpOut()->GetFont() ) )
751         ChgFnt( rInf.GetShell(), rInf.GetOut() );
752 
753     SwDigitModeModifier aDigitModeModifier( rInf.GetOut(), rInf.GetFont()->GetLanguage() );
754 
755     Size aTxtSize;
756     xub_StrLen nLn = ( rInf.GetLen() == STRING_LEN ? rInf.GetText().Len()
757                                                    : rInf.GetLen() );
758     rInf.SetLen( nLn );
759     if( IsCapital() && nLn )
760         aTxtSize = GetCapitalSize( rInf );
761     else
762     {
763         SV_STAT( nGetTextSize );
764         long nOldKern = rInf.GetKern();
765         const XubString &rOldTxt = rInf.GetText();
766         rInf.SetKern( CheckKerning() );
767         if ( !IsCaseMap() )
768             aTxtSize = pLastFont->GetTextSize( rInf );
769         else
770         {
771             String aTmp = CalcCaseMap( rInf.GetText() );
772             const XubString &rOldStr = rInf.GetText();
773             sal_Bool bCaseMapLengthDiffers(aTmp.Len() != rOldStr.Len());
774 
775             if(bCaseMapLengthDiffers && rInf.GetLen())
776             {
777                 // #108203#
778                 // If the length of the original string and the CaseMapped one
779                 // are different, it is necessary to handle the given text part as
780                 // a single snippet since it�s size may differ, too.
781                 xub_StrLen nOldIdx(rInf.GetIdx());
782                 xub_StrLen nOldLen(rInf.GetLen());
783                 const XubString aSnippet(rOldStr, nOldIdx, nOldLen);
784                 XubString aNewText(CalcCaseMap(aSnippet));
785 
786                 rInf.SetText( aNewText );
787                 rInf.SetIdx( 0 );
788                 rInf.SetLen( aNewText.Len() );
789 
790                 aTxtSize = pLastFont->GetTextSize( rInf );
791 
792                 rInf.SetIdx( nOldIdx );
793                 rInf.SetLen( nOldLen );
794             }
795             else
796             {
797                 rInf.SetText( aTmp );
798                 aTxtSize = pLastFont->GetTextSize( rInf );
799             }
800 
801             rInf.SetText( rOldStr );
802         }
803         rInf.SetKern( nOldKern );
804         rInf.SetText( rOldTxt );
805         // 15142: Ein Wort laenger als eine Zeile, beim Zeilenumbruch
806         //        hochgestellt, muss seine effektive Hoehe melden.
807         if( GetEscapement() )
808         {
809             const sal_uInt16 nAscent = pLastFont->GetFontAscent( rInf.GetShell(),
810                                                              rInf.GetOut() );
811             aTxtSize.Height() =
812                 (long)CalcEscHeight( (sal_uInt16)aTxtSize.Height(), nAscent);
813         }
814     }
815 
816     if (1==rInf.GetLen() && CH_TXT_ATR_FIELDSTART==rInf.GetText().GetChar(rInf.GetIdx()))
817     {
818         xub_StrLen nOldIdx(rInf.GetIdx());
819         xub_StrLen nOldLen(rInf.GetLen());
820         String aNewText=String::CreateFromAscii(CH_TXT_ATR_SUBST_FIELDSTART);
821         rInf.SetText( aNewText );
822         rInf.SetIdx( 0 );
823         rInf.SetLen( aNewText.Len() );
824         aTxtSize = pLastFont->GetTextSize( rInf );
825         rInf.SetIdx( nOldIdx );
826         rInf.SetLen( nOldLen );
827     }
828     else if (1==rInf.GetLen() && CH_TXT_ATR_FIELDEND==rInf.GetText().GetChar(rInf.GetIdx()))
829     {
830         xub_StrLen nOldIdx(rInf.GetIdx());
831         xub_StrLen nOldLen(rInf.GetLen());
832         String aNewText=String::CreateFromAscii(CH_TXT_ATR_SUBST_FIELDEND);
833         rInf.SetText( aNewText );
834         rInf.SetIdx( 0 );
835         rInf.SetLen( aNewText.Len() );
836         aTxtSize = pLastFont->GetTextSize( rInf );
837         rInf.SetIdx( nOldIdx );
838         rInf.SetLen( nOldLen );
839     }
840 
841     return aTxtSize;
842 }
843 
844 /*************************************************************************
845  *					  SwSubFont::_DrawText()
846  *************************************************************************/
847 
_DrawText(SwDrawTextInfo & rInf,const sal_Bool bGrey)848 void SwSubFont::_DrawText( SwDrawTextInfo &rInf, const sal_Bool bGrey )
849 {
850     rInf.SetGreyWave( bGrey );
851 	xub_StrLen nLn = rInf.GetText().Len();
852 	if( !rInf.GetLen() || !nLn )
853 		return;
854 	if( STRING_LEN == rInf.GetLen() )
855 		rInf.SetLen( nLn );
856 
857     FontUnderline nOldUnder = UNDERLINE_NONE;
858     SwUnderlineFont* pUnderFnt = 0;
859 
860     if( rInf.GetUnderFnt() )
861 	{
862 		nOldUnder = GetUnderline();
863 		SetUnderline( UNDERLINE_NONE );
864         pUnderFnt = rInf.GetUnderFnt();
865 	}
866 
867 	if( !pLastFont || pLastFont->GetOwner()!=pMagic )
868         ChgFnt( rInf.GetShell(), rInf.GetOut() );
869 
870     SwDigitModeModifier aDigitModeModifier( rInf.GetOut(), rInf.GetFont()->GetLanguage() );
871 
872 	Point aPos( rInf.GetPos() );
873 	const Point &rOld = rInf.GetPos();
874 	rInf.SetPos( aPos );
875 
876 	if( GetEscapement() )
877         CalcEsc( rInf, aPos );
878 
879     rInf.SetKern( CheckKerning() + rInf.GetSperren() / SPACING_PRECISION_FACTOR );
880 
881 	if( IsCapital() )
882         DrawCapital( rInf );
883 	else
884 	{
885 		SV_STAT( nDrawText );
886 		if ( !IsCaseMap() )
887 			pLastFont->DrawText( rInf );
888 		else
889 		{
890 			const XubString &rOldStr = rInf.GetText();
891 			XubString aString( CalcCaseMap( rOldStr ) );
892 			sal_Bool bCaseMapLengthDiffers(aString.Len() != rOldStr.Len());
893 
894 			if(bCaseMapLengthDiffers && rInf.GetLen())
895 			{
896 				// #108203#
897 				// If the length of the original string and the CaseMapped one
898 				// are different, it is necessary to handle the given text part as
899 				// a single snippet since it�s size may differ, too.
900 				xub_StrLen nOldIdx(rInf.GetIdx());
901 				xub_StrLen nOldLen(rInf.GetLen());
902 				const XubString aSnippet(rOldStr, nOldIdx, nOldLen);
903 				XubString aNewText = CalcCaseMap(aSnippet);
904 
905 				rInf.SetText( aNewText );
906 				rInf.SetIdx( 0 );
907 				rInf.SetLen( aNewText.Len() );
908 
909 				pLastFont->DrawText( rInf );
910 
911 				rInf.SetIdx( nOldIdx );
912 				rInf.SetLen( nOldLen );
913 			}
914 			else
915 			{
916 				rInf.SetText( aString );
917 				pLastFont->DrawText( rInf );
918 			}
919 
920 			rInf.SetText( rOldStr );
921 		}
922 	}
923 
924     if( pUnderFnt && nOldUnder != UNDERLINE_NONE )
925 	{
926 static sal_Char __READONLY_DATA sDoubleSpace[] = "  ";
927         Size aFontSize = _GetTxtSize( rInf );
928         const XubString &rOldStr = rInf.GetText();
929         XubString aStr( sDoubleSpace, RTL_TEXTENCODING_MS_1252 );
930 
931         xub_StrLen nOldIdx = rInf.GetIdx();
932 		xub_StrLen nOldLen = rInf.GetLen();
933         long nSpace = 0;
934 		if( rInf.GetSpace() )
935 		{
936             xub_StrLen nTmpEnd = nOldIdx + nOldLen;
937 			if( nTmpEnd > rOldStr.Len() )
938 				nTmpEnd = rOldStr.Len();
939 
940             const SwScriptInfo* pSI = rInf.GetScriptInfo();
941 
942             const sal_Bool bAsianFont =
943                 ( rInf.GetFont() && SW_CJK == rInf.GetFont()->GetActual() );
944             for( xub_StrLen nTmp = nOldIdx; nTmp < nTmpEnd; ++nTmp )
945             {
946                 if( CH_BLANK == rOldStr.GetChar( nTmp ) || bAsianFont ||
947                     ( nTmp + 1 < rOldStr.Len() && pSI &&
948                       i18n::ScriptType::ASIAN == pSI->ScriptType( nTmp + 1 ) ) )
949 					++nSpace;
950             }
951 
952             // if next portion if a hole portion we do not consider any
953             // extra space added because the last character was ASIAN
954             if ( nSpace && rInf.IsSpaceStop() && bAsianFont )
955                  --nSpace;
956 
957             nSpace *= rInf.GetSpace() / SPACING_PRECISION_FACTOR;
958 		}
959 
960         rInf.SetWidth( sal_uInt16(aFontSize.Width() + nSpace) );
961         rInf.SetText( aStr );
962 		rInf.SetIdx( 0 );
963 		rInf.SetLen( 2 );
964         SetUnderline( nOldUnder );
965         rInf.SetUnderFnt( 0 );
966 
967         // set position for underline font
968         rInf.SetPos( pUnderFnt->GetPos() );
969 
970         pUnderFnt->GetFont()._DrawStretchText( rInf );
971 
972         rInf.SetUnderFnt( pUnderFnt );
973         rInf.SetText( rOldStr );
974 		rInf.SetIdx( nOldIdx );
975 		rInf.SetLen( nOldLen );
976 	}
977 
978     rInf.SetPos( rOld );
979 }
980 
_DrawStretchText(SwDrawTextInfo & rInf)981 void SwSubFont::_DrawStretchText( SwDrawTextInfo &rInf )
982 {
983 	if( !rInf.GetLen() || !rInf.GetText().Len() )
984 		return;
985 
986     FontUnderline nOldUnder = UNDERLINE_NONE;
987     SwUnderlineFont* pUnderFnt = 0;
988 
989     if( rInf.GetUnderFnt() )
990 	{
991 		nOldUnder = GetUnderline();
992 		SetUnderline( UNDERLINE_NONE );
993         pUnderFnt = rInf.GetUnderFnt();
994 	}
995 
996     if ( !pLastFont || pLastFont->GetOwner() != pMagic )
997         ChgFnt( rInf.GetShell(), rInf.GetOut() );
998 
999     SwDigitModeModifier aDigitModeModifier( rInf.GetOut(), rInf.GetFont()->GetLanguage() );
1000 
1001     rInf.ApplyAutoColor();
1002 
1003 	Point aPos( rInf.GetPos() );
1004 
1005 	if( GetEscapement() )
1006         CalcEsc( rInf, aPos );
1007 
1008     rInf.SetKern( CheckKerning() + rInf.GetSperren() / SPACING_PRECISION_FACTOR );
1009     const Point &rOld = rInf.GetPos();
1010     rInf.SetPos( aPos );
1011 
1012 	if( IsCapital() )
1013 		DrawStretchCapital( rInf );
1014 	else
1015 	{
1016 		SV_STAT( nDrawStretchText );
1017 
1018         if ( rInf.GetFrm() )
1019         {
1020             if ( rInf.GetFrm()->IsRightToLeft() )
1021                 rInf.GetFrm()->SwitchLTRtoRTL( aPos );
1022 
1023             if ( rInf.GetFrm()->IsVertical() )
1024                 rInf.GetFrm()->SwitchHorizontalToVertical( aPos );
1025         }
1026 
1027 		if ( !IsCaseMap() )
1028 			rInf.GetOut().DrawStretchText( aPos, rInf.GetWidth(),
1029 							rInf.GetText(), rInf.GetIdx(), rInf.GetLen() );
1030 		else
1031 			rInf.GetOut().DrawStretchText( aPos, rInf.GetWidth(), CalcCaseMap(
1032 							rInf.GetText() ), rInf.GetIdx(), rInf.GetLen() );
1033 	}
1034 
1035     if( pUnderFnt && nOldUnder != UNDERLINE_NONE )
1036 	{
1037 static sal_Char __READONLY_DATA sDoubleSpace[] = "  ";
1038 		const XubString &rOldStr = rInf.GetText();
1039 		XubString aStr( sDoubleSpace, RTL_TEXTENCODING_MS_1252 );
1040 		xub_StrLen nOldIdx = rInf.GetIdx();
1041 		xub_StrLen nOldLen = rInf.GetLen();
1042 		rInf.SetText( aStr );
1043 		rInf.SetIdx( 0 );
1044 		rInf.SetLen( 2 );
1045 		SetUnderline( nOldUnder );
1046         rInf.SetUnderFnt( 0 );
1047 
1048         // set position for underline font
1049         rInf.SetPos( pUnderFnt->GetPos() );
1050 
1051         pUnderFnt->GetFont()._DrawStretchText( rInf );
1052 
1053         rInf.SetUnderFnt( pUnderFnt );
1054         rInf.SetText( rOldStr );
1055 		rInf.SetIdx( nOldIdx );
1056 		rInf.SetLen( nOldLen );
1057 	}
1058 
1059     rInf.SetPos( rOld );
1060 }
1061 
1062 /*************************************************************************
1063  *					  SwSubFont::_GetCrsrOfst()
1064  *************************************************************************/
1065 
_GetCrsrOfst(SwDrawTextInfo & rInf)1066 xub_StrLen SwSubFont::_GetCrsrOfst( SwDrawTextInfo& rInf )
1067 {
1068 	if ( !pLastFont || pLastFont->GetOwner()!=pMagic )
1069         ChgFnt( rInf.GetShell(), rInf.GetOut() );
1070 
1071     SwDigitModeModifier aDigitModeModifier( rInf.GetOut(), rInf.GetFont()->GetLanguage() );
1072 
1073 	xub_StrLen nLn = rInf.GetLen() == STRING_LEN ? rInf.GetText().Len()
1074 												 : rInf.GetLen();
1075 	rInf.SetLen( nLn );
1076 	xub_StrLen nCrsr = 0;
1077 	if( IsCapital() && nLn )
1078 		nCrsr = GetCapitalCrsrOfst( rInf );
1079 	else
1080 	{
1081 		const XubString &rOldTxt = rInf.GetText();
1082         long nOldKern = rInf.GetKern();
1083 		rInf.SetKern( CheckKerning() );
1084 		SV_STAT( nGetTextSize );
1085 		if ( !IsCaseMap() )
1086 			nCrsr = pLastFont->GetCrsrOfst( rInf );
1087 		else
1088 		{
1089 			String aTmp = CalcCaseMap( rInf.GetText() );
1090 			rInf.SetText( aTmp );
1091 			nCrsr = pLastFont->GetCrsrOfst( rInf );
1092 		}
1093 		rInf.SetKern( nOldKern );
1094 		rInf.SetText( rOldTxt );
1095 	}
1096 	return nCrsr;
1097 }
1098 
1099 /*************************************************************************
1100  *                    SwSubFont::CalcEsc()
1101  *************************************************************************/
1102 
CalcEsc(SwDrawTextInfo & rInf,Point & rPos)1103 void SwSubFont::CalcEsc( SwDrawTextInfo& rInf, Point& rPos )
1104 {
1105     long nOfst;
1106 
1107     sal_uInt16 nDir = UnMapDirection(
1108                 GetOrientation(), rInf.GetFrm() && rInf.GetFrm()->IsVertical() );
1109 
1110     switch ( GetEscapement() )
1111     {
1112     case DFLT_ESC_AUTO_SUB :
1113         nOfst = nOrgHeight - nOrgAscent -
1114             pLastFont->GetFontHeight( rInf.GetShell(), rInf.GetOut() ) +
1115             pLastFont->GetFontAscent( rInf.GetShell(), rInf.GetOut() );
1116 
1117         switch ( nDir )
1118         {
1119         case 0 :
1120             rPos.Y() += nOfst;
1121             break;
1122         case 900 :
1123             rPos.X() += nOfst;
1124             break;
1125         case 2700 :
1126             rPos.X() -= nOfst;
1127             break;
1128         }
1129 
1130         break;
1131     case DFLT_ESC_AUTO_SUPER :
1132         nOfst = pLastFont->GetFontAscent( rInf.GetShell(), rInf.GetOut() ) -
1133                 nOrgAscent;
1134 
1135 
1136         switch ( nDir )
1137         {
1138         case 0 :
1139             rPos.Y() += nOfst;
1140             break;
1141         case 900 :
1142             rPos.X() += nOfst;
1143             break;
1144         case 2700 :
1145             rPos.X() -= nOfst;
1146             break;
1147         }
1148 
1149         break;
1150     default :
1151         nOfst = ((long)nOrgHeight * GetEscapement()) / 100L;
1152 
1153         switch ( nDir )
1154         {
1155         case 0 :
1156             rPos.Y() -= nOfst;
1157             break;
1158         case 900 :
1159             rPos.X() -= nOfst;
1160             break;
1161         case 2700 :
1162             rPos.X() += nOfst;
1163             break;
1164         }
1165     }
1166 }
1167 
1168 // used during painting of small capitals
Shift(sal_uInt16 nDir)1169 void SwDrawTextInfo::Shift( sal_uInt16 nDir )
1170 {
1171     ASSERT( bPos, "DrawTextInfo: Undefined Position" );
1172     ASSERT( bSize, "DrawTextInfo: Undefined Width" );
1173 
1174     const sal_Bool bBidiPor = ( GetFrm() && GetFrm()->IsRightToLeft() ) !=
1175                           ( 0 != ( TEXT_LAYOUT_BIDI_RTL & GetpOut()->GetLayoutMode() ) );
1176 
1177     nDir = bBidiPor ?
1178             1800 :
1179             UnMapDirection( nDir, GetFrm() && GetFrm()->IsVertical() );
1180 
1181     switch ( nDir )
1182     {
1183     case 0 :
1184         ((Point*)pPos)->X() += GetSize().Width();
1185         break;
1186     case 900 :
1187         ASSERT( ((Point*)pPos)->Y() >= GetSize().Width(), "Going underground" );
1188         ((Point*)pPos)->Y() -= GetSize().Width();
1189         break;
1190     case 1800 :
1191         ((Point*)pPos)->X() -= GetSize().Width();
1192         break;
1193     case 2700 :
1194         ((Point*)pPos)->Y() += GetSize().Width();
1195         break;
1196     }
1197 }
1198 
1199 /*************************************************************************
1200  *                      SwUnderlineFont::~SwUnderlineFont
1201  *
1202  * Used for the "continuous underline" feature.
1203  *************************************************************************/
1204 
SwUnderlineFont(SwFont & rFnt,const Point & rPoint)1205 SwUnderlineFont::SwUnderlineFont( SwFont& rFnt, const Point& rPoint )
1206         : aPos( rPoint ), pFnt( &rFnt )
1207 {
1208 };
1209 
~SwUnderlineFont()1210 SwUnderlineFont::~SwUnderlineFont()
1211 {
1212      delete pFnt;
1213 }
1214 
1215 //Helper for filters to find true lineheight of a font
AttrSetToLineHeight(const IDocumentSettingAccess & rIDocumentSettingAccess,const SwAttrSet & rSet,const OutputDevice & rOut,sal_Int16 nScript)1216 long AttrSetToLineHeight( const IDocumentSettingAccess& rIDocumentSettingAccess,
1217                           const SwAttrSet &rSet,
1218                           const OutputDevice &rOut, sal_Int16 nScript)
1219 {
1220     SwFont aFont(&rSet, &rIDocumentSettingAccess);
1221     sal_uInt8 nActual;
1222     switch (nScript)
1223     {
1224         default:
1225         case i18n::ScriptType::LATIN:
1226             nActual = SW_LATIN;
1227             break;
1228         case i18n::ScriptType::ASIAN:
1229             nActual = SW_CJK;
1230             break;
1231         case i18n::ScriptType::COMPLEX:
1232             nActual = SW_CTL;
1233             break;
1234     }
1235     aFont.SetActual(nActual);
1236 
1237     OutputDevice &rMutableOut = const_cast<OutputDevice &>(rOut);
1238     const Font aOldFont(rMutableOut.GetFont());
1239 
1240     rMutableOut.SetFont(aFont.GetActualFont());
1241     long nHeight = rMutableOut.GetTextHeight();
1242 
1243     rMutableOut.SetFont(aOldFont);
1244     return nHeight;
1245 }
1246