mirror of
https://github.com/scratchfoundation/paper.js.git
synced 2025-01-03 19:45:44 -05:00
Implement unit test comparators for many more types, and start refactoring compareItems() helper.
This commit is contained in:
parent
c5a2a51f3e
commit
9adbc3774a
4 changed files with 91 additions and 188 deletions
|
@ -41,6 +41,7 @@
|
||||||
* circle.fillColor = '#ff0000';
|
* circle.fillColor = '#ff0000';
|
||||||
*/
|
*/
|
||||||
var Color = Base.extend(new function() {
|
var Color = Base.extend(new function() {
|
||||||
|
// Link color types to component indices and types:
|
||||||
var types = {
|
var types = {
|
||||||
gray: ['gray'],
|
gray: ['gray'],
|
||||||
rgb: ['red', 'green', 'blue'],
|
rgb: ['red', 'green', 'blue'],
|
||||||
|
|
|
@ -30,6 +30,15 @@ QUnit.jsDump.setParser('object', function (obj, stack) {
|
||||||
var comparators = {
|
var comparators = {
|
||||||
Null: QUnit.strictEqual,
|
Null: QUnit.strictEqual,
|
||||||
Undefined: QUnit.strictEqual,
|
Undefined: QUnit.strictEqual,
|
||||||
|
Boolean: QUnit.strictEqual,
|
||||||
|
|
||||||
|
Object: function(actual, expected, message, options) {
|
||||||
|
QUnit.push(Base.equals(actual, expected), actual, expected, message);
|
||||||
|
},
|
||||||
|
|
||||||
|
Base: function(actual, expected, message, options) {
|
||||||
|
comparators.Object(actual, expected, message, options);
|
||||||
|
},
|
||||||
|
|
||||||
Number: function(actual, expected, message, options) {
|
Number: function(actual, expected, message, options) {
|
||||||
// Compare with a default tolerance of Numerical.TOLERANCE:
|
// Compare with a default tolerance of Numerical.TOLERANCE:
|
||||||
|
@ -66,18 +75,42 @@ var comparators = {
|
||||||
comparators.Size(actual, expected, message, options);
|
comparators.Size(actual, expected, message, options);
|
||||||
},
|
},
|
||||||
|
|
||||||
|
Matrix: function(actual, expected, message, options) {
|
||||||
|
comparators.Array(actual.values, expected.values, message, options);
|
||||||
|
},
|
||||||
|
|
||||||
Color: function(actual, expected, message, options) {
|
Color: function(actual, expected, message, options) {
|
||||||
if (actual && expected) {
|
if (actual && expected) {
|
||||||
equals(actual.type, expected.type,
|
equals(actual.type, expected.type,
|
||||||
(message || '') + ' type', options);
|
(message || '') + ' type', options);
|
||||||
|
// NOTE: This also compares gradients, with identity checks and all.
|
||||||
equals(actual.components, expected.components,
|
equals(actual.components, expected.components,
|
||||||
(message || '') + ' components', options);
|
(message || '') + ' components', options);
|
||||||
} else {
|
} else {
|
||||||
equals(actual, expected, message, options);
|
equals(actual, expected, message, options);
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
Segment: function(actual, expected, message, options) {
|
||||||
|
Base.each(['handleIn', 'handleOut', 'point', 'selected'],
|
||||||
|
function(key) {
|
||||||
|
equals(actual[key], expected[key], (message || '') + ' ' + key);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
},
|
||||||
|
|
||||||
|
SegmentPoint: function(actual, expected, message, options) {
|
||||||
|
comparators.Point(actual, expected, message, options);
|
||||||
|
comparators.Boolean(actual.selected, expected.selected,
|
||||||
|
(message || '') + ' selected', options);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
var identicalAfterClone = {
|
||||||
|
Gradient: true,
|
||||||
|
Symbol: true
|
||||||
|
};
|
||||||
|
|
||||||
function getFunctionMessage(func) {
|
function getFunctionMessage(func) {
|
||||||
var message = func.toString().match(
|
var message = func.toString().match(
|
||||||
/^\s*function[^\{]*\{([\s\S]*)\}\s*$/)[1]
|
/^\s*function[^\{]*\{([\s\S]*)\}\s*$/)[1]
|
||||||
|
@ -102,13 +135,16 @@ function equals(actual, expected, message, options) {
|
||||||
}
|
}
|
||||||
// Get the comparator based on the expected value's type only and ignore the
|
// Get the comparator based on the expected value's type only and ignore the
|
||||||
// actual value's type.
|
// actual value's type.
|
||||||
var type = typeof expected;
|
var type = typeof expected,
|
||||||
var comparator = comparators[
|
cls;
|
||||||
type === 'number' && 'Number'
|
type = expected === null && 'Null'
|
||||||
|
|| type === 'number' && 'Number'
|
||||||
|
|| type === 'boolean' && 'Boolean'
|
||||||
|| type === 'undefined' && 'Undefined'
|
|| type === 'undefined' && 'Undefined'
|
||||||
|| expected === null && 'Null'
|
|
||||||
|| Array.isArray(expected) && 'Array'
|
|| Array.isArray(expected) && 'Array'
|
||||||
|| expected && expected._class];
|
|| (cls = expected && expected._class)
|
||||||
|
|| type === 'object' && 'Object';
|
||||||
|
var comparator = type && comparators[type];
|
||||||
if (comparator) {
|
if (comparator) {
|
||||||
comparator(actual, expected, message, options);
|
comparator(actual, expected, message, options);
|
||||||
} else if (expected && expected.equals) {
|
} else if (expected && expected.equals) {
|
||||||
|
@ -118,6 +154,12 @@ function equals(actual, expected, message, options) {
|
||||||
// Finally perform a strict compare
|
// Finally perform a strict compare
|
||||||
QUnit.push(actual === expected, actual, expected, message);
|
QUnit.push(actual === expected, actual, expected, message);
|
||||||
}
|
}
|
||||||
|
if (options && options.cloned && cls) {
|
||||||
|
var identical = identicalAfterClone[cls];
|
||||||
|
QUnit.push(identical ? actual === expected : actual !== expected,
|
||||||
|
actual, identical ? expected : 'not ' + expected,
|
||||||
|
(message || '') + ' identity');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function test(testName, expected) {
|
function test(testName, expected) {
|
||||||
|
@ -138,174 +180,54 @@ function asyncTest(testName, expected) {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function compareStyles(style, style2, options) {
|
|
||||||
var checkIdentity = options && options.checkIdentity;
|
|
||||||
if (checkIdentity) {
|
|
||||||
equals(function() {
|
|
||||||
return style !== style2;
|
|
||||||
}, true);
|
|
||||||
}
|
|
||||||
Base.each(['fillColor', 'strokeColor'], function(key) {
|
|
||||||
if (style[key]) {
|
|
||||||
// The color should not point to the same color object:
|
|
||||||
if (checkIdentity) {
|
|
||||||
equals(function() {
|
|
||||||
return style[key] !== style2[key];
|
|
||||||
}, true, 'The ' + key
|
|
||||||
+ ' should not point to the same color object:');
|
|
||||||
}
|
|
||||||
if (style[key] instanceof Color) {
|
|
||||||
if (style[key].type === 'gradient' && checkIdentity) {
|
|
||||||
equals(function() {
|
|
||||||
return style[key].gradient === style2[key].gradient;
|
|
||||||
}, true, 'The ' + key
|
|
||||||
+ '.gradient should point to the same object:');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
equals(style[key], style2[key], 'Compare Style#' + key);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
compareObjects(['strokeCap', 'strokeJoin', 'dashArray', 'dashOffset',
|
|
||||||
'miterLimit', 'strokeOverprint', 'fillOverprint',
|
|
||||||
'fontSize', 'font', 'leading', 'justification'],
|
|
||||||
style, style2, 'Compare Style', options);
|
|
||||||
}
|
|
||||||
|
|
||||||
function compareObjects(keys, obj, obj2, message, options) {
|
|
||||||
if (options && options.checkIdentity) {
|
|
||||||
equals(function() {
|
|
||||||
return obj !== obj2;
|
|
||||||
}, true);
|
|
||||||
}
|
|
||||||
Base.each(keys, function(key) {
|
|
||||||
equals(obj[key], obj2[key], message + '#' + key, options);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
function compareSegmentPoints(segmentPoint, segmentPoint2, options) {
|
|
||||||
compareObjects(['x', 'y', 'selected'], segmentPoint, segmentPoint2,
|
|
||||||
'Compare SegmentPoint', options);
|
|
||||||
}
|
|
||||||
|
|
||||||
function compareSegments(segment, segment2, options) {
|
|
||||||
if (options.checkIdentity) {
|
|
||||||
equals(function() {
|
|
||||||
return segment !== segment2;
|
|
||||||
}, true);
|
|
||||||
}
|
|
||||||
equals(function() {
|
|
||||||
return segment.selected == segment2.selected;
|
|
||||||
}, true);
|
|
||||||
Base.each(['handleIn', 'handleOut', 'point'], function(key) {
|
|
||||||
compareSegmentPoints(segment[key], segment2[key]);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
function compareSegmentLists(segmentList, segmentList2, options) {
|
|
||||||
var checkIdentity = options && options.checkIdentity;
|
|
||||||
if (checkIdentity) {
|
|
||||||
equals(function() {
|
|
||||||
return segmentList !== segmentList2;
|
|
||||||
}, true);
|
|
||||||
}
|
|
||||||
equals(segmentList.toString(), segmentList2.toString(),
|
|
||||||
'Compare Item#segments');
|
|
||||||
if (checkIdentity) {
|
|
||||||
for (var i = 0, l = segmentList.length; i < l; i++) {
|
|
||||||
var segment = segmentList[i],
|
|
||||||
segment2 = segmentList2[i];
|
|
||||||
compareSegments(segment, segment2, options);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function compareItems(item, item2, options) {
|
function compareItems(item, item2, options) {
|
||||||
var checkIdentity = options && options.checkIdentity;
|
if (options && options.cloned)
|
||||||
if (checkIdentity) {
|
QUnit.notStrictEqual(item.id, item2.id, 'Compare Item#id');
|
||||||
equals(function() {
|
|
||||||
return item !== item2;
|
|
||||||
}, true);
|
|
||||||
|
|
||||||
equals(function() {
|
QUnit.strictEqual(item.constructor, item2.constructor,
|
||||||
return item.id !== item2.id;
|
'Compare Item#constructor');
|
||||||
}, true);
|
// When item was cloned and had a name, the name will be versioned
|
||||||
}
|
equals(options && options.cloned && item.name ? item.name + ' 1'
|
||||||
|
: item.name, item2.name, 'Compare Item#name');
|
||||||
equals(function() {
|
Base.each(['bounds', 'position', 'data', 'matrix', 'opacity', 'locked',
|
||||||
return item.constructor == item2.constructor;
|
'visible', 'blendMode', 'selected', 'fullySelected', 'clipMask',
|
||||||
}, true);
|
'guide'],
|
||||||
|
function(key) {
|
||||||
var itemProperties = ['opacity', 'locked', 'visible', 'blendMode', 'name',
|
equals(item[key], item2[key], 'Compare Item#' + key, options);
|
||||||
'selected', 'clipMask', 'guide'];
|
|
||||||
Base.each(itemProperties, function(key) {
|
|
||||||
var value = item[key];
|
|
||||||
// When item was cloned and had a name, the name will be versioned
|
|
||||||
equals(
|
|
||||||
key == 'name' && options && options.cloned && value
|
|
||||||
? value + ' 1'
|
|
||||||
: value,
|
|
||||||
item2[key],
|
|
||||||
'compare Item#' + key);
|
|
||||||
});
|
|
||||||
|
|
||||||
if (checkIdentity) {
|
|
||||||
equals(function() {
|
|
||||||
return item.bounds !== item2.bounds;
|
|
||||||
}, true);
|
|
||||||
}
|
|
||||||
|
|
||||||
equals(item.bounds.toString(), item2.bounds.toString(),
|
|
||||||
'Compare Item#bounds');
|
|
||||||
|
|
||||||
if (checkIdentity) {
|
|
||||||
equals(function() {
|
|
||||||
return item.position !== item2.position;
|
|
||||||
}, true);
|
|
||||||
}
|
|
||||||
|
|
||||||
equals(item.position.toString(), item2.position.toString(),
|
|
||||||
'Compare Item#position');
|
|
||||||
|
|
||||||
equals(function() {
|
|
||||||
return Base.equals(item.data, item2.data);
|
|
||||||
}, true);
|
|
||||||
|
|
||||||
if (item.matrix) {
|
|
||||||
if (checkIdentity) {
|
|
||||||
equals(function() {
|
|
||||||
return item.matrix !== item2.matrix;
|
|
||||||
}, true);
|
|
||||||
}
|
}
|
||||||
equals(item.matrix.toString(), item2.matrix.toString(),
|
);
|
||||||
'Compare Item#matrix');
|
|
||||||
}
|
// Style
|
||||||
|
Base.each(['fillColor', 'strokeColor', 'strokeCap', 'strokeJoin',
|
||||||
|
'dashArray', 'dashOffset', 'miterLimit',
|
||||||
|
'fontSize', 'font', 'leading', 'justification'],
|
||||||
|
function(key) {
|
||||||
|
equals(item.style[key], item2.style[key], 'Compare Style#' + key,
|
||||||
|
options);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
// Path specific
|
// Path specific
|
||||||
if (item instanceof Path) {
|
if (item instanceof Path) {
|
||||||
var keys = ['closed', 'fullySelected', 'clockwise'];
|
Base.each(['segments', 'closed', 'clockwise', 'length'],
|
||||||
for (var i = 0, l = keys.length; i < l; i++) {
|
function(key) {
|
||||||
var key = keys[i];
|
equals(item[key], item2[key], 'Compare Path#' + key, options);
|
||||||
equals(item[key], item2[key], 'Compare Path#' + key);
|
}
|
||||||
}
|
);
|
||||||
equals(item.length, item2.length, 'Compare Path#length');
|
|
||||||
compareSegmentLists(item.segments, item2.segments, options);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Shape specific
|
// Shape specific
|
||||||
if (item instanceof Shape) {
|
if (item instanceof Shape) {
|
||||||
var keys = ['shape', 'size', 'radius'];
|
Base.each(['shape', 'size', 'radius'],
|
||||||
for (var i = 0, l = keys.length; i < l; i++) {
|
function(key) {
|
||||||
var key = keys[i];
|
equals(item[key], item2[key], 'Compare Shape#' + key, options);
|
||||||
equals(item[key], item2[key], 'Compare Shape#' + key);
|
}
|
||||||
}
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Group specific
|
// Group specific
|
||||||
if (item instanceof Group) {
|
if (item instanceof Group) {
|
||||||
equals(function() {
|
equals(item.clipped, item2.clipped, 'Compare Group#clipped', options);
|
||||||
return item.clipped == item2.clipped;
|
|
||||||
}, true);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Layer specific
|
// Layer specific
|
||||||
|
@ -324,29 +246,20 @@ function compareItems(item, item2, options) {
|
||||||
options,
|
options,
|
||||||
'Compare Symbol#definition');
|
'Compare Symbol#definition');
|
||||||
} else {
|
} else {
|
||||||
equals(function() {
|
equals(item.symbol, item2.symbol, 'Compare PlacedSymbol#symbol',
|
||||||
return item.symbol == item2.symbol;
|
options);
|
||||||
}, true);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Raster specific
|
// Raster specific
|
||||||
if (item instanceof Raster) {
|
if (item instanceof Raster) {
|
||||||
equals(item.size.toString(), item2.size.toString(),
|
equals(item.size, item2.size, 'Compare Raster#size');
|
||||||
'Compare Raster#size');
|
|
||||||
equals(item.width, item2.width, 'Compare Raster#width');
|
equals(item.width, item2.width, 'Compare Raster#width');
|
||||||
equals(item.height, item2.height, 'Compare Raster#height');
|
equals(item.height, item2.height, 'Compare Raster#height');
|
||||||
|
equals(item.ppi, item2.ppi, 'Compare Raster#ppi');
|
||||||
equals(item.ppi.toString(), item2.ppi.toString(),
|
|
||||||
'Compare Raster#ppi');
|
|
||||||
|
|
||||||
equals(item.source, item2.source, 'Compare Raster#source');
|
equals(item.source, item2.source, 'Compare Raster#source');
|
||||||
if (options.checkIdentity) {
|
equals(item.image, item2.image, 'Compare Raster#image');
|
||||||
equals(item.image, item2.image, 'Compare Raster#image');
|
equals(item.toDataURL(), item2.toDataURL(),
|
||||||
}
|
|
||||||
equals(item.size.toString(), item2.size.toString(),
|
|
||||||
'Compare Raster#size');
|
|
||||||
equals(item.toDataURL() == item2.toDataURL(), true,
|
|
||||||
'Compare Raster#toDataUrl()');
|
'Compare Raster#toDataUrl()');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -357,18 +270,7 @@ function compareItems(item, item2, options) {
|
||||||
|
|
||||||
// PointText specific:
|
// PointText specific:
|
||||||
if (item instanceof PointText) {
|
if (item instanceof PointText) {
|
||||||
if (options.checkIdentity) {
|
equals(item.point, item2.point, 'Compare Item#point');
|
||||||
equals(function() {
|
|
||||||
return item.point !== item2.point;
|
|
||||||
}, true);
|
|
||||||
}
|
|
||||||
equals(item.point.toString(), item2.point.toString(),
|
|
||||||
'Compare Item#point');
|
|
||||||
}
|
|
||||||
|
|
||||||
if (item.style) {
|
|
||||||
// Style
|
|
||||||
compareStyles(item.style, item2.style, options);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check length of children and recursively compare them:
|
// Check length of children and recursively compare them:
|
||||||
|
|
|
@ -24,7 +24,7 @@ function cloneAndCompare(item) {
|
||||||
return copy.parent.children[copy.name] == copy;
|
return copy.parent.children[copy.name] == copy;
|
||||||
}, true);
|
}, true);
|
||||||
}
|
}
|
||||||
compareItems(item, copy, { cloned: true, checkIdentity: true });
|
compareItems(item, copy, { cloned: true });
|
||||||
// Remove the cloned item to restore the document:
|
// Remove the cloned item to restore the document:
|
||||||
copy.remove();
|
copy.remove();
|
||||||
}
|
}
|
||||||
|
|
|
@ -15,7 +15,7 @@ module('SVGImport');
|
||||||
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'));
|
||||||
compareItems(item, item.clone(), { cloned: true, checkIdentity: true });
|
compareItems(item, item.clone(), { cloned: true });
|
||||||
});
|
});
|
||||||
|
|
||||||
test('make an svg line', function() {
|
test('make an svg line', function() {
|
||||||
|
|
Loading…
Reference in a new issue