mirror of
https://github.com/scratchfoundation/paper.js.git
synced 2025-01-03 19:45:44 -05:00
Implement caching of internal, untransformed bounds.
This commit is contained in:
parent
5197dd81c5
commit
e238d23194
4 changed files with 28 additions and 24 deletions
|
@ -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
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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();
|
||||||
|
|
|
@ -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);
|
||||||
|
|
Loading…
Reference in a new issue