diff --git a/src/openseadragon.js b/src/openseadragon.js index 837595aa..5d17b7f8 100644 --- a/src/openseadragon.js +++ b/src/openseadragon.js @@ -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, diff --git a/src/tiledimage.js b/src/tiledimage.js index 93b361f6..fb6a12ac 100644 --- a/src/tiledimage.js +++ b/src/tiledimage.js @@ -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) { /** * - Needs documentation - @@ -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 diff --git a/src/viewer.js b/src/viewer.js index 5d2d52bc..17c8184c 100644 --- a/src/viewer.js +++ b/src/viewer.js @@ -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,