mirror of
https://github.com/scratchfoundation/paper.js.git
synced 2025-01-19 14:10:14 -05:00
Merge remote-tracking branch 'origin/boolean-fix' into develop
This commit is contained in:
commit
2cf637d13b
33 changed files with 2293 additions and 1470 deletions
|
@ -92,7 +92,7 @@
|
||||||
text.content = 'ring.' + operation + '(square)';
|
text.content = 'ring.' + operation + '(square)';
|
||||||
}
|
}
|
||||||
result.selected = true;
|
result.selected = true;
|
||||||
result.fillColor = colors[curIndex % operations.length];
|
result.fillColor = colors[curIndex % colors.length];
|
||||||
result.moveBelow(text);
|
result.moveBelow(text);
|
||||||
|
|
||||||
// If the result is a group, color each of its children differently:
|
// If the result is a group, color each of its children differently:
|
||||||
|
|
|
@ -269,7 +269,7 @@
|
||||||
// // annotatePath(pathB)
|
// // annotatePath(pathB)
|
||||||
// // pathB.translate([ 300, 0 ]);
|
// // pathB.translate([ 300, 0 ]);
|
||||||
// // pathB.segments.filter(function(a) { return a._ixPair; }).map(
|
// // pathB.segments.filter(function(a) { return a._ixPair; }).map(
|
||||||
// // function(a) { a._ixPair.getIntersection()._segment.selected = true; });
|
// // function(a) { a._ixPair.intersection._segment.selected = true; });
|
||||||
|
|
||||||
// console.time('unite');
|
// console.time('unite');
|
||||||
// var nup = unite(pathA, pathB);
|
// var nup = unite(pathA, pathB);
|
||||||
|
@ -360,14 +360,12 @@
|
||||||
|
|
||||||
function disperse(path, distance) {
|
function disperse(path, distance) {
|
||||||
distance = distance || 10;
|
distance = distance || 10;
|
||||||
if (! path instanceof CompoundPath || ! path instanceof Group) { return; }
|
|
||||||
var center = path.bounds.center;
|
var center = path.bounds.center;
|
||||||
var children = path.children, i ,len;
|
var children = path.children, i ,len;
|
||||||
for (var i = 0, len = children.length; i < len; i++) {
|
for (var i = 0, len = children && children.length; i < len; i++) {
|
||||||
var cCenter = children[i].bounds.center;
|
var cCenter = children[i].bounds.center;
|
||||||
var vec = cCenter.subtract(center);
|
var vec = cCenter.subtract(center);
|
||||||
vec = (vec.isClose([0,0], 0.5))? vec : vec.normalize(distance);
|
children[i].translate(vec.length < 0.5 ? vec : vec.normalize(distance));
|
||||||
children[i].translate(vec);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -35,7 +35,7 @@
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"gulp": "^3.9.0",
|
"gulp": "^3.9.0",
|
||||||
"gulp-qunit": "^1.2.1",
|
"gulp-qunit": "^1.2.1",
|
||||||
"prepro": "~0.8.3",
|
"prepro": "~0.9.0",
|
||||||
"qunitjs": "~1.15.0",
|
"qunitjs": "~1.15.0",
|
||||||
"uglify-js": "~2.4.24"
|
"uglify-js": "~2.4.24"
|
||||||
},
|
},
|
||||||
|
|
|
@ -48,30 +48,30 @@ var Line = Base.extend(/** @lends Line# */{
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The starting point of the line
|
* The starting point of the line.
|
||||||
*
|
*
|
||||||
* @name Line#point
|
|
||||||
* @type Point
|
* @type Point
|
||||||
|
* @bean
|
||||||
*/
|
*/
|
||||||
getPoint: function() {
|
getPoint: function() {
|
||||||
return new Point(this._px, this._py);
|
return new Point(this._px, this._py);
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The vector of the line
|
* The direction of the line as a vector.
|
||||||
*
|
*
|
||||||
* @name Line#vector
|
|
||||||
* @type Point
|
* @type Point
|
||||||
|
* @bean
|
||||||
*/
|
*/
|
||||||
getVector: function() {
|
getVector: function() {
|
||||||
return new Point(this._vx, this._vy);
|
return new Point(this._vx, this._vy);
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The length of the line
|
* The length of the line.
|
||||||
*
|
*
|
||||||
* @name Line#length
|
|
||||||
* @type Number
|
* @type Number
|
||||||
|
* @bean
|
||||||
*/
|
*/
|
||||||
getLength: function() {
|
getLength: function() {
|
||||||
return this.getVector().getLength();
|
return this.getVector().getLength();
|
||||||
|
@ -113,35 +113,47 @@ var Line = Base.extend(/** @lends Line# */{
|
||||||
},
|
},
|
||||||
|
|
||||||
isCollinear: function(line) {
|
isCollinear: function(line) {
|
||||||
// TODO: Tests showed that 1e-10 might work well here, but we want to
|
return Point.isCollinear(this._vx, this._vy, line._vx, line._vy);
|
||||||
// keep it in sync with Point#isCollinear()
|
},
|
||||||
return this._vx * line._vy - this._vy * line._vx
|
|
||||||
< /*#=*/Numerical.TOLERANCE;
|
isOrthogonal: function(line) {
|
||||||
|
return Point.isOrthogonal(this._vx, this._vy, line._vx, line._vy);
|
||||||
},
|
},
|
||||||
|
|
||||||
statics: /** @lends Line */{
|
statics: /** @lends Line */{
|
||||||
intersect: function(apx, apy, avx, avy, bpx, bpy, bvx, bvy, asVector,
|
intersect: function(p1x, p1y, v1x, v1y, p2x, p2y, v2x, v2y, asVector,
|
||||||
isInfinite) {
|
isInfinite) {
|
||||||
// Convert 2nd points to vectors if they are not specified as such.
|
// Convert 2nd points to vectors if they are not specified as such.
|
||||||
if (!asVector) {
|
if (!asVector) {
|
||||||
avx -= apx;
|
v1x -= p1x;
|
||||||
avy -= apy;
|
v1y -= p1y;
|
||||||
bvx -= bpx;
|
v2x -= p2x;
|
||||||
bvy -= bpy;
|
v2y -= p2y;
|
||||||
}
|
}
|
||||||
var cross = avx * bvy - avy * bvx;
|
var cross = v1x * v2y - v1y * v2x;
|
||||||
// Avoid divisions by 0, and errors when getting too close to 0
|
// Avoid divisions by 0, and errors when getting too close to 0
|
||||||
if (!Numerical.isZero(cross)) {
|
if (!Numerical.isZero(cross)) {
|
||||||
var dx = apx - bpx,
|
var dx = p1x - p2x,
|
||||||
dy = apy - bpy,
|
dy = p1y - p2y,
|
||||||
ta = (bvx * dy - bvy * dx) / cross,
|
u1 = (v2x * dy - v2y * dx) / cross,
|
||||||
tb = (avx * dy - avy * dx) / cross;
|
u2 = (v1x * dy - v1y * dx) / cross,
|
||||||
// Check the ranges of t parameters if the line is not allowed
|
// Check the ranges of the u parameters if the line is not
|
||||||
// to extend beyond the definition points.
|
// allowed to extend beyond the definition points, but
|
||||||
if (isInfinite || 0 <= ta && ta <= 1 && 0 <= tb && tb <= 1)
|
// compare with EPSILON tolerance over the [0, 1] bounds.
|
||||||
|
epsilon = /*#=*/Numerical.EPSILON,
|
||||||
|
uMin = -epsilon,
|
||||||
|
uMax = 1 + epsilon;
|
||||||
|
if (isInfinite
|
||||||
|
|| uMin < u1 && u1 < uMax && uMin < u2 && u2 < uMax) {
|
||||||
|
if (!isInfinite) {
|
||||||
|
// Address the tolerance at the bounds by clipping to
|
||||||
|
// the actual range.
|
||||||
|
u1 = u1 <= 0 ? 0 : u1 >= 1 ? 1 : u1;
|
||||||
|
}
|
||||||
return new Point(
|
return new Point(
|
||||||
apx + ta * avx,
|
p1x + u1 * v1x,
|
||||||
apy + ta * avy);
|
p1y + u1 * v1y);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -157,9 +169,7 @@ var Line = Base.extend(/** @lends Line# */{
|
||||||
ccw = v2x * vx + v2y * vy; // ccw = v2.dot(v1);
|
ccw = v2x * vx + v2y * vy; // ccw = v2.dot(v1);
|
||||||
if (ccw > 0) {
|
if (ccw > 0) {
|
||||||
// ccw = v2.subtract(v1).dot(v1);
|
// ccw = v2.subtract(v1).dot(v1);
|
||||||
v2x -= vx;
|
ccw = (v2x - vx) * vx + (v2y - vy) * vy;
|
||||||
v2y -= vy;
|
|
||||||
ccw = v2x * vx + v2y * vy;
|
|
||||||
if (ccw < 0)
|
if (ccw < 0)
|
||||||
ccw = 0;
|
ccw = 0;
|
||||||
}
|
}
|
||||||
|
@ -172,11 +182,13 @@ var Line = Base.extend(/** @lends Line# */{
|
||||||
vx -= px;
|
vx -= px;
|
||||||
vy -= py;
|
vy -= py;
|
||||||
}
|
}
|
||||||
return Numerical.isZero(vx)
|
// Based on the error analysis by @iconexperience outlined in
|
||||||
? vy >= 0 ? px - x : x - px
|
// https://github.com/paperjs/paper.js/issues/799
|
||||||
: Numerical.isZero(vy)
|
return vx === 0
|
||||||
? vx >= 0 ? y - py : py - y
|
? vy >= 0 ? px - x : x - px
|
||||||
: (vx * (y - py) - vy * (x - px)) / Math.sqrt(vx * vx + vy * vy);
|
: vy === 0
|
||||||
|
? vx >= 0 ? y - py : py - y
|
||||||
|
: (vx * (y - py) - vy * (x - px)) / Math.sqrt(vx * vx + vy * vy);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
|
@ -460,11 +460,11 @@ var Point = Base.extend(/** @lends Point# */{
|
||||||
return this.clone();
|
return this.clone();
|
||||||
angle = angle * Math.PI / 180;
|
angle = angle * Math.PI / 180;
|
||||||
var point = center ? this.subtract(center) : this,
|
var point = center ? this.subtract(center) : this,
|
||||||
s = Math.sin(angle),
|
sin = Math.sin(angle),
|
||||||
c = Math.cos(angle);
|
cos = Math.cos(angle);
|
||||||
point = new Point(
|
point = new Point(
|
||||||
point.x * c - point.y * s,
|
point.x * cos - point.y * sin,
|
||||||
point.x * s + point.y * c
|
point.x * sin + point.y * cos
|
||||||
);
|
);
|
||||||
return center ? point.add(center) : point;
|
return center ? point.add(center) : point;
|
||||||
},
|
},
|
||||||
|
@ -690,7 +690,9 @@ var Point = Base.extend(/** @lends Point# */{
|
||||||
* @param {Number} tolerance the maximum distance allowed
|
* @param {Number} tolerance the maximum distance allowed
|
||||||
* @return {Boolean} {@true if it is within the given distance}
|
* @return {Boolean} {@true if it is within the given distance}
|
||||||
*/
|
*/
|
||||||
isClose: function(point, tolerance) {
|
isClose: function(/* point, tolerance */) {
|
||||||
|
var point = Point.read(arguments),
|
||||||
|
tolerance = Base.read(arguments);
|
||||||
return this.getDistance(point) < tolerance;
|
return this.getDistance(point) < tolerance;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -701,11 +703,9 @@ var Point = Base.extend(/** @lends Point# */{
|
||||||
* @param {Point} point the vector to check against
|
* @param {Point} point the vector to check against
|
||||||
* @return {Boolean} {@true it is collinear}
|
* @return {Boolean} {@true it is collinear}
|
||||||
*/
|
*/
|
||||||
isCollinear: function(point) {
|
isCollinear: function(/* point */) {
|
||||||
// NOTE: Numerical.EPSILON is too small, breaking shape-path-shape
|
var point = Point.read(arguments);
|
||||||
// conversion test. But tolerance is probably too large?
|
return Point.isCollinear(this.x, this.y, point.x, point.y);
|
||||||
// TODO: Tests showed that 1e-10 might work well here.
|
|
||||||
return Math.abs(this.cross(point)) < /*#=*/Numerical.TOLERANCE;
|
|
||||||
},
|
},
|
||||||
|
|
||||||
// TODO: Remove version with typo after a while (deprecated June 2015)
|
// TODO: Remove version with typo after a while (deprecated June 2015)
|
||||||
|
@ -718,10 +718,9 @@ var Point = Base.extend(/** @lends Point# */{
|
||||||
* @param {Point} point the vector to check against
|
* @param {Point} point the vector to check against
|
||||||
* @return {Boolean} {@true it is orthogonal}
|
* @return {Boolean} {@true it is orthogonal}
|
||||||
*/
|
*/
|
||||||
isOrthogonal: function(point) {
|
isOrthogonal: function(/* point */) {
|
||||||
// NOTE: Numerical.EPSILON is too small, breaking shape-path-shape
|
var point = Point.read(arguments);
|
||||||
// conversion test.
|
return Point.isOrthogonal(this.x, this.y, point.x, point.y);
|
||||||
return Math.abs(this.dot(point)) < /*#=*/Numerical.TOLERANCE;
|
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -767,23 +766,19 @@ var Point = Base.extend(/** @lends Point# */{
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the projection of the point on another point.
|
* Returns the projection of the point onto another point.
|
||||||
* Both points are interpreted as vectors.
|
* Both points are interpreted as vectors.
|
||||||
*
|
*
|
||||||
* @param {Point} point
|
* @param {Point} point
|
||||||
* @return {Point} the projection of the point on another point
|
* @return {Point} the projection of the point onto another point
|
||||||
*/
|
*/
|
||||||
project: function(/* point */) {
|
project: function(/* point */) {
|
||||||
var point = Point.read(arguments);
|
var point = Point.read(arguments),
|
||||||
if (point.isZero()) {
|
scale = point.isZero() ? 0 : this.dot(point) / point.dot(point);
|
||||||
return new Point(0, 0);
|
return new Point(
|
||||||
} else {
|
point.x * scale,
|
||||||
var scale = this.dot(point) / point.dot(point);
|
point.y * scale
|
||||||
return new Point(
|
);
|
||||||
point.x * scale,
|
|
||||||
point.y * scale
|
|
||||||
);
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -920,6 +915,23 @@ var Point = Base.extend(/** @lends Point# */{
|
||||||
*/
|
*/
|
||||||
random: function() {
|
random: function() {
|
||||||
return new Point(Math.random(), Math.random());
|
return new Point(Math.random(), Math.random());
|
||||||
|
},
|
||||||
|
|
||||||
|
isCollinear: function(x1, y1, x2, y2) {
|
||||||
|
// NOTE: We use normalized vectors so that the epsilon comparison is
|
||||||
|
// reliable. We could instead scale the epsilon based on the vector
|
||||||
|
// length. But instead of normalizing the vectors before calculating
|
||||||
|
// the cross product, we can scale the epsilon accordingly.
|
||||||
|
return Math.abs(x1 * y2 - y1 * x2)
|
||||||
|
<= Math.sqrt((x1 * x1 + y1 * y1) * (x2 * x2 + y2 * y2))
|
||||||
|
* /*#=*/Numerical.TRIGONOMETRIC_EPSILON;
|
||||||
|
},
|
||||||
|
|
||||||
|
isOrthogonal: function(x1, y1, x2, y2) {
|
||||||
|
// See Point.isCollinear()
|
||||||
|
return Math.abs(x1 * x2 + y1 * y2)
|
||||||
|
<= Math.sqrt((x1 * x1 + y1 * y1) * (x2 * x2 + y2 * y2))
|
||||||
|
* /*#=*/Numerical.TRIGONOMETRIC_EPSILON;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}, Base.each(['round', 'ceil', 'floor', 'abs'], function(name) {
|
}, Base.each(['round', 'ceil', 'floor', 'abs'], function(name) {
|
||||||
|
|
|
@ -475,7 +475,7 @@ var Rectangle = Base.extend(/** @lends Rectangle# */{
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The area of the rectangle in square points.
|
* The area of the rectangle.
|
||||||
*
|
*
|
||||||
* @type Number
|
* @type Number
|
||||||
* @bean
|
* @bean
|
||||||
|
@ -857,7 +857,8 @@ var LinkedRectangle = Rectangle.extend({
|
||||||
this._owner[this._setter](this);
|
this._owner[this._setter](this);
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
}, new function() {
|
},
|
||||||
|
new function() {
|
||||||
var proto = Rectangle.prototype;
|
var proto = Rectangle.prototype;
|
||||||
|
|
||||||
return Base.each(['x', 'y', 'width', 'height'], function(key) {
|
return Base.each(['x', 'y', 'width', 'height'], function(key) {
|
||||||
|
|
|
@ -120,7 +120,7 @@ var PaperScope = Base.extend(/** @lends PaperScope# */{
|
||||||
*
|
*
|
||||||
* @type String
|
* @type String
|
||||||
*/
|
*/
|
||||||
version: '/*#=*/__options.version',
|
version: /*#=*/__options.version,
|
||||||
|
|
||||||
// DOCS: PaperScope#settings
|
// DOCS: PaperScope#settings
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -1144,7 +1144,7 @@ var Item = Base.extend(Emitter, /** @lends Item# */{
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @bean
|
* @bean
|
||||||
* @deprecated use {@link #getApplyMatrix()} instead.
|
* @deprecated use {@link #applyMatrix} instead.
|
||||||
*/
|
*/
|
||||||
getTransformContent: '#getApplyMatrix',
|
getTransformContent: '#getApplyMatrix',
|
||||||
setTransformContent: '#setApplyMatrix',
|
setTransformContent: '#setApplyMatrix',
|
||||||
|
@ -1638,10 +1638,10 @@ var Item = Base.extend(Emitter, /** @lends Item# */{
|
||||||
intersects: function(item, _matrix) {
|
intersects: function(item, _matrix) {
|
||||||
if (!(item instanceof Item))
|
if (!(item instanceof Item))
|
||||||
return false;
|
return false;
|
||||||
// Tell _getIntersections to return as soon as some intersections are
|
// Tell getIntersections() to return as soon as some intersections are
|
||||||
// found, because all we care for here is there are some or none:
|
// found, because all we care for here is there are some or none:
|
||||||
return this._asPathItem()._getIntersections(item._asPathItem(),
|
return this._asPathItem().getIntersections(item._asPathItem(), null,
|
||||||
_matrix || item._matrix, [], true).length > 0;
|
_matrix || item._matrix, true).length > 0;
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1652,7 +1652,7 @@ var Item = Base.extend(Emitter, /** @lends Item# */{
|
||||||
* and may contain a combination of the following values:
|
* and may contain a combination of the following values:
|
||||||
*
|
*
|
||||||
* @option [options.tolerance={@link PaperScope#settings}.hitTolerance]
|
* @option [options.tolerance={@link PaperScope#settings}.hitTolerance]
|
||||||
* {Number} the tolerance of the hit-test in points
|
* {Number} the tolerance of the hit-test
|
||||||
* @option options.class {Function} only hit-test again a certain item class
|
* @option options.class {Function} only hit-test again a certain item class
|
||||||
* and its sub-classes: {@code Group, Layer, Path, CompoundPath,
|
* and its sub-classes: {@code Group, Layer, Path, CompoundPath,
|
||||||
* Shape, Raster, PlacedSymbol, PointText}, etc
|
* Shape, Raster, PlacedSymbol, PointText}, etc
|
||||||
|
@ -2618,6 +2618,16 @@ var Item = Base.extend(Emitter, /** @lends Item# */{
|
||||||
return item ? item.isDescendant(this) : false;
|
return item ? item.isDescendant(this) : false;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the item is an a sibling of the specified item.
|
||||||
|
*
|
||||||
|
* @param {Item} item the item to check against
|
||||||
|
* @return {Boolean} {@true if the item is aa sibling of the specified item}
|
||||||
|
*/
|
||||||
|
isSibling: function(item) {
|
||||||
|
return this._parent === item._parent;
|
||||||
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Checks whether the item is grouped with the specified item.
|
* Checks whether the item is grouped with the specified item.
|
||||||
*
|
*
|
||||||
|
|
|
@ -207,7 +207,7 @@ var Raster = Item.extend(/** @lends Raster# */{
|
||||||
/**
|
/**
|
||||||
* @private
|
* @private
|
||||||
* @bean
|
* @bean
|
||||||
* @deprecated use {@link #getResolution()} instead.
|
* @deprecated use {@link #resolution} instead.
|
||||||
*/
|
*/
|
||||||
getPpi: '#getResolution',
|
getPpi: '#getResolution',
|
||||||
|
|
||||||
|
|
|
@ -64,7 +64,7 @@ var Shape = Item.extend(/** @lends Shape# */{
|
||||||
/**
|
/**
|
||||||
* @private
|
* @private
|
||||||
* @bean
|
* @bean
|
||||||
* @deprecated use {@link #getType()} instead.
|
* @deprecated use {@link #type} instead.
|
||||||
*/
|
*/
|
||||||
getShape: '#getType',
|
getShape: '#getType',
|
||||||
setShape: '#setType',
|
setShape: '#setType',
|
||||||
|
@ -277,7 +277,6 @@ var Shape = Item.extend(/** @lends Shape# */{
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
new function() { // Scope for _contains() and _hitTestSelf() code.
|
new function() { // Scope for _contains() and _hitTestSelf() code.
|
||||||
|
|
||||||
// Returns the center of the quarter corner ellipse for rounded rectangle,
|
// Returns the center of the quarter corner ellipse for rounded rectangle,
|
||||||
// if the point lies within its bounding box.
|
// if the point lies within its bounding box.
|
||||||
function getCornerCenter(that, point, expand) {
|
function getCornerCenter(that, point, expand) {
|
||||||
|
|
|
@ -102,6 +102,15 @@ var CompoundPath = PathItem.extend(/** @lends CompoundPath# */{
|
||||||
},
|
},
|
||||||
|
|
||||||
insertChildren: function insertChildren(index, items, _preserve) {
|
insertChildren: function insertChildren(index, items, _preserve) {
|
||||||
|
// Convert CompoundPath items in the children list by adding their
|
||||||
|
// children to the list and removing their parent.
|
||||||
|
for (var i = items.length - 1; i >= 0; i--) {
|
||||||
|
var item = items[i];
|
||||||
|
if (item instanceof CompoundPath) {
|
||||||
|
items.splice.apply(items, [i, 1].concat(item.removeChildren()));
|
||||||
|
item.remove();
|
||||||
|
}
|
||||||
|
}
|
||||||
// Pass on 'path' for _type, to make sure that only paths are added as
|
// Pass on 'path' for _type, to make sure that only paths are added as
|
||||||
// children.
|
// children.
|
||||||
items = insertChildren.base.call(this, index, items, _preserve, Path);
|
items = insertChildren.base.call(this, index, items, _preserve, Path);
|
||||||
|
@ -131,16 +140,23 @@ var CompoundPath = PathItem.extend(/** @lends CompoundPath# */{
|
||||||
this._children[i].smooth();
|
this._children[i].smooth();
|
||||||
},
|
},
|
||||||
|
|
||||||
|
// DOCS: reduce()
|
||||||
|
// TEST: reduce()
|
||||||
reduce: function reduce() {
|
reduce: function reduce() {
|
||||||
if (this._children.length === 0) { // Replace with a simple empty Path
|
var children = this._children;
|
||||||
|
for (var i = children.length - 1; i >= 0; i--) {
|
||||||
|
var path = children[i].reduce();
|
||||||
|
if (path.isEmpty())
|
||||||
|
children.splice(i, 1);
|
||||||
|
}
|
||||||
|
if (children.length === 0) { // Replace with a simple empty Path
|
||||||
var path = new Path(Item.NO_INSERT);
|
var path = new Path(Item.NO_INSERT);
|
||||||
path.insertAbove(this);
|
path.insertAbove(this);
|
||||||
path.setStyle(this._style);
|
path.setStyle(this._style);
|
||||||
this.remove();
|
this.remove();
|
||||||
return path;
|
return path;
|
||||||
} else {
|
|
||||||
return reduce.base.call(this);
|
|
||||||
}
|
}
|
||||||
|
return reduce.base.call(this);
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -220,8 +236,8 @@ var CompoundPath = PathItem.extend(/** @lends CompoundPath# */{
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The area of the path in square points. Self-intersecting paths can
|
* The area that the path's geometry is covering. Self-intersecting paths
|
||||||
* contain sub-areas that cancel each other out.
|
* can contain sub-areas that cancel each other out.
|
||||||
*
|
*
|
||||||
* @type Number
|
* @type Number
|
||||||
* @bean
|
* @bean
|
||||||
|
@ -298,7 +314,8 @@ var CompoundPath = PathItem.extend(/** @lends CompoundPath# */{
|
||||||
: matrix.chain(mx));
|
: matrix.chain(mx));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}, new function() { // Injection scope for PostScript-like drawing functions
|
},
|
||||||
|
new function() { // Injection scope for PostScript-like drawing functions
|
||||||
/**
|
/**
|
||||||
* Helper method that returns the current path and checks if a moveTo()
|
* Helper method that returns the current path and checks if a moveTo()
|
||||||
* command is required first.
|
* command is required first.
|
||||||
|
|
1395
src/path/Curve.js
1395
src/path/Curve.js
File diff suppressed because it is too large
Load diff
|
@ -13,16 +13,16 @@
|
||||||
/**
|
/**
|
||||||
* @name CurveLocation
|
* @name CurveLocation
|
||||||
*
|
*
|
||||||
* @class CurveLocation objects describe a location on {@link Curve}
|
* @class CurveLocation objects describe a location on {@link Curve} objects,
|
||||||
* objects, as defined by the curve {@link #parameter}, a value between
|
* as defined by the curve-time {@link #parameter}, a value between {@code 0}
|
||||||
* {@code 0} (beginning of the curve) and {@code 1} (end of the curve). If
|
* (beginning of the curve) and {@code 1} (end of the curve). If the curve is
|
||||||
* the curve is part of a {@link Path} item, its {@link #index} inside the
|
* part of a {@link Path} item, its {@link #index} inside the
|
||||||
* {@link Path#curves} array is also provided.
|
* {@link Path#curves} array is also provided.
|
||||||
*
|
*
|
||||||
* The class is in use in many places, such as
|
* The class is in use in many places, such as
|
||||||
* {@link Path#getLocationAt(offset, isParameter)},
|
* {@link Path#getLocationAt(offset)},
|
||||||
* {@link Path#getLocationOf(point)},
|
* {@link Path#getLocationOf(point)},
|
||||||
* {@link Path#getNearestLocation(point),
|
* {@link Path#getNearestLocation(point)},
|
||||||
* {@link PathItem#getIntersections(path)},
|
* {@link PathItem#getIntersections(path)},
|
||||||
* etc.
|
* etc.
|
||||||
*/
|
*/
|
||||||
|
@ -41,22 +41,35 @@ var CurveLocation = Base.extend(/** @lends CurveLocation# */{
|
||||||
* @param {Number} parameter
|
* @param {Number} parameter
|
||||||
* @param {Point} [point]
|
* @param {Point} [point]
|
||||||
*/
|
*/
|
||||||
initialize: function CurveLocation(curve, parameter, point, _curve2,
|
initialize: function CurveLocation(curve, parameter, point,
|
||||||
_parameter2, _point2, _distance) {
|
_overlap, _distance) {
|
||||||
|
// Merge intersections very close to the end of a curve with the
|
||||||
|
// beginning of the next curve.
|
||||||
|
if (parameter > /*#=*/(1 - Numerical.CURVETIME_EPSILON)) {
|
||||||
|
var next = curve.getNext();
|
||||||
|
if (next) {
|
||||||
|
parameter = 0;
|
||||||
|
curve = next;
|
||||||
|
}
|
||||||
|
}
|
||||||
// Define this CurveLocation's unique id.
|
// Define this CurveLocation's unique id.
|
||||||
// NOTE: We do not use the same pool as the rest of the library here,
|
// NOTE: We do not use the same pool as the rest of the library here,
|
||||||
// since this is only required to be unique at runtime among other
|
// since this is only required to be unique at runtime among other
|
||||||
// CurveLocation objects.
|
// CurveLocation objects.
|
||||||
this._id = UID.get(CurveLocation);
|
this._id = UID.get(CurveLocation);
|
||||||
|
this._setCurve(curve);
|
||||||
|
this._parameter = parameter;
|
||||||
|
this._point = point || curve.getPointAt(parameter, true);
|
||||||
|
this._overlap = _overlap;
|
||||||
|
this._distance = _distance;
|
||||||
|
this._intersection = this._next = this._prev = null;
|
||||||
|
},
|
||||||
|
|
||||||
|
_setCurve: function(curve) {
|
||||||
var path = curve._path;
|
var path = curve._path;
|
||||||
this._version = path ? path._version : 0;
|
this._version = path ? path._version : 0;
|
||||||
this._curve = curve;
|
this._curve = curve;
|
||||||
this._parameter = parameter;
|
this._segment = null; // To be determined, see #getSegment()
|
||||||
this._point = point || curve.getPointAt(parameter, true);
|
|
||||||
this._curve2 = _curve2;
|
|
||||||
this._parameter2 = _parameter2;
|
|
||||||
this._point2 = _point2;
|
|
||||||
this._distance = _distance;
|
|
||||||
// Also store references to segment1 and segment2, in case path
|
// Also store references to segment1 and segment2, in case path
|
||||||
// splitting / dividing is going to happen, in which case the segments
|
// splitting / dividing is going to happen, in which case the segments
|
||||||
// can be used to determine the new curves, see #getCurve(true)
|
// can be used to determine the new curves, see #getCurve(true)
|
||||||
|
@ -64,31 +77,40 @@ var CurveLocation = Base.extend(/** @lends CurveLocation# */{
|
||||||
this._segment2 = curve._segment2;
|
this._segment2 = curve._segment2;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
_setSegment: function(segment) {
|
||||||
|
this._setCurve(segment.getCurve());
|
||||||
|
this._segment = segment;
|
||||||
|
this._parameter = segment === this._segment1 ? 0 : 1;
|
||||||
|
// To avoid issues with imprecision in getCurve() / trySegment()
|
||||||
|
this._point = segment._point.clone();
|
||||||
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The segment of the curve which is closer to the described location.
|
* The segment of the curve which is closer to the described location.
|
||||||
*
|
*
|
||||||
* @type Segment
|
* @type Segment
|
||||||
* @bean
|
* @bean
|
||||||
*/
|
*/
|
||||||
getSegment: function(_preferFirst) {
|
getSegment: function() {
|
||||||
if (!this._segment) {
|
// Request curve first, so _segment gets invalidated if it's out of sync
|
||||||
var curve = this.getCurve(),
|
var curve = this.getCurve(),
|
||||||
parameter = this.getParameter();
|
segment = this._segment;
|
||||||
if (parameter === 1) {
|
if (!segment) {
|
||||||
this._segment = curve._segment2;
|
var parameter = this.getParameter();
|
||||||
} else if (parameter === 0 || _preferFirst) {
|
if (parameter === 0) {
|
||||||
this._segment = curve._segment1;
|
segment = curve._segment1;
|
||||||
} else if (parameter == null) {
|
} else if (parameter === 1) {
|
||||||
return null;
|
segment = curve._segment2;
|
||||||
} else {
|
} else if (parameter != null) {
|
||||||
// Determine the closest segment by comparing curve lengths
|
// Determine the closest segment by comparing curve lengths
|
||||||
this._segment = curve.getPartLength(0, parameter)
|
segment = curve.getPartLength(0, parameter)
|
||||||
< curve.getPartLength(parameter, 1)
|
< curve.getPartLength(parameter, 1)
|
||||||
? curve._segment1
|
? curve._segment1
|
||||||
: curve._segment2;
|
: curve._segment2;
|
||||||
}
|
}
|
||||||
|
this._segment = segment;
|
||||||
}
|
}
|
||||||
return this._segment;
|
return segment;
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -99,29 +121,34 @@ var CurveLocation = Base.extend(/** @lends CurveLocation# */{
|
||||||
*/
|
*/
|
||||||
getCurve: function() {
|
getCurve: function() {
|
||||||
var curve = this._curve,
|
var curve = this._curve,
|
||||||
path = curve && curve._path;
|
path = curve && curve._path,
|
||||||
|
that = this;
|
||||||
if (path && path._version !== this._version) {
|
if (path && path._version !== this._version) {
|
||||||
// If the path's segments have changed in the meantime, clear the
|
// If the path's segments have changed in the meantime, clear the
|
||||||
// internal _parameter value and force refetching of the correct
|
// internal _parameter value and force refetching of the correct
|
||||||
// curve again here.
|
// curve again here.
|
||||||
curve = null;
|
curve = this._parameter = this._curve = this._offset = null;
|
||||||
this._parameter = null;
|
|
||||||
}
|
}
|
||||||
if (!curve) {
|
|
||||||
// If we're asked to get the curve uncached, access current curve
|
// If path is out of sync, access current curve objects through segment1
|
||||||
// objects through segment1 / segment2. Since path splitting or
|
// / segment2. Since path splitting or dividing might have happened in
|
||||||
// dividing might have happened in the meantime, try segment1's
|
// the meantime, try segment1's curve, and see if _point lies on it
|
||||||
// curve, and see if _point lies on it still, otherwise assume it's
|
// still, otherwise assume it's the curve before segment2.
|
||||||
// the curve before segment2.
|
function trySegment(segment) {
|
||||||
curve = this._segment1.getCurve();
|
var curve = segment && segment.getCurve();
|
||||||
if (curve.getParameterOf(this._point) == null)
|
if (curve && (that._parameter = curve.getParameterOf(that._point))
|
||||||
curve = this._segment2.getPrevious().getCurve();
|
!= null) {
|
||||||
this._curve = curve;
|
// Fetch path again as it could be on a new one through split()
|
||||||
// Fetch path again as it could be on a new one through split()
|
that._setCurve(curve);
|
||||||
path = curve._path;
|
that._segment = segment;
|
||||||
this._version = path ? path._version : 0;
|
return curve;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return curve;
|
|
||||||
|
return curve
|
||||||
|
|| trySegment(this._segment)
|
||||||
|
|| trySegment(this._segment1)
|
||||||
|
|| trySegment(this._segment2.getPrevious());
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -136,8 +163,8 @@ var CurveLocation = Base.extend(/** @lends CurveLocation# */{
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The index of the curve within the {@link Path#curves} list, if the
|
* The index of the {@link #curve} within the {@link Path#curves} list, if
|
||||||
* curve is part of a {@link Path} item.
|
* it is part of a {@link Path} item.
|
||||||
*
|
*
|
||||||
* @type Index
|
* @type Index
|
||||||
* @bean
|
* @bean
|
||||||
|
@ -148,9 +175,9 @@ var CurveLocation = Base.extend(/** @lends CurveLocation# */{
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The curve parameter, as used by various bezier curve calculations. It is
|
* The curve-time parameter, as used by various bezier curve calculations.
|
||||||
* value between {@code 0} (beginning of the curve) and {@code 1} (end of
|
* It is value between {@code 0} (beginning of the curve) and {@code 1}
|
||||||
* the curve).
|
* (end of the curve).
|
||||||
*
|
*
|
||||||
* @type Number
|
* @type Number
|
||||||
* @bean
|
* @bean
|
||||||
|
@ -183,8 +210,13 @@ var CurveLocation = Base.extend(/** @lends CurveLocation# */{
|
||||||
* @bean
|
* @bean
|
||||||
*/
|
*/
|
||||||
getOffset: function() {
|
getOffset: function() {
|
||||||
var path = this.getPath();
|
var offset = this._offset;
|
||||||
return path ? path._getOffset(this) : this.getCurveOffset();
|
if (offset == null) {
|
||||||
|
var path = this.getPath();
|
||||||
|
offset = this._offset = path ? path._getOffset(this)
|
||||||
|
: this.getCurveOffset();
|
||||||
|
}
|
||||||
|
return offset;
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -209,37 +241,31 @@ var CurveLocation = Base.extend(/** @lends CurveLocation# */{
|
||||||
* @bean
|
* @bean
|
||||||
*/
|
*/
|
||||||
getIntersection: function() {
|
getIntersection: function() {
|
||||||
var intersection = this._intersection;
|
return this._intersection;
|
||||||
if (!intersection && this._curve2) {
|
|
||||||
// If we have the parameter on the other curve use that for
|
|
||||||
// intersection rather than the point.
|
|
||||||
this._intersection = intersection = new CurveLocation(this._curve2,
|
|
||||||
this._parameter2, this._point2 || this._point);
|
|
||||||
intersection._overlap = this._overlap;
|
|
||||||
intersection._intersection = this;
|
|
||||||
}
|
|
||||||
return intersection;
|
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The tangential vector to the {@link #curve} at the given location.
|
* The tangential vector to the {@link #curve} at the given location.
|
||||||
*
|
*
|
||||||
* @name Item#tangent
|
* @name CurveLocation#getTangent
|
||||||
* @type Point
|
* @type Point
|
||||||
|
* @bean
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The normal vector to the {@link #curve} at the given location.
|
* The normal vector to the {@link #curve} at the given location.
|
||||||
*
|
*
|
||||||
* @name Item#normal
|
* @name CurveLocation#getNormal
|
||||||
* @type Point
|
* @type Point
|
||||||
|
* @bean
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The curvature of the {@link #curve} at the given location.
|
* The curvature of the {@link #curve} at the given location.
|
||||||
*
|
*
|
||||||
* @name Item#curvature
|
* @name CurveLocation#getCurvature
|
||||||
* @type Number
|
* @type Number
|
||||||
|
* @bean
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -274,23 +300,40 @@ var CurveLocation = Base.extend(/** @lends CurveLocation# */{
|
||||||
* @param {CurveLocation} location
|
* @param {CurveLocation} location
|
||||||
* @return {Boolean} {@true if the locations are equal}
|
* @return {Boolean} {@true if the locations are equal}
|
||||||
*/
|
*/
|
||||||
equals: function(loc) {
|
equals: function(loc, _ignoreOther) {
|
||||||
var abs = Math.abs,
|
var res = this === loc,
|
||||||
// Use the same tolerance for curve time parameter comparisons as
|
epsilon = /*#=*/Numerical.GEOMETRIC_EPSILON;
|
||||||
// in Curve.js when considering two locations the same.
|
// NOTE: We need to compare both by (index + parameter) and by proximity
|
||||||
tolerance = /*#=*/Numerical.TOLERANCE;
|
// of points. See:
|
||||||
return this === loc
|
// https://github.com/paperjs/paper.js/issues/784#issuecomment-143161586
|
||||||
|| loc instanceof CurveLocation
|
if (!res && loc instanceof CurveLocation
|
||||||
// Call getCurve() and getParameter() to keep in sync
|
&& this.getPath() === loc.getPath()
|
||||||
&& this.getCurve() === loc.getCurve()
|
&& this.getPoint().isClose(loc.getPoint(), epsilon)) {
|
||||||
&& abs(this.getParameter() - loc.getParameter()) < tolerance
|
// The position is the same, but it could still be in a different
|
||||||
// _curve2/_parameter2 are only used for Boolean operations
|
// location on the path. Perform more thorough checks now:
|
||||||
// and don't need syncing there.
|
var c1 = this.getCurve(),
|
||||||
// TODO: That's not quite true though... Rework this!
|
c2 = loc.getCurve(),
|
||||||
&& this._curve2 === loc._curve2
|
abs = Math.abs,
|
||||||
&& abs((this._parameter2 || 0) - (loc._parameter2 || 0))
|
// We need to wrap diff around the path's beginning / end:
|
||||||
< tolerance
|
diff = abs(
|
||||||
|| false;
|
((c1.isLast() && c2.isFirst() ? -1 : c1.getIndex())
|
||||||
|
+ this.getParameter()) -
|
||||||
|
((c2.isLast() && c1.isFirst() ? -1 : c2.getIndex())
|
||||||
|
+ loc.getParameter()));
|
||||||
|
res = (diff < /*#=*/Numerical.CURVETIME_EPSILON
|
||||||
|
// If diff isn't close enough, compare the actual offsets of
|
||||||
|
// both locations to determine if they're in the same spot,
|
||||||
|
// taking into account the wrapping around path ends too.
|
||||||
|
// This is necessary in order to handle very short consecutive
|
||||||
|
// curves (length ~< 1e-7), which would lead to diff > 1.
|
||||||
|
|| ((diff = abs(this.getOffset() - loc.getOffset())) < epsilon
|
||||||
|
|| abs(this.getPath().getLength() - diff) < epsilon))
|
||||||
|
&& (_ignoreOther
|
||||||
|
|| (!this._intersection && !loc._intersection
|
||||||
|
|| this._intersection && this._intersection.equals(
|
||||||
|
loc._intersection, true)));
|
||||||
|
}
|
||||||
|
return res;
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -313,37 +356,207 @@ var CurveLocation = Base.extend(/** @lends CurveLocation# */{
|
||||||
return '{ ' + parts.join(', ') + ' }';
|
return '{ ' + parts.join(', ') + ' }';
|
||||||
},
|
},
|
||||||
|
|
||||||
statics: {
|
|
||||||
sort: function(locations) {
|
/**
|
||||||
var tolerance = /*#=*/Numerical.TOLERANCE;
|
* {@grouptitle Tests}
|
||||||
locations.sort(function compare(l1, l2) {
|
* Checks if the location is an intersection with another curve and is
|
||||||
var curve1 = l1._curve,
|
* merely touching the other curve, as opposed to crossing it.
|
||||||
curve2 = l2._curve,
|
*
|
||||||
path1 = curve1._path,
|
* @return {Boolean} {@true if the location is an intersection that is
|
||||||
path2 = curve2._path;
|
* merely touching another curve}
|
||||||
// Sort by path-id, curve, parameter, curve2, parameter2 so we
|
* @see #isCrossing()
|
||||||
// can easily remove duplicates with calls to equals() after.
|
*/
|
||||||
return path1 === path2
|
isTouching: function() {
|
||||||
? curve1 === curve2
|
var inter = this._intersection;
|
||||||
? Math.abs(l1._parameter - l2._parameter) < tolerance
|
if (inter && this.getTangent().isCollinear(inter.getTangent())) {
|
||||||
? l1._curve2 === l2._curve2
|
// Only consider two straight curves as touching if their lines
|
||||||
? l1._parameter2 - l2._parameter2
|
// don't intersect.
|
||||||
: l1._curve2.getIndex() - l2._curve2.getIndex()
|
var curve1 = this.getCurve(),
|
||||||
: l1._parameter - l2._parameter
|
curve2 = inter.getCurve();
|
||||||
: curve1.getIndex() - curve2.getIndex()
|
return !(curve1.isStraight() && curve2.isStraight()
|
||||||
// Sort by path id to group all locs on the same path.
|
&& curve1.getLine().intersect(curve2.getLine()));
|
||||||
: path1._id - path2._id;
|
}
|
||||||
|
return false;
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the location is an intersection with another curve and is
|
||||||
|
* crossing the other curve, as opposed to just touching it.
|
||||||
|
*
|
||||||
|
* @return {Boolean} {@true if the location is an intersection that is
|
||||||
|
* crossing another curve}
|
||||||
|
* @see #isTouching()
|
||||||
|
*/
|
||||||
|
isCrossing: function(_report) {
|
||||||
|
// Implementation based on work by Andy Finnell:
|
||||||
|
// http://losingfight.com/blog/2011/07/09/how-to-implement-boolean-operations-on-bezier-paths-part-3/
|
||||||
|
// https://bitbucket.org/andyfinnell/vectorboolean
|
||||||
|
var inter = this._intersection;
|
||||||
|
if (!inter)
|
||||||
|
return false;
|
||||||
|
// TODO: Make getCurve() and getParameter() sync work in boolean ops
|
||||||
|
// before and after splitting!!!
|
||||||
|
var t1 = this._parameter,
|
||||||
|
t2 = inter._parameter,
|
||||||
|
tMin = /*#=*/Numerical.CURVETIME_EPSILON,
|
||||||
|
tMax = 1 - tMin;
|
||||||
|
// If the intersection is in the middle of the path, it is either a
|
||||||
|
// tangent or a crossing, no need for the detailed corner check below.
|
||||||
|
// But we do need a check for the edge case of tangents?
|
||||||
|
if (t1 >= tMin && t1 <= tMax || t2 >= tMin && t2 <= tMax)
|
||||||
|
return !this.isTouching();
|
||||||
|
// Values for getTangentAt() that are almost 0 and 1.
|
||||||
|
// NOTE: Even though getTangentAt() has code to support 0 and 1 instead
|
||||||
|
// of tMin and tMax, we still need to use this instead, as other issues
|
||||||
|
// emerge from switching to 0 and 1 in edge cases.
|
||||||
|
// NOTE: VectorBoolean has code that slowly shifts these points inwards
|
||||||
|
// until the resulting tangents are not ambiguous. Do we need this too?
|
||||||
|
var c2 = this._curve,
|
||||||
|
c1 = c2.getPrevious(),
|
||||||
|
c4 = inter._curve,
|
||||||
|
c3 = c4.getPrevious(),
|
||||||
|
PI = Math.PI;
|
||||||
|
if (!c1 || !c3)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (_report) {
|
||||||
|
new Path.Circle({
|
||||||
|
center: this.getPoint(),
|
||||||
|
radius: 10,
|
||||||
|
strokeColor: 'red'
|
||||||
|
});
|
||||||
|
new Path({
|
||||||
|
segments: [c1.getSegment1(), c1.getSegment2(), c2.getSegment2()],
|
||||||
|
strokeColor: 'red',
|
||||||
|
strokeWidth: 4
|
||||||
|
});
|
||||||
|
new Path({
|
||||||
|
segments: [c3.getSegment1(), c3.getSegment2(), c4.getSegment2()],
|
||||||
|
strokeColor: 'orange',
|
||||||
|
strokeWidth: 4
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function isInRange(angle, min, max) {
|
||||||
|
return min < max
|
||||||
|
? angle > min && angle < max
|
||||||
|
// The range wraps around -PI / PI:
|
||||||
|
: angle > min && angle <= PI || angle >= -PI && angle < max;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Calculate angles for all four tangents at the intersection point
|
||||||
|
var a1 = c1.getTangentAt(tMax, true).negate().getAngleInRadians(),
|
||||||
|
a2 = c2.getTangentAt(tMin, true).getAngleInRadians(),
|
||||||
|
a3 = c3.getTangentAt(tMax, true).negate().getAngleInRadians(),
|
||||||
|
a4 = c4.getTangentAt(tMin, true).getAngleInRadians();
|
||||||
|
|
||||||
|
// Count how many times curve2 angles appear between the curve1 angles
|
||||||
|
// If each pair of angles split the other two, then the edges cross.
|
||||||
|
return (isInRange(a3, a1, a2) ^ isInRange(a4, a1, a2))
|
||||||
|
&& (isInRange(a3, a2, a1) ^ isInRange(a4, a2, a1));
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the location is an intersection with another curve and is
|
||||||
|
* part of an overlap between the two involved paths.
|
||||||
|
*
|
||||||
|
* @return {Boolean} {@true if the location is an intersection that is
|
||||||
|
* part of an overlap between the two involved paths}
|
||||||
|
* @see #isCrossing()
|
||||||
|
* @see #isTouching()
|
||||||
|
*/
|
||||||
|
isOverlap: function() {
|
||||||
|
return !!this._overlap;
|
||||||
}
|
}
|
||||||
}, Base.each(Curve.evaluateMethods, function(name) {
|
}, Base.each(Curve.evaluateMethods, function(name) {
|
||||||
// Produce getters for #getTangent() / #getNormal() / #getCurvature()
|
// Produce getters for #getTangent() / #getNormal() / #getCurvature()
|
||||||
if (name !== 'getPoint') {
|
var get = name + 'At';
|
||||||
var get = name + 'At';
|
this[name] = function() {
|
||||||
this[name] = function() {
|
var parameter = this.getParameter(),
|
||||||
var parameter = this.getParameter(),
|
curve = this.getCurve();
|
||||||
curve = this.getCurve();
|
return parameter != null && curve && curve[get](parameter, true);
|
||||||
return parameter != null && curve && curve[get](parameter, true);
|
};
|
||||||
};
|
}, {
|
||||||
|
// Do not override the existing #getPoint():
|
||||||
|
preserve: true
|
||||||
|
}),
|
||||||
|
new function() { // Scope for statics
|
||||||
|
|
||||||
|
function insert(locations, loc, merge) {
|
||||||
|
// Insert-sort by path-id, curve, parameter so we can easily merge
|
||||||
|
// duplicates with calls to equals() after.
|
||||||
|
var length = locations.length,
|
||||||
|
l = 0,
|
||||||
|
r = length - 1,
|
||||||
|
abs = Math.abs;
|
||||||
|
|
||||||
|
function search(index, dir) {
|
||||||
|
// If we reach the beginning/end of the list, also compare with the
|
||||||
|
// location at the other end, as paths are circular lists.
|
||||||
|
// NOTE: When merging, the locations array will only contain
|
||||||
|
// locations on the same path, so it is fine that check for the end
|
||||||
|
// to address circularity. See PathItem#getIntersections()
|
||||||
|
for (var i = index + dir; i >= -1 && i <= length; i += dir) {
|
||||||
|
// Wrap the index around, to match the other ends:
|
||||||
|
var loc2 = locations[((i % length) + length) % length];
|
||||||
|
// Once we're outside the spot, we can stop searching.
|
||||||
|
if (!loc.getPoint().isClose(loc2.getPoint(),
|
||||||
|
/*#=*/Numerical.GEOMETRIC_EPSILON))
|
||||||
|
break;
|
||||||
|
if (loc.equals(loc2))
|
||||||
|
return loc2;
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
while (l <= r) {
|
||||||
|
var m = (l + r) >>> 1,
|
||||||
|
loc2 = locations[m],
|
||||||
|
found;
|
||||||
|
// See if the two locations are actually the same, and merge if
|
||||||
|
// they are. If they aren't check the other neighbors with search()
|
||||||
|
if (merge && (found = loc.equals(loc2) ? loc2
|
||||||
|
: (search(m, -1) || search(m, 1)))) {
|
||||||
|
// We're done, don't insert, merge with the found location
|
||||||
|
// instead, and carry over overlap:
|
||||||
|
if (loc._overlap) {
|
||||||
|
found._overlap = found._intersection._overlap = true;
|
||||||
|
}
|
||||||
|
return found;
|
||||||
|
}
|
||||||
|
var path1 = loc.getPath(),
|
||||||
|
path2 = loc2.getPath(),
|
||||||
|
// NOTE: equals() takes the intersection location into account,
|
||||||
|
// while this calculation of diff doesn't!
|
||||||
|
diff = path1 === path2
|
||||||
|
//Sort by both index and parameter. The two values added
|
||||||
|
// together provides a convenient sorting index.
|
||||||
|
? (loc.getIndex() + loc.getParameter())
|
||||||
|
- (loc2.getIndex() + loc2.getParameter())
|
||||||
|
// Sort by path id to group all locs on same path.
|
||||||
|
: path1._id - path2._id;
|
||||||
|
if (diff < 0) {
|
||||||
|
r = m - 1;
|
||||||
|
} else {
|
||||||
|
l = m + 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// We didn't merge with a preexisting location, insert it now.
|
||||||
|
locations.splice(l, 0, loc);
|
||||||
|
return loc;
|
||||||
}
|
}
|
||||||
}, {}));
|
|
||||||
|
return { statics: {
|
||||||
|
insert: insert,
|
||||||
|
|
||||||
|
expand: function(locations) {
|
||||||
|
// Create a copy since insert() keeps modifying the array and
|
||||||
|
// inserting at sorted indices.
|
||||||
|
var expanded = locations.slice();
|
||||||
|
for (var i = 0, l = locations.length; i < l; i++) {
|
||||||
|
insert(expanded, locations[i]._intersection, false);
|
||||||
|
}
|
||||||
|
return expanded;
|
||||||
|
}
|
||||||
|
}};
|
||||||
|
});
|
||||||
|
|
252
src/path/Path.js
252
src/path/Path.js
|
@ -150,7 +150,9 @@ var Path = PathItem.extend(/** @lends Path# */{
|
||||||
if (parent)
|
if (parent)
|
||||||
parent._currentPath = undefined;
|
parent._currentPath = undefined;
|
||||||
// Clockwise state becomes undefined as soon as geometry changes.
|
// Clockwise state becomes undefined as soon as geometry changes.
|
||||||
this._length = this._clockwise = undefined;
|
// Also clear cached mono curves used for winding calculations.
|
||||||
|
this._length = this._area = this._clockwise = this._monoCurves =
|
||||||
|
undefined;
|
||||||
if (flags & /*#=*/ChangeFlag.SEGMENTS) {
|
if (flags & /*#=*/ChangeFlag.SEGMENTS) {
|
||||||
this._version++; // See CurveLocation
|
this._version++; // See CurveLocation
|
||||||
} else if (this._curves) {
|
} else if (this._curves) {
|
||||||
|
@ -159,10 +161,6 @@ var Path = PathItem.extend(/** @lends Path# */{
|
||||||
for (var i = 0, l = this._curves.length; i < l; i++)
|
for (var i = 0, l = this._curves.length; i < l; i++)
|
||||||
this._curves[i]._changed();
|
this._curves[i]._changed();
|
||||||
}
|
}
|
||||||
// Clear cached curves used for winding direction and containment
|
|
||||||
// calculation.
|
|
||||||
// NOTE: This is only needed with __options.booleanOperations
|
|
||||||
this._monoCurves = undefined;
|
|
||||||
} else if (flags & /*#=*/ChangeFlag.STROKE) {
|
} else if (flags & /*#=*/ChangeFlag.STROKE) {
|
||||||
// TODO: We could preserve the purely geometric bounds that are not
|
// TODO: We could preserve the purely geometric bounds that are not
|
||||||
// affected by stroke: _bounds.bounds and _bounds.handleBounds
|
// affected by stroke: _bounds.bounds and _bounds.handleBounds
|
||||||
|
@ -368,40 +366,6 @@ var Path = PathItem.extend(/** @lends Path# */{
|
||||||
return this._segments.length === 0;
|
return this._segments.length === 0;
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
|
||||||
* Checks if this path consists of only linear curves. This can mean that
|
|
||||||
* the curves have no handles defined, or that the handles run collinear
|
|
||||||
* with the line.
|
|
||||||
*
|
|
||||||
* @return {Boolean} {@true if the path is entirely linear}
|
|
||||||
* @see Segment#isLinear()
|
|
||||||
* @see Curve#isLinear()
|
|
||||||
*/
|
|
||||||
isLinear: function() {
|
|
||||||
var segments = this._segments;
|
|
||||||
for (var i = 0, l = segments.length; i < l; i++) {
|
|
||||||
if (!segments[i].isLinear())
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
},
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Checks if none of the curves in the path define any curve handles.
|
|
||||||
*
|
|
||||||
* @return {Boolean} {@true if the path contains no curve handles}
|
|
||||||
* @see Segment#hasHandles()
|
|
||||||
* @see Curve#hasHandles()
|
|
||||||
*/
|
|
||||||
hasHandles: function() {
|
|
||||||
var segments = this._segments;
|
|
||||||
for (var i = 0, l = segments.length; i < l; i++) {
|
|
||||||
if (segments[i].hasHandles())
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
},
|
|
||||||
|
|
||||||
_transformContent: function(matrix) {
|
_transformContent: function(matrix) {
|
||||||
var coords = new Array(6);
|
var coords = new Array(6);
|
||||||
for (var i = 0, l = this._segments.length; i < l; i++)
|
for (var i = 0, l = this._segments.length; i < l; i++)
|
||||||
|
@ -410,10 +374,10 @@ var Path = PathItem.extend(/** @lends Path# */{
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Private method that adds a segment to the segment list. It assumes that
|
* Private method that adds segments to the segment list. It assumes that
|
||||||
* the passed object is a segment already and does not perform any checks.
|
* the passed object is an array of segments already and does not perform
|
||||||
* If a curves list was requested, it will kept in sync with the segments
|
* any checks. If a curves list was requested, it will be kept in sync with
|
||||||
* list automatically.
|
* the segments list automatically.
|
||||||
*/
|
*/
|
||||||
_add: function(segs, index) {
|
_add: function(segs, index) {
|
||||||
// Local short-cuts:
|
// Local short-cuts:
|
||||||
|
@ -421,7 +385,7 @@ var Path = PathItem.extend(/** @lends Path# */{
|
||||||
curves = this._curves,
|
curves = this._curves,
|
||||||
amount = segs.length,
|
amount = segs.length,
|
||||||
append = index == null,
|
append = index == null,
|
||||||
index = append ? segments.length : index;
|
from = append ? segments.length : index;
|
||||||
// Scan through segments to add first, convert if necessary and set
|
// Scan through segments to add first, convert if necessary and set
|
||||||
// _path and _index references on them.
|
// _path and _index references on them.
|
||||||
for (var i = 0; i < amount; i++) {
|
for (var i = 0; i < amount; i++) {
|
||||||
|
@ -431,7 +395,7 @@ var Path = PathItem.extend(/** @lends Path# */{
|
||||||
if (segment._path)
|
if (segment._path)
|
||||||
segment = segs[i] = segment.clone();
|
segment = segs[i] = segment.clone();
|
||||||
segment._path = this;
|
segment._path = this;
|
||||||
segment._index = index + i;
|
segment._index = from + i;
|
||||||
// If parts of this segment are selected, adjust the internal
|
// If parts of this segment are selected, adjust the internal
|
||||||
// _selectedSegmentState now
|
// _selectedSegmentState now
|
||||||
if (segment._selectionState)
|
if (segment._selectionState)
|
||||||
|
@ -442,20 +406,15 @@ var Path = PathItem.extend(/** @lends Path# */{
|
||||||
segments.push.apply(segments, segs);
|
segments.push.apply(segments, segs);
|
||||||
} else {
|
} else {
|
||||||
// Insert somewhere else
|
// Insert somewhere else
|
||||||
segments.splice.apply(segments, [index, 0].concat(segs));
|
segments.splice.apply(segments, [from, 0].concat(segs));
|
||||||
// Adjust the indices of the segments above.
|
// Adjust the indices of the segments above.
|
||||||
for (var i = index + amount, l = segments.length; i < l; i++)
|
for (var i = from + amount, l = segments.length; i < l; i++)
|
||||||
segments[i]._index = i;
|
segments[i]._index = i;
|
||||||
}
|
}
|
||||||
// Keep the curves list in sync all the time in case it was requested
|
// Keep the curves list in sync all the time in case it was requested
|
||||||
// already.
|
// already.
|
||||||
if (curves || segs._curves) {
|
if (curves) {
|
||||||
if (!curves)
|
var start = from,
|
||||||
curves = this._curves = [];
|
|
||||||
// We need to step one index down from the inserted segment to
|
|
||||||
// get its curve, except for the first segment.
|
|
||||||
var from = index > 0 ? index - 1 : index,
|
|
||||||
start = from,
|
|
||||||
to = Math.min(from + amount, this._countCurves());
|
to = Math.min(from + amount, this._countCurves());
|
||||||
if (segs._curves) {
|
if (segs._curves) {
|
||||||
// Reuse removed curves.
|
// Reuse removed curves.
|
||||||
|
@ -810,34 +769,91 @@ var Path = PathItem.extend(/** @lends Path# */{
|
||||||
clear: '#removeSegments',
|
clear: '#removeSegments',
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The approximate length of the path in points.
|
* Checks if any of the curves in the path have curve handles set.
|
||||||
|
*
|
||||||
|
* @return {Boolean} {@true if the path has curve handles set}
|
||||||
|
* @see Segment#hasHandles()
|
||||||
|
* @see Curve#hasHandles()
|
||||||
|
*/
|
||||||
|
hasHandles: function() {
|
||||||
|
var segments = this._segments;
|
||||||
|
for (var i = 0, l = segments.length; i < l; i++) {
|
||||||
|
if (segments[i].hasHandles())
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Clears the path's handles by setting their coordinates to zero,
|
||||||
|
* turning the path into a polygon (or a polyline if it isn't closed).
|
||||||
|
*/
|
||||||
|
clearHandles: function() {
|
||||||
|
var segments = this._segments;
|
||||||
|
for (var i = 0, l = segments.length; i < l; i++)
|
||||||
|
segments[i].clearHandles();
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The approximate length of the path.
|
||||||
*
|
*
|
||||||
* @type Number
|
* @type Number
|
||||||
* @bean
|
* @bean
|
||||||
*/
|
*/
|
||||||
getLength: function() {
|
getLength: function() {
|
||||||
if (this._length == null) {
|
if (this._length == null) {
|
||||||
var curves = this.getCurves();
|
var curves = this.getCurves(),
|
||||||
this._length = 0;
|
length = 0;
|
||||||
for (var i = 0, l = curves.length; i < l; i++)
|
for (var i = 0, l = curves.length; i < l; i++)
|
||||||
this._length += curves[i].getLength();
|
length += curves[i].getLength();
|
||||||
|
this._length = length;
|
||||||
}
|
}
|
||||||
return this._length;
|
return this._length;
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The area of the path in square points. Self-intersecting paths can
|
* The area that the path's geometry is covering. Self-intersecting paths
|
||||||
* contain sub-areas that cancel each other out.
|
* can contain sub-areas that cancel each other out.
|
||||||
*
|
*
|
||||||
* @type Number
|
* @type Number
|
||||||
* @bean
|
* @bean
|
||||||
*/
|
*/
|
||||||
getArea: function() {
|
getArea: function() {
|
||||||
var curves = this.getCurves();
|
if (this._area == null) {
|
||||||
var area = 0;
|
var segments = this._segments,
|
||||||
for (var i = 0, l = curves.length; i < l; i++)
|
count = segments.length,
|
||||||
area += curves[i].getArea();
|
last = count - 1,
|
||||||
return area;
|
area = 0;
|
||||||
|
for (var i = 0, l = this._closed ? count : last; i < l; i++) {
|
||||||
|
area += Curve.getArea(Curve.getValues(
|
||||||
|
segments[i], segments[i < last ? i + 1 : 0]));
|
||||||
|
}
|
||||||
|
this._area = area;
|
||||||
|
}
|
||||||
|
return this._area;
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Specifies whether the path is oriented clock-wise.
|
||||||
|
*
|
||||||
|
* @type Boolean
|
||||||
|
* @bean
|
||||||
|
*/
|
||||||
|
isClockwise: function() {
|
||||||
|
if (this._clockwise !== undefined)
|
||||||
|
return this._clockwise;
|
||||||
|
return this.getArea() >= 0;
|
||||||
|
},
|
||||||
|
|
||||||
|
setClockwise: function(clockwise) {
|
||||||
|
// Only revers the path if its clockwise orientation is not the same
|
||||||
|
// as what it is now demanded to be.
|
||||||
|
// On-the-fly conversion to boolean:
|
||||||
|
if (this.isClockwise() != (clockwise = !!clockwise))
|
||||||
|
this.reverse();
|
||||||
|
// Reverse only flips _clockwise state if it was already set, so let's
|
||||||
|
// always set this here now.
|
||||||
|
this._clockwise = clockwise;
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1009,10 +1025,9 @@ var Path = PathItem.extend(/** @lends Path# */{
|
||||||
reduce: function() {
|
reduce: function() {
|
||||||
var curves = this.getCurves();
|
var curves = this.getCurves();
|
||||||
for (var i = curves.length - 1; i >= 0; i--) {
|
for (var i = curves.length - 1; i >= 0; i--) {
|
||||||
var curve = curves[i],
|
var curve = curves[i];
|
||||||
next;
|
if (!curve.hasHandles() && (curve.getLength() === 0
|
||||||
if (curve.isLinear() && (curve.getLength() === 0
|
|| curve.isCollinear(curve.getNext())))
|
||||||
|| (next = curve.getNext()) && curve.isCollinear(next)))
|
|
||||||
curve.remove();
|
curve.remove();
|
||||||
}
|
}
|
||||||
return this;
|
return this;
|
||||||
|
@ -1174,7 +1189,8 @@ var Path = PathItem.extend(/** @lends Path# */{
|
||||||
*
|
*
|
||||||
* @param {Number} index the index of the curve in the {@link Path#curves}
|
* @param {Number} index the index of the curve in the {@link Path#curves}
|
||||||
* array at which to split
|
* array at which to split
|
||||||
* @param {Number} parameter the parameter at which the curve will be split
|
* @param {Number} parameter the curve-time parameter at which the curve
|
||||||
|
* will be split
|
||||||
* @return {Path} the newly created path after splitting, if any
|
* @return {Path} the newly created path after splitting, if any
|
||||||
*/
|
*/
|
||||||
split: function(index, parameter) {
|
split: function(index, parameter) {
|
||||||
|
@ -1191,7 +1207,7 @@ var Path = PathItem.extend(/** @lends Path# */{
|
||||||
index = arg.index;
|
index = arg.index;
|
||||||
parameter = arg.parameter;
|
parameter = arg.parameter;
|
||||||
}
|
}
|
||||||
var tMin = /*#=*/Numerical.TOLERANCE,
|
var tMin = /*#=*/Numerical.CURVETIME_EPSILON,
|
||||||
tMax = 1 - tMin;
|
tMax = 1 - tMin;
|
||||||
if (parameter >= tMax) {
|
if (parameter >= tMax) {
|
||||||
// t == 1 is the same as t == 0 and index ++
|
// t == 1 is the same as t == 0 and index ++
|
||||||
|
@ -1235,29 +1251,6 @@ var Path = PathItem.extend(/** @lends Path# */{
|
||||||
return null;
|
return null;
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
|
||||||
* Specifies whether the path is oriented clock-wise.
|
|
||||||
*
|
|
||||||
* @type Boolean
|
|
||||||
* @bean
|
|
||||||
*/
|
|
||||||
isClockwise: function() {
|
|
||||||
if (this._clockwise !== undefined)
|
|
||||||
return this._clockwise;
|
|
||||||
return Path.isClockwise(this._segments);
|
|
||||||
},
|
|
||||||
|
|
||||||
setClockwise: function(clockwise) {
|
|
||||||
// Only revers the path if its clockwise orientation is not the same
|
|
||||||
// as what it is now demanded to be.
|
|
||||||
// On-the-fly conversion to boolean:
|
|
||||||
if (this.isClockwise() != (clockwise = !!clockwise))
|
|
||||||
this.reverse();
|
|
||||||
// Reverse only flips _clockwise state if it was already set, so let's
|
|
||||||
// always set this here now.
|
|
||||||
this._clockwise = clockwise;
|
|
||||||
},
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Reverses the orientation of the path, by reversing all its segments.
|
* Reverses the orientation of the path, by reversing all its segments.
|
||||||
*/
|
*/
|
||||||
|
@ -1414,15 +1407,47 @@ var Path = PathItem.extend(/** @lends Path# */{
|
||||||
topCenter;
|
topCenter;
|
||||||
|
|
||||||
function isCollinear(i, j) {
|
function isCollinear(i, j) {
|
||||||
return segments[i].isCollinear(segments[j]);
|
var seg1 = segments[i],
|
||||||
|
seg2 = seg1.getNext(),
|
||||||
|
seg3 = segments[j],
|
||||||
|
seg4 = seg3.getNext();
|
||||||
|
return seg1._handleOut.isZero() && seg2._handleIn.isZero()
|
||||||
|
&& seg3._handleOut.isZero() && seg4._handleIn.isZero()
|
||||||
|
&& seg2._point.subtract(seg1._point).isCollinear(
|
||||||
|
seg4._point.subtract(seg3._point));
|
||||||
}
|
}
|
||||||
|
|
||||||
function isOrthogonal(i) {
|
function isOrthogonal(i) {
|
||||||
return segments[i].isOrthogonal();
|
var seg2 = segments[i],
|
||||||
|
seg1 = seg2.getPrevious(),
|
||||||
|
seg3 = seg2.getNext();
|
||||||
|
return seg1._handleOut.isZero() && seg2._handleIn.isZero()
|
||||||
|
&& seg2._handleOut.isZero() && seg3._handleIn.isZero()
|
||||||
|
&& seg2._point.subtract(seg1._point).isOrthogonal(
|
||||||
|
seg3._point.subtract(seg2._point));
|
||||||
}
|
}
|
||||||
|
|
||||||
function isArc(i) {
|
function isArc(i) {
|
||||||
return segments[i].isOrthogonalArc();
|
var seg1 = segments[i],
|
||||||
|
seg2 = seg1.getNext(),
|
||||||
|
handle1 = seg1._handleOut,
|
||||||
|
handle2 = seg2._handleIn,
|
||||||
|
kappa = /*#=*/Numerical.KAPPA;
|
||||||
|
// Look at handle length and the distance to the imaginary corner
|
||||||
|
// point and see if it their relation is kappa.
|
||||||
|
if (handle1.isOrthogonal(handle2)) {
|
||||||
|
var pt1 = seg1._point,
|
||||||
|
pt2 = seg2._point,
|
||||||
|
// Find the corner point by intersecting the lines described
|
||||||
|
// by both handles:
|
||||||
|
corner = new Line(pt1, handle1, true).intersect(
|
||||||
|
new Line(pt2, handle2, true), true);
|
||||||
|
return corner && Numerical.isZero(handle1.getLength() /
|
||||||
|
corner.subtract(pt1).getLength() - kappa)
|
||||||
|
&& Numerical.isZero(handle2.getLength() /
|
||||||
|
corner.subtract(pt2).getLength() - kappa);
|
||||||
|
}
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
function getDistance(i, j) {
|
function getDistance(i, j) {
|
||||||
|
@ -1974,7 +1999,7 @@ var Path = PathItem.extend(/** @lends Path# */{
|
||||||
* Returns the nearest location on the path to the specified point.
|
* Returns the nearest location on the path to the specified point.
|
||||||
*
|
*
|
||||||
* @function
|
* @function
|
||||||
* @param point {Point} the point for which we search the nearest location
|
* @param {Point} point the point for which we search the nearest location
|
||||||
* @return {CurveLocation} the location on the path that's the closest to
|
* @return {CurveLocation} the location on the path that's the closest to
|
||||||
* the specified point
|
* the specified point
|
||||||
*/
|
*/
|
||||||
|
@ -1997,7 +2022,7 @@ var Path = PathItem.extend(/** @lends Path# */{
|
||||||
* Returns the nearest point on the path to the specified point.
|
* Returns the nearest point on the path to the specified point.
|
||||||
*
|
*
|
||||||
* @function
|
* @function
|
||||||
* @param point {Point} the point for which we search the nearest point
|
* @param {Point} point the point for which we search the nearest point
|
||||||
* @return {Point} the point on the path that's the closest to the specified
|
* @return {Point} the point on the path that's the closest to the specified
|
||||||
* point
|
* point
|
||||||
*
|
*
|
||||||
|
@ -2028,8 +2053,8 @@ var Path = PathItem.extend(/** @lends Path# */{
|
||||||
getNearestPoint: function(/* point */) {
|
getNearestPoint: function(/* point */) {
|
||||||
return this.getNearestLocation.apply(this, arguments).getPoint();
|
return this.getNearestLocation.apply(this, arguments).getPoint();
|
||||||
}
|
}
|
||||||
}), new function() { // Scope for drawing
|
}),
|
||||||
|
new function() { // Scope for drawing
|
||||||
// Note that in the code below we're often accessing _x and _y on point
|
// Note that in the code below we're often accessing _x and _y on point
|
||||||
// objects that were read from segments. This is because the SegmentPoint
|
// objects that were read from segments. This is because the SegmentPoint
|
||||||
// class overrides the plain x / y properties with getter / setters and
|
// class overrides the plain x / y properties with getter / setters and
|
||||||
|
@ -2228,8 +2253,8 @@ var Path = PathItem.extend(/** @lends Path# */{
|
||||||
drawHandles(ctx, this._segments, matrix, paper.settings.handleSize);
|
drawHandles(ctx, this._segments, matrix, paper.settings.handleSize);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
}, new function() { // Path Smoothing
|
},
|
||||||
|
new function() { // Path Smoothing
|
||||||
/**
|
/**
|
||||||
* Solves a tri-diagonal system for one of coordinates (x or y) of first
|
* Solves a tri-diagonal system for one of coordinates (x or y) of first
|
||||||
* bezier control points.
|
* bezier control points.
|
||||||
|
@ -2354,7 +2379,8 @@ var Path = PathItem.extend(/** @lends Path# */{
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
}, new function() { // PostScript-style drawing commands
|
},
|
||||||
|
new function() { // PostScript-style drawing commands
|
||||||
/**
|
/**
|
||||||
* Helper method that returns the current segment and checks if a moveTo()
|
* Helper method that returns the current segment and checks if a moveTo()
|
||||||
* command is required first.
|
* command is required first.
|
||||||
|
@ -2477,7 +2503,6 @@ var Path = PathItem.extend(/** @lends Path# */{
|
||||||
x = pt.x,
|
x = pt.x,
|
||||||
y = pt.y,
|
y = pt.y,
|
||||||
abs = Math.abs,
|
abs = Math.abs,
|
||||||
epsilon = /*#=*/Numerical.EPSILON,
|
|
||||||
rx = abs(radius.width),
|
rx = abs(radius.width),
|
||||||
ry = abs(radius.height),
|
ry = abs(radius.height),
|
||||||
rxSq = rx * rx,
|
rxSq = rx * rx,
|
||||||
|
@ -2494,7 +2519,7 @@ var Path = PathItem.extend(/** @lends Path# */{
|
||||||
}
|
}
|
||||||
factor = (rxSq * rySq - rxSq * ySq - rySq * xSq) /
|
factor = (rxSq * rySq - rxSq * ySq - rySq * xSq) /
|
||||||
(rxSq * ySq + rySq * xSq);
|
(rxSq * ySq + rySq * xSq);
|
||||||
if (abs(factor) < epsilon)
|
if (abs(factor) < /*#=*/Numerical.EPSILON)
|
||||||
factor = 0;
|
factor = 0;
|
||||||
if (factor < 0)
|
if (factor < 0)
|
||||||
throw new Error(
|
throw new Error(
|
||||||
|
@ -2664,21 +2689,6 @@ var Path = PathItem.extend(/** @lends Path# */{
|
||||||
|
|
||||||
// Mess with indentation in order to get more line-space below:
|
// Mess with indentation in order to get more line-space below:
|
||||||
statics: {
|
statics: {
|
||||||
/**
|
|
||||||
* Determines whether the segments describe a path in clockwise or counter-
|
|
||||||
* clockwise orientation.
|
|
||||||
*
|
|
||||||
* @private
|
|
||||||
*/
|
|
||||||
isClockwise: function(segments) {
|
|
||||||
var sum = 0;
|
|
||||||
// TODO: Check if this works correctly for all open paths.
|
|
||||||
for (var i = 0, l = segments.length; i < l; i++)
|
|
||||||
sum += Curve.getEdgeSum(Curve.getValues(
|
|
||||||
segments[i], segments[i + 1 < l ? i + 1 : 0]));
|
|
||||||
return sum > 0;
|
|
||||||
},
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the bounding rectangle of the item excluding stroke width.
|
* Returns the bounding rectangle of the item excluding stroke width.
|
||||||
*
|
*
|
||||||
|
|
File diff suppressed because it is too large
Load diff
|
@ -32,10 +32,12 @@ var PathItem = Item.extend(/** @lends PathItem# */{
|
||||||
* supported.
|
* supported.
|
||||||
*
|
*
|
||||||
* @param {PathItem} path the other item to find the intersections with
|
* @param {PathItem} path the other item to find the intersections with
|
||||||
* @param {Boolean} [sorted=false] specifies whether the returned
|
* @param {Function} [include] a callback function that can be used to
|
||||||
* {@link CurveLocation} objects should be sorted by path and offset
|
* filter out undesired locations right while they are collected.
|
||||||
|
* When defined, it shall return {@true to include a location}.
|
||||||
* @return {CurveLocation[]} the locations of all intersection between the
|
* @return {CurveLocation[]} the locations of all intersection between the
|
||||||
* paths
|
* paths
|
||||||
|
* @see #getCrossings(path)
|
||||||
* @example {@paperscript} // Finding the intersections between two paths
|
* @example {@paperscript} // Finding the intersections between two paths
|
||||||
* var path = new Path.Rectangle(new Point(30, 25), new Size(50, 50));
|
* var path = new Path.Rectangle(new Point(30, 25), new Size(50, 50));
|
||||||
* path.strokeColor = 'black';
|
* path.strokeColor = 'black';
|
||||||
|
@ -59,88 +61,100 @@ var PathItem = Item.extend(/** @lends PathItem# */{
|
||||||
* }
|
* }
|
||||||
* }
|
* }
|
||||||
*/
|
*/
|
||||||
getIntersections: function(path, _matrix) {
|
getIntersections: function(path, include, _matrix, _returnFirst) {
|
||||||
// NOTE: For self-intersection, path is null. This means you can also
|
// NOTE: For self-intersection, path is null. This means you can also
|
||||||
// just call path.getIntersections() without an argument to get self
|
// just call path.getIntersections() without an argument to get self
|
||||||
// intersections.
|
// intersections.
|
||||||
// NOTE: The hidden argument _matrix is used internally to override the
|
// NOTE: The hidden argument _matrix is used internally to override the
|
||||||
// passed path's transformation matrix.
|
// passed path's transformation matrix.
|
||||||
return Curve.filterIntersections(this._getIntersections(
|
var self = this === path || !path, // self-intersections?
|
||||||
this !== path ? path : null, _matrix, []));
|
|
||||||
},
|
|
||||||
|
|
||||||
_getIntersections: function(path, matrix, locations, returnFirst) {
|
|
||||||
var curves1 = this.getCurves(),
|
|
||||||
curves2 = path ? path.getCurves() : curves1,
|
|
||||||
matrix1 = this._matrix.orNullIfIdentity(),
|
matrix1 = this._matrix.orNullIfIdentity(),
|
||||||
matrix2 = path ? (matrix || path._matrix).orNullIfIdentity()
|
matrix2 = self ? matrix1
|
||||||
: matrix1,
|
: (_matrix || path._matrix).orNullIfIdentity();
|
||||||
length1 = curves1.length,
|
|
||||||
length2 = path ? curves2.length : length1,
|
|
||||||
values2 = [],
|
|
||||||
tMin = /*#=*/Numerical.TOLERANCE,
|
|
||||||
tMax = 1 - tMin;
|
|
||||||
// First check the bounds of the two paths. If they don't intersect,
|
// First check the bounds of the two paths. If they don't intersect,
|
||||||
// we don't need to iterate through their curves.
|
// we don't need to iterate through their curves.
|
||||||
if (path && !this.getBounds(matrix1).touches(path.getBounds(matrix2)))
|
if (!self && !this.getBounds(matrix1).touches(path.getBounds(matrix2)))
|
||||||
return [];
|
return [];
|
||||||
|
var curves1 = this.getCurves(),
|
||||||
|
curves2 = self ? curves1 : path.getCurves(),
|
||||||
|
length1 = curves1.length,
|
||||||
|
length2 = self ? length1 : curves2.length,
|
||||||
|
values2 = [],
|
||||||
|
arrays = [],
|
||||||
|
locations,
|
||||||
|
path;
|
||||||
|
// Cache values for curves2 as we re-iterate them for each in curves1.
|
||||||
for (var i = 0; i < length2; i++)
|
for (var i = 0; i < length2; i++)
|
||||||
values2[i] = curves2[i].getValues(matrix2);
|
values2[i] = curves2[i].getValues(matrix2);
|
||||||
for (var i = 0; i < length1; i++) {
|
for (var i = 0; i < length1; i++) {
|
||||||
var curve1 = curves1[i],
|
var curve1 = curves1[i],
|
||||||
values1 = path ? curve1.getValues(matrix1) : values2[i];
|
values1 = self ? values2[i] : curve1.getValues(matrix1),
|
||||||
if (!path) {
|
path1 = curve1.getPath();
|
||||||
// First check for self-intersections within the same curve
|
// NOTE: Due to the nature of Curve._getIntersections(), we need to
|
||||||
var seg1 = curve1.getSegment1(),
|
// use separate location arrays per path1, to make sure the
|
||||||
seg2 = curve1.getSegment2(),
|
// circularity checks are not getting confused by locations on
|
||||||
h1 = seg1._handleOut,
|
// separate paths. We are flattening the separate arrays at the end.
|
||||||
h2 = seg2._handleIn;
|
if (path1 !== path) {
|
||||||
// Check if extended handles of endpoints of this curve
|
path = path1;
|
||||||
// intersects each other. We cannot have a self intersection
|
locations = [];
|
||||||
// within this curve if they don't intersect due to convex-hull
|
arrays.push(locations);
|
||||||
// property.
|
}
|
||||||
if (new Line(seg1._point.subtract(h1), h1.multiply(2), true)
|
if (self) {
|
||||||
.intersect(new Line(seg2._point.subtract(h2),
|
// First check for self-intersections within the same curve.
|
||||||
h2.multiply(2), true), false)) {
|
Curve._getSelfIntersection(values1, curve1, locations, {
|
||||||
// Self intersecting is found by dividing the curve in two
|
include: include,
|
||||||
// and and then applying the normal curve intersection code.
|
// Only possible if there is only one closed curve:
|
||||||
var parts = Curve.subdivide(values1);
|
startConnected: length1 === 1 &&
|
||||||
Curve.getIntersections(
|
curve1.getPoint1().equals(curve1.getPoint2())
|
||||||
parts[0], parts[1], curve1, curve1, locations,
|
});
|
||||||
function(loc) {
|
|
||||||
if (loc._parameter <= tMax) {
|
|
||||||
// Since the curve was split above, we need to
|
|
||||||
// adjust the parameters for both locations.
|
|
||||||
loc._parameter /= 2;
|
|
||||||
loc._parameter2 = 0.5 + loc._parameter2 / 2;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
// Check for intersections with other curves. For self intersection,
|
// Check for intersections with other curves. For self intersection,
|
||||||
// we can start at i + 1 instead of 0
|
// we can start at i + 1 instead of 0
|
||||||
for (var j = path ? 0 : i + 1; j < length2; j++) {
|
for (var j = self ? i + 1 : 0; j < length2; j++) {
|
||||||
// There might be already one location from the above
|
// There might be already one location from the above
|
||||||
// self-intersection check:
|
// self-intersection check:
|
||||||
if (returnFirst && locations.length)
|
if (_returnFirst && locations.length)
|
||||||
break;
|
return locations;
|
||||||
Curve.getIntersections(
|
var curve2 = curves2[j];
|
||||||
values1, values2[j], curve1, curves2[j], locations,
|
// Avoid end point intersections on consecutive curves when
|
||||||
// Avoid end point intersections on consecutive curves when
|
// self intersecting.
|
||||||
// self intersecting.
|
Curve._getIntersections(
|
||||||
!path && (j === i + 1 || j === length2 - 1 && i === 0)
|
values1, values2[j], curve1, curve2, locations,
|
||||||
&& function(loc) {
|
{
|
||||||
var t = loc._parameter;
|
include: include,
|
||||||
return t >= tMin && t <= tMax;
|
// Do not compare indices here to determine connection,
|
||||||
}
|
// since one array of curves can contain curves from
|
||||||
|
// separate sup-paths of a compound path.
|
||||||
|
startConnected: self && curve1.getPrevious() === curve2,
|
||||||
|
endConnected: self && curve1.getNext() === curve2
|
||||||
|
}
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// Now flatten the list of location arrays to one array and return it.
|
||||||
|
locations = [];
|
||||||
|
for (var i = 0, l = arrays.length; i < l; i++) {
|
||||||
|
locations.push.apply(locations, arrays[i]);
|
||||||
|
}
|
||||||
return locations;
|
return locations;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns all crossings between two {@link PathItem} items as an array
|
||||||
|
* of {@link CurveLocation} objects. {@link CompoundPath} items are also
|
||||||
|
* supported.
|
||||||
|
* Crossings are intersections where the paths actually are crossing each
|
||||||
|
* other, as opposed to simply touching.
|
||||||
|
*
|
||||||
|
* @param {PathItem} path the other item to find the crossings with
|
||||||
|
* @see #getIntersections(path)
|
||||||
|
*/
|
||||||
|
getCrossings: function(path) {
|
||||||
|
return this.getIntersections(path, function(inter) {
|
||||||
|
return inter.isCrossing();
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
_asPathItem: function() {
|
_asPathItem: function() {
|
||||||
// See Item#_asPathItem()
|
// See Item#_asPathItem()
|
||||||
return this;
|
return this;
|
||||||
|
|
|
@ -58,7 +58,7 @@ var PathIterator = Base.extend({
|
||||||
// appears to offer a good trade-off between speed and
|
// appears to offer a good trade-off between speed and
|
||||||
// precision for display purposes.
|
// precision for display purposes.
|
||||||
&& !Curve.isFlatEnough(curve, tolerance || 0.25)) {
|
&& !Curve.isFlatEnough(curve, tolerance || 0.25)) {
|
||||||
var split = Curve.subdivide(curve),
|
var split = Curve.subdivide(curve, 0.5),
|
||||||
halfT = (minT + maxT) / 2;
|
halfT = (minT + maxT) / 2;
|
||||||
// Recursively subdivide and compute parts again.
|
// Recursively subdivide and compute parts again.
|
||||||
computeParts(split[0], index, minT, halfT);
|
computeParts(split[0], index, minT, halfT);
|
||||||
|
|
|
@ -119,7 +119,7 @@ var Segment = Base.extend(/** @lends Segment# */{
|
||||||
// Nothing
|
// Nothing
|
||||||
} else if (count === 1) {
|
} else if (count === 1) {
|
||||||
// Note: This copies from existing segments through accessors.
|
// Note: This copies from existing segments through accessors.
|
||||||
if (arg0.point) {
|
if ('point' in arg0) {
|
||||||
point = arg0.point;
|
point = arg0.point;
|
||||||
handleIn = arg0.handleIn;
|
handleIn = arg0.handleIn;
|
||||||
handleOut = arg0.handleOut;
|
handleOut = arg0.handleOut;
|
||||||
|
@ -145,9 +145,10 @@ var Segment = Base.extend(/** @lends Segment# */{
|
||||||
},
|
},
|
||||||
|
|
||||||
_serialize: function(options) {
|
_serialize: function(options) {
|
||||||
// If it is straight, only serialize point, otherwise handles too.
|
// If it is has no handles, only serialize point, otherwise handles too.
|
||||||
return Base.serialize(this.isStraight() ? this._point
|
return Base.serialize(this.hasHandles()
|
||||||
: [this._point, this._handleIn, this._handleOut],
|
? [this._point, this._handleIn, this._handleOut]
|
||||||
|
: this._point,
|
||||||
options, true);
|
options, true);
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -229,87 +230,27 @@ var Segment = Base.extend(/** @lends Segment# */{
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Checks whether the segment has curve handles defined, meaning it is not
|
* Checks if the segment has any curve handles set.
|
||||||
* a straight segment.
|
|
||||||
*
|
*
|
||||||
* @return {Boolean} {@true if the segment has handles defined}
|
* @return {Boolean} {@true if the segment has handles set}
|
||||||
|
* @see Segment#getHandleIn()
|
||||||
|
* @see Segment#getHandleOut()
|
||||||
* @see Curve#hasHandles()
|
* @see Curve#hasHandles()
|
||||||
* @see Path#hasHandles()
|
* @see Path#hasHandles()
|
||||||
*/
|
*/
|
||||||
hasHandles: function() {
|
hasHandles: function() {
|
||||||
return !this.isStraight();
|
return !this._handleIn.isZero() || !this._handleOut.isZero();
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Checks whether the segment is straight, meaning it has no curve handles
|
* Clears the segment's handles by setting their coordinates to zero,
|
||||||
* defined. If two straight segments follow each each other in a path, the
|
* turning the segment into a corner.
|
||||||
* curve between them will appear as a straight line.
|
|
||||||
* Note that this is not the same as {@link #isLinear()}, which performs a
|
|
||||||
* full linearity check that includes handles running collinear to the line
|
|
||||||
* direction.
|
|
||||||
*
|
|
||||||
* @return {Boolean} {@true if the segment is straight}
|
|
||||||
* @see Curve#isStraight()
|
|
||||||
*/
|
*/
|
||||||
isStraight: function() {
|
clearHandles: function() {
|
||||||
return this._handleIn.isZero() && this._handleOut.isZero();
|
this._handleIn.set(0, 0);
|
||||||
|
this._handleOut.set(0, 0);
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
|
||||||
* Checks if the curve that starts in this segment appears as a line. This
|
|
||||||
* can mean that it has no handles defined, or that the handles run
|
|
||||||
* collinear with the line.
|
|
||||||
*
|
|
||||||
* @return {Boolean} {@true if the curve is linear}
|
|
||||||
* @see Curve#isLinear()
|
|
||||||
* @see Path#isLinear()
|
|
||||||
*/
|
|
||||||
isLinear: function() {
|
|
||||||
return Segment.isLinear(this, this.getNext());
|
|
||||||
},
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Checks if the the two segments are the beginning of two lines that are
|
|
||||||
* collinear, meaning they run in parallel.
|
|
||||||
*
|
|
||||||
* @param {Segment} the other segment to check against
|
|
||||||
* @return {Boolean} {@true if the two lines are collinear}
|
|
||||||
* @see Curve#isCollinear(curve)
|
|
||||||
*/
|
|
||||||
isCollinear: function(segment) {
|
|
||||||
return Segment.isCollinear(this, this.getNext(),
|
|
||||||
segment, segment.getNext());
|
|
||||||
},
|
|
||||||
|
|
||||||
// TODO: Remove version with typo after a while (deprecated June 2015)
|
|
||||||
isColinear: '#isCollinear',
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Checks if the segment is connecting two lines that are orthogonal,
|
|
||||||
* meaning they connect at an 90° angle.
|
|
||||||
*
|
|
||||||
* @return {Boolean} {@true if the two lines connected by this segment are
|
|
||||||
* orthogonal}
|
|
||||||
*/
|
|
||||||
isOrthogonal: function() {
|
|
||||||
return Segment.isOrthogonal(this.getPrevious(), this, this.getNext());
|
|
||||||
},
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Checks if the segment is the beginning of an orthogonal arc, as used in
|
|
||||||
* the construction of circles and ellipses.
|
|
||||||
*
|
|
||||||
* @return {Boolean} {@true if the segment is the beginning of an orthogonal
|
|
||||||
* arc}
|
|
||||||
* @see Curve#isOrthogonalArc()
|
|
||||||
*/
|
|
||||||
isOrthogonalArc: function() {
|
|
||||||
return Segment.isOrthogonalArc(this, this.getNext());
|
|
||||||
},
|
|
||||||
|
|
||||||
// TODO: Remove a while (deprecated August 2015)
|
|
||||||
isArc: '#isOrthogonalArc',
|
|
||||||
|
|
||||||
_selectionState: 0,
|
_selectionState: 0,
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -452,10 +393,45 @@ var Segment = Base.extend(/** @lends Segment# */{
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the reversed the segment, without modifying the segment itself.
|
* Checks if the this is the first segment in the {@link Path#segments}
|
||||||
|
* array.
|
||||||
|
*
|
||||||
|
* @return {Boolean} {@true if this is the first segment}
|
||||||
|
*/
|
||||||
|
isFirst: function() {
|
||||||
|
return this._index === 0;
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the this is the last segment in the {@link Path#segments}
|
||||||
|
* array.
|
||||||
|
*
|
||||||
|
* @return {Boolean} {@true if this is the last segment}
|
||||||
|
*/
|
||||||
|
isLast: function() {
|
||||||
|
var path = this._path;
|
||||||
|
return path && this._index === path._segments.length - 1 || false;
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reverses the {@link #handleIn} and {@link #handleOut} vectors of this
|
||||||
|
* segment. Note: the actual segment is modified, no copy is created.
|
||||||
* @return {Segment} the reversed segment
|
* @return {Segment} the reversed segment
|
||||||
*/
|
*/
|
||||||
reverse: function() {
|
reverse: function() {
|
||||||
|
var handleIn = this._handleIn,
|
||||||
|
handleOut = this._handleOut,
|
||||||
|
inX = handleIn._x,
|
||||||
|
inY = handleIn._y;
|
||||||
|
handleIn.set(handleOut._x, handleOut._y);
|
||||||
|
handleOut.set(inX, inY);
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the reversed the segment, without modifying the segment itself.
|
||||||
|
* @return {Segment} the reversed segment
|
||||||
|
*/
|
||||||
|
reversed: function() {
|
||||||
return new Segment(this._point, this._handleOut, this._handleIn);
|
return new Segment(this._point, this._handleOut, this._handleIn);
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -564,53 +540,5 @@ var Segment = Base.extend(/** @lends Segment# */{
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return coords;
|
return coords;
|
||||||
},
|
|
||||||
|
|
||||||
statics: {
|
|
||||||
// These statics are shared between Segment and Curve, for versions of
|
|
||||||
// these methods that are implemented in both places.
|
|
||||||
|
|
||||||
isLinear: function(seg1, seg2) {
|
|
||||||
var l = seg2._point.subtract(seg1._point);
|
|
||||||
return l.isCollinear(seg1._handleOut)
|
|
||||||
&& l.isCollinear(seg2._handleIn);
|
|
||||||
},
|
|
||||||
|
|
||||||
isCollinear: function(seg1, seg2, seg3, seg4) {
|
|
||||||
// TODO: This assumes isStraight(), while isLinear() allows handles!
|
|
||||||
return seg1._handleOut.isZero() && seg2._handleIn.isZero()
|
|
||||||
&& seg3._handleOut.isZero() && seg4._handleIn.isZero()
|
|
||||||
&& seg2._point.subtract(seg1._point).isCollinear(
|
|
||||||
seg4._point.subtract(seg3._point));
|
|
||||||
},
|
|
||||||
|
|
||||||
isOrthogonal: function(seg1, seg2, seg3) {
|
|
||||||
// TODO: This assumes isStraight(), while isLinear() allows handles!
|
|
||||||
return seg1._handleOut.isZero() && seg2._handleIn.isZero()
|
|
||||||
&& seg2._handleOut.isZero() && seg3._handleIn.isZero()
|
|
||||||
&& seg2._point.subtract(seg1._point).isOrthogonal(
|
|
||||||
seg3._point.subtract(seg2._point));
|
|
||||||
},
|
|
||||||
|
|
||||||
isOrthogonalArc: function(seg1, seg2) {
|
|
||||||
var handle1 = seg1._handleOut,
|
|
||||||
handle2 = seg2._handleIn,
|
|
||||||
kappa = /*#=*/Numerical.KAPPA;
|
|
||||||
// Look at handle length and the distance to the imaginary corner
|
|
||||||
// point and see if it their relation is kappa.
|
|
||||||
if (handle1.isOrthogonal(handle2)) {
|
|
||||||
var pt1 = seg1._point,
|
|
||||||
pt2 = seg2._point,
|
|
||||||
// Find the corner point by intersecting the lines described
|
|
||||||
// by both handles:
|
|
||||||
corner = new Line(pt1, handle1, true).intersect(
|
|
||||||
new Line(pt2, handle2, true), true);
|
|
||||||
return corner && Numerical.isZero(handle1.getLength() /
|
|
||||||
corner.subtract(pt1).getLength() - kappa)
|
|
||||||
&& Numerical.isZero(handle2.getLength() /
|
|
||||||
corner.subtract(pt2).getLength() - kappa);
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
},
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
|
@ -304,7 +304,7 @@ var Project = PaperScopeItem.extend(/** @lends Project# */{
|
||||||
* and may contain a combination of the following values:
|
* and may contain a combination of the following values:
|
||||||
*
|
*
|
||||||
* @option [options.tolerance={@link PaperScope#settings}.hitTolerance]
|
* @option [options.tolerance={@link PaperScope#settings}.hitTolerance]
|
||||||
* {Number} the tolerance of the hit-test in points
|
* {Number} the tolerance of the hit-test
|
||||||
* @option options.class {Function} only hit-test again a certain item class
|
* @option options.class {Function} only hit-test again a certain item class
|
||||||
* and its sub-classes: {@code Group, Layer, Path, CompoundPath,
|
* and its sub-classes: {@code Group, Layer, Path, CompoundPath,
|
||||||
* Shape, Raster, PlacedSymbol, PointText}, etc
|
* Shape, Raster, PlacedSymbol, PointText}, etc
|
||||||
|
|
|
@ -133,7 +133,7 @@ var Symbol = Base.extend(/** @lends Symbol# */{
|
||||||
/**
|
/**
|
||||||
* Places in instance of the symbol in the project.
|
* Places in instance of the symbol in the project.
|
||||||
*
|
*
|
||||||
* @param [position] The position of the placed symbol
|
* @param {Point} [position] the position of the placed symbol
|
||||||
* @return {PlacedSymbol}
|
* @return {PlacedSymbol}
|
||||||
*/
|
*/
|
||||||
place: function(position) {
|
place: function(position) {
|
||||||
|
|
|
@ -1112,7 +1112,8 @@ var Color = Base.extend(new function() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}, new function() {
|
},
|
||||||
|
new function() {
|
||||||
var operators = {
|
var operators = {
|
||||||
add: function(a, b) {
|
add: function(a, b) {
|
||||||
return a + b;
|
return a + b;
|
||||||
|
|
|
@ -309,7 +309,7 @@ var Style = Base.extend(new function() {
|
||||||
/**
|
/**
|
||||||
* @private
|
* @private
|
||||||
* @bean
|
* @bean
|
||||||
* @deprecated use {@link #getFontFamily()} instead.
|
* @deprecated use {@link #fontFamily} instead.
|
||||||
*/
|
*/
|
||||||
getFont: '#getFontFamily',
|
getFont: '#getFontFamily',
|
||||||
setFont: '#setFontFamily',
|
setFont: '#setFontFamily',
|
||||||
|
@ -582,7 +582,7 @@ var Style = Base.extend(new function() {
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The font size of text content, as {@Number} in pixels, or as {@String}
|
* The font size of text content, as a number in pixels, or as a string
|
||||||
* with optional units {@code 'px'}, {@code 'pt'} and {@code 'em'}.
|
* with optional units {@code 'px'}, {@code 'pt'} and {@code 'em'}.
|
||||||
*
|
*
|
||||||
* @name Style#fontSize
|
* @name Style#fontSize
|
||||||
|
@ -592,12 +592,12 @@ var Style = Base.extend(new function() {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
* The font-family to be used in text content, as one {@String}.
|
* The font-family to be used in text content, as one string.
|
||||||
* @deprecated use {@link #fontFamily} instead.
|
|
||||||
*
|
*
|
||||||
* @name Style#font
|
* @name Style#font
|
||||||
* @default 'sans-serif'
|
* @default 'sans-serif'
|
||||||
* @type String
|
* @type String
|
||||||
|
* @deprecated use {@link #fontFamily} instead.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -120,7 +120,7 @@ var TextItem = Item.extend(/** @lends TextItem# */{
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The font size of text content, as {@Number} in pixels, or as {@String}
|
* The font size of text content, as a number in pixels, or as a string
|
||||||
* with optional units {@code 'px'}, {@code 'pt'} and {@code 'em'}.
|
* with optional units {@code 'px'}, {@code 'pt'} and {@code 'em'}.
|
||||||
*
|
*
|
||||||
* @name TextItem#fontSize
|
* @name TextItem#fontSize
|
||||||
|
@ -130,7 +130,7 @@ var TextItem = Item.extend(/** @lends TextItem# */{
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
* The font-family to be used in text content, as one {@String}.
|
* The font-family to be used in text content, as one string.
|
||||||
* @deprecated use {@link #fontFamily} instead.
|
* @deprecated use {@link #fontFamily} instead.
|
||||||
*
|
*
|
||||||
* @name TextItem#font
|
* @name TextItem#font
|
||||||
|
@ -159,7 +159,7 @@ var TextItem = Item.extend(/** @lends TextItem# */{
|
||||||
/**
|
/**
|
||||||
* @private
|
* @private
|
||||||
* @bean
|
* @bean
|
||||||
* @deprecated use {@link #getStyle()} instead.
|
* @deprecated use {@link #style} instead.
|
||||||
*/
|
*/
|
||||||
getCharacterStyle: '#getStyle',
|
getCharacterStyle: '#getStyle',
|
||||||
setCharacterStyle: '#setStyle',
|
setCharacterStyle: '#setStyle',
|
||||||
|
@ -167,7 +167,7 @@ var TextItem = Item.extend(/** @lends TextItem# */{
|
||||||
/**
|
/**
|
||||||
* @private
|
* @private
|
||||||
* @bean
|
* @bean
|
||||||
* @deprecated use {@link #getStyle()} instead.
|
* @deprecated use {@link #style} instead.
|
||||||
*/
|
*/
|
||||||
getParagraphStyle: '#getStyle',
|
getParagraphStyle: '#getStyle',
|
||||||
setParagraphStyle: '#setStyle'
|
setParagraphStyle: '#setStyle'
|
||||||
|
|
|
@ -60,12 +60,15 @@ var Numerical = new function() {
|
||||||
var abs = Math.abs,
|
var abs = Math.abs,
|
||||||
sqrt = Math.sqrt,
|
sqrt = Math.sqrt,
|
||||||
pow = Math.pow,
|
pow = Math.pow,
|
||||||
TOLERANCE = 1e-6,
|
|
||||||
EPSILON = 1e-12,
|
EPSILON = 1e-12,
|
||||||
MACHINE_EPSILON = 1.12e-16;
|
MACHINE_EPSILON = 1.12e-16;
|
||||||
|
|
||||||
|
function clip(value, min, max) {
|
||||||
|
return value < min ? min : value > max ? max : value;
|
||||||
|
}
|
||||||
|
|
||||||
return /** @lends Numerical */{
|
return /** @lends Numerical */{
|
||||||
TOLERANCE: TOLERANCE,
|
TOLERANCE: 1e-6,
|
||||||
// Precision when comparing against 0
|
// Precision when comparing against 0
|
||||||
// References:
|
// References:
|
||||||
// http://docs.oracle.com/cd/E19957-01/806-3568/ncg_goldberg.html
|
// http://docs.oracle.com/cd/E19957-01/806-3568/ncg_goldberg.html
|
||||||
|
@ -78,28 +81,49 @@ var Numerical = new function() {
|
||||||
*/
|
*/
|
||||||
EPSILON: EPSILON,
|
EPSILON: EPSILON,
|
||||||
/**
|
/**
|
||||||
* MACHINE_EPSILON for a double precision (Javascript Number) is
|
* The machine epsilon for a double precision (Javascript Number) is
|
||||||
* 2.220446049250313e-16. (try this in the js console)
|
* 2.220446049250313e-16. (try this in the js console:
|
||||||
* (function(){for(var e=1;1<1+e/2;)e/=2;return e}())
|
* (function(){for(var e=1;1<1+e/2;)e/=2;return e}())
|
||||||
*
|
*
|
||||||
* Here the constant MACHINE_EPSILON refers to the constants 'δ' and 'ε'
|
* The constant MACHINE_EPSILON here refers to the constants δ and ε
|
||||||
* such that, the error introduced by addition, multiplication
|
* such that, the error introduced by addition, multiplication on a
|
||||||
* on a 64bit float (js Number) will be less than δ and ε. That is to
|
* 64bit float (js Number) will be less than δ and ε. That is to say,
|
||||||
* say, for all X and Y representable by a js Number object, S and P
|
* for all X and Y representable by a js Number object, S and P be their
|
||||||
* be their 'exact' sum and product respectively, then
|
* 'exact' sum and product respectively, then
|
||||||
* |S - (x+y)| <= δ|S|, and |P - (x*y)| <= ε|P|.
|
* |S - (x+y)| <= δ|S|, and |P - (x*y)| <= ε|P|.
|
||||||
* This amounts to about half of the actual MACHINE_EPSILON
|
* This amounts to about half of the actual machine epsilon.
|
||||||
*/
|
*/
|
||||||
MACHINE_EPSILON: MACHINE_EPSILON,
|
MACHINE_EPSILON: MACHINE_EPSILON,
|
||||||
|
/**
|
||||||
|
* The epsilon to be used when handling curve-time parameters. This
|
||||||
|
* cannot be smaller, because errors add up to around 8e-7 in the bezier
|
||||||
|
* fat-line clipping code as a result of recursive sub-division.
|
||||||
|
*/
|
||||||
|
CURVETIME_EPSILON: 8e-7,
|
||||||
|
/**
|
||||||
|
* The epsilon to be used when performing "geometric" checks, such as
|
||||||
|
* point distances and examining cross products to check for
|
||||||
|
* collinearity.
|
||||||
|
*/
|
||||||
|
GEOMETRIC_EPSILON: 4e-7, // NOTE: 1e-7 doesn't work in some edge-cases!
|
||||||
|
/**
|
||||||
|
* The epsilon to be used when performing winding contribution checks.
|
||||||
|
*/
|
||||||
|
WINDING_EPSILON: 2e-7, // NOTE: 1e-7 doesn't work in some edge-cases!
|
||||||
|
/**
|
||||||
|
* The epsilon to be used when performing "trigonometric" checks, such
|
||||||
|
* as examining cross products to check for collinearity.
|
||||||
|
*/
|
||||||
|
TRIGONOMETRIC_EPSILON: 1e-7,
|
||||||
// Kappa, see: http://www.whizkidtech.redprince.net/bezier/circle/kappa/
|
// Kappa, see: http://www.whizkidtech.redprince.net/bezier/circle/kappa/
|
||||||
KAPPA: 4 * (sqrt(2) - 1) / 3,
|
KAPPA: 4 * (sqrt(2) - 1) / 3,
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Check if the value is 0, within a tolerance defined by
|
* Checks if the value is 0, within a tolerance defined by
|
||||||
* Numerical.EPSILON.
|
* Numerical.EPSILON.
|
||||||
*/
|
*/
|
||||||
isZero: function(val) {
|
isZero: function(val) {
|
||||||
return abs(val) <= EPSILON;
|
return val >= -EPSILON && val <= EPSILON;
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -158,7 +182,7 @@ var Numerical = new function() {
|
||||||
*
|
*
|
||||||
* References:
|
* References:
|
||||||
* Kahan W. - "To Solve a Real Cubic Equation"
|
* Kahan W. - "To Solve a Real Cubic Equation"
|
||||||
* http://www.cs.berkeley.edu/~wkahan/Math128/Cubic.pdf
|
* http://www.cs.berkeley.edu/~wkahan/Math128/Cubic.pdf
|
||||||
* Blinn J. - "How to solve a Quadratic Equation"
|
* Blinn J. - "How to solve a Quadratic Equation"
|
||||||
*
|
*
|
||||||
* @param {Number} a the quadratic term
|
* @param {Number} a the quadratic term
|
||||||
|
@ -174,10 +198,14 @@ var Numerical = new function() {
|
||||||
*/
|
*/
|
||||||
solveQuadratic: function(a, b, c, roots, min, max) {
|
solveQuadratic: function(a, b, c, roots, min, max) {
|
||||||
var count = 0,
|
var count = 0,
|
||||||
|
eMin = min - EPSILON,
|
||||||
|
eMax = max + EPSILON,
|
||||||
x1, x2 = Infinity,
|
x1, x2 = Infinity,
|
||||||
B = b,
|
B = b,
|
||||||
D;
|
D;
|
||||||
b /= 2;
|
// a, b, c are expected to be the coefficients of the equation:
|
||||||
|
// Ax² - 2Bx + C == 0, so we take b = -B/2:
|
||||||
|
b /= -2;
|
||||||
D = b * b - a * c; // Discriminant
|
D = b * b - a * c; // Discriminant
|
||||||
// If the discriminant is very small, we can try to pre-condition
|
// If the discriminant is very small, we can try to pre-condition
|
||||||
// the coefficients, so that we may get better accuracy
|
// the coefficients, so that we may get better accuracy
|
||||||
|
@ -188,8 +216,8 @@ var Numerical = new function() {
|
||||||
// We multiply with a factor to normalize the coefficients.
|
// We multiply with a factor to normalize the coefficients.
|
||||||
// The factor is just the nearest exponent of 10, big enough
|
// The factor is just the nearest exponent of 10, big enough
|
||||||
// to raise all the coefficients to nearly [-1, +1] range.
|
// to raise all the coefficients to nearly [-1, +1] range.
|
||||||
var mult = pow(10, abs(
|
var mult = pow(10,
|
||||||
Math.floor(Math.log(gmC) * Math.LOG10E)));
|
abs(Math.floor(Math.log(gmC) * Math.LOG10E)));
|
||||||
if (!isFinite(mult))
|
if (!isFinite(mult))
|
||||||
mult = 0;
|
mult = 0;
|
||||||
a *= mult;
|
a *= mult;
|
||||||
|
@ -204,29 +232,25 @@ var Numerical = new function() {
|
||||||
if (abs(B) < EPSILON)
|
if (abs(B) < EPSILON)
|
||||||
return abs(c) < EPSILON ? -1 : 0;
|
return abs(c) < EPSILON ? -1 : 0;
|
||||||
x1 = -c / B;
|
x1 = -c / B;
|
||||||
} else {
|
} else if (D >= -MACHINE_EPSILON) { // No real roots if D < 0
|
||||||
// No real roots if D < 0
|
var Q = D < 0 ? 0 : sqrt(D),
|
||||||
if (D >= -MACHINE_EPSILON) {
|
R = b + (b < 0 ? -Q : Q);
|
||||||
D = D < 0 ? 0 : D;
|
// Try to minimize floating point noise.
|
||||||
var R = sqrt(D);
|
if (R === 0) {
|
||||||
// Try to minimise floating point noise.
|
x1 = c / a;
|
||||||
if (b >= MACHINE_EPSILON && b <= MACHINE_EPSILON) {
|
x2 = -x1;
|
||||||
x1 = abs(a) >= abs(c) ? R / a : -c / R;
|
} else {
|
||||||
x2 = -x1;
|
x1 = R / a;
|
||||||
} else {
|
x2 = c / R;
|
||||||
var q = -(b + (b < 0 ? -1 : 1) * R);
|
|
||||||
x1 = q / a;
|
|
||||||
x2 = c / q;
|
|
||||||
}
|
|
||||||
// Do we actually have two real roots?
|
|
||||||
// count = D > MACHINE_EPSILON ? 2 : 1;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (isFinite(x1) && (min == null || x1 >= min && x1 <= max))
|
// We need to include EPSILON in the comparisons with min / max,
|
||||||
roots[count++] = x1;
|
// as some solutions are ever so lightly out of bounds.
|
||||||
|
if (isFinite(x1) && (min == null || x1 > eMin && x1 < eMax))
|
||||||
|
roots[count++] = min == null ? x1 : clip(x1, min, max);
|
||||||
if (x2 !== x1
|
if (x2 !== x1
|
||||||
&& isFinite(x2) && (min == null || x2 >= min && x2 <= max))
|
&& isFinite(x2) && (min == null || x2 > eMin && x2 < eMax))
|
||||||
roots[count++] = x2;
|
roots[count++] = min == null ? x2 : clip(x2, min, max);
|
||||||
return count;
|
return count;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -321,8 +345,8 @@ var Numerical = new function() {
|
||||||
// The cubic has been deflated to a quadratic.
|
// The cubic has been deflated to a quadratic.
|
||||||
var count = Numerical.solveQuadratic(a, b1, c2, roots, min, max);
|
var count = Numerical.solveQuadratic(a, b1, c2, roots, min, max);
|
||||||
if (isFinite(x) && (count === 0 || x !== roots[count - 1])
|
if (isFinite(x) && (count === 0 || x !== roots[count - 1])
|
||||||
&& (min == null || x >= min && x <= max))
|
&& (min == null || x > min - EPSILON && x < max + EPSILON))
|
||||||
roots[count++] = x;
|
roots[count++] = min == null ? x : clip(x, min, max);
|
||||||
return count;
|
return count;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
|
@ -144,8 +144,8 @@ var CanvasView = View.extend(/** @lends CanvasView# */{
|
||||||
project._needsUpdate = false;
|
project._needsUpdate = false;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}, new function() { // Item based mouse handling:
|
},
|
||||||
|
new function() { // Item based mouse handling:
|
||||||
var downPoint,
|
var downPoint,
|
||||||
lastPoint,
|
lastPoint,
|
||||||
overPoint,
|
overPoint,
|
||||||
|
|
|
@ -675,8 +675,8 @@ var View = Base.extend(Emitter, /** @lends View# */{
|
||||||
return new CanvasView(project, element);
|
return new CanvasView(project, element);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}, new function() {
|
},
|
||||||
// Injection scope for mouse events on the browser
|
new function() { // Injection scope for mouse events on the browser
|
||||||
/*#*/ if (__options.environment == 'browser') {
|
/*#*/ if (__options.environment == 'browser') {
|
||||||
var tool,
|
var tool,
|
||||||
prevFocus,
|
prevFocus,
|
||||||
|
|
|
@ -286,7 +286,7 @@ function asyncTest(testName, expected) {
|
||||||
var project = new Project();
|
var project = new Project();
|
||||||
expected(function() {
|
expected(function() {
|
||||||
project.remove();
|
project.remove();
|
||||||
start();
|
QUnit.start();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
|
@ -161,7 +161,7 @@ test('Curve#getParameterAt()', function() {
|
||||||
var t2 = curve.getParameterAt(o2);
|
var t2 = curve.getParameterAt(o2);
|
||||||
equals(t1, t2, 'Curve parameter at offset ' + o1
|
equals(t1, t2, 'Curve parameter at offset ' + o1
|
||||||
+ ' should be the same value as at offset' + o2,
|
+ ' should be the same value as at offset' + o2,
|
||||||
Numerical.TOLERANCE);
|
Numerical.CURVETIME_EPSILON);
|
||||||
}
|
}
|
||||||
|
|
||||||
equals(curve.getParameterAt(curve.length + 1), null,
|
equals(curve.getParameterAt(curve.length + 1), null,
|
||||||
|
@ -178,3 +178,48 @@ test('Curve#getLocationAt()', function() {
|
||||||
'Should return null when offset is out of range.');
|
'Should return null when offset is out of range.');
|
||||||
// 'Should return null when point is not on the curve.');
|
// 'Should return null when point is not on the curve.');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test('Curve#isStraight()', function() {
|
||||||
|
equals(function() {
|
||||||
|
return new Curve([100, 100], null, null, [200, 200]).isStraight();
|
||||||
|
}, true);
|
||||||
|
equals(function() {
|
||||||
|
return new Curve([100, 100], [-50, -50], null, [200, 200]).isStraight();
|
||||||
|
}, false);
|
||||||
|
equals(function() {
|
||||||
|
return new Curve([100, 100], [50, 50], null, [200, 200]).isStraight();
|
||||||
|
}, true);
|
||||||
|
equals(function() {
|
||||||
|
return new Curve([100, 100], [50, 50], [-50, -50], [200, 200]).isStraight();
|
||||||
|
}, true);
|
||||||
|
equals(function() {
|
||||||
|
return new Curve([100, 100], [50, 50], [50, 50], [200, 200]).isStraight();
|
||||||
|
}, false);
|
||||||
|
equals(function() {
|
||||||
|
return new Curve([100, 100], null, [-50, -50], [200, 200]).isStraight();
|
||||||
|
}, true);
|
||||||
|
equals(function() {
|
||||||
|
return new Curve([100, 100], null, [50, 50], [200, 200]).isStraight();
|
||||||
|
}, false);
|
||||||
|
equals(function() {
|
||||||
|
return new Curve([100, 100], null, null, [100, 100]).isStraight();
|
||||||
|
}, true);
|
||||||
|
equals(function() {
|
||||||
|
return new Curve([100, 100], [50, 50], null, [100, 100]).isStraight();
|
||||||
|
}, false);
|
||||||
|
equals(function() {
|
||||||
|
return new Curve([100, 100], null, [-50, -50], [100, 100]).isStraight();
|
||||||
|
}, false);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Curve#isLinear()', function() {
|
||||||
|
equals(function() {
|
||||||
|
return new Curve([100, 100], [100 / 3, 100 / 3], [-100 / 3, -100 / 3], [200, 200]).isLinear();
|
||||||
|
}, true);
|
||||||
|
equals(function() {
|
||||||
|
return new Curve([100, 100], null, null, [100, 100]).isLinear();
|
||||||
|
}, true);
|
||||||
|
equals(function() {
|
||||||
|
return new Curve([100, 100], null, null, [200, 200]).isLinear();
|
||||||
|
}, false);
|
||||||
|
});
|
||||||
|
|
|
@ -16,8 +16,7 @@ test('Decomposition: rotate()', function() {
|
||||||
var m = new Matrix().rotate(a),
|
var m = new Matrix().rotate(a),
|
||||||
s = 'new Matrix().rotate(' + a + ')';
|
s = 'new Matrix().rotate(' + a + ')';
|
||||||
equals(m.getRotation(), Base.pick(ea, a),
|
equals(m.getRotation(), Base.pick(ea, a),
|
||||||
s + '.getRotation()',
|
s + '.getRotation()');
|
||||||
Numerical.TOLERANCE);
|
|
||||||
equals(m.getScaling(), new Point(1, 1),
|
equals(m.getScaling(), new Point(1, 1),
|
||||||
s + '.getScaling()');
|
s + '.getScaling()');
|
||||||
}
|
}
|
||||||
|
@ -42,8 +41,7 @@ test('Decomposition: scale()', function() {
|
||||||
var m = new Matrix().scale(sx, sy),
|
var m = new Matrix().scale(sx, sy),
|
||||||
s = 'new Matrix().scale(' + sx + ', ' + sy + ')';
|
s = 'new Matrix().scale(' + sx + ', ' + sy + ')';
|
||||||
equals(m.getRotation(), ea || 0,
|
equals(m.getRotation(), ea || 0,
|
||||||
s + '.getRotation()',
|
s + '.getRotation()');
|
||||||
Numerical.TOLERANCE);
|
|
||||||
equals(m.getScaling(), new Point(Base.pick(ex, sx), Base.pick(ey, sy)),
|
equals(m.getScaling(), new Point(Base.pick(ex, sx), Base.pick(ey, sy)),
|
||||||
s + '.getScaling()');
|
s + '.getScaling()');
|
||||||
}
|
}
|
||||||
|
@ -64,8 +62,7 @@ test('Decomposition: rotate() & scale()', function() {
|
||||||
var m = new Matrix().scale(sx, sy).rotate(a),
|
var m = new Matrix().scale(sx, sy).rotate(a),
|
||||||
s = 'new Matrix().scale(' + sx + ', ' + sy + ').rotate(' + a + ')';
|
s = 'new Matrix().scale(' + sx + ', ' + sy + ').rotate(' + a + ')';
|
||||||
equals(m.getRotation(), ea || a,
|
equals(m.getRotation(), ea || a,
|
||||||
s + '.getRotation()',
|
s + '.getRotation()');
|
||||||
Numerical.TOLERANCE);
|
|
||||||
equals(m.getScaling(), new Point(Base.pick(ex, sx), Base.pick(ey, sy)),
|
equals(m.getScaling(), new Point(Base.pick(ex, sx), Base.pick(ey, sy)),
|
||||||
s + '.getScaling()');
|
s + '.getScaling()');
|
||||||
}
|
}
|
||||||
|
|
|
@ -51,5 +51,5 @@ test('ring.subtract(square); #610', function() {
|
||||||
var ring = outer.subtract(inner);
|
var ring = outer.subtract(inner);
|
||||||
var result = ring.subtract(square);
|
var result = ring.subtract(square);
|
||||||
|
|
||||||
equals(result.pathData, 'M-10,131.62689c-68.2302,-5.11075 -122,-62.08951 -122,-131.62689c0,-69.53737 53.7698,-126.51614 122,-131.62689l0,32.12064c-50.53323,5.01724 -90,47.65277 -90,99.50625c0,51.85348 39.46677,94.489 90,99.50625z');
|
equals(result.pathData, 'M-132,0c0,-69.53737 53.7698,-126.51614 122,-131.62689l0,32.12064c-50.53323,5.01724 -90,47.65277 -90,99.50625c0,51.85348 39.46677,94.489 90,99.50625l0,32.12064c-68.2302,-5.11075 -122,-62.08951 -122,-131.62689z');
|
||||||
});
|
});
|
||||||
|
|
|
@ -107,10 +107,10 @@ test('Curve list after removing a segment - 2', function() {
|
||||||
}, 1, 'After removing the last segment, we should be left with one curve');
|
}, 1, 'After removing the last segment, we should be left with one curve');
|
||||||
});
|
});
|
||||||
|
|
||||||
test('Splitting a straight path should produce straight segments', function() {
|
test('Splitting a straight path should produce segments without handles', function() {
|
||||||
var path = new Path.Line([0, 0], [50, 50]);
|
var path1 = new Path.Line([0, 0], [50, 50]);
|
||||||
var path2 = path.split(0, 0.5);
|
var path2 = path1.split(0, 0.5);
|
||||||
equals(function() {
|
equals(function() {
|
||||||
return path2.firstSegment.isStraight();
|
return !path1.lastSegment.hasHandles() && !path2.firstSegment.hasHandles();
|
||||||
}, true);
|
}, true);
|
||||||
});
|
});
|
||||||
|
|
|
@ -43,7 +43,7 @@ test('new Segment(size)', function() {
|
||||||
|
|
||||||
test('segment.reverse()', function() {
|
test('segment.reverse()', function() {
|
||||||
var segment = new Segment(new Point(10, 10), new Point(5, 5), new Point(15, 15));
|
var segment = new Segment(new Point(10, 10), new Point(5, 5), new Point(15, 15));
|
||||||
segment = segment.reverse();
|
segment.reverse();
|
||||||
equals(segment.toString(), '{ point: { x: 10, y: 10 }, handleIn: { x: 15, y: 15 }, handleOut: { x: 5, y: 5 } }');
|
equals(segment.toString(), '{ point: { x: 10, y: 10 }, handleIn: { x: 15, y: 15 }, handleOut: { x: 5, y: 5 } }');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -14,7 +14,7 @@ module('TextItem');
|
||||||
|
|
||||||
test('PointText', function() {
|
test('PointText', function() {
|
||||||
var text = new PointText({
|
var text = new PointText({
|
||||||
fontFamily: 'Arial',
|
fontFamily: 'Helvetica, Arial',
|
||||||
fontSize: 14,
|
fontSize: 14,
|
||||||
point: [100, 100],
|
point: [100, 100],
|
||||||
content: 'Hello World!'
|
content: 'Hello World!'
|
||||||
|
@ -22,7 +22,7 @@ test('PointText', function() {
|
||||||
equals(text.fillColor, new Color(0, 0, 0), 'text.fillColor should be black by default');
|
equals(text.fillColor, new Color(0, 0, 0), 'text.fillColor should be black by default');
|
||||||
equals(text.point, new Point(100, 100), 'text.point');
|
equals(text.point, new Point(100, 100), 'text.point');
|
||||||
equals(text.bounds.point, new Point(100, 87.4), 'text.bounds.point');
|
equals(text.bounds.point, new Point(100, 87.4), 'text.bounds.point');
|
||||||
equals(text.bounds.size, new Size(77, 16.8), 'text.bounds.size', { tolerance: 1.0 });
|
equals(text.bounds.size, new Size(76, 16.8), 'text.bounds.size', { tolerance: 1.0 });
|
||||||
equals(function() {
|
equals(function() {
|
||||||
return text.hitTest(text.bounds.center) != null;
|
return text.hitTest(text.bounds.center) != null;
|
||||||
}, true);
|
}, true);
|
||||||
|
|
Loading…
Reference in a new issue