mirror of
https://github.com/scratchfoundation/paper.js.git
synced 2025-01-03 19:45:44 -05:00
Introduce new Item#copyAttributes() & #copyContent(), and revamp #clone() handling.
This commit is contained in:
parent
463f50a0c1
commit
2fea40f86f
13 changed files with 119 additions and 113 deletions
130
src/item/Item.js
130
src/item/Item.js
|
@ -309,7 +309,7 @@ var Item = Base.extend(Emitter, /** @lends Item# */{
|
||||||
return this._name;
|
return this._name;
|
||||||
},
|
},
|
||||||
|
|
||||||
setName: function(name, unique) {
|
setName: function(name) {
|
||||||
// Note: Don't check if the name has changed and bail out if it has not,
|
// Note: Don't check if the name has changed and bail out if it has not,
|
||||||
// because setName is used internally also to update internal structures
|
// because setName is used internally also to update internal structures
|
||||||
// when an item is moved from one parent to another.
|
// when an item is moved from one parent to another.
|
||||||
|
@ -326,12 +326,7 @@ var Item = Base.extend(Emitter, /** @lends Item# */{
|
||||||
var parent = this._parent;
|
var parent = this._parent;
|
||||||
if (name && parent) {
|
if (name && parent) {
|
||||||
var children = parent._children,
|
var children = parent._children,
|
||||||
namedChildren = parent._namedChildren,
|
namedChildren = parent._namedChildren;
|
||||||
orig = name,
|
|
||||||
i = 1;
|
|
||||||
// If unique is true, make sure we're not overriding other names
|
|
||||||
while (unique && children[name])
|
|
||||||
name = orig + ' ' + (i++);
|
|
||||||
(namedChildren[name] = namedChildren[name] || []).push(this);
|
(namedChildren[name] = namedChildren[name] || []).push(this);
|
||||||
children[name] = this;
|
children[name] = this;
|
||||||
}
|
}
|
||||||
|
@ -1437,61 +1432,82 @@ var Item = Base.extend(Emitter, /** @lends Item# */{
|
||||||
* }
|
* }
|
||||||
*/
|
*/
|
||||||
clone: function(insert) {
|
clone: function(insert) {
|
||||||
return this._clone(new this.constructor(Item.NO_INSERT), insert);
|
var copy = new this.constructor(Item.NO_INSERT);
|
||||||
},
|
copy.copyAttributes(this);
|
||||||
|
copy.copyContent(this);
|
||||||
/**
|
|
||||||
* Clones the item within the same project and places the copy above the
|
|
||||||
* item.
|
|
||||||
*
|
|
||||||
* @param {Boolean} [insert=true] specifies whether the copy should be
|
|
||||||
* inserted into the DOM. When set to {@code true}, it is inserted above the
|
|
||||||
* original
|
|
||||||
* @return {Item} the newly cloned item
|
|
||||||
*/
|
|
||||||
_clone: function(copy, insert, includeMatrix) {
|
|
||||||
var keys = ['_locked', '_visible', '_blendMode', '_opacity',
|
|
||||||
'_clipMask', '_guide'],
|
|
||||||
children = this._children;
|
|
||||||
// Copy over style
|
|
||||||
copy.setStyle(this._style);
|
|
||||||
// Clone all children and add them to the copy. tell #addChild we're
|
|
||||||
// cloning, as needed by CompoundPath#insertChild().
|
|
||||||
for (var i = 0, l = children && children.length; i < l; i++) {
|
|
||||||
copy.addChild(children[i].clone(false), true);
|
|
||||||
}
|
|
||||||
// Only copy over these fields if they are actually defined in 'this',
|
|
||||||
// meaning the default value has been overwritten (default is on
|
|
||||||
// prototype).
|
|
||||||
for (var i = 0, l = keys.length; i < l; i++) {
|
|
||||||
var key = keys[i];
|
|
||||||
if (this.hasOwnProperty(key))
|
|
||||||
copy[key] = this[key];
|
|
||||||
}
|
|
||||||
// Use Matrix#initialize to easily copy over values.
|
|
||||||
if (includeMatrix !== false)
|
|
||||||
copy._matrix.initialize(this._matrix);
|
|
||||||
// In case of Path#toShape(), we can't just set _applyMatrix as
|
|
||||||
// Shape won't allow it. Using the setter instead takes care of it.
|
|
||||||
// NOTE: This will also bake in the matrix that we just initialized,
|
|
||||||
// in case #applyMatrix is true.
|
|
||||||
copy.setApplyMatrix(this._applyMatrix);
|
|
||||||
copy.setPivot(this._pivot);
|
|
||||||
// Copy over the selection state, use setSelected so the item
|
|
||||||
// is also added to Project#selectedItems if it is selected.
|
|
||||||
copy.setSelected(this._selected);
|
|
||||||
// Copy over _data as well.
|
|
||||||
copy._data = this._data ? Base.clone(this._data) : null;
|
|
||||||
// Insert is true by default.
|
// Insert is true by default.
|
||||||
if (insert || insert === undefined)
|
if (insert || insert === undefined)
|
||||||
copy.insertAbove(this);
|
copy.insertAbove(this);
|
||||||
// Clone the name too, but make sure we're not overriding the original
|
// Make sure we're not overriding the original name in the same parent
|
||||||
// in the same parent, by passing true for the unique parameter.
|
var name = this._name,
|
||||||
if (this._name)
|
parent = this._parent;
|
||||||
copy.setName(this._name, true);
|
if (name && parent) {
|
||||||
|
var children = parent._children,
|
||||||
|
orig = name,
|
||||||
|
i = 1;
|
||||||
|
while (children[name])
|
||||||
|
name = orig + ' ' + (i++);
|
||||||
|
if (name !== orig)
|
||||||
|
copy.setName(name);
|
||||||
|
}
|
||||||
return copy;
|
return copy;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Copies the content of the specified item over to this item.
|
||||||
|
*
|
||||||
|
* @param {Item} source the item to copy the content from
|
||||||
|
*/
|
||||||
|
copyContent: function(source) {
|
||||||
|
var children = source._children;
|
||||||
|
// Clone all children and add them to the copy. tell #addChild we're
|
||||||
|
// cloning, as needed by CompoundPath#insertChild().
|
||||||
|
for (var i = 0, l = children && children.length; i < l; i++) {
|
||||||
|
this.addChild(children[i].clone(false), true);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Copies all attributes of the specified item over to this item. This
|
||||||
|
* includes its style, visibility, matrix, pivot, blend-mode, opacity,
|
||||||
|
* selection state, data, name, etc.
|
||||||
|
*
|
||||||
|
* @param {Item} source the item to copy the attributes from
|
||||||
|
*/
|
||||||
|
copyAttributes: function(source, _preConcatenate) {
|
||||||
|
// Copy over style
|
||||||
|
this.setStyle(source._style);
|
||||||
|
// Only copy over these fields if they are actually defined in 'source',
|
||||||
|
// meaning the default value has been overwritten (default is on
|
||||||
|
// prototype).
|
||||||
|
var keys = ['_locked', '_visible', '_blendMode', '_opacity',
|
||||||
|
'_clipMask', '_guide'];
|
||||||
|
for (var i = 0, l = keys.length; i < l; i++) {
|
||||||
|
var key = keys[i];
|
||||||
|
if (source.hasOwnProperty(key))
|
||||||
|
this[key] = source[key];
|
||||||
|
}
|
||||||
|
// Use Matrix#initialize to easily copy over values.
|
||||||
|
this._matrix[_preConcatenate ? 'preConcatenate' : 'initialize'](
|
||||||
|
source._matrix);
|
||||||
|
// We can't just set _applyMatrix as many item types won't allow it,
|
||||||
|
// e.g. creating a Shape in Path#toShape().
|
||||||
|
// Using the setter instead takes care of it.
|
||||||
|
// NOTE: This will also bake in the matrix that we just initialized,
|
||||||
|
// in case #applyMatrix is true.
|
||||||
|
this.setApplyMatrix(source._applyMatrix);
|
||||||
|
this.setPivot(source._pivot);
|
||||||
|
// Copy over the selection state, use setSelected so the item
|
||||||
|
// is also added to Project#selectedItems if it is selected.
|
||||||
|
this.setSelected(source._selected);
|
||||||
|
// Copy over data and name as well.
|
||||||
|
var data = source._data,
|
||||||
|
name = source._name;
|
||||||
|
this._data = data ? Base.clone(data) : null;
|
||||||
|
if (name)
|
||||||
|
this.setName(name);
|
||||||
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* When passed a project, copies the item to the project,
|
* When passed a project, copies the item to the project,
|
||||||
* or duplicates it within the same project. When passed an item,
|
* or duplicates it within the same project. When passed an item,
|
||||||
|
@ -2299,7 +2315,7 @@ var Item = Base.extend(Emitter, /** @lends Item# */{
|
||||||
if (this._children && this._children.length === 1) {
|
if (this._children && this._children.length === 1) {
|
||||||
var child = this._children[0].reduce();
|
var child = this._children[0].reduce();
|
||||||
child.insertAbove(this);
|
child.insertAbove(this);
|
||||||
child.setStyle(this._style);
|
child.copyAttributes(this);
|
||||||
this.remove();
|
this.remove();
|
||||||
return child;
|
return child;
|
||||||
}
|
}
|
||||||
|
|
|
@ -84,6 +84,10 @@ var PlacedSymbol = Item.extend(/** @lends PlacedSymbol# */{
|
||||||
return this._symbol === item._symbol;
|
return this._symbol === item._symbol;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
copyContent: function(source) {
|
||||||
|
this.setSymbol(source._symbol);
|
||||||
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The symbol that the placed symbol refers to.
|
* The symbol that the placed symbol refers to.
|
||||||
*
|
*
|
||||||
|
@ -99,16 +103,11 @@ var PlacedSymbol = Item.extend(/** @lends PlacedSymbol# */{
|
||||||
this._changed(/*#=*/Change.GEOMETRY);
|
this._changed(/*#=*/Change.GEOMETRY);
|
||||||
},
|
},
|
||||||
|
|
||||||
clone: function(insert) {
|
|
||||||
var copy = new PlacedSymbol(Item.NO_INSERT);
|
|
||||||
copy.setSymbol(this._symbol);
|
|
||||||
return this._clone(copy, insert);
|
|
||||||
},
|
|
||||||
|
|
||||||
isEmpty: function() {
|
isEmpty: function() {
|
||||||
return this._symbol._definition.isEmpty();
|
return this._symbol._definition.isEmpty();
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|
||||||
_getBounds: function(getter, matrix, cacheItem) {
|
_getBounds: function(getter, matrix, cacheItem) {
|
||||||
var definition = this.symbol._definition;
|
var definition = this.symbol._definition;
|
||||||
// Redirect the call to the symbol definition to calculate the bounds
|
// Redirect the call to the symbol definition to calculate the bounds
|
||||||
|
|
|
@ -101,21 +101,20 @@ var Raster = Item.extend(/** @lends Raster# */{
|
||||||
return this.getSource() === item.getSource();
|
return this.getSource() === item.getSource();
|
||||||
},
|
},
|
||||||
|
|
||||||
clone: function(insert) {
|
copyContent: function(source) {
|
||||||
var copy = new Raster(Item.NO_INSERT),
|
var image = source._image,
|
||||||
image = this._image,
|
canvas = source._canvas;
|
||||||
canvas = this._canvas;
|
|
||||||
if (image) {
|
if (image) {
|
||||||
copy.setImage(image);
|
this.setImage(image);
|
||||||
} else if (canvas) {
|
} else if (canvas) {
|
||||||
// If the Raster contains a Canvas object, we need to create a new
|
// If the Raster contains a Canvas object, we need to create a new
|
||||||
// one and draw this raster's canvas on it.
|
// one and draw this raster's canvas on it.
|
||||||
var copyCanvas = CanvasProvider.getCanvas(this._size);
|
var copyCanvas = CanvasProvider.getCanvas(source._size);
|
||||||
copyCanvas.getContext('2d').drawImage(canvas, 0, 0);
|
copyCanvas.getContext('2d').drawImage(canvas, 0, 0);
|
||||||
copy.setImage(copyCanvas);
|
this.setImage(copyCanvas);
|
||||||
}
|
}
|
||||||
copy._crossOrigin = this._crossOrigin;
|
// TODO: Shouldn't this be copied with attributes instead of content?
|
||||||
return this._clone(copy, insert);
|
this._crossOrigin = source._crossOrigin;
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -39,12 +39,10 @@ var Shape = Item.extend(/** @lends Shape# */{
|
||||||
&& Base.equals(this._radius, item._radius);
|
&& Base.equals(this._radius, item._radius);
|
||||||
},
|
},
|
||||||
|
|
||||||
clone: function(insert) {
|
copyContent: function(source) {
|
||||||
var copy = new Shape(Item.NO_INSERT);
|
this.setType(source._type);
|
||||||
copy.setType(this._type);
|
this.setSize(source._size);
|
||||||
copy.setSize(this._size);
|
this.setRadius(source._radius);
|
||||||
copy.setRadius(this._radius);
|
|
||||||
return this._clone(copy, insert);
|
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -169,17 +167,20 @@ var Shape = Item.extend(/** @lends Shape# */{
|
||||||
* @see Path#toShape(insert)
|
* @see Path#toShape(insert)
|
||||||
*/
|
*/
|
||||||
toPath: function(insert) {
|
toPath: function(insert) {
|
||||||
var path = this._clone(new Path[Base.capitalize(this._type)]({
|
// TODO: Move to Path.createTYPE creators instead of fake constructors.
|
||||||
|
var path = new Path[Base.capitalize(this._type)]({
|
||||||
center: new Point(),
|
center: new Point(),
|
||||||
size: this._size,
|
size: this._size,
|
||||||
radius: this._radius,
|
radius: this._radius,
|
||||||
insert: false
|
insert: false
|
||||||
}), insert);
|
});
|
||||||
|
path.copyAttributes(this);
|
||||||
// The created path will inherit #applyMatrix from this Shape, hence it
|
// The created path will inherit #applyMatrix from this Shape, hence it
|
||||||
// will always be false.
|
// will always be false.
|
||||||
// Respect the setting of paper.settings.applyMatrix for new paths:
|
// Respect the setting of paper.settings.applyMatrix for new paths:
|
||||||
if (paper.settings.applyMatrix)
|
if (paper.settings.applyMatrix)
|
||||||
path.setApplyMatrix(true);
|
path.setApplyMatrix(true);
|
||||||
|
path.insertAbove(this);
|
||||||
return path;
|
return path;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|
|
@ -152,10 +152,8 @@ var CompoundPath = PathItem.extend(/** @lends CompoundPath# */{
|
||||||
}
|
}
|
||||||
if (children.length === 0) { // Replace with a simple empty Path
|
if (children.length === 0) { // Replace with a simple empty Path
|
||||||
var path = new Path(Item.NO_INSERT);
|
var path = new Path(Item.NO_INSERT);
|
||||||
|
path.copyAttributes(this);
|
||||||
path.insertAbove(this);
|
path.insertAbove(this);
|
||||||
// TODO: Consider using Item#_clone() for this, but find a way to
|
|
||||||
// not clone children / name (content).
|
|
||||||
path.setStyle(this._style);
|
|
||||||
this.remove();
|
this.remove();
|
||||||
return path;
|
return path;
|
||||||
}
|
}
|
||||||
|
|
|
@ -131,13 +131,12 @@ var Path = PathItem.extend(/** @lends Path# */{
|
||||||
&& Base.equals(this._segments, item._segments);
|
&& Base.equals(this._segments, item._segments);
|
||||||
},
|
},
|
||||||
|
|
||||||
clone: function(insert) {
|
copyContent: function(source) {
|
||||||
var copy = new Path(Item.NO_INSERT);
|
this.setSegments(source._segments);
|
||||||
copy.setSegments(this._segments);
|
this._closed = source._closed;
|
||||||
copy._closed = this._closed;
|
var clockwise = source._clockwise;
|
||||||
if (this._clockwise !== undefined)
|
if (clockwise !== undefined)
|
||||||
copy._clockwise = this._clockwise;
|
this._clockwise = clockwise;
|
||||||
return this._clone(copy, insert);
|
|
||||||
},
|
},
|
||||||
|
|
||||||
_changed: function _changed(flags) {
|
_changed: function _changed(flags) {
|
||||||
|
@ -1247,8 +1246,7 @@ var Path = PathItem.extend(/** @lends Path# */{
|
||||||
// Pass true for _preserve, in case of CompoundPath, to avoid
|
// Pass true for _preserve, in case of CompoundPath, to avoid
|
||||||
// reversing of path direction, which would mess with segments!
|
// reversing of path direction, which would mess with segments!
|
||||||
path.insertAbove(this, true);
|
path.insertAbove(this, true);
|
||||||
// Use _clone to copy over all other attributes, including style
|
path.copyAttributes(this);
|
||||||
this._clone(path);
|
|
||||||
}
|
}
|
||||||
path._add(segs, 0);
|
path._add(segs, 0);
|
||||||
// Add dividing segment again. In case of a closed path, that's the
|
// Add dividing segment again. In case of a closed path, that's the
|
||||||
|
@ -1496,14 +1494,17 @@ var Path = PathItem.extend(/** @lends Path# */{
|
||||||
|
|
||||||
if (type) {
|
if (type) {
|
||||||
var center = this.getPosition(true),
|
var center = this.getPosition(true),
|
||||||
shape = this._clone(new type({
|
shape = new type({
|
||||||
center: center,
|
center: center,
|
||||||
size: size,
|
size: size,
|
||||||
radius: radius,
|
radius: radius,
|
||||||
insert: false
|
insert: false
|
||||||
}), insert, false);
|
});
|
||||||
|
// Pass `true` to preconcatenate the matrix to the center-transform.
|
||||||
|
shape.copyAttributes(this, true);
|
||||||
// Determine and apply the shape's angle of rotation.
|
// Determine and apply the shape's angle of rotation.
|
||||||
shape.rotate(topCenter.subtract(center).getAngle() + 90);
|
shape.rotate(topCenter.subtract(center).getAngle() + 90);
|
||||||
|
shape.insertAbove(this);
|
||||||
return shape;
|
return shape;
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
|
|
|
@ -67,9 +67,7 @@ PathItem.inject(new function() {
|
||||||
&& path1.getIndex() < path2.getIndex()
|
&& path1.getIndex() < path2.getIndex()
|
||||||
? path2 : path1);
|
? path2 : path1);
|
||||||
// Copy over the left-hand item's style and we're done.
|
// Copy over the left-hand item's style and we're done.
|
||||||
// TODO: Consider using Item#_clone() for this, but find a way to not
|
result.copyAttributes(path1);
|
||||||
// clone children / name (content).
|
|
||||||
result.setStyle(path1._style);
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -861,9 +859,7 @@ PathItem.inject(new function() {
|
||||||
item = new CompoundPath(Item.NO_INSERT);
|
item = new CompoundPath(Item.NO_INSERT);
|
||||||
item.setChildren(paths);
|
item.setChildren(paths);
|
||||||
item = item.reduce();
|
item = item.reduce();
|
||||||
// TODO: Consider using Item#_clone() for this, but find a way to
|
item.copyAttributes(this);
|
||||||
// not clone children / name (content).
|
|
||||||
item.setStyle(this._style);
|
|
||||||
this.replaceWith(item);
|
this.replaceWith(item);
|
||||||
}
|
}
|
||||||
return item;
|
return item;
|
||||||
|
|
|
@ -58,10 +58,6 @@ var PointText = TextItem.extend(/** @lends PointText# */{
|
||||||
TextItem.apply(this, arguments);
|
TextItem.apply(this, arguments);
|
||||||
},
|
},
|
||||||
|
|
||||||
clone: function(insert) {
|
|
||||||
return this._clone(new PointText(Item.NO_INSERT), insert);
|
|
||||||
},
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The PointText's anchor point
|
* The PointText's anchor point
|
||||||
*
|
*
|
||||||
|
|
|
@ -50,9 +50,8 @@ var TextItem = Item.extend(/** @lends TextItem# */{
|
||||||
return this._content === item._content;
|
return this._content === item._content;
|
||||||
},
|
},
|
||||||
|
|
||||||
_clone: function _clone(copy, insert, includeMatrix) {
|
copyContent: function(source) {
|
||||||
copy.setContent(this._content);
|
this.setContent(source._content);
|
||||||
return _clone.base.call(this, copy, insert, includeMatrix);
|
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -39,9 +39,10 @@ function compareItem(actual, expected, message, options, properties) {
|
||||||
QUnit.notStrictEqual(actual.id, 'not ' + expected.id, message + '.id');
|
QUnit.notStrictEqual(actual.id, 'not ' + expected.id, message + '.id');
|
||||||
QUnit.strictEqual(actual.constructor, expected.constructor,
|
QUnit.strictEqual(actual.constructor, expected.constructor,
|
||||||
message + '.constructor');
|
message + '.constructor');
|
||||||
|
equals(actual.name,
|
||||||
// When item was cloned and had a name, the name will be versioned
|
// When item was cloned and had a name, the name will be versioned
|
||||||
equals(options && options.cloned && actual.name ? actual.name + ' 1'
|
options && options.cloned && expected.name
|
||||||
: actual.name, expected.name,
|
? expected.name + ' 1' : expected.name,
|
||||||
message + '.name');
|
message + '.name');
|
||||||
compareProperties(actual, expected, ['children', 'bounds', 'position',
|
compareProperties(actual, expected, ['children', 'bounds', 'position',
|
||||||
'matrix', 'data', 'opacity', 'locked', 'visible', 'blendMode',
|
'matrix', 'data', 'opacity', 'locked', 'visible', 'blendMode',
|
||||||
|
|
|
@ -24,7 +24,7 @@ function cloneAndCompare(item) {
|
||||||
return copy.parent.children[copy.name] == copy;
|
return copy.parent.children[copy.name] == copy;
|
||||||
}, true);
|
}, true);
|
||||||
}
|
}
|
||||||
equals(item, copy, 'item.clone()', { cloned: true });
|
equals(copy, item, 'item.clone()', { cloned: true });
|
||||||
// Remove the cloned item to restore the document:
|
// Remove the cloned item to restore the document:
|
||||||
copy.remove();
|
copy.remove();
|
||||||
}
|
}
|
||||||
|
|
|
@ -117,5 +117,5 @@ test('Import SVG polyline', function() {
|
||||||
test('Import complex CompoundPath and clone', function() {
|
test('Import complex CompoundPath and clone', function() {
|
||||||
var svg = createSVG('<path id="path" fill="red" d="M4,14h20v-2H4V14z M15,26h7v-2h-7V26z M15,22h9v-2h-9V22z M15,18h9v-2h-9V18z M4,26h9V16H4V26z M28,10V6H0v22c0,0,0,4,4,4 h25c0,0,3-0.062,3-4V10H28z M4,30c-2,0-2-2-2-2V8h24v20c0,0.921,0.284,1.558,0.676,2H4z"/>;');
|
var svg = createSVG('<path id="path" fill="red" d="M4,14h20v-2H4V14z M15,26h7v-2h-7V26z M15,22h9v-2h-9V22z M15,18h9v-2h-9V18z M4,26h9V16H4V26z M28,10V6H0v22c0,0,0,4,4,4 h25c0,0,3-0.062,3-4V10H28z M4,30c-2,0-2-2-2-2V8h24v20c0,0.921,0.284,1.558,0.676,2H4z"/>;');
|
||||||
var item = paper.project.importSVG(svg.getElementById('path'));
|
var item = paper.project.importSVG(svg.getElementById('path'));
|
||||||
equals(item, item.clone(), null, { cloned: true });
|
equals(item.clone(), item, null, { cloned: true });
|
||||||
});
|
});
|
||||||
|
|
|
@ -50,6 +50,6 @@ test('shape.toPath().toShape()', function() {
|
||||||
};
|
};
|
||||||
|
|
||||||
Base.each(shapes, function(shape, name) {
|
Base.each(shapes, function(shape, name) {
|
||||||
equals(shape, shape.toPath().toShape(), name + '.toPath().toShape()');
|
equals(shape.toPath().toShape(), shape, name + '.toPath().toShape()');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
Loading…
Reference in a new issue