Mailspring/internal_packages/composer/lib/participants-text-field.cjsx

137 lines
4.3 KiB
CoffeeScript

React = require 'react'
_ = require 'underscore-plus'
{Contact,
ContactStore} = require 'inbox-exports'
{TokenizingTextField} = require 'ui-components'
module.exports =
ParticipantsTextField = React.createClass
displayName: 'ParticipantsTextField'
propTypes:
# The tab index of the ParticipantsTextField
tabIndex: React.PropTypes.string,
# The name of the field, used for both display purposes and also
# to modify the `participants` provided.
field: React.PropTypes.string,
# Whether or not the field should be visible. Defaults to true.
visible: React.PropTypes.bool
# An object containing arrays of participants. Typically, this is
# {to: [], cc: [], bcc: []}. Each ParticipantsTextField needs all of
# the values, because adding an element to one field may remove it
# from another.
participants: React.PropTypes.object.isRequired,
# The function to call with an updated `participants` object when
# changes are made.
change: React.PropTypes.func.isRequired,
getDefaultProps: ->
visible: true
render: ->
classSet = {}
classSet[@props.field] = true
<div className="participants-text-field" style={zIndex: 1000-@props.tabIndex, display: @props.visible and 'inline' or 'none'}>
<TokenizingTextField
ref="textField"
prompt={@props.field}
classSet={classSet}
tabIndex={@props.tabIndex}
tokens={@props.participants[@props.field]}
onRemove={@props.onRemove}
tokenKey={ (p) -> p.email }
tokenContent={@_componentForParticipant}
completionsForInput={ (input) -> ContactStore.searchContacts(input) }
completionContent={@_completionContent}
add={@_add}
remove={@_remove}
showMenu={@_showContextMenu} />
</div>
# Public. Can be called by any component that has a ref to this one to
# focus the input field.
focus: -> @refs.textField.focus()
_completionContent: (p) ->
if p.name?.length > 0 and p.name isnt p.email
<div className="completion-participant">
<span className="participant-name">{p.name}</span>
<span className="participant-email">({p.email})</span>
</div>
else
<div className="completion-participant">
<span className="participant-name">{p.email}</span>
</div>
_componentForParticipant: (p) ->
if p.name?.length > 0 and p.name isnt p.email
<div className="participant">
<span className="participant-primary">{p.name}</span>&nbsp;&nbsp;
<span className="participant-secondary">({p.email})</span>
</div>
else
<div className="participant">
<span className="participant-primary">{p.email}</span>
</div>
_remove: (values) ->
field = @props.field
updates = {}
updates[field] = _.reject @props.participants[field], (p) ->
return true if p.email in values
return true if p.email in _.map values, (o) -> o.email
false
@props.change(updates)
_add: (values) ->
values = _.compact _.map values, (value) ->
if value instanceof Contact
return value
else if /.+@.+\..+/.test(value)
return new Contact(email: value.trim(), name: value.trim())
else
return null
updates = {}
for field in Object.keys(@props.participants)
updates[field] = [].concat(@props.participants[field])
for value in values
# first remove the participant from all the fields. This ensures
# that drag and drop isn't "drag and copy." and you can't have the
# same recipient in multiple places.
for field in Object.keys(@props.participants)
updates[field] = _.reject updates[field], (p) ->
p.email is value.email
# add the participant to field
updates[@props.field] = _.union(updates[@props.field], [value])
@props.change(updates)
""
_showContextMenu: (participant) ->
remote = require('remote')
Menu = remote.require('menu')
MenuItem = remote.require('menu-item')
menu = new Menu()
menu.append(new MenuItem(
label: "Copy #{participant.email}"
click: -> require('clipboard').writeText(participant.email)
))
menu.append(new MenuItem(
type: 'separator'
))
menu.append(new MenuItem(
label: 'Remove',
click: => @_remove(participant)
))
menu.popup(remote.getCurrentWindow())