From bebdcec7f104a1ef953516d8f3b5d8878b4733be Mon Sep 17 00:00:00 2001 From: MinhHNguyen Date: Thu, 24 Aug 2017 16:30:29 -0700 Subject: [PATCH] Fix: IE11 image size default (#333) --- src/lib/viewers/image/ImageBaseViewer.js | 64 ++++++++++++------- src/lib/viewers/image/ImageViewer.js | 14 ++-- .../image/__tests__/ImageBaseViewer-test.js | 57 +++++++++++------ 3 files changed, 86 insertions(+), 49 deletions(-) diff --git a/src/lib/viewers/image/ImageBaseViewer.js b/src/lib/viewers/image/ImageBaseViewer.js index 95f9d96bc..77ae4e189 100644 --- a/src/lib/viewers/image/ImageBaseViewer.js +++ b/src/lib/viewers/image/ImageBaseViewer.js @@ -3,6 +3,7 @@ import Controls from '../../Controls'; import BaseViewer from '../BaseViewer'; import Browser from '../../Browser'; import { ICON_ZOOM_IN, ICON_ZOOM_OUT } from '../../icons/icons'; +import { get } from '../../util'; import { CLASS_INVISIBLE } from '../../constants'; @@ -43,13 +44,17 @@ class ImageBaseViewer extends BaseViewer { return; } - this.setOriginalImageSize(this.imageEl); - this.loadUI(); - this.zoom(); - - this.imageEl.classList.remove(CLASS_INVISIBLE); - this.loaded = true; - this.emit('load'); + const loadOriginalDimensions = this.setOriginalImageSize(this.imageEl); + loadOriginalDimensions + .then(() => { + this.loadUI(); + this.zoom(); + + this.imageEl.classList.remove(CLASS_INVISIBLE); + this.loaded = true; + this.emit('load'); + }) + .catch(this.errorHandler); } /** @@ -177,27 +182,38 @@ class ImageBaseViewer extends BaseViewer { * @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; + if (imageEl.naturalWidth && imageEl.naturalHeight) { + imageEl.setAttribute('originalWidth', imageEl.naturalWidth); + imageEl.setAttribute('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; + // Case when natural dimensions are not assigned + // By default, assigned width and height in Chrome/Safari/Firefox will be 300x150. + // IE11 workaround. Dimensions only displayed if the image is attached to the document. + get(imageEl.src, {}, 'text') + .then((imageAsText) => { + const parser = new DOMParser(); + const svgEl = parser.parseFromString(imageAsText, 'image/svg+xml'); + + try { + // Assume svgEl is an instanceof an SVG with a viewBox and preserveAspectRatio of meet + // where the height is the limiting axis + const viewBox = svgEl.documentElement.getAttribute('viewBox'); + const [, , w, h] = viewBox.split(' '); + const aspectRatio = h ? w / h : w; + imageEl.setAttribute('originalWidth', Math.round(aspectRatio * 150)); + imageEl.setAttribute('originalHeight', 150); + resolve(); + } catch (e) { + // Assume 300x150 that chrome does by default + imageEl.setAttribute('originalWidth', 300); + imageEl.setAttribute('originalHeight', 150); + resolve(e); + } + }) + .catch(reject); } }); diff --git a/src/lib/viewers/image/ImageViewer.js b/src/lib/viewers/image/ImageViewer.js index e4107e91d..490ba0f7f 100644 --- a/src/lib/viewers/image/ImageViewer.js +++ b/src/lib/viewers/image/ImageViewer.js @@ -183,10 +183,14 @@ 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.originalHeight : this.imageEl.originalWidth; + const originalWidth = this.isRotated() + ? this.imageEl.getAttribute('originalHeight') + : this.imageEl.getAttribute('originalWidth'); newWidth = Math.min(viewport.width, originalWidth); } else { - const originalHeight = this.isRotated() ? this.imageEl.originalWidth : this.imageEl.originalHeight; + const originalHeight = this.isRotated() + ? this.imageEl.getAttribute('originalWidth') + : this.imageEl.getAttribute('originalHeight'); newHeight = Math.min(viewport.height, originalHeight); } } @@ -231,7 +235,9 @@ class ImageViewer extends ImageBaseViewer { * @return {void} */ setScale(width, height) { - this.scale = width ? width / this.imageEl.originalWidth : height / this.imageEl.originalHeight; + this.scale = width + ? width / this.imageEl.getAttribute('originalWidth') + : height / this.imageEl.getAttribute('originalHeight'); this.rotationAngle = this.currentRotationAngle % 3600 % 360; this.emit('scale', { scale: this.scale, @@ -390,7 +396,7 @@ class ImageViewer extends ImageBaseViewer { handleOrientationChange() { this.adjustImageZoomPadding(); - this.scale = this.imageEl.clientWidth / this.imageEl.originalWidth; + this.scale = this.imageEl.clientWidth / this.imageEl.getAttribute('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 98aa59848..50c8831f4 100644 --- a/src/lib/viewers/image/__tests__/ImageBaseViewer-test.js +++ b/src/lib/viewers/image/__tests__/ImageBaseViewer-test.js @@ -3,6 +3,7 @@ import ImageBaseViewer from '../ImageBaseViewer'; import BaseViewer from '../../BaseViewer'; import Browser from '../../../Browser'; import fullscreen from '../../../Fullscreen'; +import * as util from '../../../util'; import { ICON_ZOOM_IN, ICON_ZOOM_OUT } from '../../../icons/icons'; const CSS_CLASS_PANNING = 'panning'; @@ -11,8 +12,6 @@ const CSS_CLASS_PANNABLE = 'pannable'; let stubs = {}; -const imageUrl = ''; - const sandbox = sinon.sandbox.create(); let imageBase; let containerEl; @@ -224,31 +223,41 @@ describe('lib/viewers/image/ImageBaseViewer', () => { it('should use the naturalHeight and naturalWidth when available', (done) => { const imageEl = { naturalWidth: 100, - naturalHeight: 100 + naturalHeight: 100, + setAttribute: (name, value) => { + imageEl[name] = value; + }, + getAttribute: (name) => imageEl[name] }; const promise = imageBase.setOriginalImageSize(imageEl); promise.should.be.fulfilled.then(() => { - expect(imageEl.originalWidth).to.equal(imageEl.naturalWidth); - expect(imageEl.originalHeight).to.equal(imageEl.naturalHeight); + expect(imageEl.getAttribute('originalWidth')).to.equal(imageEl.naturalWidth); + expect(imageEl.getAttribute('originalHeight')).to.equal(imageEl.naturalHeight); done(); + }).catch(() => { + Assert.fail(); }); }); - it('should work when naturalHeight and naturalWidth are undefined', (done) => { + it('should default to 300x150 when naturalHeight and naturalWidth are 0x0', (done) => { const imageEl = { - naturalWidth: undefined, - naturalHeight: undefined, - src: imageUrl + naturalWidth: 0, + naturalHeight: 0, + setAttribute: (name, value) => { + imageEl[name] = value; + }, + getAttribute: (name) => imageEl[name] }; - const imageUrlWidth = 12; - const imageUrlHeight = 12; + const getStub = sandbox.stub(util, 'get').returns(Promise.resolve('not real a image')); const promise = imageBase.setOriginalImageSize(imageEl); promise.should.be.fulfilled.then(() => { - expect(imageEl.originalWidth).to.equal(imageUrlWidth); - expect(imageEl.originalHeight).to.equal(imageUrlHeight); + expect(imageEl.getAttribute('originalWidth')).to.equal(300); + expect(imageEl.getAttribute('originalHeight')).to.equal(150); done(); + }).catch(() => { + Assert.fail(); }); }); }); @@ -532,14 +541,15 @@ describe('lib/viewers/image/ImageBaseViewer', () => { describe('finishLoading()', () => { beforeEach(() => { imageBase.loaded = false; - stubs.emit = sandbox.stub(imageBase, 'emit'); stubs.zoom = sandbox.stub(imageBase, 'zoom'); stubs.loadUI = sandbox.stub(imageBase, 'loadUI'); stubs.setOriginalImageSize = sandbox.stub(imageBase, 'setOriginalImageSize'); + stubs.errorHandler = sandbox.stub(imageBase, 'errorHandler'); }); it('should do nothing if already destroyed', () => { imageBase.destroyed = true; + stubs.emit = sandbox.stub(imageBase, 'emit'); imageBase.finishLoading(); expect(imageBase.loaded).to.be.false; @@ -547,17 +557,22 @@ describe('lib/viewers/image/ImageBaseViewer', () => { expect(stubs.zoom).to.not.have.been.called; expect(stubs.setOriginalImageSize).to.not.have.been.called; expect(stubs.loadUI).to.not.have.been.called; + expect(stubs.errorHandler).to.not.have.been.called; }); - it('should load UI if not destroyed', () => { - imageBase.finishLoading(); + it('should load UI if not destroyed', (done) => { + imageBase.on('load', () => { + expect(stubs.errorHandler).to.not.have.been.called; + expect(imageBase.loaded).to.be.true; + expect(stubs.zoom).to.have.been.called; + expect(stubs.loadUI).to.have.been.called; + done(); + }); + stubs.setOriginalImageSize.returns(Promise.resolve()); + imageBase.destroyed = false; - expect(imageBase.loaded).to.be.true; - expect(stubs.emit).to.have.been.calledWith('load'); - expect(stubs.emit).to.have.been.calledWith('load'); + imageBase.finishLoading(); expect(stubs.setOriginalImageSize).to.have.been.called; - expect(stubs.zoom).to.have.been.called; - expect(stubs.loadUI).to.have.been.called; }); });