2011-06-05 14:27:18 -04:00
|
|
|
/*
|
2013-01-28 21:03:27 -05:00
|
|
|
* Paper.js - The Swiss Army Knife of Vector Graphics Scripting.
|
2011-06-05 14:27:18 -04:00
|
|
|
* http://paperjs.org/
|
2011-06-30 06:01:51 -04:00
|
|
|
*
|
2014-01-03 19:47:16 -05:00
|
|
|
* Copyright (c) 2011 - 2014, Juerg Lehni & Jonathan Puckey
|
|
|
|
* http://scratchdisk.com/ & http://jonathanpuckey.com/
|
2011-06-30 06:01:51 -04:00
|
|
|
*
|
2011-07-01 06:17:45 -04:00
|
|
|
* Distributed under the MIT license. See LICENSE file for details.
|
|
|
|
*
|
2011-06-05 14:27:18 -04:00
|
|
|
* All rights reserved.
|
|
|
|
*/
|
|
|
|
|
|
|
|
// An Algorithm for Automatically Fitting Digitized Curves
|
|
|
|
// by Philip J. Schneider
|
|
|
|
// from "Graphics Gems", Academic Press, 1990
|
2011-11-10 12:30:18 -05:00
|
|
|
// Modifications and optimisations of original algorithm by Juerg Lehni.
|
|
|
|
|
2013-12-29 09:44:26 -05:00
|
|
|
/**
|
|
|
|
* @name PathFitter
|
|
|
|
* @class
|
|
|
|
* @private
|
|
|
|
*/
|
2011-06-05 14:27:18 -04:00
|
|
|
var PathFitter = Base.extend({
|
2014-08-16 13:24:54 -04:00
|
|
|
initialize: function(path, error) {
|
2014-12-26 00:42:46 -05:00
|
|
|
var points = this.points = [],
|
|
|
|
segments = path._segments,
|
2014-08-16 13:24:54 -04:00
|
|
|
prev;
|
|
|
|
// Copy over points from path and filter out adjacent duplicates.
|
|
|
|
for (var i = 0, l = segments.length; i < l; i++) {
|
|
|
|
var point = segments[i].point.clone();
|
|
|
|
if (!prev || !prev.equals(point)) {
|
2014-12-26 00:42:46 -05:00
|
|
|
points.push(point);
|
2014-08-16 13:24:54 -04:00
|
|
|
prev = point;
|
|
|
|
}
|
|
|
|
}
|
2014-12-15 08:56:25 -05:00
|
|
|
|
2014-12-26 00:42:46 -05:00
|
|
|
// We need to duplicate the first and last segment when simplifying a
|
|
|
|
// closed path.
|
|
|
|
if (path._closed) {
|
|
|
|
this.closed = true;
|
|
|
|
points.unshift(points[points.length - 1]);
|
|
|
|
points.push(points[1]); // The point previously at index 0 is now 1.
|
2014-12-15 08:56:25 -05:00
|
|
|
}
|
|
|
|
|
2014-08-16 13:24:54 -04:00
|
|
|
this.error = error;
|
|
|
|
},
|
2011-06-05 14:27:18 -04:00
|
|
|
|
2014-08-16 13:24:54 -04:00
|
|
|
fit: function() {
|
|
|
|
var points = this.points,
|
2014-12-26 00:42:46 -05:00
|
|
|
length = points.length,
|
|
|
|
segments = this.segments = length > 0
|
|
|
|
? [new Segment(points[0])] : [];
|
2014-08-16 13:24:54 -04:00
|
|
|
if (length > 1)
|
|
|
|
this.fitCubic(0, length - 1,
|
|
|
|
// Left Tangent
|
|
|
|
points[1].subtract(points[0]).normalize(),
|
|
|
|
// Right Tangent
|
|
|
|
points[length - 2].subtract(points[length - 1]).normalize());
|
2014-12-15 08:56:25 -05:00
|
|
|
|
2014-12-26 00:42:46 -05:00
|
|
|
// Remove the duplicated segments for closed paths again.
|
|
|
|
if (this.closed) {
|
|
|
|
segments.shift();
|
|
|
|
segments.pop();
|
2014-12-15 08:56:25 -05:00
|
|
|
}
|
|
|
|
|
2014-12-26 00:42:46 -05:00
|
|
|
return segments;
|
2014-08-16 13:24:54 -04:00
|
|
|
},
|
2011-06-05 14:27:18 -04:00
|
|
|
|
2014-08-16 13:24:54 -04:00
|
|
|
// Fit a Bezier curve to a (sub)set of digitized points
|
|
|
|
fitCubic: function(first, last, tan1, tan2) {
|
|
|
|
// Use heuristic if region only has two points in it
|
|
|
|
if (last - first == 1) {
|
|
|
|
var pt1 = this.points[first],
|
|
|
|
pt2 = this.points[last],
|
|
|
|
dist = pt1.getDistance(pt2) / 3;
|
|
|
|
this.addCurve([pt1, pt1.add(tan1.normalize(dist)),
|
|
|
|
pt2.add(tan2.normalize(dist)), pt2]);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// Parameterize points, and attempt to fit curve
|
|
|
|
var uPrime = this.chordLengthParameterize(first, last),
|
|
|
|
maxError = Math.max(this.error, this.error * this.error),
|
2015-02-22 14:05:06 -05:00
|
|
|
split,
|
|
|
|
parametersInOrder = true;
|
2014-08-16 13:24:54 -04:00
|
|
|
// Try 4 iterations
|
|
|
|
for (var i = 0; i <= 4; i++) {
|
|
|
|
var curve = this.generateBezier(first, last, uPrime, tan1, tan2);
|
|
|
|
// Find max deviation of points to fitted curve
|
|
|
|
var max = this.findMaxError(first, last, curve, uPrime);
|
2015-02-22 14:05:06 -05:00
|
|
|
if (max.error < this.error && parametersInOrder) {
|
2014-08-16 13:24:54 -04:00
|
|
|
this.addCurve(curve);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
split = max.index;
|
|
|
|
// If error not too large, try reparameterization and iteration
|
|
|
|
if (max.error >= maxError)
|
|
|
|
break;
|
2015-02-22 14:05:06 -05:00
|
|
|
parametersInOrder = this.reparameterize(first, last, uPrime, curve);
|
2014-08-16 13:24:54 -04:00
|
|
|
maxError = max.error;
|
|
|
|
}
|
|
|
|
// Fitting failed -- split at max error point and fit recursively
|
|
|
|
var V1 = this.points[split - 1].subtract(this.points[split]),
|
|
|
|
V2 = this.points[split].subtract(this.points[split + 1]),
|
|
|
|
tanCenter = V1.add(V2).divide(2).normalize();
|
|
|
|
this.fitCubic(first, split, tan1, tanCenter);
|
|
|
|
this.fitCubic(split, last, tanCenter.negate(), tan2);
|
|
|
|
},
|
2011-06-05 14:27:18 -04:00
|
|
|
|
2014-08-16 13:24:54 -04:00
|
|
|
addCurve: function(curve) {
|
|
|
|
var prev = this.segments[this.segments.length - 1];
|
|
|
|
prev.setHandleOut(curve[1].subtract(curve[0]));
|
|
|
|
this.segments.push(
|
|
|
|
new Segment(curve[3], curve[2].subtract(curve[3])));
|
|
|
|
},
|
2011-06-05 14:27:18 -04:00
|
|
|
|
2014-08-16 13:24:54 -04:00
|
|
|
// Use least-squares method to find Bezier control points for region.
|
|
|
|
generateBezier: function(first, last, uPrime, tan1, tan2) {
|
|
|
|
var epsilon = /*#=*/Numerical.EPSILON,
|
|
|
|
pt1 = this.points[first],
|
|
|
|
pt2 = this.points[last],
|
|
|
|
// Create the C and X matrices
|
|
|
|
C = [[0, 0], [0, 0]],
|
|
|
|
X = [0, 0];
|
2011-06-05 14:27:18 -04:00
|
|
|
|
2014-08-16 13:24:54 -04:00
|
|
|
for (var i = 0, l = last - first + 1; i < l; i++) {
|
|
|
|
var u = uPrime[i],
|
|
|
|
t = 1 - u,
|
|
|
|
b = 3 * u * t,
|
|
|
|
b0 = t * t * t,
|
|
|
|
b1 = b * t,
|
|
|
|
b2 = b * u,
|
|
|
|
b3 = u * u * u,
|
|
|
|
a1 = tan1.normalize(b1),
|
|
|
|
a2 = tan2.normalize(b2),
|
|
|
|
tmp = this.points[first + i]
|
|
|
|
.subtract(pt1.multiply(b0 + b1))
|
|
|
|
.subtract(pt2.multiply(b2 + b3));
|
|
|
|
C[0][0] += a1.dot(a1);
|
|
|
|
C[0][1] += a1.dot(a2);
|
|
|
|
// C[1][0] += a1.dot(a2);
|
|
|
|
C[1][0] = C[0][1];
|
|
|
|
C[1][1] += a2.dot(a2);
|
|
|
|
X[0] += a1.dot(tmp);
|
|
|
|
X[1] += a2.dot(tmp);
|
|
|
|
}
|
2011-06-05 14:27:18 -04:00
|
|
|
|
2014-08-16 13:24:54 -04:00
|
|
|
// Compute the determinants of C and X
|
|
|
|
var detC0C1 = C[0][0] * C[1][1] - C[1][0] * C[0][1],
|
|
|
|
alpha1, alpha2;
|
|
|
|
if (Math.abs(detC0C1) > epsilon) {
|
|
|
|
// Kramer's rule
|
|
|
|
var detC0X = C[0][0] * X[1] - C[1][0] * X[0],
|
|
|
|
detXC1 = X[0] * C[1][1] - X[1] * C[0][1];
|
|
|
|
// Derive alpha values
|
|
|
|
alpha1 = detXC1 / detC0C1;
|
|
|
|
alpha2 = detC0X / detC0C1;
|
|
|
|
} else {
|
|
|
|
// Matrix is under-determined, try assuming alpha1 == alpha2
|
|
|
|
var c0 = C[0][0] + C[0][1],
|
|
|
|
c1 = C[1][0] + C[1][1];
|
|
|
|
if (Math.abs(c0) > epsilon) {
|
|
|
|
alpha1 = alpha2 = X[0] / c0;
|
|
|
|
} else if (Math.abs(c1) > epsilon) {
|
|
|
|
alpha1 = alpha2 = X[1] / c1;
|
|
|
|
} else {
|
|
|
|
// Handle below
|
|
|
|
alpha1 = alpha2 = 0;
|
|
|
|
}
|
|
|
|
}
|
2011-06-05 14:27:18 -04:00
|
|
|
|
2014-08-16 13:24:54 -04:00
|
|
|
// If alpha negative, use the Wu/Barsky heuristic (see text)
|
|
|
|
// (if alpha is 0, you get coincident control points that lead to
|
|
|
|
// divide by zero in any subsequent NewtonRaphsonRootFind() call.
|
2015-04-04 09:54:34 -04:00
|
|
|
var segLength = pt2.getDistance(pt1),
|
|
|
|
eps = epsilon * segLength,
|
|
|
|
handle1,
|
|
|
|
handle2;
|
|
|
|
if (alpha1 < eps || alpha2 < eps) {
|
2014-08-16 13:24:54 -04:00
|
|
|
// fall back on standard (probably inaccurate) formula,
|
|
|
|
// and subdivide further if needed.
|
|
|
|
alpha1 = alpha2 = segLength / 3;
|
2015-04-04 09:54:34 -04:00
|
|
|
} else {
|
|
|
|
// Check if the found control points are in the right order when
|
|
|
|
// projected onto the line through pt1 and pt2.
|
|
|
|
var line = pt2.subtract(pt1);
|
2015-02-22 14:05:06 -05:00
|
|
|
// Control points 1 and 2 are positioned an alpha distance out
|
|
|
|
// on the tangent vectors, left and right, respectively
|
2015-04-04 09:54:34 -04:00
|
|
|
handle1 = tan1.normalize(alpha1);
|
|
|
|
handle2 = tan2.normalize(alpha2);
|
|
|
|
if (handle1.dot(line) - handle2.dot(line) > segLength * segLength) {
|
|
|
|
// Fall back to the Wu/Barsky heuristic above.
|
|
|
|
alpha1 = alpha2 = segLength / 3;
|
|
|
|
handle1 = handle2 = null; // Force recalculation
|
|
|
|
}
|
2015-02-22 14:05:06 -05:00
|
|
|
}
|
|
|
|
|
2014-08-16 13:24:54 -04:00
|
|
|
// First and last control points of the Bezier curve are
|
|
|
|
// positioned exactly at the first and last data points
|
2015-04-04 09:54:34 -04:00
|
|
|
return [pt1, pt1.add(handle1 || tan1.normalize(alpha1)),
|
|
|
|
pt2.add(handle2 || tan2.normalize(alpha2)), pt2];
|
2014-08-16 13:24:54 -04:00
|
|
|
},
|
2011-06-05 14:27:18 -04:00
|
|
|
|
2014-08-16 13:24:54 -04:00
|
|
|
// Given set of points and their parameterization, try to find
|
|
|
|
// a better parameterization.
|
|
|
|
reparameterize: function(first, last, u, curve) {
|
|
|
|
for (var i = first; i <= last; i++) {
|
|
|
|
u[i - first] = this.findRoot(curve, this.points[i], u[i - first]);
|
|
|
|
}
|
2015-02-22 14:05:06 -05:00
|
|
|
// Detect if the new parameterization has reordered the points.
|
2015-04-04 09:54:34 -04:00
|
|
|
// In that case, we would fit the points of the path in the wrong order.
|
|
|
|
for (var i = 1, l = u.length; i < l; i++) {
|
|
|
|
if (u[i] <= u[i - 1])
|
|
|
|
return false;
|
2015-02-22 14:05:06 -05:00
|
|
|
}
|
|
|
|
return true;
|
2014-08-16 13:24:54 -04:00
|
|
|
},
|
2011-06-05 14:27:18 -04:00
|
|
|
|
2014-08-16 13:24:54 -04:00
|
|
|
// Use Newton-Raphson iteration to find better root.
|
|
|
|
findRoot: function(curve, point, u) {
|
|
|
|
var curve1 = [],
|
|
|
|
curve2 = [];
|
|
|
|
// Generate control vertices for Q'
|
|
|
|
for (var i = 0; i <= 2; i++) {
|
|
|
|
curve1[i] = curve[i + 1].subtract(curve[i]).multiply(3);
|
|
|
|
}
|
|
|
|
// Generate control vertices for Q''
|
|
|
|
for (var i = 0; i <= 1; i++) {
|
|
|
|
curve2[i] = curve1[i + 1].subtract(curve1[i]).multiply(2);
|
|
|
|
}
|
|
|
|
// Compute Q(u), Q'(u) and Q''(u)
|
|
|
|
var pt = this.evaluate(3, curve, u),
|
|
|
|
pt1 = this.evaluate(2, curve1, u),
|
|
|
|
pt2 = this.evaluate(1, curve2, u),
|
|
|
|
diff = pt.subtract(point),
|
|
|
|
df = pt1.dot(pt1) + diff.dot(pt2);
|
|
|
|
// Compute f(u) / f'(u)
|
|
|
|
if (Math.abs(df) < /*#=*/Numerical.TOLERANCE)
|
|
|
|
return u;
|
|
|
|
// u = u - f(u) / f'(u)
|
|
|
|
return u - diff.dot(pt1) / df;
|
|
|
|
},
|
2011-06-05 14:27:18 -04:00
|
|
|
|
2014-08-16 13:24:54 -04:00
|
|
|
// Evaluate a bezier curve at a particular parameter value
|
|
|
|
evaluate: function(degree, curve, t) {
|
|
|
|
// Copy array
|
|
|
|
var tmp = curve.slice();
|
|
|
|
// Triangle computation
|
|
|
|
for (var i = 1; i <= degree; i++) {
|
|
|
|
for (var j = 0; j <= degree - i; j++) {
|
|
|
|
tmp[j] = tmp[j].multiply(1 - t).add(tmp[j + 1].multiply(t));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return tmp[0];
|
|
|
|
},
|
2011-06-05 14:27:18 -04:00
|
|
|
|
2014-08-16 13:24:54 -04:00
|
|
|
// Assign parameter values to digitized points
|
|
|
|
// using relative distances between points.
|
|
|
|
chordLengthParameterize: function(first, last) {
|
|
|
|
var u = [0];
|
|
|
|
for (var i = first + 1; i <= last; i++) {
|
|
|
|
u[i - first] = u[i - first - 1]
|
|
|
|
+ this.points[i].getDistance(this.points[i - 1]);
|
|
|
|
}
|
|
|
|
for (var i = 1, m = last - first; i <= m; i++) {
|
|
|
|
u[i] /= u[m];
|
|
|
|
}
|
|
|
|
return u;
|
|
|
|
},
|
2011-06-05 14:27:18 -04:00
|
|
|
|
2014-08-16 13:24:54 -04:00
|
|
|
// Find the maximum squared distance of digitized points to fitted curve.
|
|
|
|
findMaxError: function(first, last, curve, u) {
|
|
|
|
var index = Math.floor((last - first + 1) / 2),
|
|
|
|
maxDist = 0;
|
|
|
|
for (var i = first + 1; i < last; i++) {
|
|
|
|
var P = this.evaluate(3, curve, u[i - first]);
|
|
|
|
var v = P.subtract(this.points[i]);
|
|
|
|
var dist = v.x * v.x + v.y * v.y; // squared
|
|
|
|
if (dist >= maxDist) {
|
|
|
|
maxDist = dist;
|
|
|
|
index = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return {
|
|
|
|
error: maxDist,
|
|
|
|
index: index
|
|
|
|
};
|
|
|
|
}
|
2011-06-05 14:27:18 -04:00
|
|
|
});
|