diff --git a/boolean/.exrc b/boolean/.exrc deleted file mode 100644 index 59c101b9..00000000 --- a/boolean/.exrc +++ /dev/null @@ -1,3 +0,0 @@ - -set tags+=./tags - diff --git a/boolean/.gitignore b/boolean/.gitignore deleted file mode 100644 index 6e92f57d..00000000 --- a/boolean/.gitignore +++ /dev/null @@ -1 +0,0 @@ -tags diff --git a/boolean/Boolean.js b/boolean/Boolean.js deleted file mode 100644 index c6428fa4..00000000 --- a/boolean/Boolean.js +++ /dev/null @@ -1,696 +0,0 @@ - - -/*! - * - * 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 and CompoundPath objects - * - Boolean Union - * - Boolean Intersection - * - Boolean Subtraction - * - Resolving a self-intersecting Path - * - * Not supported yet ( which I would like to see supported ) - * - Boolean operations on self-intersecting Paths, these has to be resolved first - * - Paths are clones of each other that ovelap exactly on top of each other! - * - * ------ - * Harikrishnan Gopalakrishnan - * http://hkrish.com/playground/paperjs/booleanStudy.html - * - * ------ - * 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; - }, - - // 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; - } -}; - -/** - * 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 - */ - function Node( _point, _handleIn, _handleOut, _id, isBaseContour ){ - this.id = _id; - this.isBaseContour = isBaseContour; - 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; - this.isBaseContourB = false; - // 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 - if( !this.linkIn || !this.linkOut ){ - // markPoint( this.point, this._intersectionID ); - throw { name: 'Boolean Error', message: 'No matching link found at ixID: ' + - this._intersectionID + " point: " + this.point.toString() }; - } - this.linkIn.nodeOut = this; // linkIn.nodeEnd - this.linkOut.nodeIn = this; // linkOut.nodeStart - this.handleIn = this.handleIn || this.handleBIn; - this.handleOut = this.handleOut || this.handleBOut; - this.isBaseContour = this.isBaseContour | this.isBaseContourB; - } - 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 - */ - function Link( _nodeIn, _nodeOut, _id, isBaseContour, _winding ) { - this.id = _id; - this.isBaseContour = isBaseContour; - this.winding = _winding; - 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 - */ - function makeGraph( path, id, isBaseContour ){ - var graph = []; - var segs = path.segments, prevNode = null, firstNode = null, nuLink, nuNode, - winding = path.clockwise; - for( i = 0, l = segs.length; i < l; i++ ){ - // var nuSeg = segs[i].clone(); - var nuSeg = segs[i]; - nuNode = new Node( nuSeg.point, nuSeg.handleIn, nuSeg.handleOut, id, isBaseContour ); - if( prevNode ) { - nuLink = new Link( prevNode, nuNode, id, isBaseContour, winding ); - graph.push( nuLink ); - } - prevNode = nuNode; - if( !firstNode ){ - firstNode = nuNode; - } - } - // the path is closed - nuLink = new Link( prevNode, firstNode, id, isBaseContour, winding ); - 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 - */ - function boolUnion( path1, path2 ){ - 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 - */ - function boolIntersection( path1, path2 ){ - return computeBoolean( path1, path2, BooleanOps.Intersection ); -} - - -/** - * Calculates path1—path2 - * Boolean API. - * @param {Path} path1 - * @param {Path} path2 - * @return {CompoundPath} path1 path2 - */ - function boolSubtract( path1, path2 ){ - return computeBoolean( path1, path2, BooleanOps.Subtraction ); -} - -/** - * To deal with a HTML 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 opposit winding direction ( already handled by paperjs ) - * islands has to have same winding direction ( as the first child of the path ) - * - * Does NOT handle selfIntersecting CompoundPaths. - * - * @param {CompoundPath} path - Input CompoundPath, Note: This path could be modified if need be. - * @return {boolean} the winding direction of the base contour( true if clockwise ) - */ - function reorientCompoundPath( path ){ - if( !(path instanceof CompoundPath) ){ return path.clockwise; } - var children = path.children, len = children.length, baseWinding; - var bounds = new Array( len ); - var tmparray = new Array( len ); - baseWinding = children[0].clockwise; - // Omit the first path - for (i = 0; i < len; i++) { - bounds[i] = children[i].bounds; - tmparray[i] = 0; - } - for (i = 0; i < len; i++) { - var p1 = children[i]; - for (j = 0; j < len; j++) { - var p2 = children[j]; - if( i !== j && bounds[i].contains( bounds[j] ) ){ - tmparray[j]++; - } - } - } - for (i = 1; i < len; i++) { - if ( tmparray[i] % 2 === 0 ) { - children[i].clockwise = baseWinding; - } - } - return baseWinding; -} - -/** - * 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 - */ - function computeBoolean( _path1, _path2, operator ){ - IntersectionID = 1; - UNIQUE_ID = 1; - - // We work on duplicate paths since the algorithm may modify the original paths - var path1 = _path1.clone(); - var path2 = _path2.clone(); - - var i, j, k, l, lnk, crv, node, nuNode, leftLink, rightLink; - var path1Clockwise = true, path2Clockwise = true; - - // If one of the operands is empty, resolve self-intersections on the second operand - var childCount1 = (_path1 instanceof CompoundPath)? _path1.children.length : _path1.curves.length; - var childCount2 = (_path2 instanceof CompoundPath)? _path2.children.length : _path2.curves.length; - var resolveSelfIntersections = !childCount1 | !childCount2; - - // Reorient the compound paths, i.e. make all the islands wind in the same direction - // and holes in the opposit direction. - // Do this only if we are not resolving selfIntersections: - // Resolving self-intersections work on compound paths, but, we might get different results! - if( !resolveSelfIntersections ){ - path1Clockwise = reorientCompoundPath( path1 ); - path2Clockwise = reorientCompoundPath( path2 ); - } - - // Cache the bounding rectangle of paths - // so we can make the test for containment quite a bit faster - path1._bounds = (childCount1)? path1.bounds : null; - path2._bounds = (childCount2)? path2.bounds : null; - - // 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 - var graph = [], path1Children, path2Children, base; - if( path1 instanceof CompoundPath ){ - path1Children = path1.children; - for (i = 0, base = true, l = path1Children.length; i < l; i++, base = false) { - path1Children[i].closed = true; - graph = graph.concat( makeGraph( path1Children[i], 1, base ) ); - } - } else { - path1.closed = true; - path1Clockwise = path1.clockwise; - // path1.clockwise = true; - graph = graph.concat( makeGraph( path1, 1, true ) ); - } - - // if operator === BooleanOps.Subtraction, then reverse path2 - // so that the nodes and links will link correctly - var reverse = ( operator === BooleanOps.Subtraction )? true: false; - path2Clockwise = (reverse)? !path2Clockwise : path2Clockwise; - if( path2 instanceof CompoundPath ){ - path2Children = path2.children; - for (i = 0, base = true, l = path2Children.length; i < l; i++, base = false) { - path2Children[i].closed = true; - if( reverse ){path2Children[i].reverse(); } - graph = graph.concat( makeGraph( path2Children[i], 2, base ) ); - } - } else { - path2.closed = true; - // path2.clockwise = true; - if( reverse ){ path2.reverse(); } - path2Clockwise = path2.clockwise; - graph = graph.concat( makeGraph( path2, 2, true ) ); - } - - window.g = graph; - - // Sort function to sort intersections according to the 'parameter'(t) in a link (curve) - function ixSort( a, b ){ return a.parameter - b.parameter; } - - /* - * Pass 1: - * Calculate the intersections for all graphs - */ - var ix, loc, loc2, ixCount = 0; - 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( !resolveSelfIntersections && graph[j].id === graph[i].id ){ continue; } - var c2 = graph[j].getCurve(); - var v2 = c2.getValues(); - loc = []; - Curve.getIntersections( v1, v2, c1, loc ); - if( loc.length ){ - for (k = 0, l=loc.length; k= 0; i--) { - if( graph[i].intersections.length ){ - ix = graph[i].intersections; - // Sort the intersections if there is more than one - if( graph[i].intersections.length > 1 ){ ix.sort( ixSort ); } - // Remove the graph link, this link has to be split and replaced with the splits - lnk = graph.splice( i, 1 )[0]; - - nix = lnk.nodeIn.point.x; niy = lnk.nodeIn.point.y; - nox = lnk.nodeOut.point.x; noy = lnk.nodeOut.point.y; - niho = lnk.nodeIn.handleOut; nohi = lnk.nodeOut.handleIn; - nihox = nihoy = nohix = nohiy = 0; - isLinear = true; - if( niho ){ nihox = niho.x; nihoy = niho.y; isLinear = false; } - if( nohi ){ nohix = nohi.x; nohiy = nohi.y; isLinear = false; } - values = [ nix, niy, nihox + nix, nihoy + niy, - nohix + nox, nohiy + noy, nox, noy ]; - - for (j =0, l=ix.length; j discard cases 1 and 2 - * * Intersection -> discard case 3 - * * Path1-Path2 -> discard cases 2, 3[Path2] - */ - - // step 1: discard invalid links according to the boolean operator - for ( i = graph.length - 1; i >= 0; i-- ) { - var insidePath1 = false, insidePath2 = false, contains; - lnk = graph[i]; - // if( lnk.SKIP_OPERATOR ) { continue; } - if( !lnk.INVALID ) { - crv = lnk.getCurve(); - // var midPoint = new Point(lnk.nodeIn.point); - var midPoint = crv.getPoint( 0.5 ); - // If on a base curve, consider points on the curve and inside, - // if not —for example a hole, points on the curve falls outside - if( lnk.id !== 1 ){ - contains = path1.contains( midPoint ); - insidePath1 = (lnk.winding === path1Clockwise)? contains : - contains && !testOnCurve( path1, midPoint ); - } - if( lnk.id !== 2 ){ - contains = path2.contains( midPoint ); - insidePath2 = (lnk.winding === path2Clockwise)? contains : - contains && !testOnCurve( path2, midPoint ); - } - } - if( lnk.INVALID || !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; - otherNode.isBaseContourB = node.isBaseContour; - 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(); - var firstNode = true, nextNode, foundBasePath = false; - while( firstNode ){ - firstNode = nextNode = null; - len = graph.length; - while( len-- ){ - lnk = graph[len]; - if( !lnk.INVALID && !lnk.nodeIn.visited && !firstNode ){ - if( !foundBasePath && lnk.isBaseContour ){ - firstNode = lnk.nodeIn; - foundBasePath = true; - break; - } else if(foundBasePath){ - firstNode = lnk.nodeIn; - break; - } - } - } - if( firstNode ){ - var path = new Path(); - path.add( firstNode.getSegment( true ) ); - firstNode.visited = true; - nextNode = firstNode.linkOut.nodeOut; - var linkCount = graph.length + 1; - while( firstNode.uniqueID !== nextNode.uniqueID && linkCount-- ){ - path.add( nextNode.getSegment( true ) ); - nextNode.visited = true; - if( !nextNode.linkOut ){ - throw { name: 'Boolean Error', message: 'No link found at node id: ' + nextNode.id }; - } - nextNode = nextNode.linkOut.nodeOut; - } - path.closed = true; - // path.clockwise = true; - if( path.segments.length > 1 && linkCount >= 0 ){ // avoid stray segments and incomplete paths - if( path.segments.length === 2 ){ - } - if( path.segments.length > 2 || !path.curves[0].isLinear() ){ - boolResult.addChild( path ); - } - } - } - } - boolResult = boolResult.reduce(); - - // Remove the paths we duplicated - path1.remove(); - path2.remove(); - - return boolResult; -} - - -// var getLineIntersections = function(v1, v2, curve, locations) { -// var result, a1x, a2x, b1x, b2x, a1y, a2y, b1y, b2y; -// a1x = v1[0]; a1y = v1[1]; -// a2x = v1[6]; a2y = v1[7]; -// b1x = v2[0]; b1y = v2[1]; -// b2x = v2[6]; b2y = v2[7]; -// var ua_t = (b2x - b1x) * (a1y - b1y) - (b2y - b1y) * (a1x - b1x); -// var ub_t = (a2x - a1x) * (a1y - b1y) - (a2y - a1y) * (a1x - b1x); -// var u_b = (b2y - b1y) * (a2x - a1x) - (b2x - b1x) * (a2y - a1y); -// if ( u_b !== 0 ) { -// var ua = ua_t / u_b; -// var ub = ub_t / u_b; -// if ( 0 <= ua && ua <= 1 && 0 <= ub && ub <= 1 ) { -// locations.push( new CurveLocation(curve, null, new Point(a1x + ua * (a2x - a1x), a1y + ua * (a2y - a1y))) ); -// } -// } -// }; - -function testOnCurve( path, point ){ - var res = 0; - var crv = path.getCurves(); - var i = 0; - var bounds = path._bounds; - if( bounds && bounds.contains( point ) ){ - for( i = 0; i < crv.length && !res; i++ ){ - var crvi = crv[i]; - if( crvi.bounds.contains( point ) && crvi.getParameterOf( point ) ){ - res = 1; - } - } - } - return res; -} diff --git a/boolean/Boolean2.js b/boolean/Boolean2.js deleted file mode 100644 index 5266772b..00000000 --- a/boolean/Boolean2.js +++ /dev/null @@ -1,332 +0,0 @@ - - -function sortIx( a, b ) { return b.parameter - a.parameter; } - -function splitPath( _ixs, other ) { - other = other || false; - var i, j, k, l, len, ixs, ix, path, crv, vals; - var ixPoint, nuSeg; - var paths = {}, lastPathId = null; - for (i = 0, l = _ixs.length; i < l; i++) { - ix = ( other )? _ixs[i].getIntersection() : _ixs[i]; - if( !paths[ix.path.id] ){ - paths[ix.path.id] = ix.path; - } - if( !ix.curve._ixParams ){ix.curve._ixParams = []; } - ix.curve._ixParams.push( { parameter: ix.parameter, pair: ix.getIntersection() } ); - } - for (k in paths) { - if( !paths.hasOwnProperty( k ) ){ continue; } - path = paths[k]; - var lastNode = path.lastSegment, firstNode = path.firstSegment; - var nextNode = null, left = null, right = null, parts = null, isLinear; - var handleIn, handleOut; - while( nextNode !== firstNode){ - nextNode = ( nextNode )? nextNode.previous: lastNode; - if( nextNode.curve._ixParams ){ - ixs = nextNode.curve._ixParams; - ixs.sort( sortIx ); - crv = nextNode.getCurve(); - isLinear = crv.isLinear(); - crv = vals = null; - for (i = 0, l = ixs.length; i < l; i++) { - ix = ixs[i]; - crv = nextNode.getCurve(); - if( !vals ) vals = crv.getValues(); - if( ix.parameter === 0.0 || ix.parameter === 1.0 ){ - // Intersection is on an existing node - // no need to create a new segment, - // we just link the corresponding intersections together - nuSeg = ( ix.parameter === 0.0 )? crv.segment1 : crv.segment2; - nuSeg._ixPair = ix.pair; - nuSeg._ixPair._segment = nuSeg; - } else { - parts = Curve.subdivide( vals, ix.parameter ); - left = parts[0]; - right = parts[1]; - handleIn = handleOut = null; - ixPoint = new Point( right[0], right[1] ); - if( !isLinear ){ - crv.segment1.handleOut = new Point( left[2] - left[0], left[3] - left[1] ); - crv.segment2.handleIn = new Point( right[4] - right[6], right[5] - right[7] ); - handleIn = new Point( left[4] - ixPoint.x, left[5] - ixPoint.y ); - handleOut = new Point( right[2] - ixPoint.x, right[3] - ixPoint.y ); - } - nuSeg = new Segment( ixPoint, handleIn, handleOut ); - nuSeg._ixPair = ix.pair; - nuSeg._ixPair._segment = nuSeg; - path.insert( nextNode.index + 1, nuSeg ); - } - for (j = i + 1; j < l; j++) { - ixs[j].parameter = ixs[j].parameter / ix.parameter; - } - vals = left; - } - } - } - } -} - -/** - * To deal with a HTML 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 opposit winding direction ( already handled by paperjs ) - * islands has to have same winding direction ( as the first child of the path ) - * - * Does NOT handle selfIntersecting CompoundPaths. - * - * @param {CompoundPath} path - Input CompoundPath, Note: This path could be modified if need be. - * @return {boolean} the winding direction of the base contour( true if clockwise ) - */ - function reorientCompoundPath( path ){ - if( !(path instanceof CompoundPath) ){ - path.closed = true; - return path.clockwise; - } - var children = path.children, len = children.length, baseWinding; - var bounds = new Array( len ); - var tmparray = new Array( len ); - baseWinding = children[0].clockwise; - // Omit the first path - for (i = 0; i < len; i++) { - children[i].closed = true; - bounds[i] = children[i].bounds; - tmparray[i] = 0; - } - for (i = 0; i < len; i++) { - var p1 = children[i]; - for (j = 0; j < len; j++) { - var p2 = children[j]; - if( i !== j && bounds[i].contains( bounds[j] ) ){ - tmparray[j]++; - } - } - } - for (i = 1; i < len; i++) { - if ( tmparray[i] % 2 === 0 ) { - children[i].clockwise = baseWinding; - } - } - return baseWinding; -} - -function reversePath( path ){ - var baseWinding; - if( path instanceof CompoundPath ){ - var children = path.children, i, len; - for (i = 0, len = children.length; i < len; i++) { - children[i].reverse(); - children[i]._curves = null; - } - baseWinding = children[0].clockwise; - } else { - path.reverse(); - baseWinding = path.clockwise; - path._curves = null; - } - return baseWinding; -} - -function computeBoolean( path1, path2, operator, _splitCache ){ - var _path1, _path2, path1Clockwise, path2Clockwise; - var ixs, path1Id, path2Id; - // 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. - _path1 = path1.clone(); - _path2 = path2.clone(); - _path1.style = _path2.style = null; - _path1.selected = _path2.selected = false; - path1Clockwise = reorientCompoundPath( _path1 ); - path2Clockwise = reorientCompoundPath( _path2 ); - path1Id = _path1.id; - path2Id = _path2.id; - // Calculate all the intersections - ixs = ( _splitCache && _splitCache.intersections )? - _splitCache.intersections : _path1.getIntersections( _path2 ); - // if we have a empty _splitCache object as an operand, - // skip calculating boolean and cache the intersections - if( _splitCache && !_splitCache.intersections ){ - _splitCache.intersections = ixs; - return; - } - splitPath( ixs ); - splitPath( ixs, true ); - path1Id = _path1.id; - path2Id = _path2.id; - // Do operator specific calculations before we begin - if( operator.name === "subtraction" ) { - path2Clockwise = reversePath( _path2 ); - } - - var i, j, len, path, crv; - var paths = []; - if( _path1 instanceof CompoundPath ){ - paths = paths.concat( _path1.children ); - } else { - paths = [ _path1 ]; - } - if( _path2 instanceof CompoundPath ){ - paths = paths.concat( _path2.children ); - } else { - paths.push( _path2 ); - } - // step 1: discard invalid links according to the boolean operator - var lastNode, firstNode, nextNode, midPoint, insidePath1, insidePath2; - var thisId, thisWinding, contains, subtractionOp = (operator.name === 'subtraction'); - for (i = 0, len = paths.length; i < len; i++) { - insidePath1 = insidePath2 = false; - path = paths[i]; - thisId = ( path.parent instanceof CompoundPath )? path.parent.id : path.id; - thisWinding = path.clockwise; - lastNode = path.lastSegment; - firstNode = path.firstSegment; - nextNode = null; - while( nextNode !== firstNode){ - nextNode = ( nextNode )? nextNode.previous: lastNode; - crv = nextNode.curve; - midPoint = crv.getPoint( 0.5 ); - if( thisId !== path1Id ){ - contains = _path1.contains( midPoint ); - insidePath1 = (thisWinding === path1Clockwise || subtractionOp )? contains : - contains && !testOnCurve( _path1, midPoint ); - } - if( thisId !== path2Id ){ - contains = _path2.contains( midPoint ); - insidePath2 = (thisWinding === path2Clockwise )? contains : - contains && !testOnCurve( _path2, midPoint ); - } - if( !operator( thisId === path1Id, insidePath1, insidePath2 ) ){ - crv._INVALID = true; - // markPoint( midPoint, '+' ); - } - } - } - - // Final step: Retrieve the resulting paths from the graph - var boolResult = new CompoundPath(); - var node, nuNode, nuPath, nodeList = [], handle; - for (i = 0, len = paths.length; i < len; i++) { - nodeList = nodeList.concat( paths[i].segments ); - } - for (i = 0, len = nodeList.length; i < len; i++) { - node = nodeList[i]; - if( node.curve._INVALID || node._visited ){ continue; } - path = node.path; - thisId = ( path.parent instanceof CompoundPath )? path.parent.id : path.id; - thisWinding = path.clockwise; - nuPath = new Path(); - firstNode = null; - firstNode_ix = null; - if( node.previous.curve._INVALID ) { - node.handleIn = ( node._ixPair )? - node._ixPair.getIntersection()._segment.handleIn : [ 0, 0 ]; - } - while( node && !node._visited && ( node !== firstNode && node !== firstNode_ix ) ){ - node._visited = true; - firstNode = ( firstNode )? firstNode: node; - firstNode_ix = ( !firstNode_ix && firstNode._ixPair )? - firstNode._ixPair.getIntersection()._segment: firstNode_ix; - // node._ixPair is this node's intersection CurveLocation object - // node._ixPair.getIntersection() is the other CurveLocation object this node intersects with - nextNode = ( node._ixPair && node.curve._INVALID )? node._ixPair.getIntersection()._segment : node; - if( node._ixPair ) { - nextNode._visited = true; - nuNode = new Segment( node.point, node.handleIn, nextNode.handleOut ); - nuPath.add( nuNode ); - node = nextNode; - path = node.path; - thisWinding = path.clockwise; - } else { - nuPath.add( node ); - } - node = node.next; - } - if( nuPath.segments.length > 1 ) { - // avoid stray segments and incomplete paths - if( nuPath.segments.length > 2 || !nuPath.curves[0].isLinear() ){ - nuPath.closed = true; - boolResult.addChild( nuPath, true ); - } - } - } - // Delete the proxies - _path1.remove(); - _path2.remove(); - // And then, we are done. - return boolResult.reduce(); -} - -// Bottleneck no: 2 -function testOnCurve( path, point ){ - var res = 0; - var crv = path.getCurves(); - var i = 0; - var bounds = path.bounds; - if( bounds && bounds.contains( point ) ){ - for( i = 0; i < crv.length && !res; i++ ){ - var crvi = crv[i]; - if( crvi.bounds.contains( point ) && crvi.getParameterOf( point ) ){ - res = 1; - } - } - } - return res; -} - -/** - * A boolean operator is a binary operator function of the form - * f( isPath1:boolean, isInsidePath1:Boolean, isInsidePath2:Boolean ) :Boolean - * - * Boolean operator determines whether a curve segment in the operands is part - * of the boolean result, and will be called for each curve segment in the graph after - * all the intersections between the operands are calculated and curves in the operands - * are split at intersections. - * - * These functions should have a name ( "union", "subtraction" etc. below ), if we need to - * do operator specific operations on paths inside the computeBoolean function. - * for example: if the name of the operator is "subtraction" then we need to reverse the second - * operand. Subtraction is neither associative nor commutative. - * - * The boolean operator should return a Boolean value indicating whether to keep the curve or not. - * return true - keep the curve - * return false - discard the curve - */ - - function unite( path1, path2, _cache ){ - var unionOp = function union( isPath1, isInsidePath1, isInsidePath2 ){ - return ( isInsidePath1 || isInsidePath2 )? false : true; - }; - return computeBoolean( path1, path2, unionOp, _cache ); -} - -function intersect( path1, path2, _cache ){ - var intersectionOp = function intersection( isPath1, isInsidePath1, isInsidePath2 ){ - return ( !isInsidePath1 && !isInsidePath2 )? false : true; - }; - return computeBoolean( path1, path2, intersectionOp, _cache ); -} - -function subtract( path1, path2, _cache ){ - var subtractionOp = function subtraction( isPath1, isInsidePath1, isInsidePath2 ){ - return ( (isPath1 && isInsidePath2) || (!isPath1 && !isInsidePath1) )? false : true; - }; - return computeBoolean( path1, path2, subtractionOp, _cache ); -} - -// a.k.a. eXclusiveOR -function exclude( path1, path2 ){ - var res1 = subtract( path1, path2 ); - var res2 = subtract( path2, path1 ); - var res = new Group( [res1, res2] ); - return res; -} - -function divide( path1, path2 ){ - var res1 = subtract( path1, path2 ); - var res2 = intersect( path1, path2 ); - var res = new Group( [res1, res2] ); - return res; -} diff --git a/boolean/COPYING b/boolean/COPYING deleted file mode 100644 index dccb91dc..00000000 --- a/boolean/COPYING +++ /dev/null @@ -1,9 +0,0 @@ -The MIT License (MIT) - -Copyright (c) 2013 Harikrishnan Gopalakrishnan - -Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: - -The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. - -THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. diff --git a/boolean/README.md b/boolean/README.md deleted file mode 100644 index 21267208..00000000 --- a/boolean/README.md +++ /dev/null @@ -1,31 +0,0 @@ - -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 and CompoundPath objects -- Boolean Union -- Boolean Intersection -- Boolean Subtraction -- Resolving a self-intersecting Path - -Not supported yet ( which I would like to see supported ) -- Boolean operations between self-intersecting Paths -- Paths are clones of each other that ovelap exactly on top of each other! - -This is meant to be integrated into the paperjs library in the near future. - ------- -Harikrishnan Gopalakrishnan -http://hkrish.com/playground/paperjs/booleanStudy.html - ------- -Paperjs -Copyright (c) 2011, Juerg Lehni & Jonathan Puckey -http://paperjs.org/license/ diff --git a/boolean/booleanStudy.html b/boolean/booleanStudy.html deleted file mode 100644 index c6f93c28..00000000 --- a/boolean/booleanStudy.html +++ /dev/null @@ -1,457 +0,0 @@ - - - - - Boolean Study - - - - - - -
-

paperjs - Boolean Tests

- -
-
-

Vector boolean operations on paperjs objects.
- Still under development, mostly written for clarity and compatibility, - not optimised for performance, and has to be tested heavily.

-

--
- hari

-
- - - - - - - - - - - - - - - - - - - - - diff --git a/boolean/booleanTests.js b/boolean/booleanTests.js deleted file mode 100644 index 056c0c97..00000000 --- a/boolean/booleanTests.js +++ /dev/null @@ -1,429 +0,0 @@ - -paper.install(window); - - - -function runTests() { - var caption, pathA, pathB, group; - - var container = document.getElementById( 'container' ); - - caption = prepareTest( 'Overlapping circles', container ); - pathA = new Path.Circle(new Point(80, 110), 50); - pathB = new Path.Circle(new Point(150, 110), 70); - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'Disjoint circles', container ); - pathA = new Path.Circle(new Point(60, 110), 50); - pathB = new Path.Circle(new Point(170, 110), 50); - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'Overlapping circles - enveloping', container ); - pathA = new Path.Circle(new Point(110, 110), 100); - pathB = new Path.Circle(new Point(120, 110), 60); - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'Polygon and square', container ); - pathA = new Path.RegularPolygon(new Point(80, 110), 12, 80); - pathB = new Path.Rectangle(new Point(100, 80), [80, 80] ); - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'Circle and square (overlaps exactly on existing segments)', container ); - pathA = new Path.Circle(new Point(110, 110), 80); - pathB = new Path.Rectangle(new Point(110, 110), [80, 80] ); - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'Circle and square (existing segments overlaps on curves)', container ); - pathA = new Path.Circle(new Point(110, 110), 80); - pathB = new Path.Rectangle(new Point(110, 110), [100, 100] ); - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'Square and square (one segment overlaps on a line)', container ); - pathA = new Path.Rectangle(new Point(80, 125), [50, 50] ); - pathA.rotate( 45 ); - pathB = new Path.Rectangle(new Point(pathA.segments[2].point.x, 110), [80, 80] ); - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'Rectangle and rectangle (overlaps exactly on existing curves)', container ); - pathA = new Path.Rectangle(new Point(30.5, 50.5), [100, 150]); - pathB = new Path.Rectangle(new Point(130.5, 60.5), [100, 150]); - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'Circle and banana (multiple intersections within same curve segment)', container ); - pathA = new Path.Circle(new Point(80, 110), 80); - pathB = new Path.Circle(new Point(130, 110), 80 ); - pathB.segments[3].point = pathB.segments[3].point.add( [ 0, -120 ] ); - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'Overlapping stars 1', container ); - pathA = new Path.Star(new Point(80, 110), 10, 20, 80); - pathB = new Path.Star(new Point(120, 110), 10, 30, 100); - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'Overlapping stars 2', container ); - pathA = new Path.Star(new Point(110, 110), 20, 20, 80); - pathB = new Path.Star(new Point(110, 110), 6, 30, 100); - testBooleanStatic( pathA, pathB, caption ); - - // caption = prepareTest( 'Circles overlap exactly over each other', container ); - // pathA = new Path.Circle(new Point(110, 110), 100); - // pathB = new Path.Circle(new Point(110, 110), 100 ); - // // pathB.translate([0.5,0]) - // testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'Maximum possible intersections between 2 cubic bezier curve segments - 9', container ); - pathA = new Path(); - pathA.add( new Segment( [173, 44], [-281, 268], [-86, 152] ) ); - pathA.add( new Segment( [47, 93], [-89, 100], [240, -239] ) ); - pathA.closed = true; - pathB = pathA.clone(); - pathB.rotate( -90 ); - pathA.translate( [-10,0] ); - pathB.translate( [10,0] ); - testBooleanStatic( pathA, pathB, caption ); - annotatePath( pathA, null, '#008' ); - annotatePath( pathB, null, '#800' ); - view.draw(); - - caption = prepareTest( 'SVG gears', container ); - group = paper.project.importSVG( document.getElementById( 'svggears' ) ); - pathA = group.children[0]; - pathB = group.children[1]; - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'Glyphs imported from SVG', container ); - group = paper.project.importSVG( document.getElementById( 'glyphsys' ) ); - pathA = group.children[0]; - pathB = group.children[1]; - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'CompoundPaths 1', container ); - group = paper.project.importSVG( document.getElementById( 'glyphsacirc' ) ); - pathA = group.children[0]; - pathB = group.children[1]; - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'CompoundPaths 2 - holes', container ); - group = paper.project.importSVG( document.getElementById( 'glyphsacirc' ) ); - pathA = group.children[0]; - pathB = new CompoundPath(); - group.children[1].clockwise = true; - pathB.addChild(group.children[1]); - var npath = new Path.Circle([110, 110], 30); - pathB.addChild( npath ); - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'CompoundPaths 3 !', container ); - group = paper.project.importSVG( document.getElementById( 'svggreenland' ) ); - pathA = group.children[0]; - pathB = group.children[1]; - pathB.scale( 0.5, 1 ).translate( [25.5, 0] ); - // pathA.scale( 2 ); - // pathB.scale( 2 ); - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'CompoundPaths 4 - holes and islands 1', container ); - group = paper.project.importSVG( document.getElementById( 'glyphsacirc' ) ); - pathA = group.children[0]; - pathB = new CompoundPath(); - group.children[1].clockwise = true; - pathB.addChild(group.children[1]); - var npath = new Path.Circle([40, 80], 20); - pathB.addChild( npath ); - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'CompoundPaths 5 - holes and islands 2', container ); - group = paper.project.importSVG( document.getElementById( 'glyphsacirc' ) ); - pathA = group.children[0]; - pathB = new CompoundPath(); - group.children[1].clockwise = true; - pathB.addChild(group.children[1]); - var npath = new Path.Circle([40, 80], 20); - pathB.addChild( npath ); - npath = new Path.Circle([120, 110], 30); - pathB.addChild( npath ); - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'CompoundPaths 6 - holes and islands 3', container ); - group = paper.project.importSVG( document.getElementById( 'glyphsacirc' ) ); - pathA = group.children[0]; - pathB = new CompoundPath(); - var npath = new Path.Circle([110, 110], 100); - pathB.addChild( npath ); - npath = new Path.Circle([110, 110], 60); - pathB.addChild( npath ); - npath = new Path.Circle([110, 110], 30); - pathB.addChild( npath ); - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'CompoundPaths 6 - holes and islands 4 (curves overlap exactly on existing curves)', container ); - pathA = new Path.Rectangle(new Point(50.5, 50.5), [100, 120]); - pathB = new CompoundPath(); - pathB.addChild( new Path.Rectangle(new Point(140.5, 30.5), [100, 150]) ); - pathB.addChild( new Path.Rectangle(new Point(150.5, 65.5), [50, 100]) ); - // pathB = new Path.Rectangle(new Point(150.5, 80.5), [80, 80] ); - testBooleanStatic( pathA, pathB, caption ); - - - // // To resolve self intersection on a single path, - // // pass an empty second operand and do a Union operation - // caption = prepareTest( 'Self-intersecting paths 1 - Resolve self-intersection on single path', container ); - // pathA = new Path.Star(new Point(110, 110), 10, 20, 80); - // pathA.smooth(); - // pathB = new Path(); - // testBooleanStatic( pathA, pathB, caption, false, true, true, true ); - - // caption = prepareTest( 'Self-intersecting paths 2 - Resolve self-intersecting CompoundPath', container ); - // pathA = new CompoundPath(); - // pathA.addChild( new Path.Circle([100, 110], 60) ); - // pathA.addChild( new Path.Circle([160, 110], 30) ); - // pathB = new Path(); - // testBooleanStatic( pathA, pathB, caption, false, true, true, true ); - - - // var tool = new Tool(); - // tool.onMouseMove = function( e ){ - // var hi = project.hitTest( e.point ); - // if( hi ){ - // var item = hi.item; - // if( item instanceof PathItem ){ - // var txt = new PointText( e.point.add([0, -10]) ); - // txt.justification = 'center'; - // txt.content = item.id; - // txt.fillColor = '#000'; - // txt.removeOnMove(); - // } - // } - // }; - - window.a = pathA; - window.b = pathB; - - // // pathA.selected = true; - // // pathA.fullySelected = true; - // // pathB.selected = true; - // pathA.style = pathStyleBoolean; - // pathB.style = pathStyleBoolean; - - // // reorientCompoundPath( pathB ) - - // // var ixs = pathA.getIntersections( pathB ); - // // ixs.map( function(a){ console.log( "( " + a.path.id + " , " + a.curve.index + " , "+ a.parameter +" )" ); - // // markPoint( a.point, " " ) } ); - // // ixs.map( function(a){ markPoint( a.point, " " ); } ); - - // // splitPath( ixs, true ); - // // splitPath( ixs ); - - // // annotatePath( pathB ) - // // pathB.translate( [ 300, 0 ] ); - // // pathB.segments.filter( function(a){ return a._ixPair; } ).map( - // // function(a){ a._ixPair.getIntersection()._segment.selected = true; }); - - // console.time('unite'); - // var nup = unite( pathA, pathB ); - // console.timeEnd('unite'); - // // nup.style = booleanStyle; - // window.p = nup; - - // view.draw(); - - - function prepareTest( testName, parentNode ){ - console.log( '\n' + testName ); - var caption = document.createElement('h3'); - caption.appendChild( document.createTextNode( testName ) ); - var canvas = document.createElement('CANVAS'); - parentNode.appendChild( caption ); - parentNode.appendChild( canvas ); - paper.setup( canvas ); - return caption; - } -} - -var booleanStyle = { - fillColor: new Color( 1, 0, 0, 0.5 ), - strokeColor: new Color( 0, 0, 0 ), - strokeWidth: 1.5 -}; - -var pathStyleNormal = { - strokeColor: new Color( 0, 0, 0 ), - fillColor: new Color( 0, 0, 0, 0.1 ), - strokeWidth: 1 -}; - -var pathStyleBoolean = { - strokeColor: new Color( 0.8 ), - fillColor: new Color( 0, 0, 0, 0.0 ), - strokeWidth: 1 -}; - -// Better if path1 and path2 fit nicely inside a 200x200 pixels rect -function testBooleanStatic( path1, path2, caption, noUnion, noIntersection, noSubtraction, _disperse ) { - // try{ - path1.style = path2.style = pathStyleNormal; - - if( !noUnion ) { - var _p1U = path1.clone().translate( [250, 0] ); - var _p2U = path2.clone().translate( [250, 0] ); - _p1U.style = _p2U.style = pathStyleBoolean; - console.time( 'Union' ); - var boolPathU = unite( _p1U, _p2U ); - console.timeEnd( 'Union' ); - boolPathU.style = booleanStyle; - if( _disperse ){ disperse( boolPathU ); } - } - - if( !noIntersection ) { - var _p1I = path1.clone().translate( [500, 0] ); - var _p2I = path2.clone().translate( [500, 0] ); - _p1I.style = _p2I.style = pathStyleBoolean; - console.time( 'Intersection' ); - var boolPathI = intersect( _p1I, _p2I ); - console.timeEnd( 'Intersection' ); - boolPathI.style = booleanStyle; - } - - if( !noSubtraction ) { - var _p1S = path1.clone().translate( [750, 0] ); - var _p2S = path2.clone().translate( [750, 0] ); - _p1S.style = _p2S.style = pathStyleBoolean; - console.time( 'Subtraction' ); - var boolPathS = subtract( _p1S, _p2S ); - console.timeEnd( 'Subtraction' ); - boolPathS.style = booleanStyle; - } - - if( !noSubtraction ) { - var _p1E = path1.clone().translate( [250, 220] ); - var _p2E = path2.clone().translate( [250, 220] ); - _p1E.style = _p2E.style = pathStyleBoolean; - console.time( 'Exclusion' ); - var boolPathE = exclude( _p1E, _p2E ); - console.timeEnd( 'Exclusion' ); - boolPathE.style = booleanStyle; - } - - if( !noSubtraction && !noIntersection) { - var _p1D = path1.clone().translate( [500, 220] ); - var _p2D = path2.clone().translate( [500, 220] ); - _p1D.style = _p2D.style = pathStyleBoolean; - console.time( 'Division' ); - var boolPathD = divide( _p1D, _p2D ); - console.timeEnd( 'Division' ); - disperse( boolPathD ); - boolPathD.style = booleanStyle; - } - // } catch( e ){ - // console.error( e.name + ": " + e.message ); - // if( caption ) { caption.className += ' error'; } - // // paper.project.view.element.className += ' hide'; - // } finally { - console.timeEnd( 'Union' ); - console.timeEnd( 'Intersection' ); - console.timeEnd( 'Subtraction' ); - view.draw(); - // } -} - -function disperse( path, distance ){ - distance = distance || 10; - if( ! path instanceof CompoundPath || ! path instanceof Group ){ return; } - var center = path.bounds.center; - var children = path.children, i ,len; - for (i = 0, len = children.length; i < len; i++) { - var cCenter = children[i].bounds.center; - var vec = cCenter.subtract( center ); - vec = ( vec.isClose( [0,0], 0.5 ) )? vec : vec.normalize( distance ); - children[i].translate( vec ); - } -} - -// ============================================================== -// On screen debug helpers -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(); - } -} - -function annotatePath( path, t, c, tc, remove ) { - if( !path ) return; - var crvs = path.curves; - for (i = crvs.length - 1; i >= 0; i--) { - annotateCurve( crvs[i], t, c, tc, remove ); - } - var segs = path.segments; - for (i = segs.length - 1; i >= 0; i--) { - annotateSegment( segs[i], t, c, tc, remove, true ); - } -} - -function annotateSegment( s, t, c, tc, remove, skipCurves ) { - if( !s ) return; - c = c || '#000'; - tc = tc || '#ccc'; - t = t || s.index; - if( remove === undefined ){ remove = true; } - var crv = s.curve; - var t1 = crv.getNormal( 0 ).normalize( 10 ); - var p = s.point.clone().add( t1 ); - var cir = new Path.Circle( s.point, 2 ); - cir.style.fillColor = c; - cir.style.strokeColor = tc; - var text = new PointText( p ); - text.justification = 'center'; - text.fillColor = c; - text.content = t; - if( remove ) { - cir.removeOnMove(); - text.removeOnMove(); - } - if( !skipCurves ) { - annotateCurve( s.curveIn, null, c, tc, remove ); - annotateCurve( s.curveOut, null, c, tc, remove ); - } -} - -function annotateCurve( crv, t, c, tc, remove ) { - if( !crv ) return; - c = c || '#000'; - tc = tc || '#ccc'; - t = t || crv.index; - if( remove === undefined ){ remove = true; } - var p = crv.getPoint( 0.57 ); - var t1 = crv.getTangent( 0.57 ).normalize( -10 ); - var p2 = p.clone().add( t1 ); - var l = new Path.Line( p, p2 ).rotate( 30, p ); - var l2 = new Path.Line( p, p2 ).rotate( -30, p ); - p = crv.getPoint( 0.43 ); - var cir = new Path.Circle( p, 8 ); - var text = new PointText( p.subtract( [0, -4] ) ); - text.justification = 'center'; - text.fillColor = tc; - text.content = t; - l.style.strokeColor = l2.style.strokeColor = c; - cir.style.fillColor = c; - if( remove ) { - l.removeOnMove(); - l2.removeOnMove(); - cir.removeOnMove(); - text.removeOnMove(); - } -} diff --git a/boolean/booleanTests_paper.js b/boolean/booleanTests_paper.js deleted file mode 100644 index 25715122..00000000 --- a/boolean/booleanTests_paper.js +++ /dev/null @@ -1,366 +0,0 @@ - -paper.install(window); - - - -function runTests() { - var caption, pathA, pathB, group; - - var container = document.getElementById( 'container' ); - - caption = prepareTest( 'Overlapping circles', container ); - pathA = new Path.Circle(new Point(80, 110), 50); - pathB = new Path.Circle(new Point(150, 110), 70); - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'Disjoint circles', container ); - pathA = new Path.Circle(new Point(60, 110), 50); - pathB = new Path.Circle(new Point(170, 110), 50); - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'Overlapping circles - enveloping', container ); - pathA = new Path.Circle(new Point(110, 110), 100); - pathB = new Path.Circle(new Point(120, 110), 60); - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'Polygon and square', container ); - pathA = new Path.RegularPolygon(new Point(80, 110), 12, 80); - pathB = new Path.Rectangle(new Point(100, 80), [80, 80] ); - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'Circle and square (overlaps exactly on existing segments)', container ); - pathA = new Path.Circle(new Point(110, 110), 80); - pathB = new Path.Rectangle(new Point(110, 110), [80, 80] ); - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'Circle and square (existing segments overlaps on curves)', container ); - pathA = new Path.Circle(new Point(110, 110), 80); - pathB = new Path.Rectangle(new Point(110, 110), [100, 100] ); - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'Square and square (one segment overlaps on a line)', container ); - pathA = new Path.Rectangle(new Point(80, 125), [50, 50] ); - pathA.rotate( 45 ); - pathB = new Path.Rectangle(new Point(pathA.segments[2].point.x, 110), [80, 80] ); - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'Rectangle and rectangle (overlaps exactly on existing curves)', container ); - pathA = new Path.Rectangle(new Point(30.5, 50.5), [100, 150]); - pathB = new Path.Rectangle(new Point(130.5, 60.5), [100, 150]); - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'Circle and banana (multiple intersections within same curve segment)', container ); - pathA = new Path.Circle(new Point(80, 110), 80); - pathB = new Path.Circle(new Point(130, 110), 80 ); - pathB.segments[3].point = pathB.segments[3].point.add( [ 0, -120 ] ); - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'Overlapping stars 1', container ); - pathA = new Path.Star(new Point(80, 110), 10, 20, 80); - pathB = new Path.Star(new Point(120, 110), 10, 30, 100); - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'Overlapping stars 2', container ); - pathA = new Path.Star(new Point(110, 110), 20, 20, 80); - pathB = new Path.Star(new Point(110, 110), 6, 30, 100); - testBooleanStatic( pathA, pathB, caption ); - - // caption = prepareTest( 'Circles overlap exactly over each other', container ); - // pathA = new Path.Circle(new Point(110, 110), 100); - // pathB = new Path.Circle(new Point(110, 110), 100 ); - // // pathB.translate([0.5,0]) - // testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'Maximum possible intersections between 2 cubic bezier curve segments - 9', container ); - pathA = new Path(); - pathA.add( new Segment( [173, 44], [-281, 268], [-86, 152] ) ); - pathA.add( new Segment( [47, 93], [-89, 100], [240, -239] ) ); - pathA.closed = true; - pathB = pathA.clone(); - pathB.rotate( -90 ); - pathA.translate( [-10,0] ); - pathB.translate( [10,0] ); - testBooleanStatic( pathA, pathB, caption ); - annotatePath( pathA, null, '#008' ); - annotatePath( pathB, null, '#800' ); - view.draw(); - - caption = prepareTest( 'SVG gears', container ); - group = paper.project.importSVG( document.getElementById( 'svggears' ) ); - pathA = group.children[0]; - pathB = group.children[1]; - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'Glyphs imported from SVG', container ); - group = paper.project.importSVG( document.getElementById( 'glyphsys' ) ); - pathA = group.children[0]; - pathB = group.children[1]; - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'CompoundPaths 1', container ); - group = paper.project.importSVG( document.getElementById( 'glyphsacirc' ) ); - pathA = group.children[0]; - pathB = group.children[1]; - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'CompoundPaths 2 - holes', container ); - group = paper.project.importSVG( document.getElementById( 'glyphsacirc' ) ); - pathA = group.children[0]; - pathB = new CompoundPath(); - group.children[1].clockwise = true; - pathB.addChild(group.children[1]); - var npath = new Path.Circle([110, 110], 30); - pathB.addChild( npath ); - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'CompoundPaths 3 !', container ); - group = paper.project.importSVG( document.getElementById( 'svggreenland' ) ); - pathA = group.children[0]; - pathB = group.children[1]; - pathB.scale( 0.5, 1 ).translate( [25.5, 0] ); - // pathA.scale( 2 ); - // pathB.scale( 2 ); - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'CompoundPaths 4 - holes and islands 1', container ); - group = paper.project.importSVG( document.getElementById( 'glyphsacirc' ) ); - pathA = group.children[0]; - pathB = new CompoundPath(); - group.children[1].clockwise = true; - pathB.addChild(group.children[1]); - var npath = new Path.Circle([40, 80], 20); - pathB.addChild( npath ); - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'CompoundPaths 5 - holes and islands 2', container ); - group = paper.project.importSVG( document.getElementById( 'glyphsacirc' ) ); - pathA = group.children[0]; - pathB = new CompoundPath(); - group.children[1].clockwise = true; - pathB.addChild(group.children[1]); - var npath = new Path.Circle([40, 80], 20); - pathB.addChild( npath ); - npath = new Path.Circle([120, 110], 30); - pathB.addChild( npath ); - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'CompoundPaths 6 - holes and islands 3', container ); - group = paper.project.importSVG( document.getElementById( 'glyphsacirc' ) ); - pathA = group.children[0]; - pathB = new CompoundPath(); - var npath = new Path.Circle([110, 110], 100); - pathB.addChild( npath ); - npath = new Path.Circle([110, 110], 60); - pathB.addChild( npath ); - npath = new Path.Circle([110, 110], 30); - pathB.addChild( npath ); - testBooleanStatic( pathA, pathB, caption ); - - caption = prepareTest( 'CompoundPaths 6 - holes and islands 4 (curves overlap exactly on existing curves)', container ); - pathA = new Path.Rectangle(new Point(50.5, 50.5), [100, 120]); - pathB = new CompoundPath(); - pathB.addChild( new Path.Rectangle(new Point(140.5, 30.5), [100, 150]) ); - pathB.addChild( new Path.Rectangle(new Point(150.5, 60.5), [50, 100]) ); - // pathB = new Path.Rectangle(new Point(150.5, 80.5), [80, 80] ); - testBooleanStatic( pathA, pathB, caption ); - - - // To resolve self intersection on a single path, - // pass an empty second operand and do a Union operation - caption = prepareTest( 'Self-intersecting paths 1 - Resolve self-intersection on single path', container ); - pathA = new Path.Star(new Point(110, 110), 10, 20, 80); - pathA.smooth(); - pathB = new Path(); - testBooleanStatic( pathA, pathB, caption, false, true, true, true ); - - caption = prepareTest( 'Self-intersecting paths 2 - Resolve self-intersecting CompoundPath', container ); - pathA = new CompoundPath(); - pathA.addChild( new Path.Circle([100, 110], 60) ); - pathA.addChild( new Path.Circle([160, 110], 30) ); - pathB = new Path(); - testBooleanStatic( pathA, pathB, caption, false, true, true, true ); - - window.a = pathA; - window.b = pathB; - - - function prepareTest( testName, parentNode ){ - console.log( '\n' + testName ); - var caption = document.createElement('h3'); - caption.appendChild( document.createTextNode( testName ) ); - var canvas = document.createElement('CANVAS'); - parentNode.appendChild( caption ); - parentNode.appendChild( canvas ); - paper.setup( canvas ); - return caption; - } -} - -var booleanStyle = { - fillColor: new Color( 1, 0, 0, 0.5 ), - strokeColor: new Color( 0, 0, 0 ), - strokeWidth: 1.5 -}; - -var pathStyleNormal = { - strokeColor: new Color( 0, 0, 0 ), - fillColor: new Color( 0, 0, 0, 0.1 ), - strokeWidth: 1 -}; - -var pathStyleBoolean = { - strokeColor: new Color( 0.8 ), - fillColor: new Color( 0, 0, 0, 0.0 ), - strokeWidth: 1 -}; - -// Better if path1 and path2 fit nicely inside a 200x200 pixels rect -function testBooleanStatic( path1, path2, caption, noUnion, noIntersection, noSubtraction, _disperse ) { - // try{ - path1.style = path2.style = pathStyleNormal; - - if( !noUnion ) { - var _p1U = path1.clone().translate( [250, 0] ); - var _p2U = path2.clone().translate( [250, 0] ); - _p1U.style = _p2U.style = pathStyleBoolean; - console.time( 'Union' ); - var boolPathU = _p1U.unite( _p2U ); - console.timeEnd( 'Union' ); - boolPathU.style = booleanStyle; - if( _disperse ){ disperse( boolPathU ); } - window.p = boolPathU; - } - - if( !noIntersection ) { - var _p1I = path1.clone().translate( [500, 0] ); - var _p2I = path2.clone().translate( [500, 0] ); - _p1I.style = _p2I.style = pathStyleBoolean; - console.time( 'Intersection' ); - var boolPathI = _p1I.intersect( _p2I ); - console.timeEnd( 'Intersection' ); - if( _disperse ){ disperse( boolPathI ); } - boolPathI.style = booleanStyle; - } - - if( !noSubtraction ) { - var _p1S = path1.clone().translate( [750, 0] ); - var _p2S = path2.clone().translate( [750, 0] ); - _p1S.style = _p2S.style = pathStyleBoolean; - console.time( 'Subtraction' ); - var boolPathS = _p1S.subtract( _p2S ); - console.timeEnd( 'Subtraction' ); - if( _disperse ){ disperse( boolPathS ); } - boolPathS.style = booleanStyle; - } - // } catch( e ){ - // console.error( e.name + ": " + e.message ); - // if( caption ) { caption.className += ' error'; } - // // paper.project.view.element.className += ' hide'; - // } finally { - console.timeEnd( 'Union' ); - console.timeEnd( 'Intersection' ); - console.timeEnd( 'Subtraction' ); - view.draw(); - // } -} - -function disperse( path, distance ){ - distance = distance || 10; - if( ! path instanceof CompoundPath ){ return; } - var center = path.bounds.center; - var children = path.children, i ,len; - for (i = 0, len = children.length; i < len; i++) { - var cCenter = children[i].bounds.center; - var vec = cCenter.subtract( center ); - vec = ( vec.isClose( [0,0], 0.5 ) )? vec : vec.normalize( distance ); - children[i].translate( vec ); - } -} - -// ============================================================== -// On screen debug helpers -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(); - } -} - -function annotatePath( path, t, c, tc, remove ) { - if( !path ) return; - var crvs = path.curves; - for (i = crvs.length - 1; i >= 0; i--) { - annotateCurve( crvs[i], t, c, tc, remove ); - } - var segs = path.segments; - for (i = segs.length - 1; i >= 0; i--) { - annotateSegment( segs[i], t, c, tc, remove, true ); - } -} - -function annotateSegment( s, t, c, tc, remove, skipCurves ) { - if( !s ) return; - c = c || '#000'; - tc = tc || '#ccc'; - t = t || s.index; - if( remove === undefined ){ remove = true; } - var crv = s.curve; - var t1 = crv.getNormal( 0 ).normalize( 10 ); - var p = s.point.clone().add( t1 ); - var cir = new Path.Circle( s.point, 2 ); - cir.style.fillColor = c; - cir.style.strokeColor = tc; - var text = new PointText( p ); - text.justification = 'center'; - text.fillColor = c; - text.content = t; - if( remove ) { - cir.removeOnMove(); - text.removeOnMove(); - } - if( !skipCurves ) { - annotateCurve( s.curveIn, null, c, tc, remove ); - annotateCurve( s.curveOut, null, c, tc, remove ); - } -} - -function annotateCurve( crv, t, c, tc, remove ) { - if( !crv ) return; - c = c || '#000'; - tc = tc || '#ccc'; - t = t || crv.index; - if( remove === undefined ){ remove = true; } - var p = crv.getPoint( 0.57 ); - var t1 = crv.getTangent( 0.57 ).normalize( -10 ); - var p2 = p.clone().add( t1 ); - var l = new Path.Line( p, p2 ).rotate( 30, p ); - var l2 = new Path.Line( p, p2 ).rotate( -30, p ); - p = crv.getPoint( 0.43 ); - var cir = new Path.Circle( p, 8 ); - var text = new PointText( p.subtract( [0, -4] ) ); - text.justification = 'center'; - text.fillColor = tc; - text.content = t; - l.style.strokeColor = l2.style.strokeColor = c; - cir.style.fillColor = c; - if( remove ) { - l.removeOnMove(); - l2.removeOnMove(); - cir.removeOnMove(); - text.removeOnMove(); - } -} diff --git a/boolean/mpatch.js b/boolean/mpatch.js deleted file mode 100644 index 388a874a..00000000 --- a/boolean/mpatch.js +++ /dev/null @@ -1,63 +0,0 @@ - -paper.PathItem.prototype.getIntersections = function(path) { - // First check the bounds of the two paths. If they don't intersect, - // we don't need to iterate through their curves. - if (!this.getBounds().touches(path.getBounds())) - return []; - var locations = [], - curves1 = this.getCurves(), - curves2 = path.getCurves(), - length2 = curves2.length, - values2 = []; - for (var i = 0; i < length2; i++) - values2[i] = curves2[i].getValues(); - for (var i = 0, l = curves1.length; i < l; i++) { - var curve = curves1[i], - values1 = curve.getValues(); - for (var j = 0; j < length2; j++) - Curve.getIntersections(values1, values2[j], curve, curves2[j], locations); - } - return locations; -}; - - -paper.Curve.getIntersections = function(v1, v2, curve, curve2, locations) { - var bounds1 = this.getBounds(v1), - bounds2 = this.getBounds(v2); - if (bounds1.touches(bounds2)) { - // See if both curves are flat enough to be treated as lines, either - // because they have no control points at all, or are "flat enough" - if ((this.isLinear(v1) - || this.isFlatEnough(v1, /*#=*/ Numerical.TOLERANCE)) - && (this.isLinear(v2) - || this.isFlatEnough(v2, /*#=*/ Numerical.TOLERANCE))) { - // See if the parametric equations of the lines interesct. - 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)); - if (point) { - // Avoid duplicates when hitting segments (closed paths too) - var first = locations[0], - last = locations[locations.length - 1]; - if ((!first || !point.equals(first._point)) - && (!last || !point.equals(last._point))){ - // Passing null for parameter leads to lazy determination - // of parameter values in CurveLocation#getParameter() - // only once they are requested. - var cloc = new CurveLocation(curve, null, point); - var cloc2 = new CurveLocation(curve2, null, point) - cloc2._ixPair = cloc; - cloc._ixPair = cloc2; - locations.push( cloc ); - } - } - } else { - // Subdivide both curves, and see if they intersect. - var v1s = this.subdivide(v1), - v2s = this.subdivide(v2); - for (var i = 0; i < 2; i++) - for (var j = 0; j < 2; j++) - this.getIntersections(v1s[i], v2s[j], curve, curve2, locations); - } - } - return locations; - };