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_basegfx.hxx" 26 #include <cstdio> 27 #include <osl/diagnose.h> 28 #include <basegfx/polygon/b2dlinegeometry.hxx> 29 #include <basegfx/point/b2dpoint.hxx> 30 #include <basegfx/vector/b2dvector.hxx> 31 #include <basegfx/polygon/b2dpolygontools.hxx> 32 #include <basegfx/polygon/b2dpolypolygontools.hxx> 33 #include <basegfx/range/b2drange.hxx> 34 #include <basegfx/matrix/b2dhommatrix.hxx> 35 #include <basegfx/curve/b2dcubicbezier.hxx> 36 #include <basegfx/matrix/b2dhommatrixtools.hxx> 37 #include <com/sun/star/drawing/LineCap.hpp> 38 #include <basegfx/polygon/b2dpolypolygoncutter.hxx> 39 40 ////////////////////////////////////////////////////////////////////////////// 41 42 namespace basegfx 43 { 44 namespace tools 45 { createAreaGeometryForLineStartEnd(const B2DPolygon & rCandidate,const B2DPolyPolygon & rArrow,bool bStart,double fWidth,double fCandidateLength,double fDockingPosition,double * pConsumedLength)46 B2DPolyPolygon createAreaGeometryForLineStartEnd( 47 const B2DPolygon& rCandidate, 48 const B2DPolyPolygon& rArrow, 49 bool bStart, 50 double fWidth, 51 double fCandidateLength, 52 double fDockingPosition, // 0->top, 1->bottom 53 double* pConsumedLength) 54 { 55 B2DPolyPolygon aRetval; 56 OSL_ENSURE(rCandidate.count() > 1L, "createAreaGeometryForLineStartEnd: Line polygon has too less points (!)"); 57 OSL_ENSURE(rArrow.count() > 0L, "createAreaGeometryForLineStartEnd: Empty arrow PolyPolygon (!)"); 58 OSL_ENSURE(fWidth > 0.0, "createAreaGeometryForLineStartEnd: Width too small (!)"); 59 OSL_ENSURE(fDockingPosition >= 0.0 && fDockingPosition <= 1.0, 60 "createAreaGeometryForLineStartEnd: fDockingPosition out of range [0.0 .. 1.0] (!)"); 61 62 if(fWidth < 0.0) 63 { 64 fWidth = -fWidth; 65 } 66 67 if(rCandidate.count() > 1 && rArrow.count() && !fTools::equalZero(fWidth)) 68 { 69 if(fDockingPosition < 0.0) 70 { 71 fDockingPosition = 0.0; 72 } 73 else if(fDockingPosition > 1.0) 74 { 75 fDockingPosition = 1.0; 76 } 77 78 // init return value from arrow 79 aRetval.append(rArrow); 80 81 // get size of the arrow 82 const B2DRange aArrowSize(getRange(rArrow)); 83 84 // build ArrowTransform; center in X, align with axis in Y 85 B2DHomMatrix aArrowTransform(basegfx::tools::createTranslateB2DHomMatrix( 86 -aArrowSize.getCenter().getX(), -aArrowSize.getMinimum().getY())); 87 88 // scale to target size 89 const double fArrowScale(fWidth / (aArrowSize.getRange().getX())); 90 aArrowTransform.scale(fArrowScale, fArrowScale); 91 92 // get arrow size in Y 93 B2DPoint aUpperCenter(aArrowSize.getCenter().getX(), aArrowSize.getMaximum().getY()); 94 aUpperCenter *= aArrowTransform; 95 const double fArrowYLength(B2DVector(aUpperCenter).getLength()); 96 97 // move arrow to have docking position centered 98 aArrowTransform.translate(0.0, -fArrowYLength * fDockingPosition); 99 100 // prepare polygon length 101 if(fTools::equalZero(fCandidateLength)) 102 { 103 fCandidateLength = getLength(rCandidate); 104 } 105 106 // get the polygon vector we want to plant this arrow on 107 const double fConsumedLength(fArrowYLength * (1.0 - fDockingPosition)); 108 const B2DVector aHead(rCandidate.getB2DPoint((bStart) ? 0L : rCandidate.count() - 1L)); 109 const B2DVector aTail(getPositionAbsolute(rCandidate, 110 (bStart) ? fConsumedLength : fCandidateLength - fConsumedLength, fCandidateLength)); 111 112 // from that vector, take the needed rotation and add rotate for arrow to transformation 113 const B2DVector aTargetDirection(aHead - aTail); 114 const double fRotation(atan2(aTargetDirection.getY(), aTargetDirection.getX()) + (90.0 * F_PI180)); 115 116 // rotate around docking position 117 aArrowTransform.rotate(fRotation); 118 119 // move arrow docking position to polygon head 120 aArrowTransform.translate(aHead.getX(), aHead.getY()); 121 122 // transform retval and close 123 aRetval.transform(aArrowTransform); 124 aRetval.setClosed(true); 125 126 // if pConsumedLength is asked for, fill it 127 if(pConsumedLength) 128 { 129 *pConsumedLength = fConsumedLength; 130 } 131 } 132 133 return aRetval; 134 } 135 } // end of namespace tools 136 } // end of namespace basegfx 137 138 ////////////////////////////////////////////////////////////////////////////// 139 140 namespace basegfx 141 { 142 // anonymus namespace for local helpers 143 namespace 144 { impIsSimpleEdge(const B2DCubicBezier & rCandidate,double fMaxCosQuad,double fMaxPartOfEdgeQuad)145 bool impIsSimpleEdge(const B2DCubicBezier& rCandidate, double fMaxCosQuad, double fMaxPartOfEdgeQuad) 146 { 147 // isBezier() is true, already tested by caller 148 const B2DVector aEdge(rCandidate.getEndPoint() - rCandidate.getStartPoint()); 149 150 if(aEdge.equalZero()) 151 { 152 // start and end point the same, but control vectors used -> baloon curve loop 153 // is not a simple edge 154 return false; 155 } 156 157 // get tangentA and scalar with edge 158 const B2DVector aTangentA(rCandidate.getTangent(0.0)); 159 const double fScalarAE(aEdge.scalar(aTangentA)); 160 161 if(fTools::lessOrEqual(fScalarAE, 0.0)) 162 { 163 // angle between TangentA and Edge is bigger or equal 90 degrees 164 return false; 165 } 166 167 // get self-scalars for E and A 168 const double fScalarE(aEdge.scalar(aEdge)); 169 const double fScalarA(aTangentA.scalar(aTangentA)); 170 const double fLengthCompareE(fScalarE * fMaxPartOfEdgeQuad); 171 172 if(fTools::moreOrEqual(fScalarA, fLengthCompareE)) 173 { 174 // length of TangentA is more than fMaxPartOfEdge of length of edge 175 return false; 176 } 177 178 if(fTools::lessOrEqual(fScalarAE * fScalarAE, fScalarA * fScalarE * fMaxCosQuad)) 179 { 180 // angle between TangentA and Edge is bigger or equal angle defined by fMaxCos 181 return false; 182 } 183 184 // get tangentB and scalar with edge 185 const B2DVector aTangentB(rCandidate.getTangent(1.0)); 186 const double fScalarBE(aEdge.scalar(aTangentB)); 187 188 if(fTools::lessOrEqual(fScalarBE, 0.0)) 189 { 190 // angle between TangentB and Edge is bigger or equal 90 degrees 191 return false; 192 } 193 194 // get self-scalar for B 195 const double fScalarB(aTangentB.scalar(aTangentB)); 196 197 if(fTools::moreOrEqual(fScalarB, fLengthCompareE)) 198 { 199 // length of TangentB is more than fMaxPartOfEdge of length of edge 200 return false; 201 } 202 203 if(fTools::lessOrEqual(fScalarBE * fScalarBE, fScalarB * fScalarE * fMaxCosQuad)) 204 { 205 // angle between TangentB and Edge is bigger or equal defined by fMaxCos 206 return false; 207 } 208 209 return true; 210 } 211 impSubdivideToSimple(const B2DCubicBezier & rCandidate,B2DPolygon & rTarget,double fMaxCosQuad,double fMaxPartOfEdgeQuad,sal_uInt32 nMaxRecursionDepth)212 void impSubdivideToSimple(const B2DCubicBezier& rCandidate, B2DPolygon& rTarget, double fMaxCosQuad, double fMaxPartOfEdgeQuad, sal_uInt32 nMaxRecursionDepth) 213 { 214 if(!nMaxRecursionDepth || impIsSimpleEdge(rCandidate, fMaxCosQuad, fMaxPartOfEdgeQuad)) 215 { 216 rTarget.appendBezierSegment(rCandidate.getControlPointA(), rCandidate.getControlPointB(), rCandidate.getEndPoint()); 217 } 218 else 219 { 220 B2DCubicBezier aLeft, aRight; 221 rCandidate.split(0.5, &aLeft, &aRight); 222 223 impSubdivideToSimple(aLeft, rTarget, fMaxCosQuad, fMaxPartOfEdgeQuad, nMaxRecursionDepth - 1); 224 impSubdivideToSimple(aRight, rTarget, fMaxCosQuad, fMaxPartOfEdgeQuad, nMaxRecursionDepth - 1); 225 } 226 } 227 subdivideToSimple(const B2DPolygon & rCandidate,double fMaxCosQuad,double fMaxPartOfEdgeQuad)228 B2DPolygon subdivideToSimple(const B2DPolygon& rCandidate, double fMaxCosQuad, double fMaxPartOfEdgeQuad) 229 { 230 const sal_uInt32 nPointCount(rCandidate.count()); 231 232 if(rCandidate.areControlPointsUsed() && nPointCount) 233 { 234 const sal_uInt32 nEdgeCount(rCandidate.isClosed() ? nPointCount : nPointCount - 1); 235 B2DPolygon aRetval; 236 B2DCubicBezier aEdge; 237 238 // prepare edge for loop 239 aEdge.setStartPoint(rCandidate.getB2DPoint(0)); 240 aRetval.append(aEdge.getStartPoint()); 241 242 for(sal_uInt32 a(0); a < nEdgeCount; a++) 243 { 244 // fill B2DCubicBezier 245 const sal_uInt32 nNextIndex((a + 1) % nPointCount); 246 aEdge.setControlPointA(rCandidate.getNextControlPoint(a)); 247 aEdge.setControlPointB(rCandidate.getPrevControlPoint(nNextIndex)); 248 aEdge.setEndPoint(rCandidate.getB2DPoint(nNextIndex)); 249 250 // get rid of unnecessary bezier segments 251 aEdge.testAndSolveTrivialBezier(); 252 253 if(aEdge.isBezier()) 254 { 255 // before splitting recursively with internal simple criteria, use 256 // ExtremumPosFinder to remove those 257 ::std::vector< double > aExtremumPositions; 258 259 aExtremumPositions.reserve(4); 260 aEdge.getAllExtremumPositions(aExtremumPositions); 261 262 const sal_uInt32 nCount(aExtremumPositions.size()); 263 264 if(nCount) 265 { 266 if(nCount > 1) 267 { 268 // create order from left to right 269 ::std::sort(aExtremumPositions.begin(), aExtremumPositions.end()); 270 } 271 272 for(sal_uInt32 b(0); b < nCount;) 273 { 274 // split aEdge at next split pos 275 B2DCubicBezier aLeft; 276 const double fSplitPos(aExtremumPositions[b++]); 277 278 aEdge.split(fSplitPos, &aLeft, &aEdge); 279 aLeft.testAndSolveTrivialBezier(); 280 281 // consume left part 282 if(aLeft.isBezier()) 283 { 284 impSubdivideToSimple(aLeft, aRetval, fMaxCosQuad, fMaxPartOfEdgeQuad, 6); 285 } 286 else 287 { 288 aRetval.append(aLeft.getEndPoint()); 289 } 290 291 if(b < nCount) 292 { 293 // correct the remaining split positions to fit to shortened aEdge 294 const double fScaleFactor(1.0 / (1.0 - fSplitPos)); 295 296 for(sal_uInt32 c(b); c < nCount; c++) 297 { 298 aExtremumPositions[c] = (aExtremumPositions[c] - fSplitPos) * fScaleFactor; 299 } 300 } 301 } 302 303 // test the shortened rest of aEdge 304 aEdge.testAndSolveTrivialBezier(); 305 306 // consume right part 307 if(aEdge.isBezier()) 308 { 309 impSubdivideToSimple(aEdge, aRetval, fMaxCosQuad, fMaxPartOfEdgeQuad, 6); 310 } 311 else 312 { 313 aRetval.append(aEdge.getEndPoint()); 314 } 315 } 316 else 317 { 318 impSubdivideToSimple(aEdge, aRetval, fMaxCosQuad, fMaxPartOfEdgeQuad, 6); 319 } 320 } 321 else 322 { 323 // straight edge, add point 324 aRetval.append(aEdge.getEndPoint()); 325 } 326 327 // prepare edge for next step 328 aEdge.setStartPoint(aEdge.getEndPoint()); 329 } 330 331 // copy closed flag and check for double points 332 aRetval.setClosed(rCandidate.isClosed()); 333 aRetval.removeDoublePoints(); 334 335 return aRetval; 336 } 337 else 338 { 339 return rCandidate; 340 } 341 } 342 createAreaGeometryForEdge(const B2DCubicBezier & rEdge,double fHalfLineWidth,bool bStartRound,bool bEndRound,bool bStartSquare,bool bEndSquare)343 B2DPolygon createAreaGeometryForEdge( 344 const B2DCubicBezier& rEdge, 345 double fHalfLineWidth, 346 bool bStartRound, 347 bool bEndRound, 348 bool bStartSquare, 349 bool bEndSquare) 350 { 351 // create polygon for edge 352 // Unfortunately, while it would be geometrically correct to not add 353 // the in-between points EdgeEnd and EdgeStart, it leads to rounding 354 // errors when converting to integer polygon coordinates for painting 355 if(rEdge.isBezier()) 356 { 357 // prepare target and data common for upper and lower 358 B2DPolygon aBezierPolygon; 359 const B2DVector aPureEdgeVector(rEdge.getEndPoint() - rEdge.getStartPoint()); 360 const double fEdgeLength(aPureEdgeVector.getLength()); 361 const bool bIsEdgeLengthZero(fTools::equalZero(fEdgeLength)); 362 B2DVector aTangentA(rEdge.getTangent(0.0)); aTangentA.normalize(); 363 B2DVector aTangentB(rEdge.getTangent(1.0)); aTangentB.normalize(); 364 const B2DVector aNormalizedPerpendicularA(getPerpendicular(aTangentA)); 365 const B2DVector aNormalizedPerpendicularB(getPerpendicular(aTangentB)); 366 367 // create upper displacement vectors and check if they cut 368 const B2DVector aPerpendStartA(aNormalizedPerpendicularA * -fHalfLineWidth); 369 const B2DVector aPerpendEndA(aNormalizedPerpendicularB * -fHalfLineWidth); 370 double fCutA(0.0); 371 const tools::CutFlagValue aCutA(tools::findCut( 372 rEdge.getStartPoint(), aPerpendStartA, 373 rEdge.getEndPoint(), aPerpendEndA, 374 CUTFLAG_ALL, &fCutA)); 375 const bool bCutA(CUTFLAG_NONE != aCutA); 376 377 // create lower displacement vectors and check if they cut 378 const B2DVector aPerpendStartB(aNormalizedPerpendicularA * fHalfLineWidth); 379 const B2DVector aPerpendEndB(aNormalizedPerpendicularB * fHalfLineWidth); 380 double fCutB(0.0); 381 const tools::CutFlagValue aCutB(tools::findCut( 382 rEdge.getEndPoint(), aPerpendEndB, 383 rEdge.getStartPoint(), aPerpendStartB, 384 CUTFLAG_ALL, &fCutB)); 385 const bool bCutB(CUTFLAG_NONE != aCutB); 386 387 // check if cut happens 388 const bool bCut(bCutA || bCutB); 389 B2DPoint aCutPoint; 390 391 // create left edge 392 if(bStartRound || bStartSquare) 393 { 394 if(bStartRound) 395 { 396 basegfx::B2DPolygon aStartPolygon(tools::createHalfUnitCircle()); 397 398 aStartPolygon.transform( 399 tools::createScaleShearXRotateTranslateB2DHomMatrix( 400 fHalfLineWidth, fHalfLineWidth, 401 0.0, 402 atan2(aTangentA.getY(), aTangentA.getX()) + F_PI2, 403 rEdge.getStartPoint().getX(), rEdge.getStartPoint().getY())); 404 aBezierPolygon.append(aStartPolygon); 405 } 406 else // bStartSquare 407 { 408 const basegfx::B2DPoint aStart(rEdge.getStartPoint() - (aTangentA * fHalfLineWidth)); 409 410 if(bCutB) 411 { 412 aBezierPolygon.append(rEdge.getStartPoint() + aPerpendStartB); 413 } 414 415 aBezierPolygon.append(aStart + aPerpendStartB); 416 aBezierPolygon.append(aStart + aPerpendStartA); 417 418 if(bCutA) 419 { 420 aBezierPolygon.append(rEdge.getStartPoint() + aPerpendStartA); 421 } 422 } 423 } 424 else 425 { 426 // append original in-between point 427 aBezierPolygon.append(rEdge.getStartPoint()); 428 } 429 430 // create upper edge. 431 { 432 if(bCutA) 433 { 434 // calculate cut point and add 435 aCutPoint = rEdge.getStartPoint() + (aPerpendStartA * fCutA); 436 aBezierPolygon.append(aCutPoint); 437 } 438 else 439 { 440 // create scaled bezier segment 441 const B2DPoint aStart(rEdge.getStartPoint() + aPerpendStartA); 442 const B2DPoint aEnd(rEdge.getEndPoint() + aPerpendEndA); 443 const B2DVector aEdge(aEnd - aStart); 444 const double fLength(aEdge.getLength()); 445 const double fScale(bIsEdgeLengthZero ? 1.0 : fLength / fEdgeLength); 446 const B2DVector fRelNext(rEdge.getControlPointA() - rEdge.getStartPoint()); 447 const B2DVector fRelPrev(rEdge.getControlPointB() - rEdge.getEndPoint()); 448 449 aBezierPolygon.append(aStart); 450 aBezierPolygon.appendBezierSegment(aStart + (fRelNext * fScale), aEnd + (fRelPrev * fScale), aEnd); 451 } 452 } 453 454 // create right edge 455 if(bEndRound || bEndSquare) 456 { 457 if(bEndRound) 458 { 459 basegfx::B2DPolygon aEndPolygon(tools::createHalfUnitCircle()); 460 461 aEndPolygon.transform( 462 tools::createScaleShearXRotateTranslateB2DHomMatrix( 463 fHalfLineWidth, fHalfLineWidth, 464 0.0, 465 atan2(aTangentB.getY(), aTangentB.getX()) - F_PI2, 466 rEdge.getEndPoint().getX(), rEdge.getEndPoint().getY())); 467 aBezierPolygon.append(aEndPolygon); 468 } 469 else // bEndSquare 470 { 471 const basegfx::B2DPoint aEnd(rEdge.getEndPoint() + (aTangentB * fHalfLineWidth)); 472 473 if(bCutA) 474 { 475 aBezierPolygon.append(rEdge.getEndPoint() + aPerpendEndA); 476 } 477 478 aBezierPolygon.append(aEnd + aPerpendEndA); 479 aBezierPolygon.append(aEnd + aPerpendEndB); 480 481 if(bCutB) 482 { 483 aBezierPolygon.append(rEdge.getEndPoint() + aPerpendEndB); 484 } 485 } 486 } 487 else 488 { 489 // append original in-between point 490 aBezierPolygon.append(rEdge.getEndPoint()); 491 } 492 493 // create lower edge. 494 { 495 if(bCutB) 496 { 497 // calculate cut point and add 498 aCutPoint = rEdge.getEndPoint() + (aPerpendEndB * fCutB); 499 aBezierPolygon.append(aCutPoint); 500 } 501 else 502 { 503 // create scaled bezier segment 504 const B2DPoint aStart(rEdge.getEndPoint() + aPerpendEndB); 505 const B2DPoint aEnd(rEdge.getStartPoint() + aPerpendStartB); 506 const B2DVector aEdge(aEnd - aStart); 507 const double fLength(aEdge.getLength()); 508 const double fScale(bIsEdgeLengthZero ? 1.0 : fLength / fEdgeLength); 509 const B2DVector fRelNext(rEdge.getControlPointB() - rEdge.getEndPoint()); 510 const B2DVector fRelPrev(rEdge.getControlPointA() - rEdge.getStartPoint()); 511 512 aBezierPolygon.append(aStart); 513 aBezierPolygon.appendBezierSegment(aStart + (fRelNext * fScale), aEnd + (fRelPrev * fScale), aEnd); 514 } 515 } 516 517 // close 518 aBezierPolygon.setClosed(true); 519 520 if(bStartRound || bEndRound) 521 { 522 // double points possible when round caps are used at start or end 523 aBezierPolygon.removeDoublePoints(); 524 } 525 526 if(bCut && ((bStartRound || bStartSquare) && (bEndRound || bEndSquare))) 527 { 528 // When cut exists and both ends are extended with caps, a self-intersecting polygon 529 // is created; one cut point is known, but there is a 2nd one in the caps geometry. 530 // Solve by using tooling. 531 // Remark: This nearly never happens due to curve preparations to extreme points 532 // and maximum angle turning, but I constructed a test case and checkd that it is 533 // working propery. 534 const B2DPolyPolygon aTemp(tools::solveCrossovers(aBezierPolygon)); 535 const sal_uInt32 nTempCount(aTemp.count()); 536 537 if(nTempCount) 538 { 539 if(nTempCount > 1) 540 { 541 // as expected, multiple polygons (with same orientation). Remove 542 // the one which contains aCutPoint, or better take the one without 543 for (sal_uInt32 a(0); a < nTempCount; a++) 544 { 545 aBezierPolygon = aTemp.getB2DPolygon(a); 546 547 const sal_uInt32 nCandCount(aBezierPolygon.count()); 548 549 for(sal_uInt32 b(0); b < nCandCount; b++) 550 { 551 if(aCutPoint.equal(aBezierPolygon.getB2DPoint(b))) 552 { 553 aBezierPolygon.clear(); 554 break; 555 } 556 } 557 558 if(aBezierPolygon.count()) 559 { 560 break; 561 } 562 } 563 564 OSL_ENSURE(aBezierPolygon.count(), "Error in line geometry creation, could not solve self-intersection (!)"); 565 } 566 else 567 { 568 // none found, use result 569 aBezierPolygon = aTemp.getB2DPolygon(0); 570 } 571 } 572 else 573 { 574 OSL_ENSURE(false, "Error in line geometry creation, could not solve self-intersection (!)"); 575 } 576 } 577 578 // return 579 return aBezierPolygon; 580 } 581 else 582 { 583 // Get start and end point, create tangent and set to needed length 584 B2DVector aTangent(rEdge.getEndPoint() - rEdge.getStartPoint()); 585 aTangent.setLength(fHalfLineWidth); 586 587 // prepare return value 588 B2DPolygon aEdgePolygon; 589 590 // buffered angle 591 double fAngle(0.0); 592 bool bAngle(false); 593 594 // buffered perpendicular 595 B2DVector aPerpend; 596 bool bPerpend(false); 597 598 // create left vertical 599 if(bStartRound) 600 { 601 aEdgePolygon = tools::createHalfUnitCircle(); 602 fAngle = atan2(aTangent.getY(), aTangent.getX()); 603 bAngle = true; 604 aEdgePolygon.transform( 605 tools::createScaleShearXRotateTranslateB2DHomMatrix( 606 fHalfLineWidth, fHalfLineWidth, 607 0.0, 608 fAngle + F_PI2, 609 rEdge.getStartPoint().getX(), rEdge.getStartPoint().getY())); 610 } 611 else 612 { 613 aPerpend.setX(-aTangent.getY()); 614 aPerpend.setY(aTangent.getX()); 615 bPerpend = true; 616 617 if(bStartSquare) 618 { 619 const basegfx::B2DPoint aStart(rEdge.getStartPoint() - aTangent); 620 621 aEdgePolygon.append(aStart + aPerpend); 622 aEdgePolygon.append(aStart - aPerpend); 623 } 624 else 625 { 626 aEdgePolygon.append(rEdge.getStartPoint() + aPerpend); 627 aEdgePolygon.append(rEdge.getStartPoint()); // keep the in-between point for numerical reasons 628 aEdgePolygon.append(rEdge.getStartPoint() - aPerpend); 629 } 630 } 631 632 // create right vertical 633 if(bEndRound) 634 { 635 basegfx::B2DPolygon aEndPolygon(tools::createHalfUnitCircle()); 636 637 if(!bAngle) 638 { 639 fAngle = atan2(aTangent.getY(), aTangent.getX()); 640 } 641 642 aEndPolygon.transform( 643 tools::createScaleShearXRotateTranslateB2DHomMatrix( 644 fHalfLineWidth, fHalfLineWidth, 645 0.0, 646 fAngle - F_PI2, 647 rEdge.getEndPoint().getX(), rEdge.getEndPoint().getY())); 648 aEdgePolygon.append(aEndPolygon); 649 } 650 else 651 { 652 if(!bPerpend) 653 { 654 aPerpend.setX(-aTangent.getY()); 655 aPerpend.setY(aTangent.getX()); 656 } 657 658 if(bEndSquare) 659 { 660 const basegfx::B2DPoint aEnd(rEdge.getEndPoint() + aTangent); 661 662 aEdgePolygon.append(aEnd - aPerpend); 663 aEdgePolygon.append(aEnd + aPerpend); 664 } 665 else 666 { 667 aEdgePolygon.append(rEdge.getEndPoint() - aPerpend); 668 aEdgePolygon.append(rEdge.getEndPoint()); // keep the in-between point for numerical reasons 669 aEdgePolygon.append(rEdge.getEndPoint() + aPerpend); 670 } 671 } 672 673 // close and return 674 aEdgePolygon.setClosed(true); 675 676 return aEdgePolygon; 677 } 678 } 679 createAreaGeometryForJoin(const B2DVector & rTangentPrev,const B2DVector & rTangentEdge,const B2DVector & rPerpendPrev,const B2DVector & rPerpendEdge,const B2DPoint & rPoint,double fHalfLineWidth,B2DLineJoin eJoin,double fMiterMinimumAngle)680 B2DPolygon createAreaGeometryForJoin( 681 const B2DVector& rTangentPrev, 682 const B2DVector& rTangentEdge, 683 const B2DVector& rPerpendPrev, 684 const B2DVector& rPerpendEdge, 685 const B2DPoint& rPoint, 686 double fHalfLineWidth, 687 B2DLineJoin eJoin, 688 double fMiterMinimumAngle) 689 { 690 OSL_ENSURE(fHalfLineWidth > 0.0, "createAreaGeometryForJoin: LineWidth too small (!)"); 691 OSL_ENSURE(B2DLINEJOIN_NONE != eJoin, "createAreaGeometryForJoin: B2DLINEJOIN_NONE not allowed (!)"); 692 693 // LineJoin from tangent rPerpendPrev to tangent rPerpendEdge in rPoint 694 B2DPolygon aEdgePolygon; 695 const B2DPoint aStartPoint(rPoint + rPerpendPrev); 696 const B2DPoint aEndPoint(rPoint + rPerpendEdge); 697 698 // test if for Miter, the angle is too small and the fallback 699 // to bevel needs to be used 700 if(B2DLINEJOIN_MITER == eJoin) 701 { 702 const double fAngle(fabs(rPerpendPrev.angle(rPerpendEdge))); 703 704 if((F_PI - fAngle) < fMiterMinimumAngle) 705 { 706 // fallback to bevel 707 eJoin = B2DLINEJOIN_BEVEL; 708 } 709 } 710 711 switch(eJoin) 712 { 713 case B2DLINEJOIN_MITER : 714 { 715 aEdgePolygon.append(aEndPoint); 716 aEdgePolygon.append(rPoint); 717 aEdgePolygon.append(aStartPoint); 718 719 // Look for the cut point between start point along rTangentPrev and 720 // end point along rTangentEdge. -rTangentEdge should be used, but since 721 // the cut value is used for interpolating along the first edge, the negation 722 // is not needed since the same fCut will be found on the first edge. 723 // If it exists, insert it to complete the mitered fill polygon. 724 double fCutPos(0.0); 725 tools::findCut(aStartPoint, rTangentPrev, aEndPoint, rTangentEdge, CUTFLAG_ALL, &fCutPos); 726 727 if(0.0 != fCutPos) 728 { 729 const B2DPoint aCutPoint(aStartPoint + (rTangentPrev * fCutPos)); 730 aEdgePolygon.append(aCutPoint); 731 } 732 733 break; 734 } 735 case B2DLINEJOIN_ROUND : 736 { 737 // use tooling to add needed EllipseSegment 738 double fAngleStart(atan2(rPerpendPrev.getY(), rPerpendPrev.getX())); 739 double fAngleEnd(atan2(rPerpendEdge.getY(), rPerpendEdge.getX())); 740 741 // atan2 results are [-PI .. PI], consolidate to [0.0 .. 2PI] 742 if(fAngleStart < 0.0) 743 { 744 fAngleStart += F_2PI; 745 } 746 747 if(fAngleEnd < 0.0) 748 { 749 fAngleEnd += F_2PI; 750 } 751 752 const B2DPolygon aBow(tools::createPolygonFromEllipseSegment(rPoint, fHalfLineWidth, fHalfLineWidth, fAngleStart, fAngleEnd)); 753 754 if(aBow.count() > 1) 755 { 756 // #i101491# 757 // use the original start/end positions; the ones from bow creation may be numerically 758 // different due to their different creation. To guarantee good merging quality with edges 759 // and edge roundings (and to reduce point count) 760 aEdgePolygon = aBow; 761 aEdgePolygon.setB2DPoint(0, aStartPoint); 762 aEdgePolygon.setB2DPoint(aEdgePolygon.count() - 1, aEndPoint); 763 aEdgePolygon.append(rPoint); 764 765 break; 766 } 767 else 768 { 769 // wanted fall-through to default 770 } 771 } 772 default: // B2DLINEJOIN_BEVEL 773 { 774 aEdgePolygon.append(aEndPoint); 775 aEdgePolygon.append(rPoint); 776 aEdgePolygon.append(aStartPoint); 777 778 break; 779 } 780 } 781 782 // create last polygon part for edge 783 aEdgePolygon.setClosed(true); 784 785 return aEdgePolygon; 786 } 787 } // end of anonymus namespace 788 789 namespace tools 790 { createAreaGeometry(const B2DPolygon & rCandidate,double fHalfLineWidth,B2DLineJoin eJoin,com::sun::star::drawing::LineCap eCap,double fMaxAllowedAngle,double fMaxPartOfEdge,double fMiterMinimumAngle)791 B2DPolyPolygon createAreaGeometry( 792 const B2DPolygon& rCandidate, 793 double fHalfLineWidth, 794 B2DLineJoin eJoin, 795 com::sun::star::drawing::LineCap eCap, 796 double fMaxAllowedAngle, 797 double fMaxPartOfEdge, 798 double fMiterMinimumAngle) 799 { 800 if(fMaxAllowedAngle > F_PI2) 801 { 802 fMaxAllowedAngle = F_PI2; 803 } 804 else if(fMaxAllowedAngle < 0.01 * F_PI2) 805 { 806 fMaxAllowedAngle = 0.01 * F_PI2; 807 } 808 809 if(fMaxPartOfEdge > 1.0) 810 { 811 fMaxPartOfEdge = 1.0; 812 } 813 else if(fMaxPartOfEdge < 0.01) 814 { 815 fMaxPartOfEdge = 0.01; 816 } 817 818 if(fMiterMinimumAngle > F_PI) 819 { 820 fMiterMinimumAngle = F_PI; 821 } 822 else if(fMiterMinimumAngle < 0.01 * F_PI) 823 { 824 fMiterMinimumAngle = 0.01 * F_PI; 825 } 826 827 B2DPolygon aCandidate(rCandidate); 828 const double fMaxCos(cos(fMaxAllowedAngle)); 829 830 aCandidate.removeDoublePoints(); 831 aCandidate = subdivideToSimple(aCandidate, fMaxCos * fMaxCos, fMaxPartOfEdge * fMaxPartOfEdge); 832 833 const sal_uInt32 nPointCount(aCandidate.count()); 834 835 if(nPointCount) 836 { 837 B2DPolyPolygon aRetval; 838 const bool bEventuallyCreateLineJoin(B2DLINEJOIN_NONE != eJoin); 839 const bool bIsClosed(aCandidate.isClosed()); 840 const sal_uInt32 nEdgeCount(bIsClosed ? nPointCount : nPointCount - 1); 841 const bool bLineCap(!bIsClosed && com::sun::star::drawing::LineCap_BUTT != eCap); 842 843 if(nEdgeCount) 844 { 845 B2DCubicBezier aEdge; 846 B2DCubicBezier aPrev; 847 848 // prepare edge 849 aEdge.setStartPoint(aCandidate.getB2DPoint(0)); 850 851 if(bIsClosed && bEventuallyCreateLineJoin) 852 { 853 // prepare previous edge 854 const sal_uInt32 nPrevIndex(nPointCount - 1); 855 aPrev.setStartPoint(aCandidate.getB2DPoint(nPrevIndex)); 856 aPrev.setControlPointA(aCandidate.getNextControlPoint(nPrevIndex)); 857 aPrev.setControlPointB(aCandidate.getPrevControlPoint(0)); 858 aPrev.setEndPoint(aEdge.getStartPoint()); 859 } 860 861 for(sal_uInt32 a(0); a < nEdgeCount; a++) 862 { 863 // fill current Edge 864 const sal_uInt32 nNextIndex((a + 1) % nPointCount); 865 aEdge.setControlPointA(aCandidate.getNextControlPoint(a)); 866 aEdge.setControlPointB(aCandidate.getPrevControlPoint(nNextIndex)); 867 aEdge.setEndPoint(aCandidate.getB2DPoint(nNextIndex)); 868 869 // check and create linejoin 870 if(bEventuallyCreateLineJoin && (bIsClosed || 0 != a)) 871 { 872 B2DVector aTangentPrev(aPrev.getTangent(1.0)); aTangentPrev.normalize(); 873 B2DVector aTangentEdge(aEdge.getTangent(0.0)); aTangentEdge.normalize(); 874 B2VectorOrientation aOrientation(getOrientation(aTangentPrev, aTangentEdge)); 875 876 if(ORIENTATION_NEUTRAL == aOrientation) 877 { 878 // they are parallell or empty; if they are both not zero and point 879 // in opposite direction, a half-circle is needed 880 if(!aTangentPrev.equalZero() && !aTangentEdge.equalZero()) 881 { 882 const double fAngle(fabs(aTangentPrev.angle(aTangentEdge))); 883 884 if(fTools::equal(fAngle, F_PI)) 885 { 886 // for half-circle production, fallback to positive 887 // orientation 888 aOrientation = ORIENTATION_POSITIVE; 889 } 890 } 891 } 892 893 if(ORIENTATION_POSITIVE == aOrientation) 894 { 895 const B2DVector aPerpendPrev(getPerpendicular(aTangentPrev) * -fHalfLineWidth); 896 const B2DVector aPerpendEdge(getPerpendicular(aTangentEdge) * -fHalfLineWidth); 897 898 aRetval.append( 899 createAreaGeometryForJoin( 900 aTangentPrev, 901 aTangentEdge, 902 aPerpendPrev, 903 aPerpendEdge, 904 aEdge.getStartPoint(), 905 fHalfLineWidth, 906 eJoin, 907 fMiterMinimumAngle)); 908 } 909 else if(ORIENTATION_NEGATIVE == aOrientation) 910 { 911 const B2DVector aPerpendPrev(getPerpendicular(aTangentPrev) * fHalfLineWidth); 912 const B2DVector aPerpendEdge(getPerpendicular(aTangentEdge) * fHalfLineWidth); 913 914 aRetval.append( 915 createAreaGeometryForJoin( 916 aTangentEdge, 917 aTangentPrev, 918 aPerpendEdge, 919 aPerpendPrev, 920 aEdge.getStartPoint(), 921 fHalfLineWidth, 922 eJoin, 923 fMiterMinimumAngle)); 924 } 925 } 926 927 // create geometry for edge 928 const bool bLast(a + 1 == nEdgeCount); 929 930 if(bLineCap) 931 { 932 const bool bFirst(!a); 933 934 aRetval.append( 935 createAreaGeometryForEdge( 936 aEdge, 937 fHalfLineWidth, 938 bFirst && com::sun::star::drawing::LineCap_ROUND == eCap, 939 bLast && com::sun::star::drawing::LineCap_ROUND == eCap, 940 bFirst && com::sun::star::drawing::LineCap_SQUARE == eCap, 941 bLast && com::sun::star::drawing::LineCap_SQUARE == eCap)); 942 } 943 else 944 { 945 aRetval.append( 946 createAreaGeometryForEdge( 947 aEdge, 948 fHalfLineWidth, 949 false, 950 false, 951 false, 952 false)); 953 } 954 955 // prepare next step 956 if(!bLast) 957 { 958 if(bEventuallyCreateLineJoin) 959 { 960 aPrev = aEdge; 961 } 962 963 aEdge.setStartPoint(aEdge.getEndPoint()); 964 } 965 } 966 } 967 968 return aRetval; 969 } 970 else 971 { 972 return B2DPolyPolygon(rCandidate); 973 } 974 } 975 } // end of namespace tools 976 } // end of namespace basegfx 977 978 ////////////////////////////////////////////////////////////////////////////// 979 // eof 980