From c5ad9b25bf67d381e217a0183448f74bd20b584d Mon Sep 17 00:00:00 2001 From: zakharov-aa Date: Wed, 28 Sep 2016 09:40:28 +0300 Subject: [PATCH 01/14] IE8 fix for tileSources type: 'image' --- src/imagetilesource.js | 15 +++++++++++---- 1 file changed, 11 insertions(+), 4 deletions(-) diff --git a/src/imagetilesource.js b/src/imagetilesource.js index 3562bf50..5ae9e869 100644 --- a/src/imagetilesource.js +++ b/src/imagetilesource.js @@ -72,6 +72,13 @@ $.TileSource.apply(this, [options]); }; + + /* IE8 fix for tileSources type: 'image' */ + $.getNatural = function (DOMelement) { + var img = new Image(); + img.src = DOMelement.src; + return { width: img.width, height: img.height }; + }; $.extend($.ImageTileSource.prototype, $.TileSource.prototype, /** @lends OpenSeadragon.ImageTileSource.prototype */{ /** @@ -114,8 +121,8 @@ } $.addEvent(image, 'load', function () { - _this.width = image.naturalWidth; - _this.height = image.naturalHeight; + _this.width = $.getNatural(image).width; + _this.height = $.getNatural(image).height; _this.aspectRatio = _this.width / _this.height; _this.dimensions = new $.Point(_this.width, _this.height); _this._tileWidth = _this.width; @@ -202,8 +209,8 @@ _buildLevels: function () { var levels = [{ url: this._image.src, - width: this._image.naturalWidth, - height: this._image.naturalHeight + width: $.getNatural(this._image).width, + height: $.getNatural(this._image).height }]; if (!this.buildPyramid || !$.supportsCanvas || !this.useCanvas) { From ffa4bbc8df739c6308c4c1258cc8ba7f72546d67 Mon Sep 17 00:00:00 2001 From: zakharov-aa Date: Wed, 28 Sep 2016 10:39:20 +0300 Subject: [PATCH 02/14] IE8 naturalWidth and naturalHeight more clear fix --- src/imagetilesource.js | 23 ++++++++++------------- 1 file changed, 10 insertions(+), 13 deletions(-) diff --git a/src/imagetilesource.js b/src/imagetilesource.js index 5ae9e869..417243c6 100644 --- a/src/imagetilesource.js +++ b/src/imagetilesource.js @@ -72,13 +72,6 @@ $.TileSource.apply(this, [options]); }; - - /* IE8 fix for tileSources type: 'image' */ - $.getNatural = function (DOMelement) { - var img = new Image(); - img.src = DOMelement.src; - return { width: img.width, height: img.height }; - }; $.extend($.ImageTileSource.prototype, $.TileSource.prototype, /** @lends OpenSeadragon.ImageTileSource.prototype */{ /** @@ -121,8 +114,9 @@ } $.addEvent(image, 'load', function () { - _this.width = $.getNatural(image).width; - _this.height = $.getNatural(image).height; + /* IE8 fix since it has no naturalWidth and naturalHeight */ + _this.width = Object.prototype.hasOwnProperty.call(image, 'naturalWidth') ? image.naturalWidth : image.width; + _this.height = Object.prototype.hasOwnProperty.call(image, 'naturalHeight') ? image.naturalHeight : image.height; _this.aspectRatio = _this.width / _this.height; _this.dimensions = new $.Point(_this.width, _this.height); _this._tileWidth = _this.width; @@ -209,8 +203,9 @@ _buildLevels: function () { var levels = [{ url: this._image.src, - width: $.getNatural(this._image).width, - height: $.getNatural(this._image).height + /* IE8 fix since it has no naturalWidth and naturalHeight */ + width: Object.prototype.hasOwnProperty.call(this._image, 'naturalWidth') ? this._image.naturalWidth : this._image.width, + height: Object.prototype.hasOwnProperty.call(this._image, 'naturalHeight') ? this._image.naturalHeight : this._image.height }]; if (!this.buildPyramid || !$.supportsCanvas || !this.useCanvas) { @@ -219,8 +214,10 @@ return levels; } - var currentWidth = this._image.naturalWidth; - var currentHeight = this._image.naturalHeight; + /* IE8 fix since it has no naturalWidth and naturalHeight */ + var currentWidth = Object.prototype.hasOwnProperty.call(this._image, 'naturalWidth') ? this._image.naturalWidth : this._image.width; + var currentHeight = Object.prototype.hasOwnProperty.call(this._image, 'naturalHeight') ? this._image.naturalHeight : this._image.height; + var bigCanvas = document.createElement("canvas"); var bigContext = bigCanvas.getContext("2d"); From e50beb435bc3c2f00665600952a36895968de8f1 Mon Sep 17 00:00:00 2001 From: Andreas Wagner Date: Thu, 20 Oct 2016 11:32:13 +0200 Subject: [PATCH 03/14] Update legacytilesource.js Remove check for filetype based on uri or mimetype strings (maybe the filetype is not known before actually downloading the resource). --- src/legacytilesource.js | 7 +------ 1 file changed, 1 insertion(+), 6 deletions(-) diff --git a/src/legacytilesource.js b/src/legacytilesource.js index 04579034..ad2a0ffe 100644 --- a/src/legacytilesource.js +++ b/src/legacytilesource.js @@ -204,12 +204,7 @@ function filterFiles( files ){ file = files[ i ]; if( file.height && file.width && - file.url && ( - file.url.toLowerCase().match(/^.*\.(png|jpg|jpeg|gif)(?:\?.*)?$/) || ( - file.mimetype && - file.mimetype.toLowerCase().match(/^.*\/(png|jpg|jpeg|gif)$/) - ) - ) ){ + file.url ){ //This is sufficient to serve as a level filtered.push({ url: file.url, From 9e39ab7f5d041d24c0cfe7ee7e69775352fa71a0 Mon Sep 17 00:00:00 2001 From: Ian Gilman Date: Fri, 21 Oct 2016 09:36:59 -0700 Subject: [PATCH 04/14] Changelog for #1041 --- changelog.txt | 1 + 1 file changed, 1 insertion(+) diff --git a/changelog.txt b/changelog.txt index 3d1a4036..6a24a0f1 100644 --- a/changelog.txt +++ b/changelog.txt @@ -16,6 +16,7 @@ OPENSEADRAGON CHANGELOG * Fixed problem with "sparse image" DZI files (#995) * Optimization: Use the squared distance when comparing tiles (#1027) * Fix IndexSizeError on IE and Edge that occurred under certain circumstances (e.g. multi-image with transparency) (#1035) +* ImageTileSource now works in IE8 (#1041) 2.2.1: From 5bfccec7a35a4e4e8f3322cf7e37b4978e9a6b01 Mon Sep 17 00:00:00 2001 From: Antoine Vandecreme Date: Sun, 23 Oct 2016 17:45:39 +0200 Subject: [PATCH 05/14] Disable tile edge smoothing with tiled image rotation --- src/tiledimage.js | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/src/tiledimage.js b/src/tiledimage.js index 90c9222a..3eb9fafa 100644 --- a/src/tiledimage.js +++ b/src/tiledimage.js @@ -1544,7 +1544,9 @@ function drawTiles( tiledImage, lastDrawn ) { var zoom = tiledImage.viewport.getZoom(true); var imageZoom = tiledImage.viewportToImageZoom(zoom); - if (imageZoom > tiledImage.smoothTileEdgesMinZoom && !tiledImage.iOSDevice) { + // TODO: support tile edge smoothing with tiled image rotation. + if (imageZoom > tiledImage.smoothTileEdgesMinZoom && !tiledImage.iOSDevice && + tiledImage.getRotation() == 0) { // When zoomed in a lot (>100%) the tile edges are visible. // So we have to composite them at ~100% and scale them up together. // Note: Disabled on iOS devices per default as it causes a native crash From 77310b0229b2853b835c8c872d14f841735b1d78 Mon Sep 17 00:00:00 2001 From: Antoine Vandecreme Date: Sun, 23 Oct 2016 18:25:14 +0200 Subject: [PATCH 06/14] Remove TODO --- src/tiledimage.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/tiledimage.js b/src/tiledimage.js index 3eb9fafa..0d155325 100644 --- a/src/tiledimage.js +++ b/src/tiledimage.js @@ -959,7 +959,7 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag var targetRenderPixelRatio = viewport.deltaPixelsFromPointsNoRotate( this.source.getPixelRatio(level), false - ).x * this._scaleSpring.current.value; //TODO: shouldn't that be target.value? + ).x * this._scaleSpring.current.value; var targetZeroRatio = viewport.deltaPixelsFromPointsNoRotate( this.source.getPixelRatio( @@ -969,7 +969,7 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag ) ), false - ).x * this._scaleSpring.current.value; //TODO: shouldn't that be target.value? + ).x * this._scaleSpring.current.value; var optimalRatio = this.immediateRender ? 1 : targetZeroRatio; var levelOpacity = Math.min(1, (currentRenderPixelRatio - 0.5) / 0.5); From a9b60057ea990c6c9ec0c63f28bc7c77327fa8a8 Mon Sep 17 00:00:00 2001 From: Antoine Vandecreme Date: Sun, 23 Oct 2016 22:25:16 +0200 Subject: [PATCH 07/14] Fix wrapping. --- src/tiledimage.js | 21 ++++++++++----------- src/tilesource.js | 11 +++++++++-- 2 files changed, 19 insertions(+), 13 deletions(-) diff --git a/src/tiledimage.js b/src/tiledimage.js index 0d155325..992cc895 100644 --- a/src/tiledimage.js +++ b/src/tiledimage.js @@ -1058,16 +1058,12 @@ function updateLevel(tiledImage, haveDrawn, drawLevel, level, levelOpacity, resetCoverage(tiledImage.coverage, level); - if (tiledImage.wrapHorizontal) { - topLeftTile.x -= 1; // left invisible column (othervise we will have empty space after scroll at left) - } else { + if (!tiledImage.wrapHorizontal) { // Adjust for floating point error topLeftTile.x = Math.max(topLeftTile.x, 0); bottomRightTile.x = Math.min(bottomRightTile.x, numberOfTiles.x - 1); } - if (tiledImage.wrapVertical) { - topLeftTile.y -= 1; // top invisible row (othervise we will have empty space after scroll at top) - } else { + if (!tiledImage.wrapVertical) { // Adjust for floating point error topLeftTile.y = Math.max(topLeftTile.y, 0); bottomRightTile.y = Math.min(bottomRightTile.y, numberOfTiles.y - 1); @@ -1078,11 +1074,14 @@ function updateLevel(tiledImage, haveDrawn, drawLevel, level, levelOpacity, for (var x = topLeftTile.x; x <= bottomRightTile.x; x++) { for (var y = topLeftTile.y; y <= bottomRightTile.y; y++) { - var tileBounds = tiledImage.source.getTileBounds(level, x, y); - - if (drawArea.intersection(tileBounds) === null) { - // This tile is outside of the viewport, no need to draw it - continue; + // Optimisation disabled with wrapping because getTileBounds does not + // work correctly with x and y outside of the number of tiles + if (!tiledImage.wrapHorizontal && !tiledImage.wrapVertical) { + var tileBounds = tiledImage.source.getTileBounds(level, x, y); + if (drawArea.intersection(tileBounds) === null) { + // This tile is outside of the viewport, no need to draw it + continue; + } } best = updateTile( diff --git a/src/tilesource.js b/src/tilesource.js index fd0084c4..fa52f61e 100644 --- a/src/tilesource.js +++ b/src/tilesource.js @@ -346,10 +346,17 @@ $.TileSource.prototype = { */ getTileAtPoint: function(level, point) { var widthScaled = this.dimensions.x * this.getLevelScale(level); - var pixelX = point.x * widthScaled; - var pixelY = point.y * widthScaled; + var pixelX = $.positiveModulo(point.x, 1) * widthScaled; + var pixelY = $.positiveModulo(point.y, 1 / this.aspectRatio) * widthScaled; + var x = Math.floor(pixelX / this.getTileWidth()); var y = Math.floor(pixelY / this.getTileHeight()); + + // Fix for wrapping + var numTiles = this.getNumTiles(level); + x += numTiles.x * Math.floor(point.x); + y += numTiles.y * Math.floor(point.y * this.aspectRatio); + return new $.Point(x, y); }, From 824c9ab2115a52cab718cb83ee4c3f465754c99f Mon Sep 17 00:00:00 2001 From: Ian Gilman Date: Mon, 24 Oct 2016 10:13:52 -0700 Subject: [PATCH 08/14] Changelog for #1056 --- changelog.txt | 1 + 1 file changed, 1 insertion(+) diff --git a/changelog.txt b/changelog.txt index 6a24a0f1..f8ff41a1 100644 --- a/changelog.txt +++ b/changelog.txt @@ -17,6 +17,7 @@ OPENSEADRAGON CHANGELOG * Optimization: Use the squared distance when comparing tiles (#1027) * Fix IndexSizeError on IE and Edge that occurred under certain circumstances (e.g. multi-image with transparency) (#1035) * ImageTileSource now works in IE8 (#1041) +* LegacyTileSource now allows any image URLs regardless of type (#1056) 2.2.1: From b992d545721c9343c7db70285c93bab6ba2cac09 Mon Sep 17 00:00:00 2001 From: Antoine Vandecreme Date: Mon, 24 Oct 2016 21:34:29 +0200 Subject: [PATCH 09/14] Fix jshint error --- src/tiledimage.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/tiledimage.js b/src/tiledimage.js index 992cc895..12d62950 100644 --- a/src/tiledimage.js +++ b/src/tiledimage.js @@ -1545,7 +1545,7 @@ function drawTiles( tiledImage, lastDrawn ) { var imageZoom = tiledImage.viewportToImageZoom(zoom); // TODO: support tile edge smoothing with tiled image rotation. if (imageZoom > tiledImage.smoothTileEdgesMinZoom && !tiledImage.iOSDevice && - tiledImage.getRotation() == 0) { + tiledImage.getRotation() === 0) { // When zoomed in a lot (>100%) the tile edges are visible. // So we have to composite them at ~100% and scale them up together. // Note: Disabled on iOS devices per default as it causes a native crash From 5ac1502ccd03cc857f5398225cc8f52c1344853d Mon Sep 17 00:00:00 2001 From: Antoine Vandecreme Date: Mon, 24 Oct 2016 22:03:31 +0200 Subject: [PATCH 10/14] Take pixelDensityRatio into account when rotating --- src/drawer.js | 22 +++++++++++++--------- src/tiledimage.js | 30 ++++++++++++++++++------------ 2 files changed, 31 insertions(+), 21 deletions(-) diff --git a/src/drawer.js b/src/drawer.js index 7029320d..9440c280 100644 --- a/src/drawer.js +++ b/src/drawer.js @@ -477,13 +477,14 @@ $.Drawer.prototype = { context.fillStyle = this.debugGridColor; if ( this.viewport.degrees !== 0 ) { - this._offsetForRotation(this.viewport.degrees); + this._offsetForRotation({degrees: this.viewport.degrees}); } if (tiledImage.getRotation() !== 0) { - this._offsetForRotation( - tiledImage.getRotation(), - tiledImage.viewport.pixelFromPointNoRotate( - tiledImage._getRotationPoint(true), true)); + this._offsetForRotation({ + degrees: tiledImage.getRotation(), + point: tiledImage.viewport.pixelFromPointNoRotate( + tiledImage._getRotationPoint(true), true) + }); } context.strokeRect( @@ -588,13 +589,16 @@ $.Drawer.prototype = { }, // private - _offsetForRotation: function(degrees, point, useSketch) { - point = point || this.getCanvasCenter(); - var context = this._getContext(useSketch); + _offsetForRotation: function(options) { + var point = options.point ? + options.point.times($.pixelDensityRatio) : + this.getCanvasCenter(); + + var context = this._getContext(options.useSketch); context.save(); context.translate(point.x, point.y); - context.rotate(Math.PI / 180 * degrees); + context.rotate(Math.PI / 180 * options.degrees); context.translate(-point.x, -point.y); }, diff --git a/src/tiledimage.js b/src/tiledimage.js index 12d62950..c6659c77 100644 --- a/src/tiledimage.js +++ b/src/tiledimage.js @@ -1573,15 +1573,18 @@ function drawTiles( tiledImage, lastDrawn ) { // avoid interpolation if (!sketchScale) { if (tiledImage.viewport.degrees !== 0) { - tiledImage._drawer._offsetForRotation( - tiledImage.viewport.degrees, useSketch); + tiledImage._drawer._offsetForRotation({ + degrees: tiledImage.viewport.degrees, + useSketch: useSketch + }); } if (tiledImage._degrees !== 0) { - tiledImage._drawer._offsetForRotation( - tiledImage._degrees, - tiledImage.viewport.pixelFromPointNoRotate( + tiledImage._drawer._offsetForRotation({ + degrees: tiledImage._degrees, + point: tiledImage.viewport.pixelFromPointNoRotate( tiledImage._getRotationPoint(true), true), - useSketch); + useSketch: useSketch + }); } } @@ -1663,15 +1666,18 @@ function drawTiles( tiledImage, lastDrawn ) { if (useSketch) { if (sketchScale) { if (tiledImage.viewport.degrees !== 0) { - tiledImage._drawer._offsetForRotation( - tiledImage.viewport.degrees, false); + tiledImage._drawer._offsetForRotation({ + degrees: tiledImage.viewport.degrees, + useSketch: false + }); } if (tiledImage._degrees !== 0) { - tiledImage._drawer._offsetForRotation( - tiledImage._degrees, - tiledImage.viewport.pixelFromPointNoRotate( + tiledImage._drawer._offsetForRotation({ + degrees: tiledImage._degrees, + point: tiledImage.viewport.pixelFromPointNoRotate( tiledImage._getRotationPoint(true), true), - false); + useSketch: false + }); } } tiledImage._drawer.blendSketch({ From 4b487170104a577ebafbab9f53e4f7b2f758f72c Mon Sep 17 00:00:00 2001 From: Antoine Vandecreme Date: Tue, 25 Oct 2016 21:41:42 +0200 Subject: [PATCH 11/14] Rollback sketchCanvas scaling when tiled image rotated --- src/drawer.js | 4 +--- src/tiledimage.js | 16 ++-------------- src/world.js | 10 ---------- 3 files changed, 3 insertions(+), 27 deletions(-) diff --git a/src/drawer.js b/src/drawer.js index 9440c280..73772a88 100644 --- a/src/drawer.js +++ b/src/drawer.js @@ -323,7 +323,6 @@ $.Drawer.prototype = { this.sketchCanvas.height = sketchCanvasSize.y; this.sketchContext = this.sketchCanvas.getContext( "2d" ); - // FIXME: should check if any tiled image get rotated as well. // If the viewport is not currently rotated, the sketchCanvas // will have the same size as the main canvas. However, if // the viewport get rotated later on, we will need to resize it. @@ -621,8 +620,7 @@ $.Drawer.prototype = { // private _calculateSketchCanvasSize: function() { var canvasSize = this._calculateCanvasSize(); - if (this.viewport.getRotation() === 0 && - !this.viewer.world._hasRotatedItem()) { + if (this.viewport.getRotation() === 0) { return canvasSize; } // If the viewport is rotated, we need a larger sketch canvas in order diff --git a/src/tiledimage.js b/src/tiledimage.js index c6659c77..ca6b8eeb 100644 --- a/src/tiledimage.js +++ b/src/tiledimage.js @@ -550,20 +550,8 @@ $.extend($.TiledImage.prototype, $.EventSource.prototype, /** @lends OpenSeadrag }, // private - // Convert rectangle in tiled image coordinates to viewport coordinates. - _tiledImageToViewportRectangle: function(rect) { - var scale = this._scaleSpring.current.value; - return new $.Rect( - rect.x * scale + this._xSpring.current.value, - rect.y * scale + this._ySpring.current.value, - rect.width * scale, - rect.height * scale, - rect.degrees) - .rotate(this.getRotation(), this._getRotationPoint(true)); - }, - - // private - // Convert rectangle in viewport coordinates to tiled image coordinates. + // Convert rectangle in viewport coordinates to this tiled image point + // coordinates (x in [0, 1] and y in [0, aspectRatio]) _viewportToTiledImageRectangle: function(rect) { var scale = this._scaleSpring.current.value; rect = rect.rotate(-this.getRotation(), this._getRotationPoint(true)); diff --git a/src/world.js b/src/world.js index ad6285d9..e68590b3 100644 --- a/src/world.js +++ b/src/world.js @@ -436,16 +436,6 @@ $.extend( $.World.prototype, $.EventSource.prototype, /** @lends OpenSeadragon.W * @property {?Object} userData - Arbitrary subscriber-defined object. */ this.raiseEvent( 'remove-item', { item: item } ); - }, - - // private - _hasRotatedItem: function() { - for (var i = 0; i < this._items.length; i++) { - if (this._items[i].getRotation() !== 0) { - return true; - } - } - return false; } }); From 8240db3546b05f509035f15cc6eca5cffd9bcdaa Mon Sep 17 00:00:00 2001 From: Ian Gilman Date: Wed, 26 Oct 2016 09:59:20 -0700 Subject: [PATCH 12/14] changelog for #1006 --- changelog.txt | 1 + 1 file changed, 1 insertion(+) diff --git a/changelog.txt b/changelog.txt index f8ff41a1..7b85bf7f 100644 --- a/changelog.txt +++ b/changelog.txt @@ -4,6 +4,7 @@ OPENSEADRAGON CHANGELOG 2.2.2: (in progress) * BREAKING CHANGE: Tile.distance has been removed (#1027) +* You can now set the rotation of individual tiled images (#1006) * Fixed CORS bug in IE 10 (#967) * Added support for commonjs (#984) * Added an option to addTiledImage to change the crossOriginPolicy (#981) From 04eafa92756975cc2994185cd72c79cd1a930750 Mon Sep 17 00:00:00 2001 From: Edminsson Date: Mon, 31 Oct 2016 09:20:50 +0100 Subject: [PATCH 13/14] Edge smoothing and IE8 #1063 --- src/tiledimage.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/tiledimage.js b/src/tiledimage.js index b0969e44..e8092083 100644 --- a/src/tiledimage.js +++ b/src/tiledimage.js @@ -1533,7 +1533,7 @@ function drawTiles( tiledImage, lastDrawn ) { var imageZoom = tiledImage.viewportToImageZoom(zoom); // TODO: support tile edge smoothing with tiled image rotation. if (imageZoom > tiledImage.smoothTileEdgesMinZoom && !tiledImage.iOSDevice && - tiledImage.getRotation() === 0) { + tiledImage.getRotation() === 0 && $.supportsCanvas) { // When zoomed in a lot (>100%) the tile edges are visible. // So we have to composite them at ~100% and scale them up together. // Note: Disabled on iOS devices per default as it causes a native crash From 0a292491cc2159daf1bd26bc7538e6d4c2bbff6f Mon Sep 17 00:00:00 2001 From: Ian Gilman Date: Mon, 31 Oct 2016 09:57:44 -0700 Subject: [PATCH 14/14] Changelog for #1064 --- changelog.txt | 1 + 1 file changed, 1 insertion(+) diff --git a/changelog.txt b/changelog.txt index 7b85bf7f..99ee6973 100644 --- a/changelog.txt +++ b/changelog.txt @@ -19,6 +19,7 @@ OPENSEADRAGON CHANGELOG * Fix IndexSizeError on IE and Edge that occurred under certain circumstances (e.g. multi-image with transparency) (#1035) * ImageTileSource now works in IE8 (#1041) * LegacyTileSource now allows any image URLs regardless of type (#1056) +* Fixed error in IE8 when zooming in (due to edge smoothing) (#1064) 2.2.1: