Mailspring/spec/mailbox-perspective-spec.es6
Ben Gotow 36ab9d593b feat(unread/spam): New items in the sidebar for unread and spam
Summary:
Adds a new unified "Spam" folder and a unified "Unread" view,
which shows all the messages in your inbox which are unread.

Test Plan: Run tests

Reviewers: evan, juan

Reviewed By: juan

Differential Revision: https://phab.nylas.com/D2901
2016-04-19 11:32:33 -07:00

271 lines
9.7 KiB
JavaScript

import {
AccountStore,
MailboxPerspective,
TaskFactory,
Category,
CategoryStore,
} from 'nylas-exports'
import {Default} from '../internal_packages/thread-list/lib/category-removal-target-rulesets'
describe('MailboxPerspective', ()=> {
beforeEach(()=> {
this.accountIds = ['a1', 'a2']
this.accounts = {
'a1': {
id: 'a1',
defaultFinishedCategory: () => ({displayName: 'archive'}),
categoryIcon: ()=> null,
},
'a2': {
id: 'a2',
defaultFinishedCategory: () => ({displayName: 'trash2'}),
categoryIcon: ()=> null,
},
}
this.perspective = new MailboxPerspective(this.accountIds)
spyOn(AccountStore, 'accountForId').andCallFake((accId) => this.accounts[accId])
});
describe('isEqual', ()=> {
// TODO
});
describe('canArchiveThreads', ()=> {
it('returns false if the perspective is archive', ()=> {
const accounts = [
{canArchiveThreads: () => true},
{canArchiveThreads: () => true},
]
spyOn(AccountStore, 'accountsForItems').andReturn(accounts)
spyOn(this.perspective, 'isArchive').andReturn(true)
expect(this.perspective.canArchiveThreads()).toBe(false)
});
it('returns false if one of the accounts associated with the threads cannot archive', ()=> {
const accounts = [
{canArchiveThreads: () => true},
{canArchiveThreads: () => false},
]
spyOn(AccountStore, 'accountsForItems').andReturn(accounts)
spyOn(this.perspective, 'isArchive').andReturn(false)
expect(this.perspective.canArchiveThreads()).toBe(false)
});
it('returns true otherwise', ()=> {
const accounts = [
{canArchiveThreads: () => true},
{canArchiveThreads: () => true},
]
spyOn(AccountStore, 'accountsForItems').andReturn(accounts)
spyOn(this.perspective, 'isArchive').andReturn(false)
expect(this.perspective.canArchiveThreads()).toBe(true)
});
});
describe('canMoveThreadsTo', ()=> {
it('returns false if the perspective is the target folder', ()=> {
const accounts = [
{id: 'a'},
{id: 'b'},
]
spyOn(AccountStore, 'accountsForItems').andReturn(accounts)
spyOn(this.perspective, 'categoriesSharedName').andReturn('trash')
expect(this.perspective.canMoveThreadsTo([], 'trash')).toBe(false)
});
it('returns false if one of the accounts associated with the threads does not have the folder', ()=> {
const accounts = [
{id: 'a'},
{id: 'b'},
]
spyOn(CategoryStore, 'getStandardCategory').andReturn(null)
spyOn(AccountStore, 'accountsForItems').andReturn(accounts)
spyOn(this.perspective, 'categoriesSharedName').andReturn('inbox')
expect(this.perspective.canMoveThreadsTo([], 'trash')).toBe(false)
});
it('returns true otherwise', ()=> {
const accounts = [
{id: 'a'},
{id: 'b'},
]
const category = {id: 'cat'};
spyOn(CategoryStore, 'getStandardCategory').andReturn(category)
spyOn(AccountStore, 'accountsForItems').andReturn(accounts)
spyOn(this.perspective, 'categoriesSharedName').andReturn('inbox')
expect(this.perspective.canMoveThreadsTo([], 'trash')).toBe(true)
});
});
describe('canReceiveThreadsFromAccountIds', ()=> {
it('returns true if the thread account ids are included in the current account ids', ()=> {
expect(this.perspective.canReceiveThreadsFromAccountIds(['a1'])).toBe(true)
});
it('returns false otherwise', ()=> {
expect(this.perspective.canReceiveThreadsFromAccountIds(['a4'])).toBe(false)
expect(this.perspective.canReceiveThreadsFromAccountIds([])).toBe(false)
expect(this.perspective.canReceiveThreadsFromAccountIds()).toBe(false)
});
});
describe('tasksForRemovingItems', ()=> {
beforeEach(()=> {
this.categories = {
'a1': {
'archive': new Category({name: 'archive', displayName: 'archive', accountId: 'a1'}),
'inbox': new Category({name: 'inbox', displayName: 'inbox1', accountId: 'a1'}),
'trash': new Category({name: 'trash', displayName: 'trash1', accountId: 'a1'}),
'category': new Category({name: null, displayName: 'folder1', accountId: 'a1'}),
},
'a2': {
'archive': new Category({name: 'all', displayName: 'all', accountId: 'a2'}),
'inbox': new Category({name: 'inbox', displayName: 'inbox2', accountId: 'a2'}),
'trash': new Category({name: 'trash', displayName: 'trash2', accountId: 'a2'}),
'category': new Category({name: null, displayName: 'label2', accountId: 'a2'}),
},
}
this.threads = [
{accountId: 'a1'},
{accountId: 'a2'},
]
spyOn(TaskFactory, 'tasksForApplyingCategories')
spyOn(CategoryStore, 'getTrashCategory').andCallFake((accId)=> {
return this.categories[accId].trash
})
});
function assertMoved(accId) {
expect(TaskFactory.tasksForApplyingCategories).toHaveBeenCalled()
const {args} = TaskFactory.tasksForApplyingCategories.calls[0]
const {categoriesToRemove, categoriesToAdd} = args[0]
const assertor = {
from(originName) {
expect(categoriesToRemove(accId)[0].displayName).toEqual(originName)
return assertor
},
to(destinationName) {
expect(categoriesToAdd(accId)[0].displayName).toEqual(destinationName)
return assertor
},
}
return assertor
}
it('moves to finished category if viewing inbox', ()=> {
const perspective = MailboxPerspective.forCategories([
this.categories.a1.inbox,
this.categories.a2.inbox,
])
perspective.tasksForRemovingItems(this.threads, Default)
assertMoved('a1').from('inbox1').to('archive')
assertMoved('a2').from('inbox2').to('trash2')
});
it('moves to trash if viewing archive', ()=> {
const perspective = MailboxPerspective.forCategories([
this.categories.a1.archive,
this.categories.a2.archive,
])
perspective.tasksForRemovingItems(this.threads, Default)
assertMoved('a1').from('archive').to('trash1')
assertMoved('a2').from('all').to('trash2')
})
it('deletes permanently if viewing trash', ()=> {
// TODO
// Not currently possible
});
it('moves to default finished category if viewing category', ()=> {
const perspective = MailboxPerspective.forCategories([
this.categories.a1.category,
this.categories.a2.category,
])
perspective.tasksForRemovingItems(this.threads, Default)
assertMoved('a1').from('folder1').to('archive')
assertMoved('a2').from('label2').to('trash2')
})
it('unstars if viewing starred', ()=> {
spyOn(TaskFactory, 'taskForInvertingStarred').andReturn({some: 'task'})
const perspective = MailboxPerspective.forStarred(this.accountIds)
const tasks = perspective.tasksForRemovingItems(this.threads, Default)
expect(tasks).toEqual([{some: 'task'}])
});
it('does nothing when viewing spam or sent', ()=> {
['spam', 'sent'].forEach((invalid)=> {
const perspective = MailboxPerspective.forCategories([
new Category({name: invalid, accountId: 'a1'}),
new Category({name: invalid, accountId: 'a2'}),
])
const tasks = perspective.tasksForRemovingItems(this.threads, Default)
expect(TaskFactory.tasksForApplyingCategories).not.toHaveBeenCalled()
expect(tasks).toEqual([])
})
});
describe('when perspective is category perspective', ()=> {
it('overrides default ruleset', ()=> {
const customRuleset = {
all: ()=> ({displayName: 'my category'}),
}
const perspective = MailboxPerspective.forCategories([
this.categories.a1.category,
])
spyOn(perspective, 'categoriesSharedName').andReturn('all')
perspective.tasksForRemovingItems(this.threads, customRuleset)
assertMoved('a1').to('my category')
});
it('does not create tasks if any name in the ruleset is null', ()=> {
const customRuleset = {
all: null,
}
const perspective = MailboxPerspective.forCategories([
this.categories.a1.category,
])
spyOn(perspective, 'categoriesSharedName').andReturn('all')
const tasks = perspective.tasksForRemovingItems(this.threads, customRuleset)
expect(tasks).toEqual([])
});
});
});
describe('CategoryMailboxPerspective', ()=> {
beforeEach(()=> {
this.categories = [
new Category({displayName: 'c1', accountId: 'a1'}),
new Category({displayName: 'c2', accountId: 'a2'}),
new Category({displayName: 'c3', accountId: 'a2'}),
]
this.perspective = MailboxPerspective.forCategories(this.categories)
});
describe('canReceiveThreadsFromAccountIds', ()=> {
it('returns true if the thread account ids are included in the current account ids', ()=> {
expect(this.perspective.canReceiveThreadsFromAccountIds(['a1'])).toBe(true)
});
it('returns false otherwise', ()=> {
expect(this.perspective.canReceiveThreadsFromAccountIds(['a4'])).toBe(false)
expect(this.perspective.canReceiveThreadsFromAccountIds([])).toBe(false)
expect(this.perspective.canReceiveThreadsFromAccountIds()).toBe(false)
});
it('returns false if it is a locked category', ()=> {
this.perspective._categories.push(
new Category({name: 'sent', displayName: 'c4', accountId: 'a1'})
)
expect(this.perspective.canReceiveThreadsFromAccountIds(['a2'])).toBe(false)
});
});
describe('receiveThreads', ()=> {
// TODO
});
});
});