mirror of
https://github.com/Foundry376/Mailspring.git
synced 2025-01-04 07:10:06 +08:00
76036e82bd
Summary: - In Gmail all threads /must/ belong to either All Mail, Trash and Spam, and they are mutually exclusive, so we need to make sure that any add/remove label operation still guarantees that constraint - Update ChangeLabelsTask to modify the set of labels to add and remove based on this rule - Update tasksFor archiving, moving to trash and moving to spam so they don't affect any other labels in the thread, as gmail does. - Removing from view /will/ remove any current labels, but will also move between all mail and trash as needed - Remove Inbox, Trash and Spam from the CategoryPicker, as Gmail does Test Plan: - Unit tests Reviewers: drew, evan, bengotow Reviewed By: drew, evan, bengotow Differential Revision: https://phab.nylas.com/D2715
165 lines
3.8 KiB
JavaScript
165 lines
3.8 KiB
JavaScript
import _ from 'underscore'
|
|
import {
|
|
Thread,
|
|
Actions,
|
|
Message,
|
|
TaskFactory,
|
|
DatabaseStore,
|
|
FocusedPerspectiveStore,
|
|
} from 'nylas-exports'
|
|
|
|
export default class ThreadListContextMenu {
|
|
constructor({threadIds = [], accountIds = []}) {
|
|
this.threadIds = threadIds
|
|
this.accountIds = accountIds
|
|
}
|
|
|
|
menuItemTemplate() {
|
|
return DatabaseStore.modelify(Thread, this.threadIds)
|
|
.then((threads) => {
|
|
this.threads = threads;
|
|
|
|
return Promise.all([
|
|
this.replyItem(),
|
|
this.replyAllItem(),
|
|
this.forwardItem(),
|
|
{type: 'separator'},
|
|
this.archiveItem(),
|
|
this.trashItem(),
|
|
this.markAsReadItem(),
|
|
this.starItem(),
|
|
// this.moveToOrLabelItem(),
|
|
// {type: 'separator'},
|
|
// this.extensionItems(),
|
|
])
|
|
}).then((menuItems) => {
|
|
return _.filter(_.compact(menuItems), (item, index) => {
|
|
if ((index === 0 || index === menuItems.length - 1) && item.type === "separator") {
|
|
return false
|
|
}
|
|
return true
|
|
});
|
|
});
|
|
}
|
|
|
|
replyItem() {
|
|
if (this.threadIds.length !== 1) { return null }
|
|
return {
|
|
label: "Reply",
|
|
click: () => {
|
|
Actions.composeReply({threadId: this.threadIds[0], popout: true});
|
|
},
|
|
}
|
|
}
|
|
|
|
replyAllItem() {
|
|
if (this.threadIds.length !== 1) { return null }
|
|
DatabaseStore.findBy(Message, {threadId: this.threadIds[0]})
|
|
.order(Message.attributes.date.descending())
|
|
.limit(1)
|
|
.then((message) => {
|
|
if (message && message.canReplyAll()) {
|
|
return {
|
|
label: "Reply All",
|
|
click: () => {
|
|
Actions.composeReplyAll({threadId: this.threadIds[0], popout: true});
|
|
},
|
|
}
|
|
}
|
|
return null
|
|
})
|
|
}
|
|
|
|
forwardItem() {
|
|
if (this.threadIds.length !== 1) { return null }
|
|
return {
|
|
label: "Forward",
|
|
click: () => {
|
|
Actions.composeForward({threadId: this.threadIds[0], popout: true});
|
|
},
|
|
}
|
|
}
|
|
|
|
archiveItem() {
|
|
const perspective = FocusedPerspectiveStore.current()
|
|
const canArchiveThreads = perspective.canArchiveThreads(this.threads)
|
|
if (!canArchiveThreads) {
|
|
return null
|
|
}
|
|
return {
|
|
label: "Archive",
|
|
click: () => {
|
|
const tasks = TaskFactory.tasksForArchiving({
|
|
threads: this.threads,
|
|
})
|
|
Actions.queueTasks(tasks)
|
|
},
|
|
}
|
|
}
|
|
|
|
trashItem() {
|
|
const perspective = FocusedPerspectiveStore.current()
|
|
const canTrashThreads = perspective.canTrashThreads(this.threads)
|
|
if (!canTrashThreads) {
|
|
return null
|
|
}
|
|
return {
|
|
label: "Trash",
|
|
click: () => {
|
|
const tasks = TaskFactory.tasksForMovingToTrash({
|
|
threads: this.threads,
|
|
})
|
|
Actions.queueTasks(tasks)
|
|
},
|
|
}
|
|
}
|
|
|
|
markAsReadItem() {
|
|
const unread = _.every(this.threads, (t) => {
|
|
return _.isMatch(t, {unread: false})
|
|
});
|
|
const dir = unread ? "Unread" : "Read"
|
|
|
|
return {
|
|
label: `Mark as ${dir}`,
|
|
click: () => {
|
|
const task = TaskFactory.taskForInvertingUnread({
|
|
threads: this.threads,
|
|
})
|
|
Actions.queueTask(task)
|
|
},
|
|
}
|
|
}
|
|
|
|
starItem() {
|
|
const starred = _.every(this.threads, (t) => {
|
|
return _.isMatch(t, {starred: false})
|
|
});
|
|
|
|
let dir = ""
|
|
let star = "Star"
|
|
if (!starred) {
|
|
dir = "Remove "
|
|
star = (this.threadIds.length > 1) ? "Stars" : "Star"
|
|
}
|
|
|
|
|
|
return {
|
|
label: `${dir}${star}`,
|
|
click: () => {
|
|
const task = TaskFactory.taskForInvertingStarred({
|
|
threads: this.threads,
|
|
})
|
|
Actions.queueTask(task)
|
|
},
|
|
}
|
|
}
|
|
|
|
displayMenu() {
|
|
const {remote} = require('electron')
|
|
this.menuItemTemplate().then((template) => {
|
|
remote.Menu.buildFromTemplate(template)
|
|
.popup(remote.getCurrentWindow());
|
|
});
|
|
}
|
|
}
|