diff --git a/src/document/DocumentAnnotator.ts b/src/document/DocumentAnnotator.ts index d31ee1b9e..73dcf5e49 100644 --- a/src/document/DocumentAnnotator.ts +++ b/src/document/DocumentAnnotator.ts @@ -2,6 +2,7 @@ import BaseAnnotator from '../common/BaseAnnotator'; import BaseManager from '../common/BaseManager'; import { centerRegion, isRegion, RegionManager } from '../region'; import { getAnnotation } from '../store/annotations'; +import { HighlightManager } from '../highlight'; import { scrollToLocation } from '../utils/scroll'; import './DocumentAnnotator.scss'; @@ -30,6 +31,7 @@ export default class DocumentAnnotator extends BaseAnnotator { // Lazily instantiate managers as pages are added or re-rendered if (managers.size === 0) { managers.add(new RegionManager({ location: pageNumber, referenceEl: pageReferenceEl })); + managers.add(new HighlightManager({ location: pageNumber, referenceEl: pageReferenceEl })); } return managers; diff --git a/src/document/__tests__/DocumentAnnotator-test.ts b/src/document/__tests__/DocumentAnnotator-test.ts index 825d53f40..c322d2acb 100644 --- a/src/document/__tests__/DocumentAnnotator-test.ts +++ b/src/document/__tests__/DocumentAnnotator-test.ts @@ -6,6 +6,7 @@ import { annotations as regions } from '../../region/__mocks__/data'; import { fetchAnnotationsAction } from '../../store'; import { scrollToLocation } from '../../utils/scroll'; +jest.mock('../../highlight/HighlightManager'); jest.mock('../../region/RegionManager'); jest.mock('../../utils/scroll'); @@ -69,7 +70,7 @@ describe('DocumentAnnotator', () => { test('should create new managers given a new page element', () => { const managers = annotator.getPageManagers(getPage()); - expect(managers.size).toBe(1); + expect(managers.size).toBe(2); expect(managers.values().next().value).toBeInstanceOf(RegionManager); }); diff --git a/src/highlight/HighlightAnnotations.scss b/src/highlight/HighlightAnnotations.scss new file mode 100644 index 000000000..7db6045a9 --- /dev/null +++ b/src/highlight/HighlightAnnotations.scss @@ -0,0 +1,8 @@ +.ba-HighlightAnnotations-creator { + position: absolute; + top: 0; + left: 0; + width: 100%; + height: 100%; + pointer-events: auto; +} diff --git a/src/highlight/HighlightAnnotations.tsx b/src/highlight/HighlightAnnotations.tsx new file mode 100644 index 000000000..9d1fdd15a --- /dev/null +++ b/src/highlight/HighlightAnnotations.tsx @@ -0,0 +1,20 @@ +import * as React from 'react'; +import HighlightCreator from './HighlightCreator'; + +import './HighlightAnnotations.scss'; + +type Props = { + isCreating: boolean; +}; + +export default class HighlightAnnotations extends React.PureComponent { + static defaultProps = { + isCreating: false, + }; + + render(): JSX.Element { + const { isCreating } = this.props; + + return <>{isCreating && }; + } +} diff --git a/src/highlight/HighlightContainer.tsx b/src/highlight/HighlightContainer.tsx new file mode 100644 index 000000000..1491fd893 --- /dev/null +++ b/src/highlight/HighlightContainer.tsx @@ -0,0 +1,14 @@ +import { connect } from 'react-redux'; +import { AppState, getAnnotationMode } from '../store'; +import HighlightAnnotations from './HighlightAnnotations'; +import withProviders from '../common/withProviders'; + +export type Props = { + isCreating: boolean; +}; + +export const mapStateToProps = (state: AppState): Props => ({ + isCreating: getAnnotationMode(state) === 'highlight', +}); + +export default connect(mapStateToProps)(withProviders(HighlightAnnotations)); diff --git a/src/highlight/HighlightCreator.scss b/src/highlight/HighlightCreator.scss new file mode 100644 index 000000000..64e319577 --- /dev/null +++ b/src/highlight/HighlightCreator.scss @@ -0,0 +1,8 @@ +$text_cursor_32: ''; +$text_cursor_32_2x: ''; +$text_cursor_32_3x: ''; + +.ba-HighlightCreator { + cursor: url($text_cursor_32) 16 16, text; /* Legacy */ + cursor: image-set(url($text_cursor_32) 1x, url($text_cursor_32_2x) 2x, url($text_cursor_32_3x) 3x) 16 16, text; /* Webkit */ /* stylelint-disable-line */ +} diff --git a/src/highlight/HighlightCreator.tsx b/src/highlight/HighlightCreator.tsx new file mode 100644 index 000000000..87ba1dcc4 --- /dev/null +++ b/src/highlight/HighlightCreator.tsx @@ -0,0 +1,11 @@ +import * as React from 'react'; +import classNames from 'classnames'; +import './HighlightCreator.scss'; + +type Props = { + className?: string; +}; + +export default function HighlightCreator({ className }: Props): JSX.Element { + return
; +} diff --git a/src/highlight/HighlightManager.tsx b/src/highlight/HighlightManager.tsx new file mode 100644 index 000000000..5640a96a5 --- /dev/null +++ b/src/highlight/HighlightManager.tsx @@ -0,0 +1,49 @@ +import * as React from 'react'; +import * as ReactDOM from 'react-dom'; +import BaseManager, { Options, Props } from '../common/BaseManager'; +import HighlightContainer from './HighlightContainer'; + +export default class HighlightManager implements BaseManager { + location: number; + + reactEl: HTMLElement; + + constructor({ location = 1, referenceEl }: Options) { + this.location = location; + this.reactEl = this.insert(referenceEl); + } + + destroy(): void { + ReactDOM.unmountComponentAtNode(this.reactEl); + + this.reactEl.remove(); + } + + exists(parentEl: HTMLElement): boolean { + return parentEl.contains(this.reactEl); + } + + insert(referenceEl: HTMLElement): HTMLElement { + // Find the nearest applicable reference and document elements + const documentEl = referenceEl.ownerDocument || document; + const parentEl = referenceEl.parentNode || documentEl; + + // Construct a layer element where we can inject a root React component + const rootLayerEl = documentEl.createElement('div'); + rootLayerEl.classList.add('ba-Layer'); + rootLayerEl.classList.add('ba-Layer--highlight'); + rootLayerEl.dataset.testid = 'ba-Layer--highlight'; + rootLayerEl.setAttribute('data-resin-feature', 'annotations'); + + // Insert the new layer element immediately after the reference element + return parentEl.insertBefore(rootLayerEl, referenceEl.nextSibling); + } + + render(props: Props): void { + ReactDOM.render(, this.reactEl); + } + + style(styles: Partial): CSSStyleDeclaration { + return Object.assign(this.reactEl.style, styles); + } +} diff --git a/src/highlight/__tests__/HighlightAnnotations-test.tsx b/src/highlight/__tests__/HighlightAnnotations-test.tsx new file mode 100644 index 000000000..05d4deb31 --- /dev/null +++ b/src/highlight/__tests__/HighlightAnnotations-test.tsx @@ -0,0 +1,29 @@ +import React from 'react'; +import { shallow, ShallowWrapper } from 'enzyme'; +import HighlightAnnotations from '../HighlightAnnotations'; +import HighlightCreator from '../HighlightCreator'; + +jest.mock('../HighlightCreator'); + +describe('components/highlight/HighlightAnnotations', () => { + const defaults = { + isCreating: false, + }; + + const getWrapper = (props = {}): ShallowWrapper => shallow(); + + describe('render()', () => { + test('should render a RegionCreator if in creation mode', () => { + const wrapper = getWrapper({ isCreating: true }); + const creator = wrapper.find(HighlightCreator); + + expect(creator.hasClass('ba-HighlightAnnotations-creator')).toBe(true); + }); + + test('should not render creation components if not in creation mode', () => { + const wrapper = getWrapper({ isCreating: false }); + + expect(wrapper.exists(HighlightCreator)).toBe(false); + }); + }); +}); diff --git a/src/highlight/__tests__/HighlightContainer-test.tsx b/src/highlight/__tests__/HighlightContainer-test.tsx new file mode 100644 index 000000000..669ad46af --- /dev/null +++ b/src/highlight/__tests__/HighlightContainer-test.tsx @@ -0,0 +1,29 @@ +import * as React from 'react'; +import { IntlShape } from 'react-intl'; +import { mount, ReactWrapper } from 'enzyme'; +import HighlightAnnotations from '../HighlightAnnotations'; +import HighlightContainer, { Props } from '../HighlightContainer'; +import { createStore } from '../../store'; + +jest.mock('../../common/withProviders'); +jest.mock('../HighlightAnnotations'); + +describe('HighlightContainer', () => { + const defaults = { + intl: {} as IntlShape, + location: 1, + store: createStore(), + }; + const getWrapper = (props = {}): ReactWrapper => mount(); + + describe('render', () => { + test('should connect the underlying component and wrap it with a root provider', () => { + const wrapper = getWrapper(); + + expect(wrapper.exists('RootProvider')).toBe(true); + expect(wrapper.find(HighlightAnnotations).props()).toMatchObject({ + isCreating: false, + }); + }); + }); +}); diff --git a/src/highlight/__tests__/HighlightCreator-test.tsx b/src/highlight/__tests__/HighlightCreator-test.tsx new file mode 100644 index 000000000..38bd04dc1 --- /dev/null +++ b/src/highlight/__tests__/HighlightCreator-test.tsx @@ -0,0 +1,15 @@ +import React from 'react'; +import { shallow, ShallowWrapper } from 'enzyme'; +import HighlightCreator from '../HighlightCreator'; + +describe('HighlightCreator', () => { + const getWrapper = (props = {}): ShallowWrapper => shallow(); + + describe('render', () => { + test('should add class', () => { + const wrapper = getWrapper(); + + expect(wrapper.hasClass('ba-HighlightCreator')).toBe(true); + }); + }); +}); diff --git a/src/highlight/__tests__/HighlightManager-test.ts b/src/highlight/__tests__/HighlightManager-test.ts new file mode 100644 index 000000000..0482ea352 --- /dev/null +++ b/src/highlight/__tests__/HighlightManager-test.ts @@ -0,0 +1,75 @@ +import ReactDOM from 'react-dom'; +import { createIntl } from 'react-intl'; +import HighlightManager from '../HighlightManager'; +import { createStore } from '../../store'; +import { Options } from '../../common/BaseManager'; + +jest.mock('react-dom', () => ({ + render: jest.fn(), + unmountComponentAtNode: jest.fn(), +})); + +describe('HighlightManager', () => { + const intl = createIntl({ locale: 'en' }); + const rootEl = document.createElement('div'); + const getOptions = (options: Partial = {}): Options => ({ + referenceEl: rootEl.querySelector('.reference') as HTMLElement, + ...options, + }); + const getLayer = (): HTMLElement => rootEl.querySelector('[data-testid="ba-Layer--highlight"]') as HTMLElement; + const getWrapper = (options?: Partial): HighlightManager => new HighlightManager(getOptions(options)); + + beforeEach(() => { + rootEl.classList.add('root'); + rootEl.innerHTML = '
'; // referenceEl + }); + + describe('constructor', () => { + test('should set all necessary properties', () => { + const wrapper = getWrapper(); + + expect(wrapper.location).toEqual(1); + expect(wrapper.reactEl).toEqual(getLayer()); + }); + }); + + describe('destroy()', () => { + test('should unmount the React node and remove the root element', () => { + const wrapper = getWrapper(); + + wrapper.destroy(); + + expect(ReactDOM.unmountComponentAtNode).toHaveBeenCalledWith(wrapper.reactEl); + }); + }); + + describe('exists()', () => { + test('should return a boolean based on its presence in the page element', () => { + const wrapper = getWrapper(); + + expect(wrapper.exists(rootEl)).toBe(true); + expect(wrapper.exists(document.createElement('div'))).toBe(false); + }); + }); + + describe('render()', () => { + test('should format the props and pass them to the underlying components', () => { + const wrapper = getWrapper(); + + wrapper.render({ intl, store: createStore() }); + + expect(ReactDOM.render).toHaveBeenCalled(); + }); + }); + + describe('style', () => { + test('should assign the style object to the root element', () => { + const wrapper = getWrapper(); + + wrapper.style({ left: '5px', top: '10px' }); + + expect(getLayer().style.left).toEqual('5px'); + expect(getLayer().style.top).toEqual('10px'); + }); + }); +}); diff --git a/src/highlight/index.ts b/src/highlight/index.ts new file mode 100644 index 000000000..b7172c405 --- /dev/null +++ b/src/highlight/index.ts @@ -0,0 +1 @@ +export { default as HighlightManager } from './HighlightManager'; diff --git a/src/store/common/types.ts b/src/store/common/types.ts index 26fdcc1c0..b31eefaca 100644 --- a/src/store/common/types.ts +++ b/src/store/common/types.ts @@ -1,4 +1,5 @@ export enum Mode { + HIGHLIGHT = 'highlight', NONE = 'none', REGION = 'region', }