2013-04-18 14:27:19 -04:00
|
|
|
|
|
|
|
|
|
|
|
/*!
|
|
|
|
*
|
|
|
|
* Vector boolean operations on paperjs objects
|
|
|
|
* This is mostly written for clarity (I hope it is clear) and compatibility,
|
|
|
|
* not optimised for performance, and has to be tested heavily for stability.
|
|
|
|
* (Looking up to Java's Area path boolean algorithms for stability,
|
|
|
|
* but the code is too complex —mainly because the operations are stored and
|
|
|
|
* enumerable, such as quadraticCurveTo, cubicCurveTo etc.; and is largely
|
|
|
|
* undocumented to directly adapt from)
|
|
|
|
*
|
|
|
|
* Supported
|
|
|
|
* - paperjs Path objects
|
|
|
|
* - Boolean Union operations
|
|
|
|
* - Boolean Intersection operations
|
|
|
|
* - handles path complexity quite nicely
|
|
|
|
*
|
|
|
|
* Not supported yet ( which I would like to see supported )
|
|
|
|
* - Compound Paths as input ( however compound paths are correctly handled in the output )
|
|
|
|
* - Self-intersecting Paths
|
|
|
|
* - Boolean Subtraction operation ( depends on compound paths as input )
|
|
|
|
* - Paths are clones of each other that ovelap exactly on top of each other!
|
|
|
|
*
|
|
|
|
* In the Not-supported-yet list, the first three can be easily implemented,
|
|
|
|
* as for the last point, I need help! Thanks! :)
|
|
|
|
*
|
|
|
|
* ------
|
|
|
|
* Harikrishnan Gopalakrishnan
|
2013-04-20 13:54:45 -04:00
|
|
|
* http://hkrish.com/playground/paperjs/booleanStudy.html
|
2013-04-18 14:27:19 -04:00
|
|
|
*
|
|
|
|
* ------
|
|
|
|
* Paperjs
|
|
|
|
* Copyright (c) 2011, Juerg Lehni & Jonathan Puckey
|
|
|
|
* http://paperjs.org/license/
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* BooleanOps defines the boolean operator functions to use.
|
|
|
|
* A boolean operator is a function f( link:Link, isInsidePath1:Boolean, isInsidePath2:Boolean ) :
|
|
|
|
* should return a Boolean value indicating whether to keep the link or not.
|
|
|
|
* return true - keep the path
|
|
|
|
* return false - discard the path
|
|
|
|
*/
|
|
|
|
var BooleanOps = {
|
|
|
|
Union: function( lnk, isInsidePath1, isInsidePath2 ){
|
|
|
|
if( isInsidePath1 || isInsidePath2 ){
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
|
|
|
|
Intersection: function( lnk, isInsidePath1, isInsidePath2 ){
|
|
|
|
if( !isInsidePath1 && !isInsidePath2 ){
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
2013-04-19 13:49:44 -04:00
|
|
|
},
|
|
|
|
|
|
|
|
// path1 - path2
|
|
|
|
Subtraction: function( lnk, isInsidePath1, isInsidePath2 ){
|
|
|
|
var lnkid = lnk.id;
|
|
|
|
if( lnkid === 1 && isInsidePath2 ){
|
|
|
|
return false;
|
|
|
|
} else if( lnkid === 2 && !isInsidePath1 ){
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
2013-04-18 14:27:19 -04:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The datastructure for boolean computation:
|
|
|
|
* Graph - List of Links
|
|
|
|
* Link - Connects 2 Nodes, represents a Curve
|
|
|
|
* Node - Connects 2 Links, represents a Segment
|
|
|
|
*/
|
|
|
|
|
|
|
|
var NORMAL_NODE = 1;
|
|
|
|
var INTERSECTION_NODE = 2;
|
|
|
|
var IntersectionID = 1;
|
|
|
|
var UNIQUE_ID = 1;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Nodes in the graph are analogous to Segment objects
|
|
|
|
* with additional linkage information to track intersections etc.
|
|
|
|
* (enough to do a complete graph traversal)
|
|
|
|
* @param {Point} _point
|
|
|
|
* @param {Point} _handleIn
|
|
|
|
* @param {Point} _handleOut
|
|
|
|
* @param {Any} _id
|
|
|
|
*/
|
2013-04-19 13:49:44 -04:00
|
|
|
function Node( _point, _handleIn, _handleOut, _id, isBaseContour ){
|
2013-04-18 14:27:19 -04:00
|
|
|
this.id = _id;
|
2013-04-19 13:49:44 -04:00
|
|
|
this.isBaseContour = isBaseContour;
|
2013-04-18 14:27:19 -04:00
|
|
|
this.type = NORMAL_NODE;
|
|
|
|
this.point = _point;
|
|
|
|
this.handleIn = _handleIn; // handleIn
|
|
|
|
this.handleOut = _handleOut; // handleOut
|
|
|
|
this.linkIn = null; // aka linkIn
|
|
|
|
this.linkOut = null; // linkOut
|
|
|
|
this.uniqueID = ++UNIQUE_ID;
|
|
|
|
|
|
|
|
// In case of an intersection this will be a merged node.
|
|
|
|
// And we need space to save the "other Node's" parameters before merging.
|
|
|
|
this.idB = null;
|
2013-04-19 13:49:44 -04:00
|
|
|
this.isBaseContourB = false;
|
2013-04-18 14:27:19 -04:00
|
|
|
// this.pointB = this.point; // point should be the same
|
|
|
|
this.handleBIn = null;
|
|
|
|
this.handleBOut = null;
|
|
|
|
this.linkBIn = null;
|
|
|
|
this.linkBOut = null;
|
|
|
|
|
|
|
|
this._segment = null;
|
|
|
|
|
|
|
|
this.getSegment = function( recalculate ){
|
|
|
|
if( this.type === INTERSECTION_NODE && recalculate ){
|
|
|
|
// point this.linkIn and this.linkOut to those active ones
|
|
|
|
// also point this.handleIn and this.handleOut to correct in and out handles
|
|
|
|
// If a link is null, make sure the corresponding handle is also null
|
|
|
|
this.handleIn = (this.linkIn)? this.handleIn : null;
|
|
|
|
this.handleOut = (this.linkOut)? this.handleOut : null;
|
|
|
|
this.handleBIn = (this.linkBIn)? this.handleBIn : null;
|
|
|
|
this.handleBOut = (this.linkBOut)? this.handleBOut : null;
|
|
|
|
// Select the valid links
|
|
|
|
this.linkIn = this.linkIn || this.linkBIn; // linkIn
|
|
|
|
this.linkOut = this.linkOut || this.linkBOut; // linkOut
|
|
|
|
// Also update the references in links to point to "this" Node
|
2013-04-21 11:45:58 -04:00
|
|
|
if( !this.linkIn || !this.linkOut ){
|
|
|
|
throw { name: 'Boolean Error', message: 'No matching link found at ixID: ' +
|
|
|
|
this._intersectionID + " point: " + this.point.toString() };
|
|
|
|
}
|
2013-04-18 14:27:19 -04:00
|
|
|
this.linkIn.nodeOut = this; // linkIn.nodeEnd
|
|
|
|
this.linkOut.nodeIn = this; // linkOut.nodeStart
|
|
|
|
this.handleIn = this.handleIn || this.handleBIn;
|
|
|
|
this.handleOut = this.handleOut || this.handleBOut;
|
2013-04-19 13:49:44 -04:00
|
|
|
this.isBaseContour = this.isBaseContour | this.isBaseContourB;
|
2013-04-18 14:27:19 -04:00
|
|
|
}
|
|
|
|
this._segment = this._segment || new Segment( this.point, this.handleIn, this.handleOut );
|
|
|
|
return this._segment;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Links in the graph are analogous to CUrve objects
|
|
|
|
* @param {Node} _nodeIn
|
|
|
|
* @param {Node} _nodeOut
|
|
|
|
* @param {Any} _id
|
|
|
|
*/
|
2013-04-19 13:49:44 -04:00
|
|
|
function Link( _nodeIn, _nodeOut, _id, isBaseContour ) {
|
2013-04-18 14:27:19 -04:00
|
|
|
this.id = _id;
|
2013-04-19 13:49:44 -04:00
|
|
|
this.isBaseContour = isBaseContour;
|
2013-04-18 14:27:19 -04:00
|
|
|
this.nodeIn = _nodeIn; // nodeStart
|
|
|
|
this.nodeOut = _nodeOut; // nodeEnd
|
|
|
|
this.nodeIn.linkOut = this; // nodeStart.linkOut
|
|
|
|
this.nodeOut.linkIn = this; // nodeEnd.linkIn
|
|
|
|
this._curve = null;
|
|
|
|
this.intersections = [];
|
|
|
|
|
|
|
|
// for reusing the paperjs function we need to (temperorily) build a Curve object from this Link
|
|
|
|
// for performance reasons we cache it.
|
|
|
|
this.getCurve = function() {
|
|
|
|
this._curve = this._curve || new Curve( this.nodeIn.getSegment(), this.nodeOut.getSegment() );
|
|
|
|
return this._curve;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* makes a graph. Only works on paths, for compound paths we need to
|
|
|
|
* make graphs for each of the child paths and merge them.
|
|
|
|
* @param {Path} path
|
|
|
|
* @param {Integer} id
|
|
|
|
* @return {Array} Links
|
|
|
|
*/
|
2013-04-19 13:49:44 -04:00
|
|
|
function makeGraph( path, id, isBaseContour ){
|
2013-04-18 14:27:19 -04:00
|
|
|
var graph = [];
|
|
|
|
var segs = path.segments, prevNode = null, firstNode = null, nuLink, nuNode;
|
|
|
|
for( i = 0, l = segs.length; i < l; i++ ){
|
2013-04-21 08:38:57 -04:00
|
|
|
// var nuSeg = segs[i].clone();
|
|
|
|
var nuSeg = segs[i];
|
2013-04-19 13:49:44 -04:00
|
|
|
nuNode = new Node( nuSeg.point, nuSeg.handleIn, nuSeg.handleOut, id, isBaseContour );
|
2013-04-18 14:27:19 -04:00
|
|
|
if( prevNode ) {
|
2013-04-19 13:49:44 -04:00
|
|
|
nuLink = new Link( prevNode, nuNode, id, isBaseContour );
|
2013-04-18 14:27:19 -04:00
|
|
|
graph.push( nuLink );
|
|
|
|
}
|
|
|
|
prevNode = nuNode;
|
|
|
|
if( !firstNode ){
|
|
|
|
firstNode = nuNode;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// the path is closed
|
2013-04-19 13:49:44 -04:00
|
|
|
nuLink = new Link( prevNode, firstNode, id, isBaseContour );
|
2013-04-18 14:27:19 -04:00
|
|
|
graph.push( nuLink );
|
|
|
|
return graph;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Calculates the Union of two paths
|
|
|
|
* Boolean API.
|
|
|
|
* @param {Path} path1
|
|
|
|
* @param {Path} path2
|
|
|
|
* @return {CompoundPath} union of path1 & path2
|
|
|
|
*/
|
2013-04-21 08:38:57 -04:00
|
|
|
function boolUnion( path1, path2 ){
|
2013-04-18 14:27:19 -04:00
|
|
|
return computeBoolean( path1, path2, BooleanOps.Union );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Calculates the Intersection between two paths
|
|
|
|
* Boolean API.
|
|
|
|
* @param {Path} path1
|
|
|
|
* @param {Path} path2
|
|
|
|
* @return {CompoundPath} Intersection of path1 & path2
|
|
|
|
*/
|
2013-04-21 08:38:57 -04:00
|
|
|
function boolIntersection( path1, path2 ){
|
2013-04-18 14:27:19 -04:00
|
|
|
return computeBoolean( path1, path2, BooleanOps.Intersection );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-04-19 13:49:44 -04:00
|
|
|
/**
|
|
|
|
* Calculates path1—path2
|
|
|
|
* Boolean API.
|
|
|
|
* @param {Path} path1
|
|
|
|
* @param {Path} path2
|
|
|
|
* @return {CompoundPath} path1 <minus> path2
|
|
|
|
*/
|
2013-04-21 08:38:57 -04:00
|
|
|
function boolSubtract( path1, path2 ){
|
2013-04-19 13:49:44 -04:00
|
|
|
return computeBoolean( path1, path2, BooleanOps.Subtraction );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-04-18 14:27:19 -04:00
|
|
|
/**
|
|
|
|
* Actual function that computes the boolean
|
|
|
|
* @param {Path} _path1 (cannot be self-intersecting at the moment)
|
|
|
|
* @param {Path} _path2 (cannot be self-intersecting at the moment)
|
|
|
|
* @param {BooleanOps type} operator
|
|
|
|
* @return {CompoundPath} boolean result
|
|
|
|
*/
|
2013-04-21 08:38:57 -04:00
|
|
|
function computeBoolean( _path1, _path2, operator ){
|
2013-04-18 14:27:19 -04:00
|
|
|
IntersectionID = 1;
|
|
|
|
UNIQUE_ID = 1;
|
|
|
|
|
|
|
|
// The boolean operation may modify the original paths
|
|
|
|
var path1 = _path1.clone();
|
|
|
|
var path2 = _path2.clone();
|
2013-04-19 08:46:27 -04:00
|
|
|
// if( !path1.clockwise ){ path1.reverse(); }
|
|
|
|
// if( !path2.clockwise ){ path2.reverse(); }
|
|
|
|
//
|
|
|
|
var i, j, k, l, lnk, crv, node, nuNode, leftLink, rightLink;
|
2013-04-18 14:27:19 -04:00
|
|
|
|
|
|
|
// Prepare the graphs. Graphs are list of Links that retains
|
|
|
|
// full connectivity information. The order of links in a graph is not important
|
|
|
|
// That allows us to sort and merge graphs and 'splice' links with their splits easily.
|
|
|
|
// Also, this is the place to resolve self-intersecting paths
|
2013-04-19 13:49:44 -04:00
|
|
|
var graph = [], path1Children, path2Children, base;
|
2013-04-19 08:46:27 -04:00
|
|
|
if( path1 instanceof CompoundPath ){
|
|
|
|
path1Children = path1.children;
|
2013-04-19 13:49:44 -04:00
|
|
|
for (i = 0, base = true, l = path1Children.length; i < l; i++, base = false) {
|
2013-04-19 08:46:27 -04:00
|
|
|
path1Children[i].closed = true;
|
2013-04-19 13:49:44 -04:00
|
|
|
graph = graph.concat( makeGraph( path1Children[i], 1, base ) );
|
2013-04-19 08:46:27 -04:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
path1.closed = true;
|
2013-04-19 13:49:44 -04:00
|
|
|
// path1.clockwise = true;
|
|
|
|
graph = graph.concat( makeGraph( path1, 1, 1, true ) );
|
2013-04-19 08:46:27 -04:00
|
|
|
}
|
|
|
|
|
2013-04-19 13:49:44 -04:00
|
|
|
// if operator === BooleanOps.Subtraction, then reverse path2
|
|
|
|
// so that the nodes and links will link correctly
|
|
|
|
var reverse = ( operator === BooleanOps.Subtraction )? true: false;
|
2013-04-19 08:46:27 -04:00
|
|
|
if( path2 instanceof CompoundPath ){
|
|
|
|
path2Children = path2.children;
|
2013-04-19 13:49:44 -04:00
|
|
|
for (i = 0, base = true, l = path2Children.length; i < l; i++, base = false) {
|
2013-04-19 08:46:27 -04:00
|
|
|
path2Children[i].closed = true;
|
2013-04-19 13:49:44 -04:00
|
|
|
if( reverse ){ path2Children[i].reverse(); }
|
|
|
|
graph = graph.concat( makeGraph( path2Children[i], 2, i + 1, base ) );
|
2013-04-19 08:46:27 -04:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
path2.closed = true;
|
2013-04-19 13:49:44 -04:00
|
|
|
// path2.clockwise = true;
|
|
|
|
if( reverse ){ path2.reverse(); }
|
|
|
|
graph = graph.concat( makeGraph( path2, 2, 1, true ) );
|
2013-04-19 08:46:27 -04:00
|
|
|
}
|
|
|
|
|
2013-04-21 07:51:51 -04:00
|
|
|
// console.log( "Total curves: " + graph.length );
|
2013-04-19 20:04:51 -04:00
|
|
|
|
2013-04-18 14:27:19 -04:00
|
|
|
// Sort function to sort intersections according to the 'parameter'(t) in a link (curve)
|
2013-04-21 07:51:51 -04:00
|
|
|
function ixSort( a, b ){ return a.parameter - b.parameter; }
|
2013-04-18 14:27:19 -04:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Pass 1:
|
|
|
|
* Calculate the intersections for all graphs
|
|
|
|
* TODO: test if this takes are of self intersecting paths - NO
|
|
|
|
* And since it doesn't take self-intersecting curves, we need to only calculate
|
|
|
|
* intersections if the "id" of the links differ.
|
|
|
|
* The rest of the algorithm can easily be modified to resolve self-intersections
|
|
|
|
*/
|
|
|
|
for ( i = graph.length - 1; i >= 0; i--) {
|
|
|
|
var c1 = graph[i].getCurve();
|
|
|
|
var v1 = c1.getValues();
|
|
|
|
for ( j = i -1; j >= 0; j-- ) {
|
|
|
|
if( graph[j].id === graph[i].id ){ continue; }
|
|
|
|
var c2 = graph[j].getCurve();
|
|
|
|
var v2 = c2.getValues();
|
|
|
|
var loc = [];
|
2013-04-21 07:51:51 -04:00
|
|
|
Curve._addIntersections( v1, v2, c1, loc );
|
2013-04-18 14:27:19 -04:00
|
|
|
if( loc.length ){
|
|
|
|
for (k = 0, l=loc.length; k<l; k++) {
|
2013-04-21 07:51:51 -04:00
|
|
|
graph[i].intersections.push( loc[k] );
|
|
|
|
var loc2 = new CurveLocation( c2, null, loc[k].point );
|
2013-04-21 08:57:29 -04:00
|
|
|
// TODO: change this to loc2._id when CurveLocation has an id property
|
2013-04-21 07:51:51 -04:00
|
|
|
loc2.id = loc[k].id;
|
2013-04-18 14:27:19 -04:00
|
|
|
graph[j].intersections.push( loc2 );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-04-19 13:49:44 -04:00
|
|
|
|
2013-04-18 14:27:19 -04:00
|
|
|
/*
|
|
|
|
* Pass 2:
|
|
|
|
* Walk the graph, sort the intersections on each individual link.
|
|
|
|
* for each link that intersects with another one, replace it with new split links.
|
|
|
|
*/
|
2013-04-21 08:57:29 -04:00
|
|
|
var ix, ixPoint, ixHandleI, ixHandleOut, param, isLinear, parts, left, right;
|
2013-04-18 14:27:19 -04:00
|
|
|
for ( i = graph.length - 1; i >= 0; i--) {
|
|
|
|
if( graph[i].intersections.length ){
|
2013-04-21 08:57:29 -04:00
|
|
|
ix = graph[i].intersections;
|
2013-04-19 13:49:44 -04:00
|
|
|
// Sort the intersections if there is more than one
|
|
|
|
if( graph[i].intersections.length > 1 ){ ix.sort( ixSort ); }
|
2013-04-18 14:27:19 -04:00
|
|
|
// Remove the graph link, this link has to be split and replaced with the splits
|
|
|
|
lnk = graph.splice( i, 1 )[0];
|
|
|
|
for (j =0, l=ix.length; j<l && lnk; j++) {
|
|
|
|
crv = lnk.getCurve();
|
|
|
|
// We need to recalculate parameter after each curve split
|
|
|
|
// This operation (except for recalculating the curve parameter),
|
|
|
|
// is fairly similar to Curve.split method, except that it operates on Node and Link objects.
|
2013-04-21 08:57:29 -04:00
|
|
|
param = crv.getParameterOf( ix[j].point );
|
2013-04-19 20:04:51 -04:00
|
|
|
// var param = crv.getNearestLocation( ix[j] ).parameter;
|
2013-04-18 14:27:19 -04:00
|
|
|
if( param === 0.0 || param === 1.0) {
|
|
|
|
// Intersection falls on an existing node
|
|
|
|
// there is no need to split the link
|
|
|
|
nuNode = ( param === 0.0 )? lnk.nodeIn : lnk.nodeOut;
|
|
|
|
nuNode.type = INTERSECTION_NODE;
|
2013-04-21 07:51:51 -04:00
|
|
|
nuNode._intersectionID = ix[j].id;
|
2013-04-18 14:27:19 -04:00
|
|
|
if( param === 1.0 ){
|
|
|
|
leftLink = null;
|
|
|
|
rightLink = lnk;
|
|
|
|
} else {
|
|
|
|
leftLink = lnk;
|
|
|
|
rightLink = null;
|
|
|
|
}
|
|
|
|
} else {
|
2013-04-21 08:57:29 -04:00
|
|
|
isLinear = crv.isLinear();
|
|
|
|
parts = Curve.subdivide(crv.getValues(), param);
|
|
|
|
left = parts[0];
|
|
|
|
right = parts[1];
|
2013-04-18 14:27:19 -04:00
|
|
|
// Make new link and convert handles from absolute to relative
|
2013-04-21 08:57:29 -04:00
|
|
|
ixPoint = new Point( left[6], left[7] );
|
|
|
|
if( !isLinear ){
|
|
|
|
ixHandleIn = new Point(left[4] - ixPoint.x, left[5] - ixPoint.y);
|
|
|
|
ixHandleOut = new Point(right[2] - ixPoint.x, right[3] - ixPoint.y);
|
|
|
|
} else {
|
|
|
|
ixHandleIn = ixHandleOut = null;
|
|
|
|
}
|
|
|
|
nuNode = new Node( ixPoint, ixHandleIn, ixHandleOut, lnk.id, lnk.isBaseContour );
|
2013-04-18 14:27:19 -04:00
|
|
|
nuNode.type = INTERSECTION_NODE;
|
2013-04-21 07:51:51 -04:00
|
|
|
nuNode._intersectionID = ix[j].id;
|
2013-04-18 14:27:19 -04:00
|
|
|
// clear the cached Segment on original end nodes and Update their handles
|
|
|
|
lnk.nodeIn._segment = null;
|
2013-04-21 08:57:29 -04:00
|
|
|
if( !isLinear ){
|
|
|
|
var tmppnt = lnk.nodeIn.point;
|
|
|
|
lnk.nodeIn.handleOut = new Point( left[2] - tmppnt.x, left[3] - tmppnt.y );
|
|
|
|
lnk.nodeOut._segment = null;
|
|
|
|
tmppnt = lnk.nodeOut.point;
|
|
|
|
lnk.nodeOut.handleIn = new Point( right[4] - tmppnt.x, right[5] - tmppnt.y );
|
|
|
|
}
|
2013-04-18 14:27:19 -04:00
|
|
|
// Make new links after the split
|
2013-04-19 13:49:44 -04:00
|
|
|
leftLink = new Link( lnk.nodeIn, nuNode, lnk.id, lnk.isBaseContour );
|
|
|
|
rightLink = new Link( nuNode, lnk.nodeOut, lnk.id, lnk.isBaseContour );
|
2013-04-18 14:27:19 -04:00
|
|
|
}
|
|
|
|
// Add the first split link back to the graph, since we sorted the intersections
|
|
|
|
// already, this link should contain no more intersections to the left.
|
|
|
|
if( leftLink ){
|
|
|
|
graph.splice( i, 0, leftLink );
|
|
|
|
}
|
|
|
|
// continue with the second split link, to see if
|
|
|
|
// there are more intersections to deal with
|
|
|
|
lnk = rightLink;
|
|
|
|
}
|
|
|
|
// Add the last split link back to the graph
|
|
|
|
if( lnk ){
|
|
|
|
graph.splice( i, 0, lnk );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Pass 3:
|
|
|
|
* Merge matching intersection Node Pairs (type is INTERSECTION_NODE &&
|
|
|
|
* a._intersectionID == b._intersectionID )
|
|
|
|
*
|
|
|
|
* Mark each Link(Curve) according to whether it is
|
|
|
|
* case 1. inside Path1 ( and only Path1 )
|
|
|
|
* 2. inside Path2 ( and only Path2 )
|
2013-04-20 13:54:45 -04:00
|
|
|
* 3. outside (normal case)
|
2013-04-18 14:27:19 -04:00
|
|
|
*
|
|
|
|
* Take a test function "operator" which will discard links
|
|
|
|
* according to the above
|
2013-04-20 13:54:45 -04:00
|
|
|
* * Union -> discard cases 1 and 2
|
|
|
|
* * Intersection -> discard case 3
|
|
|
|
* * Path1-Path2 -> discard cases 2, 3[Path2]
|
2013-04-18 14:27:19 -04:00
|
|
|
*/
|
|
|
|
|
|
|
|
// step 1: discard invalid links according to the boolean operator
|
|
|
|
for ( i = graph.length - 1; i >= 0; i--) {
|
|
|
|
lnk = graph[i];
|
|
|
|
crv = lnk.getCurve();
|
|
|
|
// var midPoint = new Point(lnk.nodeIn.point);
|
|
|
|
var midPoint = crv.getPoint( 0.5 );
|
|
|
|
var insidePath1 = (lnk.id === 1 )? false : path1.contains( midPoint );
|
|
|
|
var insidePath2 = (lnk.id === 2 )? false : path2.contains( midPoint );
|
|
|
|
if( !operator( lnk, insidePath1, insidePath2 ) ){
|
|
|
|
// lnk = graph.splice( i, 1 )[0];
|
|
|
|
lnk.INVALID = true;
|
|
|
|
lnk.nodeIn.linkOut = null;
|
|
|
|
lnk.nodeOut.linkIn = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// step 2: Match nodes according to their _intersectionID and merge them together
|
|
|
|
var len = graph.length;
|
|
|
|
while( len-- ){
|
|
|
|
node = graph[len].nodeIn;
|
|
|
|
if( node.type === INTERSECTION_NODE ){
|
|
|
|
var otherNode = null;
|
|
|
|
for (i = len - 1; i >= 0; i--) {
|
|
|
|
var tmpnode = graph[i].nodeIn;
|
|
|
|
if( tmpnode._intersectionID === node._intersectionID &&
|
|
|
|
tmpnode.uniqueID !== node.uniqueID ) {
|
|
|
|
otherNode = tmpnode;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if( otherNode ) {
|
|
|
|
//Check if it is a self-intersecting Node
|
|
|
|
if( node.id === otherNode.id ){
|
|
|
|
// Swap the outgoing links, this will resolve a knot and create two paths,
|
|
|
|
// the portion of the original path on one side of a self crossing is counter-clockwise,
|
|
|
|
// so one of the resulting paths will also be counter-clockwise
|
|
|
|
var tmp = otherNode.linkOut;
|
|
|
|
otherNode.linkOut = node.linkOut;
|
|
|
|
node.linkOut = tmp;
|
|
|
|
tmp = otherNode.handleOut;
|
|
|
|
otherNode.handleOut = node.handleOut;
|
|
|
|
node.handleOut = tmp;
|
|
|
|
node.type = otherNode.type = NORMAL_NODE;
|
|
|
|
node._intersectionID = null;
|
|
|
|
node._segment = otherNode._segment = null;
|
|
|
|
} else {
|
|
|
|
// Merge the nodes together, by adding this node's information to the other node
|
|
|
|
otherNode.idB = node.id;
|
2013-04-19 13:49:44 -04:00
|
|
|
otherNode.isBaseContourB = node.isBaseContour;
|
2013-04-18 14:27:19 -04:00
|
|
|
otherNode.handleBIn = node.handleIn;
|
|
|
|
otherNode.handleBOut = node.handleOut;
|
|
|
|
otherNode.linkBIn = node.linkIn;
|
|
|
|
otherNode.linkBOut = node.linkOut;
|
|
|
|
otherNode._segment = null;
|
|
|
|
if( node.linkIn ){ node.linkIn.nodeOut = otherNode; }
|
|
|
|
if( node.linkOut ){ node.linkOut.nodeIn = otherNode; }
|
|
|
|
// Clear this node's intersectionID, so that we won't iterate over it again
|
|
|
|
node._intersectionID = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Final step: Retrieve the resulting paths from the graph
|
|
|
|
var boolResult = new CompoundPath();
|
2013-04-19 08:46:27 -04:00
|
|
|
var firstNode = true, nextNode, foundBasePath = false;
|
2013-04-18 14:27:19 -04:00
|
|
|
while( firstNode ){
|
|
|
|
firstNode = nextNode = null;
|
|
|
|
len = graph.length;
|
|
|
|
while( len-- ){
|
2013-04-21 11:45:58 -04:00
|
|
|
lnk = graph[len];
|
|
|
|
if( !lnk.INVALID && !lnk.nodeIn.visited && !firstNode ){
|
|
|
|
if( !foundBasePath && lnk.isBaseContour === 1 ){
|
|
|
|
firstNode = lnk.nodeIn;
|
2013-04-19 08:46:27 -04:00
|
|
|
foundBasePath = true;
|
|
|
|
break;
|
|
|
|
} else if(foundBasePath){
|
2013-04-21 11:45:58 -04:00
|
|
|
firstNode = lnk.nodeIn;
|
2013-04-19 08:46:27 -04:00
|
|
|
break;
|
|
|
|
}
|
2013-04-18 14:27:19 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if( firstNode ){
|
|
|
|
var path = new Path();
|
|
|
|
path.add( firstNode.getSegment( true ) );
|
|
|
|
firstNode.visited = true;
|
|
|
|
nextNode = firstNode.linkOut.nodeOut;
|
|
|
|
while( firstNode.uniqueID !== nextNode.uniqueID ){
|
|
|
|
path.add( nextNode.getSegment( true ) );
|
|
|
|
nextNode.visited = true;
|
|
|
|
nextNode = nextNode.linkOut.nodeOut;
|
|
|
|
}
|
|
|
|
path.closed = true;
|
|
|
|
// path.clockwise = true;
|
|
|
|
boolResult.addChild( path );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
boolResult = boolResult.reduce();
|
|
|
|
|
2013-04-21 11:45:58 -04:00
|
|
|
// Remove the paths we duplicated
|
2013-04-21 08:38:57 -04:00
|
|
|
path1.remove();
|
|
|
|
path2.remove();
|
|
|
|
|
2013-04-18 14:27:19 -04:00
|
|
|
return boolResult;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-04-19 08:46:27 -04:00
|
|
|
function markPoint( pnt, t, c, tc, remove ) {
|
|
|
|
if( !pnt ) return;
|
|
|
|
c = c || '#000';
|
|
|
|
if( remove === undefined ){ remove = true; }
|
|
|
|
var cir = new Path.Circle( pnt, 2 );
|
|
|
|
cir.style.fillColor = c;
|
|
|
|
cir.style.strokeColor = tc;
|
|
|
|
if( t !== undefined || t !== null ){
|
|
|
|
var text = new PointText( pnt.add([0, -3]) );
|
|
|
|
text.justification = 'center';
|
|
|
|
text.fillColor = c;
|
|
|
|
text.content = t;
|
|
|
|
if( remove ){
|
|
|
|
text.removeOnMove();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if( remove ) {
|
|
|
|
cir.removeOnMove();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-04-18 14:27:19 -04:00
|
|
|
// Same as the paperjs' Numerical class,
|
|
|
|
// added here because I can't access the original from this scope
|
|
|
|
var Numerical = {
|
|
|
|
TOLERANCE : 10e-6
|
|
|
|
};
|
|
|
|
|
|
|
|
// paperjs' Curve._addIntersections modified to return just intersection Point with a
|
|
|
|
// unique id.
|
2013-04-21 07:51:51 -04:00
|
|
|
paper.Curve._addIntersections = function(v1, v2, curve, locations) {
|
2013-04-21 08:38:57 -04:00
|
|
|
var bounds1 = Curve.getBounds(v1),
|
|
|
|
bounds2 = Curve.getBounds(v2);
|
|
|
|
if (bounds1.touches(bounds2)) {
|
2013-04-21 07:51:51 -04:00
|
|
|
// See if both curves are flat enough to be treated as lines.
|
|
|
|
if (Curve.isFlatEnough(v1, /*#=*/ Numerical.TOLERANCE)
|
2013-04-21 08:38:57 -04:00
|
|
|
&& Curve.isFlatEnough(v2, /*#=*/ Numerical.TOLERANCE)) {
|
2013-04-21 07:51:51 -04:00
|
|
|
// See if the parametric equations of the lines interesct.
|
2013-04-21 08:38:57 -04:00
|
|
|
var point = new Line(v1[0], v1[1], v1[6], v1[7], false)
|
|
|
|
.intersect(new Line(v2[0], v2[1], v2[6], v2[7], false),
|
2013-04-18 14:27:19 -04:00
|
|
|
// Filter out beginnings of the curves, to avoid
|
|
|
|
// duplicate solutions where curves join.
|
|
|
|
true, false);
|
2013-04-21 08:38:57 -04:00
|
|
|
if (point){
|
2013-04-21 07:51:51 -04:00
|
|
|
// Passing null for parameter leads to lazy determination of
|
|
|
|
// parameter values in CurveLocation#getParameter() only
|
|
|
|
// once they are requested.
|
|
|
|
var cl = new CurveLocation(curve, null, point);
|
|
|
|
cl.id = UNIQUE_ID++;
|
|
|
|
locations.push( cl );
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Subdivide both curves, and see if they intersect.
|
|
|
|
var v1s = Curve.subdivide(v1),
|
2013-04-21 08:38:57 -04:00
|
|
|
v2s = Curve.subdivide(v2);
|
2013-04-21 07:51:51 -04:00
|
|
|
for (var i = 0; i < 2; i++)
|
|
|
|
for (var j = 0; j < 2; j++)
|
|
|
|
this._addIntersections(v1s[i], v2s[j], curve, locations);
|
2013-04-21 08:38:57 -04:00
|
|
|
}
|
2013-04-21 07:51:51 -04:00
|
|
|
}
|
2013-04-21 08:38:57 -04:00
|
|
|
return locations;
|
|
|
|
};
|