Skip to content

Commit

Permalink
feat(highlight): Change cursor in Highlight Text mode (#544)
Browse files Browse the repository at this point in the history
* feat(highlight): Change cursor in Highlight Text mode

* feat(highlight): Add tests

* feat(highlight): Address feedbacks
  • Loading branch information
Mingze authored Jul 28, 2020
1 parent 24660b8 commit 9535ee4
Show file tree
Hide file tree
Showing 14 changed files with 264 additions and 1 deletion.
2 changes: 2 additions & 0 deletions src/document/DocumentAnnotator.ts
Original file line number Diff line number Diff line change
Expand Up @@ -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';

Expand Down Expand Up @@ -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;
Expand Down
3 changes: 2 additions & 1 deletion src/document/__tests__/DocumentAnnotator-test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -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');

Expand Down Expand Up @@ -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);
});

Expand Down
8 changes: 8 additions & 0 deletions src/highlight/HighlightAnnotations.scss
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
.ba-HighlightAnnotations-creator {
position: absolute;
top: 0;
left: 0;
width: 100%;
height: 100%;
pointer-events: auto;
}
20 changes: 20 additions & 0 deletions src/highlight/HighlightAnnotations.tsx
Original file line number Diff line number Diff line change
@@ -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<Props> {
static defaultProps = {
isCreating: false,
};

render(): JSX.Element {
const { isCreating } = this.props;

return <>{isCreating && <HighlightCreator className="ba-HighlightAnnotations-creator" />}</>;
}
}
14 changes: 14 additions & 0 deletions src/highlight/HighlightContainer.tsx
Original file line number Diff line number Diff line change
@@ -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));
8 changes: 8 additions & 0 deletions src/highlight/HighlightCreator.scss
Original file line number Diff line number Diff line change
@@ -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 */
}
11 changes: 11 additions & 0 deletions src/highlight/HighlightCreator.tsx
Original file line number Diff line number Diff line change
@@ -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 <div className={classNames(className, 'ba-HighlightCreator')} data-testid="ba-HighlightCreator" />;
}
49 changes: 49 additions & 0 deletions src/highlight/HighlightManager.tsx
Original file line number Diff line number Diff line change
@@ -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(<HighlightContainer {...props} />, this.reactEl);
}

style(styles: Partial<CSSStyleDeclaration>): CSSStyleDeclaration {
return Object.assign(this.reactEl.style, styles);
}
}
29 changes: 29 additions & 0 deletions src/highlight/__tests__/HighlightAnnotations-test.tsx
Original file line number Diff line number Diff line change
@@ -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(<HighlightAnnotations {...defaults} {...props} />);

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);
});
});
});
29 changes: 29 additions & 0 deletions src/highlight/__tests__/HighlightContainer-test.tsx
Original file line number Diff line number Diff line change
@@ -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<Props> => mount(<HighlightContainer {...defaults} {...props} />);

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,
});
});
});
});
15 changes: 15 additions & 0 deletions src/highlight/__tests__/HighlightCreator-test.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
import React from 'react';
import { shallow, ShallowWrapper } from 'enzyme';
import HighlightCreator from '../HighlightCreator';

describe('HighlightCreator', () => {
const getWrapper = (props = {}): ShallowWrapper => shallow(<HighlightCreator {...props} />);

describe('render', () => {
test('should add class', () => {
const wrapper = getWrapper();

expect(wrapper.hasClass('ba-HighlightCreator')).toBe(true);
});
});
});
75 changes: 75 additions & 0 deletions src/highlight/__tests__/HighlightManager-test.ts
Original file line number Diff line number Diff line change
@@ -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> = {}): Options => ({
referenceEl: rootEl.querySelector('.reference') as HTMLElement,
...options,
});
const getLayer = (): HTMLElement => rootEl.querySelector('[data-testid="ba-Layer--highlight"]') as HTMLElement;
const getWrapper = (options?: Partial<Options>): HighlightManager => new HighlightManager(getOptions(options));

beforeEach(() => {
rootEl.classList.add('root');
rootEl.innerHTML = '<div class="reference" />'; // 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');
});
});
});
1 change: 1 addition & 0 deletions src/highlight/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
export { default as HighlightManager } from './HighlightManager';
1 change: 1 addition & 0 deletions src/store/common/types.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
export enum Mode {
HIGHLIGHT = 'highlight',
NONE = 'none',
REGION = 'region',
}
Expand Down

0 comments on commit 9535ee4

Please sign in to comment.