Part 1 of large refactoring of bounds handling.

This commit is contained in:
Jürg Lehni 2016-02-12 17:59:37 +01:00
parent cb79232e12
commit 55c5f42716
12 changed files with 175 additions and 133 deletions

View file

@ -92,7 +92,7 @@ Base.inject(/** @lends Base# */{
this[key] = value; this[key] = value;
} }
} }
return true; return props;
} }
}, },

View file

@ -167,21 +167,13 @@ var Group = Item.extend(/** @lends Group# */{
child.setClipMask(clipped); child.setClipMask(clipped);
}, },
_getBounds: function _getBounds(getter, matrix, cacheItem, internal) { _getBounds: function _getBounds(matrix, options) {
var clipItem = this._getClipItem(), var clipItem = this._getClipItem();
// We need to fall-back to bounds getter that do not take stroke
// into account
clipBoundsGetter = {
getStrokeBounds: 'getBounds',
getRoughBounds: 'getHandleBounds',
getInternalRoughBounds: 'getInternalBounds'
};
return clipItem return clipItem
? clipItem._getCachedBounds(clipBoundsGetter[getter] || getter, ? clipItem._getCachedBounds(
matrix && matrix.appended(clipItem._matrix), matrix && matrix.appended(clipItem._matrix),
cacheItem, internal) Base.set({}, options, { stroke: false }))
: _getBounds.base.call(this, getter, matrix, cacheItem, : _getBounds.base.call(this, matrix, options);
internal);
}, },
_hitTestChildren: function _hitTestChildren(point, options) { _hitTestChildren: function _hitTestChildren(point, options) {

View file

@ -780,55 +780,14 @@ new function() { // // Scope to inject various item event handlers
}, },
_pivot: null, _pivot: null,
}, Base.each(['bounds', 'strokeBounds', 'handleBounds', 'roughBounds', }, Base.each({ // Produce getters for bounds properties:
'internalBounds', 'internalHandleBounds', 'internalRoughBounds'], getStrokeBounds: { stroke: true },
function(key) { getHandleBounds: { handle: true },
// Produce getters for bounds properties. These handle caching, matrices getInternalBounds: { internal: true }
// and redirect the call to the private _getBounds, which can be },
// overridden by subclasses, see below. function(options, key) {
// Treat internalBounds and internalRoughBounds untransformed, as this[key] = function(matrix) {
// required by the code that uses these methods internally, but make return this.getBounds(matrix, options);
// sure they can be cached like all the others as well.
// Pass on the getter that these version actually use, untransformed,
// as `internal`.
// NOTE: These need to be versions of other methods, as otherwise the
// cache gets messed up.
var getter = 'get' + Base.capitalize(key),
match = key.match(/^internal(.*)$/),
internal = match ? 'get' + match[1] : null,
// Determine if the stroke is involved in the calculation of the
// bounds: strokeBounds, roughBounds,internalRoughBounds
stroke = /^stroke|rough/i.test(key);
this[getter] = function(_matrix) {
// TODO: If we're getting stroke based bounds (strokeBounds,
// roughBounds, internalRoughBounds), and the object does not have
// a stroke, fall back to the bounds getter without the stroke:
// strokeBounds -> bounds
// roughBounds -> handleBounds
// internalRoughBounds -> internalHandleBounds
var boundsGetter = this._boundsGetter,
// Allow subclasses to override _boundsGetter if they use the
// same calculations for multiple type of bounds.
// The default is getter:
name = !internal && (typeof boundsGetter === 'string'
? boundsGetter : boundsGetter && boundsGetter[getter])
|| getter,
// We can only cache the bounds if the path uses stroke-scaling,
// or if no stroke is involved in the calculation of the bounds.
// When strokeScaling is false, the bounds are affected by the
// zoom level of the view, hence we can't cache.
canCache = !stroke || this.getStrokeScaling(),
// If we're caching bounds, pass on this item as cacheItem, so
// the children can setup _boundsCache structures for it.
bounds = this._getCachedBounds(name, _matrix, canCache && this,
internal);
// If we're returning 'bounds', create a LinkedRectangle that uses
// the setBounds() setter to update the Item whenever the bounds are
// changed:
return key === 'bounds'
? new LinkedRectangle(bounds.x, bounds.y, bounds.width,
bounds.height, this, 'setBounds')
: bounds;
}; };
}, },
/** @lends Item# */{ /** @lends Item# */{
@ -836,13 +795,41 @@ new function() { // // Scope to inject various item event handlers
// See _matrix parameter above. // See _matrix parameter above.
beans: true, beans: true,
getBounds: function(matrix, options) {
var hasMatrix = options || matrix instanceof Matrix,
opts = Base.set({}, hasMatrix ? options : matrix,
this._boundsOptions);
// We can only cache the bounds if the path uses stroke-scaling, or if
// no stroke is involved in the calculation of the bounds.
// When strokeScaling is false, the bounds are affected by the zoom
// level of the view, hence we can't cache.
// TODO: Look more into handling of stroke-scaling, e.g. on groups with
// some children that have strokeScaling, as well as SymbolItem with
// SymbolDefinition that have strokeScaling!
// TODO: Once that is resolved, we should be able to turn off
// opts.stroke if a resolved item definition does not have a stroke,
// allowing the code to share caches between #strokeBounds and #bounds.
if (!opts.stroke || this.getStrokeScaling())
opts.cacheItem = this;
// If we're caching bounds, pass on this item as cacheItem, so
// the children can setup _boundsCache structures for it.
var bounds = this._getCachedBounds(hasMatrix && matrix, opts);
// If we're returning '#bounds', create a LinkedRectangle that uses
// the setBounds() setter to update the Item whenever the bounds are
// changed:
return arguments.length === 0
? new LinkedRectangle(bounds.x, bounds.y, bounds.width,
bounds.height, this, 'setBounds')
: bounds;
},
/** /**
* Protected method used in all the bounds getters. It loops through all the * Protected method used in all the bounds getters. It loops through all the
* children, gets their bounds and finds the bounds around all of them. * children, gets their bounds and finds the bounds around all of them.
* Subclasses override it to define calculations for the various required * Subclasses override it to define calculations for the various required
* bounding types. * bounding types.
*/ */
_getBounds: function(getter, matrix, cacheItem, internal) { _getBounds: function(matrix, options) {
// NOTE: We cannot cache these results here, since we do not get // NOTE: We cannot cache these results here, since we do not get
// _changed() notifications here for changing geometry in children. // _changed() notifications here for changing geometry in children.
// But cacheName is used in sub-classes such as SymbolItem and Raster. // But cacheName is used in sub-classes such as SymbolItem and Raster.
@ -854,7 +841,7 @@ new function() { // // Scope to inject various item event handlers
// Call _updateBoundsCache() even when the group only holds empty / // Call _updateBoundsCache() even when the group only holds empty /
// invisible items), so future changes in these items will cause right // invisible items), so future changes in these items will cause right
// handling of _boundsCache. // handling of _boundsCache.
Item._updateBoundsCache(this, cacheItem); Item._updateBoundsCache(this, options.cacheItem);
var x1 = Infinity, var x1 = Infinity,
x2 = -x1, x2 = -x1,
y1 = x1, y1 = x1,
@ -862,9 +849,8 @@ new function() { // // Scope to inject various item event handlers
for (var i = 0, l = children.length; i < l; i++) { for (var i = 0, l = children.length; i < l; i++) {
var child = children[i]; var child = children[i];
if (child._visible && !child.isEmpty()) { if (child._visible && !child.isEmpty()) {
var rect = child._getCachedBounds(getter, var rect = child._getCachedBounds(
matrix && matrix.appended(child._matrix), matrix && matrix.appended(child._matrix), options);
cacheItem, internal);
x1 = Math.min(rect.x, x1); x1 = Math.min(rect.x, x1);
y1 = Math.min(rect.y, y1); y1 = Math.min(rect.y, y1);
x2 = Math.max(rect.x + rect.width, x2); x2 = Math.max(rect.x + rect.width, x2);
@ -901,29 +887,38 @@ new function() { // // Scope to inject various item event handlers
* Private method that deals with the calling of _getBounds, recursive * Private method that deals with the calling of _getBounds, recursive
* matrix concatenation and handles all the complicated caching mechanisms. * matrix concatenation and handles all the complicated caching mechanisms.
*/ */
_getCachedBounds: function(getter, matrix, cacheItem, internal) { _getCachedBounds: function(matrix, options) {
// See if we can cache these bounds. We only cache the bounds // See if we can cache these bounds. We only cache the bounds
// transformed with the internally stored _matrix, (the default if no // transformed with the internally stored _matrix, (the default if no
// matrix is passed). // matrix is passed).
matrix = matrix && matrix._orNullIfIdentity(); matrix = matrix && matrix._orNullIfIdentity();
// Do not transform by the internal matrix if there is a internal getter // Do not transform by the internal matrix for internal, untransformed
var _matrix = internal ? null : this._matrix._orNullIfIdentity(), // bounds.
cache = cacheItem && (!matrix || matrix.equals(_matrix)) && getter; var internal = options.internal,
cacheItem = options.cacheItem,
_matrix = internal ? null : this._matrix._orNullIfIdentity(),
// Create a key for caching, reflecting all bounds options.
cacheKey = cacheItem && (!matrix || matrix.equals(_matrix)) && [
options.stroke ? 1 : 0,
options.handle ? 1 : 0,
internal ? 1 : 0
].join('');
// NOTE: This needs to happen before returning cached values, since even // NOTE: This needs to happen before returning cached values, since even
// then, _boundsCache needs to be kept up-to-date. // then, _boundsCache needs to be kept up-to-date.
Item._updateBoundsCache(this._parent || this._parentSymbol, cacheItem); Item._updateBoundsCache(this._parent || this._parentSymbol, cacheItem);
if (cache && this._bounds && this._bounds[cache]) if (cacheKey && this._bounds && cacheKey in this._bounds)
return this._bounds[cache].clone(); return this._bounds[cacheKey].rect.clone();
var bounds = this._getBounds(internal || getter, matrix || _matrix, var bounds = this._getBounds(matrix || _matrix, options);
cacheItem, internal);
// If we can cache the result, update the _bounds cache structure // If we can cache the result, update the _bounds cache structure
// before returning // before returning
if (cache) { if (cacheKey) {
if (!this._bounds) if (!this._bounds)
this._bounds = {}; this._bounds = {};
var cached = this._bounds[cache] = bounds.clone(); var cached = this._bounds[cacheKey] = {
// Mark as internal, so Item#transform() won't transform it! rect: bounds.clone(),
cached._internal = !!internal; // Mark as internal, so Item#transform() won't transform it
internal: options.internal
};
} }
return bounds; return bounds;
}, },
@ -933,10 +928,9 @@ new function() { // // Scope to inject various item event handlers
* when calculating bounds: the item's matrix if {@link #strokeScaling} is * when calculating bounds: the item's matrix if {@link #strokeScaling} is
* `true`, otherwise the shiftless, inverted view matrix. * `true`, otherwise the shiftless, inverted view matrix.
*/ */
_getStrokeMatrix: function(matrix, internal) { _getStrokeMatrix: function(matrix, options) {
return this.getStrokeScaling() ? matrix return this.getStrokeScaling() ? matrix : (options && options.internal
: (internal ? this : this._parent).getViewMatrix() ? this : this._parent).getViewMatrix().invert()._shiftless();
.invert()._shiftless();
}, },
statics: { statics: {
@ -951,7 +945,7 @@ new function() { // // Scope to inject various item event handlers
* times the same structure. * times the same structure.
*/ */
_updateBoundsCache: function(parent, item) { _updateBoundsCache: function(parent, item) {
if (parent) { if (parent && item) {
// Set-up the parent's boundsCache structure if it does not // Set-up the parent's boundsCache structure if it does not
// exist yet and add the item to it. // exist yet and add the item to it.
var id = item._id, var id = item._id,
@ -1780,7 +1774,8 @@ new function() { // // Scope to inject various item event handlers
// Transform point to local coordinates. // Transform point to local coordinates.
point = matrix._inverseTransform(point); point = matrix._inverseTransform(point);
// If the matrix is non-reversible, point will now be `null`: // If the matrix is non-reversible, point will now be `null`:
if (!point || !this._children && !this.getInternalRoughBounds() if (!point || !this._children &&
!this.getBounds({ internal: true, stroke: true, handle: true })
.expand(tolerancePadding.multiply(2))._containsPoint(point)) .expand(tolerancePadding.multiply(2))._containsPoint(point))
return null; return null;
// Filter for type, guides and selected items if that's required. // Filter for type, guides and selected items if that's required.
@ -2036,7 +2031,7 @@ new function() { // // Scope to inject various item event handlers
if (obj) { if (obj) {
// Create a copy of the match object that doesn't contain // Create a copy of the match object that doesn't contain
// these special properties: // these special properties:
match = Base.set({}, match, { match = new Base()._set(match, {
recursive: true, inside: true, overlapping: true recursive: true, inside: true, overlapping: true
}); });
} }
@ -3248,12 +3243,14 @@ new function() { // // Scope to inject various item event handlers
// Transform the old bound by looping through all the cached bounds // Transform the old bound by looping through all the cached bounds
// in _bounds and transform each. // in _bounds and transform each.
for (var key in bounds) { for (var key in bounds) {
var rect = bounds[key]; var cache = bounds[key];
// If these are internal bounds, only transform them if this // If these are internal bounds, only transform them if this
// item applied its matrix. // item applied its matrix.
if (applyMatrix || !rect._internal) if (applyMatrix || !cache.internal) {
var rect = cache.rect;
matrix._transformBounds(rect, rect); matrix._transformBounds(rect, rect);
} }
}
// If we have cached bounds, update _position again as its // If we have cached bounds, update _position again as its
// center. We need to take into account _boundsGetter here too, in // center. We need to take into account _boundsGetter here too, in
// case another getter is assigned to it, e.g. 'getStrokeBounds'. // case another getter is assigned to it, e.g. 'getStrokeBounds'.
@ -4147,7 +4144,8 @@ new function() { // // Scope to inject various item event handlers
this._drawSelected(ctx, mx, selectedItems); this._drawSelected(ctx, mx, selectedItems);
if (this._boundsSelected) { if (this._boundsSelected) {
var half = size / 2, var half = size / 2,
coords = mx._transformCorners(this.getInternalBounds()); coords = mx._transformCorners(
this.getInternalBounds());
// Now draw a rectangle that connects the transformed // Now draw a rectangle that connects the transformed
// bounds corners, and draw the corners. // bounds corners, and draw the corners.
ctx.beginPath(); ctx.beginPath();

View file

@ -23,7 +23,7 @@ var Raster = Item.extend(/** @lends Raster# */{
_canApplyMatrix: false, _canApplyMatrix: false,
// Raster doesn't make the distinction between the different bounds, // Raster doesn't make the distinction between the different bounds,
// so use the same name for all of them // so use the same name for all of them
_boundsGetter: 'getBounds', _boundsOptions: { stroke: false, handle: false },
_boundsSelected: true, _boundsSelected: true,
_serializeFields: { _serializeFields: {
crossOrigin: null, // NOTE: Needs to be set before source to work! crossOrigin: null, // NOTE: Needs to be set before source to work!
@ -690,7 +690,7 @@ var Raster = Item.extend(/** @lends Raster# */{
* @type Function * @type Function
*/ */
_getBounds: function(getter, matrix) { _getBounds: function(matrix, options) {
var rect = new Rectangle(this._size).setCenter(0, 0); var rect = new Rectangle(this._size).setCenter(0, 0);
return matrix ? matrix._transformBounds(rect) : rect; return matrix ? matrix._transformBounds(rect) : rect;
}, },

View file

@ -275,18 +275,19 @@ var Shape = Item.extend(/** @lends Shape# */{
return !(this.hasFill() && this.hasStroke()); return !(this.hasFill() && this.hasStroke());
}, },
_getBounds: function(getter, matrix, cacheItem, internal) { _getBounds: function(matrix, options) {
var rect = new Rectangle(this._size).setCenter(0, 0), var rect = new Rectangle(this._size).setCenter(0, 0),
style = this._style, style = this._style,
strokeWidth = style.hasStroke() && strokeWidth = options.stroke && style.hasStroke()
/^get(?:Stroke|Rough)Bounds$/.test(getter) && && style.getStrokeWidth();
style.getStrokeWidth();
// If we're getting the strokeBounds, include the stroke width before // If we're getting the strokeBounds, include the stroke width before
// or after transforming the rect, based on strokeScaling. // or after transforming the rect, based on strokeScaling.
if (matrix) if (matrix)
rect = matrix._transformBounds(rect); rect = matrix._transformBounds(rect);
return strokeWidth ? rect.expand(Path._getStrokePadding( return strokeWidth
strokeWidth, this._getStrokeMatrix(matrix, internal))) : rect; ? rect.expand(Path._getStrokePadding(strokeWidth,
this._getStrokeMatrix(matrix, options)))
: rect;
} }
}, },
new function() { // Scope for _contains() and _hitTestSelf() code. new function() { // Scope for _contains() and _hitTestSelf() code.

View file

@ -23,7 +23,7 @@ var SymbolItem = Item.extend(/** @lends SymbolItem# */{
_applyMatrix: false, _applyMatrix: false,
_canApplyMatrix: false, _canApplyMatrix: false,
// SymbolItem uses strokeBounds for bounds // SymbolItem uses strokeBounds for bounds
_boundsGetter: { getBounds: 'getStrokeBounds' }, _boundsOptions: { stroke: true },
_boundsSelected: true, _boundsSelected: true,
_serializeFields: { _serializeFields: {
symbol: null symbol: null
@ -114,12 +114,11 @@ var SymbolItem = Item.extend(/** @lends SymbolItem# */{
}, },
_getBounds: function(getter, matrix, cacheItem, internal) { _getBounds: function(matrix, options) {
var item = this._definition._item; var item = this._definition._item;
// Redirect the call to the definition item to calculate the bounds. // Redirect the call to the definition item to calculate the bounds.
return item._getCachedBounds(getter, return item._getCachedBounds(matrix && matrix.appended(item._matrix),
matrix && matrix.appended(item._matrix), options);
cacheItem, internal);
}, },
_hitTestSelf: function(point, options) { _hitTestSelf: function(point, options) {

View file

@ -2512,8 +2512,15 @@ new function() { // PostScript-style drawing commands
// Curve. But not all of them use all these parameters, and some define // Curve. But not all of them use all these parameters, and some define
// additional ones after. // additional ones after.
_getBounds: function(getter, matrix) { _getBounds: function(matrix, options) {
return Path[getter](this._segments, this._closed, this, matrix); var method = options.handle
? options.stroke
? 'getRoughBounds'
: 'getHandleBounds'
: options.stroke
? 'getStrokeBounds'
: 'getBounds';
return Path[method](this._segments, this._closed, this, matrix, options);
}, },
// Mess with indentation in order to get more line-space below: // Mess with indentation in order to get more line-space below:
@ -2523,7 +2530,7 @@ statics: {
* *
* @private * @private
*/ */
getBounds: function(segments, closed, path, matrix, strokePadding) { getBounds: function(segments, closed, path, matrix, options, strokePadding) {
var first = segments[0]; var first = segments[0];
// If there are no segments, return "empty" rectangle, just like groups, // If there are no segments, return "empty" rectangle, just like groups,
// since #bounds is assumed to never return null. // since #bounds is assumed to never return null.
@ -2564,18 +2571,18 @@ statics: {
* *
* @private * @private
*/ */
getStrokeBounds: function(segments, closed, path, matrix, internal) { getStrokeBounds: function(segments, closed, path, matrix, options) {
var style = path._style; var style = path._style;
if (!style.hasStroke()) if (!style.hasStroke())
return Path.getBounds(segments, closed, path, matrix); return Path.getBounds(segments, closed, path, matrix, options);
var length = segments.length - (closed ? 0 : 1), var length = segments.length - (closed ? 0 : 1),
strokeWidth = style.getStrokeWidth(), strokeWidth = style.getStrokeWidth(),
strokeRadius = strokeWidth / 2, strokeRadius = strokeWidth / 2,
strokeMatrix = path._getStrokeMatrix(matrix, internal), strokeMatrix = path._getStrokeMatrix(matrix, options),
strokePadding = Path._getStrokePadding(strokeWidth, strokeMatrix), strokePadding = Path._getStrokePadding(strokeWidth, strokeMatrix),
// Start with normal path bounds with added stroke padding. Then we // Start with normal path bounds with added stroke padding. Then we
// only need to look at each segment and handle join / cap / miter. // only need to look at each segment and handle join / cap / miter.
bounds = Path.getBounds(segments, closed, path, matrix, bounds = Path.getBounds(segments, closed, path, matrix, options,
strokePadding), strokePadding),
join = style.getStrokeJoin(), join = style.getStrokeJoin(),
cap = style.getStrokeCap(), cap = style.getStrokeCap(),
@ -2736,8 +2743,8 @@ statics: {
* *
* @private * @private
*/ */
getHandleBounds: function(segments, closed, path, matrix, strokePadding, getHandleBounds: function(segments, closed, path, matrix, options,
joinPadding) { strokePadding, joinPadding) {
var coords = new Array(6), var coords = new Array(6),
x1 = Infinity, x1 = Infinity,
x2 = -x1, x2 = -x1,
@ -2772,7 +2779,7 @@ statics: {
* *
* @private * @private
*/ */
getRoughBounds: function(segments, closed, path, matrix, internal) { getRoughBounds: function(segments, closed, path, matrix, options) {
// Delegate to handleBounds, but pass on radius values for stroke and // Delegate to handleBounds, but pass on radius values for stroke and
// joins. Handle miter joins specially, by passing the largest radius // joins. Handle miter joins specially, by passing the largest radius
// possible. // possible.
@ -2780,14 +2787,14 @@ statics: {
strokeRadius = style.hasStroke() ? style.getStrokeWidth() / 2 : 0, strokeRadius = style.hasStroke() ? style.getStrokeWidth() / 2 : 0,
joinRadius = strokeRadius, joinRadius = strokeRadius,
strokeMatrix = strokeRadius && strokeMatrix = strokeRadius &&
path._getStrokeMatrix(matrix, internal); path._getStrokeMatrix(matrix, options);
if (strokeRadius > 0) { if (strokeRadius > 0) {
if (style.getStrokeJoin() === 'miter') if (style.getStrokeJoin() === 'miter')
joinRadius = strokeRadius * style.getMiterLimit(); joinRadius = strokeRadius * style.getMiterLimit();
if (style.getStrokeCap() === 'square') if (style.getStrokeCap() === 'square')
joinRadius = Math.max(joinRadius, strokeRadius * Math.sqrt(2)); joinRadius = Math.max(joinRadius, strokeRadius * Math.sqrt(2));
} }
return Path.getHandleBounds(segments, closed, path, matrix, return Path.getHandleBounds(segments, closed, path, matrix, options,
Path._getStrokePadding(strokeRadius, strokeMatrix), Path._getStrokePadding(strokeRadius, strokeMatrix),
Path._getStrokePadding(joinRadius, strokeMatrix)); Path._getStrokePadding(joinRadius, strokeMatrix));
} }

View file

@ -176,7 +176,8 @@ var PathItem = Item.extend(/** @lends PathItem# */{
// Check the transformed point against the untransformed (internal) // Check the transformed point against the untransformed (internal)
// handle bounds, which is the fastest rough bounding box to calculate // handle bounds, which is the fastest rough bounding box to calculate
// for a quick check before calculating the actual winding. // for a quick check before calculating the actual winding.
var winding = point.isInside(this.getInternalHandleBounds()) var winding = point.isInside(
this.getBounds({ internal: true, handle: true }))
&& this._getWinding(point); && this._getWinding(point);
return !!(this.getFillRule() === 'evenodd' ? winding & 1 : winding); return !!(this.getFillRule() === 'evenodd' ? winding & 1 : winding);
/*#*/ } // !__options.nativeContains && __options.booleanOperations /*#*/ } // !__options.nativeContains && __options.booleanOperations

View file

@ -102,7 +102,7 @@ var PointText = TextItem.extend(/** @lends PointText# */{
} }
}, },
_getBounds: function(getter, matrix) { _getBounds: function(matrix, options) {
var style = this._style, var style = this._style,
lines = this._lines, lines = this._lines,
numLines = lines.length, numLines = lines.length,

View file

@ -31,7 +31,7 @@ var TextItem = Item.extend(/** @lends TextItem# */{
}, },
// TextItem doesn't make the distinction between the different bounds, // TextItem doesn't make the distinction between the different bounds,
// so use the same name for all of them // so use the same name for all of them
_boundsGetter: 'getBounds', _boundsOptions: { stroke: false, handle: false },
initialize: function TextItem(arg) { initialize: function TextItem(arg) {
this._content = ''; this._content = '';

View file

@ -11,7 +11,7 @@
*/ */
var isNode = typeof global === 'object', var isNode = typeof global === 'object',
isPhantom = !!window.callPhantom, isPhantom = typeof window === 'object' && !!window.callPhantom,
root; root;
if (isNode) { if (isNode) {

View file

@ -585,25 +585,69 @@ test('path.strokeBounds without strokeScaling and zoomed view', function() {
center: [0, 0], center: [0, 0],
radius: 100, radius: 100,
strokeColor: 'black', strokeColor: 'black',
strokeWidth: 15, strokeWidth: 20,
strokeScaling: false,
applyMatrix: false
});
view.zoom = 2;
equals(path.strokeBounds, new Rectangle(-105, -105, 210, 210),
'path.strokeBounds with zoomed view');
view.zoom = 1;
equals(path.strokeBounds, new Rectangle(-110, -110, 220, 220),
'path.strokeBounds without zoomed view');
path.scale(0.5, 1);
view.zoom = 2;
// Internal stroke bounds need to apply stroke deformation with
// strokeScaling:
equals(path.getBounds({ internal: true, stroke: true }),
new Rectangle(-110, -105, 220, 210),
'path.getBounds({ internal: true, stroke: true })'
+ ' with path.applyMatrix = false, path.scale(0.5, 1);');
path.applyMatrix = true;
equals(path.getBounds({ internal: true, stroke: true }),
new Rectangle(-55, -105, 110, 210),
'path.getBounds({ internal: true, stroke: true })'
+ ' with path.applyMatrix = true, path.scale(0.5, 1);');
});
test('shape.strokeBounds without strokeScaling and zoomed view', function() {
var shape = new Shape.Circle({
center: [0, 0],
radius: 100,
strokeColor: 'black',
strokeWidth: 20,
strokeScaling: false strokeScaling: false
}); });
view.zoom = 2; view.zoom = 2;
new Path.Rectangle({ equals(shape.strokeBounds, new Rectangle(-105, -105, 210, 210),
rectangle: path.strokeBounds, 'shape.strokeBounds with zoomed view');
strokeColor: 'red',
strokeScaling: false
});
equals(path.strokeBounds, new Rectangle(-103.75, -103.75, 207.5, 207.5),
'path.strokeBounds with zoomed view');
view.zoom = 1; view.zoom = 1;
equals(path.strokeBounds, new Rectangle(-107.5, -107.5, 215, 215), equals(shape.strokeBounds, new Rectangle(-110, -110, 220, 220),
'path.strokeBounds without zoomed view'); 'shape.strokeBounds without zoomed view');
shape.scale(0.5, 1);
view.zoom = 2;
// Internal stroke bounds need to apply stroke deformation with
// strokeScaling:
equals(shape.getBounds({ internal: true, stroke: true }),
new Rectangle(-110, -105, 220, 210),
'shape.getBounds({ internal: true, stroke: true })'
+ ' with shape.scale(0.5, 1);');
}); });
test('path.internalBounds', function() { test('path.internalBounds', function() {