Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fix: Unregister drawing on mode cancel #288

Merged
merged 6 commits into from
Nov 15, 2018
Merged
Show file tree
Hide file tree
Changes from 3 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
21 changes: 17 additions & 4 deletions src/controllers/AnnotationModeController.js
Original file line number Diff line number Diff line change
Expand Up @@ -599,6 +599,21 @@ class AnnotationModeController extends EventEmitter {
Object.keys(this.annotations).forEach((pageNum) => this.renderPage(pageNum));
}

/**
* Unregisters and destroys the specified thread
*
* @param {AnnotationThread} thread thread to destroy
* @return {void}
*/
destroyThread(thread: AnnotationThread) {
pramodsum marked this conversation as resolved.
Show resolved Hide resolved
if (!thread) {
return;
}

this.unregisterThread(thread);
thread.destroy();
}

/**
* Renders annotations from memory for a specified page.
*
Expand All @@ -614,8 +629,7 @@ class AnnotationModeController extends EventEmitter {
pageThreads.forEach((thread, index) => {
// Destroy any pending threads that may exist on re-render
if (thread.state === STATES.pending || thread.id === this.pendingThreadID) {
this.unregisterThread(thread);
thread.destroy();
this.destroyThread(thread);
return;
}

Expand Down Expand Up @@ -645,10 +659,9 @@ class AnnotationModeController extends EventEmitter {
const pageThreads = this.annotations[pageNum].all() || [];
pageThreads.forEach((thread) => {
if (thread.state === STATES.pending || thread.id === this.pendingThreadID) {
this.unregisterThread(thread);
this.destroyThread(thread);
hadPendingThreads = true;
this.pendingThreadID = null;
thread.destroy();
}
});
});
Expand Down
11 changes: 4 additions & 7 deletions src/controllers/DrawingModeController.js
Original file line number Diff line number Diff line change
Expand Up @@ -92,10 +92,7 @@ class DrawingModeController extends AnnotationModeController {
* @return {void}
*/
cancelDrawing(): void {
if (this.currentThread) {
this.currentThread.destroy();
}

this.destroyThread(this.currentThread);
this.exit();
}

Expand All @@ -108,7 +105,8 @@ class DrawingModeController extends AnnotationModeController {
if (
this.currentThread &&
this.currentThread.state === STATES.pending &&
this.currentThread.pathContainer.length > 0
this.currentThread.pathContainer &&
this.currentThread.pathContainer.undoStack.length > 0
) {
this.currentThread.save(TYPES.draw);
}
Expand Down Expand Up @@ -305,8 +303,7 @@ class DrawingModeController extends AnnotationModeController {
this.unbindListeners();
this.bindListeners();
} else {
this.unregisterThread(thread);
thread.destroy();
this.destroyThread(thread);
this.renderPage(thread.location.page);
}

Expand Down
6 changes: 1 addition & 5 deletions src/controllers/PointModeController.js
Original file line number Diff line number Diff line change
Expand Up @@ -64,11 +64,7 @@ class PointModeController extends AnnotationModeController {
this.annotatedElement.classList.remove(CLASS_ANNOTATION_POINT_MODE);

const thread = this.getThreadByID(this.pendingThreadID);
if (thread) {
this.unregisterThread(thread);
thread.destroy();
}

this.destroyThread(thread);
this.pendingThreadID = null;
}

Expand Down
60 changes: 60 additions & 0 deletions src/controllers/__tests__/DrawingModeController-test.js
Original file line number Diff line number Diff line change
Expand Up @@ -87,6 +87,66 @@ describe('controllers/DrawingModeController', () => {
});
});

describe('cancelDrawing()', () => {
beforeEach(() => {
controller.exit = jest.fn();
controller.unregisterThread = jest.fn();
thread = new DrawingThread();
});

it('should exit drawing mode', () => {
controller.cancelDrawing();
expect(controller.exit).toBeCalled();
expect(controller.unregisterThread).not.toBeCalled();
});

it('should destroy the current thread', () => {
controller.currentThread = thread;
controller.cancelDrawing();
expect(controller.exit).toBeCalled();
expect(controller.unregisterThread).toBeCalled();
expect(controller.currentThread.destroy).toBeCalled();
});
});

describe('postDrawing()', () => {
beforeEach(() => {
controller.exit = jest.fn();
thread = new DrawingThread();
thread.state = STATES.pending;
});

it('should exit drawing mode', () => {
controller.postDrawing();
expect(controller.exit).toBeCalled();
expect(thread.save).not.toBeCalled();
});

it('should not save non-pending threads', () => {
thread.state = STATES.inactive;
controller.currentThread = thread;
controller.postDrawing();
expect(controller.exit).toBeCalled();
expect(thread.save).not.toBeCalled();
});

it('should not save pending threads without paths', () => {
thread.pathContainer = { undoStack: [] };
controller.currentThread = thread;
controller.postDrawing();
expect(controller.exit).toBeCalled();
expect(thread.save).not.toBeCalled();
});

it('should save the current pending thread', () => {
thread.pathContainer = { undoStack: [{}] };
controller.currentThread = thread;
controller.postDrawing();
expect(controller.exit).toBeCalled();
expect(thread.save).toBeCalled();
});
});

describe('setupHandlers()', () => {
it('should successfully contain draw mode handlers if undo and redo buttons exist', () => {
controller.annotatedElement = {};
Expand Down