React = require 'react' _ = require 'underscore' {Utils, Contact, ContactStore} = require 'nylas-exports' {TokenizingTextField, Menu} = require 'nylas-component-kit' class ParticipantsTextField extends React.Component @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, # 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, className: React.PropTypes.string @defaultProps: visible: true shouldComponentUpdate: (nextProps, nextState) => not Utils.isEqualReact(nextProps, @props) or not Utils.isEqualReact(nextState, @state) render: => classSet = {} classSet[@props.field] = true
p.email } tokenIsValid={ (p) -> ContactStore.isValidContact(p) } tokenNode={@_tokenNode} onRequestCompletions={ (input) -> ContactStore.searchContacts(input) } completionNode={@_completionNode} onAdd={@_add} onRemove={@_remove} onEdit={@_edit} onEmptied={@props.onEmptied} onTokenAction={@_showContextMenu} tabIndex={@props.tabIndex} menuClassSet={classSet} menuPrompt={@props.field} />
# Public. Can be called by any component that has a ref to this one to # focus the input field. focus: => @refs.textField.focus() _completionNode: (p) => _tokenNode: (p) => if p.name?.length > 0 and p.name isnt p.email
{p.name}  
else
{p.email}
_tokensForString: (string, options = {}) => # If the input is a string, parse out email addresses and build # an array of contact objects. For each email address wrapped in # parentheses, look for a preceding name, if one exists. if string.length is 0 return [] contacts = ContactStore.parseContactsInString(string, options) if contacts.length > 0 return contacts else # If no contacts are returned, treat the entire string as a single # (malformed) contact object. return [new Contact(email: string, name: null)] _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) _edit: (token, replacementString) => field = @props.field tokenIndex = @props.participants[field].indexOf(token) replacements = @_tokensForString(replacementString) updates = {} updates[field] = [].concat(@props.participants[field]) updates[field].splice(tokenIndex, 1, replacements...) @props.change(updates) _add: (values, options={}) => # If the input is a string, parse out email addresses and build # an array of contact objects. For each email address wrapped in # parentheses, look for a preceding name, if one exists. if _.isString(values) values = @_tokensForString(values, options) # Safety check: remove anything from the incoming values that isn't # a Contact. We should never receive anything else in the values array. values = _.filter values, (value) -> value instanceof Contact 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') # Warning: Menu is already initialized as Menu.cjsx! MenuClass = remote.require('menu') MenuItem = remote.require('menu-item') menu = new MenuClass() 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()) module.exports = ParticipantsTextField