From ede8e429867042a6e0107204c0b2a4abf1979354 Mon Sep 17 00:00:00 2001 From: Jared Stoffan Date: Wed, 4 Nov 2020 17:30:21 -0800 Subject: [PATCH] feat(controls): Create react core controls for image viewers (#1285) --- src/lib/viewers/image/ImageBaseViewer.js | 12 ++++++- src/lib/viewers/image/ImageControls.tsx | 17 ++++++++++ src/lib/viewers/image/ImageViewer.js | 31 ++++++++++++++--- src/lib/viewers/image/MultiImageControls.tsx | 16 +++++++++ src/lib/viewers/image/MultiImageViewer.js | 31 ++++++++++++++--- .../image/__tests__/ImageBaseViewer-test.js | 24 ++++++++++++-- .../image/__tests__/ImageControls-test.jsx | 24 ++++++++++++++ .../image/__tests__/ImageViewer-test.js | 25 ++++++++++++-- .../__tests__/MultiImageControls-test.jsx | 24 ++++++++++++++ .../image/__tests__/MultiImageViewer-test.js | 33 +++++++++++++++---- 10 files changed, 217 insertions(+), 20 deletions(-) create mode 100644 src/lib/viewers/image/ImageControls.tsx create mode 100644 src/lib/viewers/image/MultiImageControls.tsx create mode 100644 src/lib/viewers/image/__tests__/ImageControls-test.jsx create mode 100644 src/lib/viewers/image/__tests__/MultiImageControls-test.jsx diff --git a/src/lib/viewers/image/ImageBaseViewer.js b/src/lib/viewers/image/ImageBaseViewer.js index 380fc99f3..a4b7fe95e 100644 --- a/src/lib/viewers/image/ImageBaseViewer.js +++ b/src/lib/viewers/image/ImageBaseViewer.js @@ -1,6 +1,7 @@ import BaseViewer from '../BaseViewer'; import Browser from '../../Browser'; import Controls from '../../Controls'; +import ControlsRoot from '../controls/controls-root'; import PreviewError from '../../PreviewError'; import ZoomControls from '../../ZoomControls'; @@ -78,7 +79,12 @@ class ImageBaseViewer extends BaseViewer { const loadOriginalDimensions = this.setOriginalImageSize(this.imageEl); loadOriginalDimensions.then(() => { this.zoom(); - this.loadUI(); + + if (this.options.useReactControls) { + this.loadUIReact(); + } else { + this.loadUI(); + } this.imageEl.classList.remove(CLASS_INVISIBLE); this.loaded = true; @@ -203,6 +209,10 @@ class ImageBaseViewer extends BaseViewer { this.zoomControls.init(this.scale, { onZoomIn: this.zoomIn, onZoomOut: this.zoomOut }); } + loadUIReact() { + this.controls = new ControlsRoot({ containerEl: this.containerEl }); + } + /** * Sets the original image width and height on the img element. Can be removed when * naturalHeight and naturalWidth attributes work correctly in IE 11. diff --git a/src/lib/viewers/image/ImageControls.tsx b/src/lib/viewers/image/ImageControls.tsx new file mode 100644 index 000000000..b4bba1b9f --- /dev/null +++ b/src/lib/viewers/image/ImageControls.tsx @@ -0,0 +1,17 @@ +import React from 'react'; +import ControlsBar from '../controls/controls-bar'; +import FullscreenToggle, { Props as FullscreenToggleProps } from '../controls/fullscreen'; +import ZoomControls, { Props as ZoomControlsProps } from '../controls/zoom'; + +export type Props = FullscreenToggleProps & ZoomControlsProps; + +export default function ImageControls({ onFullscreenToggle, onZoomIn, onZoomOut, scale }: Props): JSX.Element { + return ( + + + {/* TODO: RotateControl */} + + {/* TODO: AnnotationControls (separate group) */} + + ); +} diff --git a/src/lib/viewers/image/ImageViewer.js b/src/lib/viewers/image/ImageViewer.js index c1e248907..6344a2c5c 100644 --- a/src/lib/viewers/image/ImageViewer.js +++ b/src/lib/viewers/image/ImageViewer.js @@ -1,6 +1,8 @@ -import ImageBaseViewer from './ImageBaseViewer'; +import React from 'react'; import AnnotationControls, { AnnotationMode } from '../../AnnotationControls'; import AnnotationControlsFSM, { AnnotationInput, AnnotationState, stateModeMap } from '../../AnnotationControlsFSM'; +import ImageBaseViewer from './ImageBaseViewer'; +import ImageControls from './ImageControls'; import { CLASS_INVISIBLE, DISCOVERABILITY_ATTRIBUTE } from '../../constants'; import { ICON_FULLSCREEN_IN, ICON_FULLSCREEN_OUT, ICON_ROTATE_LEFT } from '../../icons/icons'; import './Image.scss'; @@ -338,9 +340,8 @@ class ImageViewer extends ImageBaseViewer { ? width / this.imageEl.getAttribute('originalWidth') : height / this.imageEl.getAttribute('originalHeight'); this.rotationAngle = (this.currentRotationAngle % 3600) % 360; - if (this.zoomControls) { - this.zoomControls.setCurrentScale(this.scale); - } + this.renderUI(); + this.emit('scale', { scale: this.scale, rotationAngle: this.rotationAngle, @@ -378,6 +379,28 @@ class ImageViewer extends ImageBaseViewer { } } + loadUIReact() { + super.loadUIReact(); + this.renderUI(); + } + + renderUI() { + if (this.zoomControls) { + this.zoomControls.setCurrentScale(this.scale); + } + + if (this.controls && this.options.useReactControls) { + this.controls.render( + , + ); + } + } + /** * Determines if Image file has been rotated 90 or 270 degrees to the left * diff --git a/src/lib/viewers/image/MultiImageControls.tsx b/src/lib/viewers/image/MultiImageControls.tsx new file mode 100644 index 000000000..9506ac91d --- /dev/null +++ b/src/lib/viewers/image/MultiImageControls.tsx @@ -0,0 +1,16 @@ +import React from 'react'; +import ControlsBar from '../controls/controls-bar'; +import FullscreenToggle, { Props as FullscreenToggleProps } from '../controls/fullscreen'; +import ZoomControls, { Props as ZoomControlsProps } from '../controls/zoom'; + +export type Props = FullscreenToggleProps & ZoomControlsProps; + +export default function MultiImageControls({ onFullscreenToggle, onZoomIn, onZoomOut, scale }: Props): JSX.Element { + return ( + + + {/* TODO: PageControls */} + + + ); +} diff --git a/src/lib/viewers/image/MultiImageViewer.js b/src/lib/viewers/image/MultiImageViewer.js index 078ec48f1..89a0d0d6c 100644 --- a/src/lib/viewers/image/MultiImageViewer.js +++ b/src/lib/viewers/image/MultiImageViewer.js @@ -1,9 +1,10 @@ +import React from 'react'; import ImageBaseViewer from './ImageBaseViewer'; +import MultiImageControls from './MultiImageControls'; import PageControls from '../../PageControls'; -import { ICON_FULLSCREEN_IN, ICON_FULLSCREEN_OUT } from '../../icons/icons'; import { CLASS_INVISIBLE, CLASS_MULTI_IMAGE_PAGE, CLASS_IS_SCROLLABLE } from '../../constants'; +import { ICON_FULLSCREEN_IN, ICON_FULLSCREEN_OUT } from '../../icons/icons'; import { pageNumberFromScroll } from '../../util'; - import './MultiImage.scss'; const PADDING_BUFFER = 100; @@ -246,9 +247,7 @@ class MultiImageViewer extends ImageBaseViewer { // Grab the first page image dimensions const imageEl = this.singleImageEls[0]; this.scale = width ? width / imageEl.naturalWidth : height / imageEl.naturalHeight; - if (this.zoomControls) { - this.zoomControls.setCurrentScale(this.scale); - } + this.renderUI(); this.emit('scale', { scale: this.scale }); } @@ -265,6 +264,28 @@ class MultiImageViewer extends ImageBaseViewer { this.bindPageControlListeners(); } + loadUIReact() { + super.loadUIReact(); + this.renderUI(); + } + + renderUI() { + if (this.zoomControls) { + this.zoomControls.setCurrentScale(this.scale); + } + + if (this.controls && this.options.useReactControls) { + this.controls.render( + , + ); + } + } + /** * Binds listeners for the page and the rest of the document controls. * diff --git a/src/lib/viewers/image/__tests__/ImageBaseViewer-test.js b/src/lib/viewers/image/__tests__/ImageBaseViewer-test.js index 887167786..a142b3b35 100644 --- a/src/lib/viewers/image/__tests__/ImageBaseViewer-test.js +++ b/src/lib/viewers/image/__tests__/ImageBaseViewer-test.js @@ -522,6 +522,7 @@ describe('lib/viewers/image/ImageBaseViewer', () => { imageBase.loaded = false; stubs.zoom = jest.spyOn(imageBase, 'zoom'); stubs.loadUI = jest.spyOn(imageBase, 'loadUI'); + stubs.loadUIReact = jest.spyOn(imageBase, 'loadUIReact'); stubs.setOriginalImageSize = jest.spyOn(imageBase, 'setOriginalImageSize'); imageBase.options = { file: { @@ -543,21 +544,40 @@ describe('lib/viewers/image/ImageBaseViewer', () => { expect(stubs.zoom).not.toBeCalled(); expect(stubs.setOriginalImageSize).not.toBeCalled(); expect(stubs.loadUI).not.toBeCalled(); + expect(stubs.loadUIReact).not.toBeCalled(); }); test('should load UI if not destroyed', done => { + stubs.setOriginalImageSize.mockResolvedValue(undefined); + imageBase.on(VIEWER_EVENT.load, () => { expect(imageBase.loaded).toBe(true); expect(stubs.zoom).toBeCalled(); expect(stubs.loadUI).toBeCalled(); + expect(stubs.loadUIReact).not.toBeCalled(); done(); }); - stubs.setOriginalImageSize.mockResolvedValue(undefined); - imageBase.destroyed = false; + imageBase.destroyed = false; imageBase.finishLoading(); + expect(stubs.setOriginalImageSize).toBeCalled(); }); + + test('should load react UI if option is provided', done => { + stubs.setOriginalImageSize.mockResolvedValue(undefined); + + imageBase.on(VIEWER_EVENT.load, () => { + expect(imageBase.loaded).toBe(true); + expect(stubs.zoom).toBeCalled(); + expect(stubs.loadUI).not.toBeCalled(); + expect(stubs.loadUIReact).toBeCalled(); + done(); + }); + imageBase.destroyed = false; + imageBase.options.useReactControls = true; + imageBase.finishLoading('', true); + }); }); describe('disableViewerControls()', () => { diff --git a/src/lib/viewers/image/__tests__/ImageControls-test.jsx b/src/lib/viewers/image/__tests__/ImageControls-test.jsx new file mode 100644 index 000000000..6cb3acee7 --- /dev/null +++ b/src/lib/viewers/image/__tests__/ImageControls-test.jsx @@ -0,0 +1,24 @@ +import React from 'react'; +import { shallow } from 'enzyme'; +import ControlsBar from '../../controls/controls-bar'; +import FullscreenToggle from '../../controls/fullscreen'; +import ImageControls from '../ImageControls'; +import ZoomControls from '../../controls/zoom'; + +describe('ImageControls', () => { + describe('render', () => { + test('should return a valid wrapper', () => { + const onToggle = jest.fn(); + const onZoomIn = jest.fn(); + const onZoomOut = jest.fn(); + const wrapper = shallow( + , + ); + + expect(wrapper.exists(ControlsBar)); + expect(wrapper.find(FullscreenToggle).prop('onFullscreenToggle')).toEqual(onToggle); + expect(wrapper.find(ZoomControls).prop('onZoomIn')).toEqual(onZoomIn); + expect(wrapper.find(ZoomControls).prop('onZoomOut')).toEqual(onZoomOut); + }); + }); +}); diff --git a/src/lib/viewers/image/__tests__/ImageViewer-test.js b/src/lib/viewers/image/__tests__/ImageViewer-test.js index c2f453e7a..0a5c86437 100644 --- a/src/lib/viewers/image/__tests__/ImageViewer-test.js +++ b/src/lib/viewers/image/__tests__/ImageViewer-test.js @@ -1,9 +1,13 @@ -/* eslint-disable no-unused-expressions */ +import React from 'react'; import AnnotationControls, { AnnotationMode } from '../../../AnnotationControls'; import AnnotationControlsFSM, { AnnotationState, stateModeMap } from '../../../AnnotationControlsFSM'; -import ImageViewer from '../ImageViewer'; import BaseViewer from '../../BaseViewer'; import Browser from '../../../Browser'; +import ControlsRoot from '../../controls/controls-root'; +import ImageControls from '../ImageControls'; +import ImageViewer from '../ImageViewer'; + +jest.mock('../../controls/controls-root'); const imageUrl = ''; @@ -411,6 +415,23 @@ describe('lib/viewers/image/ImageViewer', () => { ); }); + describe('loadUIReact()', () => { + test('should create controls root and render the react controls', () => { + image.options.useReactControls = true; + image.loadUIReact(); + + expect(image.controls).toBeInstanceOf(ControlsRoot); + expect(image.controls.render).toBeCalledWith( + , + ); + }); + }); + describe('isRotated()', () => { test('should return false if image is not rotated', () => { const result = image.isRotated(); diff --git a/src/lib/viewers/image/__tests__/MultiImageControls-test.jsx b/src/lib/viewers/image/__tests__/MultiImageControls-test.jsx new file mode 100644 index 000000000..af03ab878 --- /dev/null +++ b/src/lib/viewers/image/__tests__/MultiImageControls-test.jsx @@ -0,0 +1,24 @@ +import React from 'react'; +import { shallow } from 'enzyme'; +import ControlsBar from '../../controls/controls-bar'; +import FullscreenToggle from '../../controls/fullscreen'; +import MultiImageControls from '../MultiImageControls'; +import ZoomControls from '../../controls/zoom'; + +describe('MultiImageControls', () => { + describe('render', () => { + test('should return a valid wrapper', () => { + const onToggle = jest.fn(); + const onZoomIn = jest.fn(); + const onZoomOut = jest.fn(); + const wrapper = shallow( + , + ); + + expect(wrapper.exists(ControlsBar)); + expect(wrapper.find(FullscreenToggle).prop('onFullscreenToggle')).toEqual(onToggle); + expect(wrapper.find(ZoomControls).prop('onZoomIn')).toEqual(onZoomIn); + expect(wrapper.find(ZoomControls).prop('onZoomOut')).toEqual(onZoomOut); + }); + }); +}); diff --git a/src/lib/viewers/image/__tests__/MultiImageViewer-test.js b/src/lib/viewers/image/__tests__/MultiImageViewer-test.js index 75971979e..81b5ecc50 100644 --- a/src/lib/viewers/image/__tests__/MultiImageViewer-test.js +++ b/src/lib/viewers/image/__tests__/MultiImageViewer-test.js @@ -1,14 +1,18 @@ -/* eslint-disable no-unused-expressions */ +import React from 'react'; +import * as util from '../../../util'; +import BaseViewer from '../../BaseViewer'; +import Browser from '../../../Browser'; +import ControlsRoot from '../../controls/controls-root'; +import ImageBaseViewer from '../ImageBaseViewer'; +import MultiImageControls from '../MultiImageControls'; import MultiImageViewer from '../MultiImageViewer'; import PageControls from '../../../PageControls'; +import ZoomControls from '../../../ZoomControls'; import fullscreen from '../../../Fullscreen'; import { CLASS_MULTI_IMAGE_PAGE } from '../../../constants'; -import BaseViewer from '../../BaseViewer'; -import ImageBaseViewer from '../ImageBaseViewer'; -import Browser from '../../../Browser'; -import * as util from '../../../util'; import { ICON_FULLSCREEN_IN, ICON_FULLSCREEN_OUT } from '../../../icons/icons'; -import ZoomControls from '../../../ZoomControls'; + +jest.mock('../../controls/controls-root'); const CLASS_INVISIBLE = 'bp-is-invisible'; @@ -377,6 +381,23 @@ describe('lib/viewers/image/MultiImageViewer', () => { }); }); + describe('loadUIReact()', () => { + test('should create controls root and render the react controls', () => { + multiImage.options.useReactControls = true; + multiImage.loadUIReact(); + + expect(multiImage.controls).toBeInstanceOf(ControlsRoot); + expect(multiImage.controls.render).toBeCalledWith( + , + ); + }); + }); + describe('bindPageControlListeners()', () => { beforeEach(() => { multiImage.currentPageNumber = 1;