diff --git a/__tests__/src/components/CanvasLayers.test.js b/__tests__/src/components/CanvasLayers.test.js index a9e9bb4cc43272f6818be8a89da175babec71369..e831f31a994d879be0cc51a1f000a0d8a989384b 100644 --- a/__tests__/src/components/CanvasLayers.test.js +++ b/__tests__/src/components/CanvasLayers.test.js @@ -1,11 +1,9 @@ import React from 'react'; import { shallow } from 'enzyme'; -import { Utils } from 'manifesto.js'; import Input from '@material-ui/core/Input'; import Slider from '@material-ui/core/Slider'; import Typography from '@material-ui/core/Typography'; import { DragDropContext, Droppable, Draggable } from 'react-beautiful-dnd'; -import manifestFixtureHamilton from '../../fixtures/version-2/hamilton.json'; import { CanvasLayers } from '../../../src/components/CanvasLayers'; import IIIFThumbnail from '../../../src/containers/IIIFThumbnail'; @@ -13,7 +11,7 @@ import IIIFThumbnail from '../../../src/containers/IIIFThumbnail'; function createWrapper(props) { return shallow( <CanvasLayers - canvas={{ id: 'foo' }} + canvasId="foo" classes={{}} index={0} label="A Canvas Label" @@ -39,7 +37,7 @@ describe('CanvasLayers', () => { it('renders canvas layers in a list', () => { const wrapper = createWrapper({ - canvas: Utils.parseManifest(manifestFixtureHamilton).getSequences()[0].getCanvasByIndex(0), + canvasId: 'https://prtd.app/hamilton/canvas/p1.json', layers: [ { id: 'https://prtd.app/image/iiif/2/hamilton%2fHL_524_1r_00_PSC/full/862,1024/0/default.jpg' }, { id: 'https://prtd.app/image/iiif/2/hamilton%2fHL_524_1r_00_TS_Blue/full/862,1024/0/default.png' }, @@ -96,7 +94,7 @@ describe('CanvasLayers', () => { beforeEach(() => { updateLayers = jest.fn(); const wrapper = createWrapper({ - canvas: Utils.parseManifest(manifestFixtureHamilton).getSequences()[0].getCanvasByIndex(0), + canvasId: 'https://prtd.app/hamilton/canvas/p1.json', layers: [ { id: 'https://prtd.app/image/iiif/2/hamilton%2fHL_524_1r_00_PSC/full/862,1024/0/default.jpg' }, { id: 'https://prtd.app/image/iiif/2/hamilton%2fHL_524_1r_00_TS_Blue/full/862,1024/0/default.png' }, diff --git a/__tests__/src/components/IIIFThumbnail.test.js b/__tests__/src/components/IIIFThumbnail.test.js index 6975f5a4f14c8f7bec7ef066a025daa44c01ffe6..fa6eb72fa33ef5fb6e93b254b532ac2ede40cc57 100644 --- a/__tests__/src/components/IIIFThumbnail.test.js +++ b/__tests__/src/components/IIIFThumbnail.test.js @@ -18,9 +18,9 @@ function createWrapper(props) { describe('IIIFThumbnail', () => { let wrapper; const url = 'http://example.com/iiif/image'; - const image = { height: 120, url, width: 100 }; + const thumbnail = { height: 120, url, width: 100 }; beforeEach(() => { - wrapper = createWrapper({ image }); + wrapper = createWrapper({ thumbnail }); }); it('renders properly', () => { @@ -55,36 +55,36 @@ describe('IIIFThumbnail', () => { }); it('can be constrained by maxHeight', () => { - wrapper = createWrapper({ image, maxHeight: 100 }); + wrapper = createWrapper({ maxHeight: 100, thumbnail }); expect(wrapper.find('img').props().style).toMatchObject({ height: 100, width: 'auto' }); }); it('can be constrained by maxWidth', () => { - wrapper = createWrapper({ image, maxWidth: 80 }); + wrapper = createWrapper({ maxWidth: 80, thumbnail }); expect(wrapper.find('img').props().style).toMatchObject({ height: 'auto', width: 80 }); }); it('can be constrained by maxWidth and maxHeight', () => { - wrapper = createWrapper({ image, maxHeight: 90, maxWidth: 50 }); + wrapper = createWrapper({ maxHeight: 90, maxWidth: 50, thumbnail }); expect(wrapper.find('img').props().style).toMatchObject({ height: 60, width: 50 }); }); it('relaxes constraints when the image dimensions are unknown', () => { - wrapper = createWrapper({ image: { url } }); + wrapper = createWrapper({ thumbnail: { url } }); expect(wrapper.find('img').props().style).toMatchObject({ height: 'auto', width: 'auto' }); }); it('constrains what it can when the image dimensions are unknown', () => { - wrapper = createWrapper({ image: { height: 120, url }, maxHeight: 90 }); + wrapper = createWrapper({ maxHeight: 90, thumbnail: { height: 120, url } }); expect(wrapper.find('img').props().style).toMatchObject({ height: 90, width: 'auto' }); }); it('renders a provided label', () => { wrapper = createWrapper({ - classes: { label: 'label' }, image, label: 'Some label', labelled: true, + classes: { label: 'label' }, label: 'Some label', labelled: true, thumbnail, }); expect( wrapper.find('div.label').at(0).matchesElement( @@ -94,7 +94,7 @@ describe('IIIFThumbnail', () => { }); it('renders children', () => { - wrapper = createWrapper({ children: <span id="hi" />, image }); + wrapper = createWrapper({ children: <span id="hi" />, thumbnail }); expect(wrapper.find('span').length).toEqual(1); }); }); diff --git a/__tests__/src/components/LayersPanel.test.js b/__tests__/src/components/LayersPanel.test.js index d4877ce1f25ea03bc445adde540dab650c013f1f..769bde8c999b3eee6f71639f07bbe27682d98da4 100644 --- a/__tests__/src/components/LayersPanel.test.js +++ b/__tests__/src/components/LayersPanel.test.js @@ -19,11 +19,8 @@ function createWrapper(props) { describe('LayersPanel', () => { it('renders layers for each canvas', () => { - const canvases = [ - { id: 'a' }, - { id: 'b' }, - ]; - const wrapper = createWrapper({ canvases }); + const canvasIds = ['a', 'b']; + const wrapper = createWrapper({ canvasIds }); expect(wrapper.find(CanvasLayers).length).toBe(2); expect(wrapper.find(CanvasLayers).at(0).props()).toMatchObject({ diff --git a/__tests__/src/components/SidebarIndexItem.test.js b/__tests__/src/components/SidebarIndexItem.test.js index c26f2e5dd8009b351dde28436ae73fa20a0bf8fc..fb4295ffed531ae77b04bc7022d0e0fa90f4403a 100644 --- a/__tests__/src/components/SidebarIndexItem.test.js +++ b/__tests__/src/components/SidebarIndexItem.test.js @@ -7,7 +7,7 @@ import { SidebarIndexItem } from '../../../src/components/SidebarIndexItem'; function createWrapper(props) { return shallow( <SidebarIndexItem - canvas={{ label: 'yolo' }} + label="yolo" classes={{}} {...props} />, diff --git a/__tests__/src/components/SidebarIndexList.test.js b/__tests__/src/components/SidebarIndexList.test.js index da8dc6a7af70133e1aa8cb304d633b3f7d98b39d..03cad91b6e943a2cf814c316b11e2667a94443e1 100644 --- a/__tests__/src/components/SidebarIndexList.test.js +++ b/__tests__/src/components/SidebarIndexList.test.js @@ -23,7 +23,7 @@ function createWrapper(props) { setCanvas={() => {}} config={{ canvasNavigation: { height: 100 } }} updateVariant={() => {}} - selectedCanvases={[canvases[1]]} + selectedCanvasIds={[canvases[1].id]} {...props} />, ); diff --git a/__tests__/src/components/SidebarIndexThumbnail.test.js b/__tests__/src/components/SidebarIndexThumbnail.test.js index 997df3013c3606f6ebd927b5998674689ce0fb29..a2c219a2f9df6def5bb5b21297121d070a623f7e 100644 --- a/__tests__/src/components/SidebarIndexThumbnail.test.js +++ b/__tests__/src/components/SidebarIndexThumbnail.test.js @@ -10,8 +10,8 @@ import IIIFThumbnail from '../../../src/containers/IIIFThumbnail'; function createWrapper(props) { return shallow( <SidebarIndexThumbnail - canvas={{ label: 'yolo' }} - otherCanvas={Utils.parseManifest(fixture).getSequences()[0].getCanvases()[1]} + canvas={Utils.parseManifest(fixture).getSequences()[0].getCanvases()[1]} + label="yolo" classes={{}} config={{ canvasNavigation: { height: 200, width: 100 } }} {...props} diff --git a/__tests__/src/components/ThumbnailCanvasGrouping.test.js b/__tests__/src/components/ThumbnailCanvasGrouping.test.js index dfcfcb7312159ae3d22e1b8fbf7c73ffaf003c2d..ba27c5a71cbeaeab2f1a0c60f4c0c81271ef5ff8 100644 --- a/__tests__/src/components/ThumbnailCanvasGrouping.test.js +++ b/__tests__/src/components/ThumbnailCanvasGrouping.test.js @@ -28,7 +28,7 @@ describe('ThumbnailCanvasGrouping', () => { let setCanvas; const data = { canvasGroupings: new CanvasGroupings(Utils.parseManifest(manifestJson) - .getSequences()[0].getCanvases()), + .getSequences()[0].getCanvases()).groupings(), height: 131, position: 'far-bottom', }; diff --git a/__tests__/src/components/ThumbnailNavigation.test.js b/__tests__/src/components/ThumbnailNavigation.test.js index e86aa08966ab0bf28fc79e3e42d9d3f2986fc5c4..4bfbf173ff2859e039055e7ee0f7b78d9a7b9952 100644 --- a/__tests__/src/components/ThumbnailNavigation.test.js +++ b/__tests__/src/components/ThumbnailNavigation.test.js @@ -12,7 +12,9 @@ function createWrapper(props, fixture = manifestJson) { return shallow( <ThumbnailNavigation canvasGroupings={ - new CanvasGroupings(Utils.parseManifest(fixture).getSequences()[0].getCanvases()) + new CanvasGroupings( + Utils.parseManifest(fixture).getSequences()[0].getCanvases(), + ).groupings() } canvasIndex={1} classes={{}} diff --git a/__tests__/src/components/WindowSideBarAnnotationsPanel.test.js b/__tests__/src/components/WindowSideBarAnnotationsPanel.test.js index 10d8e32c09bb8e37b2db55d8a7d17a5a21fd6472..0ed16619527ba0b1cda0bb35ee209ccd60fefaf1 100644 --- a/__tests__/src/components/WindowSideBarAnnotationsPanel.test.js +++ b/__tests__/src/components/WindowSideBarAnnotationsPanel.test.js @@ -44,10 +44,7 @@ describe('WindowSideBarAnnotationsPanel', () => { it('renders a CanvasAnnotations for every selected canvas', () => { wrapper = createWrapper({ - selectedCanvases: [ - { id: 'abc', index: 0 }, - { id: 'xyz', index: 1 }, - ], + canvasIds: ['abc', 'xyz'], }); expect(wrapper.find(CanvasAnnotations).length).toBe(2); diff --git a/__tests__/src/components/WindowSideBarInfoPanel.test.js b/__tests__/src/components/WindowSideBarInfoPanel.test.js index 9b8ee70dff2644bec4b158cbe4f47f46fc953035..a986abb29bed4df5bdfc8941d9be18fe28d238b2 100644 --- a/__tests__/src/components/WindowSideBarInfoPanel.test.js +++ b/__tests__/src/components/WindowSideBarInfoPanel.test.js @@ -34,7 +34,7 @@ describe('WindowSideBarInfoPanel', () => { }); it('renders the canvas elements', () => { - wrapper = createWrapper({ selectedCanvases: [{ id: '1' }, { id: '2' }] }); + wrapper = createWrapper({ canvasIds: ['1', '2'] }); expect(wrapper.find(CanvasInfo).length).toBe(2); let canvasInfo = wrapper.find(CanvasInfo).at(0); diff --git a/__tests__/src/selectors/viewer.test.js b/__tests__/src/selectors/viewer.test.js new file mode 100644 index 0000000000000000000000000000000000000000..ece25446b81a8393bf996b80cf51eb49e2d0e523 --- /dev/null +++ b/__tests__/src/selectors/viewer.test.js @@ -0,0 +1,38 @@ +import { + getCurrentCanvasWorld, +} from '../../../src/state/selectors/viewer'; + +describe('getCurrentCanvasWorld', () => { + it('returns a CanvasWorld', () => { + const windowId = 'id'; + const state = { + manifests: { + a: { + json: { + '@context': 'http://iiif.io/api/presentation/3/context.json', + id: 'whatever', + items: [ + { '@id': 'a1', type: 'Canvas' }, + { '@id': 'a2', type: 'Canvas' }, + ], + type: 'Manifest', + viewingDirection: 'sideways', + }, + }, + }, + windows: { + [windowId]: { + manifestId: 'a', + visibleCanvases: [ + 'a1', 'a2', + ], + }, + }, + }; + + const actual = getCurrentCanvasWorld(state, { windowId }); + expect(actual.canvases.length).toEqual(2); + expect(Object.keys(actual.layers).length).toEqual(2); + expect(actual.viewingDirection).toEqual('sideways'); + }); +}); diff --git a/src/components/CanvasLayers.js b/src/components/CanvasLayers.js index 7024334c49909bb7b1f6b92c6d4e9b3264c413ec..d2eb7d44d8e1aa471a211b3fc14955006f290e14 100644 --- a/src/components/CanvasLayers.js +++ b/src/components/CanvasLayers.js @@ -51,7 +51,7 @@ export class CanvasLayers extends Component { /** */ onDragEnd(result) { const { - canvas, layers, updateLayers, windowId, + canvasId, layers, updateLayers, windowId, } = this.props; if (!result.destination) return; if (result.destination.droppableId !== this.droppableId) return; @@ -68,26 +68,26 @@ export class CanvasLayers extends Component { return acc; }, {}); - updateLayers(windowId, canvas.id, payload); + updateLayers(windowId, canvasId, payload); } /** */ setLayerVisibility(layerId, value) { const { - canvas, updateLayers, windowId, + canvasId, updateLayers, windowId, } = this.props; const payload = { [layerId]: { visibility: value }, }; - updateLayers(windowId, canvas.id, payload); + updateLayers(windowId, canvasId, payload); } /** */ moveToTop(layerId) { const { - canvas, layers, updateLayers, windowId, + canvasId, layers, updateLayers, windowId, } = this.props; const sortedLayers = reorder(layers.map(l => l.id), layers.findIndex(l => l.id === layerId), 0); @@ -97,20 +97,20 @@ export class CanvasLayers extends Component { return acc; }, {}); - updateLayers(windowId, canvas.id, payload); + updateLayers(windowId, canvasId, payload); } /** */ handleOpacityChange(layerId, value) { const { - canvas, updateLayers, windowId, + canvasId, updateLayers, windowId, } = this.props; const payload = { [layerId]: { opacity: value / 100.0 }, }; - updateLayers(windowId, canvas.id, payload); + updateLayers(windowId, canvasId, payload); } /** @private */ @@ -261,9 +261,7 @@ export class CanvasLayers extends Component { } CanvasLayers.propTypes = { - canvas: PropTypes.shape({ - id: PropTypes.string, - }).isRequired, + canvasId: PropTypes.string.isRequired, classes: PropTypes.objectOf(PropTypes.string), index: PropTypes.number.isRequired, label: PropTypes.string.isRequired, diff --git a/src/components/IIIFThumbnail.js b/src/components/IIIFThumbnail.js index bd43f70ac2d2841708deab9325783632c7442afc..789c1499bb7be76aedc4c36f28feaf09ad8d255f 100644 --- a/src/components/IIIFThumbnail.js +++ b/src/components/IIIFThumbnail.js @@ -4,11 +4,21 @@ import 'intersection-observer'; // polyfill needed for Safari import Typography from '@material-ui/core/Typography'; import IntersectionObserver from '@researchgate/react-intersection-observer'; import classNames from 'classnames'; +import getThumbnail from '../lib/ThumbnailFactory'; /** * Uses InteractionObserver to "lazy" load canvas thumbnails that are in view. */ export class IIIFThumbnail extends Component { + /** */ + static getUseableLabel(resource, index) { + return (resource + && resource.getLabel + && resource.getLabel().length > 0) + ? resource.getLabel().map(label => label.value)[0] + : String(index + 1); + } + /** */ constructor(props) { @@ -17,14 +27,33 @@ export class IIIFThumbnail extends Component { this.handleIntersection = this.handleIntersection.bind(this); } + /** */ + componentDidMount() { + this.setState(state => ({ ...state, image: this.image() })); + } + + /** */ + componentDidUpdate(prevProps) { + const { maxHeight, maxWidth, resource } = this.props; + + if ( + prevProps.maxHeight !== maxHeight + || prevProps.maxWidth !== maxWidth + || prevProps.resource !== resource) { + this.setState(state => ({ ...state, image: this.image() })); // eslint-disable-line + } + } + /** * */ imageStyles() { const { - maxHeight, maxWidth, style, image, + maxHeight, maxWidth, style, } = this.props; + const image = this.image(); + const styleProps = { height: 'auto', width: 'auto' }; if (!image) return { ...style, height: maxHeight || 'auto', width: maxWidth || 'auto' }; @@ -74,9 +103,29 @@ export class IIIFThumbnail extends Component { if (loaded || !event.isIntersecting) return; - this.setState({ - loaded: true, - }); + this.setState(state => ({ ...state, loaded: true })); + } + + /** */ + image() { + const { + thumbnail, resource, maxHeight, maxWidth, + } = this.props; + + if (thumbnail) return thumbnail; + + const image = getThumbnail(resource, { maxHeight, maxWidth }); + + if (image && image.url) return image; + + return undefined; + } + + /** */ + label() { + const { label, resource } = this.props; + + return label || IIIFThumbnail.getUseableLabel(resource); } /** @@ -85,13 +134,15 @@ export class IIIFThumbnail extends Component { const { children, classes, - image, - label, + imagePlaceholder, labelled, + thumbnail, variant, } = this.props; - const { loaded } = this.state; + const { image, loaded } = this.state; + + const { url: src = imagePlaceholder } = (loaded && (thumbnail || image)) || {}; return ( <div className={classNames(classes.root, { [classes[`${variant}Root`]]: variant })}> @@ -99,15 +150,15 @@ export class IIIFThumbnail extends Component { <img alt="" role="presentation" - src={(loaded && image && image.url) || IIIFThumbnail.defaultImgPlaceholder} + src={src} style={this.imageStyles()} className={classes.image} /> </IntersectionObserver> - { labelled && label && ( + { labelled && ( <div className={classNames(classes.label, { [classes[`${variant}Label`]]: variant })}> <Typography variant="caption" classes={{ root: classNames(classes.caption, { [classes[`${variant}Caption`]]: variant }) }}> - {label} + {this.label()} </Typography> </div> )} @@ -117,33 +168,34 @@ export class IIIFThumbnail extends Component { } } -// Transparent "gray" -IIIFThumbnail.defaultImgPlaceholder = ''; - IIIFThumbnail.propTypes = { children: PropTypes.node, classes: PropTypes.objectOf(PropTypes.string), - image: PropTypes.shape({ - height: PropTypes.number, - url: PropTypes.string.isRequired, - width: PropTypes.number, - }), + imagePlaceholder: PropTypes.string, label: PropTypes.string, labelled: PropTypes.bool, maxHeight: PropTypes.number, maxWidth: PropTypes.number, + resource: PropTypes.object.isRequired, // eslint-disable-line react/forbid-prop-types style: PropTypes.object, // eslint-disable-line react/forbid-prop-types + thumbnail: PropTypes.shape({ + height: PropTypes.number, + url: PropTypes.string.isRequired, + width: PropTypes.number, + }), variant: PropTypes.oneOf(['inside', 'outside']), }; IIIFThumbnail.defaultProps = { children: null, classes: {}, - image: null, + // Transparent "gray" + imagePlaceholder: '', label: undefined, labelled: false, maxHeight: null, maxWidth: null, style: {}, + thumbnail: null, variant: null, }; diff --git a/src/components/LayersPanel.js b/src/components/LayersPanel.js index 4ef013f034e128e2614a91ce603c4d0654c6a392..09cda41e93b4ed1a75c3b9f29db94ca0c97157fe 100644 --- a/src/components/LayersPanel.js +++ b/src/components/LayersPanel.js @@ -12,7 +12,7 @@ export class LayersPanel extends Component { */ render() { const { - canvases, + canvasIds, id, t, windowId, @@ -24,12 +24,12 @@ export class LayersPanel extends Component { id={id} windowId={windowId} > - {canvases.map((canvas, index) => ( + {canvasIds.map((canvasId, index) => ( <CanvasLayers - canvasId={canvas.id} + canvasId={canvasId} index={index} - key={canvas.id} - totalSize={canvases.length} + key={canvasId} + totalSize={canvasIds.length} windowId={windowId} /> ))} @@ -39,14 +39,12 @@ export class LayersPanel extends Component { } LayersPanel.propTypes = { - canvases: PropTypes.arrayOf(PropTypes.shape({ - id: PropTypes.string, - })), + canvasIds: PropTypes.arrayOf(PropTypes.string), id: PropTypes.string.isRequired, t: PropTypes.func.isRequired, windowId: PropTypes.string.isRequired, }; LayersPanel.defaultProps = { - canvases: [], + canvasIds: [], }; diff --git a/src/components/SidebarIndexItem.js b/src/components/SidebarIndexItem.js index bad724843d4a6fbb43ba38e419b8a3fdf51c1616..909b218a95b8fe3d3604ddb4d7105b89f4fbfe2b 100644 --- a/src/components/SidebarIndexItem.js +++ b/src/components/SidebarIndexItem.js @@ -8,7 +8,7 @@ export class SidebarIndexItem extends Component { /** */ render() { const { - classes, canvas, + classes, label, } = this.props; return ( @@ -17,7 +17,7 @@ export class SidebarIndexItem extends Component { className={classNames(classes.label)} variant="body1" > - {canvas.label} + {label} </Typography> </> ); @@ -25,6 +25,6 @@ export class SidebarIndexItem extends Component { } SidebarIndexItem.propTypes = { - canvas: PropTypes.object.isRequired, // eslint-disable-line react/forbid-prop-types classes: PropTypes.objectOf(PropTypes.string).isRequired, + label: PropTypes.string.isRequired, }; diff --git a/src/components/SidebarIndexList.js b/src/components/SidebarIndexList.js index 0972d74abaf7227379cd7568afe48653776d3a9d..667b2764fc930c5dda597616a8f77213bee0d70d 100644 --- a/src/components/SidebarIndexList.js +++ b/src/components/SidebarIndexList.js @@ -25,7 +25,7 @@ export class SidebarIndexList extends Component { canvases, classes, containerRef, - selectedCanvases, + selectedCanvasIds, setCanvas, variant, windowId, @@ -56,15 +56,15 @@ export class SidebarIndexList extends Component { onClick={onClick} button component="li" - selected={!!selectedCanvases.find(c => c.id === canvas.id)} + selected={selectedCanvasIds.includes(canvas.id)} > <ScrollTo containerRef={containerRef} key={`${canvas.id}-${variant}`} offsetTop={96} // offset for the height of the form above - scrollTo={!!selectedCanvases.find(c => c.id === canvas.id)} + scrollTo={selectedCanvasIds.includes(canvas.id)} > - <Item canvas={canvas} otherCanvas={canvases[canvasIndex]} /> + <Item label={canvas.label} canvas={canvases[canvasIndex]} /> </ScrollTo> </MenuItem> ); @@ -79,13 +79,13 @@ SidebarIndexList.propTypes = { canvases: PropTypes.array.isRequired, // eslint-disable-line react/forbid-prop-types classes: PropTypes.objectOf(PropTypes.string).isRequired, containerRef: PropTypes.oneOf([PropTypes.func, PropTypes.object]).isRequired, - selectedCanvases: PropTypes.arrayOf(PropTypes.shape({ id: PropTypes.string })), + selectedCanvasIds: PropTypes.arrayOf(PropTypes.string), setCanvas: PropTypes.func.isRequired, variant: PropTypes.oneOf(['item', 'thumbnail']), windowId: PropTypes.string.isRequired, }; SidebarIndexList.defaultProps = { - selectedCanvases: [], + selectedCanvasIds: [], variant: 'item', }; diff --git a/src/components/SidebarIndexThumbnail.js b/src/components/SidebarIndexThumbnail.js index c2ab64128277c054228e349045a06de5299faf50..5ac21782e7c5403d3259abf863246def01c510a7 100644 --- a/src/components/SidebarIndexThumbnail.js +++ b/src/components/SidebarIndexThumbnail.js @@ -9,14 +9,15 @@ export class SidebarIndexThumbnail extends Component { /** */ render() { const { - classes, otherCanvas, canvas, height, width, + classes, canvas, height, label, width, } = this.props; return ( <> <div style={{ minWidth: 50 }}> <IIIFThumbnail - resource={otherCanvas} + label={label} + resource={canvas} className={classNames(classes.clickable)} maxHeight={height} maxWidth={width} @@ -26,7 +27,7 @@ export class SidebarIndexThumbnail extends Component { className={classNames(classes.label)} variant="body1" > - {canvas.label} + {label} </Typography> </> ); @@ -37,7 +38,7 @@ SidebarIndexThumbnail.propTypes = { canvas: PropTypes.object.isRequired, // eslint-disable-line react/forbid-prop-types classes: PropTypes.objectOf(PropTypes.string).isRequired, height: PropTypes.number, - otherCanvas: PropTypes.object.isRequired, // eslint-disable-line react/forbid-prop-types + label: PropTypes.string.isRequired, width: PropTypes.number, }; diff --git a/src/components/ThumbnailCanvasGrouping.js b/src/components/ThumbnailCanvasGrouping.js index 958e2e5f2a4d10e31a48a932f5b4f878ec06da50..cc966ff815eba1093a823238a750ce70f5ac309c 100644 --- a/src/components/ThumbnailCanvasGrouping.js +++ b/src/components/ThumbnailCanvasGrouping.js @@ -36,7 +36,7 @@ export class ThumbnailCanvasGrouping extends PureComponent { const { canvasGroupings, position, height, } = data; - const currentGroupings = canvasGroupings.groupings()[index]; + const currentGroupings = canvasGroupings[index]; const SPACING = 8; return ( <div diff --git a/src/components/ThumbnailNavigation.js b/src/components/ThumbnailNavigation.js index 48accae8e21f7708b802ab5d503cea106392a211..252e458243868b6f5b6c6ece55639dd510548595 100644 --- a/src/components/ThumbnailNavigation.js +++ b/src/components/ThumbnailNavigation.js @@ -48,7 +48,7 @@ export class ThumbnailNavigation extends Component { */ calculateScaledSize(index) { const { thumbnailNavigation, canvasGroupings, position } = this.props; - const canvases = canvasGroupings.groupings()[index]; + const canvases = canvasGroupings[index]; const world = new CanvasWorld(canvases); const bounds = world.worldBounds(); switch (position) { @@ -125,7 +125,7 @@ export class ThumbnailNavigation extends Component { /** */ itemCount() { const { canvasGroupings } = this.props; - return canvasGroupings.groupings().length; + return canvasGroupings.length; } /** */ @@ -231,7 +231,7 @@ export class ThumbnailNavigation extends Component { } ThumbnailNavigation.propTypes = { - canvasGroupings: PropTypes.object.isRequired, // eslint-disable-line react/forbid-prop-types + canvasGroupings: PropTypes.array.isRequired, // eslint-disable-line react/forbid-prop-types canvasIndex: PropTypes.number.isRequired, classes: PropTypes.objectOf(PropTypes.string).isRequired, hasNextCanvas: PropTypes.bool, diff --git a/src/components/WindowSideBarAnnotationsPanel.js b/src/components/WindowSideBarAnnotationsPanel.js index 6cf9ac8bf0bdea6f1f25d0cd8bfd09e4bc7e1c41..06068abe31e400d2b9500238310850fdc64962ea 100644 --- a/src/components/WindowSideBarAnnotationsPanel.js +++ b/src/components/WindowSideBarAnnotationsPanel.js @@ -15,7 +15,7 @@ export class WindowSideBarAnnotationsPanel extends Component { */ render() { const { - annotationCount, classes, selectedCanvases, t, windowId, id, + annotationCount, classes, canvasIds, t, windowId, id, } = this.props; return ( <CompanionWindow @@ -29,12 +29,12 @@ export class WindowSideBarAnnotationsPanel extends Component { <Typography component="p" variant="subtitle2">{t('showingNumAnnotations', { number: annotationCount })}</Typography> </div> - {selectedCanvases.map((canvas, index) => ( + {canvasIds.map((canvasId, index) => ( <CanvasAnnotations - canvasId={canvas.id} - key={canvas.id} + canvasId={canvasId} + key={canvasId} index={index} - totalSize={selectedCanvases.length} + totalSize={canvasIds.length} windowId={windowId} /> ))} @@ -45,15 +45,14 @@ export class WindowSideBarAnnotationsPanel extends Component { WindowSideBarAnnotationsPanel.propTypes = { annotationCount: PropTypes.number.isRequired, + canvasIds: PropTypes.arrayOf(PropTypes.string), classes: PropTypes.objectOf(PropTypes.string).isRequired, - id: PropTypes.string.isRequired, - selectedCanvases: PropTypes.arrayOf(PropTypes.shape({ id: PropTypes.string })), t: PropTypes.func, windowId: PropTypes.string.isRequired, }; WindowSideBarAnnotationsPanel.defaultProps = { - selectedCanvases: [], + canvasIds: [], t: key => key, }; diff --git a/src/components/WindowSideBarInfoPanel.js b/src/components/WindowSideBarInfoPanel.js index 176e0761c564c214b4f95cbd43a309a5d380490a..8136fc66abd662197443db92568866885a0e5305 100644 --- a/src/components/WindowSideBarInfoPanel.js +++ b/src/components/WindowSideBarInfoPanel.js @@ -20,11 +20,11 @@ export class WindowSideBarInfoPanel extends Component { const { windowId, id, + canvasIds, classes, collectionPath, t, locale, - selectedCanvases, setLocale, availableLocales, showLocalePicker, @@ -48,13 +48,13 @@ export class WindowSideBarInfoPanel extends Component { )} > { - selectedCanvases.map((canvas, index) => ( - <div key={canvas.id} className={classes.section}> + canvasIds.map((canvasId, index) => ( + <div key={canvasId} className={classes.section}> <CanvasInfo id={id} - canvasId={canvas.id} + canvasId={canvasId} index={index} - totalSize={selectedCanvases.length} + totalSize={canvasIds.length} windowId={windowId} /> </div> @@ -80,11 +80,11 @@ export class WindowSideBarInfoPanel extends Component { WindowSideBarInfoPanel.propTypes = { availableLocales: PropTypes.arrayOf(PropTypes.string), + canvasIds: PropTypes.arrayOf(PropTypes.string), classes: PropTypes.objectOf(PropTypes.string), collectionPath: PropTypes.arrayOf(PropTypes.string), id: PropTypes.string.isRequired, locale: PropTypes.string, - selectedCanvases: PropTypes.arrayOf(PropTypes.shape({ id: PropTypes.string })), setLocale: PropTypes.func, showLocalePicker: PropTypes.bool, t: PropTypes.func, @@ -93,10 +93,10 @@ WindowSideBarInfoPanel.propTypes = { WindowSideBarInfoPanel.defaultProps = { availableLocales: [], + canvasIds: [], classes: {}, collectionPath: [], locale: '', - selectedCanvases: [], setLocale: undefined, showLocalePicker: false, t: key => key, diff --git a/src/containers/AnnotationsOverlay.js b/src/containers/AnnotationsOverlay.js index 43be6e15931dddb888d88d183ac1805caea27df5..8619925686bfcb71677d7d68e7d65ad71e4d89d6 100644 --- a/src/containers/AnnotationsOverlay.js +++ b/src/containers/AnnotationsOverlay.js @@ -4,18 +4,15 @@ import { withTranslation } from 'react-i18next'; import { withPlugins } from '../extend/withPlugins'; import { AnnotationsOverlay } from '../components/AnnotationsOverlay'; import * as actions from '../state/actions'; -import CanvasWorld from '../lib/CanvasWorld'; import { getWindow, - getSequenceViewingDirection, - getLayersForVisibleCanvases, - getVisibleCanvases, getSearchAnnotationsForWindow, getCompanionWindowsForContent, getTheme, getConfig, getPresentAnnotationsOnSelectedCanvases, getSelectedAnnotationId, + getCurrentCanvasWorld, } from '../state/selectors'; /** @@ -25,11 +22,7 @@ import { */ const mapStateToProps = (state, { windowId }) => ({ annotations: getPresentAnnotationsOnSelectedCanvases(state, { windowId }), - canvasWorld: new CanvasWorld( - getVisibleCanvases(state, { windowId }), - getLayersForVisibleCanvases(state, { windowId }), - getSequenceViewingDirection(state, { windowId }), - ), + canvasWorld: getCurrentCanvasWorld(state, { windowId }), drawAnnotations: getConfig(state).window.forceDrawAnnotations || getCompanionWindowsForContent(state, { content: 'annotations', windowId }).length > 0, drawSearchAnnotations: getConfig(state).window.forceDrawAnnotations || getCompanionWindowsForContent(state, { content: 'search', windowId }).length > 0, highlightAllAnnotations: getWindow(state, { windowId }).highlightAllAnnotations, diff --git a/src/containers/CanvasLayers.js b/src/containers/CanvasLayers.js index 7e36dc1d6f52f381037ae086f9cb9df7226c773d..aba5f5c7c7703a0f989801e99022498742ca27cf 100644 --- a/src/containers/CanvasLayers.js +++ b/src/containers/CanvasLayers.js @@ -4,7 +4,6 @@ import { withTranslation } from 'react-i18next'; import { withStyles } from '@material-ui/core/styles'; import * as actions from '../state/actions'; import { - getCanvas, getCanvasLabel, getLayers, getSortedLayers, @@ -13,11 +12,7 @@ import { CanvasLayers } from '../components/CanvasLayers'; /** For connect */ const mapStateToProps = (state, { canvasId, windowId }) => ({ - canvas: getCanvas(state, { canvasId, windowId }), - label: getCanvasLabel(state, { - canvasId, - windowId, - }), + label: getCanvasLabel(state, { canvasId, windowId }), layerMetadata: getLayers(state, { canvasId, windowId }), layers: getSortedLayers(state, { canvasId, windowId }), }); diff --git a/src/containers/IIIFThumbnail.js b/src/containers/IIIFThumbnail.js index 706e920b84b9f4441eedf45dc49fdaec4b22e9d4..3caffe62d464c24ad6d14306d0ab8e8892cd473c 100644 --- a/src/containers/IIIFThumbnail.js +++ b/src/containers/IIIFThumbnail.js @@ -1,25 +1,8 @@ import { compose } from 'redux'; -import { connect } from 'react-redux'; import { withTranslation } from 'react-i18next'; import { withStyles } from '@material-ui/core/styles'; import { withPlugins } from '../extend/withPlugins'; import { IIIFThumbnail } from '../components/IIIFThumbnail'; -import getThumbnail from '../lib/ThumbnailFactory'; - -/** */ -function getLabel(resource) { - return resource.getLabel().length > 0 - ? resource.getLabel().map(label => label.value)[0] - : String(resource.index + 1); -} - -/** */ -const mapStateToProps = (state, { - label, labelled, maxHeight, maxWidth, resource, thumbnail, -}) => ({ - image: thumbnail || getThumbnail(resource, { maxHeight, maxWidth }), - label: labelled && (label || getLabel(resource)), -}); /** * Styles for withStyles HOC @@ -65,7 +48,6 @@ const styles = theme => ({ const enhance = compose( withStyles(styles), withTranslation(), - connect(mapStateToProps), withPlugins('IIIFThumbnail'), ); diff --git a/src/containers/LayersPanel.js b/src/containers/LayersPanel.js index b438a5a6af4c725a5958bb1eb4cae122bd278cf6..16a1a9b01b5cec0d1005c357debb1fd72dcf801d 100644 --- a/src/containers/LayersPanel.js +++ b/src/containers/LayersPanel.js @@ -5,14 +5,14 @@ import { withStyles } from '@material-ui/core/styles'; import { withPlugins } from '../extend/withPlugins'; import { LayersPanel } from '../components/LayersPanel'; import { - getVisibleCanvases, + getVisibleCanvasIds, } from '../state/selectors'; /** * mapStateToProps - to hook up connect */ const mapStateToProps = (state, { id, windowId }) => ({ - canvases: getVisibleCanvases(state, { windowId }), + canvasIds: getVisibleCanvasIds(state, { windowId }), }); /** diff --git a/src/containers/OpenSeadragonViewer.js b/src/containers/OpenSeadragonViewer.js index 253c10ee6b5a3b902181edca81af6b7b2740f302..bf5d3fef6b091af9709e994c6e6a49d2d206b1f3 100644 --- a/src/containers/OpenSeadragonViewer.js +++ b/src/containers/OpenSeadragonViewer.js @@ -6,18 +6,15 @@ import flatten from 'lodash/flatten'; import { withPlugins } from '../extend/withPlugins'; import { OpenSeadragonViewer } from '../components/OpenSeadragonViewer'; import * as actions from '../state/actions'; -import CanvasWorld from '../lib/CanvasWorld'; import { getVisibleCanvasNonTiledResources, getCurrentCanvas, getCanvasLabel, - getSequenceViewingDirection, - getLayersForVisibleCanvases, - getVisibleCanvases, getViewer, getConfig, getCompanionWindowsForContent, selectInfoResponses, + getCurrentCanvasWorld, } from '../state/selectors'; /** @@ -26,12 +23,7 @@ import { * @private */ const mapStateToProps = (state, { windowId }) => { - const canvasWorld = new CanvasWorld( - getVisibleCanvases(state, { windowId }), - getLayersForVisibleCanvases(state, { windowId }), - getSequenceViewingDirection(state, { windowId }), - ); - + const canvasWorld = getCurrentCanvasWorld(state, { windowId }); const infoResponses = selectInfoResponses(state); const imageServiceIds = flatten(canvasWorld.canvases.map(c => c.imageServiceIds)); diff --git a/src/containers/SidebarIndexList.js b/src/containers/SidebarIndexList.js index 053de13c60219b02d3b25bc0b51542c36c60b34f..866e7d61d7229e239182378c8cd3e075a5a85cfa 100644 --- a/src/containers/SidebarIndexList.js +++ b/src/containers/SidebarIndexList.js @@ -7,21 +7,18 @@ import * as actions from '../state/actions'; import { getCompanionWindow, getCanvases, - getVisibleCanvases, + getVisibleCanvasIds, } from '../state/selectors'; import { SidebarIndexList } from '../components/SidebarIndexList'; /** * mapStateToProps - to hook up connect */ -const mapStateToProps = (state, { id, windowId }) => { - const canvases = getCanvases(state, { windowId }); - return { - canvases, - selectedCanvases: getVisibleCanvases(state, { windowId }), - variant: getCompanionWindow(state, { companionWindowId: id, windowId }).variant, - }; -}; +const mapStateToProps = (state, { id, windowId }) => ({ + canvases: getCanvases(state, { windowId }), + selectedCanvasIds: getVisibleCanvasIds(state, { windowId }), + variant: getCompanionWindow(state, { companionWindowId: id, windowId }).variant, +}); /** * mapStateToProps - used to hook up connect to state diff --git a/src/containers/ThumbnailNavigation.js b/src/containers/ThumbnailNavigation.js index eb82793df90cd4a35c6432dfe4b9a59d339bd6cb..e3be9a8f3363f1068771976c0d71929445d9e771 100644 --- a/src/containers/ThumbnailNavigation.js +++ b/src/containers/ThumbnailNavigation.js @@ -3,13 +3,12 @@ import { connect } from 'react-redux'; import { withTranslation } from 'react-i18next'; import { withStyles } from '@material-ui/core/styles'; import { withPlugins } from '../extend/withPlugins'; -import CanvasGroupings from '../lib/CanvasGroupings'; import * as actions from '../state/actions'; import { ThumbnailNavigation } from '../components/ThumbnailNavigation'; import { getCompanionWindow, getWindow, getNextCanvasGrouping, getPreviousCanvasGrouping, - getCanvases, getCanvasIndex, getWindowViewType, + getCanvasGroupings, getCanvasIndex, getWindowViewType, getSequenceViewingDirection, getConfig, } from '../state/selectors'; @@ -18,24 +17,18 @@ import { * @memberof ThumbnailNavigation * @private */ -const mapStateToProps = (state, { windowId }) => { - const viewType = getWindowViewType(state, { windowId }); - return { - canvasGroupings: new CanvasGroupings( - getCanvases(state, { windowId }), - viewType, - ), - canvasIndex: getCanvasIndex(state, { windowId }), - hasNextCanvas: !!getNextCanvasGrouping(state, { windowId }), - hasPreviousCanvas: !!getPreviousCanvasGrouping(state, { windowId }), - position: getCompanionWindow(state, { - companionWindowId: getWindow(state, { windowId }).thumbnailNavigationId, - }).position, - thumbnailNavigation: getConfig(state).thumbnailNavigation, - view: viewType, - viewingDirection: getSequenceViewingDirection(state, { windowId }), - }; -}; +const mapStateToProps = (state, { windowId }) => ({ + canvasGroupings: getCanvasGroupings(state, { windowId }), + canvasIndex: getCanvasIndex(state, { windowId }), + hasNextCanvas: !!getNextCanvasGrouping(state, { windowId }), + hasPreviousCanvas: !!getPreviousCanvasGrouping(state, { windowId }), + position: getCompanionWindow(state, { + companionWindowId: getWindow(state, { windowId }).thumbnailNavigationId, + }).position, + thumbnailNavigation: getConfig(state).thumbnailNavigation, + view: getWindowViewType(state, { windowId }), + viewingDirection: getSequenceViewingDirection(state, { windowId }), +}); /** * mapDispatchToProps - used to hook up connect to action creators diff --git a/src/containers/WindowSideBarAnnotationsPanel.js b/src/containers/WindowSideBarAnnotationsPanel.js index 854f5175882ed828f177d762ef796abd6a4f7bab..904cea2f93053bcbca8d2c0903931f934ccc6e9d 100644 --- a/src/containers/WindowSideBarAnnotationsPanel.js +++ b/src/containers/WindowSideBarAnnotationsPanel.js @@ -4,7 +4,7 @@ import { withTranslation } from 'react-i18next'; import { withStyles } from '@material-ui/core/styles'; import { withPlugins } from '../extend/withPlugins'; import { - getVisibleCanvases, + getVisibleCanvasIds, getAnnotationResourcesByMotivation, } from '../state/selectors'; import { WindowSideBarAnnotationsPanel } from '../components/WindowSideBarAnnotationsPanel'; @@ -19,7 +19,7 @@ const mapStateToProps = (state, { windowId }) => ({ state, { windowId }, ).length, - selectedCanvases: getVisibleCanvases(state, { windowId }), + canvasIds: getVisibleCanvasIds(state, { windowId }), }); /** */ diff --git a/src/containers/WindowSideBarCanvasPanel.js b/src/containers/WindowSideBarCanvasPanel.js index a55744b1cb0b8b51008831ce1b70ede2629642e3..e26303a543965736c65bfbfb8cfe203d5f466cad 100644 --- a/src/containers/WindowSideBarCanvasPanel.js +++ b/src/containers/WindowSideBarCanvasPanel.js @@ -8,8 +8,6 @@ import { WindowSideBarCanvasPanel } from '../components/WindowSideBarCanvasPanel import { getCompanionWindow, getDefaultSidebarVariant, - getCanvases, - getVisibleCanvases, getSequenceTreeStructure, getWindow, getManifestoInstance, @@ -19,7 +17,6 @@ import { * mapStateToProps - to hook up connect */ const mapStateToProps = (state, { id, windowId }) => { - const canvases = getCanvases(state, { windowId }); const treeStructure = getSequenceTreeStructure(state, { windowId }); const window = getWindow(state, { windowId }); const { config } = state; @@ -27,10 +24,8 @@ const mapStateToProps = (state, { id, windowId }) => { const collectionPath = window.collectionPath || []; const collectionId = collectionPath && collectionPath[collectionPath.length - 1]; return { - canvases, collection: collectionId && getManifestoInstance(state, { manifestId: collectionId }), config, - selectedCanvases: getVisibleCanvases(state, { windowId }), showToc: treeStructure && treeStructure.nodes && treeStructure.nodes.length > 0, variant: companionWindow.variant || getDefaultSidebarVariant(state, { windowId }), diff --git a/src/containers/WindowSideBarInfoPanel.js b/src/containers/WindowSideBarInfoPanel.js index 0a69e62d0ebe70f5cc9079aea63e48c2725b1c94..9fc498ce117521c64e176a2559bebd8c8b40692d 100644 --- a/src/containers/WindowSideBarInfoPanel.js +++ b/src/containers/WindowSideBarInfoPanel.js @@ -8,7 +8,7 @@ import { getCompanionWindow, getManifestLocale, getMetadataLocales, - getVisibleCanvases, + getVisibleCanvasIds, getWindowConfig, getWindow, } from '../state/selectors'; @@ -21,10 +21,10 @@ import { WindowSideBarInfoPanel } from '../components/WindowSideBarInfoPanel'; */ const mapStateToProps = (state, { id, windowId }) => ({ availableLocales: getMetadataLocales(state, { companionWindowId: id, windowId }), + canvasIds: getVisibleCanvasIds(state, { windowId }), collectionPath: (getWindow(state, { windowId }) || {}).collectionPath, locale: getCompanionWindow(state, { companionWindowId: id }).locale || getManifestLocale(state, { windowId }), - selectedCanvases: getVisibleCanvases(state, { windowId }), showLocalePicker: getWindowConfig(state, { windowId }).showLocalePicker, }); diff --git a/src/state/selectors/canvases.js b/src/state/selectors/canvases.js index 684676bcd8fb15756f7a5246b2c6b535ef12046c..4a3c9924889b25d1caa071bc965ae2a557d018a8 100644 --- a/src/state/selectors/canvases.js +++ b/src/state/selectors/canvases.js @@ -70,7 +70,7 @@ export const getVisibleCanvases = createSelector( * @param {string} props.windowId * @return {Array} */ -const getCanvasGroupings = createSelector( +export const getCanvasGroupings = createSelector( [ getCanvases, getWindowViewType, diff --git a/src/state/selectors/index.js b/src/state/selectors/index.js index 866141957ffc7875f157663ae457c88428e38c5b..bad2660352779524c7324cf25d3f33f32ec1cef9 100644 --- a/src/state/selectors/index.js +++ b/src/state/selectors/index.js @@ -12,3 +12,4 @@ export * from './layers'; export * from './sequences'; export * from './auth'; export * from './utils'; +export * from './viewer'; diff --git a/src/state/selectors/viewer.js b/src/state/selectors/viewer.js new file mode 100644 index 0000000000000000000000000000000000000000..27b5992553d5d5f2d739f35d2f863c90d46e99d7 --- /dev/null +++ b/src/state/selectors/viewer.js @@ -0,0 +1,14 @@ +import { createSelector } from 'reselect'; +import CanvasWorld from '../../lib/CanvasWorld'; + +import { getVisibleCanvases } from './canvases'; +import { getLayersForVisibleCanvases } from './layers'; +import { getSequenceViewingDirection } from './sequences'; + +/** Instantiate a manifesto instance */ +export const getCurrentCanvasWorld = createSelector( + getVisibleCanvases, + getLayersForVisibleCanvases, + getSequenceViewingDirection, + (canvases, layers, viewingDirection) => new CanvasWorld(canvases, layers, viewingDirection), +);