mirror of
https://github.com/Foundry376/Mailspring.git
synced 2025-01-04 07:10:06 +08:00
251d7c44d1
* Initial commit * SVG-based graph components * Add histogram, pull data into graphs * Loading animation, timespan descriptions * Improvements to read receipt / link tracking section * Initial pass at subject line analysis * Fixes to subject-line stats * Fix theme `ui-variables` include paths * Add “Share this report” button * Add “Learn More” button * Make it more clear how to edit your shortcuts * Merge activity-list and new activity-dashboard, move in sidebar
158 lines
5.1 KiB
CoffeeScript
158 lines
5.1 KiB
CoffeeScript
_ = require 'underscore'
|
|
_str = require 'underscore.string'
|
|
{WorkspaceStore,
|
|
MailboxPerspective,
|
|
FocusedPerspectiveStore,
|
|
SyncbackCategoryTask,
|
|
DestroyCategoryTask,
|
|
CategoryStore,
|
|
Actions,
|
|
Utils,
|
|
RegExpUtils} = require 'mailspring-exports'
|
|
{OutlineViewItem} = require 'mailspring-component-kit'
|
|
|
|
SidebarActions = require './sidebar-actions'
|
|
|
|
idForCategories = (categories) ->
|
|
_.pluck(categories, 'id').join('-')
|
|
|
|
countForItem = (perspective) ->
|
|
unreadCountEnabled = AppEnv.config.get('core.workspace.showUnreadForAllCategories')
|
|
if perspective.isInbox() or unreadCountEnabled
|
|
return perspective.unreadCount()
|
|
return 0
|
|
|
|
isItemSelected = (perspective) ->
|
|
FocusedPerspectiveStore.current().isEqual(perspective)
|
|
|
|
isItemCollapsed = (id) ->
|
|
if AppEnv.savedState.sidebarKeysCollapsed[id] isnt undefined
|
|
AppEnv.savedState.sidebarKeysCollapsed[id]
|
|
else
|
|
true
|
|
|
|
toggleItemCollapsed = (item) ->
|
|
return unless item.children.length > 0
|
|
SidebarActions.setKeyCollapsed(item.id, not isItemCollapsed(item.id))
|
|
|
|
onDeleteItem = (item) ->
|
|
# TODO Delete multiple categories at once
|
|
return if item.deleted is true
|
|
category = item.perspective.category()
|
|
return unless category
|
|
|
|
Actions.queueTask(new DestroyCategoryTask({
|
|
path: category.path,
|
|
accountId: category.accountId,
|
|
}))
|
|
|
|
onEditItem = (item, value) ->
|
|
return unless value
|
|
return if item.deleted is true
|
|
category = item.perspective.category()
|
|
return unless category
|
|
re = RegExpUtils.subcategorySplitRegex()
|
|
match = re.exec(category.displayName)
|
|
lastMatch = match
|
|
while match
|
|
lastMatch = match
|
|
match = re.exec(category.displayName)
|
|
if lastMatch
|
|
newDisplayName = category.displayName.slice(0, lastMatch.index + 1) + value
|
|
else
|
|
newDisplayName = value
|
|
if newDisplayName is category.displayName
|
|
return
|
|
|
|
Actions.queueTask(SyncbackCategoryTask.forRenaming({
|
|
accountId: category.accountId,
|
|
path: category.path,
|
|
newName: newDisplayName,
|
|
}))
|
|
|
|
|
|
class SidebarItem
|
|
|
|
@forPerspective: (id, perspective, opts = {}) ->
|
|
counterStyle = OutlineViewItem.CounterStyles.Alt if perspective.isInbox()
|
|
|
|
return Object.assign({
|
|
id: id
|
|
name: perspective.name
|
|
contextMenuLabel: perspective.name
|
|
count: countForItem(perspective)
|
|
iconName: perspective.iconName
|
|
children: []
|
|
perspective: perspective
|
|
selected: isItemSelected(perspective)
|
|
collapsed: isItemCollapsed(id) ? true
|
|
counterStyle: counterStyle
|
|
onDelete: if opts.deletable then onDeleteItem else undefined
|
|
onEdited: if opts.editable then onEditItem else undefined
|
|
onCollapseToggled: toggleItemCollapsed
|
|
|
|
onDrop: (item, event) ->
|
|
jsonString = event.dataTransfer.getData('nylas-threads-data')
|
|
jsonData = null
|
|
try
|
|
jsonData = JSON.parse(jsonString)
|
|
catch err
|
|
console.error("JSON parse error: #{err}")
|
|
return unless jsonData
|
|
item.perspective.receiveThreadIds(jsonData.threadIds)
|
|
|
|
shouldAcceptDrop: (item, event) ->
|
|
target = item.perspective
|
|
current = FocusedPerspectiveStore.current()
|
|
return false unless event.dataTransfer.types.includes('nylas-threads-data')
|
|
return false if target.isEqual(current)
|
|
|
|
# We can't inspect the drag payload until drop, so we use a dataTransfer
|
|
# type to encode the account IDs of threads currently being dragged.
|
|
accountsType = event.dataTransfer.types.find((t) => t.startsWith('nylas-accounts='))
|
|
accountIds = (accountsType || "").replace('nylas-accounts=', '').split(',')
|
|
return target.canReceiveThreadsFromAccountIds(accountIds)
|
|
|
|
onSelect: (item) ->
|
|
Actions.focusMailboxPerspective(item.perspective)
|
|
}, opts)
|
|
|
|
|
|
@forCategories: (categories = [], opts = {}) ->
|
|
id = idForCategories(categories)
|
|
contextMenuLabel = _str.capitalize(categories[0]?.displayType())
|
|
perspective = MailboxPerspective.forCategories(categories)
|
|
|
|
opts.deletable ?= true
|
|
opts.editable ?= true
|
|
opts.contextMenuLabel = contextMenuLabel
|
|
@forPerspective(id, perspective, opts)
|
|
|
|
@forStarred: (accountIds, opts = {}) ->
|
|
perspective = MailboxPerspective.forStarred(accountIds)
|
|
id = 'Starred'
|
|
id += "-#{opts.name}" if opts.name
|
|
@forPerspective(id, perspective, opts)
|
|
|
|
@forUnread: (accountIds, opts = {}) ->
|
|
categories = accountIds.map (accId) =>
|
|
CategoryStore.getCategoryByRole(accId, 'inbox')
|
|
|
|
# NOTE: It's possible for an account to not yet have an `inbox`
|
|
# category. Since the `SidebarStore` triggers on `AccountStore`
|
|
# changes, it'll trigger the exact moment an account is added to the
|
|
# config. However, the API has not yet come back with the list of
|
|
# `categories` for that account.
|
|
categories = _.compact(categories)
|
|
|
|
perspective = MailboxPerspective.forUnread(categories)
|
|
id = 'Unread'
|
|
id += "-#{opts.name}" if opts.name
|
|
@forPerspective(id, perspective, opts)
|
|
|
|
@forDrafts: (accountIds, opts = {}) ->
|
|
perspective = MailboxPerspective.forDrafts(accountIds)
|
|
id = "Drafts-#{opts.name}"
|
|
@forPerspective(id, perspective, opts)
|
|
|
|
module.exports = SidebarItem
|