summaryrefslogtreecommitdiff
path: root/tools/source/generic/poly.cxx
diff options
context:
space:
mode:
Diffstat (limited to 'tools/source/generic/poly.cxx')
-rw-r--r--tools/source/generic/poly.cxx134
1 files changed, 103 insertions, 31 deletions
diff --git a/tools/source/generic/poly.cxx b/tools/source/generic/poly.cxx
index 39a8fec07cb8..4d786d6aadda 100644
--- a/tools/source/generic/poly.cxx
+++ b/tools/source/generic/poly.cxx
@@ -4,9 +4,9 @@
*
* $RCSfile: poly.cxx,v $
*
- * $Revision: 1.13 $
+ * $Revision: 1.14 $
*
- * last change: $Author: kz $ $Date: 2006-12-13 15:10:36 $
+ * last change: $Author: kz $ $Date: 2007-05-09 13:22:58 $
*
* The Contents of this file are made available subject to
* the terms of GNU Lesser General Public License Version 2.1.
@@ -2183,50 +2183,122 @@ void Polygon::Write( SvStream& rOStream ) const
}
// -----------------------------------------------------------------------
+// #i74631# numerical correction method for B2DPolygon
+bool impCorrectVectorAForContinuity(basegfx::B2DVector& rNewVectorA, BYTE nCFlag, const basegfx::B2DPolygon& rTarget, sal_uInt32 nCurrentIndex, sal_uInt32 nPreviousIndex)
+{
+ if(POLY_SMOOTH == nCFlag || POLY_SYMMTR == nCFlag)
+ {
+ // get previous vectorB
+ const basegfx::B2DVector aPreviousVectorB(rTarget.getControlVectorB(nPreviousIndex));
+
+ if(!aPreviousVectorB.equalZero())
+ {
+ // aPreviousVectorB is used. Make it relative to the new start point
+ const basegfx::B2DVector aRelativeVectorB((rTarget.getB2DPoint(nPreviousIndex) + aPreviousVectorB) - rTarget.getB2DPoint(nCurrentIndex));
+
+ if(!aRelativeVectorB.equalZero())
+ {
+ if(POLY_SMOOTH == nCFlag)
+ {
+ // apply inverse direction of aRelativeVectorB to aVectorA, keep length
+ const double fOriginalLength(rNewVectorA.getLength());
+ rNewVectorA = -aRelativeVectorB;
+ rNewVectorA.setLength(fOriginalLength);
+ }
+ else // POLY_SYMMTR
+ {
+ // apply inverse aRelativeVectorB to rNewVectorA
+ rNewVectorA = -aRelativeVectorB;
+ }
+
+ return true;
+ }
+ }
+ }
+
+ return false;
+}
+
+// -----------------------------------------------------------------------
// convert to ::basegfx::B2DPolygon and return
::basegfx::B2DPolygon Polygon::getB2DPolygon() const
{
- ::basegfx::B2DPolygon aRetval;
+ basegfx::B2DPolygon aRetval;
const sal_uInt16 nCount(mpImplPolygon->mnPoints);
- for(sal_uInt16 a(0L); a < nCount;)
+ if(mpImplPolygon->mpFlagAry)
{
- // get point
- Point aPointA = mpImplPolygon->mpPointAry[a++];
+ // handling for curves. Will work with non-curves too (add test for mpFlagAry
+ // again where commented below), but the more used non-curve case is faster this way
+ for(sal_uInt16 a(0); a < nCount;)
+ {
+ // get start point and add
+ const BYTE nStartPointFlag(mpImplPolygon->mpFlagAry[a]);
+ const Point aStartPoint(mpImplPolygon->mpPointAry[a++]);
+ const basegfx::B2DPoint aB2DStartPoint(aStartPoint.X(), aStartPoint.Y());
+ aRetval.append(aB2DStartPoint);
+
+ // test flags of next point if available, maybe it's a control point
+ if(a < nCount && /*mpImplPolygon->mpFlagAry &&*/ POLY_CONTROL == mpImplPolygon->mpFlagAry[a])
+ {
+ // check if there are two control points
+ if(a + 1 < nCount && POLY_CONTROL == mpImplPolygon->mpFlagAry[a + 1])
+ {
+ // get the two control points and calculate the vectors
+ const Point aControlA(mpImplPolygon->mpPointAry[a++]);
+ const Point aControlB(mpImplPolygon->mpPointAry[a++]);
+ basegfx::B2DVector aVectorA(aControlA.X() - aStartPoint.X(), aControlA.Y() - aStartPoint.Y());
+ const basegfx::B2DVector aVectorB(aControlB.X() - aStartPoint.X(), aControlB.Y() - aStartPoint.Y());
+ const sal_uInt32 nDestIndex(aRetval.count() - 1L);
+
+ if(nDestIndex)
+ {
+ // #i74631# check if we have a C1 or C2 flagged continuity in the source. If yes,
+ // we will need to correct the to-be-set VectorA to have the numerically
+ // correct value for double precision
+ impCorrectVectorAForContinuity(aVectorA, nStartPointFlag, aRetval, nDestIndex, nDestIndex - 1);
+ }
+
+ // add Vectors
+ aRetval.setControlVectorA(nDestIndex, aVectorA);
+ aRetval.setControlVectorB(nDestIndex, aVectorB);
+ }
+ else
+ {
+ // we have an invalid polygon (!), inform the user
+ DBG_ASSERT(false, "Polygon::getB2DPolygon(): invalid source polygon (curve edge with only one control point detected)" );
+ }
+ }
+ }
+
+ // set closed flag
+ ::basegfx::tools::checkClosed(aRetval);
- // test flags of next point if available, maybe it's a control point
- if(a < nCount && mpImplPolygon->mpFlagAry && POLY_CONTROL == mpImplPolygon->mpFlagAry[a])
+ if(aRetval.isClosed() && nCount && aRetval.count())
{
- // get two control points
- Point aControlA = mpImplPolygon->mpPointAry[a++];
+ // #i74631# when a closed poly is produced we need to evtl. correct continuity in start point, too
+ const BYTE nStartPointFlag(mpImplPolygon->mpFlagAry[0]);
+ basegfx::B2DVector aStartVector(aRetval.getControlVectorA(0));
- DBG_ASSERT( a < nCount, "Polygon::getB2DPolygon(): invalid polygon (polygon is ending with only one control point)" );
- if ( a < nCount )
+ if(impCorrectVectorAForContinuity(aStartVector, nStartPointFlag, aRetval, 0, aRetval.count() - 1))
{
- Point aControlB = mpImplPolygon->mpPointAry[a++];
-
- // add point A
- ::basegfx::B2DPoint aPoA(aPointA.X(), aPointA.Y());
- aRetval.append(aPoA);
-
- // calculate Vectors and add them
- const sal_uInt32 nDestIndex(aRetval.count() - 1L);
- ::basegfx::B2DVector aVeA(aControlA.X() - aPointA.X(), aControlA.Y() - aPointA.Y());
- aRetval.setControlVectorA(nDestIndex, aVeA);
- ::basegfx::B2DVector aVeB(aControlB.X() - aPointA.X(), aControlB.Y() - aPointA.Y());
- aRetval.setControlVectorB(nDestIndex, aVeB);
+ aRetval.setControlVectorA(0, aStartVector);
}
}
- else
+ }
+ else
+ {
+ // extra handling for non-curves for speedup
+ for(sal_uInt16 a(0); a < nCount; a++)
{
- // add point A
- ::basegfx::B2DPoint aPoA(aPointA.X(), aPointA.Y());
- aRetval.append(aPoA);
+ // get point and add
+ const Point aPoint(mpImplPolygon->mpPointAry[a]);
+ aRetval.append(basegfx::B2DPoint(aPoint.X(), aPoint.Y()));
}
- }
- // set closed flag
- ::basegfx::tools::checkClosed(aRetval);
+ // set closed flag
+ ::basegfx::tools::checkClosed(aRetval);
+ }
return aRetval;
}