mirror of
https://github.com/scratchfoundation/paper.js.git
synced 2025-01-23 15:59:45 -05:00
First attempt at implementing handling of multiple intersections in the same location.
Relates to #787, works pretty well already for many situations.
This commit is contained in:
parent
18c5a06f45
commit
2c8634793c
1 changed files with 49 additions and 9 deletions
|
@ -71,7 +71,7 @@ PathItem.inject(new function() {
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
var scaleFactor = 0.5; // 1 / 3000;
|
var scaleFactor = 1.0; // 1 / 3000;
|
||||||
var textAngle = 33;
|
var textAngle = 33;
|
||||||
var fontSize = 5;
|
var fontSize = 5;
|
||||||
|
|
||||||
|
@ -87,8 +87,6 @@ PathItem.inject(new function() {
|
||||||
function computeBoolean(path1, path2, operation) {
|
function computeBoolean(path1, path2, operation) {
|
||||||
segmentOffset = {};
|
segmentOffset = {};
|
||||||
pathIndices = {};
|
pathIndices = {};
|
||||||
pathIndex = 0;
|
|
||||||
pathCount = 1;
|
|
||||||
|
|
||||||
// We do not modify the operands themselves, but create copies instead,
|
// We do not modify the operands themselves, but create copies instead,
|
||||||
// fas produced by the calls to preparePath().
|
// fas produced by the calls to preparePath().
|
||||||
|
@ -220,6 +218,12 @@ PathItem.inject(new function() {
|
||||||
if (segment._intersection) {
|
if (segment._intersection) {
|
||||||
console.log('Segment already has an intersection: '
|
console.log('Segment already has an intersection: '
|
||||||
+ segment._intersection + ', ' + loc._intersection);
|
+ segment._intersection + ', ' + loc._intersection);
|
||||||
|
// Create a chain of possible intersections linked through
|
||||||
|
// _next:
|
||||||
|
var inter = segment._intersection;
|
||||||
|
while (inter._next)
|
||||||
|
inter = inter._next;
|
||||||
|
inter._next = loc;
|
||||||
} else {
|
} else {
|
||||||
segment._intersection = loc._intersection;
|
segment._intersection = loc._intersection;
|
||||||
}
|
}
|
||||||
|
@ -451,6 +455,8 @@ PathItem.inject(new function() {
|
||||||
* @return {Path[]} the contours traced
|
* @return {Path[]} the contours traced
|
||||||
*/
|
*/
|
||||||
function tracePaths(segments, operation) {
|
function tracePaths(segments, operation) {
|
||||||
|
pathIndex = 0;
|
||||||
|
pathCount = 1;
|
||||||
|
|
||||||
function labelSegment(seg, text, color) {
|
function labelSegment(seg, text, color) {
|
||||||
var point = seg.point;
|
var point = seg.point;
|
||||||
|
@ -492,6 +498,7 @@ PathItem.inject(new function() {
|
||||||
+ ' op: ' + (operator && operator(seg._winding))
|
+ ' op: ' + (operator && operator(seg._winding))
|
||||||
+ ' ov: ' + (inter && inter._overlap || 0)
|
+ ' ov: ' + (inter && inter._overlap || 0)
|
||||||
+ ' wi: ' + seg._winding
|
+ ' wi: ' + seg._winding
|
||||||
|
+ ' mu: ' + !!(inter && inter._next)
|
||||||
, color);
|
, color);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -516,16 +523,23 @@ PathItem.inject(new function() {
|
||||||
var paths = [],
|
var paths = [],
|
||||||
operator = operators[operation];
|
operator = operators[operation];
|
||||||
for (var i = 0, l = segments.length; i < l; i++) {
|
for (var i = 0, l = segments.length; i < l; i++) {
|
||||||
var seg = segments[i];
|
var seg = segments[i],
|
||||||
if (seg._visited || operator && !operator(seg._winding))
|
inter = seg._intersection;
|
||||||
|
// Do not start a chain with already visited segments, intersections
|
||||||
|
// with multiple possibilities (TODO: Is that really a problem?),
|
||||||
|
// and segments that are not going to be part of the resulting
|
||||||
|
// operation.
|
||||||
|
if (seg._visited /* || inter && inter._next*/
|
||||||
|
|| operator && !operator(seg._winding))
|
||||||
continue;
|
continue;
|
||||||
var path = new Path(Item.NO_INSERT),
|
var path = new Path(Item.NO_INSERT),
|
||||||
start = seg,
|
start = seg,
|
||||||
inter = seg._intersection,
|
otherStart = null,
|
||||||
other = inter && inter._segment,
|
|
||||||
otherStart = other,
|
|
||||||
added = false; // Whether a first segment as added already
|
added = false; // Whether a first segment as added already
|
||||||
do {
|
do {
|
||||||
|
var other = inter && inter._segment;
|
||||||
|
if (!otherStart)
|
||||||
|
otherStart = other;
|
||||||
var handleIn = added && seg._handleIn;
|
var handleIn = added && seg._handleIn;
|
||||||
if (!added || !other || other === start) {
|
if (!added || !other || other === start) {
|
||||||
// TODO: Is (other === start) check really required?
|
// TODO: Is (other === start) check really required?
|
||||||
|
@ -574,7 +588,8 @@ PathItem.inject(new function() {
|
||||||
console.error('Unable to switch to intersecting segment, '
|
console.error('Unable to switch to intersecting segment, '
|
||||||
+ 'aborting #' + pathCount + '.'
|
+ 'aborting #' + pathCount + '.'
|
||||||
+ (path ? path._segments.length + 1 : 1)
|
+ (path ? path._segments.length + 1 : 1)
|
||||||
+ ' id: ' + seg._path._id + '.' + seg._index);
|
+ ', id: ' + seg._path._id + '.' + seg._index
|
||||||
|
+ ', multiple: ' + (!!inter._next));
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
// Add the current segment to the path, and mark the added
|
// Add the current segment to the path, and mark the added
|
||||||
|
@ -583,6 +598,31 @@ PathItem.inject(new function() {
|
||||||
seg._visited = added = true;
|
seg._visited = added = true;
|
||||||
seg = seg.getNext();
|
seg = seg.getNext();
|
||||||
inter = seg && seg._intersection;
|
inter = seg && seg._intersection;
|
||||||
|
// If there are multiple possible intersections, find the one
|
||||||
|
// that's either connecting back to start or is not visited yet,
|
||||||
|
// and will be part of the boolean result:
|
||||||
|
function check(inter, ignoreOther) {
|
||||||
|
if (!inter)
|
||||||
|
return null;
|
||||||
|
var seg = inter._segment,
|
||||||
|
next = seg.getNext();
|
||||||
|
console.log('Multiple'
|
||||||
|
+ ', seg: ' + seg._path._id + '.' + seg._index
|
||||||
|
+ ', next: ' + next._path._id + '.' + next._index
|
||||||
|
+ ', visited:' + !!next._visited
|
||||||
|
+ ', operator:' + (operator && operator(next._winding))
|
||||||
|
+ ', start: ' + (next === start)
|
||||||
|
+ ', next: ' + (!!inter._next));
|
||||||
|
return next === start || !next._visited
|
||||||
|
&& (!operator || operator(next._winding))
|
||||||
|
? inter
|
||||||
|
// If no match, check the other intersection first,
|
||||||
|
// then carry on with the next linked intersection.
|
||||||
|
: !ignoreOther && check(inter._intersection, true)
|
||||||
|
|| check(inter._next);
|
||||||
|
}
|
||||||
|
inter = check(inter) || inter;
|
||||||
|
|
||||||
other = inter && inter._segment;
|
other = inter && inter._segment;
|
||||||
if (seg === start || seg === otherStart) {
|
if (seg === start || seg === otherStart) {
|
||||||
drawSegment(seg, 'done', i, 'red');
|
drawSegment(seg, 'done', i, 'red');
|
||||||
|
|
Loading…
Reference in a new issue