Implement caching of internal, untransformed bounds.

This commit is contained in:
Jürg Lehni 2013-12-09 10:53:19 +01:00
parent 5197dd81c5
commit e238d23194
4 changed files with 28 additions and 24 deletions

View file

@ -798,22 +798,26 @@ var Item = Base.extend(Callback, /** @lends Item# */{
// No need for _changed() since the only thing this affects is _position // No need for _changed() since the only thing this affects is _position
delete this._position; delete this._position;
} }
}, Base.each(['getBounds', 'getStrokeBounds', 'getHandleBounds', 'getRoughBounds'], }, Base.each(['getBounds', 'getStrokeBounds', 'getHandleBounds',
function(name) { 'getRoughBounds', 'getInternalBounds'],
function(key) {
// Produce getters for bounds properties. These handle caching, matrices // Produce getters for bounds properties. These handle caching, matrices
// and redirect the call to the private _getBounds, which can be // and redirect the call to the private _getBounds, which can be
// overridden by subclasses, see below. // overridden by subclasses, see below.
this[name] = function(/* matrix */) { var internal = key === 'getInternalBounds';
this[key] = function(/* matrix */) {
var getter = this._boundsGetter, var getter = this._boundsGetter,
// Allow subclasses to override _boundsGetter if they use // Allow subclasses to override _boundsGetter if they use
// the same calculations for multiple type of bounds. // the same calculations for multiple type of bounds.
// The default is name: // The default is key:
bounds = this._getCachedBounds(typeof getter == 'string' bounds = this._getCachedBounds(!internal
? getter : getter && getter[name] || name, arguments[0]); && (typeof getter === 'string'
? getter : getter && getter[key])
|| key, arguments[0], null, internal);
// If we're returning 'bounds', create a LinkedRectangle that uses // If we're returning 'bounds', create a LinkedRectangle that uses
// the setBounds() setter to update the Item whenever the bounds are // the setBounds() setter to update the Item whenever the bounds are
// changed: // changed:
return name === 'getBounds' return key === 'getBounds'
? new LinkedRectangle(bounds.x, bounds.y, bounds.width, ? new LinkedRectangle(bounds.x, bounds.y, bounds.width,
bounds.height, this, 'setBounds') bounds.height, this, 'setBounds')
: bounds; : bounds;
@ -879,12 +883,12 @@ var Item = Base.extend(Callback, /** @lends Item# */{
* 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) { _getCachedBounds: function(getter, matrix, cacheItem, internal) {
// 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();
var _matrix = this._matrix.orNullIfIdentity(), var _matrix = internal ? null : this._matrix.orNullIfIdentity(),
cache = (!matrix || matrix.equals(_matrix)) && getter; cache = (!matrix || matrix.equals(_matrix)) && getter;
// Set up a boundsCache structure that keeps track of items that keep // Set up a boundsCache structure that keeps track of items that keep
// cached bounds that depend on this item. We store this in our parent, // cached bounds that depend on this item. We store this in our parent,
@ -925,13 +929,16 @@ var Item = Base.extend(Callback, /** @lends Item# */{
: matrix; : matrix;
// If we're caching bounds on this item, pass it on as cacheItem, so the // If we're caching bounds on this item, pass it on as cacheItem, so the
// children can setup the _boundsCache structures for it. // children can setup the _boundsCache structures for it.
var bounds = this._getBounds(getter, matrix, cache ? this : cacheItem); var bounds = this._getBounds(getter === 'getInternalBounds'
? 'getBounds' : getter, matrix, cache ? this : cacheItem);
// 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 (cache) {
if (!this._bounds) if (!this._bounds)
this._bounds = {}; this._bounds = {};
this._bounds[cache] = bounds.clone(); var cached = this._bounds[cache] = bounds.clone();
// Mark as internal, so Item#transform() won't transform it!
cached._internal = internal;
} }
return bounds; return bounds;
}, },
@ -1542,11 +1549,7 @@ var Item = Base.extend(Callback, /** @lends Item# */{
} }
// We only implement it here for items with rectangular content, // We only implement it here for items with rectangular content,
// for anything else we need to override #contains() // for anything else we need to override #contains()
// TODO: There currently is no caching for the results of direct calls return point.isInside(this.getInternalBounds());
// to this._getBounds('getBounds') (without the application of the
// internal matrix). Performance improvements could be achieved if
// these were cached too. See #_getCachedBounds().
return point.isInside(this._getBounds('getBounds'));
}, },
/** /**
@ -1622,7 +1625,7 @@ var Item = Base.extend(Callback, /** @lends Item# */{
if (checkSelf && (options.center || options.bounds) && this._parent) { if (checkSelf && (options.center || options.bounds) && this._parent) {
// Don't get the transformed bounds, check against transformed // Don't get the transformed bounds, check against transformed
// points instead // points instead
var bounds = this._getBounds('getBounds'); var bounds = this.getInternalBounds();
if (options.center) if (options.center)
res = checkBounds('center', 'Center'); res = checkBounds('center', 'Center');
if (!res && options.bounds) { if (!res && options.bounds) {
@ -2677,6 +2680,8 @@ var Item = Base.extend(Callback, /** @lends Item# */{
// 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 rect = bounds[key];
// See _getCachedBounds for an explanation of this:
if (!rect._internal)
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

View file

@ -1716,6 +1716,8 @@ var Path = PathItem.extend(/** @lends Path# */{
if (!closed && !this.hasFill() if (!closed && !this.hasFill()
// We need to call the internal _getBounds, to get non- // We need to call the internal _getBounds, to get non-
// transformed bounds. // transformed bounds.
// TODO: Implement caching for internal rough bounds and switch
// hit-testing code to using this too.
|| !this._getBounds('getRoughBounds')._containsPoint(point)) || !this._getBounds('getRoughBounds')._containsPoint(point))
return 0; return 0;
// Use the crossing number algorithm, by counting the crossings of the // Use the crossing number algorithm, by counting the crossings of the

View file

@ -462,11 +462,8 @@ var Project = PaperScopeItem.extend(/** @lends Project# */{
if (item._drawSelected) if (item._drawSelected)
item._drawSelected(ctx, mx); item._drawSelected(ctx, mx);
if (item._boundsSelected) { if (item._boundsSelected) {
// We need to call the internal _getBounds, to get non-
// transformed bounds.
// TODO: Implement caching for these too?
var coords = mx._transformCorners( var coords = mx._transformCorners(
item._getBounds('getBounds')); item.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

@ -20,8 +20,8 @@ test('PointText', function() {
content: 'Hello World!' content: 'Hello World!'
}); });
equals(text.fillColor, { red: 0, green: 0, blue: 0 }, 'text.fillColor should be black by default'); equals(text.fillColor, { red: 0, green: 0, blue: 0 }, 'text.fillColor should be black by default');
equals(text.point, { x: 100, y: 100 }); comparePoints(text.point, { x: 100, y: 100 });
equals(text.bounds, { x: 100, y: 87.4, width: 55, height: 16.8 }); compareRectangles(text.bounds, { x: 100, y: 87.4, width: 77, height: 16.8 });
equals(function() { equals(function() {
return text.hitTest(text.bounds.center) != null; return text.hitTest(text.bounds.center) != null;
}, true); }, true);