mirror of
https://github.com/scratchfoundation/paper.js.git
synced 2025-01-19 14:10:14 -05:00
Run BooleanOperations.html code asynchronously for immediate feedback.
This commit is contained in:
parent
6da97b894f
commit
eb8ebca097
1 changed files with 235 additions and 230 deletions
|
@ -7,177 +7,216 @@
|
||||||
<script type="text/javascript">
|
<script type="text/javascript">
|
||||||
paper.install(window);
|
paper.install(window);
|
||||||
|
|
||||||
|
var operations = {
|
||||||
|
union: true,
|
||||||
|
intersection: true,
|
||||||
|
subtraction: true,
|
||||||
|
exclusion: true,
|
||||||
|
division: true
|
||||||
|
};
|
||||||
|
|
||||||
function runTests() {
|
function runTests() {
|
||||||
var caption, pathA, pathB, group;
|
var pathA, pathB, group;
|
||||||
|
|
||||||
var container = document.getElementById('container');
|
var container = document.getElementById('container');
|
||||||
|
|
||||||
caption = prepareTest('Overlapping circles', container);
|
function runTest(testName, handler) {
|
||||||
pathA = new Path.Circle(new Point(80, 110), 50);
|
console.log('\n' + testName);
|
||||||
pathB = new Path.Circle(new Point(150, 110), 70);
|
var caption = document.createElement('h3');
|
||||||
testBooleanStatic(pathA, pathB, caption);
|
var canvas = document.createElement('canvas');
|
||||||
|
caption.appendChild(document.createTextNode(testName));
|
||||||
|
container.appendChild(caption);
|
||||||
|
container.appendChild(canvas);
|
||||||
|
setTimeout(function() {
|
||||||
|
paper.setup(canvas);
|
||||||
|
var paths = handler();
|
||||||
|
testBooleanStatic(paths[0], paths[1]);
|
||||||
|
}, 0);
|
||||||
|
return caption;
|
||||||
|
}
|
||||||
|
|
||||||
caption = prepareTest('Disjoint circles', container);
|
runTest('Overlapping circles', function() {
|
||||||
pathA = new Path.Circle(new Point(60, 110), 50);
|
return [
|
||||||
pathB = new Path.Circle(new Point(170, 110), 50);
|
new Path.Circle(new Point(80, 110), 50),
|
||||||
testBooleanStatic(pathA, pathB, caption);
|
new Path.Circle(new Point(150, 110), 70)
|
||||||
|
];
|
||||||
|
});
|
||||||
|
|
||||||
caption = prepareTest('Overlapping circles - enveloping', container);
|
runTest('Disjoint circles', function() {
|
||||||
pathA = new Path.Circle(new Point(110, 110), 100);
|
return [
|
||||||
pathB = new Path.Circle(new Point(120, 110), 60);
|
new Path.Circle(new Point(60, 110), 50),
|
||||||
testBooleanStatic(pathA, pathB, caption);
|
new Path.Circle(new Point(170, 110), 50)
|
||||||
|
];
|
||||||
|
});
|
||||||
|
|
||||||
caption = prepareTest('Polygon and square', container);
|
runTest('Overlapping circles - enveloping', function() {
|
||||||
pathA = new Path.RegularPolygon(new Point(80, 110), 12, 80);
|
return [
|
||||||
pathB = new Path.Rectangle(new Point(100, 80), [80, 80]);
|
new Path.Circle(new Point(110, 110), 100),
|
||||||
testBooleanStatic(pathA, pathB, caption);
|
new Path.Circle(new Point(120, 110), 60)
|
||||||
|
];
|
||||||
|
});
|
||||||
|
|
||||||
caption = prepareTest('Circle and square (overlaps exactly on existing segments)', container);
|
runTest('Polygon and square', function() {
|
||||||
pathA = new Path.Circle(new Point(110, 110), 80);
|
return [
|
||||||
pathB = new Path.Rectangle(new Point(110, 110), [80, 80]);
|
new Path.RegularPolygon(new Point(80, 110), 12, 80),
|
||||||
testBooleanStatic(pathA, pathB, caption);
|
new Path.Rectangle(new Point(100, 80), [80, 80])
|
||||||
|
];
|
||||||
|
});
|
||||||
|
|
||||||
caption = prepareTest('Circle and square (existing segments overlaps on curves)', container);
|
runTest('Circle and square (overlaps exactly on existing segments)', function() {
|
||||||
pathA = new Path.Circle(new Point(110, 110), 80);
|
return [
|
||||||
pathB = new Path.Rectangle(new Point(110, 110), [100, 100]);
|
new Path.Circle(new Point(110, 110), 80),
|
||||||
testBooleanStatic(pathA, pathB, caption);
|
new Path.Rectangle(new Point(110, 110), [80, 80])
|
||||||
|
];
|
||||||
|
});
|
||||||
|
|
||||||
caption = prepareTest('Square and square (one segment overlaps on a line)', container);
|
runTest('Circle and square (existing segments overlaps on curves)', function() {
|
||||||
pathA = new Path.Rectangle(new Point(80, 125), [50, 50]);
|
return [
|
||||||
pathA.rotate(45);
|
new Path.Circle(new Point(110, 110), 80),
|
||||||
pathB = new Path.Rectangle(new Point(pathA.segments[2].point.x, 110), [80, 80]);
|
new Path.Rectangle(new Point(110, 110), [100, 100])
|
||||||
testBooleanStatic(pathA, pathB, caption);
|
];
|
||||||
|
});
|
||||||
|
|
||||||
caption = prepareTest('Rectangle and rectangle (overlaps exactly on existing curves)', container);
|
runTest('Square and square (one segment overlaps on a line)', function() {
|
||||||
pathA = new Path.Rectangle(new Point(30.5, 50.5), [100, 150]);
|
var pathA = new Path.Rectangle(new Point(80, 125), [50, 50]).rotate(45);
|
||||||
pathB = new Path.Rectangle(new Point(130.5, 60.5), [100, 150]);
|
var pathB = new Path.Rectangle(new Point(pathA.segments[2].point.x, 110), [80, 80]);
|
||||||
testBooleanStatic(pathA, pathB, caption);
|
return [pathA, pathB];
|
||||||
|
});
|
||||||
|
|
||||||
caption = prepareTest('Circle and banana (multiple intersections within same curve segment)', container);
|
runTest('Rectangle and rectangle (overlaps exactly on existing curves)', function() {
|
||||||
pathA = new Path.Circle(new Point(80, 110), 80);
|
return [
|
||||||
pathB = new Path.Circle(new Point(130, 110), 80);
|
new Path.Rectangle(new Point(30.5, 50.5), [100, 150]),
|
||||||
pathB.segments[3].point = pathB.segments[3].point.add([ 0, -120 ]);
|
new Path.Rectangle(new Point(130.5, 60.5), [100, 150])
|
||||||
testBooleanStatic(pathA, pathB, caption);
|
];
|
||||||
|
});
|
||||||
|
|
||||||
caption = prepareTest('Overlapping stars 1', container);
|
runTest('Circle and banana (multiple intersections within same curve segment)', function() {
|
||||||
pathA = new Path.Star(new Point(80, 110), 10, 20, 80);
|
var pathA = new Path.Circle(new Point(80, 110), 80);
|
||||||
pathB = new Path.Star(new Point(120, 110), 10, 30, 100);
|
var pathB = new Path.Circle(new Point(130, 110), 80);
|
||||||
testBooleanStatic(pathA, pathB, caption);
|
pathB.segments[3].point = pathB.segments[3].point.add([ 0, -120 ]);
|
||||||
|
return [pathA, pathB];
|
||||||
|
});
|
||||||
|
|
||||||
caption = prepareTest('Overlapping stars 2', container);
|
runTest('Overlapping stars 1', function() {
|
||||||
pathA = new Path.Star(new Point(110, 110), 20, 20, 80);
|
return [
|
||||||
pathB = new Path.Star(new Point(110, 110), 6, 30, 100);
|
new Path.Star(new Point(80, 110), 10, 20, 80),
|
||||||
testBooleanStatic(pathA, pathB, caption);
|
new Path.Star(new Point(120, 110), 10, 30, 100)
|
||||||
|
];
|
||||||
|
});
|
||||||
|
|
||||||
// caption = prepareTest('Circles overlap exactly over each other', container);
|
runTest('Overlapping stars 2', function() {
|
||||||
|
return [
|
||||||
|
new Path.Star(new Point(110, 110), 20, 20, 80),
|
||||||
|
new Path.Star(new Point(110, 110), 6, 30, 100)
|
||||||
|
];
|
||||||
|
});
|
||||||
|
|
||||||
|
// runTest('Circles overlap exactly over each other');
|
||||||
// pathA = new Path.Circle(new Point(110, 110), 100);
|
// pathA = new Path.Circle(new Point(110, 110), 100);
|
||||||
// pathB = new Path.Circle(new Point(110, 110), 100);
|
// pathB = new Path.Circle(new Point(110, 110), 100);
|
||||||
// // pathB.translate([0.5,0])
|
// // pathB.translate([0.5,0])
|
||||||
// testBooleanStatic(pathA, pathB, caption);
|
// testBooleanStatic(pathA, pathB);
|
||||||
|
|
||||||
caption = prepareTest('Maximum possible intersections between 2 cubic bezier curve segments - 9', container);
|
runTest('Maximum possible intersections between 2 cubic bezier curve segments - 9', function() {
|
||||||
pathA = new Path();
|
var pathA = new Path();
|
||||||
pathA.add(new Segment([173, 44], [-281, 268], [-86, 152]));
|
pathA.add(new Segment([173, 44], [-281, 268], [-86, 152]));
|
||||||
pathA.add(new Segment([47, 93], [-89, 100], [240, -239]));
|
pathA.add(new Segment([47, 93], [-89, 100], [240, -239]));
|
||||||
pathA.closed = true;
|
pathA.closed = true;
|
||||||
pathB = pathA.clone();
|
var pathB = pathA.clone();
|
||||||
pathB.rotate(-90);
|
pathB.rotate(-90);
|
||||||
pathA.translate([-10,0]);
|
pathA.translate([-10,0]);
|
||||||
pathB.translate([10,0]);
|
pathB.translate([10,0]);
|
||||||
testBooleanStatic(pathA, pathB, caption);
|
annotatePath(pathA, null, '#008');
|
||||||
annotatePath(pathA, null, '#008');
|
annotatePath(pathB, null, '#800');
|
||||||
annotatePath(pathB, null, '#800');
|
return [pathA, pathB];
|
||||||
view.draw();
|
});
|
||||||
|
|
||||||
caption = prepareTest('SVG gears', container);
|
runTest('SVG gears', function() {
|
||||||
group = paper.project.importSVG(document.getElementById('svggears'));
|
var group = paper.project.importSVG(document.getElementById('svggears'));
|
||||||
pathA = group.children[0];
|
return group.children;
|
||||||
pathB = group.children[1];
|
});
|
||||||
testBooleanStatic(pathA, pathB, caption);
|
|
||||||
|
|
||||||
caption = prepareTest('Glyphs imported from SVG', container);
|
runTest('Glyphs imported from SVG', function() {
|
||||||
group = paper.project.importSVG(document.getElementById('glyphsys'));
|
var group = paper.project.importSVG(document.getElementById('glyphsys'));
|
||||||
pathA = group.children[0];
|
return group.children;
|
||||||
pathB = group.children[1];
|
});
|
||||||
testBooleanStatic(pathA, pathB, caption);
|
|
||||||
|
|
||||||
caption = prepareTest('CompoundPaths 1', container);
|
runTest('CompoundPaths 1', function() {
|
||||||
group = paper.project.importSVG(document.getElementById('glyphsacirc'));
|
var group = paper.project.importSVG(document.getElementById('glyphsacirc'));
|
||||||
pathA = group.children[0];
|
return group.children;
|
||||||
pathB = group.children[1];
|
});
|
||||||
testBooleanStatic(pathA, pathB, caption);
|
|
||||||
|
|
||||||
caption = prepareTest('CompoundPaths 2 - holes', container);
|
runTest('CompoundPaths 2 - holes', function() {
|
||||||
group = paper.project.importSVG(document.getElementById('glyphsacirc'));
|
var group = paper.project.importSVG(document.getElementById('glyphsacirc'));
|
||||||
pathA = group.children[0];
|
var pathA = group.children[0];
|
||||||
pathB = new CompoundPath();
|
var pathB = new CompoundPath();
|
||||||
group.children[1].clockwise = true;
|
group.children[1].clockwise = true;
|
||||||
pathB.addChild(group.children[1]);
|
pathB.addChild(group.children[1]);
|
||||||
var npath = new Path.Circle([110, 110], 30);
|
pathB.addChild(new Path.Circle([110, 110], 30));
|
||||||
pathB.addChild(npath);
|
return [pathA, pathB];
|
||||||
testBooleanStatic(pathA, pathB, caption);
|
});
|
||||||
|
|
||||||
caption = prepareTest('CompoundPaths 3 !', container);
|
runTest('CompoundPaths 3 !', function() {
|
||||||
group = paper.project.importSVG(document.getElementById('svggreenland'));
|
var group = paper.project.importSVG(document.getElementById('svggreenland'));
|
||||||
pathA = group.children[0];
|
group.children[1].scale(0.5, 1).translate([25.5, 0]);
|
||||||
pathB = group.children[1];
|
return group.children;
|
||||||
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);
|
runTest('CompoundPaths 4 - holes and islands 1', function() {
|
||||||
group = paper.project.importSVG(document.getElementById('glyphsacirc'));
|
var group = paper.project.importSVG(document.getElementById('glyphsacirc'));
|
||||||
pathA = group.children[0];
|
var pathA = group.children[0];
|
||||||
pathB = new CompoundPath();
|
var pathB = new CompoundPath();
|
||||||
group.children[1].clockwise = true;
|
group.children[1].clockwise = true;
|
||||||
pathB.addChild(group.children[1]);
|
pathB.addChild(group.children[1]);
|
||||||
var npath = new Path.Circle([40, 80], 20);
|
pathB.addChild(new Path.Circle([40, 80], 20));
|
||||||
pathB.addChild(npath);
|
return [pathA, pathB];
|
||||||
testBooleanStatic(pathA, pathB, caption);
|
});
|
||||||
|
|
||||||
caption = prepareTest('CompoundPaths 5 - holes and islands 2', container);
|
runTest('CompoundPaths 5 - holes and islands 2', function() {
|
||||||
group = paper.project.importSVG(document.getElementById('glyphsacirc'));
|
var group = paper.project.importSVG(document.getElementById('glyphsacirc'));
|
||||||
pathA = group.children[0];
|
var pathA = group.children[0];
|
||||||
pathB = new CompoundPath();
|
var pathB = new CompoundPath();
|
||||||
group.children[1].clockwise = true;
|
group.children[1].clockwise = true;
|
||||||
pathB.addChild(group.children[1]);
|
pathB.addChild(group.children[1]);
|
||||||
var npath = new Path.Circle([40, 80], 20);
|
var npath = new Path.Circle([40, 80], 20);
|
||||||
pathB.addChild(npath);
|
pathB.addChild(npath);
|
||||||
npath = new Path.Circle([120, 110], 30);
|
npath = new Path.Circle([120, 110], 30);
|
||||||
pathB.addChild(npath);
|
pathB.addChild(npath);
|
||||||
testBooleanStatic(pathA, pathB, caption);
|
return [pathA, pathB];
|
||||||
|
});
|
||||||
|
|
||||||
caption = prepareTest('CompoundPaths 6 - holes and islands 3', container);
|
runTest('CompoundPaths 6 - holes and islands 3', function() {
|
||||||
group = paper.project.importSVG(document.getElementById('glyphsacirc'));
|
var group = paper.project.importSVG(document.getElementById('glyphsacirc'));
|
||||||
pathA = group.children[0];
|
var pathA = group.children[0];
|
||||||
pathB = new CompoundPath();
|
var pathB = new CompoundPath();
|
||||||
var npath = new Path.Circle([110, 110], 100);
|
var npath = new Path.Circle([110, 110], 100);
|
||||||
pathB.addChild(npath);
|
pathB.addChild(npath);
|
||||||
npath = new Path.Circle([110, 110], 60);
|
npath = new Path.Circle([110, 110], 60);
|
||||||
pathB.addChild(npath);
|
pathB.addChild(npath);
|
||||||
npath = new Path.Circle([110, 110], 30);
|
npath = new Path.Circle([110, 110], 30);
|
||||||
pathB.addChild(npath);
|
pathB.addChild(npath);
|
||||||
testBooleanStatic(pathA, pathB, caption);
|
return [pathA, pathB];
|
||||||
|
});
|
||||||
|
|
||||||
caption = prepareTest('CompoundPaths 6 - holes and islands 4 (curves overlap exactly on existing curves)', container);
|
runTest('CompoundPaths 6 - holes and islands 4 (curves overlap exactly on existing curves)', function() {
|
||||||
pathA = new Path.Rectangle(new Point(50.5, 50.5), [100, 120]);
|
var pathA = new Path.Rectangle(new Point(50.5, 50.5), [100, 120]);
|
||||||
pathB = new CompoundPath();
|
var pathB = new CompoundPath();
|
||||||
pathB.addChild(new Path.Rectangle(new Point(140.5, 30.5), [100, 150]));
|
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.addChild(new Path.Rectangle(new Point(150.5, 65.5), [50, 100]));
|
||||||
// pathB = new Path.Rectangle(new Point(150.5, 80.5), [80, 80]);
|
// pathB = new Path.Rectangle(new Point(150.5, 80.5), [80, 80])
|
||||||
testBooleanStatic(pathA, pathB, caption);
|
return [pathA, pathB];
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
// // To resolve self intersection on a single path,
|
// // To resolve self intersection on a single path,
|
||||||
// // pass an empty second operand and do a Union operation
|
// // pass an empty second operand and do a Union operation
|
||||||
// caption = prepareTest('Self-intersecting paths 1 - Resolve self-intersection on single path', container);
|
// runTest('Self-intersecting paths 1 - Resolve self-intersection on single path');
|
||||||
// pathA = new Path.Star(new Point(110, 110), 10, 20, 80);
|
// pathA = new Path.Star(new Point(110, 110), 10, 20, 80);
|
||||||
// pathA.smooth();
|
// pathA.smooth();
|
||||||
// pathB = new Path();
|
// pathB = new Path();
|
||||||
// testBooleanStatic(pathA, pathB, caption, false, true, true, true);
|
// testBooleanStatic(pathA, pathB, caption, false, true, true, true);
|
||||||
|
|
||||||
// caption = prepareTest('Self-intersecting paths 2 - Resolve self-intersecting CompoundPath', container);
|
// runTest('Self-intersecting paths 2 - Resolve self-intersecting CompoundPath');
|
||||||
// pathA = new CompoundPath();
|
// pathA = new CompoundPath();
|
||||||
// pathA.addChild(new Path.Circle([100, 110], 60));
|
// pathA.addChild(new Path.Circle([100, 110], 60));
|
||||||
// pathA.addChild(new Path.Circle([160, 110], 30));
|
// pathA.addChild(new Path.Circle([160, 110], 30));
|
||||||
|
@ -200,9 +239,6 @@
|
||||||
// }
|
// }
|
||||||
// };
|
// };
|
||||||
|
|
||||||
window.a = pathA;
|
|
||||||
window.b = pathB;
|
|
||||||
|
|
||||||
// // pathA.selected = true;
|
// // pathA.selected = true;
|
||||||
// // pathA.fullySelected = true;
|
// // pathA.fullySelected = true;
|
||||||
// // pathB.selected = true;
|
// // pathB.selected = true;
|
||||||
|
@ -228,21 +264,8 @@
|
||||||
// var nup = unite(pathA, pathB);
|
// var nup = unite(pathA, pathB);
|
||||||
// console.timeEnd('unite');
|
// console.timeEnd('unite');
|
||||||
// // nup.style = booleanStyle;
|
// // nup.style = booleanStyle;
|
||||||
// window.p = nup;
|
|
||||||
|
|
||||||
// view.draw();
|
// 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 = {
|
var booleanStyle = {
|
||||||
|
@ -264,71 +287,64 @@
|
||||||
};
|
};
|
||||||
|
|
||||||
// Better if path1 and path2 fit nicely inside a 200x200 pixels rect
|
// Better if path1 and path2 fit nicely inside a 200x200 pixels rect
|
||||||
function testBooleanStatic(path1, path2, caption, noUnion, noIntersection, noSubtraction, _disperse) {
|
function testBooleanStatic(path1, path2) {
|
||||||
// try{
|
path1.style = path2.style = pathStyleNormal;
|
||||||
path1.style = path2.style = pathStyleNormal;
|
|
||||||
|
|
||||||
if (!noUnion) {
|
if (operations.union) {
|
||||||
var _p1U = path1.clone().translate([250, 0]);
|
var _p1U = path1.clone().translate([250, 0]);
|
||||||
var _p2U = path2.clone().translate([250, 0]);
|
var _p2U = path2.clone().translate([250, 0]);
|
||||||
_p1U.style = _p2U.style = pathStyleBoolean;
|
_p1U.style = _p2U.style = pathStyleBoolean;
|
||||||
console.time('Union');
|
console.time('Union');
|
||||||
var boolPathU = _p1U.unite(_p2U);
|
var boolPathU = _p1U.unite(_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 = _p1I.intersect(_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 = _p1S.subtract(_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 = _p1E.exclude(_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 = _p1D.divide(_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('Union');
|
||||||
console.timeEnd('Intersection');
|
boolPathU.style = booleanStyle;
|
||||||
console.timeEnd('Subtraction');
|
|
||||||
view.draw();
|
view.draw();
|
||||||
// }
|
}
|
||||||
|
|
||||||
|
if (operations.intersection) {
|
||||||
|
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');
|
||||||
|
boolPathI.style = booleanStyle;
|
||||||
|
view.draw();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (operations.subtraction) {
|
||||||
|
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');
|
||||||
|
boolPathS.style = booleanStyle;
|
||||||
|
view.draw();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (operations.exclusion) {
|
||||||
|
var _p1E = path1.clone().translate([250, 220]);
|
||||||
|
var _p2E = path2.clone().translate([250, 220]);
|
||||||
|
_p1E.style = _p2E.style = pathStyleBoolean;
|
||||||
|
console.time('Exclusion');
|
||||||
|
var boolPathE = _p1E.exclude(_p2E);
|
||||||
|
console.timeEnd('Exclusion');
|
||||||
|
boolPathE.style = booleanStyle;
|
||||||
|
view.draw();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (operations.division) {
|
||||||
|
var _p1D = path1.clone().translate([500, 220]);
|
||||||
|
var _p2D = path2.clone().translate([500, 220]);
|
||||||
|
_p1D.style = _p2D.style = pathStyleBoolean;
|
||||||
|
console.time('Division');
|
||||||
|
var boolPathD = _p1D.divide(_p2D);
|
||||||
|
console.timeEnd('Division');
|
||||||
|
disperse(boolPathD);
|
||||||
|
boolPathD.style = booleanStyle;
|
||||||
|
view.draw();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function disperse(path, distance) {
|
function disperse(path, distance) {
|
||||||
|
@ -436,6 +452,8 @@
|
||||||
body {
|
body {
|
||||||
height: 100%;
|
height: 100%;
|
||||||
overflow: auto;
|
overflow: auto;
|
||||||
|
font-family: 'Helvetica Neue';
|
||||||
|
font-weight: 300;
|
||||||
}
|
}
|
||||||
#container {
|
#container {
|
||||||
display: block;
|
display: block;
|
||||||
|
@ -443,18 +461,14 @@
|
||||||
margin: 0 auto 50px;
|
margin: 0 auto 50px;
|
||||||
}
|
}
|
||||||
h1, h3 {
|
h1, h3 {
|
||||||
font-family: 'Helvetica Neue';
|
|
||||||
font-weight: 300;
|
font-weight: 300;
|
||||||
margin: 50px 0 20px;
|
margin: 50px 0 20px;
|
||||||
}
|
}
|
||||||
footer{
|
footer {
|
||||||
display: block;
|
display: block;
|
||||||
width: 1000px;
|
width: 1000px;
|
||||||
margin: 30px auto;
|
margin: 30px auto;
|
||||||
color: #999;
|
color: #999;
|
||||||
font-family: 'Helvetica Neue';
|
|
||||||
font-weight: 300;
|
|
||||||
font-style: italic;
|
|
||||||
}
|
}
|
||||||
canvas {
|
canvas {
|
||||||
cursor: crosshair;
|
cursor: crosshair;
|
||||||
|
@ -476,30 +490,21 @@
|
||||||
<button id="testStart" value="Start tests" onClick="runTests();">Start tests</button>
|
<button id="testStart" value="Start tests" onClick="runTests();">Start tests</button>
|
||||||
</div>
|
</div>
|
||||||
<footer>
|
<footer>
|
||||||
<p>Vector boolean operations on paperjs objects.</p>
|
|
||||||
<p>
|
|
||||||
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.
|
|
||||||
</p>
|
|
||||||
<p>Supported</p>
|
<p>Supported</p>
|
||||||
<ul>
|
<ul>
|
||||||
<li>paperjs Path and CompoundPath objects</li>
|
<li>All PathItem classes (Path and CompoundPath)</li>
|
||||||
<li>Boolean Union</li>
|
<li>Boolean Union</li>
|
||||||
<li>Boolean Intersection</li>
|
<li>Boolean Intersection</li>
|
||||||
<li>Boolean Subtraction</li>
|
<li>Boolean Subtraction</li>
|
||||||
<li>Resolving a self-intersecting Path</li>
|
<li>Resolving a self-intersecting Path</li>
|
||||||
</ul>
|
</ul>
|
||||||
<p>Not supported yet ( which I would like to see supported )</p>
|
<p>Not supported yet</p>
|
||||||
<ul>
|
<ul>
|
||||||
<li>Boolean operations between self-intersecting Paths</li>
|
<li>Boolean operations between self-intersecting Paths</li>
|
||||||
<li>Paths are clones of each other that ovelap exactly on top of each other!</li>
|
<li>Paths that are clones of each other and lie in the exact same position</li>
|
||||||
</ul>
|
</ul>
|
||||||
<p>
|
<p>
|
||||||
This is meant to be integrated into the paperjs library in the near future.
|
<a href="http://hkrish.com">Harikrishnan Gopalakrishnan</a>
|
||||||
</p>
|
|
||||||
<p>
|
|
||||||
Harikrishnan Gopalakrishnan<br>
|
|
||||||
http://hkrish.com/playground/paperjs/booleanStudy.html
|
|
||||||
<p>
|
<p>
|
||||||
</footer>
|
</footer>
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue