mirror of
https://github.com/scratchfoundation/paper.js.git
synced 2025-01-22 07:19:57 -05:00
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:
parent
fc72c05e69
commit
a0417040f8
10 changed files with 151 additions and 217 deletions
|
@ -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);
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -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();
|
||||||
|
|
|
@ -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.
|
||||||
|
|
|
@ -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);
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|
|
@ -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;
|
||||||
},
|
},
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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);
|
||||||
|
|
|
@ -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() {
|
||||||
|
|
|
@ -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() {
|
||||||
|
|
|
@ -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: [
|
||||||
|
|
Loading…
Reference in a new issue