From a4c46ceec433edf0c5de03ea8d36857a455cafd2 Mon Sep 17 00:00:00 2001 From: Regina Henschel Date: Sat, 15 Dec 2018 17:51:25 +0100 Subject: tdf#121761, tdf#121952 Accurate ellipsequadrant in custom shape The current solution uses one, bad valued Bezier curve and does not toggle direction. The patch uses the more accurate solution from basegfx and simplifies the decision tree. In addition it extracts the calculation in double from GetPoint, so they can be used directly instead of double->long->double conversion. Change-Id: I298f49415d1b7624b36ca561647f2e58af5eb5c6 Reviewed-on: https://gerrit.libreoffice.org/65203 Tested-by: Jenkins Reviewed-by: Regina Henschel --- svx/source/customshapes/EnhancedCustomShape2d.cxx | 189 ++++++++++++---------- 1 file changed, 104 insertions(+), 85 deletions(-) (limited to 'svx/source') diff --git a/svx/source/customshapes/EnhancedCustomShape2d.cxx b/svx/source/customshapes/EnhancedCustomShape2d.cxx index 12448ca95054..c01c49949f94 100644 --- a/svx/source/customshapes/EnhancedCustomShape2d.cxx +++ b/svx/source/customshapes/EnhancedCustomShape2d.cxx @@ -917,40 +917,32 @@ bool EnhancedCustomShape2d::SetAdjustValueAsDouble( const double& rValue, const return bRetValue; } -Point EnhancedCustomShape2d::GetPoint( const css::drawing::EnhancedCustomShapeParameterPair& rPair, +basegfx::B2DPoint EnhancedCustomShape2d::GetPointAsB2DPoint( const css::drawing::EnhancedCustomShapeParameterPair& rPair, const bool bScale, const bool bReplaceGeoSize ) const { - Point aRetValue; - sal_uInt32 nPass = 0; - do + double fValX, fValY; + // width + GetParameter(fValX, rPair.First, bReplaceGeoSize, false); + fValX -= nCoordLeft; + if (bScale) { - sal_uInt32 nIndex = nPass; - - double fVal; - const EnhancedCustomShapeParameter& rParameter = nIndex ? rPair.Second : rPair.First; - if ( nPass ) // height - { - GetParameter( fVal, rParameter, false, bReplaceGeoSize ); - fVal -= nCoordTop; - if ( bScale ) - { - fVal *= fYScale; - } - aRetValue.setY( static_cast(fVal) ); - } - else // width - { - GetParameter( fVal, rParameter, bReplaceGeoSize, false ); - fVal -= nCoordLeft; - if ( bScale ) - { - fVal *= fXScale; - } - aRetValue.setX( static_cast(fVal) ); - } + fValX *= fXScale; } - while ( ++nPass < 2 ); - return aRetValue; + // height + GetParameter(fValY, rPair.Second, false, bReplaceGeoSize); + fValY -= nCoordTop; + if (bScale) + { + fValY *= fYScale; + } + return basegfx::B2DPoint(fValX,fValY); +} + +Point EnhancedCustomShape2d::GetPoint( const css::drawing::EnhancedCustomShapeParameterPair& rPair, + const bool bScale, const bool bReplaceGeoSize ) const +{ + basegfx::B2DPoint aPoint(GetPointAsB2DPoint(rPair, bScale, bReplaceGeoSize)); + return Point(static_cast(aPoint.getX()), static_cast(aPoint.getY())); } void EnhancedCustomShape2d::GetParameter( double& rRetValue, const EnhancedCustomShapeParameter& rParameter, @@ -1841,69 +1833,96 @@ void EnhancedCustomShape2d::CreateSubPath( case ELLIPTICALQUADRANTX : case ELLIPTICALQUADRANTY : { - bool bFirstDirection(true); - basegfx::B2DPoint aControlPointA; - basegfx::B2DPoint aControlPointB; - - for ( sal_uInt16 i = 0; ( i < nPntCount ) && ( rSrcPt < nCoordSize ); i++ ) + if (nPntCount && (rSrcPt < nCoordSize)) { - sal_uInt32 nModT = ( nCommand == ELLIPTICALQUADRANTX ) ? 1 : 0; - Point aCurrent( GetPoint( seqCoordinates[ rSrcPt ], true, true ) ); - - if ( rSrcPt ) // we need a previous point + // The arc starts at the previous point and ends at the point given in the parameter. + basegfx::B2DPoint aStart; + basegfx::B2DPoint aEnd; + sal_uInt16 i = 0; + if (rSrcPt) { - Point aPrev( GetPoint( seqCoordinates[ rSrcPt - 1 ], true, true ) ); - sal_Int32 nX, nY; - nX = aCurrent.X() - aPrev.X(); - nY = aCurrent.Y() - aPrev.Y(); - if ( ( nY ^ nX ) & 0x80000000 ) - { - if ( !i ) - bFirstDirection = true; - else if ( !bFirstDirection ) - nModT ^= 1; - } - else - { - if ( !i ) - bFirstDirection = false; - else if ( bFirstDirection ) - nModT ^= 1; - } - if ( nModT ) // get the right corner + aStart = GetPointAsB2DPoint(seqCoordinates[rSrcPt - 1], true, true); + } + else + { // no previous point, path is ill-structured. But we want to show as much as possible. + // Thus make a moveTo to the point given as parameter and continue from there. + aStart = GetPointAsB2DPoint(seqCoordinates[static_cast(rSrcPt)], true, true); + aNewB2DPolygon.append(aStart); + rSrcPt++; + i++; + } + // If there are several points, then the direction changes with every point. + bool bIsXDirection(nCommand == ELLIPTICALQUADRANTX); + basegfx::B2DPolygon aArc; + for ( ; ( i < nPntCount ) && ( rSrcPt < nCoordSize ); i++ ) + { + aEnd = GetPointAsB2DPoint(seqCoordinates[rSrcPt], true, true); + basegfx::B2DPoint aCenter; + double fRadiusX = fabs(aEnd.getX() - aStart.getX()); + double fRadiusY = fabs(aEnd.getY() - aStart.getY()); + if (bIsXDirection) { - nX = aCurrent.X(); - nY = aPrev.Y(); + aCenter = basegfx::B2DPoint(aStart.getX(),aEnd.getY()); + if (aEnd.getX()=aStart.getX() + { + if (aEnd.getY()=aStart.getX() + { + if (aEnd.getY()> 1; - sal_Int32 nYVec = ( nY - aPrev.Y() ) >> 1; - Point aControl1( aPrev.X() + nXVec, aPrev.Y() + nYVec ); - - aControlPointA = basegfx::B2DPoint(aControl1.X(), aControl1.Y()); - - nXVec = ( nX - aCurrent.X() ) >> 1; - nYVec = ( nY - aCurrent.Y() ) >> 1; - Point aControl2( aCurrent.X() + nXVec, aCurrent.Y() + nYVec ); - - aControlPointB = basegfx::B2DPoint(aControl2.X(), aControl2.Y()); - - aNewB2DPolygon.appendBezierSegment( - aControlPointA, - aControlPointB, - basegfx::B2DPoint(aCurrent.X(), aCurrent.Y())); - } - else - { - aNewB2DPolygon.append(basegfx::B2DPoint(aCurrent.X(), aCurrent.Y())); + aNewB2DPolygon.append(aArc); + rSrcPt++; + bIsXDirection = !bIsXDirection; + aStart = aEnd; } - - rSrcPt++; } + // else error in path syntax, do nothing } break; -- cgit