-
Notifications
You must be signed in to change notification settings - Fork 47
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(highlight): Change cursor in Highlight Text mode (#544)
* 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
Showing
14 changed files
with
264 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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" />}</>; | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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)); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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 */ | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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" />; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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); | ||
}); | ||
}); | ||
}); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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, | ||
}); | ||
}); | ||
}); | ||
}); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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); | ||
}); | ||
}); | ||
}); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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'); | ||
}); | ||
}); | ||
}); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
export { default as HighlightManager } from './HighlightManager'; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,4 +1,5 @@ | ||
export enum Mode { | ||
HIGHLIGHT = 'highlight', | ||
NONE = 'none', | ||
REGION = 'region', | ||
} | ||
|