Skip to content

Commit

Permalink
feat(controls): Create react core controls for image viewers (#1285)
Browse files Browse the repository at this point in the history
  • Loading branch information
jstoffan authored Nov 5, 2020
1 parent c6bdc5f commit ede8e42
Show file tree
Hide file tree
Showing 10 changed files with 217 additions and 20 deletions.
12 changes: 11 additions & 1 deletion src/lib/viewers/image/ImageBaseViewer.js
Original file line number Diff line number Diff line change
@@ -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';

Expand Down Expand Up @@ -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;
Expand Down Expand Up @@ -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.
Expand Down
17 changes: 17 additions & 0 deletions src/lib/viewers/image/ImageControls.tsx
Original file line number Diff line number Diff line change
@@ -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 (
<ControlsBar>
<ZoomControls onZoomIn={onZoomIn} onZoomOut={onZoomOut} scale={scale} />
{/* TODO: RotateControl */}
<FullscreenToggle onFullscreenToggle={onFullscreenToggle} />
{/* TODO: AnnotationControls (separate group) */}
</ControlsBar>
);
}
31 changes: 27 additions & 4 deletions src/lib/viewers/image/ImageViewer.js
Original file line number Diff line number Diff line change
@@ -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';
Expand Down Expand Up @@ -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,
Expand Down Expand Up @@ -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(
<ImageControls
onFullscreenToggle={this.toggleFullscreen}
onZoomIn={this.zoomIn}
onZoomOut={this.zoomOut}
scale={this.scale}
/>,
);
}
}

/**
* Determines if Image file has been rotated 90 or 270 degrees to the left
*
Expand Down
16 changes: 16 additions & 0 deletions src/lib/viewers/image/MultiImageControls.tsx
Original file line number Diff line number Diff line change
@@ -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 (
<ControlsBar>
<ZoomControls onZoomIn={onZoomIn} onZoomOut={onZoomOut} scale={scale} />
{/* TODO: PageControls */}
<FullscreenToggle onFullscreenToggle={onFullscreenToggle} />
</ControlsBar>
);
}
31 changes: 26 additions & 5 deletions src/lib/viewers/image/MultiImageViewer.js
Original file line number Diff line number Diff line change
@@ -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;
Expand Down Expand Up @@ -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 });
}

Expand All @@ -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(
<MultiImageControls
onFullscreenToggle={this.toggleFullscreen}
onZoomIn={this.zoomIn}
onZoomOut={this.zoomOut}
scale={this.scale}
/>,
);
}
}

/**
* Binds listeners for the page and the rest of the document controls.
*
Expand Down
24 changes: 22 additions & 2 deletions src/lib/viewers/image/__tests__/ImageBaseViewer-test.js
Original file line number Diff line number Diff line change
Expand Up @@ -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: {
Expand All @@ -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()', () => {
Expand Down
24 changes: 24 additions & 0 deletions src/lib/viewers/image/__tests__/ImageControls-test.jsx
Original file line number Diff line number Diff line change
@@ -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(
<ImageControls onFullscreenToggle={onToggle} onZoomIn={onZoomIn} onZoomOut={onZoomOut} />,
);

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);
});
});
});
25 changes: 23 additions & 2 deletions src/lib/viewers/image/__tests__/ImageViewer-test.js
Original file line number Diff line number Diff line change
@@ -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 =
'';
Expand Down Expand Up @@ -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(
<ImageControls
onFullscreenToggle={image.toggleFullscreen}
onZoomIn={image.zoomIn}
onZoomOut={image.zoomOut}
scale={1}
/>,
);
});
});

describe('isRotated()', () => {
test('should return false if image is not rotated', () => {
const result = image.isRotated();
Expand Down
24 changes: 24 additions & 0 deletions src/lib/viewers/image/__tests__/MultiImageControls-test.jsx
Original file line number Diff line number Diff line change
@@ -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(
<MultiImageControls onFullscreenToggle={onToggle} onZoomIn={onZoomIn} onZoomOut={onZoomOut} />,
);

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);
});
});
});
33 changes: 27 additions & 6 deletions src/lib/viewers/image/__tests__/MultiImageViewer-test.js
Original file line number Diff line number Diff line change
@@ -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';

Expand Down Expand Up @@ -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(
<MultiImageControls
onFullscreenToggle={multiImage.toggleFullscreen}
onZoomIn={multiImage.zoomIn}
onZoomOut={multiImage.zoomOut}
scale={1}
/>,
);
});
});

describe('bindPageControlListeners()', () => {
beforeEach(() => {
multiImage.currentPageNumber = 1;
Expand Down

0 comments on commit ede8e42

Please sign in to comment.