Mailspring/app/spec/stores/draft-store-spec.es6
Ben Gotow 1a3cca8d0a
Totally overhauled composer based on Slate (#524)
* Remove the composer contenteditable, replace with basic <textarea>

* Beginning broader cleanup of draft session

* DraftJS composer with color, style support

* Serialization/unserialization of basic styles, toolbar working

* WIP

* Switch to draft-js-plugins approach, need to revisit HTML

* Move HTML conversion functionality into plugins

* Add spellcheck context menu to editor

* Initial work on quoted text

* Further work on quoted text

* BLOCK approach

* Entity approach - better, does not bump out to top level

* Hiding and showing quoted text via CSS

* Get rid of ability to inject another subject line component

* Clean up specs, DraftFactory to ES6

* Remove old initial focus hack

* Fix focusing, initial text selection

* Remove participant “collapsing” support, it can be confusing

* Correctly terminate links on carriage returns

* Initial signature support, allow removal of uneditable blocks

* Sync body string with body editorstate

* Simplify draft editor session, finish signatures

* Templates

* Minor fixes

* Simplify link/open tracking, ensure it works

* Reorg composer, rework template editor

* Omg the slowness is all the stupid emoji button

* Polish and small fixes

* Performance improvements, new templates UI

* Don’t assume nodes are elements

* Fix for sending drafts twice due to back-to-back saves

* Fix order of operations on app quit to save drafts reliably

* Improve DraftJS-Convert whitespace handling

* Use contentID throughout attachment lifecycle

* Try to fix images

* Switch to Slate instead of DraftJS… much better

* Fix newline handling

* Bug fixes

* Cleanup

* Finish templates plugin

* Clean up text editing / support for Gmail email styles

* Support for color + size on the same node, clean trailing whitespace

* Restore emoji typeahead / emoji picker

* Fix scrolling in template editor

* Fix specs

* Fix newlines

* Re-implement spellcheck to be faster

* Make spellcheck decorator changes invisible to the undo/redo stack

* Remove comment

* Polish themplates panel

* Fix #521
2018-01-11 15:55:56 -08:00

575 lines
20 KiB
JavaScript

import {
Thread,
Actions,
Contact,
Message,
Account,
DraftStore,
DatabaseStore,
SoundRegistry,
DestroyDraftTask,
ComposerExtension,
ExtensionRegistry,
FocusedContentStore,
DatabaseWriter,
} from 'mailspring-exports';
import { remote } from 'electron';
import DraftFactory from '../../src/flux/stores/draft-factory';
class TestExtension extends ComposerExtension {
static prepareNewDraft({ draft }) {
draft.body = `Edited by TestExtension! ${draft.body}`;
}
}
xdescribe('DraftStore', function draftStore() {
beforeEach(() => {
this.fakeThread = new Thread({ id: 'fake-thread', headerMessageId: 'fake-thread' });
this.fakeMessage = new Message({ id: 'fake-message', headerMessageId: 'fake-message' });
spyOn(AppEnv, 'newWindow').andCallFake(() => {});
spyOn(DatabaseWriter.prototype, 'persistModel').andReturn(Promise.resolve());
spyOn(DatabaseStore, 'run').andCallFake(query => {
if (query._klass === Thread) {
return Promise.resolve(this.fakeThread);
}
if (query._klass === Message) {
return Promise.resolve(this.fakeMessage);
}
if (query._klass === Contact) {
return Promise.resolve(null);
}
return Promise.reject(new Error(`Not Stubbed for class ${query._klass.name}`));
});
for (const headerMessageId of Object.keys(DraftStore._draftSessions)) {
const sess = DraftStore._draftSessions[headerMessageId];
if (sess.teardown) {
DraftStore._doneWithSession(sess);
}
}
DraftStore._draftSessions = {};
});
describe('creating and opening drafts', () => {
beforeEach(() => {
const draft = new Message({ id: 'A', subject: 'B', headerMessageId: 'A', body: '123' });
this.newDraft = draft;
spyOn(DraftFactory, 'createDraftForReply').andReturn(Promise.resolve(draft));
spyOn(DraftFactory, 'createOrUpdateDraftForReply').andReturn(Promise.resolve(draft));
spyOn(DraftFactory, 'createDraftForForward').andReturn(Promise.resolve(draft));
spyOn(DraftFactory, 'createDraft').andReturn(Promise.resolve(draft));
});
it('should always attempt to focus the new draft', () => {
DraftStore._onComposeReply({
threadId: this.fakeThread.id,
messageId: this.fakeMessage.id,
type: 'reply',
behavior: 'prefer-existing',
});
advanceClock();
advanceClock();
});
describe('context', () => {
it('can accept IDs for thread and message arguments', () => {
DraftStore._onComposeReply({
threadId: this.fakeThread.id,
messageId: this.fakeMessage.id,
type: 'reply',
behavior: 'prefer-existing',
});
advanceClock();
expect(DraftFactory.createOrUpdateDraftForReply).toHaveBeenCalledWith({
thread: this.fakeThread,
message: this.fakeMessage,
type: 'reply',
behavior: 'prefer-existing',
});
});
it('can accept models for thread and message arguments', () => {
DraftStore._onComposeReply({
thread: this.fakeThread,
message: this.fakeMessage,
type: 'reply',
behavior: 'prefer-existing',
});
advanceClock();
expect(DraftFactory.createOrUpdateDraftForReply).toHaveBeenCalledWith({
thread: this.fakeThread,
message: this.fakeMessage,
type: 'reply',
behavior: 'prefer-existing',
});
});
it('can accept only a thread / threadId, and use the last message on the thread', () => {
DraftStore._onComposeReply({
thread: this.fakeThread,
type: 'reply',
behavior: 'prefer-existing',
});
advanceClock();
expect(DraftFactory.createOrUpdateDraftForReply).toHaveBeenCalledWith({
thread: this.fakeThread,
message: this.fakeMessage,
type: 'reply',
behavior: 'prefer-existing',
});
});
});
describe('popout behavior', () => {
it('can popout a reply', () => {
runs(() => {
DraftStore._onComposeReply({
threadId: this.fakeThread.id,
messageId: this.fakeMessage.id,
type: 'reply',
popout: true,
});
});
waitsFor(() => {
return DatabaseWriter.prototype.persistModel.callCount > 0;
});
runs(() => {
expect(AppEnv.newWindow).toHaveBeenCalledWith({
title: 'Message',
hidden: true,
windowKey: `composer-A`,
windowType: 'composer',
windowProps: { draftHeaderMessageId: 'A', draftJSON: this.newDraft.toJSON() },
});
});
});
it('can popout a forward', () => {
runs(() => {
DraftStore._onComposeForward({
threadId: this.fakeThread.id,
messageId: this.fakeMessage.id,
popout: true,
});
});
waitsFor(() => {
return DatabaseWriter.prototype.persistModel.callCount > 0;
});
runs(() => {
expect(AppEnv.newWindow).toHaveBeenCalledWith({
title: 'Message',
hidden: true,
windowKey: `composer-A`,
windowType: 'composer',
windowProps: { draftHeaderMessageId: 'A', draftJSON: this.newDraft.toJSON() },
});
});
});
});
});
describe('onDestroyDraft', () => {
beforeEach(() => {
this.draftSessionTeardown = jasmine.createSpy('draft teardown');
this.session = {
draft() {
return { pristine: false };
},
changes: {
commit() {
return Promise.resolve();
},
teardown() {},
},
teardown: this.draftSessionTeardown,
};
DraftStore._draftSessions = { abc: this.session };
spyOn(Actions, 'queueTask');
});
it('should teardown the draft session, ensuring no more saves are made', () => {
DraftStore._onDestroyDraft('abc');
expect(this.draftSessionTeardown).toHaveBeenCalled();
});
it('should not throw if the draft session is not in the window', () => {
expect(() => DraftStore._onDestroyDraft('other')).not.toThrow();
});
it('should queue a destroy draft task', () => {
DraftStore._onDestroyDraft('abc');
expect(Actions.queueTask).toHaveBeenCalled();
expect(Actions.queueTask.mostRecentCall.args[0] instanceof DestroyDraftTask).toBe(true);
});
it('should clean up the draft session', () => {
spyOn(DraftStore, '_doneWithSession');
DraftStore._onDestroyDraft('abc');
expect(DraftStore._doneWithSession).toHaveBeenCalledWith(this.session);
});
it("should close the window if it's a popout", () => {
spyOn(AppEnv, 'close');
spyOn(AppEnv, 'isComposerWindow').andReturn(true);
DraftStore._onDestroyDraft('abc');
expect(AppEnv.close).toHaveBeenCalled();
});
it("should NOT close the window if isn't a popout", () => {
spyOn(AppEnv, 'close');
spyOn(AppEnv, 'isComposerWindow').andReturn(false);
DraftStore._onDestroyDraft('abc');
expect(AppEnv.close).not.toHaveBeenCalled();
});
});
describe('before unloading', () => {
it('should destroy pristine drafts', () => {
DraftStore._draftSessions = {
abc: {
changes: {},
draft() {
return { pristine: true };
},
},
};
spyOn(Actions, 'queueTask');
DraftStore._onBeforeUnload();
expect(Actions.queueTask).toHaveBeenCalled();
expect(Actions.queueTask.mostRecentCall.args[0] instanceof DestroyDraftTask).toBe(true);
});
describe('when drafts return unresolved commit promises', () => {
beforeEach(() => {
this.resolve = null;
DraftStore._draftSessions = {
abc: {
changes: {
commit: () =>
new Promise(resolve => {
this.resolve = resolve;
}),
},
draft() {
return { pristine: false };
},
},
};
});
it('should return false and call window.close itself', () => {
const callback = jasmine.createSpy('callback');
expect(DraftStore._onBeforeUnload(callback)).toBe(false);
expect(callback).not.toHaveBeenCalled();
this.resolve();
advanceClock(1000);
advanceClock(1000);
expect(callback).toHaveBeenCalled();
});
});
describe('when drafts return immediately fulfilled commit promises', () => {
beforeEach(() => {
DraftStore._draftSessions = {
abc: {
changes: { commit: () => Promise.resolve() },
draft() {
return { pristine: false };
},
},
};
});
it('should still wait one tick before firing AppEnv.close again', () => {
const callback = jasmine.createSpy('callback');
expect(DraftStore._onBeforeUnload(callback)).toBe(false);
expect(callback).not.toHaveBeenCalled();
advanceClock();
advanceClock(1000);
expect(callback).toHaveBeenCalled();
});
});
describe('when there are no drafts', () => {
beforeEach(() => {
DraftStore._draftSessions = {};
});
it('should return true and allow the window to close', () => {
expect(DraftStore._onBeforeUnload()).toBe(true);
});
});
});
describe('sending a draft', () => {
beforeEach(() => {
this.draft = new Message({
headerMessageId: 'local-123',
threadId: 'thread-123',
replyToHeaderMessageId: 'message-123@localhost',
files: ['stub'],
});
DraftStore._draftSessions = {};
DraftStore._draftsSending = {};
this.forceCommit = false;
const session = {
prepare() {
return Promise.resolve(session);
},
teardown() {},
draft: () => this.draft,
changes: {
commit: ({ force } = {}) => {
this.forceCommit = force;
return Promise.resolve();
},
},
};
DraftStore._draftSessions[this.draft.headerMessageId] = session;
spyOn(DraftStore, '_doneWithSession').andCallThrough();
spyOn(DraftStore, 'trigger');
spyOn(SoundRegistry, 'playSound');
spyOn(Actions, 'queueTask');
});
it('plays a sound immediately when sending draft', () => {
spyOn(AppEnv.config, 'get').andReturn(true);
DraftStore._onSendDraft(this.draft.headerMessageId);
advanceClock();
expect(AppEnv.config.get).toHaveBeenCalledWith('core.sending.sounds');
expect(SoundRegistry.playSound).toHaveBeenCalledWith('hit-send');
});
it("doesn't plays a sound if the setting is off", () => {
spyOn(AppEnv.config, 'get').andReturn(false);
DraftStore._onSendDraft(this.draft.headerMessageId);
advanceClock();
expect(AppEnv.config.get).toHaveBeenCalledWith('core.sending.sounds');
expect(SoundRegistry.playSound).not.toHaveBeenCalled();
});
it('sets the sending state when sending', () => {
spyOn(AppEnv, 'isMainWindow').andReturn(true);
DraftStore._onSendDraft(this.draft.headerMessageId);
advanceClock();
expect(DraftStore.isSendingDraft(this.draft.headerMessageId)).toBe(true);
});
// Since all changes haven't been applied yet, we want to ensure that
// no view of the draft renders the draft as if its sending, but with
// the wrong text.
it('does NOT trigger until the latest changes have been applied', () => {
spyOn(AppEnv, 'isMainWindow').andReturn(true);
runs(() => {
DraftStore._onSendDraft(this.draft.headerMessageId);
expect(DraftStore.trigger).not.toHaveBeenCalled();
});
waitsFor(() => {
return Actions.queueTask.calls.length > 0;
});
runs(() => {
// Normally, the session.changes.commit will persist to the
// Database. Since that's stubbed out, we need to manually invoke
// to database update event to get the trigger (which we want to
// test) to fire
DraftStore._onDataChanged({
objectClass: 'Message',
objects: [{ draft: true }],
});
expect(DraftStore.isSendingDraft(this.draft.headerMessageId)).toBe(true);
expect(DraftStore.trigger).toHaveBeenCalled();
expect(DraftStore.trigger.calls.length).toBe(1);
});
});
it("returns false if the draft hasn't been seen", () => {
spyOn(AppEnv, 'isMainWindow').andReturn(true);
expect(DraftStore.isSendingDraft(this.draft.headerMessageId)).toBe(false);
});
it("closes the window if it's a popout", () => {
spyOn(AppEnv, 'getWindowType').andReturn('composer');
spyOn(AppEnv, 'isMainWindow').andReturn(false);
spyOn(AppEnv, 'close');
runs(() => {
return DraftStore._onSendDraft(this.draft.headerMessageId);
});
waitsFor('Mailspring to close', () => AppEnv.close.calls.length > 0);
});
it("doesn't close the window if it's inline", () => {
spyOn(AppEnv, 'getWindowType').andReturn('other');
spyOn(AppEnv, 'isMainWindow').andReturn(false);
spyOn(AppEnv, 'close');
spyOn(AppEnv, 'isComposerWindow').andCallThrough();
runs(() => {
DraftStore._onSendDraft(this.draft.headerMessageId);
});
waitsFor(() => AppEnv.isComposerWindow.calls.length > 0);
runs(() => {
expect(AppEnv.close).not.toHaveBeenCalled();
});
});
it("resets the sending state if there's an error", () => {
spyOn(AppEnv, 'isMainWindow').andReturn(false);
DraftStore._draftsSending[this.draft.headerMessageId] = true;
Actions.draftDeliveryFailed({
errorMessage: 'boohoo',
draftheaderMessageId: this.draft.headerMessageId,
});
expect(DraftStore.isSendingDraft(this.draft.headerMessageId)).toBe(false);
expect(DraftStore.trigger).toHaveBeenCalledWith(this.draft.headerMessageId);
});
it("displays a popup in the main window if there's an error", () => {
spyOn(AppEnv, 'isMainWindow').andReturn(true);
spyOn(FocusedContentStore, 'focused').andReturn({ id: 't1' });
spyOn(remote.dialog, 'showMessageBox');
spyOn(Actions, 'composePopoutDraft');
DraftStore._draftsSending[this.draft.headerMessageId] = true;
Actions.draftDeliveryFailed({
threadId: 't1',
errorMessage: 'boohoo',
draftheaderMessageId: this.draft.headerMessageId,
});
advanceClock(400);
expect(DraftStore.isSendingDraft(this.draft.headerMessageId)).toBe(false);
expect(DraftStore.trigger).toHaveBeenCalledWith(this.draft.headerMessageId);
expect(remote.dialog.showMessageBox).toHaveBeenCalled();
const dialogArgs = remote.dialog.showMessageBox.mostRecentCall.args[1];
expect(dialogArgs.detail).toEqual('boohoo');
expect(Actions.composePopoutDraft).not.toHaveBeenCalled();
});
it("re-opens the draft if you're not looking at the thread", () => {
spyOn(AppEnv, 'isMainWindow').andReturn(true);
spyOn(FocusedContentStore, 'focused').andReturn({ id: 't1' });
spyOn(Actions, 'composePopoutDraft');
DraftStore._draftsSending[this.draft.headerMessageId] = true;
Actions.draftDeliveryFailed({
threadId: 't2',
errorMessage: 'boohoo',
draftheaderMessageId: this.draft.headerMessageId,
});
advanceClock(400);
expect(Actions.composePopoutDraft).toHaveBeenCalled();
const call = Actions.composePopoutDraft.calls[0];
expect(call.args[0]).toBe(this.draft.headerMessageId);
expect(call.args[1]).toEqual({ errorMessage: 'boohoo' });
});
it('re-opens the draft if there is no thread id', () => {
spyOn(AppEnv, 'isMainWindow').andReturn(true);
spyOn(Actions, 'composePopoutDraft');
DraftStore._draftsSending[this.draft.headerMessageId] = true;
spyOn(FocusedContentStore, 'focused').andReturn(null);
Actions.draftDeliveryFailed({
errorMessage: 'boohoo',
draftheaderMessageId: this.draft.headerMessageId,
});
advanceClock(400);
expect(Actions.composePopoutDraft).toHaveBeenCalled();
const call = Actions.composePopoutDraft.calls[0];
expect(call.args[0]).toBe(this.draft.headerMessageId);
expect(call.args[1]).toEqual({ errorMessage: 'boohoo' });
});
});
describe('session teardown', () => {
beforeEach(() => {
spyOn(AppEnv, 'isMainWindow').andReturn(true);
this.draftTeardown = jasmine.createSpy('draft teardown');
this.session = {
headerMessageId: 'abc',
draft() {
return { pristine: false };
},
changes: {
commit() {
return Promise.resolve();
},
reset() {},
},
teardown: this.draftTeardown,
};
DraftStore._draftSessions = { abc: this.session };
DraftStore._doneWithSession(this.session);
});
it('removes from the list of draftSessions', () => {
expect(DraftStore._draftSessions.abc).toBeUndefined();
});
it('Calls teardown on the session', () => {
expect(this.draftTeardown).toHaveBeenCalled();
});
});
describe('mailto handling', () => {
beforeEach(() => {
spyOn(AppEnv, 'isMainWindow').andReturn(true);
});
describe('extensions', () => {
beforeEach(() => {
ExtensionRegistry.Composer.register(TestExtension);
});
afterEach(() => {
ExtensionRegistry.Composer.unregister(TestExtension);
});
it('should give extensions a chance to customize the draft via ext.prepareNewDraft', () => {
waitsForPromise(() => {
return DraftStore._onHandleMailtoLink({}, 'mailto:bengotow@gmail.com').then(() => {
const received = DatabaseWriter.prototype.persistModel.mostRecentCall.args[0];
expect(received.body.indexOf('Edited by TestExtension!')).toBe(0);
});
});
});
});
it('should call through to DraftFactory and popout a new draft', () => {
const draft = new Message({ headerMessageId: 'A', body: '123' });
spyOn(DraftFactory, 'createDraftForMailto').andReturn(Promise.resolve(draft));
spyOn(DraftStore, '_onPopoutDraft');
waitsForPromise(() => {
return DraftStore._onHandleMailtoLink({}, 'mailto:bengotow@gmail.com').then(() => {
const received = DatabaseWriter.prototype.persistModel.mostRecentCall.args[0];
expect(received).toEqual(draft);
expect(DraftStore._onPopoutDraft).toHaveBeenCalled();
});
});
});
});
describe('mailfiles handling', () => {
it('should popout a new draft', () => {
const defaultMe = new Contact();
spyOn(DraftStore, '_onPopoutDraft');
spyOn(Account.prototype, 'defaultMe').andReturn(defaultMe);
spyOn(Actions, 'addAttachment').andCallFake(({ onCreated }) => onCreated());
DraftStore._onHandleMailFiles({}, ['/Users/ben/file1.png', '/Users/ben/file2.png']);
waitsFor(() => DatabaseWriter.prototype.persistModel.callCount > 0);
runs(() => {
const { body, subject, from } = DatabaseWriter.prototype.persistModel.calls[0].args[0];
expect({ body, subject, from }).toEqual({ body: '', subject: '', from: [defaultMe] });
expect(DraftStore._onPopoutDraft).toHaveBeenCalled();
});
});
it('should call addAttachment for each provided file path', () => {
spyOn(Actions, 'addAttachment');
DraftStore._onHandleMailFiles({}, ['/Users/ben/file1.png', '/Users/ben/file2.png']);
waitsFor(() => Actions.addAttachment.callCount === 2);
runs(() => {
expect(Actions.addAttachment.calls[0].args[0].filePath).toEqual('/Users/ben/file1.png');
expect(Actions.addAttachment.calls[1].args[0].filePath).toEqual('/Users/ben/file2.png');
});
});
});
});