Properly handle bounds caching when items are empty first.

Closes #676.
This commit is contained in:
Jürg Lehni 2015-06-15 19:04:15 +02:00
parent ba092b316e
commit c7281ee619
2 changed files with 37 additions and 30 deletions

View file

@ -892,6 +892,10 @@ var Item = Base.extend(Emitter, /** @lends Item# */{
// Scriptographer behaves weirdly then too. // Scriptographer behaves weirdly then too.
if (!children || children.length == 0) if (!children || children.length == 0)
return new Rectangle(); return new Rectangle();
// Call _updateBoundsCache() even when the group is currently empty
// (or only holds empty / invisible items), so future changes in these
// items will cause right handling of _boundsCache.
Item._updateBoundsCache(this, cacheItem);
var x1 = Infinity, var x1 = Infinity,
x2 = -x1, x2 = -x1,
y1 = x1, y1 = x1,
@ -945,32 +949,9 @@ var Item = Base.extend(Emitter, /** @lends Item# */{
// Do not transform by the internal matrix if there is a internalGetter. // Do not transform by the internal matrix if there is a internalGetter.
var _matrix = internalGetter ? null : this._matrix.orNullIfIdentity(), var _matrix = internalGetter ? 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
// cached bounds that depend on this item. We store this in the parent,
// for multiple reasons:
// The parent receives CHILDREN change notifications for when its
// children are added or removed and can thus clear the cache, and we
// save a lot of memory, e.g. when grouping 100 items and asking the
// group for its bounds. If stored on the children, we would have 100
// times the same structure.
// 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.
var cacheParent = this._parent || this._parentSymbol; Item._updateBoundsCache(this._parent || this._parentSymbol, cacheItem);
if (cacheParent) {
// Set-up the parent's boundsCache structure if it does not
// exist yet and add the cacheItem to it.
var id = cacheItem._id,
ref = cacheParent._boundsCache = cacheParent._boundsCache || {
// Use both a hash-table for ids and an array for the list,
// so we can keep track of items that were added already
ids: {},
list: []
};
if (!ref.ids[id]) {
ref.list.push(cacheItem);
ref.ids[id] = cacheItem;
}
}
if (cache && this._bounds && this._bounds[cache]) if (cache && this._bounds && this._bounds[cache])
return this._bounds[cache].clone(); return this._bounds[cache].clone();
// 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
@ -993,9 +974,37 @@ var Item = Base.extend(Emitter, /** @lends Item# */{
}, },
statics: { statics: {
/**
* Set up a boundsCache structure that keeps track of items that keep
* cached bounds that depend on this item. We store this in the parent,
* for multiple reasons:
* The parent receives CHILDREN change notifications for when its
* children are added or removed and can thus clear the cache, and we
* save a lot of memory, e.g. when grouping 100 items and asking the
* group for its bounds. If stored on the children, we would have 100
* times the same structure.
*/
_updateBoundsCache: function(parent, item) {
if (parent) {
// Set-up the parent's boundsCache structure if it does not
// exist yet and add the item to it.
var id = item._id,
ref = parent._boundsCache = parent._boundsCache || {
// Use a hash-table for ids and an array for the list,
// so we can keep track of items that were added already
ids: {},
list: []
};
if (!ref.ids[id]) {
ref.list.push(item);
ref.ids[id] = item;
}
}
},
/** /**
* Clears cached bounds of all items that the children of this item are * Clears cached bounds of all items that the children of this item are
* contributing to. See #_getCachedBounds() for an explanation why this * contributing to. See _updateBoundsCache() for an explanation why this
* information is stored on parents, not the children themselves. * information is stored on parents, not the children themselves.
*/ */
_clearBoundsCache: function(item) { _clearBoundsCache: function(item) {
@ -1005,7 +1014,7 @@ var Item = Base.extend(Emitter, /** @lends Item# */{
if (cache) { if (cache) {
// Erase cache before looping, to prevent circular recursion. // Erase cache before looping, to prevent circular recursion.
item._bounds = item._position = item._boundsCache = undefined; item._bounds = item._position = item._boundsCache = undefined;
for (var i = 0, list = cache.list, l = list.length; i < l; i++) { for (var i = 0, list = cache.list, l = list.length; i < l; i++){
var other = list[i]; var other = list[i];
if (other !== item) { if (other !== item) {
other._bounds = other._position = undefined; other._bounds = other._position = undefined;

View file

@ -350,10 +350,8 @@ var Raster = Item.extend(/** @lends Raster# */{
// it's actually loaded and we give the code time to install event. // it's actually loaded and we give the code time to install event.
setTimeout(loaded, 0); setTimeout(loaded, 0);
} else { } else {
// Trigger the onLoad event on the image once it's loaded // Trigger the load event on the image once it's loaded
DomEvent.add(image, { DomEvent.add(image, { load: loaded });
load: loaded
});
// A new image created above? Set the source now. // A new image created above? Set the source now.
if (!image.src) if (!image.src)
image.src = src; image.src = src;