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_svx.hxx" 26 #include "EnhancedCustomShapeFontWork.hxx" 27 #include <tools/solar.h> // UINTXX 28 #include <svx/svddef.hxx> 29 #include <svx/svdogrp.hxx> 30 #include <svx/svdopath.hxx> 31 #include <vcl/metric.hxx> 32 #include <svx/svdpage.hxx> 33 #include <svx/sdasitm.hxx> 34 #include <svx/sdasaitm.hxx> 35 #include <svx/sdtfsitm.hxx> 36 #include <vcl/virdev.hxx> 37 #include <svx/svditer.hxx> 38 #include <vcl/metric.hxx> 39 #include <editeng/eeitem.hxx> 40 #include <editeng/frmdiritem.hxx> 41 #include <editeng/fontitem.hxx> 42 #include <editeng/postitem.hxx> 43 #include <editeng/wghtitem.hxx> 44 #include <editeng/charscaleitem.hxx> 45 #include "svx/EnhancedCustomShapeTypeNames.hxx" 46 #include <svx/svdorect.hxx> 47 #include <svx/svdoashp.hxx> 48 #include <editeng/outliner.hxx> 49 #include <editeng/outlobj.hxx> 50 #include <editeng/editobj.hxx> 51 #include <editeng/editeng.hxx> 52 #include <svx/svdmodel.hxx> 53 #include <vector> 54 #include <numeric> 55 #include <algorithm> 56 #include <comphelper/processfactory.hxx> 57 #ifndef _COM_SUN_STAR_I18N_SCRIPTTYPE_HDL_ 58 #include <com/sun/star/i18n/ScriptType.hdl> 59 #endif 60 #include <basegfx/polygon/b2dpolypolygontools.hxx> 61 #include <com/sun/star/lang/XMultiServiceFactory.hpp> 62 #ifndef _COM_SUN_STAR_I18N_CHARACTERITERATORMODE_HDL_ 63 #include <com/sun/star/i18n/CharacterIteratorMode.hdl> 64 #endif 65 #include <basegfx/polygon/b2dpolygontools.hxx> 66 67 using namespace com::sun::star; 68 using namespace com::sun::star::uno; 69 70 typedef std::vector< std::vector< double > > PolyPolygonDistances; 71 72 struct FWCharacterData // representing a single character 73 { 74 std::vector< PolyPolygon > vOutlines; 75 Rectangle aBoundRect; 76 }; 77 struct FWParagraphData // representing a single paragraph 78 { 79 rtl::OUString aString; 80 std::vector< FWCharacterData > vCharacters; 81 Rectangle aBoundRect; 82 sal_Int16 nFrameDirection; 83 }; 84 struct FWTextArea // representing multiple concluding paragraphs 85 { 86 std::vector< FWParagraphData > vParagraphs; 87 Rectangle aBoundRect; 88 }; 89 struct FWData // representing the whole text 90 { 91 std::vector< FWTextArea > vTextAreas; 92 double fHorizontalTextScaling; 93 sal_uInt32 nMaxParagraphsPerTextArea; 94 sal_Int32 nSingleLineHeight; 95 sal_Bool bSingleLineMode; 96 }; 97 98 99 sal_Bool InitializeFontWorkData( const SdrObject* pCustomShape, const sal_uInt16 nOutlinesCount2d, FWData& rFWData ) 100 { 101 sal_Bool bNoErr = sal_False; 102 sal_Bool bSingleLineMode = sal_False; 103 sal_uInt16 nTextAreaCount = nOutlinesCount2d; 104 if ( nOutlinesCount2d & 1 ) 105 bSingleLineMode = sal_True; 106 else 107 nTextAreaCount >>= 1; 108 109 if ( nTextAreaCount ) 110 { 111 rFWData.bSingleLineMode = bSingleLineMode; 112 113 // setting the strings 114 OutlinerParaObject* pParaObj = ((SdrObjCustomShape*)pCustomShape)->GetOutlinerParaObject(); 115 if ( pParaObj ) 116 { 117 const EditTextObject& rTextObj = pParaObj->GetTextObject(); 118 sal_Int64 nParagraphsLeft = rTextObj.GetParagraphCount(); 119 120 rFWData.nMaxParagraphsPerTextArea = ( ( nParagraphsLeft - 1 ) / nTextAreaCount ) + 1; 121 sal_Int16 j = 0; 122 while( nParagraphsLeft && nTextAreaCount ) 123 { 124 FWTextArea aTextArea; 125 sal_Int32 i, nParagraphs = ( ( nParagraphsLeft - 1 ) / nTextAreaCount ) + 1; 126 for ( i = 0; i < nParagraphs; i++, j++ ) 127 { 128 FWParagraphData aParagraphData; 129 aParagraphData.aString = rTextObj.GetText( j ); 130 131 const SfxItemSet& rParaSet = rTextObj.GetParaAttribs( j ); // retrieving some paragraph attributes 132 aParagraphData.nFrameDirection = ((SvxFrameDirectionItem&)rParaSet.Get( EE_PARA_WRITINGDIR )).GetValue(); 133 aTextArea.vParagraphs.push_back( aParagraphData ); 134 } 135 rFWData.vTextAreas.push_back( aTextArea ); 136 nParagraphsLeft -= nParagraphs; 137 nTextAreaCount--; 138 } 139 bNoErr = sal_True; 140 } 141 } 142 return bNoErr; 143 } 144 145 double GetLength( const Polygon& rPolygon ) 146 { 147 double fLength = 0; 148 if ( rPolygon.GetSize() > 1 ) 149 { 150 sal_uInt16 nCount = rPolygon.GetSize(); 151 while( --nCount ) 152 fLength += ((Polygon&)rPolygon).CalcDistance( nCount, nCount - 1 ); 153 } 154 return fLength; 155 } 156 157 158 /* CalculateHorizontalScalingFactor returns the horizontal scaling factor for 159 the whole text object, so that each text will match its corresponding 2d Outline */ 160 void CalculateHorizontalScalingFactor( const SdrObject* pCustomShape, 161 FWData& rFWData, const PolyPolygon& rOutline2d ) 162 { 163 double fScalingFactor = 1.0; 164 sal_Bool bScalingFactorDefined = sal_False; 165 166 sal_uInt16 i = 0; 167 sal_Bool bSingleLineMode = sal_False; 168 sal_uInt16 nOutlinesCount2d = rOutline2d.Count(); 169 170 Font aFont; 171 SvxFontItem& rFontItem = (SvxFontItem&)pCustomShape->GetMergedItem( EE_CHAR_FONTINFO ); 172 aFont.SetHeight( pCustomShape->GetLogicRect().GetHeight() / rFWData.nMaxParagraphsPerTextArea ); 173 aFont.SetAlign( ALIGN_TOP ); 174 aFont.SetName( rFontItem.GetFamilyName() ); 175 aFont.SetFamily( rFontItem.GetFamily() ); 176 aFont.SetStyleName( rFontItem.GetStyleName() ); 177 aFont.SetOrientation( 0 ); 178 // initializing virtual device 179 180 VirtualDevice aVirDev( 1 ); 181 aVirDev.SetMapMode( MAP_100TH_MM ); 182 aVirDev.SetFont( aFont ); 183 184 if ( nOutlinesCount2d & 1 ) 185 bSingleLineMode = sal_True; 186 187 std::vector< FWTextArea >::iterator aTextAreaIter = rFWData.vTextAreas.begin(); 188 std::vector< FWTextArea >::iterator aTextAreaIEnd = rFWData.vTextAreas.end(); 189 while( aTextAreaIter != aTextAreaIEnd ) 190 { 191 // calculating the width of the corresponding 2d text area 192 double fWidth = GetLength( rOutline2d.GetObject( i++ ) ); 193 if ( !bSingleLineMode ) 194 { 195 fWidth += GetLength( rOutline2d.GetObject( i++ ) ); 196 fWidth /= 2.0; 197 } 198 std::vector< FWParagraphData >::const_iterator aParagraphIter( aTextAreaIter->vParagraphs.begin() ); 199 std::vector< FWParagraphData >::const_iterator aParagraphIEnd( aTextAreaIter->vParagraphs.end() ); 200 while( aParagraphIter != aParagraphIEnd ) 201 { 202 double fTextWidth = aVirDev.GetTextWidth( aParagraphIter->aString ); 203 if ( fTextWidth > 0.0 ) 204 { 205 double fScale = fWidth / fTextWidth; 206 if ( !bScalingFactorDefined ) 207 { 208 fScalingFactor = fScale; 209 bScalingFactorDefined = sal_True; 210 } 211 else 212 { 213 if ( fScale < fScalingFactor ) 214 fScalingFactor = fScale; 215 } 216 } 217 aParagraphIter++; 218 } 219 aTextAreaIter++; 220 } 221 rFWData.fHorizontalTextScaling = fScalingFactor; 222 } 223 224 void GetTextAreaOutline( const FWData& rFWData, const SdrObject* pCustomShape, FWTextArea& rTextArea, sal_Bool bSameLetterHeights ) 225 { 226 sal_Bool bIsVertical = ((SdrObjCustomShape*)pCustomShape)->IsVerticalWriting(); 227 sal_Int32 nVerticalOffset = rFWData.nMaxParagraphsPerTextArea > rTextArea.vParagraphs.size() 228 ? rFWData.nSingleLineHeight / 2 : 0; 229 230 std::vector< FWParagraphData >::iterator aParagraphIter( rTextArea.vParagraphs.begin() ); 231 std::vector< FWParagraphData >::iterator aParagraphIEnd( rTextArea.vParagraphs.end() ); 232 while( aParagraphIter != aParagraphIEnd ) 233 { 234 const rtl::OUString& rText = aParagraphIter->aString; 235 if ( rText.getLength() ) 236 { 237 // generating vcl/font 238 sal_uInt16 nScriptType = i18n::ScriptType::LATIN; 239 Reference< i18n::XBreakIterator > xBI( EnhancedCustomShapeFontWork::GetBreakIterator() ); 240 if ( xBI.is() ) 241 { 242 nScriptType = xBI->getScriptType( rText, 0 ); 243 sal_uInt16 nChg = 0; 244 if( i18n::ScriptType::WEAK == nScriptType ) 245 { 246 nChg = (xub_StrLen)xBI->endOfScript( rText, nChg, nScriptType ); 247 if( nChg < rText.getLength() ) 248 nScriptType = xBI->getScriptType( rText, nChg ); 249 else 250 nScriptType = i18n::ScriptType::LATIN; 251 } 252 } 253 sal_uInt16 nFntItm = EE_CHAR_FONTINFO; 254 if ( nScriptType == i18n::ScriptType::COMPLEX ) 255 nFntItm = EE_CHAR_FONTINFO_CTL; 256 else if ( nScriptType == i18n::ScriptType::ASIAN ) 257 nFntItm = EE_CHAR_FONTINFO_CJK; 258 SvxFontItem& rFontItem = (SvxFontItem&)pCustomShape->GetMergedItem( nFntItm ); 259 Font aFont; 260 aFont.SetHeight( rFWData.nSingleLineHeight ); 261 aFont.SetAlign( ALIGN_TOP ); 262 // aFont.SetAlign( ) 263 264 aFont.SetName( rFontItem.GetFamilyName() ); 265 aFont.SetFamily( rFontItem.GetFamily() ); 266 aFont.SetStyleName( rFontItem.GetStyleName() ); 267 aFont.SetOrientation( 0 ); 268 269 SvxPostureItem& rPostureItem = (SvxPostureItem&)pCustomShape->GetMergedItem( EE_CHAR_ITALIC ); 270 aFont.SetItalic( rPostureItem.GetPosture() ); 271 272 SvxWeightItem& rWeightItem = (SvxWeightItem&)pCustomShape->GetMergedItem( EE_CHAR_WEIGHT ); 273 aFont.SetWeight( rWeightItem.GetWeight() ); 274 275 // initializing virtual device 276 VirtualDevice aVirDev( 1 ); 277 aVirDev.SetMapMode( MAP_100TH_MM ); 278 aVirDev.SetFont( aFont ); 279 aVirDev.EnableRTL( sal_True ); 280 if ( aParagraphIter->nFrameDirection == FRMDIR_HORI_RIGHT_TOP ) 281 aVirDev.SetLayoutMode( TEXT_LAYOUT_BIDI_RTL ); 282 283 SvxCharScaleWidthItem& rCharScaleWidthItem = (SvxCharScaleWidthItem&)pCustomShape->GetMergedItem( EE_CHAR_FONTWIDTH ); 284 sal_uInt16 nCharScaleWidth = rCharScaleWidthItem.GetValue(); 285 sal_Int32* pDXArry = NULL; 286 sal_Int32 nWidth = 0; 287 288 // VERTICAL 289 if ( bIsVertical ) 290 { 291 // vertical _> each single character needs to be rotated by 90 292 sal_Int32 i; 293 sal_Int32 nHeight = 0; 294 Rectangle aSingleCharacterUnion; 295 for ( i = 0; i < rText.getLength(); i++ ) 296 { 297 FWCharacterData aCharacterData; 298 rtl::OUString aCharText( (sal_Unicode)rText[ i ] ); 299 if ( aVirDev.GetTextOutlines( aCharacterData.vOutlines, aCharText, 0, 0, STRING_LEN, sal_True, nWidth, pDXArry ) ) 300 { 301 sal_Int32 nTextWidth = aVirDev.GetTextWidth( aCharText, 0, STRING_LEN ); 302 std::vector< PolyPolygon >::iterator aOutlineIter = aCharacterData.vOutlines.begin(); 303 std::vector< PolyPolygon >::iterator aOutlineIEnd = aCharacterData.vOutlines.end(); 304 if ( aOutlineIter == aOutlineIEnd ) 305 { 306 nHeight += rFWData.nSingleLineHeight; 307 } 308 else 309 { 310 while ( aOutlineIter != aOutlineIEnd ) 311 { 312 // rotating 313 aOutlineIter->Rotate( Point( nTextWidth / 2, rFWData.nSingleLineHeight / 2 ), 900 ); 314 aCharacterData.aBoundRect.Union( aOutlineIter->GetBoundRect() ); 315 aOutlineIter++; 316 } 317 aOutlineIter = aCharacterData.vOutlines.begin(); 318 aOutlineIEnd = aCharacterData.vOutlines.end(); 319 while ( aOutlineIter != aOutlineIEnd ) 320 { 321 sal_Int32 nM = - aCharacterData.aBoundRect.Left() + nHeight; 322 aOutlineIter->Move( nM, 0 ); 323 aCharacterData.aBoundRect.Move( nM, 0 ); 324 aOutlineIter++; 325 } 326 nHeight += aCharacterData.aBoundRect.GetWidth() + ( rFWData.nSingleLineHeight / 5 ); 327 aSingleCharacterUnion.Union( aCharacterData.aBoundRect ); 328 } 329 } 330 aParagraphIter->vCharacters.push_back( aCharacterData ); 331 } 332 std::vector< FWCharacterData >::iterator aCharacterIter( aParagraphIter->vCharacters.begin() ); 333 std::vector< FWCharacterData >::iterator aCharacterIEnd( aParagraphIter->vCharacters.end() ); 334 while ( aCharacterIter != aCharacterIEnd ) 335 { 336 std::vector< PolyPolygon >::iterator aOutlineIter( aCharacterIter->vOutlines.begin() ); 337 std::vector< PolyPolygon >::iterator aOutlineIEnd( aCharacterIter->vOutlines.end() ); 338 while ( aOutlineIter != aOutlineIEnd ) 339 { 340 aOutlineIter->Move( ( aSingleCharacterUnion.GetWidth() - aCharacterIter->aBoundRect.GetWidth() ) / 2, 0 ); 341 aOutlineIter++; 342 } 343 aCharacterIter++; 344 } 345 } 346 else 347 { 348 if ( ( nCharScaleWidth != 100 ) && nCharScaleWidth ) 349 { // applying character spacing 350 pDXArry = new sal_Int32[ rText.getLength() ]; 351 aVirDev.GetTextArray( rText, pDXArry, 0, STRING_LEN ); 352 FontMetric aFontMetric( aVirDev.GetFontMetric() ); 353 aFont.SetWidth( (sal_Int32)( (double)aFontMetric.GetWidth() * ( (double)100 / (double)nCharScaleWidth ) ) ); 354 aVirDev.SetFont( aFont ); 355 } 356 FWCharacterData aCharacterData; 357 if ( aVirDev.GetTextOutlines( aCharacterData.vOutlines, rText, 0, 0, STRING_LEN, sal_True, nWidth, pDXArry ) ) 358 { 359 aParagraphIter->vCharacters.push_back( aCharacterData ); 360 } 361 362 /* trying to retrieve each single character _> is not working well 363 sal_Int32 i; 364 for ( i = 0; i < rText.getLength(); i++ ) 365 { 366 FWCharacterData aCharacterData; 367 if ( aVirDev.GetTextOutlines( aCharacterData.vOutlines, rText, 0, i, 1, sal_True, nWidth, pDXArry ) ) 368 { 369 std::vector< PolyPolygon >::iterator aOutlineIter = aCharacterData.vOutlines.begin(); 370 std::vector< PolyPolygon >::iterator aOutlineIEnd = aCharacterData.vOutlines.end(); 371 while ( aOutlineIter != aOutlineIEnd ) 372 { 373 aCharacterData.aBoundRect.Union( aOutlineIter->GetBoundRect() ); 374 aOutlineIter++; 375 } 376 } 377 aParagraphIter->vCharacters.push_back( aCharacterData ); 378 } 379 */ 380 } 381 delete[] pDXArry; 382 383 // veritcal alignment 384 std::vector< FWCharacterData >::iterator aCharacterIter( aParagraphIter->vCharacters.begin() ); 385 std::vector< FWCharacterData >::iterator aCharacterIEnd ( aParagraphIter->vCharacters.end() ); 386 while ( aCharacterIter != aCharacterIEnd ) 387 { 388 std::vector< PolyPolygon >::iterator aOutlineIter( aCharacterIter->vOutlines.begin() ); 389 std::vector< PolyPolygon >::iterator aOutlineIEnd( aCharacterIter->vOutlines.end() ); 390 while( aOutlineIter != aOutlineIEnd ) 391 { 392 393 PolyPolygon& rPolyPoly = *aOutlineIter++; 394 395 if ( nVerticalOffset ) 396 rPolyPoly.Move( 0, nVerticalOffset ); 397 398 // retrieving the boundrect for the paragraph 399 Rectangle aBoundRect( rPolyPoly.GetBoundRect() ); 400 aParagraphIter->aBoundRect.Union( aBoundRect ); 401 } 402 aCharacterIter++; 403 } 404 } 405 // updating the boundrect for the text area by merging the current paragraph boundrect 406 if ( aParagraphIter->aBoundRect.IsEmpty() ) 407 { 408 if ( rTextArea.aBoundRect.IsEmpty() ) 409 rTextArea.aBoundRect = Rectangle( Point( 0, 0 ), Size( 1, rFWData.nSingleLineHeight ) ); 410 else 411 rTextArea.aBoundRect.Bottom() += rFWData.nSingleLineHeight; 412 } 413 else 414 { 415 Rectangle& rParagraphBoundRect = aParagraphIter->aBoundRect; 416 rTextArea.aBoundRect.Union( rParagraphBoundRect ); 417 418 if ( bSameLetterHeights ) 419 { 420 std::vector< FWCharacterData >::iterator aCharacterIter( aParagraphIter->vCharacters.begin() ); 421 std::vector< FWCharacterData >::iterator aCharacterIEnd( aParagraphIter->vCharacters.end() ); 422 while ( aCharacterIter != aCharacterIEnd ) 423 { 424 std::vector< PolyPolygon >::iterator aOutlineIter( aCharacterIter->vOutlines.begin() ); 425 std::vector< PolyPolygon >::iterator aOutlineIEnd( aCharacterIter->vOutlines.end() ); 426 while( aOutlineIter != aOutlineIEnd ) 427 { 428 Rectangle aPolyPolyBoundRect( aOutlineIter->GetBoundRect() ); 429 if ( aPolyPolyBoundRect.GetHeight() != rParagraphBoundRect.GetHeight() ) 430 aOutlineIter->Scale( 1.0, (double)rParagraphBoundRect.GetHeight() / aPolyPolyBoundRect.GetHeight() ); 431 aPolyPolyBoundRect = aOutlineIter->GetBoundRect(); 432 sal_Int32 nMove = aPolyPolyBoundRect.Top() - rParagraphBoundRect.Top(); 433 if ( nMove ) 434 aOutlineIter->Move( 0, -nMove ); 435 aOutlineIter++; 436 } 437 aCharacterIter++; 438 } 439 } 440 } 441 if ( bIsVertical ) 442 nVerticalOffset -= rFWData.nSingleLineHeight; 443 else 444 nVerticalOffset += rFWData.nSingleLineHeight; 445 aParagraphIter++; 446 } 447 } 448 449 void GetFontWorkOutline( FWData& rFWData, const SdrObject* pCustomShape ) 450 { 451 SdrTextHorzAdjust eHorzAdjust( ((SdrTextHorzAdjustItem&)pCustomShape->GetMergedItem( SDRATTR_TEXT_HORZADJUST )).GetValue() ); 452 SdrFitToSizeType eFTS( ((SdrTextFitToSizeTypeItem&)pCustomShape->GetMergedItem( SDRATTR_TEXT_FITTOSIZE )).GetValue() ); 453 454 std::vector< FWTextArea >::iterator aTextAreaIter = rFWData.vTextAreas.begin(); 455 std::vector< FWTextArea >::iterator aTextAreaIEnd = rFWData.vTextAreas.end(); 456 457 rFWData.nSingleLineHeight = (sal_Int32)( ( (double)pCustomShape->GetLogicRect().GetHeight() 458 / rFWData.nMaxParagraphsPerTextArea ) * rFWData.fHorizontalTextScaling ); 459 460 sal_Bool bSameLetterHeights = sal_False; 461 SdrCustomShapeGeometryItem& rGeometryItem = (SdrCustomShapeGeometryItem&)pCustomShape->GetMergedItem( SDRATTR_CUSTOMSHAPE_GEOMETRY ); 462 const rtl::OUString sTextPath( RTL_CONSTASCII_USTRINGPARAM ( "TextPath" ) ); 463 const rtl::OUString sSameLetterHeights( RTL_CONSTASCII_USTRINGPARAM ( "SameLetterHeights" ) ); 464 com::sun::star::uno::Any* pAny = rGeometryItem.GetPropertyValueByName( sTextPath, sSameLetterHeights ); 465 if ( pAny ) 466 *pAny >>= bSameLetterHeights; 467 468 while ( aTextAreaIter != aTextAreaIEnd ) 469 { 470 GetTextAreaOutline( rFWData, pCustomShape, *aTextAreaIter, bSameLetterHeights ); 471 if ( eFTS == SDRTEXTFIT_ALLLINES ) 472 { 473 std::vector< FWParagraphData >::iterator aParagraphIter( aTextAreaIter->vParagraphs.begin() ); 474 std::vector< FWParagraphData >::iterator aParagraphIEnd( aTextAreaIter->vParagraphs.end() ); 475 while ( aParagraphIter != aParagraphIEnd ) 476 { 477 sal_Int32 nParaWidth = aParagraphIter->aBoundRect.GetWidth(); 478 if ( nParaWidth ) 479 { 480 double fScale = (double)aTextAreaIter->aBoundRect.GetWidth() / nParaWidth; 481 482 std::vector< FWCharacterData >::iterator aCharacterIter( aParagraphIter->vCharacters.begin() ); 483 std::vector< FWCharacterData >::iterator aCharacterIEnd( aParagraphIter->vCharacters.end() ); 484 while ( aCharacterIter != aCharacterIEnd ) 485 { 486 std::vector< PolyPolygon >::iterator aOutlineIter = aCharacterIter->vOutlines.begin(); 487 std::vector< PolyPolygon >::iterator aOutlineIEnd = aCharacterIter->vOutlines.end(); 488 while( aOutlineIter != aOutlineIEnd ) 489 { 490 aOutlineIter->Scale( fScale, 1.0 ); 491 aOutlineIter++; 492 } 493 aCharacterIter++; 494 } 495 } 496 aParagraphIter++; 497 } 498 } 499 else 500 { 501 switch( eHorzAdjust ) 502 { 503 case SDRTEXTHORZADJUST_RIGHT : 504 case SDRTEXTHORZADJUST_CENTER: 505 { 506 std::vector< FWParagraphData >::iterator aParagraphIter( aTextAreaIter->vParagraphs.begin() ); 507 std::vector< FWParagraphData >::iterator aParagraphIEnd( aTextAreaIter->vParagraphs.end() ); 508 while ( aParagraphIter != aParagraphIEnd ) 509 { 510 sal_Int32 nHorzDiff = 0; 511 if ( eHorzAdjust == SDRTEXTHORZADJUST_CENTER ) 512 nHorzDiff = ( aTextAreaIter->aBoundRect.GetWidth() - aParagraphIter->aBoundRect.GetWidth() ) / 2; 513 else if ( eHorzAdjust == SDRTEXTHORZADJUST_RIGHT ) 514 nHorzDiff = ( aTextAreaIter->aBoundRect.GetWidth() - aParagraphIter->aBoundRect.GetWidth() ); 515 if ( nHorzDiff ) 516 { 517 std::vector< FWCharacterData >::iterator aCharacterIter( aParagraphIter->vCharacters.begin() ); 518 std::vector< FWCharacterData >::iterator aCharacterIEnd( aParagraphIter->vCharacters.end() ); 519 while ( aCharacterIter != aCharacterIEnd ) 520 { 521 std::vector< PolyPolygon >::iterator aOutlineIter = aCharacterIter->vOutlines.begin(); 522 std::vector< PolyPolygon >::iterator aOutlineIEnd = aCharacterIter->vOutlines.end(); 523 while( aOutlineIter != aOutlineIEnd ) 524 { 525 aOutlineIter->Move( nHorzDiff, 0 ); 526 aOutlineIter++; 527 } 528 aCharacterIter++; 529 } 530 } 531 aParagraphIter++; 532 } 533 } 534 break; 535 default: 536 case SDRTEXTHORZADJUST_BLOCK : break; // don't know 537 case SDRTEXTHORZADJUST_LEFT : break; // already left aligned -> nothing to do 538 } 539 } 540 aTextAreaIter++; 541 } 542 } 543 544 basegfx::B2DPolyPolygon GetOutlinesFromShape2d( const SdrObject* pShape2d ) 545 { 546 basegfx::B2DPolyPolygon aOutlines2d; 547 548 SdrObjListIter aObjListIter( *pShape2d, IM_DEEPWITHGROUPS ); 549 while( aObjListIter.IsMore() ) 550 { 551 SdrObject* pPartObj = aObjListIter.Next(); 552 if ( pPartObj->ISA( SdrPathObj ) ) 553 { 554 basegfx::B2DPolyPolygon aCandidate(((SdrPathObj*)pPartObj)->GetPathPoly()); 555 if(aCandidate.areControlPointsUsed()) 556 { 557 aCandidate = basegfx::tools::adaptiveSubdivideByAngle(aCandidate); 558 } 559 aOutlines2d.append(aCandidate); 560 } 561 } 562 563 return aOutlines2d; 564 } 565 566 void CalcDistances( const Polygon& rPoly, std::vector< double >& rDistances ) 567 { 568 sal_uInt16 i, nCount = rPoly.GetSize(); 569 if ( nCount > 1 ) 570 { 571 for ( i = 0; i < nCount; i++ ) 572 { 573 double fDistance = i ? ((Polygon&)rPoly).CalcDistance( i, i - 1 ) : 0.0; 574 rDistances.push_back( fDistance ); 575 } 576 std::partial_sum( rDistances.begin(), rDistances.end(), rDistances.begin() ); 577 double fLength = rDistances[ rDistances.size() - 1 ]; 578 if ( fLength > 0.0 ) 579 { 580 std::vector< double >::iterator aIter = rDistances.begin(); 581 std::vector< double >::iterator aEnd = rDistances.end(); 582 while ( aIter != aEnd ) 583 *aIter++ /= fLength; 584 } 585 } 586 } 587 588 void InsertMissingOutlinePoints( const Polygon& /*rOutlinePoly*/, const std::vector< double >& rDistances, const Rectangle& rTextAreaBoundRect, Polygon& rPoly ) 589 { 590 sal_uInt16 i = 0; 591 double fLastDistance = 0.0; 592 for ( i = 0; i < rPoly.GetSize(); i++ ) 593 { 594 Point& rPoint = rPoly[ i ]; 595 double fDistance = (double)( rPoint.X() - rTextAreaBoundRect.Left() ) / (double)rTextAreaBoundRect.GetWidth(); 596 if ( i ) 597 { 598 if ( fDistance > fLastDistance ) 599 { 600 std::vector< double >::const_iterator aIter = std::upper_bound( rDistances.begin(), rDistances.end(), fLastDistance ); 601 if ( aIter != rDistances.end() && ( *aIter > fLastDistance ) && ( *aIter < fDistance ) ) 602 { 603 Point& rPt0 = rPoly[ i - 1 ]; 604 sal_Int32 fX = rPoint.X() - rPt0.X(); 605 sal_Int32 fY = rPoint.Y() - rPt0.Y(); 606 double fd = ( 1.0 / ( fDistance - fLastDistance ) ) * ( *aIter - fLastDistance ); 607 rPoly.Insert( i, Point( (sal_Int32)( rPt0.X() + fX * fd ), (sal_Int32)( rPt0.Y() + fY * fd ) ) ); 608 fDistance = *aIter; 609 } 610 } 611 else if ( fDistance < fLastDistance ) 612 { 613 std::vector< double >::const_iterator aIter = std::lower_bound( rDistances.begin(), rDistances.end(), fLastDistance ); 614 if ( aIter-- != rDistances.begin() ) 615 { 616 if ( ( *aIter > fDistance ) && ( *aIter < fLastDistance ) ) 617 { 618 Point& rPt0 = rPoly[ i - 1 ]; 619 sal_Int32 fX = rPoint.X() - rPt0.X(); 620 sal_Int32 fY = rPoint.Y() - rPt0.Y(); 621 double fd = ( 1.0 / ( fDistance - fLastDistance ) ) * ( *aIter - fLastDistance ); 622 rPoly.Insert( i, Point( (sal_Int32)( rPt0.X() + fX * fd ), (sal_Int32)( rPt0.Y() + fY * fd ) ) ); 623 fDistance = *aIter; 624 } 625 } 626 } 627 } 628 fLastDistance = fDistance; 629 } 630 } 631 632 void GetPoint( const Polygon& rPoly, const std::vector< double >& rDistances, const double& fX, double& fx1, double& fy1 ) 633 { 634 fy1 = fx1 = 0.0; 635 if ( rPoly.GetSize() ) 636 { 637 std::vector< double >::const_iterator aIter = std::lower_bound( rDistances.begin(), rDistances.end(), fX ); 638 sal_uInt16 nIdx = sal::static_int_cast<sal_uInt16>( std::distance( rDistances.begin(), aIter ) ); 639 if ( aIter == rDistances.end() ) 640 nIdx--; 641 const Point& rPt = rPoly[ nIdx ]; 642 fx1 = rPt.X(); 643 fy1 = rPt.Y(); 644 if ( nIdx && ( aIter != rDistances.end() ) && ( *aIter != fX ) ) 645 { 646 nIdx = sal::static_int_cast<sal_uInt16>( std::distance( rDistances.begin(), aIter ) ); 647 double fDist0 = *( aIter - 1 ); 648 double fd = ( 1.0 / ( *aIter - fDist0 ) ) * ( fX - fDist0 ); 649 const Point& rPt2 = rPoly[ nIdx - 1 ]; 650 double fWidth = rPt.X() - rPt2.X(); 651 double fHeight= rPt.Y() - rPt2.Y(); 652 fWidth *= fd; 653 fHeight*= fd; 654 fx1 = rPt2.X() + fWidth; 655 fy1 = rPt2.Y() + fHeight; 656 } 657 } 658 } 659 660 void FitTextOutlinesToShapeOutlines( const PolyPolygon& aOutlines2d, FWData& rFWData ) 661 { 662 std::vector< FWTextArea >::iterator aTextAreaIter = rFWData.vTextAreas.begin(); 663 std::vector< FWTextArea >::iterator aTextAreaIEnd = rFWData.vTextAreas.end(); 664 665 sal_uInt16 nOutline2dIdx = 0; 666 while( aTextAreaIter != aTextAreaIEnd ) 667 { 668 Rectangle rTextAreaBoundRect = aTextAreaIter->aBoundRect; 669 sal_Int32 nLeft = rTextAreaBoundRect.Left(); 670 sal_Int32 nTop = rTextAreaBoundRect.Top(); 671 sal_Int32 nWidth = rTextAreaBoundRect.GetWidth(); 672 sal_Int32 nHeight= rTextAreaBoundRect.GetHeight(); 673 if ( rFWData.bSingleLineMode && nHeight && nWidth ) 674 { 675 if ( nOutline2dIdx >= aOutlines2d.Count() ) 676 break; 677 const Polygon& rOutlinePoly( aOutlines2d[ nOutline2dIdx++ ] ); 678 const sal_uInt16 nPointCount = rOutlinePoly.GetSize(); 679 if ( nPointCount > 1 ) 680 { 681 std::vector< double > vDistances; 682 vDistances.reserve( nPointCount ); 683 CalcDistances( rOutlinePoly, vDistances ); 684 if ( !vDistances.empty() ) 685 { 686 std::vector< FWParagraphData >::iterator aParagraphIter( aTextAreaIter->vParagraphs.begin() ); 687 std::vector< FWParagraphData >::iterator aParagraphIEnd( aTextAreaIter->vParagraphs.end() ); 688 while( aParagraphIter != aParagraphIEnd ) 689 { 690 std::vector< FWCharacterData >::iterator aCharacterIter( aParagraphIter->vCharacters.begin() ); 691 std::vector< FWCharacterData >::iterator aCharacterIEnd( aParagraphIter->vCharacters.end() ); 692 while ( aCharacterIter != aCharacterIEnd ) 693 { 694 std::vector< PolyPolygon >::iterator aOutlineIter = aCharacterIter->vOutlines.begin(); 695 std::vector< PolyPolygon >::iterator aOutlineIEnd = aCharacterIter->vOutlines.end(); 696 while( aOutlineIter != aOutlineIEnd ) 697 { 698 PolyPolygon& rPolyPoly = *aOutlineIter; 699 Rectangle aBoundRect( rPolyPoly.GetBoundRect() ); 700 double fx1 = aBoundRect.Left() - nLeft; 701 double fx2 = aBoundRect.Right() - nLeft; 702 double fy1, fy2; 703 double fM1 = fx1 / (double)nWidth; 704 double fM2 = fx2 / (double)nWidth; 705 706 GetPoint( rOutlinePoly, vDistances, fM1, fx1, fy1 ); 707 GetPoint( rOutlinePoly, vDistances, fM2, fx2, fy2 ); 708 709 double fvx = ( fy2 - fy1 ); 710 double fvy = - ( fx2 - fx1 ); 711 fx1 = fx1 + ( ( fx2 - fx1 ) * 0.5 ); 712 fy1 = fy1 + ( ( fy2 - fy1 ) * 0.5 ); 713 714 double fAngle = atan2( -fvx, -fvy ); 715 double fL = hypot( fvx, fvy ); 716 fvx = fvx / fL; 717 fvy = fvy / fL; 718 fL = (double)( aTextAreaIter->aBoundRect.GetHeight() / 2.0 + aTextAreaIter->aBoundRect.Top() ) - aParagraphIter->aBoundRect.Center().Y(); 719 fvx *= fL; 720 fvy *= fL; 721 rPolyPoly.Rotate( Point( aBoundRect.Center().X(), aParagraphIter->aBoundRect.Center().Y() ), sin( fAngle ), cos( fAngle ) ); 722 rPolyPoly.Move( (sal_Int32)( ( fx1 + fvx )- aBoundRect.Center().X() ), (sal_Int32)( ( fy1 + fvy ) - aParagraphIter->aBoundRect.Center().Y() ) ); 723 724 aOutlineIter++; 725 } 726 aCharacterIter++; 727 } 728 aParagraphIter++; 729 } 730 } 731 } 732 } 733 else 734 { 735 if ( ( nOutline2dIdx + 1 ) >= aOutlines2d.Count() ) 736 break; 737 const Polygon& rOutlinePoly( aOutlines2d[ nOutline2dIdx++ ] ); 738 const Polygon& rOutlinePoly2( aOutlines2d[ nOutline2dIdx++ ] ); 739 const sal_uInt16 nPointCount = rOutlinePoly.GetSize(); 740 const sal_uInt16 nPointCount2 = rOutlinePoly2.GetSize(); 741 if ( ( nPointCount > 1 ) && ( nPointCount2 > 1 ) ) 742 { 743 std::vector< double > vDistances; 744 vDistances.reserve( nPointCount ); 745 std::vector< double > vDistances2; 746 vDistances2.reserve( nPointCount2 ); 747 CalcDistances( rOutlinePoly, vDistances ); 748 CalcDistances( rOutlinePoly2, vDistances2 ); 749 std::vector< FWParagraphData >::iterator aParagraphIter = aTextAreaIter->vParagraphs.begin(); 750 std::vector< FWParagraphData >::iterator aParagraphIEnd = aTextAreaIter->vParagraphs.end(); 751 while( aParagraphIter != aParagraphIEnd ) 752 { 753 std::vector< FWCharacterData >::iterator aCharacterIter( aParagraphIter->vCharacters.begin() ); 754 std::vector< FWCharacterData >::iterator aCharacterIEnd( aParagraphIter->vCharacters.end() ); 755 while ( aCharacterIter != aCharacterIEnd ) 756 { 757 std::vector< PolyPolygon >::iterator aOutlineIter = aCharacterIter->vOutlines.begin(); 758 std::vector< PolyPolygon >::iterator aOutlineIEnd = aCharacterIter->vOutlines.end(); 759 while( aOutlineIter != aOutlineIEnd ) 760 { 761 PolyPolygon& rPolyPoly = *aOutlineIter; 762 sal_uInt16 i, nPolyCount = rPolyPoly.Count(); 763 for ( i = 0; i < nPolyCount; i++ ) 764 { 765 // #i35928# 766 basegfx::B2DPolygon aCandidate(rPolyPoly[ i ].getB2DPolygon()); 767 768 if(aCandidate.areControlPointsUsed()) 769 { 770 aCandidate = basegfx::tools::adaptiveSubdivideByAngle(aCandidate); 771 } 772 773 // create local polygon copy to work on 774 Polygon aLocalPoly(aCandidate); 775 776 InsertMissingOutlinePoints( rOutlinePoly, vDistances, rTextAreaBoundRect, aLocalPoly ); 777 InsertMissingOutlinePoints( rOutlinePoly2, vDistances2, rTextAreaBoundRect, aLocalPoly ); 778 779 sal_uInt16 j, _nPointCount = aLocalPoly.GetSize(); 780 for ( j = 0; j < _nPointCount; j++ ) 781 { 782 Point& rPoint = aLocalPoly[ j ]; 783 rPoint.X() -= nLeft; 784 rPoint.Y() -= nTop; 785 double fX = (double)rPoint.X() / (double)nWidth; 786 double fY = (double)rPoint.Y() / (double)nHeight; 787 788 double fx1, fy1, fx2, fy2; 789 GetPoint( rOutlinePoly, vDistances, fX, fx1, fy1 ); 790 GetPoint( rOutlinePoly2, vDistances2, fX, fx2, fy2 ); 791 double fWidth = fx2 - fx1; 792 double fHeight= fy2 - fy1; 793 rPoint.X() = (sal_Int32)( fx1 + fWidth * fY ); 794 rPoint.Y() = (sal_Int32)( fy1 + fHeight* fY ); 795 } 796 797 // write back polygon 798 rPolyPoly[ i ] = aLocalPoly; 799 } 800 aOutlineIter++; 801 } 802 aCharacterIter++; 803 } 804 aParagraphIter++; 805 } 806 } 807 } 808 aTextAreaIter++; 809 } 810 } 811 812 SdrObject* CreateSdrObjectFromParagraphOutlines( const FWData& rFWData, const SdrObject* pCustomShape ) 813 { 814 SdrObject* pRet = NULL; 815 if ( !rFWData.vTextAreas.empty() ) 816 { 817 pRet = new SdrObjGroup(); 818 // SJ: not setting model, so we save a lot of broadcasting and the model is not modified any longer 819 // pRet->SetModel( pCustomShape->GetModel() ); 820 std::vector< FWTextArea >::const_iterator aTextAreaIter = rFWData.vTextAreas.begin(); 821 std::vector< FWTextArea >::const_iterator aTextAreaIEnd = rFWData.vTextAreas.end(); 822 while ( aTextAreaIter != aTextAreaIEnd ) 823 { 824 std::vector< FWParagraphData >::const_iterator aParagraphIter = aTextAreaIter->vParagraphs.begin(); 825 std::vector< FWParagraphData >::const_iterator aParagraphIEnd = aTextAreaIter->vParagraphs.end(); 826 while ( aParagraphIter != aParagraphIEnd ) 827 { 828 std::vector< FWCharacterData >::const_iterator aCharacterIter( aParagraphIter->vCharacters.begin() ); 829 std::vector< FWCharacterData >::const_iterator aCharacterIEnd( aParagraphIter->vCharacters.end() ); 830 while ( aCharacterIter != aCharacterIEnd ) 831 { 832 std::vector< PolyPolygon >::const_iterator aOutlineIter = aCharacterIter->vOutlines.begin(); 833 std::vector< PolyPolygon >::const_iterator aOutlineIEnd = aCharacterIter->vOutlines.end(); 834 while( aOutlineIter != aOutlineIEnd ) 835 { 836 SdrObject* pPathObj = new SdrPathObj( OBJ_POLY, aOutlineIter->getB2DPolyPolygon() ); 837 // SJ: not setting model, so we save a lot of broadcasting and the model is not modified any longer 838 // pPathObj->SetModel( pCustomShape->GetModel() ); 839 ((SdrObjGroup*)pRet)->GetSubList()->NbcInsertObject( pPathObj ); 840 aOutlineIter++; 841 } 842 aCharacterIter++; 843 } 844 aParagraphIter++; 845 } 846 aTextAreaIter++; 847 } 848 849 Point aP( pCustomShape->GetSnapRect().Center() ); 850 Size aS( pCustomShape->GetLogicRect().GetSize() ); 851 aP.X() -= aS.Width() / 2; 852 aP.Y() -= aS.Height() / 2; 853 Rectangle aLogicRect( aP, aS ); 854 855 SfxItemSet aSet( pCustomShape->GetMergedItemSet() ); 856 aSet.ClearItem( SDRATTR_TEXTDIRECTION ); //SJ: vertical writing is not required, by removing this item no outliner is created 857 aSet.Put(SdrShadowItem(sal_False)); // #i37011# NO shadow for FontWork geometry 858 pRet->SetMergedItemSet( aSet ); // * otherwise we would crash, because the outliner tries to create a Paraobject, but there is no model 859 } 860 return pRet; 861 } 862 863 ::com::sun::star::uno::Reference < ::com::sun::star::i18n::XBreakIterator > EnhancedCustomShapeFontWork::mxBreakIterator = 0; 864 865 Reference < i18n::XBreakIterator > EnhancedCustomShapeFontWork::GetBreakIterator() 866 { 867 if ( !mxBreakIterator.is() ) 868 { 869 Reference< lang::XMultiServiceFactory > xMSF = ::comphelper::getProcessServiceFactory(); 870 Reference < XInterface > xI = xMSF->createInstance( rtl::OUString::createFromAscii( "com.sun.star.i18n.BreakIterator" ) ); 871 if ( xI.is() ) 872 { 873 Any x = xI->queryInterface( ::getCppuType((const Reference< i18n::XBreakIterator >*)0) ); 874 x >>= mxBreakIterator; 875 } 876 } 877 return mxBreakIterator; 878 } 879 880 SdrObject* EnhancedCustomShapeFontWork::CreateFontWork( const SdrObject* pShape2d, const SdrObject* pCustomShape ) 881 { 882 SdrObject* pRet = NULL; 883 884 Rectangle aLogicRect( pCustomShape->GetLogicRect() ); 885 PolyPolygon aOutlines2d( GetOutlinesFromShape2d( pShape2d ) ); 886 sal_uInt16 nOutlinesCount2d = aOutlines2d.Count(); 887 if ( nOutlinesCount2d ) 888 { 889 FWData aFWData; 890 if ( InitializeFontWorkData( pCustomShape, nOutlinesCount2d, aFWData ) ) 891 { 892 /* retrieves the horizontal scaling factor that has to be used 893 to fit each paragraph text into its corresponding 2d outline */ 894 CalculateHorizontalScalingFactor( pCustomShape, aFWData, aOutlines2d ); 895 896 /* retrieving the Outlines for the each Paragraph. */ 897 898 GetFontWorkOutline( aFWData, pCustomShape ); 899 900 FitTextOutlinesToShapeOutlines( aOutlines2d, aFWData ); 901 902 pRet = CreateSdrObjectFromParagraphOutlines( aFWData, pCustomShape ); 903 } 904 } 905 return pRet; 906 } 907