Mailspring/internal_packages/thread-list/lib/thread-toolbar-buttons.cjsx
Ben Gotow a48ddd51f8 refactor(menus): Major prune of keymaps/commands, real conditional menus
Summary:
Keymaps & menus CSON => JSON, remove AtomKeymaps, CommandRegistry use of CSS selectors, use Mousetrap instead

Important Notes:

- The `application:` prefix is reserved for commands which are handled in the application process. Don't use it for other things. You will not receive the events in the window.

- Maintaining dynamic menus seems to come with quite an overhead, because Electron updates the entire menu every time. In the future, we'll need https://github.com/electron/electron/issues/528 to really make things nice. I will be tracking this upstream.

- The format for keyboard shortcuts has changed. `cmd-X` is now `command+shift+x`

Test Plan: Run tests

Reviewers: juan, evan

Reviewed By: evan

Differential Revision: https://phab.nylas.com/D2917
2016-04-24 12:16:25 -05:00

242 lines
6.1 KiB
CoffeeScript

_ = require 'underscore'
React = require "react"
classNames = require 'classnames'
ThreadListStore = require './thread-list-store'
{RetinaImg} = require 'nylas-component-kit'
{Actions,
TaskFactory,
AccountStore,
CategoryStore,
FocusedContentStore,
FocusedPerspectiveStore} = require "nylas-exports"
class ArchiveButton extends React.Component
@displayName: 'ArchiveButton'
@containerRequired: false
@propTypes:
items: React.PropTypes.array.isRequired
render: ->
allowed = FocusedPerspectiveStore.current().canArchiveThreads(@props.items)
return <span /> unless allowed
<button
tabIndex={-1}
style={order:-107}
className="btn btn-toolbar"
title="Archive"
onClick={@_onArchive}>
<RetinaImg name="toolbar-archive.png" mode={RetinaImg.Mode.ContentIsMask} />
</button>
_onArchive: (event) =>
tasks = TaskFactory.tasksForArchiving
threads: @props.items
Actions.queueTasks(tasks)
Actions.popSheet()
event.stopPropagation()
return
class TrashButton extends React.Component
@displayName: 'TrashButton'
@containerRequired: false
@propTypes:
items: React.PropTypes.array.isRequired
render: ->
allowed = FocusedPerspectiveStore.current().canMoveThreadsTo(@props.items, 'trash')
return <span /> unless allowed
<button tabIndex={-1}
style={order:-106}
className="btn btn-toolbar"
title="Move to Trash"
onClick={@_onRemove}>
<RetinaImg name="toolbar-trash.png" mode={RetinaImg.Mode.ContentIsMask} />
</button>
_onRemove: (event) =>
tasks = TaskFactory.tasksForMovingToTrash
threads: @props.items
Actions.queueTasks(tasks)
Actions.popSheet()
event.stopPropagation()
return
class MarkAsSpamButton extends React.Component
@displayName: 'MarkAsSpamButton'
@containerRequired: false
@propTypes:
items: React.PropTypes.array.isRequired
render: ->
allowed = FocusedPerspectiveStore.current().canMoveThreadsTo(@props.items, 'spam')
return <span /> unless allowed
<button tabIndex={-1}
style={order:-105}
className="btn btn-toolbar"
title="Mark as Spam"
onClick={@_onClick}>
<RetinaImg name="toolbar-spam.png" mode={RetinaImg.Mode.ContentIsMask} />
</button>
_onClick: (event) =>
tasks = TaskFactory.tasksForMarkingAsSpam
threads: @props.items
Actions.queueTasks(tasks)
Actions.popSheet()
event.stopPropagation()
return
class ToggleStarredButton extends React.Component
@displayName: 'ToggleStarredButton'
@containerRequired: false
@propTypes:
items: React.PropTypes.array.isRequired
render: ->
postClickStarredState = _.every @props.items, (t) -> t.starred is false
title = "Unstar"
imageName = "toolbar-star-selected.png"
if postClickStarredState
title = "Star"
imageName = "toolbar-star.png"
<button tabIndex={-1}
style={order:-103}
className="btn btn-toolbar"
title={title}
onClick={@_onStar}>
<RetinaImg name={imageName} mode={RetinaImg.Mode.ContentIsMask} />
</button>
_onStar: (event) =>
task = TaskFactory.taskForInvertingStarred(threads: @props.items)
Actions.queueTask(task)
event.stopPropagation()
return
class ToggleUnreadButton extends React.Component
@displayName: 'ToggleUnreadButton'
@containerRequired: false
@propTypes:
items: React.PropTypes.array.isRequired
render: =>
postClickUnreadState = _.every @props.items, (t) -> _.isMatch(t, {unread: false})
fragment = if postClickUnreadState then "unread" else "read"
<button tabIndex={-1}
style={order:-104}
className="btn btn-toolbar"
title="Mark as #{fragment}"
onClick={@_onClick}>
<RetinaImg name="toolbar-markas#{fragment}.png"
mode={RetinaImg.Mode.ContentIsMask} />
</button>
_onClick: (event) =>
task = TaskFactory.taskForInvertingUnread(threads: @props.items)
Actions.queueTask(task)
Actions.popSheet()
event.stopPropagation()
return
ThreadNavButtonMixin =
getInitialState: ->
@_getStateFromStores()
componentDidMount: ->
@_unsubscribe = ThreadListStore.listen @_onStoreChange
@_unsubscribe_focus = FocusedContentStore.listen @_onStoreChange
isFirstThread: ->
selectedId = FocusedContentStore.focusedId('thread')
ThreadListStore.dataSource().get(0)?.id is selectedId
isLastThread: ->
selectedId = FocusedContentStore.focusedId('thread')
lastIndex = ThreadListStore.dataSource().count() - 1
ThreadListStore.dataSource().get(lastIndex)?.id is selectedId
componentWillUnmount: ->
@_unsubscribe()
@_unsubscribe_focus()
_onStoreChange: ->
@setState @_getStateFromStores()
DownButton = React.createClass
displayName: 'DownButton'
mixins: [ThreadNavButtonMixin]
render: ->
<div className={@_classSet()} onClick={@_onClick} title="Next thread">
<RetinaImg name="toolbar-down-arrow.png" mode={RetinaImg.Mode.ContentIsMask} />
</div>
_classSet: ->
classNames
"btn-icon": true
"message-toolbar-arrow": true
"down": true
"disabled": @state.disabled
_onClick: ->
return if @state.disabled
NylasEnv.commands.dispatch('core:next-item')
return
_getStateFromStores: ->
disabled: @isLastThread()
UpButton = React.createClass
displayName: 'UpButton'
mixins: [ThreadNavButtonMixin]
render: ->
<div className={@_classSet()} onClick={@_onClick} title="Previous thread">
<RetinaImg name="toolbar-up-arrow.png" mode={RetinaImg.Mode.ContentIsMask} />
</div>
_classSet: ->
classNames
"btn-icon": true
"message-toolbar-arrow": true
"up": true
"disabled": @state.disabled
_onClick: ->
return if @state.disabled
NylasEnv.commands.dispatch('core:previous-item')
return
_getStateFromStores: ->
disabled: @isFirstThread()
UpButton.containerRequired = false
DownButton.containerRequired = false
module.exports = {
UpButton,
DownButton,
TrashButton,
MarkAsSpamButton,
ArchiveButton,
ToggleStarredButton,
ToggleUnreadButton
}