Mailspring/src/components/tokenizing-text-field.cjsx
Evan Morikawa 57cb02c76a feat(salesforce): associate threads with SF objects
Summary:
#### WIP! ####

This is making it all work with the association endpoint, putting
together the Salesforce Sidebar interfaces, and getting the nested
creators/updaters working.

I still need to do a bunch of UI work and actually debug the whole
workflow still

---

rename SalesforceContactStore to SalesforceSearchStore

rename SalesforceContact to SalesforceSearchResult

salesforce sidebar changes

salesforce association picker

object form store fixes

figuring out newFormItem instigators

Make SalesforceObjectFormStore declarative off SalesforceObjectStore

Make action basd handlers for SalesforceObjectStore

sidebar store create and associate

salesforce sidebar and picker fixes

association works and displays on sidebar

salesforce object form fixes

object form fixes

fix salesforce updating

Test Plan: TODO

Reviewers: bengotow

Reviewed By: bengotow

Differential Revision: https://review.inboxapp.com/D1440
2015-04-30 11:35:38 -07:00

367 lines
11 KiB
CoffeeScript

React = require 'react/addons'
classNames = require 'classnames'
_ = require 'underscore-plus'
{CompositeDisposable} = require 'event-kit'
{Contact, ContactStore} = require 'inbox-exports'
RetinaImg = require './retina-img'
{DragDropMixin} = require 'react-dnd'
Token = React.createClass
displayName: "Token"
mixins: [DragDropMixin]
propTypes:
selected: React.PropTypes.bool,
select: React.PropTypes.func.isRequired,
action: React.PropTypes.func,
item: React.PropTypes.object,
statics:
configureDragDrop: (registerType) ->
registerType('token', {
dragSource:
beginDrag: (component) ->
item: component.props.item
})
render: ->
classes = classNames
"token": true
"dragging": @getDragState('token').isDragging
"selected": @props.selected
<div {...@dragSourceFor('token')}
className={classes}
onClick={@_onSelect}>
<button className="action" onClick={@_onAction} style={marginTop: "2px"}><RetinaImg name="composer-caret.png" /></button>
{@props.children}
</div>
_onSelect: (event) ->
@props.select(@props.item)
event.preventDefault()
_onAction: (event) ->
@props.action(@props.item)
event.preventDefault()
###
The TokenizingTextField component displays a list of options as you type
and converts them into stylable tokens.
It wraps the Menu component, which takes care of the typing and keyboard
interactions.
See documentation on the propTypes for usage info.
###
module.exports =
TokenizingTextField = React.createClass
displayName: "TokenizingTextField"
propTypes:
# An array of current tokens.
#
# A token is usually an object type like a
# `Contact` or a `SalesforceObject`. The set of tokens is stored as a
# prop instead of `state`. This means that when the set of tokens
# needs to be changed, it is the parent's responsibility to make that
# change.
tokens: React.PropTypes.arrayOf(React.PropTypes.object)
# A unique ID for each token object
#
# A function that, given an object used for tokens, returns a unique
# id (key) for that object.
#
# This is necessary for React to assign each of the subitems and
# unique key.
tokenKey: React.PropTypes.func.isRequired
# What each token looks like
#
# A function that is passed an object and should return React elements
# to display that individual token.
tokenNode: React.PropTypes.func.isRequired
# The function responsible for providing a list of possible options
# given the current input.
#
# It takes the current input as a value and should return an array of
# candidate objects. These objects must be the same type as are passed
# to the `tokens` prop.
onRequestCompletions: React.PropTypes.func.isRequired
# What each suggestion looks like.
#
# This is passed through to the Menu component's `itemContent` prop.
# See components/menu.cjsx for more info.
completionNode: React.PropTypes.func.isRequired
# If the onRequestCompletions function is asynchronous, the parent will
# have to pass in the correct completions as new props.
initialCompletions: React.PropTypes.array
# Gets called when we we're ready to add whatever it is we're
# completing
#
# It's either passed an array of objects (the same ones used to
# render tokens)
#
# OR
#
# It's passed the string currently in the input field. The string case
# happens on paste and blur.
#
# The function doesn't need to return anything, but it is generally
# responible for mutating the parent's state in a way that eventually
# updates this component's `tokens` prop.
onAdd: React.PropTypes.func.isRequired
# Gets called when we remove a token
#
# It's passed an array of objects (the same ones used to render
# tokens)
#
# The function doesn't need to return anything, but it is generally
# responible for mutating the parent's state in a way that eventually
# updates this component's `tokens` prop.
onRemove: React.PropTypes.func.isRequired
# Called when we remove and there's nothing left to remove
onEmptied: React.PropTypes.func
# Gets called when the secondary action of the token gets invoked.
onTokenAction: React.PropTypes.func
# The tabIndex of the input item
tabIndex: React.PropTypes.oneOfType([
React.PropTypes.number
React.PropTypes.string
])
# A Prompt used in the head of the menu
menuPrompt: React.PropTypes.string
# A classSet hash applied to the Menu item
menuClassSet: React.PropTypes.object
mixins: [DragDropMixin]
statics:
configureDragDrop: (registerType) ->
registerType('token', {
dropTarget:
acceptDrop: (component, token) ->
component._addToken(token)
})
getInitialState: ->
completions: @props.initialCompletions ? []
inputValue: ""
selectedTokenKey: null
componentDidMount: ->
input = React.findDOMNode(@refs.input)
check = (fn) -> (event) ->
return unless event.target is input
# Wrapper to guard against events triggering on the wrong element
fn(event)
@subscriptions = new CompositeDisposable()
@subscriptions.add atom.commands.add '.tokenizing-field',
'tokenizing-field:cancel': check => @_clearInput()
'tokenizing-field:remove': check => @_removeToken()
'tokenizing-field:add-suggestion': check => @_addToken(@refs.completions.getSelectedItem() || @state.completions[0])
'tokenizing-field:add-input-value': check => @_addInputValue()
componentWillUnmount: ->
@subscriptions?.dispose()
componentWillReceiveProps: (nextProps) ->
@setState completions: nextProps.initialCompletions ? []
componentDidUpdate: ->
# Measure the width of the text in the input and
# resize the input field to fit.
input = React.findDOMNode(@refs.input)
measure = React.findDOMNode(@refs.measure)
measure.innerText = @state.inputValue
measure.style.top = input.offsetTop + "px"
measure.style.left = input.offsetLeft + "px"
input.style.width = "calc(6px + #{measure.offsetWidth}px)"
render: ->
{Menu} = require 'ui-components'
classes = classNames _.extend (@props.menuClassSet ? {}),
"tokenizing-field": true
"focused": @state.focus
"native-key-bindings": true
"empty": (@state.inputValue ? "").trim().length is 0
"has-suggestions": @state.completions.length > 0
<Menu className={classes} ref="completions"
items={@state.completions}
itemKey={ (item) -> item.id }
itemContent={@props.completionNode}
headerComponents={[@_fieldComponent()]}
onSelect={@_addToken}
/>
_fieldComponent: ->
<div key="field-component" onClick={@focus} {...@dropTargetFor('token')}>
{@_renderPrompt()}
<div className="tokenizing-field-input">
{@_fieldTokenComponents()}
<input type="text"
ref="input"
onCopy={@_onCopy}
onCut={@_onCut}
onPaste={@_onPaste}
onBlur={@_onInputBlurred}
onFocus={@_onInputFocused}
onChange={@_onInputChanged}
disabled={@props.disabled}
tabIndex={@props.tabIndex}
value={@state.inputValue} />
<span ref="measure" style={
position: 'absolute'
visibility: 'hidden'
}/>
</div>
</div>
_renderPrompt: ->
if @props.menuPrompt
<div className="tokenizing-field-label">{"#{@props.menuPrompt}:"}</div>
else
<div></div>
_fieldTokenComponents: ->
@props.tokens.map (item) =>
<Token item={item}
key={@props.tokenKey(item)}
select={@_selectToken}
action={@props.onTokenAction || @_showDefaultTokenMenu}
selected={@state.selectedTokenKey is @props.tokenKey(item)}>
{@props.tokenNode(item)}
</Token>
# Maintaining Input State
_onInputFocused: ->
@setState
completions: @_getCompletions()
focus: true
_onInputChanged: (event) ->
val = event.target.value.trimLeft()
@setState
selectedTokenKey: null
completions: @_getCompletions(val)
inputValue: val
_onInputBlurred: ->
@_addInputValue()
@setState
selectedTokenKey: null
focus: false
_clearInput: ->
@setState
completions: @_getCompletions("", clear: true)
inputValue: ""
focus: ->
React.findDOMNode(@refs.input).focus()
# Managing Tokens
_addInputValue: (input) ->
input ?= @state.inputValue
@props.onAdd(input)
@_clearInput()
_selectToken: (token) ->
@setState
selectedTokenKey: @props.tokenKey(token)
_selectedToken: ->
_.find @props.tokens, (t) =>
@props.tokenKey(t) is @state.selectedTokenKey
_addToken: (token) ->
return unless @isMounted()
return unless token
@props.onAdd([token])
@_clearInput()
@focus()
_removeToken: (token = null) ->
if @state.inputValue.trim().length is 0 and @props.tokens.length is 0 and @props.onEmptied?
@props.onEmptied()
if token
tokenToDelete = token
else if @state.selectedTokenKey
tokenToDelete = @_selectedToken()
else if @props.tokens.length > 0
@_selectToken(@props.tokens[@props.tokens.length - 1])
if tokenToDelete
@props.onRemove([tokenToDelete])
if @props.tokenKey(tokenToDelete) is @state.selectedTokenKey
@setState
selectedTokenKey: null
_showDefaultTokenMenu: (token) ->
remote = require('remote')
Menu = remote.require('menu')
MenuItem = remote.require('menu-item')
menu = new Menu()
menu.append(new MenuItem(
label: 'Remove',
click: => @_removeToken(token)
))
menu.popup(remote.getCurrentWindow())
# Copy and Paste
_onCut: (event) ->
if @state.selectedTokenKey
event.clipboardData?.setData('text/plain', @props.tokenKey(@_selectedToken()))
event.preventDefault()
@_removeToken(@_selectedToken())
_onCopy: (event) ->
if @state.selectedTokenKey
event.clipboardData.setData('text/plain', @props.tokenKey(@_selectedToken()))
event.preventDefault()
_onPaste: (event) ->
data = event.clipboardData.getData('text/plain')
@_addInputValue(data)
event.preventDefault()
# Managing Suggestions
_getCompletions: (val = @state.inputValue, {clear}={}) ->
existingKeys = _.map(@props.tokens, @props.tokenKey)
tokens = @props.onRequestCompletions(val, {clear})
if _.isArray(tokens)
_.reject tokens, (t) => @props.tokenKey(t) in existingKeys
else
# This case commonly happens when @props.onRequestCompletions returns
# a Promise object. In this case we can't synchronously return the
# new completion objects. Instead we need to wait for the parent to
# finish what it's doing and set the `initialCompletions` props.
if clear then return []
else return @props.initialCompletions ? []