On 23 Apr 2014, at 2:31 , Igor Stasenko <siguc...@gmail.com> wrote: > > > > On 23 April 2014 13:17, Henrik Johansen <henrik.s.johan...@veloxit.no> wrote: > > On 22 Apr 2014, at 2:23 , Igor Stasenko <siguc...@gmail.com> wrote: > > > as for why there's 4 arc segments instead of one, its because > > of bad approximation, when drawing more that 90 degree arcs. > > > > also, in athens, arc segment is defined with following inputs: > > - end point of previous segment (implicit) > > - angle > > - direction (clockwise/counterclockwise) > > - end point > > > > the radius, therefore calculated automatically, because with given set of > > parameters there's only one way to draw an arc connecting given points. > > > > Now if you put angle of 360 degrees, you cannot draw arc without specifying > > radius, > > because your end points will coincide, which means there's infinite number > > of ways to draw full circle passing through a single point, with any radius. > > > > cairo using different inputs for specifying arc segments.. > > - center, radius, start angle, end angle > > > > the problem with such parametrization is that it is completely separate > > from rest of commands (line/move/bezier etc).. and you will be very lucky > > if your arc will be connected with rest of your path.. because arc's > > starting point depends on start angle, instead of last point of previous > > path segment. > > > > this was the main reason to use more appropriate parametrization to get rid > > of inconsistency.. while losing ability to draw full circle with single > > command.. > > AFAICT, still doesn’t explain how to draw an ellipse with constant stroke > path width, which was the original question :) > > > Right, what is missing is elliptical arc segment type. And there's no direct > support for it in Cairo.. so it can be only approximated by other segment > types, like lines or bezier curves. > There's a work started on calculating path geometry using approximation with > line segments.. it can be used to represent any kind of curves defined > parametrically. > But it is not yet plugged into the API. > > > One way is to not use a transformed circle path altogether, but draw the > actual ellipsis path using cubic beziers: > http://www.charlespetzold.com/blog/2012/12/Bezier-Circles-and-Bezier-Ellipses.html > > ellipsisOfExtent := [:builder :anExtent | | halfX halfY | > halfX := anExtent x / 2. > halfY := anExtent y / 2. > “We expect relative builder, and start the ellipsis at > anExtent x / 2 @ 0" > builder > curveVia: 0@(halfY negated * 0.55) and: (0.45 * > halfX)@halfY negated to: halfX@ halfY negated; > curveVia: halfX* 0.55 @ 0 and: halfX@ (0.45 * halfY) > to: halfX @ halfY; > curveVia: 0 @ (halfY * 0.55 ) and: (0.45 * halfX > negated @ halfY) to: halfX negated @ halfY; > curveVia: (halfX negated * 0.55) @ 0 and: halfX > negated @ (halfY negated * 0.45) to: halfX negated @ halfY negated; > close]. > > AthensSceneView new > scene: [ :can | > | path | > > path := can > createPath: [ :builder | > builder moveTo: 10@60. > ellipsisOfExtent value: builder > value: 200@100 ]. > (can > > setStrokePaint: Color red) > width: 8 asFloat. > can drawShape: path ] ; > openInWindow > > > quite nice approximation. What is an error measure comparing to true ellipse?
From the abstract of the paper cited in the linked site: We provide a surprisingly simple cubic Bézier curve which gives a very accurate approximation to a segment of a circle. Joining the Bézier segments we obtain an approximation to the circle with continuous tangent and curvature. For 45° segments the error is approximately 2·10-6, and in general the approximation is sixth order accurate. Considering the code simply stretches a bezier circle’s control points, I’d say quite. In fact, I’d be surprised if the arc primitive in cairo is implemented using a different method. (Of course, even less error using the actual value of the formula for 90 degrees, instead of approximate values .55 / 1- 0.55.) Cheers, Henry
signature.asc
Description: Message signed with OpenPGP using GPGMail