<div dir="ltr"><div><div>You can usually avoid working with angles entirely, by using vector math.<br><br></div>For instance it is very common to make the handle slope for P1 be equal to the slope between P0 and P2. This means the distance from P1 to the handle ends is some multiple of (P2-P0), all done with xy pairs and no angles or trig needed.<br><br></div>Making the two handles for a point equal-length and parallel is a requirement if you want the second derivative of the curve to be continuous. Though it is not clear whether this results in the most attractive curve.<br><br></div><div class="gmail_extra"><br><div class="gmail_quote">On Mon, Dec 14, 2015 at 1:37 AM, Bernhard Fischer <span dir="ltr"><<a href="mailto:bf@abenteuerland.at" target="_blank">bf@abenteuerland.at</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"><span class="">On Saturday 12 December 2015 22:46:17 Lawrence D'Oliveiro wrote:<br>
> On Sat, 12 Dec 2015 08:25:30 +0100, Bernhard Fischer wrote:<br>
> > I wrote an article about calculating control points for Bsplines<br>
> > (based on polygons) and a sample program.<br>
><br>
> Always interested to see how people use geometrical calculations<br>
> together with Cairo. Just some points:<br>
><br>
> I had to deal with the same issue of calculating half the<br>
> difference between two angles in this<br>
> <<a href="https://github.com/ldo/qahirah_examples/blob/master/pattern_dash" rel="noreferrer" target="_blank">https://github.com/ldo/qahirah_examples/blob/master/pattern_dash</a>><br>
> piece of example code (output here<br>
> <<a href="http://default-cube.deviantart.com/art/Pattern-Dash-576369003" rel="noreferrer" target="_blank">http://default-cube.deviantart.com/art/Pattern-Dash-576369003</a>>),<br>
> while avoiding discontinuities around 0°. The calculation I came up with<br>
> was basically<br>
><br>
>     ((angle2 - angle1 + π) mod 2π - π) ÷ 2<br>
<br>
</span>That's why I do not divide the angles like this (I did that in an earlier<br>
variant). I calculate the end points of an intermediate line and then<br>
determine the angle of this. It eliminates the need for handling of special<br>
cases.<br>
<span class=""><br>
> Also, C, like most IEEE-754-compliant languages, has a built-in function<br>
> for calculating Euclidean distance. Instead of<br>
><br>
>     s = sqrt(pow(a, 2) + pow(b, 2))<br>
><br>
> why not just write<br>
><br>
>     s = hypot(a, b)<br>
<br>
</span>Thanks, I know that function but forgot about it. I'll change it.<br>
<span class=""><br>
> Another idea is, when averaging the angles at the corners, why not<br>
> weight the contributions from the adjacent line segments according to<br>
> their relative lengths? So shorter segments will produce sharper curves.<br>
<br>
</span>Well, there is no perfect solution. I implemented some variants and every<br>
variant creates a specific appearance of the curve. I chose the one as<br>
implemented in my demo program to fit best to my needs.<br>
I use this variant in my chart renderer Smrender. The following example have<br>
the coastline and the depth contour lines rendered with this curve style:<br>
<br>
<a href="http://www.abenteuerland.at/download/smrender/samples/murter.pdf" rel="noreferrer" target="_blank">http://www.abenteuerland.at/download/smrender/samples/murter.pdf</a><br>
<br>
<br>
Best regards,<br>
Bernhard<br>
<div class="HOEnZb"><div class="h5"><br>
--<br>
cairo mailing list<br>
<a href="mailto:cairo@cairographics.org">cairo@cairographics.org</a><br>
<a href="http://lists.cairographics.org/mailman/listinfo/cairo" rel="noreferrer" target="_blank">http://lists.cairographics.org/mailman/listinfo/cairo</a></div></div></blockquote></div><br></div>