mirror of
https://github.com/openseadragon/openseadragon.git
synced 2025-04-03 13:53:31 +03:00
Merge pull request #2686 from MichaelWGibson/animation-dest-tile-loading
During animations, load tiles in the destination region
This commit is contained in:
commit
c386eca94d
3 changed files with 164 additions and 54 deletions
|
@ -355,6 +355,9 @@
|
|||
* Specifies the animation duration per each {@link OpenSeadragon.Spring}
|
||||
* which occur when the image is dragged, zoomed or rotated.
|
||||
*
|
||||
* @property {Boolean} [loadDestinationTilesOnAnimation=true]
|
||||
* If true, tiles are loaded only at the destination of an animation.
|
||||
* If false, tiles are loaded along the animation path during the animation.
|
||||
* @property {OpenSeadragon.GestureSettings} [gestureSettingsMouse]
|
||||
* Settings for gestures generated by a mouse pointer device. (See {@link OpenSeadragon.GestureSettings})
|
||||
* @property {Boolean} [gestureSettingsMouse.dragToPan=true] - Pan on drag gesture
|
||||
|
@ -1275,6 +1278,7 @@ function OpenSeadragon( options ){
|
|||
dblClickDistThreshold: 20,
|
||||
springStiffness: 6.5,
|
||||
animationTime: 1.2,
|
||||
loadDestinationTilesOnAnimation: true,
|
||||
gestureSettingsMouse: {
|
||||
dragToPan: true,
|
||||
scrollToZoom: true,
|
||||
|
|
|
@ -176,6 +176,7 @@ $.TiledImage = function( options ) {
|
|||
wrapHorizontal: $.DEFAULT_SETTINGS.wrapHorizontal,
|
||||
wrapVertical: $.DEFAULT_SETTINGS.wrapVertical,
|
||||
immediateRender: $.DEFAULT_SETTINGS.immediateRender,
|
||||
loadDestinationTilesOnAnimation: $.DEFAULT_SETTINGS.loadDestinationTilesOnAnimation,
|
||||
blendTime: $.DEFAULT_SETTINGS.blendTime,
|
||||
alwaysBlend: $.DEFAULT_SETTINGS.alwaysBlend,
|
||||
minPixelRatio: $.DEFAULT_SETTINGS.minPixelRatio,
|
||||
|
@ -1084,6 +1085,13 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag
|
|||
return drawArea;
|
||||
},
|
||||
|
||||
getLoadArea: function() {
|
||||
var viewport = this.viewport;
|
||||
var bounds = viewport.getBounds(false);
|
||||
var drawArea = bounds.getBoundingBox();
|
||||
return drawArea;
|
||||
},
|
||||
|
||||
/**
|
||||
*
|
||||
* @returns {Array} Array of Tiles that make up the current view
|
||||
|
@ -1347,6 +1355,11 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag
|
|||
var highestLevel = levelsInterval.highestLevel; // the highest level we should draw at our current zoom
|
||||
var bestTiles = [];
|
||||
var drawArea = this.getDrawArea();
|
||||
var loadArea = drawArea;
|
||||
|
||||
if (this.loadDestinationTilesOnAnimation) {
|
||||
loadArea = this.getLoadArea();
|
||||
}
|
||||
var currentTime = $.now();
|
||||
|
||||
// reset each tile's beingDrawn flag
|
||||
|
@ -1434,6 +1447,7 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag
|
|||
levelOpacity,
|
||||
levelVisibility,
|
||||
drawArea,
|
||||
loadArea,
|
||||
currentTime,
|
||||
bestTiles
|
||||
);
|
||||
|
@ -1586,16 +1600,16 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag
|
|||
* @param {Number} levelOpacity
|
||||
* @param {Number} levelVisibility
|
||||
* @param {OpenSeadragon.Rect} drawArea
|
||||
* @param {OpenSeadragon.Rect} loadArea
|
||||
* @param {Number} currentTime
|
||||
* @param {OpenSeadragon.Tile[]} best Array of the current best tiles
|
||||
* @returns {Object} Dictionary {bestTiles: OpenSeadragon.Tile - the current "best" tiles to draw, updatedTiles: OpenSeadragon.Tile) - the updated tiles}.
|
||||
*/
|
||||
_updateLevel: function(level, levelOpacity,
|
||||
levelVisibility, drawArea, currentTime, best) {
|
||||
|
||||
var topLeftBound = drawArea.getBoundingBox().getTopLeft();
|
||||
var bottomRightBound = drawArea.getBoundingBox().getBottomRight();
|
||||
levelVisibility, drawArea, loadArea, currentTime, best) {
|
||||
|
||||
var drawTopLeftBound = drawArea.getBoundingBox().getTopLeft();
|
||||
var drawBottomRightBound = drawArea.getBoundingBox().getBottomRight();
|
||||
if (this.viewer) {
|
||||
/**
|
||||
* <em>- Needs documentation -</em>
|
||||
|
@ -1623,24 +1637,42 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag
|
|||
opacity: levelOpacity,
|
||||
visibility: levelVisibility,
|
||||
drawArea: drawArea,
|
||||
topleft: topLeftBound,
|
||||
bottomright: bottomRightBound,
|
||||
topleft: drawTopLeftBound,
|
||||
bottomright: drawBottomRightBound,
|
||||
currenttime: currentTime,
|
||||
best: best
|
||||
});
|
||||
}
|
||||
|
||||
this._resetCoverage(this.coverage, level);
|
||||
this._resetCoverage(this.loadingCoverage, level);
|
||||
var updatedTiles = this._updateDrawArea(level,
|
||||
levelVisibility, drawArea, currentTime);
|
||||
|
||||
var bestTiles = this._updateLoadArea(level, loadArea, currentTime, best);
|
||||
|
||||
return {
|
||||
bestTiles: bestTiles,
|
||||
updatedTiles: updatedTiles
|
||||
};
|
||||
},
|
||||
|
||||
/**
|
||||
* Visit all tiles in an a given area on a given level.
|
||||
* @private
|
||||
* @param {Number} level
|
||||
* @param {OpenSeadragon.Rect} area
|
||||
* @param {Function} callback - TiledImage, x, y, total
|
||||
*/
|
||||
_visitTiles: function(level, area, callback) {
|
||||
var topLeftBound = area.getBoundingBox().getTopLeft();
|
||||
var bottomRightBound = area.getBoundingBox().getBottomRight();
|
||||
|
||||
var drawCornerTiles = this._getCornerTiles(level, topLeftBound, bottomRightBound);
|
||||
var drawTopLeftTile = drawCornerTiles.topLeft;
|
||||
var drawBottomRightTile = drawCornerTiles.bottomRight;
|
||||
|
||||
|
||||
//OK, a new drawing so do your calculations
|
||||
var cornerTiles = this._getCornerTiles(level, topLeftBound, bottomRightBound);
|
||||
var topLeftTile = cornerTiles.topLeft;
|
||||
var bottomRightTile = cornerTiles.bottomRight;
|
||||
var numberOfTiles = this.source.getNumTiles(level);
|
||||
|
||||
var viewportCenter = this.viewport.pixelFromPoint(this.viewport.getCenter());
|
||||
|
||||
if (this.getFlip()) {
|
||||
// The right-most tile can be narrower than the others. When flipped,
|
||||
// this tile is now on the left. Because it is narrower than the normal
|
||||
|
@ -1648,16 +1680,15 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag
|
|||
// fill the viewport. Fix this by rendering an extra column of tiles. If we
|
||||
// are not wrapping, make sure we never render more than the number of tiles
|
||||
// in the image.
|
||||
bottomRightTile.x += 1;
|
||||
drawBottomRightTile.x += 1;
|
||||
if (!this.wrapHorizontal) {
|
||||
bottomRightTile.x = Math.min(bottomRightTile.x, numberOfTiles.x - 1);
|
||||
drawBottomRightTile.x = Math.min(drawBottomRightTile.x, numberOfTiles.x - 1);
|
||||
}
|
||||
}
|
||||
var numTiles = Math.max(0, (bottomRightTile.x - topLeftTile.x) * (bottomRightTile.y - topLeftTile.y));
|
||||
var tiles = new Array(numTiles);
|
||||
var tileIndex = 0;
|
||||
for (var x = topLeftTile.x; x <= bottomRightTile.x; x++) {
|
||||
for (var y = topLeftTile.y; y <= bottomRightTile.y; y++) {
|
||||
var numTiles = Math.max(0, (drawBottomRightTile.x - drawTopLeftTile.x) * (drawBottomRightTile.y - drawTopLeftTile.y));
|
||||
|
||||
for (var x = drawTopLeftTile.x; x <= drawBottomRightTile.x; x++) {
|
||||
for (var y = drawTopLeftTile.y; y <= drawBottomRightTile.y; y++) {
|
||||
|
||||
var flippedX;
|
||||
if (this.getFlip()) {
|
||||
|
@ -1667,30 +1698,83 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag
|
|||
flippedX = x;
|
||||
}
|
||||
|
||||
if (drawArea.intersection(this.getTileBounds(level, flippedX, y)) === null) {
|
||||
// This tile is outside of the viewport, no need to draw it
|
||||
if (area.intersection(this.getTileBounds(level, flippedX, y)) === null) {
|
||||
// This tile is not in the draw area
|
||||
continue;
|
||||
}
|
||||
|
||||
var result = this._updateTile(
|
||||
flippedX, y,
|
||||
level,
|
||||
levelVisibility,
|
||||
viewportCenter,
|
||||
numberOfTiles,
|
||||
currentTime,
|
||||
best
|
||||
);
|
||||
best = result.bestTiles;
|
||||
tiles[tileIndex] = result.tile;
|
||||
tileIndex += 1;
|
||||
callback(this, flippedX, y, numTiles);
|
||||
}
|
||||
}
|
||||
|
||||
return {
|
||||
bestTiles: best,
|
||||
updatedTiles: tiles
|
||||
};
|
||||
|
||||
},
|
||||
|
||||
/**
|
||||
* Updates draw information for all tiles at a given level in the area
|
||||
* @private
|
||||
* @param {Number} level
|
||||
* @param {Number} levelOpacity
|
||||
* @param {Number} levelVisibility
|
||||
* @param {OpenSeadragon.Rect} drawArea
|
||||
* @param {Number} currentTime
|
||||
* @param {OpenSeadragon.Tile[]} best Array of the current best tiles
|
||||
* @returns {OpenSeadragon.Tile[]} Updated tiles
|
||||
*/
|
||||
_updateDrawArea: function(level,
|
||||
levelVisibility, drawArea, currentTime) {
|
||||
var numberOfTiles = this.source.getNumTiles(level);
|
||||
var viewportCenter = this.viewport.pixelFromPoint(this.viewport.getCenter());
|
||||
this._resetCoverage(this.coverage, level);
|
||||
|
||||
var tiles = null;
|
||||
var tileIndex = 0;
|
||||
|
||||
this._visitTiles(level, drawArea, function(tiledImage, x, y, total) {
|
||||
if (!tiles) {
|
||||
tiles = new Array(total);
|
||||
}
|
||||
tiles[tileIndex] = tiledImage._updateTile(
|
||||
x, y,
|
||||
level,
|
||||
levelVisibility,
|
||||
viewportCenter,
|
||||
numberOfTiles,
|
||||
currentTime
|
||||
);
|
||||
|
||||
tileIndex += 1;
|
||||
|
||||
});
|
||||
|
||||
return tiles;
|
||||
},
|
||||
|
||||
/**
|
||||
* Updates load information for all tiles at a given level in the area
|
||||
* @private
|
||||
* @param {Number} level
|
||||
* @param {OpenSeadragon.Rect} loadArea
|
||||
* @param {Number} currentTime
|
||||
* @param {OpenSeadragon.Tile[]} best Array of the current best tiles to load
|
||||
* @returns {OpenSeadragon.Tile[]} The new best tiles to load
|
||||
*/
|
||||
_updateLoadArea: function(level, loadArea, currentTime, best) {
|
||||
this._resetCoverage(this.loadingCoverage, level);
|
||||
var numberOfTiles = this.source.getNumTiles(level);
|
||||
|
||||
this._visitTiles(level, loadArea, function(tiledImage, x, y, _) {
|
||||
best = tiledImage._considerTileForLoad(
|
||||
x, y,
|
||||
level,
|
||||
numberOfTiles,
|
||||
currentTime,
|
||||
best
|
||||
);
|
||||
|
||||
});
|
||||
|
||||
return best;
|
||||
},
|
||||
|
||||
/**
|
||||
|
@ -1756,11 +1840,10 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag
|
|||
* @param {OpenSeadragon.Point} viewportCenter
|
||||
* @param {Number} numberOfTiles
|
||||
* @param {Number} currentTime
|
||||
* @param {OpenSeadragon.Tile} best - The current "best" tile to draw.
|
||||
* @returns {Object} Dictionary {bestTiles: OpenSeadragon.Tile[] - the current best tiles, tile: OpenSeadragon.Tile the current tile}
|
||||
* @returns {OpenSeadragon.Tile} the updated Tile
|
||||
*/
|
||||
_updateTile: function( x, y, level,
|
||||
levelVisibility, viewportCenter, numberOfTiles, currentTime, best){
|
||||
levelVisibility, viewportCenter, numberOfTiles, currentTime){
|
||||
|
||||
const tile = this._getTile(
|
||||
x, y,
|
||||
|
@ -1790,19 +1873,12 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag
|
|||
|
||||
this._setCoverage( this.coverage, level, x, y, false );
|
||||
|
||||
var loadingCoverage = tile.loaded || tile.loading || this._isCovered(this.loadingCoverage, level, x, y);
|
||||
this._setCoverage(this.loadingCoverage, level, x, y, loadingCoverage);
|
||||
|
||||
if ( !tile.exists ) {
|
||||
return {
|
||||
bestTiles: best,
|
||||
tile: tile
|
||||
};
|
||||
return tile;
|
||||
}
|
||||
if (tile.loaded && tile.opacity === 1){
|
||||
this._setCoverage( this.coverage, level, x, y, true );
|
||||
}
|
||||
|
||||
this._positionTile(
|
||||
tile,
|
||||
this.source.tileOverlap,
|
||||
|
@ -1811,6 +1887,38 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag
|
|||
levelVisibility
|
||||
);
|
||||
|
||||
return tile;
|
||||
},
|
||||
|
||||
/**
|
||||
* Consider a tile for loading
|
||||
* @private
|
||||
* @param {Number} x
|
||||
* @param {Number} y
|
||||
* @param {Number} level
|
||||
* @param {Number} numberOfTiles
|
||||
* @param {Number} currentTime
|
||||
* @param {OpenSeadragon.Tile[]} best - The current "best" tiles to draw.
|
||||
* @returns {OpenSeadragon.Tile[]} - The updated "best" tiles to draw.
|
||||
*/
|
||||
_considerTileForLoad: function( x, y, level, numberOfTiles, currentTime, best){
|
||||
|
||||
const tile = this._getTile(
|
||||
x, y,
|
||||
level,
|
||||
currentTime,
|
||||
numberOfTiles
|
||||
);
|
||||
|
||||
|
||||
var loadingCoverage = tile.loaded || tile.loading || this._isCovered(this.loadingCoverage, level, x, y);
|
||||
this._setCoverage(this.loadingCoverage, level, x, y, loadingCoverage);
|
||||
|
||||
|
||||
if ( !tile.exists ) {
|
||||
return best;
|
||||
}
|
||||
|
||||
// Try-find will populate tile with data if equal tile exists in system
|
||||
if (!tile.loaded && !tile.loading && this._tryFindTileCacheRecord(tile)) {
|
||||
loadingCoverage = true;
|
||||
|
@ -1827,10 +1935,7 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag
|
|||
}
|
||||
}
|
||||
|
||||
return {
|
||||
bestTiles: best,
|
||||
tile: tile
|
||||
};
|
||||
return best;
|
||||
},
|
||||
|
||||
// private
|
||||
|
|
|
@ -1748,6 +1748,7 @@ $.extend( $.Viewer.prototype, $.EventSource.prototype, $.ControlDock.prototype,
|
|||
wrapHorizontal: _this.wrapHorizontal,
|
||||
wrapVertical: _this.wrapVertical,
|
||||
maxTilesPerFrame: _this.maxTilesPerFrame,
|
||||
loadDestinationTilesOnAnimation: _this.loadDestinationTilesOnAnimation,
|
||||
immediateRender: _this.immediateRender,
|
||||
blendTime: _this.blendTime,
|
||||
alwaysBlend: _this.alwaysBlend,
|
||||
|
|
Loading…
Add table
Reference in a new issue