From 2889b0d14770d2e3a48f37d2fadafb1bb68e7355 Mon Sep 17 00:00:00 2001 From: MinhHNguyen Date: Mon, 21 Aug 2017 16:35:19 -0700 Subject: [PATCH] Fix: Inconsistent natural image dimensions in IE (#324) * Fix: Inconsistent natural image dimensions in IE * Update: use promise in ImageBaseViewer test * Update: make setOriginalImageSize non-static --- src/lib/viewers/image/ImageBaseViewer.js | 37 ++++++++++++++++++ src/lib/viewers/image/ImageViewer.js | 8 ++-- .../image/__tests__/ImageBaseViewer-test.js | 38 +++++++++++++++++++ 3 files changed, 79 insertions(+), 4 deletions(-) diff --git a/src/lib/viewers/image/ImageBaseViewer.js b/src/lib/viewers/image/ImageBaseViewer.js index 5f764d49d..95f9d96bc 100644 --- a/src/lib/viewers/image/ImageBaseViewer.js +++ b/src/lib/viewers/image/ImageBaseViewer.js @@ -43,6 +43,7 @@ class ImageBaseViewer extends BaseViewer { return; } + this.setOriginalImageSize(this.imageEl); this.loadUI(); this.zoom(); @@ -167,6 +168,42 @@ class ImageBaseViewer extends BaseViewer { this.bindControlListeners(); } + /** + * Sets the original image width and height on the img element. Can be removed when + * naturalHeight and naturalWidth attributes work correctly in IE 11. + * + * @private + * @param {Image} imageEl - The image to set the original size attributes on + * @return {Promise} A promise that is resolved if the original image dimensions were set. + */ + setOriginalImageSize(imageEl) { + const image = imageEl; + const promise = new Promise((resolve, reject) => { + // Do not bother loading a new image when the natural size attributes exist + if (imageEl.naturalWidth > 0 && imageEl.naturalHeight > 0) { + image.originalWidth = imageEl.naturalWidth; + image.originalHeight = imageEl.naturalHeight; + resolve(); + } else { + const originalImg = new Image(); + image.originalWidth = 1; + image.originalHeight = 1; + + originalImg.error = () => { + reject(); + }; + originalImg.onload = () => { + image.originalWidth = originalImg.width || 1; + image.originalHeight = originalImg.height || 1; + resolve(); + }; + originalImg.src = imageEl.src; + } + }); + + return promise; + } + //-------------------------------------------------------------------------- // Event Listeners //-------------------------------------------------------------------------- diff --git a/src/lib/viewers/image/ImageViewer.js b/src/lib/viewers/image/ImageViewer.js index 43a091d16..e4107e91d 100644 --- a/src/lib/viewers/image/ImageViewer.js +++ b/src/lib/viewers/image/ImageViewer.js @@ -183,10 +183,10 @@ class ImageViewer extends ImageBaseViewer { // If the image is smaller than the new viewport, zoom up to a // max of the original file size } else if (modifyWidthInsteadOfHeight) { - const originalWidth = this.isRotated() ? this.imageEl.naturalHeight : this.imageEl.naturalWidth; + const originalWidth = this.isRotated() ? this.imageEl.originalHeight : this.imageEl.originalWidth; newWidth = Math.min(viewport.width, originalWidth); } else { - const originalHeight = this.isRotated() ? this.imageEl.naturalWidth : this.imageEl.naturalHeight; + const originalHeight = this.isRotated() ? this.imageEl.originalWidth : this.imageEl.originalHeight; newHeight = Math.min(viewport.height, originalHeight); } } @@ -231,7 +231,7 @@ class ImageViewer extends ImageBaseViewer { * @return {void} */ setScale(width, height) { - this.scale = width ? width / this.imageEl.naturalWidth : height / this.imageEl.naturalHeight; + this.scale = width ? width / this.imageEl.originalWidth : height / this.imageEl.originalHeight; this.rotationAngle = this.currentRotationAngle % 3600 % 360; this.emit('scale', { scale: this.scale, @@ -390,7 +390,7 @@ class ImageViewer extends ImageBaseViewer { handleOrientationChange() { this.adjustImageZoomPadding(); - this.scale = this.imageEl.clientWidth / this.imageEl.naturalWidth; + this.scale = this.imageEl.clientWidth / this.imageEl.originalWidth; this.rotationAngle = this.currentRotationAngle % 3600 % 360; this.emit('scale', { scale: this.scale, diff --git a/src/lib/viewers/image/__tests__/ImageBaseViewer-test.js b/src/lib/viewers/image/__tests__/ImageBaseViewer-test.js index 97c4cd08b..98aa59848 100644 --- a/src/lib/viewers/image/__tests__/ImageBaseViewer-test.js +++ b/src/lib/viewers/image/__tests__/ImageBaseViewer-test.js @@ -11,6 +11,8 @@ const CSS_CLASS_PANNABLE = 'pannable'; let stubs = {}; +const imageUrl = ''; + const sandbox = sinon.sandbox.create(); let imageBase; let containerEl; @@ -218,6 +220,39 @@ describe('lib/viewers/image/ImageBaseViewer', () => { }); }); + describe('setOriginalImageSize()', () => { + it('should use the naturalHeight and naturalWidth when available', (done) => { + const imageEl = { + naturalWidth: 100, + naturalHeight: 100 + }; + + const promise = imageBase.setOriginalImageSize(imageEl); + promise.should.be.fulfilled.then(() => { + expect(imageEl.originalWidth).to.equal(imageEl.naturalWidth); + expect(imageEl.originalHeight).to.equal(imageEl.naturalHeight); + done(); + }); + }); + + it('should work when naturalHeight and naturalWidth are undefined', (done) => { + const imageEl = { + naturalWidth: undefined, + naturalHeight: undefined, + src: imageUrl + }; + + const imageUrlWidth = 12; + const imageUrlHeight = 12; + const promise = imageBase.setOriginalImageSize(imageEl); + promise.should.be.fulfilled.then(() => { + expect(imageEl.originalWidth).to.equal(imageUrlWidth); + expect(imageEl.originalHeight).to.equal(imageUrlHeight); + done(); + }); + }); + }); + describe('bindControlListeners()', () => { it('should add the correct controls', () => { imageBase.controls = { @@ -500,6 +535,7 @@ describe('lib/viewers/image/ImageBaseViewer', () => { stubs.emit = sandbox.stub(imageBase, 'emit'); stubs.zoom = sandbox.stub(imageBase, 'zoom'); stubs.loadUI = sandbox.stub(imageBase, 'loadUI'); + stubs.setOriginalImageSize = sandbox.stub(imageBase, 'setOriginalImageSize'); }); it('should do nothing if already destroyed', () => { @@ -509,6 +545,7 @@ describe('lib/viewers/image/ImageBaseViewer', () => { expect(imageBase.loaded).to.be.false; expect(stubs.emit).to.not.have.been.called; expect(stubs.zoom).to.not.have.been.called; + expect(stubs.setOriginalImageSize).to.not.have.been.called; expect(stubs.loadUI).to.not.have.been.called; }); @@ -518,6 +555,7 @@ describe('lib/viewers/image/ImageBaseViewer', () => { expect(imageBase.loaded).to.be.true; expect(stubs.emit).to.have.been.calledWith('load'); expect(stubs.emit).to.have.been.calledWith('load'); + expect(stubs.setOriginalImageSize).to.have.been.called; expect(stubs.zoom).to.have.been.called; expect(stubs.loadUI).to.have.been.called; });