mirror of
https://github.com/scratchfoundation/paper.js.git
synced 2025-01-23 07:49:48 -05:00
Implement from / to options for 'continuous' smooth().
This commit is contained in:
parent
71c7405d6b
commit
5f11345fc9
2 changed files with 61 additions and 39 deletions
|
@ -2066,23 +2066,41 @@ var Path = PathItem.extend(/** @lends Path# */{
|
||||||
smooth: function(options) {
|
smooth: function(options) {
|
||||||
// Helper method to pick the right from / to indices.
|
// Helper method to pick the right from / to indices.
|
||||||
// Supports numbers and segment objects.
|
// Supports numbers and segment objects.
|
||||||
|
// For numbers, the `to` index is exclusive, while for segments and
|
||||||
|
// curves, it is inclusive, handled by the `offset` parameter.
|
||||||
function getIndex(value, _default) {
|
function getIndex(value, _default) {
|
||||||
return value == null
|
var index = value == null
|
||||||
? _default
|
? _default
|
||||||
: typeof value === 'number'
|
: typeof value === 'number'
|
||||||
? value
|
? value
|
||||||
: value.getIndex
|
: value.getIndex
|
||||||
? value.getIndex()
|
? value.getIndex()
|
||||||
: _default;
|
: _default;
|
||||||
|
// Handle negative values based on whether a path is open or not:
|
||||||
|
// Closed paths are wrapped around the end (allowing values to be
|
||||||
|
// negative), while open paths stay within the open range.
|
||||||
|
return index < 0 && closed
|
||||||
|
? index % length
|
||||||
|
: (Math.min(index, length - 1) % length + length) % length;
|
||||||
}
|
}
|
||||||
|
|
||||||
var opts = options || {},
|
var opts = options || {},
|
||||||
type = opts.type,
|
type = opts.type,
|
||||||
segments = this._segments,
|
segments = this._segments,
|
||||||
length = segments.length,
|
length = segments.length,
|
||||||
|
range = opts.from !== undefined || opts.to !== undefined,
|
||||||
from = getIndex(opts.from, 0),
|
from = getIndex(opts.from, 0),
|
||||||
to = getIndex(opts.to, length - 1);
|
to = getIndex(opts.to, length - 1),
|
||||||
|
closed = this._closed;
|
||||||
|
if (from > to) {
|
||||||
|
if (closed) {
|
||||||
|
from -= length;
|
||||||
|
} else {
|
||||||
|
var tmp = from;
|
||||||
|
from = to;
|
||||||
|
to = tmp;
|
||||||
|
}
|
||||||
|
}
|
||||||
if (!type || type === 'continuous') {
|
if (!type || type === 'continuous') {
|
||||||
// Continuous smoothing approach based on work by Lubos Brieda,
|
// Continuous smoothing approach based on work by Lubos Brieda,
|
||||||
// Particle In Cell Consulting LLC, but further simplified by
|
// Particle In Cell Consulting LLC, but further simplified by
|
||||||
|
@ -2090,28 +2108,28 @@ var Path = PathItem.extend(/** @lends Path# */{
|
||||||
// to process x and y coordinates simultaneously. Also added
|
// to process x and y coordinates simultaneously. Also added
|
||||||
// handling of closed paths.
|
// handling of closed paths.
|
||||||
// https://www.particleincell.com/2012/bezier-splines/
|
// https://www.particleincell.com/2012/bezier-splines/
|
||||||
var closed = this._closed,
|
var min = Math.min,
|
||||||
n = length - 1,
|
amount = to - from + 1,
|
||||||
// Add overlapping ends for closed paths.
|
n = amount - 1;
|
||||||
overlap = 0;
|
// Overlap by up to 4 points on closed paths since a current segment
|
||||||
if (length <= 2)
|
// is affected by its 4 neighbors on both sides (?).
|
||||||
|
var loop = closed && !range,
|
||||||
|
padding = loop ? min(amount, 4) : 1,
|
||||||
|
paddingLeft = padding,
|
||||||
|
paddingRight = padding,
|
||||||
|
knots = [];
|
||||||
|
if (!closed) {
|
||||||
|
// If the path is open and a range is defined, try using a
|
||||||
|
// padding of 1 on either side.
|
||||||
|
paddingLeft = min(1, from);
|
||||||
|
paddingRight = min(1, length - to - 1);
|
||||||
|
}
|
||||||
|
// Set up the knots array now, taking the paddings into account.
|
||||||
|
n += paddingLeft + paddingRight;
|
||||||
|
if (n <= 1)
|
||||||
return;
|
return;
|
||||||
if (closed) {
|
for (var i = 0, j = from - paddingLeft; i <= n; i++, j++) {
|
||||||
// Overlap by up to 4 points since a current segment is affected
|
knots[i] = segments[(j < 0 ? j + length : j) % length]._point;
|
||||||
// by 4 neighbors.
|
|
||||||
overlap = Math.min(length, 4);
|
|
||||||
n += Math.min(length, overlap) * 2;
|
|
||||||
}
|
|
||||||
var knots = [];
|
|
||||||
for (var i = 0; i < length; i++)
|
|
||||||
knots[i + overlap] = segments[i]._point;
|
|
||||||
if (closed) {
|
|
||||||
// Add the last points again at the beginning, and the first
|
|
||||||
// ones at the end.
|
|
||||||
for (var i = 0; i < overlap; i++) {
|
|
||||||
knots[i] = knots[i + length];
|
|
||||||
knots[i + length + overlap] = knots[i + overlap];
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Right-hand side vectors, with left most segment added.
|
// Right-hand side vectors, with left most segment added.
|
||||||
|
@ -2148,7 +2166,6 @@ var Path = PathItem.extend(/** @lends Path# */{
|
||||||
|
|
||||||
var px = [],
|
var px = [],
|
||||||
py = [];
|
py = [];
|
||||||
|
|
||||||
px[n_1] = rx[n_1] / b[n_1];
|
px[n_1] = rx[n_1] / b[n_1];
|
||||||
py[n_1] = ry[n_1] / b[n_1];
|
py[n_1] = ry[n_1] / b[n_1];
|
||||||
for (var i = n - 2; i >= 0; i--) {
|
for (var i = n - 2; i >= 0; i--) {
|
||||||
|
@ -2159,21 +2176,23 @@ var Path = PathItem.extend(/** @lends Path# */{
|
||||||
py[n] = (3 * knots[n]._y - py[n_1]) / 2;
|
py[n] = (3 * knots[n]._y - py[n_1]) / 2;
|
||||||
|
|
||||||
// Now update the segments
|
// Now update the segments
|
||||||
n -= overlap;
|
for (var i = paddingLeft, max = n - paddingRight, j = from;
|
||||||
for (var i = overlap; i <= n; i++) {
|
i <= max; i++, j++) {
|
||||||
var segment = segments[i - overlap],
|
var segment = segments[j < 0 ? j + length : j],
|
||||||
pt = segment._point,
|
pt = segment._point,
|
||||||
hx = px[i] - pt._x,
|
hx = px[i] - pt._x,
|
||||||
hy = py[i] - pt._y;
|
hy = py[i] - pt._y;
|
||||||
if (closed || i < n)
|
if (loop || i < max)
|
||||||
segment.setHandleOut(hx, hy);
|
segment.setHandleOut(hx, hy);
|
||||||
if (closed || i > 0)
|
if (loop || i > paddingLeft)
|
||||||
segment.setHandleIn(-hx, -hy);
|
segment.setHandleIn(-hx, -hy);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// All other smoothing methods are handled directly on the segments:
|
// All other smoothing methods are handled directly on the segments:
|
||||||
for (var i = from; i <= to; i++)
|
for (var i = from; i <= to; i++) {
|
||||||
segments[i].smooth(opts);
|
segments[i < 0 ? i + length : i].smooth(opts,
|
||||||
|
i === from, i === to);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}),
|
}),
|
||||||
|
|
|
@ -389,7 +389,8 @@ var Segment = Base.extend(/** @lends Segment# */{
|
||||||
*
|
*
|
||||||
* @see PathItem#smooth(options)
|
* @see PathItem#smooth(options)
|
||||||
*/
|
*/
|
||||||
smooth: function(options) {
|
smooth: function(options, _first, _last) {
|
||||||
|
// _first = _last = false;
|
||||||
var opts = options || {},
|
var opts = options || {},
|
||||||
type = opts.type,
|
type = opts.type,
|
||||||
factor = opts.factor,
|
factor = opts.factor,
|
||||||
|
@ -409,13 +410,13 @@ var Segment = Base.extend(/** @lends Segment# */{
|
||||||
// 0.0: the standard, uniform Catmull-Rom spline
|
// 0.0: the standard, uniform Catmull-Rom spline
|
||||||
// 0.5: the centripetal Catmull-Rom spline, guaranteeing no self-
|
// 0.5: the centripetal Catmull-Rom spline, guaranteeing no self-
|
||||||
// intersections
|
// intersections
|
||||||
// 1.0: the chordal Catmull-Rom spline.
|
// 1.0: the chordal Catmull-Rom spline
|
||||||
var a = factor === undefined ? 0.5 : factor,
|
var a = factor === undefined ? 0.5 : factor,
|
||||||
d1_a = Math.pow(d1, a),
|
d1_a = Math.pow(d1, a),
|
||||||
d1_2a = d1_a * d1_a,
|
d1_2a = d1_a * d1_a,
|
||||||
d2_a = Math.pow(d2, a),
|
d2_a = Math.pow(d2, a),
|
||||||
d2_2a = d2_a * d2_a;
|
d2_2a = d2_a * d2_a;
|
||||||
if (prev) {
|
if (!_first && prev) {
|
||||||
var A = 2 * d2_2a + 3 * d2_a * d1_a + d1_2a,
|
var A = 2 * d2_2a + 3 * d2_a * d1_a + d1_2a,
|
||||||
N = 3 * d2_a * (d2_a + d1_a);
|
N = 3 * d2_a * (d2_a + d1_a);
|
||||||
this.setHandleIn(N !== 0
|
this.setHandleIn(N !== 0
|
||||||
|
@ -424,7 +425,7 @@ var Segment = Base.extend(/** @lends Segment# */{
|
||||||
(d2_2a * p0._y + A * p1._y - d1_2a * p2._y) / N - p1._y)
|
(d2_2a * p0._y + A * p1._y - d1_2a * p2._y) / N - p1._y)
|
||||||
: new Point());
|
: new Point());
|
||||||
}
|
}
|
||||||
if (next) {
|
if (!_last && next) {
|
||||||
var A = 2 * d1_2a + 3 * d1_a * d2_a + d2_2a,
|
var A = 2 * d1_2a + 3 * d1_a * d2_a + d2_2a,
|
||||||
N = 3 * d1_a * (d1_a + d2_a);
|
N = 3 * d1_a * (d1_a + d2_a);
|
||||||
this.setHandleOut(N !== 0
|
this.setHandleOut(N !== 0
|
||||||
|
@ -442,7 +443,9 @@ var Segment = Base.extend(/** @lends Segment# */{
|
||||||
var vector = p0.subtract(p2),
|
var vector = p0.subtract(p2),
|
||||||
t = factor === undefined ? 0.4 : factor,
|
t = factor === undefined ? 0.4 : factor,
|
||||||
k = t * d1 / (d1 + d2);
|
k = t * d1 / (d1 + d2);
|
||||||
|
if (!_first)
|
||||||
this.setHandleIn(vector.multiply(k));
|
this.setHandleIn(vector.multiply(k));
|
||||||
|
if (!_last)
|
||||||
this.setHandleOut(vector.multiply(k - t));
|
this.setHandleOut(vector.multiply(k - t));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
|
Loading…
Reference in a new issue