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
JavaScript
'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