UNPKG

kepler.gl

Version:

kepler.gl is a webgl based application to visualize large scale location data in the browser

556 lines (472 loc) 56.6 kB
'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); exports.default = undefined; var _extends2 = require('babel-runtime/helpers/extends'); var _extends3 = _interopRequireDefault(_extends2); var _defineProperty2 = require('babel-runtime/helpers/defineProperty'); var _defineProperty3 = _interopRequireDefault(_defineProperty2); var _classCallCheck2 = require('babel-runtime/helpers/classCallCheck'); var _classCallCheck3 = _interopRequireDefault(_classCallCheck2); var _createClass2 = require('babel-runtime/helpers/createClass'); var _createClass3 = _interopRequireDefault(_createClass2); var _possibleConstructorReturn2 = require('babel-runtime/helpers/possibleConstructorReturn'); var _possibleConstructorReturn3 = _interopRequireDefault(_possibleConstructorReturn2); var _inherits2 = require('babel-runtime/helpers/inherits'); var _inherits3 = _interopRequireDefault(_inherits2); var _taggedTemplateLiteral2 = require('babel-runtime/helpers/taggedTemplateLiteral'); var _taggedTemplateLiteral3 = _interopRequireDefault(_taggedTemplateLiteral2); var _class, _temp; var _templateObject = (0, _taggedTemplateLiteral3.default)(['\n display: flex;\n flex-direction: column;\n background-color: ', ';\n box-shadow: ', ';\n\n :focus {\n outline: 0;\n }\n'], ['\n display: flex;\n flex-direction: column;\n background-color: ', ';\n box-shadow: ', ';\n\n :focus {\n outline: 0;\n }\n']), _templateObject2 = (0, _taggedTemplateLiteral3.default)(['\n padding: 8px;\n'], ['\n padding: 8px;\n']), _templateObject3 = (0, _taggedTemplateLiteral3.default)(['\n ', '\n :hover {\n cursor: pointer;\n background-color: ', ';\n }\n'], ['\n ', '\n :hover {\n cursor: pointer;\n background-color: ', ';\n }\n']), _templateObject4 = (0, _taggedTemplateLiteral3.default)(['\n position: absolute;\n right: 15px;\n top: 14px;\n color: ', ';\n'], ['\n position: absolute;\n right: 15px;\n top: 14px;\n color: ', ';\n']); // Copyright (c) 2018 Uber Technologies, Inc. // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to deal // in the Software without restriction, including without limitation the rights // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell // copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN // THE SOFTWARE. var _react = require('react'); var _react2 = _interopRequireDefault(_react); var _propTypes = require('prop-types'); var _propTypes2 = _interopRequireDefault(_propTypes); var _fuzzy = require('fuzzy'); var _fuzzy2 = _interopRequireDefault(_fuzzy); var _classnames = require('classnames'); var _classnames2 = _interopRequireDefault(_classnames); var _styledComponents = require('styled-components'); var _styledComponents2 = _interopRequireDefault(_styledComponents); var _window = require('global/window'); var _accessor = require('./accessor'); var _accessor2 = _interopRequireDefault(_accessor); var _keyevent = require('./keyevent'); var _keyevent2 = _interopRequireDefault(_keyevent); var _dropdownList = require('./dropdown-list'); var _dropdownList2 = _interopRequireDefault(_dropdownList); var _icons = require('../icons'); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } var DEFAULT_CLASS = 'typeahead'; /** * Copied mostly from 'react-typeahead', an auto-completing text input * * Renders an text input that shows options nearby that you can use the * keyboard or mouse to select. */ var TypeaheadWrapper = _styledComponents2.default.div(_templateObject, function (props) { return props.theme.dropdownListBgd; }, function (props) { return props.theme.dropdownListShadow; }); var InputBox = _styledComponents2.default.div(_templateObject2); var TypeaheadInput = _styledComponents2.default.input(_templateObject3, function (props) { return props.theme.secondaryInput; }, function (props) { return props.theme.secondaryInputBgd; }); var InputIcon = _styledComponents2.default.div(_templateObject4, function (props) { return props.theme.inputPlaceholderColor; }); var Typeahead = (_temp = _class = function (_Component) { (0, _inherits3.default)(Typeahead, _Component); function Typeahead(props) { (0, _classCallCheck3.default)(this, Typeahead); var _this = (0, _possibleConstructorReturn3.default)(this, (Typeahead.__proto__ || Object.getPrototypeOf(Typeahead)).call(this, props)); _this._onOptionSelected = function (option, event) { if (_this.props.searchable) { // reset entry input _this.setState({ searchResults: _this.getOptionsForValue('', _this.props.options), selection: '', entryValue: '' }); } return _this.props.onOptionSelected(option, event); }; _this._onTextEntryUpdated = function () { if (_this.props.searchable) { var value = _this.entry.value; _this.setState({ searchResults: _this.getOptionsForValue(value, _this.props.options), selection: '', entryValue: value }); } }; _this._onEnter = function (event) { var selection = _this.getSelection(); if (!selection) { return _this.props.onKeyDown(event); } return _this._onOptionSelected(selection, event); }; _this.navDown = function () { _this._nav(1); }; _this.navUp = function () { _this._nav(-1); }; _this._onChange = function (event) { if (_this.props.onChange) { _this.props.onChange(event); } _this._onTextEntryUpdated(); }; _this._onKeyDown = function (event) { // If there are no visible elements, don't perform selector navigation. // Just pass this up to the upstream onKeydown handler. // Also skip if the user is pressing the shift key, since none of our handlers are looking for shift if (!_this._hasHint() || event.shiftKey) { return _this.props.onKeyDown(event); } var handler = _this.eventMap()[event.keyCode]; if (handler) { handler(event); } else { return _this.props.onKeyDown(event); } // Don't propagate the keystroke back to the DOM/browser event.preventDefault(); }; _this._onFocus = function (event) { _this.setState({ isFocused: true }); if (_this.props.onFocus) { return _this.props.onFocus(event); } }; _this._onBlur = function (event) { _this.setState({ isFocused: false }); if (_this.props.onBlur) { return _this.props.onBlur(event); } }; _this.state = { searchResults: _this.getOptionsForValue(_this.props.initialValue, _this.props.options), // This should be called something else, 'entryValue' entryValue: _this.props.value || _this.props.initialValue, // A valid typeahead value selection: _this.props.value, // Index of the selection selectionIndex: null, // Keep track of the focus state of the input element, to determine // whether to show options when empty (if showOptionsWhenEmpty is true) isFocused: false }; return _this; } (0, _createClass3.default)(Typeahead, [{ key: 'componentDidMount', value: function componentDidMount() { this.setState({ searchResults: this.getOptionsForValue('', this.props.options) }); // call focus on entry or div to trigger key events listener if (this.entry) { this.entry.focus(); } else { this.root.focus(); } } }, { key: 'componentWillReceiveProps', value: function componentWillReceiveProps(nextProps) { var searchResults = this.getOptionsForValue(this.state.entryValue, nextProps.options); this.setState({ searchResults: searchResults }); } }, { key: '_shouldSkipSearch', value: function _shouldSkipSearch(input) { var emptyValue = !input || input.trim().length === 0; // this.state must be checked because it may not be defined yet if this function // is called from within getInitialState var isFocused = this.state && this.state.isFocused; return !(this.props.showOptionsWhenEmpty && isFocused) && emptyValue; } }, { key: 'getOptionsForValue', value: function getOptionsForValue(value, options) { if (!this.props.searchable) { // directly pass through options if can not be searched return options; } if (this._shouldSkipSearch(value)) { return options; } var searchOptions = this._generateSearchFunction(); return searchOptions(value, options); } }, { key: 'focus', value: function focus() { if (this.entry) { this.entry.focus(); } } }, { key: '_hasCustomValue', value: function _hasCustomValue() { return this.props.allowCustomValues > 0 && this.state.entryValue.length >= this.props.allowCustomValues && this.state.searchResults.indexOf(this.state.entryValue) < 0; } }, { key: '_getCustomValue', value: function _getCustomValue() { return this._hasCustomValue() ? this.state.entryValue : null; } }, { key: '_renderIncrementalSearchResults', value: function _renderIncrementalSearchResults() { return _react2.default.createElement(this.props.customListComponent, { fixedOptions: this.props.fixedOptions, options: this.props.maxVisible ? this.state.searchResults.slice(0, this.props.maxVisible) : this.state.searchResults, areResultsTruncated: this.props.maxVisible && this.state.searchResults.length > this.props.maxVisible, resultsTruncatedMessage: this.props.resultsTruncatedMessage, onOptionSelected: this._onOptionSelected, allowCustomValues: this.props.allowCustomValues, customValue: this._getCustomValue(), customClasses: this.props.customClasses, customListItemComponent: this.props.customListItemComponent, customListHeaderComponent: this.props.customListHeaderComponent, selectionIndex: this.state.selectionIndex, defaultClassNames: this.props.defaultClassNames, displayOption: this.props.displayOption, selectedItems: this.props.selectedItems }); } }, { key: 'getSelection', value: function getSelection() { var index = this.state.selectionIndex; if (this._hasCustomValue()) { if (index === 0) { return this.state.entryValue; } index--; } if (this._hasFixedOptions()) { return index < this.props.fixedOptions.length ? this.props.fixedOptions[index] : this.state.searchResults[index - this.props.fixedOptions.length]; } return this.state.searchResults[index]; } // use () => {} to avoid binding 'this' }, { key: '_onEscape', value: function _onEscape() { this.setState({ selectionIndex: null }); } }, { key: '_onTab', value: function _onTab(event) { var selection = this.getSelection(); var option = selection ? selection : this.state.searchResults.length > 0 ? this.state.searchResults[0] : null; if (option === null && this._hasCustomValue()) { option = this._getCustomValue(); } if (option !== null) { return this._onOptionSelected(option, event); } } }, { key: 'eventMap', value: function eventMap(event) { var events = {}; events[_keyevent2.default.DOM_VK_UP] = this.navUp; events[_keyevent2.default.DOM_VK_DOWN] = this.navDown; events[_keyevent2.default.DOM_VK_RETURN] = events[_keyevent2.default.DOM_VK_ENTER] = this._onEnter; events[_keyevent2.default.DOM_VK_ESCAPE] = this._onEscape; events[_keyevent2.default.DOM_VK_TAB] = this._onTab; return events; } }, { key: '_nav', value: function _nav(delta) { if (!this._hasHint()) { return; } var newIndex = this.state.selectionIndex === null ? delta === 1 ? 0 : delta : this.state.selectionIndex + delta; var length = this.props.maxVisible ? this.state.searchResults.slice(0, this.props.maxVisible).length : this.state.searchResults.length; if (this._hasCustomValue()) { length += 1; } if (newIndex < 0) { newIndex += length; } else if (newIndex >= length) { newIndex -= length; } this.setState({ selectionIndex: newIndex }); } }, { key: '_renderHiddenInput', value: function _renderHiddenInput() { if (!this.props.name) { return null; } return _react2.default.createElement('input', { type: 'hidden', name: this.props.name, value: this.state.selection }); } }, { key: '_generateSearchFunction', value: function _generateSearchFunction() { var searchOptionsProp = this.props.searchOptions; var filterOptionProp = this.props.filterOption; if (typeof searchOptionsProp === 'function') { if (filterOptionProp !== null) { _window.console.warn('searchOptions prop is being used, filterOption prop will be ignored'); } return searchOptionsProp; } else if (typeof filterOptionProp === 'function') { // use custom filter option return function (value, options) { return options.filter(function (o) { return filterOptionProp(value, o); }); }; } var mapper = typeof filterOptionProp === 'string' ? _accessor2.default.generateAccessor(filterOptionProp) : _accessor2.default.IDENTITY_FN; return function (value, options) { return _fuzzy2.default.filter(value, options, { extract: mapper }).map(function (res) { return options[res.index]; }); }; } }, { key: '_hasHint', value: function _hasHint() { return this.state.searchResults.length > 0 || this._hasCustomValue(); } }, { key: '_hasFixedOptions', value: function _hasFixedOptions() { return Array.isArray(this.props.fixedOptions) && this.props.fixedOptions.length; } }, { key: 'render', value: function render() { var _this2 = this; var inputClasses = {}; inputClasses[this.props.customClasses.input] = Boolean(this.props.customClasses.input); var inputClassList = (0, _classnames2.default)(inputClasses); var classes = (0, _defineProperty3.default)({}, DEFAULT_CLASS, this.props.defaultClassNames); classes[this.props.className] = Boolean(this.props.className); var classList = (0, _classnames2.default)(classes); return _react2.default.createElement( TypeaheadWrapper, { className: classList, innerRef: function innerRef(comp) { _this2.root = comp; }, tabIndex: '0', onKeyDown: this._onKeyDown, onKeyPress: this.props.onKeyPress, onKeyUp: this.props.onKeyUp, onFocus: this._onFocus }, this._renderHiddenInput(), this.props.searchable ? _react2.default.createElement( InputBox, null, _react2.default.createElement(TypeaheadInput, (0, _extends3.default)({ innerRef: function innerRef(comp) { _this2.entry = comp; }, type: 'text', disabled: this.props.disabled }, this.props.inputProps, { placeholder: this.props.placeholder, className: inputClassList, value: this.state.entryValue, onChange: this._onChange, onBlur: this._onBlur })), _react2.default.createElement( InputIcon, null, _react2.default.createElement(_icons.Search, { height: '18px' }) ) ) : null, this._renderIncrementalSearchResults() ); } }]); return Typeahead; }(_react.Component), _class.propTypes = { name: _propTypes2.default.string, customClasses: _propTypes2.default.object, maxVisible: _propTypes2.default.number, resultsTruncatedMessage: _propTypes2.default.string, options: _propTypes2.default.arrayOf(_propTypes2.default.any), fixedOptions: _propTypes2.default.arrayOf(_propTypes2.default.any), allowCustomValues: _propTypes2.default.number, initialValue: _propTypes2.default.string, value: _propTypes2.default.string, placeholder: _propTypes2.default.string, disabled: _propTypes2.default.bool, textarea: _propTypes2.default.bool, inputProps: _propTypes2.default.object, onOptionSelected: _propTypes2.default.func, onChange: _propTypes2.default.func, onKeyDown: _propTypes2.default.func, onKeyPress: _propTypes2.default.func, onKeyUp: _propTypes2.default.func, onFocus: _propTypes2.default.func, onBlur: _propTypes2.default.func, filterOption: _propTypes2.default.oneOfType([_propTypes2.default.string, _propTypes2.default.func]), searchOptions: _propTypes2.default.func, displayOption: _propTypes2.default.oneOfType([_propTypes2.default.string, _propTypes2.default.func]), inputDisplayOption: _propTypes2.default.oneOfType([_propTypes2.default.string, _propTypes2.default.func]), formInputOption: _propTypes2.default.oneOfType([_propTypes2.default.string, _propTypes2.default.func]), defaultClassNames: _propTypes2.default.bool, customListComponent: _propTypes2.default.oneOfType([_propTypes2.default.element, _propTypes2.default.func]), customListItemComponent: _propTypes2.default.oneOfType([_propTypes2.default.element, _propTypes2.default.func]), customListHeaderComponent: _propTypes2.default.oneOfType([_propTypes2.default.element, _propTypes2.default.func]), showOptionsWhenEmpty: _propTypes2.default.bool, searchable: _propTypes2.default.bool }, _class.defaultProps = { options: [], customClasses: {}, allowCustomValues: 0, initialValue: '', value: '', placeholder: '', disabled: false, textarea: false, inputProps: {}, onOptionSelected: function onOptionSelected(option) {}, onChange: function onChange(event) {}, onKeyDown: function onKeyDown(event) {}, onKeyPress: function onKeyPress(event) {}, onKeyUp: function onKeyUp(event) {}, onFocus: function onFocus(event) {}, onBlur: function onBlur(event) {}, filterOption: null, searchOptions: null, inputDisplayOption: null, defaultClassNames: true, customListComponent: _dropdownList2.default, customListItemComponent: _dropdownList.ListItem, customListHeaderComponent: null, showOptionsWhenEmpty: true, searchable: true, resultsTruncatedMessage: null }, _temp); exports.default = Typeahead; ; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/common/item-selector/typeahead.js"],"names":["DEFAULT_CLASS","TypeaheadWrapper","styled","div","props","theme","dropdownListBgd","dropdownListShadow","InputBox","TypeaheadInput","input","secondaryInput","secondaryInputBgd","InputIcon","inputPlaceholderColor","Typeahead","_onOptionSelected","option","event","searchable","setState","searchResults","getOptionsForValue","options","selection","entryValue","onOptionSelected","_onTextEntryUpdated","value","entry","_onEnter","getSelection","onKeyDown","navDown","_nav","navUp","_onChange","onChange","_onKeyDown","_hasHint","shiftKey","handler","eventMap","keyCode","preventDefault","_onFocus","isFocused","onFocus","_onBlur","onBlur","state","initialValue","selectionIndex","focus","root","nextProps","emptyValue","trim","length","showOptionsWhenEmpty","_shouldSkipSearch","searchOptions","_generateSearchFunction","allowCustomValues","indexOf","_hasCustomValue","fixedOptions","maxVisible","slice","resultsTruncatedMessage","_getCustomValue","customClasses","customListItemComponent","customListHeaderComponent","defaultClassNames","displayOption","selectedItems","index","_hasFixedOptions","events","KeyEvent","DOM_VK_UP","DOM_VK_DOWN","DOM_VK_RETURN","DOM_VK_ENTER","DOM_VK_ESCAPE","_onEscape","DOM_VK_TAB","_onTab","delta","newIndex","name","searchOptionsProp","filterOptionProp","filterOption","Console","warn","filter","o","mapper","Accessor","generateAccessor","IDENTITY_FN","fuzzy","extract","map","res","Array","isArray","inputClasses","Boolean","inputClassList","classes","className","classList","comp","onKeyPress","onKeyUp","_renderHiddenInput","disabled","inputProps","placeholder","_renderIncrementalSearchResults","Component","propTypes","PropTypes","string","object","number","arrayOf","any","bool","textarea","func","oneOfType","inputDisplayOption","formInputOption","customListComponent","element","defaultProps","DropdownList","ListItem"],"mappings":"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","file":"typeahead.js","sourcesContent":["// Copyright (c) 2018 Uber Technologies, Inc.\n//\n// Permission is hereby granted, free of charge, to any person obtaining a copy\n// of this software and associated documentation files (the \"Software\"), to deal\n// in the Software without restriction, including without limitation the rights\n// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n// copies of the Software, and to permit persons to whom the Software is\n// furnished to do so, subject to the following conditions:\n//\n// The above copyright notice and this permission notice shall be included in\n// all copies or substantial portions of the Software.\n//\n// THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN\n// THE SOFTWARE.\n\nimport React, {Component} from 'react';\nimport PropTypes from 'prop-types';\nimport fuzzy from 'fuzzy';\nimport classNames from 'classnames';\nimport styled from 'styled-components';\nimport {console as Console} from 'global/window';\n\nimport Accessor from './accessor';\nimport KeyEvent from './keyevent';\nimport DropdownList, {ListItem} from './dropdown-list';\nimport {Search} from 'components/common/icons';\n\nconst DEFAULT_CLASS = 'typeahead';\n/**\n * Copied mostly from 'react-typeahead', an auto-completing text input\n *\n * Renders an text input that shows options nearby that you can use the\n * keyboard or mouse to select.\n */\n\nconst TypeaheadWrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  background-color: ${props => props.theme.dropdownListBgd};\n  box-shadow: ${props => props.theme.dropdownListShadow};\n\n  :focus {\n    outline: 0;\n  }\n`;\n\nconst InputBox = styled.div`\n  padding: 8px;\n`;\n\nconst TypeaheadInput = styled.input`\n  ${props => props.theme.secondaryInput}\n  :hover {\n    cursor: pointer;\n    background-color: ${props => props.theme.secondaryInputBgd};\n  }\n`;\n\nconst InputIcon = styled.div`\n  position: absolute;\n  right: 15px;\n  top: 14px;\n  color: ${props => props.theme.inputPlaceholderColor};\n`;\n\nexport default class Typeahead extends Component {\n  static propTypes = {\n    name: PropTypes.string,\n    customClasses: PropTypes.object,\n    maxVisible: PropTypes.number,\n    resultsTruncatedMessage: PropTypes.string,\n    options: PropTypes.arrayOf(PropTypes.any),\n    fixedOptions: PropTypes.arrayOf(PropTypes.any),\n    allowCustomValues: PropTypes.number,\n    initialValue: PropTypes.string,\n    value: PropTypes.string,\n    placeholder: PropTypes.string,\n    disabled: PropTypes.bool,\n    textarea: PropTypes.bool,\n    inputProps: PropTypes.object,\n    onOptionSelected: PropTypes.func,\n    onChange: PropTypes.func,\n    onKeyDown: PropTypes.func,\n    onKeyPress: PropTypes.func,\n    onKeyUp: PropTypes.func,\n    onFocus: PropTypes.func,\n    onBlur: PropTypes.func,\n    filterOption: PropTypes.oneOfType([PropTypes.string, PropTypes.func]),\n    searchOptions: PropTypes.func,\n    displayOption: PropTypes.oneOfType([PropTypes.string, PropTypes.func]),\n    inputDisplayOption: PropTypes.oneOfType([PropTypes.string, PropTypes.func]),\n    formInputOption: PropTypes.oneOfType([PropTypes.string, PropTypes.func]),\n    defaultClassNames: PropTypes.bool,\n    customListComponent: PropTypes.oneOfType([PropTypes.element, PropTypes.func]),\n    customListItemComponent: PropTypes.oneOfType([\n      PropTypes.element,\n      PropTypes.func\n    ]),\n    customListHeaderComponent: PropTypes.oneOfType([\n      PropTypes.element,\n      PropTypes.func\n    ]),\n    showOptionsWhenEmpty: PropTypes.bool,\n    searchable: PropTypes.bool\n  };\n\n  static defaultProps = {\n    options: [],\n    customClasses: {},\n    allowCustomValues: 0,\n    initialValue: '',\n    value: '',\n    placeholder: '',\n    disabled: false,\n    textarea: false,\n    inputProps: {},\n    onOptionSelected(option) {},\n    onChange(event) {},\n    onKeyDown(event) {},\n    onKeyPress(event) {},\n    onKeyUp(event) {},\n    onFocus(event) {},\n    onBlur(event) {},\n    filterOption: null,\n    searchOptions: null,\n    inputDisplayOption: null,\n    defaultClassNames: true,\n    customListComponent: DropdownList,\n    customListItemComponent: ListItem,\n    customListHeaderComponent: null,\n    showOptionsWhenEmpty: true,\n    searchable: true,\n    resultsTruncatedMessage: null\n  };\n\n  constructor(props) {\n    super(props);\n\n    this.state = {\n      searchResults: this.getOptionsForValue(\n        this.props.initialValue,\n        this.props.options\n      ),\n\n      // This should be called something else, 'entryValue'\n      entryValue: this.props.value || this.props.initialValue,\n\n      // A valid typeahead value\n      selection: this.props.value,\n\n      // Index of the selection\n      selectionIndex: null,\n\n      // Keep track of the focus state of the input element, to determine\n      // whether to show options when empty (if showOptionsWhenEmpty is true)\n      isFocused: false\n    };\n  }\n\n  componentDidMount() {\n    this.setState({\n      searchResults: this.getOptionsForValue('', this.props.options)\n    });\n\n    // call focus on entry or div to trigger key events listener\n    if (this.entry) {\n      this.entry.focus();\n    } else {\n      this.root.focus();\n    }\n  }\n\n  componentWillReceiveProps(nextProps) {\n    const searchResults = this.getOptionsForValue(\n      this.state.entryValue,\n      nextProps.options\n    );\n\n    this.setState({searchResults});\n  }\n\n  _shouldSkipSearch(input) {\n    const emptyValue = !input || input.trim().length === 0;\n\n    // this.state must be checked because it may not be defined yet if this function\n    // is called from within getInitialState\n    const isFocused = this.state && this.state.isFocused;\n    return !(this.props.showOptionsWhenEmpty && isFocused) && emptyValue;\n  }\n\n  getOptionsForValue(value, options) {\n    if (!this.props.searchable) {\n      // directly pass through options if can not be searched\n      return options;\n    }\n    if (this._shouldSkipSearch(value)) {\n      return options;\n    }\n\n    const searchOptions = this._generateSearchFunction();\n    return searchOptions(value, options);\n  }\n\n  focus() {\n    if (this.entry) {\n      this.entry.focus();\n    }\n  }\n\n  _hasCustomValue() {\n    return (\n      this.props.allowCustomValues > 0 &&\n      this.state.entryValue.length >= this.props.allowCustomValues &&\n      this.state.searchResults.indexOf(this.state.entryValue) < 0\n    );\n  }\n\n  _getCustomValue() {\n    return this._hasCustomValue() ? this.state.entryValue : null;\n  }\n\n  _renderIncrementalSearchResults() {\n    return (\n      <this.props.customListComponent\n        fixedOptions={this.props.fixedOptions}\n        options={\n          this.props.maxVisible\n            ? this.state.searchResults.slice(0, this.props.maxVisible)\n            : this.state.searchResults\n        }\n        areResultsTruncated={\n          this.props.maxVisible &&\n          this.state.searchResults.length > this.props.maxVisible\n        }\n        resultsTruncatedMessage={this.props.resultsTruncatedMessage}\n        onOptionSelected={this._onOptionSelected}\n        allowCustomValues={this.props.allowCustomValues}\n        customValue={this._getCustomValue()}\n        customClasses={this.props.customClasses}\n        customListItemComponent={this.props.customListItemComponent}\n        customListHeaderComponent={this.props.customListHeaderComponent}\n        selectionIndex={this.state.selectionIndex}\n        defaultClassNames={this.props.defaultClassNames}\n        displayOption={this.props.displayOption}\n        selectedItems={this.props.selectedItems}\n      />\n    );\n  }\n\n  getSelection() {\n    let index = this.state.selectionIndex;\n\n    if (this._hasCustomValue()) {\n      if (index === 0) {\n        return this.state.entryValue;\n      }\n      index--;\n    }\n    if (this._hasFixedOptions()) {\n      return index < this.props.fixedOptions.length\n        ? this.props.fixedOptions[index]\n        : this.state.searchResults[index - this.props.fixedOptions.length];\n    }\n    return this.state.searchResults[index];\n  }\n\n  _onOptionSelected = (option, event) => {\n    if (this.props.searchable) {\n      // reset entry input\n      this.setState({\n        searchResults: this.getOptionsForValue('', this.props.options),\n        selection: '',\n        entryValue: ''\n      });\n    }\n\n    return this.props.onOptionSelected(option, event);\n  };\n\n  // use () => {} to avoid binding 'this'\n  _onTextEntryUpdated = () => {\n    if (this.props.searchable) {\n      const value = this.entry.value;\n\n      this.setState({\n        searchResults: this.getOptionsForValue(value, this.props.options),\n        selection: '',\n        entryValue: value\n      });\n    }\n  };\n\n  _onEnter = event => {\n    const selection = this.getSelection();\n    if (!selection) {\n      return this.props.onKeyDown(event);\n    }\n    return this._onOptionSelected(selection, event);\n  };\n\n  _onEscape() {\n    this.setState({\n      selectionIndex: null\n    });\n  }\n\n  _onTab(event) {\n    const selection = this.getSelection();\n    let option = selection\n      ? selection\n      : this.state.searchResults.length > 0\n        ? this.state.searchResults[0]\n        : null;\n\n    if (option === null && this._hasCustomValue()) {\n      option = this._getCustomValue();\n    }\n\n    if (option !== null) {\n      return this._onOptionSelected(option, event);\n    }\n  }\n\n  eventMap(event) {\n    const