2013-05-03 19:16:52 -04:00
|
|
|
/*
|
|
|
|
* Paper.js - The Swiss Army Knife of Vector Graphics Scripting.
|
|
|
|
* http://paperjs.org/
|
|
|
|
*
|
2014-01-03 19:47:16 -05:00
|
|
|
* Copyright (c) 2011 - 2014, Juerg Lehni & Jonathan Puckey
|
|
|
|
* http://scratchdisk.com/ & http://jonathanpuckey.com/
|
2013-05-03 19:16:52 -04:00
|
|
|
*
|
|
|
|
* Distributed under the MIT license. See LICENSE file for details.
|
|
|
|
*
|
|
|
|
* All rights reserved.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
2013-05-03 19:31:36 -04:00
|
|
|
* Boolean Geometric Path Operations
|
2013-05-03 19:16:52 -04:00
|
|
|
*
|
|
|
|
* This is mostly written for clarity and compatibility, not optimised for
|
|
|
|
* performance, and has to be tested heavily for stability.
|
|
|
|
*
|
|
|
|
* Supported
|
2013-05-05 19:38:18 -04:00
|
|
|
* - Path and CompoundPath items
|
2013-05-03 19:16:52 -04:00
|
|
|
* - Boolean Union
|
|
|
|
* - Boolean Intersection
|
|
|
|
* - Boolean Subtraction
|
|
|
|
* - Resolving a self-intersecting Path
|
|
|
|
*
|
|
|
|
* Not supported yet
|
|
|
|
* - Boolean operations on self-intersecting Paths
|
|
|
|
* - Paths are clones of each other that ovelap exactly on top of each other!
|
|
|
|
*
|
|
|
|
* @author Harikrishnan Gopalakrishnan
|
|
|
|
* http://hkrish.com/playground/paperjs/booleanStudy.html
|
|
|
|
*/
|
|
|
|
|
2014-02-20 13:35:11 -05:00
|
|
|
PathItem.inject(/** @lends PathItem# */{
|
|
|
|
// Boolean operators return true if a curve with the given winding
|
|
|
|
// contribution contributes to the final result or not. They are called
|
|
|
|
// for each curve in the graph after curves in the operands are
|
|
|
|
// split at intersections.
|
|
|
|
_computeBoolean: function(other, operator, subtract) {
|
|
|
|
// To deal with a HTML5 canvas requirement where CompoundPaths' child
|
|
|
|
// contours has to be of different winding direction for correctly
|
|
|
|
// filling holes. But if some individual countours are disjoint, i.e.
|
|
|
|
// islands, we have to reorient them so that:
|
|
|
|
// - The holes have opposite winding direction, already handled by paper
|
|
|
|
// - Islands have to have the same winding direction as the first child
|
|
|
|
// NOTE: Does NOT handle self-intersecting CompoundPaths.
|
|
|
|
function reorientPath(path) {
|
|
|
|
if (path instanceof CompoundPath) {
|
|
|
|
var children = path.removeChildren(),
|
|
|
|
length = children.length,
|
|
|
|
bounds = new Array(length),
|
|
|
|
counters = new Array(length),
|
|
|
|
clockwise;
|
|
|
|
children.sort(function(a, b) {
|
|
|
|
return b.getBounds().getArea() - a.getBounds().getArea();
|
|
|
|
});
|
|
|
|
path.addChildren(children);
|
|
|
|
clockwise = children[0].isClockwise();
|
|
|
|
for (var i = 0; i < length; i++) {
|
|
|
|
bounds[i] = children[i].getBounds();
|
|
|
|
counters[i] = 0;
|
|
|
|
}
|
|
|
|
for (var i = 0; i < length; i++) {
|
|
|
|
for (var j = 1; j < length; j++) {
|
|
|
|
if (i !== j && bounds[i].intersects(bounds[j]))
|
|
|
|
counters[j]++;
|
|
|
|
}
|
|
|
|
// Omit the first child
|
|
|
|
if (i > 0 && counters[i] % 2 === 0)
|
|
|
|
children[i].setClockwise(clockwise);
|
2013-05-04 02:03:00 -04:00
|
|
|
}
|
2013-05-03 19:21:44 -04:00
|
|
|
}
|
2014-02-20 13:35:11 -05:00
|
|
|
return path;
|
2013-05-03 19:21:44 -04:00
|
|
|
}
|
2013-05-03 19:16:52 -04:00
|
|
|
|
2013-05-03 19:21:44 -04:00
|
|
|
// We do not modify the operands themselves
|
|
|
|
// The result might not belong to the same type
|
|
|
|
// i.e. subtraction(A:Path, B:Path):CompoundPath etc.
|
2014-02-19 20:51:35 -05:00
|
|
|
// We call reduce() on both cloned paths to simplify compound paths and
|
|
|
|
// remove empty curves. We also apply matrices to both paths in case
|
|
|
|
// they were transformed.
|
2014-02-20 13:35:11 -05:00
|
|
|
var path1 = reorientPath(this.clone(false).reduce().applyMatrix());
|
|
|
|
path2 = this !== other
|
|
|
|
&& reorientPath(other.clone(false).reduce().applyMatrix());
|
2013-05-03 19:21:44 -04:00
|
|
|
// Do operator specific calculations before we begin
|
2014-02-20 13:10:46 -05:00
|
|
|
// Make both paths at clockwise orientation, except when subtract = true
|
|
|
|
// We need both paths at opposite orientation for subtraction.
|
2013-12-29 07:38:04 -05:00
|
|
|
if (!path1.isClockwise())
|
2013-09-22 21:18:22 -04:00
|
|
|
path1.reverse();
|
2014-02-20 13:35:11 -05:00
|
|
|
if (path2 && !(subtract ^ path2.isClockwise()))
|
2013-09-22 21:18:22 -04:00
|
|
|
path2.reverse();
|
2014-02-20 13:10:46 -05:00
|
|
|
// Split curves at intersections on both paths.
|
2014-02-20 13:35:11 -05:00
|
|
|
PathItem._splitPath(path1.getIntersections(path2 || path1, true));
|
2014-02-20 13:10:46 -05:00
|
|
|
|
2014-02-20 12:44:38 -05:00
|
|
|
var chain = [],
|
2014-02-17 14:59:38 -05:00
|
|
|
windings = [],
|
|
|
|
lengths = [],
|
|
|
|
segments = [],
|
|
|
|
// Aggregate of all curves in both operands, monotonic in y
|
2014-02-20 13:10:46 -05:00
|
|
|
monoCurves = [];
|
2014-01-25 23:39:51 -05:00
|
|
|
|
2014-02-20 13:10:46 -05:00
|
|
|
function collect(paths) {
|
|
|
|
for (var i = 0, l = paths.length; i < l; i++) {
|
|
|
|
var path = paths[i];
|
|
|
|
segments.push.apply(segments, path._segments);
|
|
|
|
monoCurves.push.apply(monoCurves, path._getMonoCurves());
|
|
|
|
}
|
2013-05-03 19:21:44 -04:00
|
|
|
}
|
2014-02-20 13:10:46 -05:00
|
|
|
|
|
|
|
// Collect all segments and monotonic curves
|
|
|
|
collect(path1._children || [path1]);
|
2014-02-20 13:35:11 -05:00
|
|
|
if (path2)
|
2014-02-20 13:10:46 -05:00
|
|
|
collect(path2._children || [path2]);
|
2013-12-29 07:38:04 -05:00
|
|
|
// Propagate the winding contribution. Winding contribution of curves
|
|
|
|
// does not change between two intersections.
|
|
|
|
// First, sort all segments with an intersection to the begining.
|
|
|
|
segments.sort(function(a, b) {
|
2014-02-19 18:32:15 -05:00
|
|
|
var _a = a._intersection,
|
|
|
|
_b = b._intersection;
|
|
|
|
return !_a && !_b || _a && _b ? 0 : _a ? -1 : 1;
|
2013-12-29 07:38:04 -05:00
|
|
|
});
|
2014-02-20 12:44:38 -05:00
|
|
|
for (var i = 0, l = segments.length; i < l; i++) {
|
|
|
|
var segment = segments[i];
|
2014-02-17 14:59:38 -05:00
|
|
|
if (segment._winding != null)
|
2013-05-04 02:03:00 -04:00
|
|
|
continue;
|
2013-12-29 07:38:04 -05:00
|
|
|
// Here we try to determine the most probable winding number
|
2014-02-20 12:44:38 -05:00
|
|
|
// contribution for this curve-chain. Once we have enough confidence
|
|
|
|
// in the winding contribution, we can propagate it until the
|
|
|
|
// intersection or end of a curve chain.
|
|
|
|
chain.length = windings.length = lengths.length = 0;
|
|
|
|
var totalLength = 0,
|
|
|
|
startSeg = segment;
|
2013-05-04 05:50:18 -04:00
|
|
|
do {
|
2014-02-20 12:44:38 -05:00
|
|
|
chain.push(segment);
|
|
|
|
lengths.push(totalLength += segment.getCurve().getLength());
|
2013-05-04 06:00:31 -04:00
|
|
|
segment = segment.getNext();
|
2014-02-17 14:59:38 -05:00
|
|
|
} while (segment && !segment._intersection && segment !== startSeg);
|
2014-02-20 12:44:38 -05:00
|
|
|
// Select the median winding of three random points along this curve
|
|
|
|
// chain, as a representative winding number. The random selection
|
|
|
|
// gives a better chance of returning a correct winding than equally
|
|
|
|
// dividing the curve chain, with the same (amortised) time.
|
|
|
|
for (var j = 0; j < 3; j++) {
|
|
|
|
var length = totalLength * Math.random(),
|
|
|
|
amount = lengths.length;
|
|
|
|
k = 0;
|
|
|
|
do {
|
|
|
|
if (lengths[k] >= length) {
|
|
|
|
if (k > 0)
|
|
|
|
length -= lengths[k - 1];
|
2014-01-05 08:26:09 -05:00
|
|
|
break;
|
2013-12-29 07:38:04 -05:00
|
|
|
}
|
2014-02-20 12:44:38 -05:00
|
|
|
} while (++k < amount);
|
|
|
|
var curve = chain[k].getCurve(),
|
|
|
|
point = curve.getPointAt(length),
|
|
|
|
hor = curve.isHorizontal(),
|
|
|
|
path = curve._path;
|
|
|
|
if (path._parent instanceof CompoundPath)
|
|
|
|
path = path._parent;
|
2014-01-05 08:26:09 -05:00
|
|
|
// While subtracting, we need to omit this curve if this
|
|
|
|
// curve is contributing to the second operand and is outside
|
|
|
|
// the first operand.
|
2014-02-20 13:35:11 -05:00
|
|
|
windings[j] = subtract && path2
|
2014-02-20 12:44:38 -05:00
|
|
|
&& (path === path1 && path2._getWinding(point, hor)
|
2014-02-20 13:10:46 -05:00
|
|
|
|| path === path2 && !path1._getWinding(point, hor))
|
2014-02-20 12:44:38 -05:00
|
|
|
? 0
|
|
|
|
: PathItem._getWinding(point, monoCurves, hor);
|
2013-05-03 19:21:44 -04:00
|
|
|
}
|
2014-01-05 08:26:09 -05:00
|
|
|
windings.sort();
|
2014-02-20 12:44:38 -05:00
|
|
|
// Assign the median winding to the entire curve chain.
|
|
|
|
var winding = windings[1];
|
|
|
|
for (var j = chain.length - 1; j >= 0; j--)
|
|
|
|
chain[j]._winding = winding;
|
2013-05-03 19:21:44 -04:00
|
|
|
}
|
2014-02-20 12:44:38 -05:00
|
|
|
// Trace closed contours and insert them into the result.
|
|
|
|
var result = new CompoundPath();
|
|
|
|
result.addChildren(PathItem._tracePaths(segments, operator), true);
|
2013-05-03 19:21:44 -04:00
|
|
|
// Delete the proxies
|
2013-05-04 06:38:19 -04:00
|
|
|
path1.remove();
|
2014-02-20 13:35:11 -05:00
|
|
|
if (path2)
|
2014-02-16 13:02:07 -05:00
|
|
|
path2.remove();
|
2013-05-03 19:21:44 -04:00
|
|
|
// And then, we are done.
|
2013-05-04 01:38:29 -04:00
|
|
|
return result.reduce();
|
2014-02-20 13:35:11 -05:00
|
|
|
},
|
2013-05-03 19:16:52 -04:00
|
|
|
|
2014-02-20 13:35:11 -05:00
|
|
|
/**
|
|
|
|
* {@grouptitle Boolean Path Operations}
|
|
|
|
*
|
|
|
|
* Merges the geometry of the specified path from this path's
|
|
|
|
* geometry and returns the result as a new path item.
|
|
|
|
*
|
|
|
|
* @param {PathItem} path the path to unite with
|
|
|
|
* @return {PathItem} the resulting path item
|
|
|
|
*/
|
|
|
|
unite: function(path) {
|
|
|
|
return this._computeBoolean(path, function(w) {
|
|
|
|
return w === 1 || w === 0;
|
|
|
|
}, false);
|
|
|
|
},
|
2013-05-04 00:21:53 -04:00
|
|
|
|
2014-02-20 13:35:11 -05:00
|
|
|
/**
|
|
|
|
* Intersects the geometry of the specified path with this path's
|
|
|
|
* geometry and returns the result as a new path item.
|
|
|
|
*
|
|
|
|
* @param {PathItem} path the path to intersect with
|
|
|
|
* @return {PathItem} the resulting path item
|
|
|
|
*/
|
|
|
|
intersect: function(path) {
|
|
|
|
return this._computeBoolean(path, function(w) {
|
|
|
|
return w === 2;
|
|
|
|
}, false);
|
|
|
|
},
|
2013-05-04 00:21:53 -04:00
|
|
|
|
2014-02-20 13:35:11 -05:00
|
|
|
/**
|
|
|
|
* Subtracts the geometry of the specified path from this path's
|
|
|
|
* geometry and returns the result as a new path item.
|
|
|
|
*
|
|
|
|
* @param {PathItem} path the path to subtract
|
|
|
|
* @return {PathItem} the resulting path item
|
|
|
|
*/
|
|
|
|
subtract: function(path) {
|
|
|
|
return this._computeBoolean(path, function(w) {
|
|
|
|
return w === 1;
|
|
|
|
}, true);
|
|
|
|
},
|
2013-05-04 00:21:53 -04:00
|
|
|
|
2014-02-20 13:35:11 -05:00
|
|
|
// Compound boolean operators combine the basic boolean operations such
|
|
|
|
// as union, intersection, subtract etc.
|
|
|
|
/**
|
|
|
|
* Excludes the intersection of the geometry of the specified path with
|
|
|
|
* this path's geometry and returns the result as a new group item.
|
|
|
|
*
|
|
|
|
* @param {PathItem} path the path to exclude the intersection of
|
|
|
|
* @return {Group} the resulting group item
|
|
|
|
*/
|
|
|
|
exclude: function(path) {
|
|
|
|
return new Group([this.subtract(path), path.subtract(this)]);
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Splits the geometry of this path along the geometry of the specified
|
|
|
|
* path returns the result as a new group item.
|
|
|
|
*
|
|
|
|
* @param {PathItem} path the path to divide by
|
|
|
|
* @return {Group} the resulting group item
|
|
|
|
*/
|
|
|
|
divide: function(path) {
|
|
|
|
return new Group([this.subtract(path), this.intersect(path)]);
|
|
|
|
}
|
2013-05-03 19:16:52 -04:00
|
|
|
});
|