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_tools.hxx" 26 27 #define _SV_POLY2_CXX 28 29 #define POLY_CLIP_INT 0 30 #define POLY_CLIP_UNION 1 31 #define POLY_CLIP_DIFF 2 32 #define POLY_CLIP_XOR 3 33 34 #include <rtl/math.hxx> 35 #include <poly.h> 36 #include <tools/poly.hxx> 37 #include <tools/debug.hxx> 38 #include <tools/stream.hxx> 39 #include <tools/vcompat.hxx> 40 #include <basegfx/polygon/b2dpolypolygon.hxx> 41 #include <basegfx/polygon/b2dpolygon.hxx> 42 #include <basegfx/polygon/b2dpolypolygoncutter.hxx> 43 44 // --------------- 45 // - PolyPolygon - 46 // --------------- 47 48 DBG_NAME( PolyPolygon ) 49 50 // ----------------------------------------------------------------------- 51 52 ImplPolyPolygon::ImplPolyPolygon( sal_uInt16 nInitSize ) 53 { 54 mnRefCount = 1; 55 mnCount = nInitSize; 56 mnSize = nInitSize; 57 mnResize = 16; 58 mpPolyAry = new SVPPOLYGON[ nInitSize ]; 59 } 60 61 // ----------------------------------------------------------------------- 62 63 ImplPolyPolygon::ImplPolyPolygon( const ImplPolyPolygon& rImplPolyPoly ) 64 { 65 mnRefCount = 1; 66 mnCount = rImplPolyPoly.mnCount; 67 mnSize = rImplPolyPoly.mnSize; 68 mnResize = rImplPolyPoly.mnResize; 69 70 if ( rImplPolyPoly.mpPolyAry ) 71 { 72 mpPolyAry = new SVPPOLYGON[mnSize]; 73 for ( sal_uInt16 i = 0; i < mnCount; i++ ) 74 mpPolyAry[i] = new Polygon( *rImplPolyPoly.mpPolyAry[i] ); 75 } 76 else 77 mpPolyAry = NULL; 78 } 79 80 // ----------------------------------------------------------------------- 81 82 ImplPolyPolygon::~ImplPolyPolygon() 83 { 84 if ( mpPolyAry ) 85 { 86 for ( sal_uInt16 i = 0; i < mnCount; i++ ) 87 delete mpPolyAry[i]; 88 delete[] mpPolyAry; 89 } 90 } 91 92 // ======================================================================= 93 94 PolyPolygon::PolyPolygon( sal_uInt16 nInitSize, sal_uInt16 nResize ) 95 { 96 DBG_CTOR( PolyPolygon, NULL ); 97 98 if ( nInitSize > MAX_POLYGONS ) 99 nInitSize = MAX_POLYGONS; 100 else if ( !nInitSize ) 101 nInitSize = 1; 102 if ( nResize > MAX_POLYGONS ) 103 nResize = MAX_POLYGONS; 104 else if ( !nResize ) 105 nResize = 1; 106 mpImplPolyPolygon = new ImplPolyPolygon( nInitSize, nResize ); 107 } 108 109 // ----------------------------------------------------------------------- 110 111 PolyPolygon::PolyPolygon( const Polygon& rPoly ) 112 { 113 DBG_CTOR( PolyPolygon, NULL ); 114 115 if ( rPoly.GetSize() ) 116 { 117 mpImplPolyPolygon = new ImplPolyPolygon( 1 ); 118 mpImplPolyPolygon->mpPolyAry[0] = new Polygon( rPoly ); 119 } 120 else 121 mpImplPolyPolygon = new ImplPolyPolygon( 16, 16 ); 122 } 123 124 // ----------------------------------------------------------------------- 125 126 PolyPolygon::PolyPolygon( const PolyPolygon& rPolyPoly ) 127 { 128 DBG_CTOR( PolyPolygon, NULL ); 129 DBG_CHKOBJ( &rPolyPoly, PolyPolygon, NULL ); 130 DBG_ASSERT( rPolyPoly.mpImplPolyPolygon->mnRefCount < 0xFFFFFFFE, "PolyPolygon: RefCount overflow" ); 131 132 mpImplPolyPolygon = rPolyPoly.mpImplPolyPolygon; 133 mpImplPolyPolygon->mnRefCount++; 134 } 135 136 // ----------------------------------------------------------------------- 137 138 PolyPolygon::~PolyPolygon() 139 { 140 DBG_DTOR( PolyPolygon, NULL ); 141 142 if ( mpImplPolyPolygon->mnRefCount > 1 ) 143 mpImplPolyPolygon->mnRefCount--; 144 else 145 delete mpImplPolyPolygon; 146 } 147 148 // ----------------------------------------------------------------------- 149 150 void PolyPolygon::Insert( const Polygon& rPoly, sal_uInt16 nPos ) 151 { 152 DBG_CHKTHIS( PolyPolygon, NULL ); 153 154 if ( mpImplPolyPolygon->mnCount >= MAX_POLYGONS ) 155 return; 156 157 if ( mpImplPolyPolygon->mnRefCount > 1 ) 158 { 159 mpImplPolyPolygon->mnRefCount--; 160 mpImplPolyPolygon = new ImplPolyPolygon( *mpImplPolyPolygon ); 161 } 162 163 if ( nPos > mpImplPolyPolygon->mnCount ) 164 nPos = mpImplPolyPolygon->mnCount; 165 166 if ( !mpImplPolyPolygon->mpPolyAry ) 167 mpImplPolyPolygon->mpPolyAry = new SVPPOLYGON[mpImplPolyPolygon->mnSize]; 168 else if ( mpImplPolyPolygon->mnCount == mpImplPolyPolygon->mnSize ) 169 { 170 sal_uInt16 nOldSize = mpImplPolyPolygon->mnSize; 171 sal_uInt16 nNewSize = nOldSize + mpImplPolyPolygon->mnResize; 172 SVPPOLYGON* pNewAry; 173 174 if ( nNewSize >= MAX_POLYGONS ) 175 nNewSize = MAX_POLYGONS; 176 pNewAry = new SVPPOLYGON[nNewSize]; 177 memcpy( pNewAry, mpImplPolyPolygon->mpPolyAry, nPos*sizeof(SVPPOLYGON) ); 178 memcpy( pNewAry+nPos+1, mpImplPolyPolygon->mpPolyAry+nPos, 179 (nOldSize-nPos)*sizeof(SVPPOLYGON) ); 180 delete[] mpImplPolyPolygon->mpPolyAry; 181 mpImplPolyPolygon->mpPolyAry = pNewAry; 182 mpImplPolyPolygon->mnSize = nNewSize; 183 } 184 else if ( nPos < mpImplPolyPolygon->mnCount ) 185 { 186 memmove( mpImplPolyPolygon->mpPolyAry+nPos+1, 187 mpImplPolyPolygon->mpPolyAry+nPos, 188 (mpImplPolyPolygon->mnCount-nPos)*sizeof(SVPPOLYGON) ); 189 } 190 191 mpImplPolyPolygon->mpPolyAry[nPos] = new Polygon( rPoly ); 192 mpImplPolyPolygon->mnCount++; 193 } 194 195 // ----------------------------------------------------------------------- 196 197 void PolyPolygon::Remove( sal_uInt16 nPos ) 198 { 199 DBG_CHKTHIS( PolyPolygon, NULL ); 200 DBG_ASSERT( nPos < Count(), "PolyPolygon::Remove(): nPos >= nSize" ); 201 if ( nPos >= Count() ) return; // not removable 202 203 if ( mpImplPolyPolygon->mnRefCount > 1 ) 204 { 205 mpImplPolyPolygon->mnRefCount--; 206 mpImplPolyPolygon = new ImplPolyPolygon( *mpImplPolyPolygon ); 207 } 208 209 delete mpImplPolyPolygon->mpPolyAry[nPos]; 210 mpImplPolyPolygon->mnCount--; 211 memmove( mpImplPolyPolygon->mpPolyAry+nPos, 212 mpImplPolyPolygon->mpPolyAry+nPos+1, 213 (mpImplPolyPolygon->mnCount-nPos)*sizeof(SVPPOLYGON) ); 214 } 215 216 // ----------------------------------------------------------------------- 217 218 void PolyPolygon::Replace( const Polygon& rPoly, sal_uInt16 nPos ) 219 { 220 DBG_CHKTHIS( PolyPolygon, NULL ); 221 DBG_ASSERT( nPos < Count(), "PolyPolygon::Replace(): nPos >= nSize" ); 222 if ( nPos >= Count() ) return; // not replaceable 223 224 if ( mpImplPolyPolygon->mnRefCount > 1 ) 225 { 226 mpImplPolyPolygon->mnRefCount--; 227 mpImplPolyPolygon = new ImplPolyPolygon( *mpImplPolyPolygon ); 228 } 229 230 delete mpImplPolyPolygon->mpPolyAry[nPos]; 231 mpImplPolyPolygon->mpPolyAry[nPos] = new Polygon( rPoly ); 232 } 233 234 // ----------------------------------------------------------------------- 235 236 const Polygon& PolyPolygon::GetObject( sal_uInt16 nPos ) const 237 { 238 DBG_CHKTHIS( PolyPolygon, NULL ); 239 DBG_ASSERT( nPos < Count(), "PolyPolygon::GetObject(): nPos >= nSize" ); 240 241 return *(mpImplPolyPolygon->mpPolyAry[nPos]); 242 } 243 244 // ----------------------------------------------------------------------- 245 246 sal_Bool PolyPolygon::IsRect() const 247 { 248 sal_Bool bIsRect = sal_False; 249 if ( Count() == 1 ) 250 bIsRect = mpImplPolyPolygon->mpPolyAry[ 0 ]->IsRect(); 251 return bIsRect; 252 } 253 254 // ----------------------------------------------------------------------- 255 256 void PolyPolygon::Clear() 257 { 258 DBG_CHKTHIS( PolyPolygon, NULL ); 259 260 if ( mpImplPolyPolygon->mnRefCount > 1 ) 261 { 262 mpImplPolyPolygon->mnRefCount--; 263 mpImplPolyPolygon = new ImplPolyPolygon( mpImplPolyPolygon->mnResize, 264 mpImplPolyPolygon->mnResize ); 265 } 266 else 267 { 268 if ( mpImplPolyPolygon->mpPolyAry ) 269 { 270 for ( sal_uInt16 i = 0; i < mpImplPolyPolygon->mnCount; i++ ) 271 delete mpImplPolyPolygon->mpPolyAry[i]; 272 delete[] mpImplPolyPolygon->mpPolyAry; 273 mpImplPolyPolygon->mpPolyAry = NULL; 274 mpImplPolyPolygon->mnCount = 0; 275 mpImplPolyPolygon->mnSize = mpImplPolyPolygon->mnResize; 276 } 277 } 278 } 279 280 // ----------------------------------------------------------------------- 281 282 void PolyPolygon::Optimize( sal_uIntPtr nOptimizeFlags, const PolyOptimizeData* pData ) 283 { 284 DBG_CHKTHIS( PolyPolygon, NULL ); 285 286 if(nOptimizeFlags && Count()) 287 { 288 // #115630# ImplDrawHatch does not work with beziers included in the polypolygon, take care of that 289 bool bIsCurve(false); 290 291 for(sal_uInt16 a(0); !bIsCurve && a < Count(); a++) 292 { 293 if((*this)[a].HasFlags()) 294 { 295 bIsCurve = true; 296 } 297 } 298 299 if(bIsCurve) 300 { 301 OSL_ENSURE(false, "Optimize does *not* support curves, falling back to AdaptiveSubdivide()..."); 302 PolyPolygon aPolyPoly; 303 304 AdaptiveSubdivide(aPolyPoly); 305 aPolyPoly.Optimize(nOptimizeFlags, pData); 306 *this = aPolyPoly; 307 } 308 else 309 { 310 double fArea; 311 const sal_Bool bEdges = ( nOptimizeFlags & POLY_OPTIMIZE_EDGES ) == POLY_OPTIMIZE_EDGES; 312 sal_uInt16 nPercent = 0; 313 314 if( bEdges ) 315 { 316 const Rectangle aBound( GetBoundRect() ); 317 318 fArea = ( aBound.GetWidth() + aBound.GetHeight() ) * 0.5; 319 nPercent = pData ? pData->GetPercentValue() : 50; 320 nOptimizeFlags &= ~POLY_OPTIMIZE_EDGES; 321 } 322 323 // watch for ref counter 324 if( mpImplPolyPolygon->mnRefCount > 1 ) 325 { 326 mpImplPolyPolygon->mnRefCount--; 327 mpImplPolyPolygon = new ImplPolyPolygon( *mpImplPolyPolygon ); 328 } 329 330 // Optimize polygons 331 for( sal_uInt16 i = 0, nPolyCount = mpImplPolyPolygon->mnCount; i < nPolyCount; i++ ) 332 { 333 if( bEdges ) 334 { 335 mpImplPolyPolygon->mpPolyAry[ i ]->Optimize( POLY_OPTIMIZE_NO_SAME ); 336 Polygon::ImplReduceEdges( *( mpImplPolyPolygon->mpPolyAry[ i ] ), fArea, nPercent ); 337 } 338 339 if( nOptimizeFlags ) 340 mpImplPolyPolygon->mpPolyAry[ i ]->Optimize( nOptimizeFlags, pData ); 341 } 342 } 343 } 344 } 345 346 // ----------------------------------------------------------------------- 347 348 void PolyPolygon::AdaptiveSubdivide( PolyPolygon& rResult, const double d ) const 349 { 350 DBG_CHKTHIS( PolyPolygon, NULL ); 351 352 rResult.Clear(); 353 354 Polygon aPolygon; 355 356 for( sal_uInt16 i = 0; i < mpImplPolyPolygon->mnCount; i++ ) 357 { 358 mpImplPolyPolygon->mpPolyAry[ i ]->AdaptiveSubdivide( aPolygon, d ); 359 rResult.Insert( aPolygon ); 360 } 361 } 362 363 // ----------------------------------------------------------------------- 364 365 void PolyPolygon::GetIntersection( const PolyPolygon& rPolyPoly, PolyPolygon& rResult ) const 366 { 367 ImplDoOperation( rPolyPoly, rResult, POLY_CLIP_INT ); 368 } 369 370 // ----------------------------------------------------------------------- 371 372 void PolyPolygon::GetUnion( const PolyPolygon& rPolyPoly, PolyPolygon& rResult ) const 373 { 374 ImplDoOperation( rPolyPoly, rResult, POLY_CLIP_UNION ); 375 } 376 377 // ----------------------------------------------------------------------- 378 379 void PolyPolygon::GetDifference( const PolyPolygon& rPolyPoly, PolyPolygon& rResult ) const 380 { 381 ImplDoOperation( rPolyPoly, rResult, POLY_CLIP_DIFF ); 382 } 383 384 // ----------------------------------------------------------------------- 385 386 void PolyPolygon::GetXOR( const PolyPolygon& rPolyPoly, PolyPolygon& rResult ) const 387 { 388 ImplDoOperation( rPolyPoly, rResult, POLY_CLIP_XOR ); 389 } 390 391 // ----------------------------------------------------------------------- 392 393 void PolyPolygon::ImplDoOperation( const PolyPolygon& rPolyPoly, PolyPolygon& rResult, sal_uIntPtr nOperation ) const 394 { 395 // Convert to B2DPolyPolygon, temporarily. It might be 396 // advantageous in the future, to have a PolyPolygon adaptor that 397 // just simulates a B2DPolyPolygon here... 398 basegfx::B2DPolyPolygon aMergePolyPolygonA( getB2DPolyPolygon() ); 399 basegfx::B2DPolyPolygon aMergePolyPolygonB( rPolyPoly.getB2DPolyPolygon() ); 400 401 // normalize the two polypolygons before. Force properly oriented 402 // polygons. 403 aMergePolyPolygonA = basegfx::tools::prepareForPolygonOperation( aMergePolyPolygonA ); 404 aMergePolyPolygonB = basegfx::tools::prepareForPolygonOperation( aMergePolyPolygonB ); 405 406 switch( nOperation ) 407 { 408 // All code extracted from svx/source/svdraw/svedtv2.cxx 409 // ----------------------------------------------------- 410 411 case POLY_CLIP_UNION: 412 { 413 // merge A and B (OR) 414 aMergePolyPolygonA = basegfx::tools::solvePolygonOperationOr(aMergePolyPolygonA, aMergePolyPolygonB); 415 break; 416 } 417 418 case POLY_CLIP_DIFF: 419 { 420 // subtract B from A (DIFF) 421 aMergePolyPolygonA = basegfx::tools::solvePolygonOperationDiff(aMergePolyPolygonA, aMergePolyPolygonB); 422 break; 423 } 424 425 case POLY_CLIP_XOR: 426 { 427 // compute XOR between poly A and B 428 aMergePolyPolygonA = basegfx::tools::solvePolygonOperationXor(aMergePolyPolygonA, aMergePolyPolygonB); 429 break; 430 } 431 432 default: 433 case POLY_CLIP_INT: 434 { 435 // cut poly 1 against polys 2..n (AND) 436 aMergePolyPolygonA = basegfx::tools::solvePolygonOperationAnd(aMergePolyPolygonA, aMergePolyPolygonB); 437 break; 438 } 439 } 440 441 rResult = PolyPolygon( aMergePolyPolygonA ); 442 } 443 444 // ----------------------------------------------------------------------- 445 446 sal_uInt16 PolyPolygon::Count() const 447 { 448 DBG_CHKTHIS( PolyPolygon, NULL ); 449 return mpImplPolyPolygon->mnCount; 450 } 451 452 // ----------------------------------------------------------------------- 453 454 void PolyPolygon::Move( long nHorzMove, long nVertMove ) 455 { 456 DBG_CHKTHIS( PolyPolygon, NULL ); 457 458 // Diese Abfrage sollte man fuer die DrawEngine durchfuehren 459 if( nHorzMove || nVertMove ) 460 { 461 // Referenzcounter beruecksichtigen 462 if ( mpImplPolyPolygon->mnRefCount > 1 ) 463 { 464 mpImplPolyPolygon->mnRefCount--; 465 mpImplPolyPolygon = new ImplPolyPolygon( *mpImplPolyPolygon ); 466 } 467 468 // Punkte verschieben 469 sal_uInt16 nPolyCount = mpImplPolyPolygon->mnCount; 470 for ( sal_uInt16 i = 0; i < nPolyCount; i++ ) 471 mpImplPolyPolygon->mpPolyAry[i]->Move( nHorzMove, nVertMove ); 472 } 473 } 474 475 // ----------------------------------------------------------------------- 476 477 void PolyPolygon::Translate( const Point& rTrans ) 478 { 479 DBG_CHKTHIS( PolyPolygon, NULL ); 480 481 // Referenzcounter beruecksichtigen 482 if( mpImplPolyPolygon->mnRefCount > 1 ) 483 { 484 mpImplPolyPolygon->mnRefCount--; 485 mpImplPolyPolygon = new ImplPolyPolygon( *mpImplPolyPolygon ); 486 } 487 488 // Punkte verschieben 489 for ( sal_uInt16 i = 0, nCount = mpImplPolyPolygon->mnCount; i < nCount; i++ ) 490 mpImplPolyPolygon->mpPolyAry[ i ]->Translate( rTrans ); 491 } 492 493 // ----------------------------------------------------------------------- 494 495 void PolyPolygon::Scale( double fScaleX, double fScaleY ) 496 { 497 DBG_CHKTHIS( PolyPolygon, NULL ); 498 499 // Referenzcounter beruecksichtigen 500 if( mpImplPolyPolygon->mnRefCount > 1 ) 501 { 502 mpImplPolyPolygon->mnRefCount--; 503 mpImplPolyPolygon = new ImplPolyPolygon( *mpImplPolyPolygon ); 504 } 505 506 // Punkte verschieben 507 for ( sal_uInt16 i = 0, nCount = mpImplPolyPolygon->mnCount; i < nCount; i++ ) 508 mpImplPolyPolygon->mpPolyAry[ i ]->Scale( fScaleX, fScaleY ); 509 } 510 511 // ----------------------------------------------------------------------- 512 513 void PolyPolygon::Rotate( const Point& rCenter, sal_uInt16 nAngle10 ) 514 { 515 DBG_CHKTHIS( PolyPolygon, NULL ); 516 nAngle10 %= 3600; 517 518 if( nAngle10 ) 519 { 520 const double fAngle = F_PI1800 * nAngle10; 521 Rotate( rCenter, sin( fAngle ), cos( fAngle ) ); 522 } 523 } 524 525 // ----------------------------------------------------------------------- 526 527 void PolyPolygon::Rotate( const Point& rCenter, double fSin, double fCos ) 528 { 529 DBG_CHKTHIS( PolyPolygon, NULL ); 530 531 // Referenzcounter beruecksichtigen 532 if( mpImplPolyPolygon->mnRefCount > 1 ) 533 { 534 mpImplPolyPolygon->mnRefCount--; 535 mpImplPolyPolygon = new ImplPolyPolygon( *mpImplPolyPolygon ); 536 } 537 538 // Punkte verschieben 539 for ( sal_uInt16 i = 0, nCount = mpImplPolyPolygon->mnCount; i < nCount; i++ ) 540 mpImplPolyPolygon->mpPolyAry[ i ]->Rotate( rCenter, fSin, fCos ); 541 } 542 543 // ----------------------------------------------------------------------- 544 545 void PolyPolygon::SlantX( long nYRef, double fSin, double fCos ) 546 { 547 DBG_CHKTHIS( PolyPolygon, NULL ); 548 549 // Referenzcounter beruecksichtigen 550 if( mpImplPolyPolygon->mnRefCount > 1 ) 551 { 552 mpImplPolyPolygon->mnRefCount--; 553 mpImplPolyPolygon = new ImplPolyPolygon( *mpImplPolyPolygon ); 554 } 555 556 // Punkte verschieben 557 for ( sal_uInt16 i = 0, nCount = mpImplPolyPolygon->mnCount; i < nCount; i++ ) 558 mpImplPolyPolygon->mpPolyAry[ i ]->SlantX( nYRef, fSin, fCos ); 559 } 560 561 // ----------------------------------------------------------------------- 562 563 void PolyPolygon::SlantY( long nXRef, double fSin, double fCos ) 564 { 565 DBG_CHKTHIS( PolyPolygon, NULL ); 566 567 // Referenzcounter beruecksichtigen 568 if( mpImplPolyPolygon->mnRefCount > 1 ) 569 { 570 mpImplPolyPolygon->mnRefCount--; 571 mpImplPolyPolygon = new ImplPolyPolygon( *mpImplPolyPolygon ); 572 } 573 574 // Punkte verschieben 575 for ( sal_uInt16 i = 0, nCount = mpImplPolyPolygon->mnCount; i < nCount; i++ ) 576 mpImplPolyPolygon->mpPolyAry[ i ]->SlantY( nXRef, fSin, fCos ); 577 } 578 579 // ----------------------------------------------------------------------- 580 581 void PolyPolygon::Distort( const Rectangle& rRefRect, const Polygon& rDistortedRect ) 582 { 583 DBG_CHKTHIS( PolyPolygon, NULL ); 584 585 // Referenzcounter beruecksichtigen 586 if( mpImplPolyPolygon->mnRefCount > 1 ) 587 { 588 mpImplPolyPolygon->mnRefCount--; 589 mpImplPolyPolygon = new ImplPolyPolygon( *mpImplPolyPolygon ); 590 } 591 592 // Punkte verschieben 593 for ( sal_uInt16 i = 0, nCount = mpImplPolyPolygon->mnCount; i < nCount; i++ ) 594 mpImplPolyPolygon->mpPolyAry[ i ]->Distort( rRefRect, rDistortedRect ); 595 } 596 597 598 // ----------------------------------------------------------------------- 599 600 void PolyPolygon::Clip( const Rectangle& rRect ) 601 { 602 // Polygon-Clippen 603 sal_uInt16 nPolyCount = mpImplPolyPolygon->mnCount; 604 sal_uInt16 i; 605 606 if ( !nPolyCount ) 607 return; 608 609 // Referenzcounter beruecksichtigen 610 if ( mpImplPolyPolygon->mnRefCount > 1 ) 611 { 612 mpImplPolyPolygon->mnRefCount--; 613 mpImplPolyPolygon = new ImplPolyPolygon( *mpImplPolyPolygon ); 614 } 615 616 // Erst jedes Polygon Clippen und dann die leeren entfernen 617 for ( i = 0; i < nPolyCount; i++ ) 618 mpImplPolyPolygon->mpPolyAry[i]->Clip( rRect ); 619 while ( nPolyCount ) 620 { 621 if ( GetObject( nPolyCount-1 ).GetSize() <= 2 ) 622 Remove( nPolyCount-1 ); 623 nPolyCount--; 624 } 625 } 626 627 // ----------------------------------------------------------------------- 628 629 Rectangle PolyPolygon::GetBoundRect() const 630 { 631 DBG_CHKTHIS( PolyPolygon, NULL ); 632 633 long nXMin=0, nXMax=0, nYMin=0, nYMax=0; 634 sal_Bool bFirst = sal_True; 635 sal_uInt16 nPolyCount = mpImplPolyPolygon->mnCount; 636 637 for ( sal_uInt16 n = 0; n < nPolyCount; n++ ) 638 { 639 const Polygon* pPoly = mpImplPolyPolygon->mpPolyAry[n]; 640 const Point* pAry = pPoly->GetConstPointAry(); 641 sal_uInt16 nPointCount = pPoly->GetSize(); 642 643 for ( sal_uInt16 i = 0; i < nPointCount; i++ ) 644 { 645 const Point* pPt = &pAry[ i ]; 646 647 if ( bFirst ) 648 { 649 nXMin = nXMax = pPt->X(); 650 nYMin = nYMax = pPt->Y(); 651 bFirst = sal_False; 652 } 653 else 654 { 655 if ( pPt->X() < nXMin ) 656 nXMin = pPt->X(); 657 if ( pPt->X() > nXMax ) 658 nXMax = pPt->X(); 659 if ( pPt->Y() < nYMin ) 660 nYMin = pPt->Y(); 661 if ( pPt->Y() > nYMax ) 662 nYMax = pPt->Y(); 663 } 664 } 665 } 666 667 if ( !bFirst ) 668 return Rectangle( nXMin, nYMin, nXMax, nYMax ); 669 else 670 return Rectangle(); 671 } 672 673 // ----------------------------------------------------------------------- 674 675 Polygon& PolyPolygon::operator[]( sal_uInt16 nPos ) 676 { 677 DBG_CHKTHIS( PolyPolygon, NULL ); 678 DBG_ASSERT( nPos < Count(), "PolyPolygon::[](): nPos >= nSize" ); 679 680 if ( mpImplPolyPolygon->mnRefCount > 1 ) 681 { 682 mpImplPolyPolygon->mnRefCount--; 683 mpImplPolyPolygon = new ImplPolyPolygon( *mpImplPolyPolygon ); 684 } 685 686 return *(mpImplPolyPolygon->mpPolyAry[nPos]); 687 } 688 689 // ----------------------------------------------------------------------- 690 691 PolyPolygon& PolyPolygon::operator=( const PolyPolygon& rPolyPoly ) 692 { 693 DBG_CHKTHIS( PolyPolygon, NULL ); 694 DBG_CHKOBJ( &rPolyPoly, PolyPolygon, NULL ); 695 DBG_ASSERT( rPolyPoly.mpImplPolyPolygon->mnRefCount < 0xFFFFFFFE, "PolyPolygon: RefCount overflow" ); 696 697 rPolyPoly.mpImplPolyPolygon->mnRefCount++; 698 699 if ( mpImplPolyPolygon->mnRefCount > 1 ) 700 mpImplPolyPolygon->mnRefCount--; 701 else 702 delete mpImplPolyPolygon; 703 704 mpImplPolyPolygon = rPolyPoly.mpImplPolyPolygon; 705 return *this; 706 } 707 708 // ----------------------------------------------------------------------- 709 710 sal_Bool PolyPolygon::operator==( const PolyPolygon& rPolyPoly ) const 711 { 712 DBG_CHKTHIS( PolyPolygon, NULL ); 713 DBG_CHKOBJ( &rPolyPoly, PolyPolygon, NULL ); 714 715 if ( rPolyPoly.mpImplPolyPolygon == mpImplPolyPolygon ) 716 return sal_True; 717 else 718 return sal_False; 719 } 720 721 // ----------------------------------------------------------------------- 722 723 sal_Bool PolyPolygon::IsEqual( const PolyPolygon& rPolyPoly ) const 724 { 725 sal_Bool bIsEqual = sal_True; 726 if ( Count() != rPolyPoly.Count() ) 727 bIsEqual = sal_False; 728 else 729 { 730 sal_uInt16 i; 731 for ( i = 0; i < Count(); i++ ) 732 { 733 if (!GetObject( i ).IsEqual( rPolyPoly.GetObject( i ) ) ) 734 { 735 bIsEqual = sal_False; 736 break; 737 } 738 } 739 } 740 return bIsEqual; 741 } 742 743 // ----------------------------------------------------------------------- 744 745 SvStream& operator>>( SvStream& rIStream, PolyPolygon& rPolyPoly ) 746 { 747 DBG_CHKOBJ( &rPolyPoly, PolyPolygon, NULL ); 748 DBG_ASSERTWARNING( rIStream.GetVersion(), "PolyPolygon::>> - Solar-Version not set on rIStream" ); 749 750 Polygon* pPoly; 751 sal_uInt16 nPolyCount; 752 753 // Anzahl der Polygone einlesen 754 rIStream >> nPolyCount; 755 756 // Daten anlegen 757 if( nPolyCount ) 758 { 759 // Referenzcounter beruecksichtigen 760 if ( rPolyPoly.mpImplPolyPolygon->mnRefCount > 1 ) 761 rPolyPoly.mpImplPolyPolygon->mnRefCount--; 762 else 763 delete rPolyPoly.mpImplPolyPolygon; 764 765 rPolyPoly.mpImplPolyPolygon = new ImplPolyPolygon( nPolyCount ); 766 767 for ( sal_uInt16 i = 0; i < nPolyCount; i++ ) 768 { 769 pPoly = new Polygon; 770 rIStream >> *pPoly; 771 rPolyPoly.mpImplPolyPolygon->mpPolyAry[i] = pPoly; 772 } 773 } 774 else 775 rPolyPoly = PolyPolygon(); 776 777 return rIStream; 778 } 779 780 // ----------------------------------------------------------------------- 781 782 SvStream& operator<<( SvStream& rOStream, const PolyPolygon& rPolyPoly ) 783 { 784 DBG_CHKOBJ( &rPolyPoly, PolyPolygon, NULL ); 785 DBG_ASSERTWARNING( rOStream.GetVersion(), "PolyPolygon::<< - Solar-Version not set on rOStream" ); 786 787 // Anzahl der Polygone rausschreiben 788 sal_uInt16 nPolyCount = rPolyPoly.mpImplPolyPolygon->mnCount; 789 rOStream << nPolyCount; 790 791 // Die einzelnen Polygone ausgeben 792 for ( sal_uInt16 i = 0; i < nPolyCount; i++ ) 793 rOStream << *(rPolyPoly.mpImplPolyPolygon->mpPolyAry[i]); 794 795 return rOStream; 796 } 797 798 // ----------------------------------------------------------------------- 799 800 void PolyPolygon::Read( SvStream& rIStream ) 801 { 802 VersionCompat aCompat( rIStream, STREAM_READ ); 803 804 DBG_CHKTHIS( PolyPolygon, NULL ); 805 DBG_ASSERTWARNING( rIStream.GetVersion(), "PolyPolygon::>> - Solar-Version not set on rIStream" ); 806 807 Polygon* pPoly; 808 sal_uInt16 nPolyCount; 809 810 // Anzahl der Polygone einlesen 811 rIStream >> nPolyCount; 812 813 // Daten anlegen 814 if( nPolyCount ) 815 { 816 // Referenzcounter beruecksichtigen 817 if ( mpImplPolyPolygon->mnRefCount > 1 ) 818 mpImplPolyPolygon->mnRefCount--; 819 else 820 delete mpImplPolyPolygon; 821 822 mpImplPolyPolygon = new ImplPolyPolygon( nPolyCount ); 823 824 for ( sal_uInt16 i = 0; i < nPolyCount; i++ ) 825 { 826 pPoly = new Polygon; 827 pPoly->ImplRead( rIStream ); 828 mpImplPolyPolygon->mpPolyAry[i] = pPoly; 829 } 830 } 831 else 832 *this = PolyPolygon(); 833 } 834 835 // ----------------------------------------------------------------------- 836 837 void PolyPolygon::Write( SvStream& rOStream ) const 838 { 839 VersionCompat aCompat( rOStream, STREAM_WRITE, 1 ); 840 841 DBG_CHKTHIS( PolyPolygon, NULL ); 842 DBG_ASSERTWARNING( rOStream.GetVersion(), "PolyPolygon::<< - Solar-Version not set on rOStream" ); 843 844 // Anzahl der Polygone rausschreiben 845 sal_uInt16 nPolyCount = mpImplPolyPolygon->mnCount; 846 rOStream << nPolyCount; 847 848 // Die einzelnen Polygone ausgeben 849 for ( sal_uInt16 i = 0; i < nPolyCount; i++ ) 850 mpImplPolyPolygon->mpPolyAry[i]->ImplWrite( rOStream );; 851 } 852 853 // ----------------------------------------------------------------------- 854 // convert to basegfx::B2DPolyPolygon and return 855 basegfx::B2DPolyPolygon PolyPolygon::getB2DPolyPolygon() const 856 { 857 basegfx::B2DPolyPolygon aRetval; 858 859 for(sal_uInt16 a(0); a < mpImplPolyPolygon->mnCount; a++) 860 { 861 Polygon* pCandidate = mpImplPolyPolygon->mpPolyAry[a]; 862 aRetval.append(pCandidate->getB2DPolygon()); 863 } 864 865 return aRetval; 866 } 867 868 // ----------------------------------------------------------------------- 869 // constructor to convert from basegfx::B2DPolyPolygon 870 PolyPolygon::PolyPolygon(const basegfx::B2DPolyPolygon& rPolyPolygon) 871 { 872 DBG_CTOR( PolyPolygon, NULL ); 873 const sal_uInt16 nCount(sal_uInt16(rPolyPolygon.count())); 874 DBG_ASSERT(sal_uInt32(nCount) == rPolyPolygon.count(), 875 "PolyPolygon::PolyPolygon: Too many sub-polygons in given basegfx::B2DPolyPolygon (!)"); 876 877 if ( nCount ) 878 { 879 mpImplPolyPolygon = new ImplPolyPolygon( nCount ); 880 881 for(sal_uInt16 a(0); a < nCount; a++) 882 { 883 basegfx::B2DPolygon aCandidate(rPolyPolygon.getB2DPolygon(sal_uInt32(a))); 884 mpImplPolyPolygon->mpPolyAry[a] = new Polygon( aCandidate ); 885 } 886 } 887 else 888 { 889 mpImplPolyPolygon = new ImplPolyPolygon( 16, 16 ); 890 } 891 } 892 893 // eof 894