How to reduce the number of points in bezier curve paths - javascript

I have a large svg path which has quadratice bezier curves.
The path data is used for drawing maps.
How to reduce the number of points of the bezier curve without distorting the overall shape?

You don't say whether you want to do this offline (pre-prepared paths) or online (on the fly). If offline is fine, use a tool like Inkscape.
If you want to calculate the simplified curve yourself, then the typical algorithm used to do this is also the same one that has been used for drawing bezier curves. The algorithm is called "flattening".
Basically the idea is to convert the bezier curves to a series of straight line segments. Because you don't want the the flatness of the line segments to be visible, you have to take into account the scale of the drawing and how curvy the bezier is. If the bezier is very curvy you have to use more line segments than if it is fairly straight.
What you typically do is divide each bezier into two using De Casteljau's algorithm. Then look at each of the two half bezier curves. If a bezier is straight enough to meet a flatness limit you decide on, then stop dividing. Otherwise, divide in half and try again.
At the end of the process you should get a polyline that is indistinguishable from the bezier version. Or if you use a "flatness test" that is a bit courser than that, you will get a rougher approximation of the shape. In your case a map.
If you google bezier flattening you can find a number of papers on the technique. And a few pages which describe how to do it in a more friendly accessible way. For example this one, which is about generating offset curves, but starts out by describing how to flatten a curve:
https://seant23.wordpress.com/2010/11/12/offset-bezier-curves/

Related

Draw curve -> Get equation

A part of the project I am working on requires an interface where a user draws a curve, and I want to output an approximate mathematical equation of that curve to perform a variety of tests on it.
I have thought of 2 approaches so far and was wondering if they are feasible or if there exists a better way of approaching the problem that I am missing/ cool mathematical trick that allows me to pull this off.
Approach 1:
Instead of having users draw a curve, give them the option of inserting bezier curve points and tweaking them to make their curve. Since bezier curves have a parametric equation that describes them, I could directly get the exact equation.
Cons:
-It's more cumbersome for users to tweak and make a bezier than simply draw a curve
Approach 2:
Get the curve drawn and extract 'n' points. (I don't think this should be hard to do).
Somehow go from these 'n' points to an equation of a curve passing through them. Naturally 'n' would be pretty large, say 100.
Is there a neat way to get an equation for the "simplest continuous function" passing through 'n' points?
Approach 1 -- insert Bezier points and tweak control points -- is used in a lot of vector drawing programs, especially CAD-type applications where precision is important and points are laid out on a grid, etc. It is a slower way to draw, but it's used in these situations, because it's really difficult to draw a freehand curve carefully.
Approach 2 -- just draw and get an smooth approximation -- is also pretty popular when precision isn't required. I'm not sure what methods are usually used for this, or even if there is a "usually", but I don't think it's particularly difficult to do a pretty good job. I would:
Connect all the points with line segments in the order in which they were drawn.
Apply the Ramer-Douglas-Peucker algorithm to reduce the number of points.
Interpolate the points with cubic spline interpolation. Be careful looking this up-there are many flavors of cubic spline interpolation and their names overlap. I would specifically prefer the versions that preserve 2nd derivative continuity.

D3.js: Animating a bezier curve

I'm using the following code to generate a bezier curve for a workflow design:
group.append("svg:path")
.attr("d","m " +(300*k) +",0 q "+(z*150)+",-300,"+z*300+", 0")
.attr("fill","blue")
.style("stroke", "blue").attr("stroke-width",2)
.attr("marker-end", "url(#arrow)").attr("fill","none").append("text");
I want to achieve this animation in the curve. Any suggestion might help.
Here is an example of animated Bezier curves. You can inspect the code that he is using on that page, but, basically the trick is as they say...
For a second-order or quadratic Bézier curve, first we find two intermediate points that are t along the lines between the three control points. Then we perform the same interpolation step again and find another point that is t along the line between those two intermediate points. Plotting this last point yields a quadratic Bézier curve. The same steps can be repeated for higher orders.

Intersection of two Bézier curves (or two curves and a line): code?

Checking if two cubic Bézier curves intersect gives a link to http://cagd.cs.byu.edu/~557/text/ch7.pdf .. sounds readable on first pass.. but it is not code.
I am wondering if someone has actually implemented this algorithm in any common programming language. I would be interested in some Javascript code (other languages OK) that can implement the algorithm using two cubic Bézier curves, or a Bézier curve and a straight line.
For a cubic Bézier curves and a straight line it probably easiest to use section 7.3 Intersection of a Parametric Curve and an Implicit Curve. You can write the straight line as a x+b y+c=0. If you Bézier curves in given by cubic p(t) and q(t) for x and y coords you can substitute those into the line equation. This gives a cubic in t which you can solve by your favourite root finding algorithm.
The similar question Checking if two cubic Bézier curves intersect has some good answers. In particular the first answer mentions the asymptote library which has code for all these. You can see the relevant source code at http://sourceforge.net/p/asymptote/code/HEAD/tree/trunk/asymptote/path.cc including code for finding roots of a cubic.

Random enclosing path with dots on path

I would like to generate a SVG/canvas using JavaScript that contains an enclosing path (a line that starts and ends at the same place) with random number of points that should be equidistant from each other (on the path, not in terms of space).
What library can be used to achieve this effect and if possible, can an example be provided?
Angela
For SVG you can use the Snap library and in particular use the getPointAtLength method to get equidistant points along a path.
http://snapsvg.io/docs/#Snap.path.getPointAtLength
In html Canvas a path can be any combination of line segments and curves (quadratic & cubic Bezier curves). Therefore to plot equidistant points on a path you must mathematically interpolate along each line segment and curve using the appropriate formulae.
For line segments this usually involves calculating the length of the line segment and then "lerping" (Google!) using your specified distance.
For curves this usually involves using the equations for the curve and plotting numerous points along that line. Then connect those points with line segments and again "lerping" along those segments at your specified distance.

Plotting mathematical functions without rendering artefacts

I don't think there's a good answer to this, but I'd like to find out if there's a better way to do this.
I need to plot a mathematical function, which is nearly flat at one end of the display, and nearly vertical at the other end. The bottom left quadrant of a circle would be a good model. I can auto-generate as many points as required.
The problem is, I can't do this without all sorts of artefacts.
I haven't tried Bezier fitting; I don't think this would be even close. My understanding is that Bezier is for one-off manually-constructed pretty graphics, and not for real curve-fitting.
That leaves polylines. There are only 2 things I can do with polylines - I can select the line length (in other words, the number of points I auto-generate), and I can disable anti-aliasing (setAttributeNS(null, "shape-rendering", "crisp-edges").
If I generate lots of points, then I get jaggies everywhere, and the result is unusable. It can also look very much like it's oscillating, which makes it appear that I've incorrectly calculated the function. The anti-aliasing doesn't make any difference, since it doesn't operate across point boundaries.
The only solution I've got is to draw fewer points, so that it's obvious that I'm drawing segments. It's no longer smooth, but at least there are no jaggies or oscillation. I draw this with the default anti-aliasing.
Any ideas?
Edit:
It seems like the only answer to this is actually Bezier curve fitting. You have to preprocess to find the parameters of the required segments, and then plot the results. Google comes up with a number of hits on curve fitting with Beziers.
You have the mathematical function, and can therefore generate as many points as you need.
I assume the problem is that because you do not know the output resolution (SVG is device independent) you do not know how many points to generate. Otherwise you could just create a polyline where each line is approximately 1 pixel long.
Fitting your mathematical function to a bezier curve is (probably) not going to get a perfect match - just like a circle cannot be matched perfectly by a cubic bezier curve. And I think the task of fitting your function to a bezier curve would not be trivial (I've never done this).
Could you rather output your mathematical function to a canvas element? Then you could write some javascript code to plot your mathematical function dependant on the output resolution. Similar to how a graphics system renders a Bezier curve.
Do you know how graphics systems render Bezier curves? They approximate the bezier curve with a polyline, and then measure the error difference between the polyline and the bezier curve. If the difference is greater than a certain tolerance - where the tolerance is determined by the output resolution - the bezier is subdivided and the process repeated for each bezier curve. When the difference between beziers and polylines is below the tolerance, the polylines are drawn. http://en.wikipedia.org/wiki/B%C3%A9zier_curve#Computer_graphics
I suppose you want to draw y=f(x) over a certain interval [a,b]
A classical solution is to take N points uniformly distributed over [a,b], to compute f over these points and draw lines (or polynoms).
It of course doesn't work in your case, since y is nearly vertical in certain area. But why don't you take more points in these areas (and less points where the function is nearly horizontal) ?
You can compute the derivative of your function (or approximate this derivative with (f(x+h)-f(x))/h and h small) and determine the step between two successive points with this derivative

Categories

Resources