Skip to content

Commit

Permalink
feat(region): Retain annotation message between pages (#498)
Browse files Browse the repository at this point in the history
* feat(region): Retain annotation message between pages

* feat(region): Address comments

Co-authored-by: mergify[bot] <37929162+mergify[bot]@users.noreply.github.com>
  • Loading branch information
Mingze and mergify[bot] authored May 27, 2020
1 parent 92588cb commit f4553f7
Show file tree
Hide file tree
Showing 14 changed files with 90 additions and 81 deletions.
36 changes: 17 additions & 19 deletions src/region/RegionAnnotations.tsx
Original file line number Diff line number Diff line change
@@ -1,22 +1,24 @@
import merge from 'lodash/merge';
import * as React from 'react';
import PopupReply from '../components/Popups/PopupReply';
import RegionCreator from './RegionCreator';
import RegionList from './RegionList';
import RegionRect, { RegionRectRef } from './RegionRect';
import { AnnotationRegion, Rect } from '../@types';
import { CreateArg } from './actions';
import { CreatorItem, CreatorStatus } from '../store/creator';
import { scaleShape } from './regionUtil';
import './RegionAnnotations.scss';

type Props = {
activeAnnotationId: string | null;
annotations: AnnotationRegion[];
createRegion: (arg: { location: number; message: string; shape: Rect }) => void;
createRegion: (arg: CreateArg) => void;
isCreating: boolean;
message: string;
page: number;
scale: number;
setActiveAnnotationId: (annotationId: string | null) => void;
setMessage: (message: string) => void;
setStaged: (staged: CreatorItem | null) => void;
setStatus: (status: CreatorStatus) => void;
staged?: CreatorItem | null;
Expand All @@ -36,11 +38,6 @@ export default class RegionAnnotations extends React.PureComponent<Props, State>

state: State = {};

setStaged(data: Partial<CreatorItem> | null): void {
const { page, setStaged, staged } = this.props;
setStaged(data ? merge({ location: page, message: '' }, staged, data) : data);
}

setStatus(status: CreatorStatus): void {
const { setStatus } = this.props;
setStatus(status);
Expand All @@ -53,44 +50,46 @@ export default class RegionAnnotations extends React.PureComponent<Props, State>
};

handleCancel = (): void => {
this.setStaged(null);
const { setMessage, setStaged } = this.props;
setMessage('');
setStaged(null);
this.setStatus(CreatorStatus.init);
};

handleChange = (text?: string): void => {
this.setStaged({ message: text });
const { setMessage } = this.props;
setMessage(text || '');
};

handleStart = (): void => {
this.setStaged(null);
const { setStaged } = this.props;
setStaged(null);
this.setStatus(CreatorStatus.init);
};

handleStop = (shape: Rect): void => {
const { scale } = this.props;

this.setStaged({ shape: scaleShape(shape, scale, true) });
const { page, scale, setStaged } = this.props;
setStaged({ location: page, shape: scaleShape(shape, scale, true) });
this.setStatus(CreatorStatus.staged);
};

handleSubmit = (): void => {
const { createRegion, staged } = this.props;
const { createRegion, message, staged } = this.props;

if (!staged) {
return;
}

createRegion(staged);
createRegion({ ...staged, message });
};

setRectRef = (rectRef: RegionRectRef): void => {
this.setState({ rectRef });
};

render(): JSX.Element {
const { activeAnnotationId, annotations, isCreating, scale, staged, status } = this.props;
const { activeAnnotationId, annotations, isCreating, message, scale, staged, status } = this.props;
const { rectRef } = this.state;
const canDraw = !staged || !staged.message;
const canReply = status !== CreatorStatus.init;
const isPending = status === CreatorStatus.pending;

Expand All @@ -108,7 +107,6 @@ export default class RegionAnnotations extends React.PureComponent<Props, State>
{/* Layer 2: Drawn (unsaved) incomplete annotation target, if any */}
{isCreating && (
<RegionCreator
canDraw={canDraw}
className="ba-RegionAnnotations-creator"
onStart={this.handleStart}
onStop={this.handleStop}
Expand All @@ -131,7 +129,7 @@ export default class RegionAnnotations extends React.PureComponent<Props, State>
onChange={this.handleChange}
onSubmit={this.handleSubmit}
reference={rectRef}
value={staged.message}
value={message}
/>
</div>
)}
Expand Down
5 changes: 5 additions & 0 deletions src/region/RegionContainer.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -7,9 +7,11 @@ import {
getActiveAnnotationId,
getAnnotationMode,
getAnnotationsForLocation,
getCreatorMessage,
getCreatorStagedForLocation,
getCreatorStatus,
setActiveAnnotationIdAction,
setMessageAction,
setStagedAction,
setStatusAction,
} from '../store';
Expand All @@ -22,6 +24,7 @@ export type Props = {
activeAnnotationId: string | null;
annotations: AnnotationRegion[];
isCreating: boolean;
message: string;
staged: CreatorItem | null;
status: CreatorStatus;
};
Expand All @@ -30,13 +33,15 @@ export const mapStateToProps = (state: AppState, { page }: { page: number }): Pr
activeAnnotationId: getActiveAnnotationId(state),
annotations: getAnnotationsForLocation(state, page).filter(isRegion),
isCreating: getAnnotationMode(state) === 'region',
message: getCreatorMessage(state),
staged: getCreatorStagedForLocation(state, page),
status: getCreatorStatus(state),
});

export const mapDispatchToProps = {
createRegion: createRegionAction,
setActiveAnnotationId: setActiveAnnotationIdAction,
setMessage: setMessageAction,
setStaged: setStagedAction,
setStatus: setStatusAction,
};
Expand Down
2 changes: 1 addition & 1 deletion src/region/RegionCreator.scss
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ $region_cursor_32: '
$region_cursor_32_2x: '';
$region_cursor_32_3x: '';

.ba-RegionCreator.is-active {
.ba-RegionCreator {
cursor: url($region_cursor_32) 16 16, crosshair; /* Legacy */
cursor: image-set(url($region_cursor_32) 1x, url($region_cursor_32_2x) 2x, url($region_cursor_32_3x) 3x) 16 16, crosshair; /* Webkit */ /* stylelint-disable-line */
}
Expand Down
30 changes: 13 additions & 17 deletions src/region/RegionCreator.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,6 @@ import { styleShape } from './regionUtil';
import './RegionCreator.scss';

type Props = {
canDraw: boolean;
className?: string;
onStart: () => void;
onStop: (shape: Rect) => void;
Expand All @@ -19,7 +18,7 @@ const MIN_Y = 0; // Minimum region y position must be positive
const MIN_SIZE = 10; // Minimum region size must be large enough to be clickable
const MOUSE_PRIMARY = 1; // Primary mouse button

export default function RegionCreator({ canDraw, className, onStart, onStop }: Props): JSX.Element {
export default function RegionCreator({ className, onStart, onStop }: Props): JSX.Element {
const [isDrawing, setIsDrawing] = React.useState<boolean>(false);
const [isHovering, setIsHovering] = React.useState<boolean>(false);
const creatorElRef = React.useRef<HTMLDivElement>(null);
Expand Down Expand Up @@ -154,18 +153,6 @@ export default function RegionCreator({ canDraw, className, onStart, onStop }: P
const handleTouchStart = ({ targetTouches }: React.TouchEvent): void => {
startDraw(targetTouches[0].clientX, targetTouches[0].clientY);
};
const eventHandlers = canDraw
? {
onClick: handleClick,
onMouseDown: handleMouseDown,
onMouseOut: handleMouseOut,
onMouseOver: handleMouseOver,
onTouchCancel: handleTouchCancel,
onTouchEnd: handleTouchEnd,
onTouchMove: handleTouchMove,
onTouchStart: handleTouchStart,
}
: {};

const renderStep = (callback: () => void): void => {
renderHandleRef.current = window.requestAnimationFrame(callback);
Expand Down Expand Up @@ -216,14 +203,23 @@ export default function RegionCreator({ canDraw, className, onStart, onStop }: P
});

return (
// eslint-disable-next-line jsx-a11y/mouse-events-have-key-events
<div
ref={creatorElRef}
className={classNames(className, 'ba-RegionCreator', { 'is-active': canDraw })}
className={classNames(className, 'ba-RegionCreator')}
data-testid="ba-RegionCreator"
{...eventHandlers}
onClick={handleClick}
onMouseDown={handleMouseDown}
onMouseOut={handleMouseOut}
onMouseOver={handleMouseOver}
onTouchCancel={handleTouchCancel}
onTouchEnd={handleTouchEnd}
onTouchMove={handleTouchMove}
onTouchStart={handleTouchStart}
role="presentation"
>
{isDrawing && <RegionRect ref={regionRectRef} className="ba-RegionCreator-rect" isActive />}
{!isDrawing && isHovering && canDraw && <PopupCursor />}
{!isDrawing && isHovering && <PopupCursor />}
</div>
);
}
24 changes: 14 additions & 10 deletions src/region/__tests__/RegionAnnotations-test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -18,13 +18,15 @@ jest.mock('../regionUtil', () => ({
scaleShape: jest.fn(value => value),
}));

describe('RegionAnnotations', () => {
describe('components/region/RegionAnnotations', () => {
const defaults = {
activeAnnotationId: null,
createRegion: jest.fn(),
message: 'test',
page: 1,
scale: 1,
setActiveAnnotationId: jest.fn(),
setMessage: jest.fn(),
setStaged: jest.fn(),
setStatus: jest.fn(),
staged: null,
Expand All @@ -40,7 +42,6 @@ describe('RegionAnnotations', () => {
const getRectRef = (): HTMLDivElement => document.createElement('div');
const getStaged = (): CreatorItem => ({
location: 1,
message: 'test',
shape: getRect(),
});
const getWrapper = (props = {}): ShallowWrapper => shallow(<RegionAnnotations {...defaults} {...props} />);
Expand All @@ -60,6 +61,7 @@ describe('RegionAnnotations', () => {
test('should reset the staged state and status', () => {
instance.handleCancel();

expect(defaults.setMessage).toHaveBeenCalledWith('');
expect(defaults.setStaged).toHaveBeenCalledWith(null);
expect(defaults.setStatus).toHaveBeenCalledWith(CreatorStatus.init);
});
Expand All @@ -69,18 +71,15 @@ describe('RegionAnnotations', () => {
test('should set the staged state with the new message', () => {
instance.handleChange('test');

expect(defaults.setStaged).toHaveBeenCalledWith(
expect.objectContaining({
message: 'test',
}),
);
expect(defaults.setMessage).toHaveBeenCalledWith('test');
});
});

describe('handleStart', () => {
test('should reset the creator status', () => {
instance.handleStart();

expect(defaults.setStaged).toHaveBeenCalledWith(null);
expect(defaults.setStatus).toHaveBeenCalledWith(CreatorStatus.init);
});
});
Expand All @@ -98,7 +97,10 @@ describe('RegionAnnotations', () => {
instance.handleStop(shape);

expect(scaleShape).toHaveBeenCalledWith(shape, defaults.scale, true); // Inverse scale
expect(defaults.setStaged).toHaveBeenCalledWith(expect.objectContaining({ shape }));
expect(defaults.setStaged).toHaveBeenCalledWith({
location: defaults.page,
shape,
});
expect(defaults.setStatus).toHaveBeenCalledWith(CreatorStatus.staged);
});
});
Expand All @@ -107,7 +109,10 @@ describe('RegionAnnotations', () => {
test('should save the staged annotation', () => {
instance.handleSubmit();

expect(defaults.createRegion).toHaveBeenCalledWith(getStaged());
expect(defaults.createRegion).toHaveBeenCalledWith({
...getStaged(),
message: defaults.message,
});
});
});

Expand All @@ -134,7 +139,6 @@ describe('RegionAnnotations', () => {
const creator = wrapper.find(RegionCreator);

expect(creator.hasClass('ba-RegionAnnotations-creator')).toBe(true);
expect(creator.prop('canDraw')).toBe(true);
});

test('should scale and render a RegionAnnotation if one has been drawn', () => {
Expand Down
2 changes: 2 additions & 0 deletions src/region/__tests__/RegionContainer-test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -27,11 +27,13 @@ describe('RegionContainer', () => {
annotations: [],
createRegion: expect.any(Function),
isCreating: false,
message: '',
page: defaults.page,
scale: defaults.scale,
staged: null,
status: CreatorStatus.init,
setActiveAnnotationId: expect.any(Function),
setMessage: expect.any(Function),
setStaged: expect.any(Function),
setStatus: expect.any(Function),
store: defaults.store,
Expand Down
28 changes: 6 additions & 22 deletions src/region/__tests__/RegionCreator-test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,6 @@ jest.mock('../regionUtil');

describe('RegionCreator', () => {
const defaults = {
canDraw: true,
onDraw: jest.fn(),
onStart: jest.fn(),
onStop: jest.fn(),
Expand Down Expand Up @@ -278,36 +277,21 @@ describe('RegionCreator', () => {
});
});

test('should add event listeners if canDraw is true', () => {
test('should add class and event listeners', () => {
const wrapper = getWrapper();
const rootEl = getWrapperRoot(wrapper);

expect(rootEl.hasClass('ba-RegionCreator')).toBe(true);

expect(rootEl.prop('onClick')).toBeDefined();
expect(rootEl.prop('onMouseDown')).toBeDefined();
expect(rootEl.prop('onMouseOut')).toBeDefined();
expect(rootEl.prop('onMouseOver')).toBeDefined();
expect(rootEl.prop('onTouchCancel')).toBeDefined();
expect(rootEl.prop('onTouchCancel')).toBeDefined();
expect(rootEl.prop('onTouchEnd')).toBeDefined();
expect(rootEl.prop('onTouchMove')).toBeDefined();
expect(rootEl.prop('onTouchStart')).toBeDefined();
});

test('should not add event listeners if canDraw is false', () => {
const wrapper = getWrapper({ canDraw: false });
const rootEl = getWrapperRoot(wrapper);

expect(rootEl.prop('onClick')).not.toBeDefined();
expect(rootEl.prop('onMouseDown')).not.toBeDefined();
expect(rootEl.prop('onTouchCancel')).not.toBeDefined();
expect(rootEl.prop('onTouchEnd')).not.toBeDefined();
expect(rootEl.prop('onTouchMove')).not.toBeDefined();
expect(rootEl.prop('onTouchStart')).not.toBeDefined();
});

test.each([true, false])('should add the is-active class based on canDraw: %s', canDraw => {
const wrapper = getWrapper({ canDraw });
const rootEl = getWrapperRoot(wrapper);

expect(rootEl.hasClass('ba-RegionCreator')).toBe(true);
expect(rootEl.hasClass('is-active')).toBe(canDraw);
});
});
});
2 changes: 1 addition & 1 deletion src/store/creator/__mocks__/creatorState.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,9 +3,9 @@ import { CreatorStatus } from '../types';
export default {
cursor: 0,
error: null,
message: 'test',
staged: {
location: 1,
message: 'test',
shape: {
height: 100,
width: 100,
Expand Down
Loading

0 comments on commit f4553f7

Please sign in to comment.