rc-select
Version:
select ui component for react
608 lines (542 loc) • 18.9 kB
JavaScript
'use strict';
Object.defineProperty(exports, '__esModule', {
value: true
});
var _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; };
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { 'default': obj }; }
function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
var _react = require('react');
var _react2 = _interopRequireDefault(_react);
var _reactDom = require('react-dom');
var _reactDom2 = _interopRequireDefault(_reactDom);
var _rcUtil = require('rc-util');
var _classnames = require('classnames');
var _classnames2 = _interopRequireDefault(_classnames);
var _OptGroup = require('./OptGroup');
var _OptGroup2 = _interopRequireDefault(_OptGroup);
var _rcAnimate = require('rc-animate');
var _rcAnimate2 = _interopRequireDefault(_rcAnimate);
var _util = require('./util');
var _SelectTrigger = require('./SelectTrigger');
var _SelectTrigger2 = _interopRequireDefault(_SelectTrigger);
function noop() {}
function filterFn(input, child) {
return String((0, _util.getPropValue)(child, this.props.optionFilterProp)).indexOf(input) > -1;
}
function saveRef(name, component) {
this[name] = component;
}
var Select = _react2['default'].createClass({
displayName: 'Select',
propTypes: {
defaultActiveFirstOption: _react.PropTypes.bool,
multiple: _react.PropTypes.bool,
filterOption: _react.PropTypes.any,
showSearch: _react.PropTypes.bool,
disabled: _react.PropTypes.bool,
showArrow: _react.PropTypes.bool,
tags: _react.PropTypes.bool,
transitionName: _react.PropTypes.string,
optionLabelProp: _react.PropTypes.string,
optionFilterProp: _react.PropTypes.string,
animation: _react.PropTypes.string,
choiceTransitionName: _react.PropTypes.string,
onChange: _react.PropTypes.func,
onSelect: _react.PropTypes.func,
onSearch: _react.PropTypes.func,
searchPlaceholder: _react.PropTypes.string,
placeholder: _react.PropTypes.any,
onDeselect: _react.PropTypes.func,
value: _react.PropTypes.oneOfType([_react.PropTypes.array, _react.PropTypes.string]),
defaultValue: _react.PropTypes.oneOfType([_react.PropTypes.array, _react.PropTypes.string]),
label: _react.PropTypes.oneOfType([_react.PropTypes.array, _react.PropTypes.any]),
defaultLabel: _react.PropTypes.oneOfType([_react.PropTypes.array, _react.PropTypes.any]),
dropdownStyle: _react.PropTypes.object,
maxTagTextLength: _react.PropTypes.number
},
getDefaultProps: function getDefaultProps() {
return {
prefixCls: 'rc-select',
filterOption: filterFn,
defaultActiveFirstOption: true,
showSearch: true,
allowClear: false,
placeholder: '',
searchPlaceholder: '',
defaultValue: [],
onChange: noop,
onSelect: noop,
onSearch: noop,
onDeselect: noop,
showArrow: true,
dropdownMatchSelectWidth: true,
dropdownStyle: {},
dropdownMenuStyle: {},
optionFilterProp: 'value',
optionLabelProp: 'value',
notFoundContent: 'Not Found'
};
},
getInitialState: function getInitialState() {
var props = this.props;
var value = [];
if ('value' in props) {
value = (0, _util.toArray)(props.value);
} else {
value = (0, _util.toArray)(props.defaultValue);
}
var label = this.getLabelFromProps(props, value, 1);
var inputValue = '';
if (props.combobox) {
inputValue = value[0] || '';
}
this.saveInputRef = saveRef.bind(this, 'inputInstance');
return { value: value, inputValue: inputValue, label: label };
},
componentWillReceiveProps: function componentWillReceiveProps(nextProps) {
if ('value' in nextProps) {
var value = (0, _util.toArray)(nextProps.value);
var label = this.getLabelFromProps(nextProps, value);
this.setState({
value: value,
label: label
});
if (nextProps.combobox) {
this.setState({
inputValue: value[0] || ''
});
}
}
},
componentDidUpdate: function componentDidUpdate() {
var state = this.state;
var props = this.props;
if (state.open && (0, _util.isMultipleOrTags)(props)) {
var inputNode = this.getInputDOMNode();
if (inputNode.value) {
inputNode.style.width = '';
inputNode.style.width = inputNode.scrollWidth + 'px';
} else {
inputNode.style.width = '';
}
}
},
componentWillUnmount: function componentWillUnmount() {
if (this.dropdownContainer) {
_reactDom2['default'].unmountComponentAtNode(this.dropdownContainer);
document.body.removeChild(this.dropdownContainer);
this.dropdownContainer = null;
}
},
onInputChange: function onInputChange(event) {
var val = event.target.value;
var props = this.props;
this.setState({
inputValue: val,
open: true
});
if ((0, _util.isCombobox)(props)) {
this.fireChange([val], [val]);
}
props.onSearch(val);
},
onDropdownVisibleChange: function onDropdownVisibleChange(open) {
this.setOpenState(open);
},
// combobox ignore
onKeyDown: function onKeyDown(event) {
var props = this.props;
if (props.disabled) {
return;
}
var keyCode = event.keyCode;
if (this.state.open && !this.getInputDOMNode()) {
this.onInputKeyDown(event);
} else if (keyCode === _rcUtil.KeyCode.ENTER || keyCode === _rcUtil.KeyCode.DOWN) {
this.setOpenState(true);
event.preventDefault();
}
},
onInputKeyDown: function onInputKeyDown(event) {
var props = this.props;
var state = this.state;
var keyCode = event.keyCode;
if ((0, _util.isMultipleOrTags)(props) && !event.target.value && keyCode === _rcUtil.KeyCode.BACKSPACE) {
var value = state.value.concat();
if (value.length) {
var label = state.label.concat();
var popValue = value.pop();
label.pop();
props.onDeselect(popValue);
this.fireChange(value, label);
}
return;
}
if (keyCode === _rcUtil.KeyCode.DOWN) {
if (!state.open) {
this.openIfHasChildren();
event.preventDefault();
event.stopPropagation();
return;
}
} else if (keyCode === _rcUtil.KeyCode.ESC) {
if (state.open) {
this.setOpenState(false);
event.preventDefault();
event.stopPropagation();
}
return;
}
if (state.open) {
var menu = this.refs.trigger.getInnerMenu();
if (menu && menu.onKeyDown(event)) {
event.preventDefault();
event.stopPropagation();
}
}
},
onMenuSelect: function onMenuSelect(_ref) {
var item = _ref.item;
var value = this.state.value;
var label = this.state.label;
var props = this.props;
var selectedValue = (0, _util.getValuePropValue)(item);
var selectedLabel = this.getLabelFromOption(item);
props.onSelect(selectedValue, item);
if ((0, _util.isMultipleOrTags)(props)) {
if (value.indexOf(selectedValue) !== -1) {
return;
}
value = value.concat([selectedValue]);
label = label.concat([selectedLabel]);
} else {
if (value[0] === selectedValue) {
this.setOpenState(false);
return;
}
value = [selectedValue];
label = [selectedLabel];
this.setOpenState(false);
}
this.fireChange(value, label);
this.setState({
inputValue: ''
});
if ((0, _util.isCombobox)(props)) {
this.setState({
inputValue: (0, _util.getPropValue)(item, props.optionLabelProp)
});
}
},
onMenuDeselect: function onMenuDeselect(_ref2) {
var item = _ref2.item;
var domEvent = _ref2.domEvent;
if (domEvent.type === 'click') {
this.removeSelected((0, _util.getValuePropValue)(item));
}
if (!(0, _util.isMultipleOrTags)(this.props)) {
this.setOpenState(false);
}
this.setState({
inputValue: ''
});
},
onPlaceholderClick: function onPlaceholderClick() {
this.getInputDOMNode().focus();
},
onClearSelection: function onClearSelection(event) {
var props = this.props;
var state = this.state;
if (props.disabled) {
return;
}
event.stopPropagation();
if (state.inputValue || state.value.length) {
this.fireChange([], []);
this.setOpenState(false);
this.setState({
inputValue: ''
});
}
},
getLabelBySingleValue: function getLabelBySingleValue(children, value) {
var _this = this;
if (value === undefined) {
return null;
}
var label = null;
_react2['default'].Children.forEach(children, function (child) {
if (child.type === _OptGroup2['default']) {
var maybe = _this.getLabelBySingleValue(child.props.children, value);
if (maybe !== null) {
label = maybe;
}
} else if ((0, _util.getValuePropValue)(child) === value) {
label = _this.getLabelFromOption(child);
}
});
return label;
},
getLabelFromOption: function getLabelFromOption(child) {
return (0, _util.getPropValue)(child, this.props.optionLabelProp);
},
getLabelFromProps: function getLabelFromProps(props, value, init) {
var label = [];
if ('label' in props) {
label = (0, _util.toArray)(props.label);
} else if (init && 'defaultLabel' in props) {
label = (0, _util.toArray)(props.defaultLabel);
} else {
label = this.getLabelByValue(props.children, value);
}
return label;
},
getVLForOnChange: function getVLForOnChange(vls) {
if (vls !== undefined) {
return (0, _util.isMultipleOrTags)(this.props) ? vls : vls[0];
}
return vls;
},
getLabelByValue: function getLabelByValue(children, values) {
var _this2 = this;
return values.map(function (value) {
var label = _this2.getLabelBySingleValue(children, value);
if (label === null) {
return value;
}
return label;
});
},
getDropdownContainer: function getDropdownContainer() {
if (!this.dropdownContainer) {
this.dropdownContainer = document.createElement('div');
document.body.appendChild(this.dropdownContainer);
}
return this.dropdownContainer;
},
getSearchPlaceholderElement: function getSearchPlaceholderElement(hidden) {
var props = this.props;
if (props.searchPlaceholder) {
return _react2['default'].createElement(
'span',
{
style: { display: hidden ? 'none' : 'block' },
onClick: this.onPlaceholderClick,
className: props.prefixCls + '-search__field__placeholder' },
props.searchPlaceholder
);
}
return null;
},
getInputElement: function getInputElement() {
var props = this.props;
return _react2['default'].createElement(
'span',
{ className: props.prefixCls + '-search__field__wrap' },
_react2['default'].createElement('input', { ref: this.saveInputRef,
onChange: this.onInputChange,
onKeyDown: this.onInputKeyDown,
value: this.state.inputValue,
disabled: props.disabled,
className: props.prefixCls + '-search__field',
role: 'textbox' }),
(0, _util.isMultipleOrTags)(props) ? null : this.getSearchPlaceholderElement(!!this.state.inputValue)
);
},
getInputDOMNode: function getInputDOMNode() {
return this.inputInstance;
},
getPopupDOMNode: function getPopupDOMNode() {
return this.refs.trigger.getPopupDOMNode();
},
getPopupMenuComponent: function getPopupMenuComponent() {
return this.refs.trigger.getInnerMenu();
},
setOpenState: function setOpenState(open) {
var _this3 = this;
var refs = this.refs;
this.setState({
open: open
}, function () {
if (open || (0, _util.isMultipleOrTagsOrCombobox)(_this3.props)) {
if (_this3.getInputDOMNode()) {
_this3.getInputDOMNode().focus();
}
} else if (refs.selection) {
refs.selection.focus();
}
});
},
removeSelected: function removeSelected(selectedValue) {
var props = this.props;
if (props.disabled) {
return;
}
var label = this.state.label.concat();
var index = this.state.value.indexOf(selectedValue);
var value = this.state.value.filter(function (singleValue) {
return singleValue !== selectedValue;
});
if (index !== -1) {
label.splice(index, 1);
}
var canMultiple = (0, _util.isMultipleOrTags)(props);
if (canMultiple) {
props.onDeselect(selectedValue);
}
this.fireChange(value, label);
},
openIfHasChildren: function openIfHasChildren() {
var props = this.props;
if (_react2['default'].Children.count(props.children) || (0, _util.isSingleMode)(props)) {
this.setOpenState(true);
}
},
fireChange: function fireChange(value, label) {
var props = this.props;
if (!('value' in props)) {
this.setState({
value: value, label: label
});
}
props.onChange(this.getVLForOnChange(value), this.getVLForOnChange(label));
},
renderTopControlNode: function renderTopControlNode() {
var _this4 = this;
var value = this.state.value;
var label = this.state.label;
var props = this.props;
var choiceTransitionName = props.choiceTransitionName;
var prefixCls = props.prefixCls;
var maxTagTextLength = props.maxTagTextLength;
// single and not combobox, input is inside dropdown
if ((0, _util.isSingleMode)(props)) {
var placeholder = _react2['default'].createElement(
'span',
{ key: 'placeholder',
className: prefixCls + '-selection__placeholder' },
props.placeholder
);
var innerNode = placeholder;
if (this.state.label[0]) {
innerNode = _react2['default'].createElement(
'span',
{ key: 'value' },
this.state.label[0]
);
}
return _react2['default'].createElement(
'span',
{ className: prefixCls + '-selection__rendered' },
innerNode
);
}
var selectedValueNodes = [];
if ((0, _util.isMultipleOrTags)(props)) {
selectedValueNodes = value.map(function (singleValue, index) {
var content = label[index];
var title = content;
if (maxTagTextLength && typeof content === 'string' && content.length > maxTagTextLength) {
content = content.slice(0, maxTagTextLength) + '...';
}
return _react2['default'].createElement(
'li',
{ className: prefixCls + '-selection__choice',
key: singleValue,
title: title },
_react2['default'].createElement(
'span',
{ className: prefixCls + '-selection__choice__content' },
content
),
_react2['default'].createElement('span', { className: prefixCls + '-selection__choice__remove',
onClick: _this4.removeSelected.bind(_this4, singleValue) })
);
});
}
selectedValueNodes.push(_react2['default'].createElement(
'li',
{ className: prefixCls + '-search ' + prefixCls + '-search--inline', key: '__input' },
this.getInputElement()
));
var className = prefixCls + '-selection__rendered';
if ((0, _util.isMultipleOrTags)(props) && choiceTransitionName) {
return _react2['default'].createElement(
_rcAnimate2['default'],
{ className: className,
component: 'ul',
transitionName: choiceTransitionName },
selectedValueNodes
);
}
return _react2['default'].createElement(
'ul',
{ className: className },
selectedValueNodes
);
},
render: function render() {
var _rootCls;
var props = this.props;
var multiple = (0, _util.isMultipleOrTags)(props);
var state = this.state;
var className = props.className;
var disabled = props.disabled;
var allowClear = props.allowClear;
var prefixCls = props.prefixCls;
var ctrlNode = this.renderTopControlNode();
var extraSelectionProps = {};
if (!(0, _util.isCombobox)(props)) {
extraSelectionProps = {
onKeyDown: this.onKeyDown,
tabIndex: 0
};
}
var rootCls = (_rootCls = {}, _defineProperty(_rootCls, className, !!className), _defineProperty(_rootCls, prefixCls, 1), _defineProperty(_rootCls, prefixCls + '-open', state.open), _defineProperty(_rootCls, prefixCls + '-combobox', (0, _util.isCombobox)(props)), _defineProperty(_rootCls, prefixCls + '-disabled', disabled), _defineProperty(_rootCls, prefixCls + '-enabled', !disabled), _rootCls);
var clear = _react2['default'].createElement('span', { key: 'clear',
className: prefixCls + '-selection__clear',
onClick: this.onClearSelection });
return _react2['default'].createElement(
_SelectTrigger2['default'],
_extends({}, props, {
options: props.children,
multiple: multiple,
disabled: disabled,
visible: state.open,
inputValue: state.inputValue,
inputElement: this.getInputElement(),
value: state.value,
onDropdownVisibleChange: this.onDropdownVisibleChange,
onMenuSelect: this.onMenuSelect,
onMenuDeselect: this.onMenuDeselect,
ref: 'trigger' }),
_react2['default'].createElement(
'span',
{
style: props.style,
className: (0, _classnames2['default'])(rootCls) },
_react2['default'].createElement(
'span',
_extends({ ref: 'selection',
key: 'selection',
className: prefixCls + '-selection ' + prefixCls + '-selection--' + (multiple ? 'multiple' : 'single'),
role: 'combobox',
'aria-autocomplete': 'list',
'aria-haspopup': 'true',
'aria-expanded': state.open
}, extraSelectionProps),
ctrlNode,
allowClear && !(0, _util.isMultipleOrTags)(props) ? clear : null,
multiple || !props.showArrow ? null : _react2['default'].createElement(
'span',
{ key: 'arrow', className: prefixCls + '-arrow', tabIndex: '-1', style: { outline: 'none' } },
_react2['default'].createElement('b', null)
),
multiple ? this.getSearchPlaceholderElement(!!this.state.inputValue || this.state.value.length) : null
)
)
);
}
});
exports['default'] = Select;
module.exports = exports['default'];