Improve handling of sub-path orientation in CompoundPath.

Remove automatic orientation on insertion, as it caused more troubles than solved problems, in favor of the new PathItem#reorient() method, or the even-odd fill-rule.

Closes #590, #1029
This commit is contained in:
Jürg Lehni 2016-07-21 15:21:45 +02:00
parent fc72c05e69
commit a0417040f8
10 changed files with 151 additions and 217 deletions

View file

@ -157,7 +157,7 @@ new function() { // Injection scope for various item event handlers
this._setProject(project); this._setProject(project);
} else { } else {
(hasProps && props.parent || project) (hasProps && props.parent || project)
._insertItem(undefined, this, true, true); ._insertItem(undefined, this, true); // _created = true
} }
// Filter out Item.NO_INSERT before _set(), for performance reasons. // Filter out Item.NO_INSERT before _set(), for performance reasons.
if (hasProps && props !== Item.NO_INSERT) { if (hasProps && props !== Item.NO_INSERT) {
@ -1373,9 +1373,9 @@ new function() { // Injection scope for various item event handlers
return this._children; return this._children;
}, },
setChildren: function(items, _preserve) { setChildren: function(items) {
this.removeChildren(); this.removeChildren();
this.addChildren(items, _preserve); this.addChildren(items);
}, },
/** /**
@ -2287,8 +2287,8 @@ new function() { // Injection scope for hit-test functions shared with project
* @param {Item} item the item to be added as a child * @param {Item} item the item to be added as a child
* @return {Item} the added item, or `null` if adding was not possible * @return {Item} the added item, or `null` if adding was not possible
*/ */
addChild: function(item, _preserve) { addChild: function(item) {
return this.insertChild(undefined, item, _preserve); return this.insertChild(undefined, item);
}, },
/** /**
@ -2300,8 +2300,8 @@ new function() { // Injection scope for hit-test functions shared with project
* @param {Item} item the item to be inserted as a child * @param {Item} item the item to be inserted as a child
* @return {Item} the inserted item, or `null` if inserting was not possible * @return {Item} the inserted item, or `null` if inserting was not possible
*/ */
insertChild: function(index, item, _preserve) { insertChild: function(index, item) {
var res = item ? this.insertChildren(index, [item], _preserve) : null; var res = item ? this.insertChildren(index, [item]) : null;
return res && res[0]; return res && res[0];
}, },
@ -2313,8 +2313,8 @@ new function() { // Injection scope for hit-test functions shared with project
* @param {Item[]} items the items to be added as children * @param {Item[]} items the items to be added as children
* @return {Item[]} the added items, or `null` if adding was not possible * @return {Item[]} the added items, or `null` if adding was not possible
*/ */
addChildren: function(items, _preserve) { addChildren: function(items) {
return this.insertChildren(this._children.length, items, _preserve); return this.insertChildren(this._children.length, items);
}, },
/** /**
@ -2327,9 +2327,7 @@ new function() { // Injection scope for hit-test functions shared with project
* @return {Item[]} the inserted items, or `null` if inserted was not * @return {Item[]} the inserted items, or `null` if inserted was not
* possible * possible
*/ */
insertChildren: function(index, items, _preserve) { insertChildren: function(index, items) {
// CompoundPath#insertChildren() requires _preserve and _type:
// _preserve avoids changing of the children's path orientation
var children = this._children; var children = this._children;
if (children && items && items.length > 0) { if (children && items && items.length > 0) {
// We need to clone items because it may be an Item#children array. // We need to clone items because it may be an Item#children array.
@ -2386,7 +2384,7 @@ new function() { // Injection scope for hit-test functions shared with project
* @param {Number} offset the offset at which the item should be inserted * @param {Number} offset the offset at which the item should be inserted
* @return {Item} the inserted item, or `null` if inserting was not possible * @return {Item} the inserted item, or `null` if inserting was not possible
*/ */
_insertAt: function(item, offset, _preserve) { _insertAt: function(item, offset) {
var res = this; var res = this;
if (res !== item) { if (res !== item) {
var owner = item && item._getOwner(); var owner = item && item._getOwner();
@ -2394,7 +2392,7 @@ new function() { // Injection scope for hit-test functions shared with project
// Notify parent of change. Don't notify item itself yet, // Notify parent of change. Don't notify item itself yet,
// as we're doing so when adding it to the new owner below. // as we're doing so when adding it to the new owner below.
res._remove(false, true); res._remove(false, true);
owner._insertItem(item._index + offset, res, _preserve); owner._insertItem(item._index + offset, res);
} else { } else {
res = null; res = null;
} }
@ -2408,8 +2406,8 @@ new function() { // Injection scope for hit-test functions shared with project
* @param {Item} item the item above which it should be inserted * @param {Item} item the item above which it should be inserted
* @return {Item} the inserted item, or `null` if inserting was not possible * @return {Item} the inserted item, or `null` if inserting was not possible
*/ */
insertAbove: function(item, _preserve) { insertAbove: function(item) {
return this._insertAt(item, 1, _preserve); return this._insertAt(item, 1);
}, },
/** /**
@ -2418,8 +2416,8 @@ new function() { // Injection scope for hit-test functions shared with project
* @param {Item} item the item below which it should be inserted * @param {Item} item the item below which it should be inserted
* @return {Item} the inserted item, or `null` if inserting was not possible * @return {Item} the inserted item, or `null` if inserting was not possible
*/ */
insertBelow: function(item, _preserve) { insertBelow: function(item) {
return this._insertAt(item, 0, _preserve); return this._insertAt(item, 0);
}, },
/** /**

View file

@ -378,13 +378,13 @@ var Project = PaperScopeItem.extend(/** @lends Project# */{
// Project#_insertItem() and Item#_insertItem() are helper functions called // Project#_insertItem() and Item#_insertItem() are helper functions called
// in Item#copyTo(), and through _getOwner() in the various Item#insert*() // in Item#copyTo(), and through _getOwner() in the various Item#insert*()
// methods. They are called the same to facilitate so duck-typing. // methods. They are called the same to facilitate so duck-typing.
_insertItem: function(index, item, _preserve, _created) { _insertItem: function(index, item, _created) {
item = this.insertLayer(index, item) item = this.insertLayer(index, item)
// Anything else than layers needs to be added to a layer first. // Anything else than layers needs to be added to a layer first.
// If none exists yet, create one now, then add the item to it. // If none exists yet, create one now, then add the item to it.
|| (this._activeLayer || this._insertItem(undefined, || (this._activeLayer || this._insertItem(undefined,
new Layer(Item.NO_INSERT), true, true)) new Layer(Item.NO_INSERT), true)) // _created = true
.insertChild(index, item, _preserve); .insertChild(index, item);
// If a layer was newly created, also activate it. // If a layer was newly created, also activate it.
if (_created && item.activate) if (_created && item.activate)
item.activate(); item.activate();

View file

@ -106,7 +106,7 @@ var CompoundPath = PathItem.extend(/** @lends CompoundPath# */{
} }
}, },
insertChildren: function insertChildren(index, items, _preserve) { insertChildren: function insertChildren(index, items) {
// If we're passed a segment array describing a simple path instead of a // If we're passed a segment array describing a simple path instead of a
// compound-path, wrap it in another array to turn it into the array // compound-path, wrap it in another array to turn it into the array
// notation for compound-paths. // notation for compound-paths.
@ -124,28 +124,13 @@ var CompoundPath = PathItem.extend(/** @lends CompoundPath# */{
if (list === items && !(item instanceof Path)) if (list === items && !(item instanceof Path))
list = Base.slice(list); list = Base.slice(list);
if (Array.isArray(item)) { if (Array.isArray(item)) {
var path = new Path({ segments: item, insert: false }); list[i] = new Path({ segments: item, insert: false });
// Fix natural clockwise value, so it's not automatically
// determined when inserted into the compound-path.
// TODO: Remove reorientation code instead.
path.setClockwise(path.isClockwise());
list[i] = path;
} else if (item instanceof CompoundPath) { } else if (item instanceof CompoundPath) {
list.splice.apply(list, [i, 1].concat(item.removeChildren())); list.splice.apply(list, [i, 1].concat(item.removeChildren()));
item.remove(); item.remove();
} }
} }
list = insertChildren.base.call(this, index, list, _preserve); return insertChildren.base.call(this, index, list);
// All children except for the bottom one (first one in list) are set
// to anti-clockwise orientation, so that they appear as holes, but
// only if their orientation was not already specified before
// (= _clockwise is defined).
for (var i = 0, l = !_preserve && list && list.length; i < l; i++) {
var item = list[i];
if (item._clockwise === undefined)
item.setClockwise(item._index === 0);
}
return list;
}, },
// DOCS: reduce() // DOCS: reduce()
@ -167,22 +152,6 @@ var CompoundPath = PathItem.extend(/** @lends CompoundPath# */{
return reduce.base.call(this); return reduce.base.call(this);
}, },
/**
* Specifies whether the compound path is oriented clock-wise.
*
* @bean
* @type Boolean
*/
isClockwise: function() {
var child = this.getFirstChild();
return child && child.isClockwise();
},
setClockwise: function(clockwise) {
if (this.isClockwise() ^ !!clockwise)
this.reverse();
},
/** /**
* Specifies whether the compound-path is fully closed, meaning all its * Specifies whether the compound-path is fully closed, meaning all its
* contained sub-paths are closed path. * contained sub-paths are closed path.

View file

@ -134,17 +134,12 @@ var Path = PathItem.extend(/** @lends Path# */{
copyContent: function(source) { copyContent: function(source) {
this.setSegments(source._segments); this.setSegments(source._segments);
this._closed = source._closed; this._closed = source._closed;
var clockwise = source._clockwise;
if (clockwise !== undefined)
this._clockwise = clockwise;
}, },
_changed: function _changed(flags) { _changed: function _changed(flags) {
_changed.base.call(this, flags); _changed.base.call(this, flags);
if (flags & /*#=*/ChangeFlag.GEOMETRY) { if (flags & /*#=*/ChangeFlag.GEOMETRY) {
// Clockwise state becomes undefined as soon as geometry changes. this._length = this._area = undefined;
// Also clear cached mono curves used for winding calculations.
this._length = this._area = this._clockwise = undefined;
if (flags & /*#=*/ChangeFlag.SEGMENTS) { if (flags & /*#=*/ChangeFlag.SEGMENTS) {
this._version++; // See CurveLocation this._version++; // See CurveLocation
} else if (this._curves) { } else if (this._curves) {
@ -854,29 +849,6 @@ var Path = PathItem.extend(/** @lends Path# */{
return area; return area;
}, },
/**
* Specifies whether the path is oriented clock-wise.
*
* @bean
* @type Boolean
*/
isClockwise: function() {
if (this._clockwise !== undefined)
return this._clockwise;
return this.getArea() >= 0;
},
setClockwise: function(clockwise) {
// Only revers the path if its clockwise orientation is not the same
// as what it is now demanded to be.
// On-the-fly conversion to boolean:
if (this.isClockwise() != (clockwise = !!clockwise))
this.reverse();
// Reverse only flips _clockwise state if it was already set, so let's
// always set this here now.
this._clockwise = clockwise;
},
/** /**
* Specifies whether an path is selected and will also return `true` if the * Specifies whether an path is selected and will also return `true` if the
* path is partially selected, i.e. one or more of its segments is selected. * path is partially selected, i.e. one or more of its segments is selected.
@ -1084,9 +1056,7 @@ var Path = PathItem.extend(/** @lends Path# */{
path = this; path = this;
} else { } else {
path = new Path(Item.NO_INSERT); path = new Path(Item.NO_INSERT);
// Pass true for _preserve, in case of CompoundPath, to avoid path.insertAbove(this);
// reversing of path direction, which would mess with segments!
path.insertAbove(this, true);
path.copyAttributes(this); path.copyAttributes(this);
} }
path._add(segs, 0); path._add(segs, 0);
@ -1264,9 +1234,6 @@ var Path = PathItem.extend(/** @lends Path# */{
} }
// Clear curves since it all has changed. // Clear curves since it all has changed.
this._curves = null; this._curves = null;
// Flip clockwise state if it's defined
if (this._clockwise !== undefined)
this._clockwise = !this._clockwise;
this._changed(/*#=*/Change.GEOMETRY); this._changed(/*#=*/Change.GEOMETRY);
}, },

View file

@ -55,7 +55,9 @@ PathItem.inject(new function() {
.transform(null, true, true); .transform(null, true, true);
if (closed) if (closed)
res.setClosed(true); res.setClosed(true);
return closed ? res.resolveCrossings().reorient(true) : res; return closed
? res.resolveCrossings().reorient(res.getFillRule() === 'nonzero')
: res;
} }
function createResult(ctor, paths, reduce, path1, path2) { function createResult(ctor, paths, reduce, path1, path2) {
@ -990,11 +992,8 @@ PathItem.inject(new function() {
var length = paths.length, var length = paths.length,
item; item;
if (length > 1 && children) { if (length > 1 && children) {
if (paths !== children) { if (paths !== children)
// TODO: Fix automatic child-orientation in CompoundPath, this.setChildren(paths);
// and stop passing true for _preserve.
this.setChildren(paths, true); // Preserve orientation
}
item = this; item = this;
} else if (length === 1 && !children) { } else if (length === 1 && !children) {
if (paths[0] !== this) if (paths[0] !== this)
@ -1005,7 +1004,7 @@ PathItem.inject(new function() {
// and attempt to replace this item with it. // and attempt to replace this item with it.
if (!item) { if (!item) {
item = new CompoundPath(Item.NO_INSERT); item = new CompoundPath(Item.NO_INSERT);
item.addChildren(paths, true); // Preserve orientation item.addChildren(paths);
item = item.reduce(); item = item.reduce();
item.copyAttributes(this); item.copyAttributes(this);
this.replaceWith(item); this.replaceWith(item);
@ -1015,49 +1014,44 @@ PathItem.inject(new function() {
/** /**
* Fixes the orientation of the sub-paths of a compound-path, assuming * Fixes the orientation of the sub-paths of a compound-path, assuming
* that non of its sub-paths intersect, by reorienting sub-paths so that * that non of its sub-paths intersect, by reorienting them so that they
* they are of different winding direction than their containing path, * are of different winding direction than their containing paths,
* except for disjoint sub-paths, i.e. islands, which are reoriented so * except for disjoint sub-paths, i.e. islands, which are oriented so
* that they have the same winding direction as the the biggest path. * that they have the same winding direction as the the biggest path.
* *
* Additionally, if the compound-path has the `'nonzero'` * @param {Boolean} [nonZero=false] controls if the non-zero fill-rule
* {@link #getFillRule()}, the winding of each nested path is counted, * is to be applied, by counting the winding of each nested path and
* and sub-paths that do not contribute to the final result are * discarding sub-paths that do not contribute to the final result
* discarded.
*
* @param {Boolean} [sort=false] controls if the sub-paths should be
* sorted according to their area from largest to smallest, or if
* normal sequence should be preserved
* @return {PahtItem} a reference to the item itself, reoriented * @return {PahtItem} a reference to the item itself, reoriented
* @see #getFillRule()
*/ */
reorient: function(sort) { reorient: function(nonZero) {
var children = this._children, var children = this._children,
length = children && children.length; length = children && children.length;
if (length > 1) { if (length > 1) {
children = this.removeChildren(); // Build a lookup table with information for each path's
// First order the paths by their areas. // original index and winding contribution.
var sorted = children.slice().sort(function (a, b) { var lookup = Base.each(children, function(path, i) {
return abs(b.getArea()) - abs(a.getArea());
}),
first = sorted[0],
paths = [first],
isNonZero = this.getFillRule() === 'nonzero',
// We only need to build a lookup table with information for
// each path if we process with non-zero fill-rule, or if we
// are to preserve the original sequence in the result.
lookup = (isNonZero || !sort) && Base.each(children,
function(path, i) {
this[path._id] = { this[path._id] = {
winding: path.isClockwise() ? 1 : -1, winding: path.isClockwise() ? 1 : -1,
index: i index: i
}; };
}, {}); }, {}),
// Walk through sorted paths, from largest to smallest. // Now sort the paths by their areas, from large to small.
// The first, largest path can be skipped. sorted = this.removeChildren().sort(function (a, b) {
return abs(b.getArea()) - abs(a.getArea());
}),
// Get reference to the first, largest path and insert it
// already.
first = sorted[0],
paths = [];
// Always insert paths at their original index. With exclusion,
// this produces null entries, but #setChildren() handles those.
paths[lookup[first._id].index] = first;
// Walk through the sorted paths, from largest to smallest.
// Skip the first path, as it is already added.
for (var i1 = 1; i1 < length; i1++) { for (var i1 = 1; i1 < length; i1++) {
var path1 = sorted[i1], var path1 = sorted[i1],
entry1 = lookup && lookup[path1._id], entry1 = lookup[path1._id],
point = path1.getInteriorPoint(), point = path1.getInteriorPoint(),
isContained = false, isContained = false,
container = null, container = null,
@ -1073,8 +1067,8 @@ PathItem.inject(new function() {
// contains the current path, and then set the // contains the current path, and then set the
// orientation to the opposite of the containing path. // orientation to the opposite of the containing path.
if (path2.contains(point)) { if (path2.contains(point)) {
var entry2 = lookup && lookup[path2._id]; var entry2 = lookup[path2._id];
if (isNonZero && !isContained) { if (nonZero && !isContained) {
entry1.winding += entry2.winding; entry1.winding += entry2.winding;
// Remove path if rule is nonzero and winding // Remove path if rule is nonzero and winding
// of path and containing path is not zero. // of path and containing path is not zero.
@ -1086,7 +1080,7 @@ PathItem.inject(new function() {
isContained = true; isContained = true;
// If the containing path is not excluded, we're // If the containing path is not excluded, we're
// done searching for the orientation defining path. // done searching for the orientation defining path.
container = !(entry2 && entry2.exclude) && path2; container = !entry2.exclude && path2;
} }
} }
if (!exclude) { if (!exclude) {
@ -1096,18 +1090,10 @@ PathItem.inject(new function() {
path1.setClockwise(container path1.setClockwise(container
? !container.isClockwise() ? !container.isClockwise()
: first.isClockwise()); : first.isClockwise());
if (!sort) {
// If asked to preserve sequence (not sort children
// according to their area), insert back at their
// original index. With exclusion this produces null
// entries, but #setChildren() can handle those.
paths[entry1.index] = path1; paths[entry1.index] = path1;
} else {
paths.push(path1);
} }
} }
} this.setChildren(paths);
this.setChildren(paths, true); // Preserve orientation
} }
return this; return this;
}, },

View file

@ -90,6 +90,24 @@ var PathItem = Item.extend(/** @lends PathItem# */{
return this; return this;
}, },
/**
* Specifies whether the path is oriented clock-wise.
*
* @bean
* @type Boolean
*/
isClockwise: function() {
return this.getArea() >= 0;
},
setClockwise: function(clockwise) {
// Only revers the path if its clockwise orientation is not the same
// as what it is now demanded to be.
// On-the-fly conversion to boolean:
if (this.isClockwise() != (clockwise = !!clockwise))
this.reverse();
},
/** /**
* The path's geometry, formatted as SVG style path data. * The path's geometry, formatted as SVG style path data.
* *
@ -97,7 +115,6 @@ var PathItem = Item.extend(/** @lends PathItem# */{
* @bean * @bean
* @type String * @type String
*/ */
setPathData: function(data) { setPathData: function(data) {
// NOTE: #getPathData() is defined in CompoundPath / Path // NOTE: #getPathData() is defined in CompoundPath / Path
// This is a very compact SVG Path Data parser that works both for Path // This is a very compact SVG Path Data parser that works both for Path

View file

@ -34,10 +34,10 @@ test('moveTo() / lineTo()', function() {
}, 2); }, 2);
}); });
test('CompoundPath() and default clockwise settings', function() { test('CompoundPath#reorient()', function() {
var path1 = new Path.Rectangle([200, 200], [100, 100]); var path1 = new Path.Rectangle([300, 300], [100, 100]);
var path2 = new Path.Rectangle([50, 50], [200, 200]); var path2 = new Path.Rectangle([50, 50], [200, 200]);
var path3 = new Path.Rectangle([0, 0], [400, 400]); var path3 = new Path.Rectangle([0, 0], [500, 500]);
equals(function() { equals(function() {
return path1.clockwise; return path1.clockwise;
@ -49,7 +49,9 @@ test('CompoundPath() and default clockwise settings', function() {
return path3.clockwise; return path3.clockwise;
}, true); }, true);
var compound = new CompoundPath(path1, path2, path3); var compound = new CompoundPath({
children: [path1, path2, path3],
}).reorient();
equals(function() { equals(function() {
return compound.lastChild == path3; return compound.lastChild == path3;
@ -59,29 +61,10 @@ test('CompoundPath() and default clockwise settings', function() {
}, true); }, true);
equals(function() { equals(function() {
return path1.clockwise; return path1.clockwise;
}, true); }, false);
equals(function() { equals(function() {
return path2.clockwise; return path2.clockwise;
}, false); }, false);
equals(function() {
return path3.clockwise;
}, false);
});
test('CompoundPath() and predefined clockwise settings', function() {
var path1 = new Path.Rectangle([200, 200], [100, 100]);
var path2 = new Path.Rectangle([50, 50], [200, 200]);
var path3 = new Path.Rectangle([0, 0], [400, 400]);
path1.clockwise = false;
path2.clockwise = true;
path3.clockwise = true;
var compound = new CompoundPath(path1, path2, path3);
equals(function() {
return path1.clockwise;
}, false);
equals(function() {
return path2.clockwise;
}, true);
equals(function() { equals(function() {
return path3.clockwise; return path3.clockwise;
}, true); }, true);

View file

@ -696,7 +696,8 @@ test('hit-testing compound-paths', function() {
}); });
var compoundPath = new CompoundPath({ var compoundPath = new CompoundPath({
children: [path1, path2], children: [path1, path2],
fillColor: 'blue' fillColor: 'blue',
fillRule: 'evenodd'
}); });
// When hit-testing a side, we should get a result on the torus // When hit-testing a side, we should get a result on the torus
equals(function() { equals(function() {

View file

@ -96,40 +96,50 @@ test('CompoundPath#contains() (donut)', function() {
new Path.Circle([0, 0], 25) new Path.Circle([0, 0], 25)
]); ]);
testPoint(path, new Point(0, -50), true, function testDonut(path, title) {
title = 'fillRule = ' + title + ': ';
testPoint(path, new Point(0, -50), true, title +
'The top center point of the outer circle should be inside the donut.'); 'The top center point of the outer circle should be inside the donut.');
testPoint(path, new Point(0, 0), false, testPoint(path, new Point(0, 0), false, title +
'The center point should be outside the donut.'); 'The center point should be outside the donut.');
testPoint(path, new Point(-35, 0), true, testPoint(path, new Point(-35, 0), true, title +
'A vertically centered point on the left side should be inside the donut.'); 'A vertically centered point on the left side should be inside the donut.');
testPoint(path, new Point(35, 0), true, testPoint(path, new Point(35, 0), true, title +
'A vertically centered point on the right side should be inside the donut.'); 'A vertically centered point on the right side should be inside the donut.');
testPoint(path, new Point(0, 49), true, testPoint(path, new Point(0, 49), true, title +
'The near bottom center point of the outer circle should be inside the donut.'); 'The near bottom center point of the outer circle should be inside the donut.');
testPoint(path, new Point(0, 50), true, testPoint(path, new Point(0, 50), true, title +
'The bottom center point of the outer circle should be inside the donut.'); 'The bottom center point of the outer circle should be inside the donut.');
testPoint(path, new Point(0, 51), false, testPoint(path, new Point(0, 51), false, title +
'The near bottom center point of the outer circle should be outside the donut.'); 'The near bottom center point of the outer circle should be outside the donut.');
testPoint(path, new Point({ length: 50, angle: 30 }), true, testPoint(path, new Point({ length: 50, angle: 30 }), true, title +
'A random point on the periphery of the outer circle should be inside the donut.'); 'A random point on the periphery of the outer circle should be inside the donut.');
testPoint(path, new Point(-25, 0), true, testPoint(path, new Point(-25, 0), true, title +
'The left center point of the inner circle should be inside the donut.'); 'The left center point of the inner circle should be inside the donut.');
testPoint(path, new Point(0, -25), true, testPoint(path, new Point(0, -25), true, title +
'The top center point of the inner circle should be inside the donut.'); 'The top center point of the inner circle should be inside the donut.');
testPoint(path, new Point(25, 0), true, testPoint(path, new Point(25, 0), true, title +
'The right center point of the inner circle should be inside the donut.'); 'The right center point of the inner circle should be inside the donut.');
testPoint(path, new Point(0, 25), true, testPoint(path, new Point(0, 25), true, title +
'The bottom center point of the inner circle should be inside the donut.'); 'The bottom center point of the inner circle should be inside the donut.');
testPoint(path, new Point(-50, -50), false, testPoint(path, new Point(-50, -50), false, title +
'The top left point of bounding box should be outside the donut.'); 'The top left point of bounding box should be outside the donut.');
testPoint(path, new Point(50, -50), false, testPoint(path, new Point(50, -50), false, title +
'The top right point of the bounding box should be outside the donut.'); 'The top right point of the bounding box should be outside the donut.');
testPoint(path, new Point(-50, 50), false, testPoint(path, new Point(-50, 50), false, title +
'The bottom left point of bounding box should be outside the donut.'); 'The bottom left point of bounding box should be outside the donut.');
testPoint(path, new Point(50, 50), false, testPoint(path, new Point(50, 50), false, title +
'The bottom right point of the bounding box should be outside the donut.'); 'The bottom right point of the bounding box should be outside the donut.');
testPoint(path, new Point(-45, 45), false, testPoint(path, new Point(-45, 45), false, title +
'The near bottom left point of bounding box should be outside the donut.'); 'The near bottom left point of bounding box should be outside the donut.');
}
path.fillRule = 'evenodd';
// testDonut(path, '\'evenodd\'');
path.reorient();
testDonut(path, '\'evenodd\' + reorient()');
path.fillRule = 'nonzero';
testDonut(path, '\'nonzero\' + reorient()');
}); });
test('Shape#contains()', function() { test('Shape#contains()', function() {

View file

@ -350,7 +350,7 @@ test('#870', function() {
}); });
test('#875', function() { test('#875', function() {
var cp = new Path({ var p1 = new Path({
segments: [ segments: [
[158.7, 389.3, 0, 0, -4.95, 4.95], [158.7, 389.3, 0, 0, -4.95, 4.95],
[158.7, 407.2, -4.95, -4.95, 4.95, 4.95], [158.7, 407.2, -4.95, -4.95, 4.95, 4.95],
@ -360,14 +360,15 @@ test('#875', function() {
], ],
closed: true closed: true
}); });
var p = new Path.Circle(260, 320, 100); var p2 = new Path.Circle(260, 320, 100);
compareBoolean(function() { return cp.subtract(p); }, compareBoolean(function() { return p1.subtract(p2); },
'M158.7,407.2c4.95,4.95 12.95,4.95 17.9,0c4.95,-4.95 4.95,-12.95 0,-17.9c-4.95,-4.95 -12.95,-4.95 -17.9,0c-4.95,4.95 -4.95,12.95 0,17.9z'); 'M158.7,407.2c4.95,4.95 12.95,4.95 17.9,0c4.95,-4.95 4.95,-12.95 0,-17.9c-4.95,-4.95 -12.95,-4.95 -17.9,0c-4.95,4.95 -4.95,12.95 0,17.9z');
}); });
test('#877', function() { test('#877', function() {
var cp = new CompoundPath([ var cp = new CompoundPath({
children: [
new Path.Circle(100, 60, 50), new Path.Circle(100, 60, 50),
new Path.Circle(100, 60, 30), new Path.Circle(100, 60, 30),
new Path({ new Path({
@ -378,8 +379,10 @@ test('#877', function() {
[120, 190] [120, 190]
], ],
closed: true closed: true
}) }),
]); ],
fillRule: 'evenodd'
});
var p = new Path({ var p = new Path({
segments: [ segments: [