A library for efficiently querying points along cubic Bezier curves.
This implements two features of the native JS SVGPathElement
API: path.getTotalLength()
, and path.getPointAtLength()
, but a lot faster than the native API.
Bezier curves aren't easy to sample by length. Normally, you sample in "Bezier space" for each segment via a number between 0 and 1, but evenly spaced values in Bezier space won't produce evenly spaced points in Cartesian coordinates. Finding a point at a specific distance normally means doing an integration of some kind, which can be slow if you are getting a lot of points. I need it to be fast, though, which is where this library comes in!
Add the library in a script tag:
<script src="https://cdn.jsdelivr.net/npm/@davepagurek/bezier-path@0.0.2"></script>
Or on OpenProcessing, add the CDN link as a library:
https://cdn.jsdelivr.net/npm/@davepagurek/bezier-path@0.0.2
You can call create()
with an array of control points. Each control point in an object including a pt
property, which is the coordinate the curve will pass through. It may optionally include left
and right
properties, which control the handles going into and out of the point, respectively.
const myPath = BezierPath.create([
{
pt: { x: 10, y: 10 },
right: { x: 50, y: 10 }
},
{
left: { x: 20, y: 20 },
pt: { x: 20, y: 30 },
right: { x: 20, y: 40 }
},
{
left: { x: 50, y: 50 },
pt: { x: 10, y: 50 },
},
]) |
https://editor.p5js.org/davepagurek/sketches/dg2o-sLeK
If you want a fully smooth curve, then the line between a control point's left
and right
coordinates must pass through its pt
coordinate. A way to ensure this is to create just one side, e.g. right
, and then mirror it for the other side. Here's an example resembling Catmull-Rom interpolation, using p5's p5.Vector
class for the points:
const pts = []
for (let i = 0; i < 5; i++) {
pts.push({
pt: createVector(random(width), random(height))
})
}
// Smooth tangents
for (let i = 0; i < 5; i++) {
const prev = pts[i-1]
const curr = pts[i]
const next = pts[i+1]
if (prev && next) {
// Change the scaling of the tangent to adjust the curve tightness
const tangent = next.pt.copy().sub(prev.pt).div(4)
curr.right = curr.pt.copy().add(tangent)
curr.left = curr.pt.copy().sub(tangent)
}
}
const myPath = BezierPath.create(pts) |
You can also import a path from an SVG embedded in the page:
const myPath = BezierPath.createFromElement(document.querySelector('path'))
You can draw a path as a polyline by querying points via getPointAtLength()
:
beginShape()
for (let i = 0; i <= 60; i++) {
const pt = myPath.getPointAtLength(
map(i, 0, 60, 0, myPath.getTotalLength())
)
vertex(pt.x, pt.y)
}
endShape()
Typically, you'll want to vary the number of sample points based on the length of the curve, e.g.:
beginShape()
const sampleRate = 3 // One point every 3px
const numSamples = ceil(myPath.getTotalLength / sampleRate)
for (let i = 0; i < numSamples; i++) {
const pt = myPath.getPointAtLength(
map(i, 0, numSamples-1, 0, myPath.getTotalLength())
)
vertex(pt.x, pt.y)
}
endShape()
As long as you have the same number of sample points per path, you can morph between two paths by lerping each sample point between the two:
const mix = map(cos(frameCount / 120 * TWO_PI), -1, 1, 0, 1)
const numSamples = ceil(max(path1.getTotalLength(), path2.getTotalLength()) / 3)
beginShape()
for (let i = 0; i < numSamples; i++) {
const pt1 = path1.getPointAtLength(
map(i, 0, numSamples-1, 0, path1.getTotalLength())
)
const pt2 = path2.getPointAtLength(
map(i, 0, numSamples-1, 0, path2.getTotalLength())
)
vertex(lerp(pt1.x, pt2.x, mix), lerp(pt1.y, pt2.y, mix))
}
endShape() |
https://editor.p5js.org/davepagurek/sketches/XvC5H7Zhi
A full path made of a number of segments.
-
getTotalLength(): number
- Returns the total length of the path
-
getPointAtLength(length: number, approximate: boolean = false): Point
- Returns a coordinate at the given length on the path
- Specifying
true
forapproximate
will use linear interpolation between internal sample points. This will be faster, but will result in a more segmented line.
-
getTangentAtLength(length: number, approximate: boolean = false): Point
- Returns the tangent vector at a given length on the path, normalized to have a length of 1
- Specifying
true
forapproximate
will use linear interpolation between tangents at internal sample points for faster execution. The result will still be normalized, so this actually will still look pretty good, if not perfectly accurate.
-
getAngleAtLength(length: number, approximate: boolean = false): number
- Like
getTangentAtLength
, but it will return the angle of the path in radians - This is equivalent to calling
Math.atan2(y, x)
on the tangent vector
- Like