mirror of
https://github.com/scratchfoundation/paper.js.git
synced 2025-01-01 02:38:43 -05:00
parent
261fc02c1d
commit
d436d07fee
3 changed files with 66 additions and 21 deletions
|
@ -187,6 +187,10 @@ var Shape = Item.extend(/** @lends Shape# */{
|
||||||
|
|
||||||
toShape: '#clone',
|
toShape: '#clone',
|
||||||
|
|
||||||
|
_asPathItem: function() {
|
||||||
|
return this.toPath(false);
|
||||||
|
},
|
||||||
|
|
||||||
_draw: function(ctx, param, viewMatrix, strokeMatrix) {
|
_draw: function(ctx, param, viewMatrix, strokeMatrix) {
|
||||||
var style = this._style,
|
var style = this._style,
|
||||||
hasFill = style.hasFill(),
|
hasFill = style.hasFill(),
|
||||||
|
|
|
@ -1716,17 +1716,18 @@ new function() { // Scope for methods that require private functions
|
||||||
},
|
},
|
||||||
new function() { // Scope for bezier intersection using fat-line clipping
|
new function() { // Scope for bezier intersection using fat-line clipping
|
||||||
|
|
||||||
function addLocation(locations, include, c1, t1, p1, c2, t2, p2, overlap) {
|
function addLocation(locations, include, c1, t1, c2, t2, overlap) {
|
||||||
// Determine if locations at the beginning / end of the curves should be
|
// Determine if locations at the beginning / end of the curves should be
|
||||||
// excluded, in case the two curves are neighbors, but do not exclude
|
// excluded, in case the two curves are neighbors, but do not exclude
|
||||||
// connecting points between two curves if they were part of overlap
|
// connecting points between two curves if they were part of overlap
|
||||||
// checks, as they could be self-overlapping.
|
// checks, as they could be self-overlapping.
|
||||||
|
// NOTE: We don't pass p1 and p2, because v1 and v2 may be transformed
|
||||||
|
// by their path.matrix, while c1 and c2 are untransformed. Passing null
|
||||||
|
// for point in CurveLocation() will do the right thing.
|
||||||
var excludeStart = !overlap && c1.getPrevious() === c2,
|
var excludeStart = !overlap && c1.getPrevious() === c2,
|
||||||
excludeEnd = !overlap && c1 !== c2 && c1.getNext() === c2,
|
excludeEnd = !overlap && c1 !== c2 && c1.getNext() === c2,
|
||||||
tMin = /*#=*/Numerical.CURVETIME_EPSILON,
|
tMin = /*#=*/Numerical.CURVETIME_EPSILON,
|
||||||
tMax = 1 - tMin;
|
tMax = 1 - tMin;
|
||||||
if (t1 == null)
|
|
||||||
t1 = c1.getTimeOf(p1);
|
|
||||||
// Check t1 and t2 against correct bounds, based on excludeStart/End:
|
// Check t1 and t2 against correct bounds, based on excludeStart/End:
|
||||||
// - excludeStart means the start of c1 connects to the end of c2
|
// - excludeStart means the start of c1 connects to the end of c2
|
||||||
// - excludeEnd means the end of c1 connects to the start of c2
|
// - excludeEnd means the end of c1 connects to the start of c2
|
||||||
|
@ -1736,14 +1737,10 @@ new function() { // Scope for bezier intersection using fat-line clipping
|
||||||
// the beginning, and would be added twice otherwise.
|
// the beginning, and would be added twice otherwise.
|
||||||
if (t1 !== null && t1 >= (excludeStart ? tMin : 0) &&
|
if (t1 !== null && t1 >= (excludeStart ? tMin : 0) &&
|
||||||
t1 <= (excludeEnd ? tMax : 1)) {
|
t1 <= (excludeEnd ? tMax : 1)) {
|
||||||
if (t2 == null)
|
|
||||||
t2 = c2.getTimeOf(p2);
|
|
||||||
if (t2 !== null && t2 >= (excludeEnd ? tMin : 0) &&
|
if (t2 !== null && t2 >= (excludeEnd ? tMin : 0) &&
|
||||||
t2 <= (excludeStart ? tMax : 1)) {
|
t2 <= (excludeStart ? tMax : 1)) {
|
||||||
var loc1 = new CurveLocation(c1, t1,
|
var loc1 = new CurveLocation(c1, t1, null, overlap),
|
||||||
p1 || c1.getPointAtTime(t1), overlap),
|
loc2 = new CurveLocation(c2, t2, null, overlap);
|
||||||
loc2 = new CurveLocation(c2, t2,
|
|
||||||
p2 || c2.getPointAtTime(t2), overlap);
|
|
||||||
// Link the two locations to each other.
|
// Link the two locations to each other.
|
||||||
loc1._intersection = loc2;
|
loc1._intersection = loc2;
|
||||||
loc2._intersection = loc1;
|
loc2._intersection = loc1;
|
||||||
|
@ -1806,8 +1803,8 @@ new function() { // Scope for bezier intersection using fat-line clipping
|
||||||
var t = (tMinNew + tMaxNew) / 2,
|
var t = (tMinNew + tMaxNew) / 2,
|
||||||
u = (uMin + uMax) / 2;
|
u = (uMin + uMax) / 2;
|
||||||
addLocation(locations, include,
|
addLocation(locations, include,
|
||||||
flip ? c2 : c1, flip ? u : t, null,
|
flip ? c2 : c1, flip ? u : t,
|
||||||
flip ? c1 : c2, flip ? t : u, null);
|
flip ? c1 : c2, flip ? t : u);
|
||||||
} else {
|
} else {
|
||||||
// Apply the result of the clipping to curve 1:
|
// Apply the result of the clipping to curve 1:
|
||||||
v1 = Curve.getPart(v1, tMinClip, tMaxClip);
|
v1 = Curve.getPart(v1, tMinClip, tMaxClip);
|
||||||
|
@ -1989,12 +1986,11 @@ new function() { // Scope for bezier intersection using fat-line clipping
|
||||||
p1 = Curve.getPoint(v1, t1),
|
p1 = Curve.getPoint(v1, t1),
|
||||||
t2 = Curve.getTimeOf(v2, p1);
|
t2 = Curve.getTimeOf(v2, p1);
|
||||||
if (t2 !== null) {
|
if (t2 !== null) {
|
||||||
var p2 = Curve.getPoint(v2, t2);
|
|
||||||
// Only use the time values if there was no recursion, and let
|
// Only use the time values if there was no recursion, and let
|
||||||
// addLocation() figure out the actual time values otherwise.
|
// addLocation() figure out the actual time values otherwise.
|
||||||
addLocation(locations, include,
|
addLocation(locations, include,
|
||||||
flip ? c2 : c1, flip ? t2 : t1, flip ? p2 : p1,
|
flip ? c2 : c1, flip ? t2 : t1,
|
||||||
flip ? c1 : c2, flip ? t1 : t2, flip ? p1 : p2);
|
flip ? c1 : c2, flip ? t1 : t2);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2004,7 +2000,9 @@ new function() { // Scope for bezier intersection using fat-line clipping
|
||||||
v1[0], v1[1], v1[6], v1[7],
|
v1[0], v1[1], v1[6], v1[7],
|
||||||
v2[0], v2[1], v2[6], v2[7]);
|
v2[0], v2[1], v2[6], v2[7]);
|
||||||
if (pt) {
|
if (pt) {
|
||||||
addLocation(locations, include, c1, null, pt, c2, null, pt);
|
addLocation(locations, include,
|
||||||
|
c1, Curve.getTimeOf(v1, pt),
|
||||||
|
c2, Curve.getTimeOf(v2, pt));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2028,8 +2026,8 @@ new function() { // Scope for bezier intersection using fat-line clipping
|
||||||
for (var i = 0; i < 2; i++) {
|
for (var i = 0; i < 2; i++) {
|
||||||
var overlap = overlaps[i];
|
var overlap = overlaps[i];
|
||||||
addLocation(locations, include,
|
addLocation(locations, include,
|
||||||
c1, overlap[0], null,
|
c1, overlap[0],
|
||||||
c2, overlap[1], null, true);
|
c2, overlap[1], true);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
var straight1 = Curve.isStraight(v1),
|
var straight1 = Curve.isStraight(v1),
|
||||||
|
@ -2065,8 +2063,8 @@ new function() { // Scope for bezier intersection using fat-line clipping
|
||||||
p2 = new Point(v2[i2], v2[i2 + 1]);
|
p2 = new Point(v2[i2], v2[i2 + 1]);
|
||||||
if (p1.isClose(p2, epsilon)) {
|
if (p1.isClose(p2, epsilon)) {
|
||||||
addLocation(locations, include,
|
addLocation(locations, include,
|
||||||
c1, t1, p1,
|
c1, t1,
|
||||||
c2, t2, p2);
|
c2, t2);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2080,8 +2078,8 @@ new function() { // Scope for bezier intersection using fat-line clipping
|
||||||
if (info.type === 'loop') {
|
if (info.type === 'loop') {
|
||||||
var roots = info.roots;
|
var roots = info.roots;
|
||||||
addLocation(locations, include,
|
addLocation(locations, include,
|
||||||
c1, roots[0], null,
|
c1, roots[0],
|
||||||
c1, roots[1], null);
|
c1, roots[1]);
|
||||||
}
|
}
|
||||||
return locations;
|
return locations;
|
||||||
}
|
}
|
||||||
|
|
|
@ -111,6 +111,49 @@ test('circle and square (existing segments overlaps on curves)', function() {
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test('intersecting paths with applyMatrix = true / false', function() {
|
||||||
|
function test(ctor, applyMatrix) {
|
||||||
|
var name = ctor.name.toLowerCase();
|
||||||
|
|
||||||
|
var item1 = new ctor.Rectangle({
|
||||||
|
point: [0, 0],
|
||||||
|
size: [200, 200],
|
||||||
|
applyMatrix: applyMatrix
|
||||||
|
});
|
||||||
|
|
||||||
|
var offset = new Point(200, 200);
|
||||||
|
|
||||||
|
item1.translate(offset);
|
||||||
|
|
||||||
|
var item2 = new ctor.Rectangle({
|
||||||
|
point: [100, 100],
|
||||||
|
size: [200, 200],
|
||||||
|
applyMatrix: applyMatrix
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!applyMatrix)
|
||||||
|
offset = new Point(0, 0);
|
||||||
|
|
||||||
|
if (ctor === Path) {
|
||||||
|
testIntersections(item1.getIntersections(item2), [{
|
||||||
|
point: new Point(0, 100).add(offset), index: 0, time: 0.5,
|
||||||
|
crossing: true
|
||||||
|
}, {
|
||||||
|
point: new Point(100, 0).add(offset), index: 1, time: 0.5,
|
||||||
|
crossing: true
|
||||||
|
}]);
|
||||||
|
}
|
||||||
|
|
||||||
|
equals(item1.intersects(item2), true,
|
||||||
|
name + '1.intersects(' + name + '2);');
|
||||||
|
}
|
||||||
|
|
||||||
|
test(Path, true);
|
||||||
|
test(Path, false);
|
||||||
|
// Also tests #intersects() on Shape
|
||||||
|
test(Shape, false);
|
||||||
|
});
|
||||||
|
|
||||||
test('#904', function() {
|
test('#904', function() {
|
||||||
var path1 = new Path([
|
var path1 = new Path([
|
||||||
[347.65684372173973, 270.4315945523045, 0, 0, 22.844385382059784, -25.115215946843847],
|
[347.65684372173973, 270.4315945523045, 0, 0, 22.844385382059784, -25.115215946843847],
|
||||||
|
|
Loading…
Reference in a new issue