Compound Paths working now. Also pulled in jLehni's changes on resolving transformations on compoundpaths

This commit is contained in:
hkrish 2013-04-19 14:46:27 +02:00
parent 7b6b94fc21
commit 27eeb24c4f
3 changed files with 109 additions and 23 deletions

View file

@ -81,8 +81,9 @@
* @param {Point} _handleOut * @param {Point} _handleOut
* @param {Any} _id * @param {Any} _id
*/ */
function Node( _point, _handleIn, _handleOut, _id ){ function Node( _point, _handleIn, _handleOut, _id, _childId ){
this.id = _id; this.id = _id;
this.childId = _childId;
this.type = NORMAL_NODE; this.type = NORMAL_NODE;
this.point = _point; this.point = _point;
this.handleIn = _handleIn; // handleIn this.handleIn = _handleIn; // handleIn
@ -131,8 +132,9 @@
* @param {Node} _nodeOut * @param {Node} _nodeOut
* @param {Any} _id * @param {Any} _id
*/ */
function Link( _nodeIn, _nodeOut, _id ) { function Link( _nodeIn, _nodeOut, _id, _childId ) {
this.id = _id; this.id = _id;
this.childId = _childId;
this.nodeIn = _nodeIn; // nodeStart this.nodeIn = _nodeIn; // nodeStart
this.nodeOut = _nodeOut; // nodeEnd this.nodeOut = _nodeOut; // nodeEnd
this.nodeIn.linkOut = this; // nodeStart.linkOut this.nodeIn.linkOut = this; // nodeStart.linkOut
@ -155,14 +157,14 @@
* @param {Integer} id * @param {Integer} id
* @return {Array} Links * @return {Array} Links
*/ */
function makeGraph( path, id ){ function makeGraph( path, id, childId ){
var graph = []; var graph = [];
var segs = path.segments, prevNode = null, firstNode = null, nuLink, nuNode; var segs = path.segments, prevNode = null, firstNode = null, nuLink, nuNode;
for( i = 0, l = segs.length; i < l; i++ ){ for( i = 0, l = segs.length; i < l; i++ ){
var nuSeg = segs[i].clone(); var nuSeg = segs[i].clone();
nuNode = new Node( nuSeg.point, nuSeg.handleIn, nuSeg.handleOut, id ); nuNode = new Node( nuSeg.point, nuSeg.handleIn, nuSeg.handleOut, id, childId );
if( prevNode ) { if( prevNode ) {
nuLink = new Link( prevNode, nuNode, id ); nuLink = new Link( prevNode, nuNode, id, childId );
graph.push( nuLink ); graph.push( nuLink );
} }
prevNode = nuNode; prevNode = nuNode;
@ -171,7 +173,7 @@
} }
} }
// the path is closed // the path is closed
nuLink = new Link( prevNode, firstNode, id ); nuLink = new Link( prevNode, firstNode, id, childId );
graph.push( nuLink ); graph.push( nuLink );
return graph; return graph;
} }
@ -215,23 +217,46 @@
// The boolean operation may modify the original paths // The boolean operation may modify the original paths
var path1 = _path1.clone(); var path1 = _path1.clone();
var path2 = _path2.clone(); var path2 = _path2.clone();
if( !path1.clockwise ){ path1.reverse(); } // if( !path1.clockwise ){ path1.reverse(); }
if( !path2.clockwise ){ path2.reverse(); } // if( !path2.clockwise ){ path2.reverse(); }
//
var i, j, k, l, lnk, crv, node, nuNode, leftLink, rightLink;
// Prepare the graphs. Graphs are list of Links that retains // Prepare the graphs. Graphs are list of Links that retains
// full connectivity information. The order of links in a graph is not important // 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. // 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 // Also, this is the place to resolve self-intersecting paths
var graph = makeGraph( path1, 1 ); var graph = [], path1Children, path2Children;
var graph2 = makeGraph( path2, 2 ); if( path1 instanceof CompoundPath ){
// Merge the two graphs. Since we have unique id's for each Link and Node, path1Children = path1.children;
// retrieveing the original graphs is rather simple. for (i = 0, l = path1Children.length; i < l; i++) {
graph = graph.concat( graph2 ); path1Children[i].closed = true;
graph = graph.concat( makeGraph( path1Children[i], 1, i + 1 ) );
}
} else {
path1.closed = true;
path1.clockwise = true;
graph = graph.concat( makeGraph( path1, 1, 1 ) );
}
// TODO: if operator === BooleanOps.subtract, then for path2, clockwise must be false
if( path2 instanceof CompoundPath ){
path2Children = path2.children;
for (i = 0, l = path2Children.length; i < l; i++) {
path2Children[i].closed = true;
graph = graph.concat( makeGraph( path2Children[i], 2, i + 1 ) );
}
} else {
path2.closed = true;
path2.clockwise = true;
graph = graph.concat( makeGraph( path2, 2, 1 ) );
}
window.g = graph
// Sort function to sort intersections according to the 'parameter'(t) in a link (curve) // Sort function to sort intersections according to the 'parameter'(t) in a link (curve)
function ixSort( a, b ){ return a._parameter - b._parameter; } function ixSort( a, b ){ return a._parameter - b._parameter; }
var i, j, k, l, lnk, crv, node, nuNode, leftLink, rightLink;
/* /*
* Pass 1: * Pass 1:
* Calculate the intersections for all graphs * Calculate the intersections for all graphs
@ -264,7 +289,6 @@
} }
} }
/* /*
* Pass 2: * Pass 2:
* Walk the graph, sort the intersections on each individual link. * Walk the graph, sort the intersections on each individual link.
@ -304,7 +328,7 @@
// TODO: check if link is linear and set handles to null // TODO: check if link is linear and set handles to null
var ixPoint = new Point( left[6], left[7] ); var ixPoint = new Point( left[6], left[7] );
nuNode = new Node( ixPoint, new Point(left[4] - ixPoint.x, left[5] - ixPoint.y), nuNode = new Node( ixPoint, new Point(left[4] - ixPoint.x, left[5] - ixPoint.y),
new Point(right[2] - ixPoint.x, right[3] - ixPoint.y), lnk.id ); new Point(right[2] - ixPoint.x, right[3] - ixPoint.y), lnk.id, lnk.childId );
nuNode.type = INTERSECTION_NODE; nuNode.type = INTERSECTION_NODE;
nuNode._intersectionID = ix[j]._intersectionID; nuNode._intersectionID = ix[j]._intersectionID;
// clear the cached Segment on original end nodes and Update their handles // clear the cached Segment on original end nodes and Update their handles
@ -315,8 +339,8 @@
tmppnt = lnk.nodeOut.point; tmppnt = lnk.nodeOut.point;
lnk.nodeOut.handleIn = new Point( right[4] - tmppnt.x, right[5] - tmppnt.y ); lnk.nodeOut.handleIn = new Point( right[4] - tmppnt.x, right[5] - tmppnt.y );
// Make new links after the split // Make new links after the split
leftLink = new Link( lnk.nodeIn, nuNode, lnk.id ); leftLink = new Link( lnk.nodeIn, nuNode, lnk.id, lnk.childId);
rightLink = new Link( nuNode, lnk.nodeOut, lnk.id ); rightLink = new Link( nuNode, lnk.nodeOut, lnk.id, lnk.childId );
} }
// Add the first split link back to the graph, since we sorted the intersections // 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. // already, this link should contain no more intersections to the left.
@ -418,17 +442,24 @@
} }
// Final step: Retrieve the resulting paths from the graph // Final step: Retrieve the resulting paths from the graph
// TODO: start from a path where childId === 1
var boolResult = new CompoundPath(); var boolResult = new CompoundPath();
var firstNode = true, nextNode; var firstNode = true, nextNode, foundBasePath = false;
while( firstNode ){ while( firstNode ){
firstNode = nextNode = null; firstNode = nextNode = null;
len = graph.length; len = graph.length;
while( len-- ){ while( len-- ){
if( !graph[len].INVALID && !graph[len].nodeIn.visited && !firstNode ){ if( !graph[len].INVALID && !graph[len].nodeIn.visited && !firstNode ){
if( !foundBasePath && graph[len].childId === 1 ){
firstNode = graph[len].nodeIn;
foundBasePath = true;
break;
} else if(foundBasePath){
firstNode = graph[len].nodeIn; firstNode = graph[len].nodeIn;
break; break;
} }
} }
}
if( firstNode ){ if( firstNode ){
var path = new Path(); var path = new Path();
path.add( firstNode.getSegment( true ) ); path.add( firstNode.getSegment( true ) );
@ -450,6 +481,27 @@
} }
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();
}
}
// Same as the paperjs' Numerical class, // Same as the paperjs' Numerical class,
// added here because I can't access the original from this scope // added here because I can't access the original from this scope
var Numerical = { var Numerical = {

View file

@ -40,5 +40,17 @@
c4.17-10.207,9.736-20.396,15.764-32.207l22.473-43.104h22.707l-52.131,89.669v66.51H82.734z"/> c4.17-10.207,9.736-20.396,15.764-32.207l22.473-43.104h22.707l-52.131,89.669v66.51H82.734z"/>
</svg> </svg>
<svg version="1.1" id="glyphsacirc" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px"
width="200px" height="200px" viewBox="0 0 200 200" enable-background="new 0 0 200 200" xml:space="preserve">
<path fill="none" d="M107.585,161.74c-3.868,0-9.281-2.318-12.117-5.156c-3.351-3.35-4.898-6.961-6.186-11.342
c-10.312,6.961-22.688,16.498-30.421,16.498c-18.304,0-30.935-15.211-30.935-31.451c0-12.631,6.703-20.621,20.881-25.52
c15.468-5.157,34.287-12.118,39.958-16.757v-4.383c0-18.818-8.766-29.13-21.913-29.13c-4.898,0-8.765,1.805-11.601,4.898
c-3.093,3.609-5.413,9.281-7.476,17.015c-1.288,4.124-3.868,5.929-7.477,5.929c-4.64,0-11.084-4.898-11.084-11.085
c0-3.609,3.094-6.703,7.991-10.312c6.961-5.156,23.976-14.952,38.67-18.045c7.733,0,15.466,2.321,21.139,6.961
c9.024,7.733,11.601,18.045,11.601,31.708v47.693c0,11.6,4.64,15.467,9.024,15.467c3.094,0,6.7-1.289,9.279-2.836l2.577,6.703
L107.585,161.74z M88.766,96.778c-5.415,2.835-17.788,8.248-23.202,10.828c-10.054,4.639-15.725,9.537-15.725,19.076
c0,13.146,10.31,19.592,18.303,19.592c6.701,0,15.725-4.383,20.623-9.279V96.778z"/>
<circle fill="none" cx="112.681" cy="105.821" r="52.076"/>
</svg>
</body> </body>
</html> </html>

View file

@ -75,6 +75,26 @@ function runTests() {
pathB = group.children[1]; pathB = group.children[1];
testBooleanStatic( pathA, pathB, caption ); testBooleanStatic( pathA, pathB, caption );
caption = prepareTest( 'CompoundPaths 1', container );
var group = paper.project.importSvg( document.getElementById( 'glyphsacirc' ) );
pathA = group.children[0];
pathB = group.children[1];
testBooleanStatic( pathA, pathB, caption );
caption = prepareTest( 'CompoundPaths 2', container );
var group = paper.project.importSvg( document.getElementById( 'glyphsacirc' ) );
pathA = group.children[0];
pathB = new CompoundPath();
pathB.addChild(group.children[1]);
var npath = new Path.Circle([110, 110], 30);
console.log(npath.clockwise)
pathB.addChild( npath );
console.log(npath.clockwise)
testBooleanStatic( pathA, pathB, caption );
window.p = pathB;
function prepareTest( testName, parentNode ){ function prepareTest( testName, parentNode ){
console.log( '\n' + testName ); console.log( '\n' + testName );
@ -115,6 +135,8 @@ function testBooleanStatic( path1, path2, caption ) {
var boolPathU = boolUnion( _p1U, _p2U ); var boolPathU = boolUnion( _p1U, _p2U );
console.timeEnd( 'Union' ); console.timeEnd( 'Union' );
window.b = boolPathU
var _p1I = path1.clone().translate( [560, 0] ); var _p1I = path1.clone().translate( [560, 0] );
var _p2I = path2.clone().translate( [560, 0] ); var _p2I = path2.clone().translate( [560, 0] );
console.time( 'Intersection' ); console.time( 'Intersection' );