mirror of
https://github.com/Foundry376/Mailspring.git
synced 2025-01-12 11:08:10 +08:00
a20b979208
Summary: Adds the following new events: - Threads Moved to Folder - isArchive - source - folderType - folderDisplayName - numThreads - numMessages - description - isUndo - Threads Changed Labels - isArchive - source - labelTypesToAdd - labelTypesToRemove - labelDisplayNamesToAdd - labelDisplayNamesToRemove - numThreads - numMessages - description - isUndo - Threads Starred - source - numThreads - description - isUndo - Threads Unstarred - source - numThreads - description - isUndo - Threads Marked as Read - source - numThreads - description - isUndo - Threads Marked as Unread - source - numThreads - description - isUndo Each new action has a "source" property that's one of the following: - Category Picker: New Category - Category Picker: Existing Category - Toolbar Button: Message List - Toolbar Button: Thread List - Send and Archive - Context Menu: Thread List - Thread List Icon - Quick Actions: Thread List - Swipe - Keyboard Shortcut - Dragged Out of List - Snooze Move - Important Icon - Label Remove Icon - Thread Selected - Mail Rules - Dragged Into List Test Plan: manual Reviewers: juan Reviewed By: juan Differential Revision: https://phab.nylas.com/D3760
183 lines
4.3 KiB
JavaScript
183 lines
4.3 KiB
JavaScript
/* eslint global-require: 0*/
|
|
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,
|
|
type: 'reply',
|
|
behavior: 'prefer-existing-if-pristine',
|
|
});
|
|
},
|
|
}
|
|
}
|
|
|
|
replyAllItem() {
|
|
if (this.threadIds.length !== 1) {
|
|
return null;
|
|
}
|
|
|
|
return 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.composeReply({
|
|
threadId: this.threadIds[0],
|
|
popout: true,
|
|
type: 'reply-all',
|
|
behavior: 'prefer-existing-if-pristine',
|
|
});
|
|
},
|
|
}
|
|
}
|
|
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 allowed = perspective.canArchiveThreads(this.threads)
|
|
if (!allowed) {
|
|
return null
|
|
}
|
|
return {
|
|
label: "Archive",
|
|
click: () => {
|
|
const tasks = TaskFactory.tasksForArchiving({
|
|
source: "Context Menu: Thread List",
|
|
threads: this.threads,
|
|
})
|
|
Actions.queueTasks(tasks)
|
|
},
|
|
}
|
|
}
|
|
|
|
trashItem() {
|
|
const perspective = FocusedPerspectiveStore.current()
|
|
const allowed = perspective.canMoveThreadsTo(this.threads, 'trash')
|
|
if (!allowed) {
|
|
return null
|
|
}
|
|
return {
|
|
label: "Trash",
|
|
click: () => {
|
|
const tasks = TaskFactory.tasksForMovingToTrash({
|
|
source: "Context Menu: Thread List",
|
|
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({
|
|
source: "Context Menu: Thread List",
|
|
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({
|
|
source: "Context Menu: Thread List",
|
|
threads: this.threads,
|
|
})
|
|
Actions.queueTask(task)
|
|
},
|
|
}
|
|
}
|
|
|
|
displayMenu() {
|
|
const {remote} = require('electron')
|
|
this.menuItemTemplate().then((template) => {
|
|
remote.Menu.buildFromTemplate(template)
|
|
.popup(remote.getCurrentWindow());
|
|
});
|
|
}
|
|
}
|