adui
Version:
<div> <img src="https://wxa.wxs.qq.com/mpweb/delivery/legacy/wxadtouch/upload/t1/od834zef_52939fc6.png" style="margin:40px 0 0 -8px; background-color: #fcfcfc; box-shadow: none;" /> </div>
470 lines • 73.5 kB
JavaScript
function _typeof(o) { "@babel/helpers - typeof"; return _typeof = "function" == typeof Symbol && "symbol" == typeof Symbol.iterator ? function (o) { return typeof o; } : function (o) { return o && "function" == typeof Symbol && o.constructor === Symbol && o !== Symbol.prototype ? "symbol" : typeof o; }, _typeof(o); }
var _excluded = ["allowClear", "className", "getPopupContainer", "onChange", "options", "placeholder", "placeholderColor", "placement", "rightIcon", "searchable", "size", "theme", "value"];
function _defineProperty(e, r, t) { return (r = _toPropertyKey(r)) in e ? Object.defineProperty(e, r, { value: t, enumerable: !0, configurable: !0, writable: !0 }) : e[r] = t, e; }
function _objectWithoutProperties(e, t) { if (null == e) return {}; var o, r, i = _objectWithoutPropertiesLoose(e, t); if (Object.getOwnPropertySymbols) { var s = Object.getOwnPropertySymbols(e); for (r = 0; r < s.length; r++) o = s[r], t.includes(o) || {}.propertyIsEnumerable.call(e, o) && (i[o] = e[o]); } return i; }
function _objectWithoutPropertiesLoose(r, e) { if (null == r) return {}; var t = {}; for (var n in r) if ({}.hasOwnProperty.call(r, n)) { if (e.includes(n)) continue; t[n] = r[n]; } return t; }
function _extends() { return _extends = Object.assign ? Object.assign.bind() : function (n) { for (var e = 1; e < arguments.length; e++) { var t = arguments[e]; for (var r in t) ({}).hasOwnProperty.call(t, r) && (n[r] = t[r]); } return n; }, _extends.apply(null, arguments); }
function _classCallCheck(a, n) { if (!(a instanceof n)) throw new TypeError("Cannot call a class as a function"); }
function _defineProperties(e, r) { for (var t = 0; t < r.length; t++) { var o = r[t]; o.enumerable = o.enumerable || !1, o.configurable = !0, "value" in o && (o.writable = !0), Object.defineProperty(e, _toPropertyKey(o.key), o); } }
function _createClass(e, r, t) { return r && _defineProperties(e.prototype, r), t && _defineProperties(e, t), Object.defineProperty(e, "prototype", { writable: !1 }), e; }
function _toPropertyKey(t) { var i = _toPrimitive(t, "string"); return "symbol" == _typeof(i) ? i : i + ""; }
function _toPrimitive(t, r) { if ("object" != _typeof(t) || !t) return t; var e = t[Symbol.toPrimitive]; if (void 0 !== e) { var i = e.call(t, r || "default"); if ("object" != _typeof(i)) return i; throw new TypeError("@@toPrimitive must return a primitive value."); } return ("string" === r ? String : Number)(t); }
function _callSuper(t, o, e) { return o = _getPrototypeOf(o), _possibleConstructorReturn(t, _isNativeReflectConstruct() ? Reflect.construct(o, e || [], _getPrototypeOf(t).constructor) : o.apply(t, e)); }
function _possibleConstructorReturn(t, e) { if (e && ("object" == _typeof(e) || "function" == typeof e)) return e; if (void 0 !== e) throw new TypeError("Derived constructors may only return object or undefined"); return _assertThisInitialized(t); }
function _assertThisInitialized(e) { if (void 0 === e) throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); return e; }
function _isNativeReflectConstruct() { try { var t = !Boolean.prototype.valueOf.call(Reflect.construct(Boolean, [], function () {})); } catch (t) {} return (_isNativeReflectConstruct = function _isNativeReflectConstruct() { return !!t; })(); }
function _getPrototypeOf(t) { return _getPrototypeOf = Object.setPrototypeOf ? Object.getPrototypeOf.bind() : function (t) { return t.__proto__ || Object.getPrototypeOf(t); }, _getPrototypeOf(t); }
function _inherits(t, e) { if ("function" != typeof e && null !== e) throw new TypeError("Super expression must either be null or a function"); t.prototype = Object.create(e && e.prototype, { constructor: { value: t, writable: !0, configurable: !0 } }), Object.defineProperty(t, "prototype", { writable: !1 }), e && _setPrototypeOf(t, e); }
function _setPrototypeOf(t, e) { return _setPrototypeOf = Object.setPrototypeOf ? Object.setPrototypeOf.bind() : function (t, e) { return t.__proto__ = e, t; }, _setPrototypeOf(t, e); }
import * as React from "react";
import PropTypes from "prop-types";
import classNames from "classnames";
import RcSelect from "rc-select";
import OptGroup from "./OptGroup";
import Option from "./Option";
import omit from "../_util/omit";
import Icon from "../icon";
import { ConfigContext } from "../config-provider";
import getPlacements from "../tooltip/placements";
import "./style";
var noop = function noop() {};
var prefix = "adui-select";
var Select = function (_React$Component) {
function Select(props) {
var _this;
_classCallCheck(this, Select);
_this = _callSuper(this, Select, [props]);
_this.hash = Math.random().toString(36).slice(2);
_this.locked = false;
_this.select = void 0;
_this.search = void 0;
_this.menu = void 0;
_this.saveSelect = function (node) {
_this.select = node;
};
_this.saveSearch = function (node) {
_this.search = node;
};
_this.onDropdownVisibleChange = function (open) {
var _this$props = _this.props,
onDropdownVisibleChange = _this$props.onDropdownVisibleChange,
openProp = _this$props.open;
var _this$state = _this.state,
value = _this$state.value,
selectId = _this$state.selectId;
if (_this.locked) {
return;
}
if (openProp === null) {
_this.setState({
open: open
});
}
if (open) {
_this.preventVisibleChange();
setTimeout(function () {
if (_this.search) {
if (value) {
var id = selectId;
if (!selectId) {
var _this$search$parentNo;
id = (_this$search$parentNo = _this.search.parentNode) === null || _this$search$parentNo === void 0 || (_this$search$parentNo = _this$search$parentNo.nextSibling) === null || _this$search$parentNo === void 0 ? void 0 : _this$search$parentNo.id;
_this.setState({
selectId: id
});
}
if (id) {
var _document$getElementB, _parent$getElementsBy;
var parent = (_document$getElementB = document.getElementById(id)) === null || _document$getElementB === void 0 ? void 0 : _document$getElementB.nextSibling;
var el = parent === null || parent === void 0 || (_parent$getElementsBy = parent.getElementsByClassName("adui-select-item-option-selected")[0]) === null || _parent$getElementsBy === void 0 ? void 0 : _parent$getElementsBy.children[0];
var listHolder = parent === null || parent === void 0 ? void 0 : parent.getElementsByClassName("rc-virtual-list-holder")[0];
if (listHolder) {
listHolder.dispatchEvent(new Event("scroll"));
if (listHolder.children[0]) {
listHolder.children[0].dispatchEvent(new Event("scroll"));
}
}
if (el) {
_this.setState({
placeholderText: el.dataset.html || el.innerHTML
});
}
}
} else {
_this.setState({
placeholderText: ""
});
}
_this.search.focus();
}
}, 150);
} else {
if (_this.search) {
_this.search.value = "";
}
_this.setState({
placeholderShow: false
});
}
if (onDropdownVisibleChange) {
onDropdownVisibleChange(open);
}
};
_this.onSelect = function (value, option) {
var _this$props2 = _this.props,
onSelect = _this$props2.onSelect,
valueProp = _this$props2.value;
if (valueProp === null) {
_this.setState({
value: value
});
}
if (onSelect && valueProp !== value) {
onSelect(value, option);
}
if (_this.search) {
_this.search.value = "";
}
if (_this.select && _this.select.setInputValue) {
_this.select.setInputValue("");
}
};
_this.handleDropdownRender = function (menu) {
_this.menu = menu;
var _this$props3 = _this.props,
searchable = _this$props3.searchable,
searchPlaceholder = _this$props3.searchPlaceholder,
searchInputProps = _this$props3.searchInputProps,
topContent = _this$props3.topContent,
bottomContent = _this$props3.bottomContent;
if (searchable) {
return React.createElement("div", null, topContent, React.createElement("div", {
className: "".concat(prefix, "-search")
}, React.createElement("input", _extends({
ref: _this.saveSearch,
placeholder: searchPlaceholder,
onChange: _this.handleSearch,
onCompositionStart: _this.handleSearchStart,
onCompositionUpdate: _this.handleSearchUpdate,
onCompositionEnd: _this.handleSearchEnd,
onKeyDown: _this.handleSearchKeyDown,
onMouseDown: _this.preventVisibleChange,
onMouseUp: _this.preventVisibleChange
}, searchInputProps || {})), React.createElement(Icon, {
icon: "search",
className: "".concat(prefix, "-icon")
})), menu, bottomContent);
}
return React.createElement(React.Fragment, null, topContent, menu, bottomContent);
};
_this.handleSearchStart = function (e) {
var onSearchCompositionStart = _this.props.onSearchCompositionStart;
var target;
if (e) {
target = e.target;
} else {
target = _this.search;
}
if (onSearchCompositionStart) {
onSearchCompositionStart(target.value);
}
};
_this.handleSearchUpdate = function (e) {
var onSearchCompositionUpdate = _this.props.onSearchCompositionUpdate;
var target;
if (e) {
target = e.target;
} else {
target = _this.search;
}
if (onSearchCompositionUpdate) {
onSearchCompositionUpdate(target.value);
}
};
_this.handleSearchEnd = function (e) {
var onSearchCompositionEnd = _this.props.onSearchCompositionEnd;
var target;
if (e) {
target = e.target;
} else {
target = _this.search;
}
if (onSearchCompositionEnd) {
onSearchCompositionEnd(target.value);
}
};
_this.handleSearch = function (e) {
var target;
if (e) {
target = e.target;
} else {
target = _this.search;
}
var val = target.value;
var _this$state2 = _this.state,
placeholderShow = _this$state2.placeholderShow,
selectId = _this$state2.selectId;
if (val && !placeholderShow) {
_this.setState({
placeholderShow: true
});
}
if (!val && placeholderShow) {
_this.setState({
placeholderShow: false
});
}
if (_this.select) {
var id = selectId;
if (!selectId) {
var _target$parentNode;
id = (_target$parentNode = target.parentNode) === null || _target$parentNode === void 0 || (_target$parentNode = _target$parentNode.nextSibling) === null || _target$parentNode === void 0 ? void 0 : _target$parentNode.id;
_this.setState({
selectId: id
});
}
var realInput = document.querySelector(".adui-select-selection-search [aria-owns=\"".concat(id, "\"]"));
if (realInput) {
var lastValue = realInput.value;
realInput.value = val;
var event = new Event("input", {
bubbles: true
});
var tracker = realInput._valueTracker;
if (tracker) {
tracker.setValue(lastValue);
}
realInput.dispatchEvent(event);
}
}
};
_this.handleSearchKeyDown = function (e) {
if (_this.select && _this.select.onInputKeyDown) {
_this.select.onInputKeyDown(e);
}
};
_this.preventVisibleChange = function () {
_this.locked = true;
setTimeout(function () {
_this.locked = false;
_this.setState({
open: true
});
}, 200);
};
var defaultOpen = props.defaultOpen,
defaultValue = props.defaultValue,
_open = props.open,
_value = props.value;
var valueState;
if (_value !== null) {
if (_value === "") {
valueState = undefined;
} else {
valueState = _value;
}
} else if (defaultValue !== null) {
valueState = defaultValue;
}
var openState;
if (_open !== null) {
openState = _open;
} else if (defaultOpen !== null) {
openState = defaultOpen;
}
_this.state = {
open: openState,
placeholderShow: false,
selectId: "",
placeholderText: "",
value: valueState
};
return _this;
}
_inherits(Select, _React$Component);
return _createClass(Select, [{
key: "render",
value: function render() {
var _this2 = this;
var _this$props4 = this.props,
allowClear = _this$props4.allowClear,
className = _this$props4.className,
getPopupContainer = _this$props4.getPopupContainer,
_onChange = _this$props4.onChange,
options = _this$props4.options,
placeholder = _this$props4.placeholder,
placeholderColor = _this$props4.placeholderColor,
placement = _this$props4.placement,
rightIcon = _this$props4.rightIcon,
searchable = _this$props4.searchable,
size = _this$props4.size,
theme = _this$props4.theme,
valueProp = _this$props4.value,
otherProps = _objectWithoutProperties(_this$props4, _excluded);
var restProps = omit(otherProps, ["defaultValue", "open", "onDropdownVisibleChange", "onSelect", "searchPlaceholder", "onSearchCompositionStart", "onSearchCompositionUpdate", "onSearchCompositionEnd", "searchInputProps", "topContent", "bottomContent"]);
var _this$state3 = this.state,
openState = _this$state3.open,
placeholderShow = _this$state3.placeholderShow,
placeholderText = _this$state3.placeholderText,
valueState = _this$state3.value;
var openProps = {};
if (typeof openState === "boolean") {
openProps.open = openState;
}
if (valueState !== null) {
openProps.value = valueProp === "" ? placeholderColor ? React.createElement("span", {
style: {
color: placeholderColor
}
}, placeholder) : placeholder : valueState;
}
var transitionName = "slide-up";
return React.createElement(ConfigContext.Consumer, null, function (_ref) {
var getPopupContainerContext = _ref.getPopupContainer;
return React.createElement(RcSelect, _extends({
allowClear: allowClear,
className: classNames(className, "".concat(prefix, "-select"), "".concat(prefix, "-").concat(size), _defineProperty(_defineProperty(_defineProperty({}, "".concat(prefix, "-value_is_empty_string"), valueState === "" || valueState === undefined), "".concat(prefix, "-light"), theme === "light"), "".concat(prefix, "-select_placeholderShow"), placeholderShow)),
"data-value": valueState || placeholder,
menuItemSelectedIcon: null,
dropdownAlign: getPlacements({
alignEdge: true
})[placement || "bottomLeft"],
dropdownClassName: "adui-select-dropdown-".concat(_this2.hash, " adui-select-dropdown-").concat(size, " ").concat(searchable ? "adui-select-dropdown-searchable" : ""),
dropdownRender: _this2.handleDropdownRender,
suffixIcon: React.createElement(React.Fragment, null, placeholderShow && (typeof (placeholderText || placeholder || "") === "string" ? React.createElement("div", {
className: "".concat(prefix, "-placeholder-text"),
style: {
color: !valueState ? placeholderColor || "var(--gray-700)" : "var(--gray-900)"
},
dangerouslySetInnerHTML: {
__html: "".concat(placeholderText || placeholder || "")
}
}) : React.createElement("div", {
className: "".concat(prefix, "-placeholder-text"),
style: {
color: !valueState ? placeholderColor || "var(--gray-700)" : "var(--gray-900)"
}
}, placeholderText || placeholder || "")), React.createElement(Icon, {
icon: rightIcon || "triangle-down",
color: "var(--transparent-gray-700)"
})),
clearIcon: React.createElement("div", {
style: {
background: "radial-gradient(\n circle at 50% 50%, #fff 50%, transparent 50%\n )"
}
}, React.createElement(Icon, {
icon: "cancel-circle",
color: "var(--transparent-gray-700)"
})),
listHeight: 250,
listItemHeight: size === "large" ? 40 : size === "medium" ? 36 : 32,
notFoundContent: "\u65E0\u5339\u914D\u7ED3\u679C",
onDropdownVisibleChange: _this2.onDropdownVisibleChange,
onSelect: _this2.onSelect,
onChange: function onChange(v, ops) {
if (v === undefined && ops === undefined && allowClear) {
_this2.onSelect(v, ops);
}
if (_onChange) {
_onChange(v, ops);
}
},
defaultActiveFirstOption: false,
getPopupContainer: getPopupContainer || getPopupContainerContext,
optionLabelProp: options ? "label" : "children",
optionFilterProp: options ? "label" : "children",
options: options,
placeholder: placeholderColor ? React.createElement("span", {
style: {
color: placeholderColor
}
}, placeholder) : placeholder,
prefixCls: "adui-select",
showSearch: true,
ref: _this2.saveSelect,
transitionName: transitionName
}, openProps, restProps));
});
}
}]);
}(React.Component);
Select.type = "Select";
Select.Option = Option;
Select.OptGroup = OptGroup;
Select.propTypes = {
allowClear: PropTypes.bool,
bottomContent: PropTypes.node,
children: PropTypes.node,
className: PropTypes.string,
defaultOpen: PropTypes.bool,
defaultValue: PropTypes.oneOfType([PropTypes.string, PropTypes.number]),
dropdownMatchSelectWidth: PropTypes.bool,
getPopupContainer: PropTypes.func,
onDropdownVisibleChange: PropTypes.func,
onPopupScroll: PropTypes.func,
onSearch: PropTypes.func,
onSearchCompositionStart: PropTypes.func,
onSearchCompositionUpdate: PropTypes.func,
onSearchCompositionEnd: PropTypes.func,
searchInputProps: PropTypes.any,
onSelect: PropTypes.func,
open: PropTypes.bool,
options: PropTypes.array,
placeholder: PropTypes.any,
placeholderColor: PropTypes.string,
placement: PropTypes.oneOf(["top", "left", "right", "bottom", "topLeft", "topRight", "bottomLeft", "bottomRight", "leftTop", "leftBottom", "rightTop", "rightBottom"]),
rightIcon: PropTypes.any,
searchPlaceholder: PropTypes.string,
searchable: PropTypes.bool,
size: PropTypes.oneOf(["mini", "small", "medium", "large"]),
theme: PropTypes.oneOf([null, "light"]),
topContent: PropTypes.node,
value: PropTypes.oneOfType([PropTypes.string, PropTypes.number])
};
Select.defaultProps = {
allowClear: false,
bottomContent: null,
children: null,
className: undefined,
defaultOpen: null,
defaultValue: null,
dropdownMatchSelectWidth: true,
getPopupContainer: null,
onDropdownVisibleChange: noop,
onPopupScroll: noop,
onSearch: noop,
onSearchCompositionStart: noop,
onSearchCompositionUpdate: noop,
onSearchCompositionEnd: noop,
searchInputProps: {},
onSelect: noop,
open: null,
options: undefined,
placeholder: "请选择",
placeholderColor: undefined,
placement: "bottomLeft",
rightIcon: "triangle-down",
searchPlaceholder: "搜索",
searchable: false,
size: "small",
theme: null,
topContent: null
};
Select.getDerivedStateFromProps = function (_ref2) {
var open = _ref2.open,
value = _ref2.value;
var newState = {};
if (open !== null) {
newState.open = open;
}
if (value !== null) {
if (value === "") {
newState.value = undefined;
} else {
newState.value = value;
}
}
return Object.keys(newState).length > 0 ? newState : null;
};
export default Select;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["React","PropTypes","classNames","RcSelect","OptGroup","Option","omit","Icon","ConfigContext","getPlacements","noop","prefix","Select","_React$Component","props","_this","_classCallCheck","_callSuper","hash","Math","random","toString","slice","locked","select","search","menu","saveSelect","node","saveSearch","onDropdownVisibleChange","open","_this$props","openProp","_this$state","state","value","selectId","setState","preventVisibleChange","setTimeout","id","_this$search$parentNo","parentNode","nextSibling","_document$getElementB","_parent$getElementsBy","parent","document","getElementById","el","getElementsByClassName","children","listHolder","dispatchEvent","Event","placeholderText","dataset","html","innerHTML","focus","placeholderShow","onSelect","option","_this$props2","valueProp","setInputValue","handleDropdownRender","_this$props3","searchable","searchPlaceholder","searchInputProps","topContent","bottomContent","createElement","className","concat","_extends","ref","placeholder","onChange","handleSearch","onCompositionStart","handleSearchStart","onCompositionUpdate","handleSearchUpdate","onCompositionEnd","handleSearchEnd","onKeyDown","handleSearchKeyDown","onMouseDown","onMouseUp","icon","Fragment","e","onSearchCompositionStart","target","onSearchCompositionUpdate","onSearchCompositionEnd","val","_this$state2","_target$parentNode","realInput","querySelector","lastValue","event","bubbles","tracker","_valueTracker","setValue","onInputKeyDown","defaultOpen","defaultValue","valueState","undefined","openState","_inherits","_createClass","key","render","_this2","_this$props4","allowClear","getPopupContainer","options","placeholderColor","placement","rightIcon","size","theme","otherProps","_objectWithoutProperties","_excluded","restProps","_this$state3","openProps","style","color","transitionName","Consumer","_ref","getPopupContainerContext","_defineProperty","menuItemSelectedIcon","dropdownAlign","alignEdge","dropdownClassName","dropdownRender","suffixIcon","dangerouslySetInnerHTML","__html","clearIcon","background","listHeight","listItemHeight","notFoundContent","v","ops","defaultActiveFirstOption","optionLabelProp","optionFilterProp","prefixCls","showSearch","Component","type","propTypes","bool","string","oneOfType","number","dropdownMatchSelectWidth","func","onPopupScroll","onSearch","any","array","oneOf","defaultProps","getDerivedStateFromProps","_ref2","newState","Object","keys","length"],"sources":["../../components/select/Select.tsx"],"sourcesContent":["/* eslint-disable no-underscore-dangle */\n/* eslint-disable react/no-danger */\n/* eslint-disable no-nested-ternary */\nimport * as React from \"react\"\nimport PropTypes from \"prop-types\"\nimport classNames from \"classnames\"\nimport RcSelect from \"rc-select\"\nimport OptGroup from \"./OptGroup\"\nimport Option from \"./Option\"\nimport omit from \"../_util/omit\"\nimport Icon, { IconNames } from \"../icon\"\nimport { ConfigContext } from \"../config-provider\"\nimport { Placement } from \"../pop-trigger\"\nimport getPlacements from \"../tooltip/placements\"\nimport \"./style\"\n\n/**\n * Select 封装于 rc-select: https://github.com/react-component/select\n * rc-select 本身 props 非常非常多。而如 onDropdownVisibleChange 这样的 prop 甚至不在文档中；\n * 调用 rc-select 的内置方法，Select 实现了功能的再封装，如内嵌搜索依靠的是 this.select.onInputChange；\n * rc-select 仍然有许多值得再封装和继续学习的余地，**未来的可维护空间是非常大的**。\n */\n\nconst noop = () => {}\nconst prefix = \"adui-select\"\n\nexport interface ISelect {\n  forcePopupAlign?: () => void\n  getRootDomNode?: () => HTMLElement\n  onInputChange?: (e: React.ChangeEvent<HTMLInputElement>) => void\n  onInputKeyDown?: (e: React.KeyboardEvent<HTMLInputElement>) => void\n  setInputValue?: (value?: string) => void\n  getPopupDOMNode: () => HTMLElement\n}\n\ntype ValueType = React.ReactText\n\nexport interface ISelectProps<T extends ValueType = ValueType> {\n  [key: string]: any\n  /**\n   * 是否提供清除功能\n   */\n  allowClear?: boolean\n  /**\n   * 下拉框底部显示的自定义元素\n   */\n  bottomContent?: React.ReactNode\n  /**\n   * 子节点\n   */\n  children?: React.ReactNode\n  /**\n   * 附加类名\n   */\n  className?: string\n  /**\n   * 内部驱动：是否展开\n   */\n  defaultOpen?: boolean | null\n  /**\n   * 内部驱动：当前选中项的值\n   */\n  defaultValue?: T | null\n  /**\n   * 下拉列表是否和选择器同宽\n   */\n  dropdownMatchSelectWidth?: boolean\n  /**\n   * 指定弹出层的父级，默认为 document.body\n   */\n  getPopupContainer?: null | ((node: HTMLElement) => HTMLElement)\n  /**\n   * 下拉列表显示变化时的 handler，参数：bool\n   */\n  onDropdownVisibleChange?: (open: boolean) => void\n  /**\n   * 下拉列表滚动时的 handler, 参数：e\n   */\n  onPopupScroll?: (e: React.MouseEvent<HTMLDivElement>) => void\n  /**\n   * 搜索时的 handler, 参数：string\n   */\n  onSearch?: (val: string) => void\n  /**\n   * 搜索 CompositionStart 的 handler, 参数：string\n   */\n  onSearchCompositionStart?: (val: string) => void\n  /**\n   * 搜索 CompositionUpdate 的 handler, 参数：string\n   */\n  onSearchCompositionUpdate?: (val: string) => void\n  /**\n   * 搜索 CompositionEnd 的 handler, 参数：string\n   */\n  onSearchCompositionEnd?: (val: string) => void\n  /**\n   * 更多自定义 search input 的 Prop\n   */\n  searchInputProps?: {\n    [key: string]: any\n  }\n  /**\n   * 选择时的 handler，参数：(value, option)\n   */\n  onSelect?: (value: T, option: React.ReactElement<any>) => void\n  /**\n   * 外部控制：是否展开\n   */\n  open?: null | boolean\n  /**\n   * 可直接传入 options，替代手动构造 children jsx 的方式，需传入每项的 label 和 value\n   */\n  options?: {\n    [key: string]: any\n    className?: string\n    disabled?: boolean\n    label: React.ReactNode\n    value: T\n  }[]\n  /**\n   * 选择框默认文字\n   */\n  placeholder?: React.ReactNode\n  /**\n   * 选择框默认文字的颜色\n   */\n  placeholderColor?: string\n  /**\n   * 设置 placement\n   */\n  placement?: Placement\n  /**\n   * 设置右图标\n   */\n  rightIcon?: IconNames\n  /**\n   * 搜索框默认文字\n   */\n  searchable?: boolean\n  /**\n   * 是否需要内嵌搜索\n   */\n  searchPlaceholder?: string\n  /**\n   * 设置尺寸，跟着 button 走\n   */\n  size?: \"mini\" | \"small\" | \"medium\" | \"large\"\n  /**\n   * 设置主题\n   */\n  theme?: null | \"light\"\n  /**\n   * 下拉框顶部显示的自定义元素\n   */\n  topContent?: React.ReactNode\n  /**\n   * 外部控制：当前选中项的值\n   */\n  value?: T | null\n}\n\nexport interface ISelectState<T extends ValueType = ValueType> {\n  open?: boolean\n  placeholderShow?: boolean\n  placeholderText?: string\n  value?: T | null\n  selectId?: string\n}\n\n/**\n * 选择器用于选择某项内容。\n * 选择器对比单选 Radio 的优势是，当选项过多时，选择器可对内容收起，并更关注于已选项。\n * 通常，当用户能够通过已选项，轻易得知其余选项的规律时（如年份、城市等），选择器 Select 是比较好的选择。\n */\nclass Select<T extends ValueType = ValueType> extends React.Component<\n  ISelectProps<T>,\n  ISelectState<T>\n> {\n  public static type = \"Select\"\n\n  public static Option: typeof Option = Option\n\n  public static OptGroup: typeof OptGroup = OptGroup\n\n  public static propTypes = {\n    /**\n     * 是否提供清除功能\n     */\n    allowClear: PropTypes.bool,\n    /**\n     * 下拉框底部显示的自定义元素\n     */\n    bottomContent: PropTypes.node,\n    /**\n     * 子节点\n     */\n    children: PropTypes.node,\n    /**\n     * 附加类名\n     */\n    className: PropTypes.string,\n    /**\n     * 内部驱动：是否展开\n     */\n    defaultOpen: PropTypes.bool,\n    /**\n     * 内部驱动：当前选中项的值\n     */\n    defaultValue: PropTypes.oneOfType([PropTypes.string, PropTypes.number]),\n    /**\n     * 下拉列表是否和选择器同宽\n     */\n    dropdownMatchSelectWidth: PropTypes.bool,\n    /**\n     * 指定弹出层的父级，默认为 document.body\n     */\n    getPopupContainer: PropTypes.func,\n    /**\n     * 下拉列表显示变化时的 handler，参数：bool\n     */\n    onDropdownVisibleChange: PropTypes.func,\n    /**\n     * 下拉列表滚动时的 handler, 参数：e\n     */\n    onPopupScroll: PropTypes.func,\n    /**\n     * 搜索时的 handler, 参数：string\n     */\n    onSearch: PropTypes.func,\n    /**\n     * 搜索 CompositionStart 的 handler, 参数：string\n     */\n    onSearchCompositionStart: PropTypes.func,\n    /**\n     * 搜索 CompositionUpdate 的 handler, 参数：string\n     */\n    onSearchCompositionUpdate: PropTypes.func,\n    /**\n     * 搜索 CompositionEnd 的 handler, 参数：string\n     */\n    onSearchCompositionEnd: PropTypes.func,\n    /**\n     * 更多自定义 search input 的 Prop\n     */\n    searchInputProps: PropTypes.any,\n    /**\n     * 选择时的 handler，参数：(value, option)\n     */\n    onSelect: PropTypes.func,\n    /**\n     * 外部控制：是否展开\n     */\n    open: PropTypes.bool,\n    /**\n     * 可直接传入 options，替代手动构造 children jsx 的方式，需传入每项的 label 和 value\n     */\n    options: PropTypes.array,\n    /**\n     * 选择框默认文字\n     */\n    placeholder: PropTypes.any,\n    /**\n     * 选择框默认文字的颜色\n     */\n    placeholderColor: PropTypes.string,\n    /**\n     * 设置 placement\n     */\n    placement: PropTypes.oneOf([\n      \"top\",\n      \"left\",\n      \"right\",\n      \"bottom\",\n      \"topLeft\",\n      \"topRight\",\n      \"bottomLeft\",\n      \"bottomRight\",\n      \"leftTop\",\n      \"leftBottom\",\n      \"rightTop\",\n      \"rightBottom\",\n    ]),\n    /**\n     * 设置右图标\n     */\n    rightIcon: PropTypes.any,\n    /**\n     * 搜索框默认文字\n     */\n    searchPlaceholder: PropTypes.string,\n    /**\n     * 是否需要内嵌搜索\n     */\n    searchable: PropTypes.bool,\n    /**\n     * 设置尺寸，跟着 button 走\n     */\n    size: PropTypes.oneOf([\"mini\", \"small\", \"medium\", \"large\"]),\n    /**\n     * 设置主题\n     */\n    theme: PropTypes.oneOf([null, \"light\"]),\n    /**\n     * 下拉框顶部显示的自定义元素\n     */\n    topContent: PropTypes.node,\n    /**\n     * 外部控制：当前选中项的值\n     */\n    value: PropTypes.oneOfType([PropTypes.string, PropTypes.number]),\n  }\n\n  public static defaultProps: ISelectProps = {\n    allowClear: false,\n    bottomContent: null,\n    children: null,\n    className: undefined,\n    defaultOpen: null,\n    defaultValue: null,\n    dropdownMatchSelectWidth: true,\n    getPopupContainer: null,\n    onDropdownVisibleChange: noop,\n    onPopupScroll: noop,\n    onSearch: noop,\n    onSearchCompositionStart: noop,\n    onSearchCompositionUpdate: noop,\n    onSearchCompositionEnd: noop,\n    searchInputProps: {},\n    onSelect: noop,\n    open: null,\n    options: undefined,\n    placeholder: \"请选择\",\n    placeholderColor: undefined,\n    placement: \"bottomLeft\",\n    rightIcon: \"triangle-down\",\n    searchPlaceholder: \"搜索\",\n    searchable: false,\n    size: \"small\",\n    theme: null,\n    topContent: null,\n  }\n\n  public static getDerivedStateFromProps = ({ open, value }: ISelectProps) => {\n    const newState: ISelectState = {}\n    if (open !== null) {\n      newState.open = open\n    }\n    if (value !== null) {\n      if (value === \"\") {\n        newState.value = undefined\n      } else {\n        newState.value = value\n      }\n    }\n    return Object.keys(newState).length > 0 ? newState : null\n  }\n\n  public hash = Math.random().toString(36).slice(2)\n\n  public locked = false\n\n  public select: ISelect\n\n  public search: HTMLInputElement\n\n  public menu: JSX.Element\n\n  constructor(props: ISelectProps<T>) {\n    super(props)\n    const { defaultOpen, defaultValue, open, value } = props\n\n    let valueState: T | undefined\n    if (value !== null) {\n      if (value === \"\") {\n        valueState = undefined\n      } else {\n        valueState = value\n      }\n    } else if (defaultValue !== null) {\n      valueState = defaultValue\n    }\n\n    let openState\n    if (open !== null) {\n      openState = open\n    } else if (defaultOpen !== null) {\n      openState = defaultOpen\n    }\n    this.state = {\n      open: openState,\n      placeholderShow: false,\n      selectId: \"\",\n      placeholderText: \"\",\n      value: valueState,\n    }\n  }\n\n  public saveSelect = (node: ISelect) => {\n    this.select = node\n  }\n\n  public saveSearch = (node: HTMLInputElement) => {\n    this.search = node\n  }\n\n  public onDropdownVisibleChange = (open: boolean) => {\n    const { onDropdownVisibleChange, open: openProp } = this.props\n    const { value, selectId } = this.state\n\n    if (this.locked) {\n      return\n    }\n    if (openProp === null) {\n      this.setState({ open })\n    }\n    if (open) {\n      this.preventVisibleChange()\n\n      setTimeout(() => {\n        if (this.search) {\n          if (value) {\n            let id = selectId\n            if (!selectId) {\n              id = (this.search.parentNode?.nextSibling as Element)?.id\n              this.setState({ selectId: id })\n            }\n            if (id) {\n              const parent = document.getElementById(id)?.nextSibling as Element\n\n              const el = parent?.getElementsByClassName(\n                \"adui-select-item-option-selected\"\n              )[0]?.children[0] as HTMLElement\n\n              const listHolder = parent?.getElementsByClassName(\n                \"rc-virtual-list-holder\"\n              )[0] as HTMLElement\n\n              if (listHolder) {\n                listHolder.dispatchEvent(new Event(\"scroll\"))\n\n                if (listHolder.children[0]) {\n                  listHolder.children[0].dispatchEvent(new Event(\"scroll\"))\n                }\n              }\n\n              if (el) {\n                this.setState({\n                  placeholderText: el.dataset.html || el.innerHTML,\n                })\n              }\n            }\n          } else {\n            this.setState({\n              placeholderText: \"\",\n            })\n          }\n          this.search.focus()\n        }\n      }, 150)\n    } else {\n      if (this.search) {\n        this.search.value = \"\"\n      }\n      this.setState({\n        placeholderShow: false,\n      })\n    }\n    if (onDropdownVisibleChange) {\n      onDropdownVisibleChange(open)\n    }\n  }\n\n  public onSelect = (value: T, option: any) => {\n    const { onSelect, value: valueProp } = this.props\n\n    if (valueProp === null) {\n      this.setState({\n        value,\n      })\n    }\n    if (onSelect && valueProp !== value) {\n      onSelect(value, option)\n    }\n\n    if (this.search) {\n      this.search.value = \"\"\n    }\n    if (this.select && this.select.setInputValue) {\n      this.select.setInputValue(\"\")\n    }\n  }\n\n  public handleDropdownRender = (menu: JSX.Element) => {\n    this.menu = menu\n    const {\n      searchable,\n      searchPlaceholder,\n      searchInputProps,\n      topContent,\n      bottomContent,\n    } = this.props\n    if (searchable) {\n      return (\n        <div>\n          {topContent}\n          <div className={`${prefix}-search`}>\n            <input\n              ref={this.saveSearch}\n              placeholder={searchPlaceholder}\n              onChange={this.handleSearch}\n              onCompositionStart={this.handleSearchStart}\n              onCompositionUpdate={this.handleSearchUpdate}\n              onCompositionEnd={this.handleSearchEnd}\n              onKeyDown={this.handleSearchKeyDown}\n              onMouseDown={this.preventVisibleChange}\n              onMouseUp={this.preventVisibleChange}\n              {...(searchInputProps || {})}\n            />\n            <Icon icon=\"search\" className={`${prefix}-icon`} />\n          </div>\n          {menu}\n          {bottomContent}\n        </div>\n      )\n    }\n    return (\n      <>\n        {topContent}\n        {menu}\n        {bottomContent}\n      </>\n    )\n  }\n\n  public handleSearchStart: React.CompositionEventHandler<HTMLInputElement> = (\n    e\n  ) => {\n    const { onSearchCompositionStart } = this.props\n    let target: HTMLInputElement\n    if (e) {\n      target = e.target as HTMLInputElement\n    } else {\n      target = this.search\n    }\n    if (onSearchCompositionStart) {\n      onSearchCompositionStart(target.value)\n    }\n  }\n\n  public handleSearchUpdate: React.CompositionEventHandler<HTMLInputElement> = (\n    e\n  ) => {\n    const { onSearchCompositionUpdate } = this.props\n    let target: HTMLInputElement\n    if (e) {\n      target = e.target as HTMLInputElement\n    } else {\n      target = this.search\n    }\n    if (onSearchCompositionUpdate) {\n      onSearchCompositionUpdate(target.value)\n    }\n  }\n\n  public handleSearchEnd: React.CompositionEventHandler<HTMLInputElement> = (\n    e\n  ) => {\n    const { onSearchCompositionEnd } = this.props\n    let target: HTMLInputElement\n    if (e) {\n      target = e.target as HTMLInputElement\n    } else {\n      target = this.search\n    }\n    if (onSearchCompositionEnd) {\n      onSearchCompositionEnd(target.value)\n    }\n  }\n\n  public handleSearch = (e?: React.ChangeEvent<HTMLInputElement>) => {\n    let target: HTMLInputElement\n    if (e) {\n      target = e.target\n    } else {\n      target = this.search\n    }\n    const val = target.value\n    const { placeholderShow, selectId } = this.state\n    if (val && !placeholderShow) {\n      this.setState({ placeholderShow: true })\n    }\n    if (!val && placeholderShow) {\n      this.setState({ placeholderShow: false })\n    }\n    if (this.select) {\n      let id = selectId\n      if (!selectId) {\n        id = (target.parentNode?.nextSibling as Element)?.id\n        this.setState({ selectId: id })\n      }\n      const realInput = document.querySelector(\n        `.adui-select-selection-search [aria-owns=\"${id}\"]`\n      ) as any\n      if (realInput) {\n        const lastValue = realInput.value\n        realInput.value = val\n        const event = new Event(\"input\", { bubbles: true })\n        // hack React16 内部定义了descriptor拦截value，此处重置状态\n        const tracker = realInput._valueTracker\n        if (tracker) {\n          tracker.setValue(lastValue)\n        }\n        realInput.dispatchEvent(event)\n      }\n    }\n  }\n\n  public handleSearchKeyDown = (e: React.KeyboardEvent<HTMLInputElement>) => {\n    if (this.select && this.select.onInputKeyDown) {\n      this.select.onInputKeyDown(e)\n    }\n  }\n\n  public preventVisibleChange = () => {\n    this.locked = true\n    setTimeout(() => {\n      this.locked = false\n      this.setState({ open: true })\n    }, 200)\n  }\n\n  public render() {\n    const {\n      allowClear,\n      className,\n      getPopupContainer,\n      onChange,\n      options,\n      placeholder,\n      placeholderColor,\n      placement,\n      rightIcon,\n      searchable,\n      size,\n      theme,\n      value: valueProp,\n      ...otherProps\n    } = this.props\n\n    const restProps = omit(otherProps, [\n      \"defaultValue\",\n      \"open\",\n      \"onDropdownVisibleChange\",\n      \"onSelect\",\n      \"searchPlaceholder\",\n      \"onSearchCompositionStart\",\n      \"onSearchCompositionUpdate\",\n      \"onSearchCompositionEnd\",\n      \"searchInputProps\",\n      \"topContent\",\n      \"bottomContent\",\n    ])\n\n    const {\n      open: openState,\n      placeholderShow,\n      placeholderText,\n      value: valueState,\n    } = this.state\n\n    // openState 可能是 undefined or null\n    const openProps: { open?: boolean; value?: React.ReactNode | null } = {}\n    if (typeof openState === \"boolean\") {\n      openProps.open = openState\n    }\n    if (valueState !== null) {\n      openProps.value =\n        valueProp === \"\" ? (\n          placeholderColor ? (\n            <span style={{ color: placeholderColor }}>{placeholder}</span>\n          ) : (\n            placeholder\n          )\n        ) : (\n          valueState\n        )\n    }\n\n    const transitionName = \"slide-up\"\n\n    return (\n      <ConfigContext.Consumer>\n        {({ getPopupContainer: getPopupContainerContext }) => (\n          <RcSelect\n            allowClear={allowClear}\n            className={classNames(\n              className,\n              `${prefix}-select`,\n              `${prefix}-${size}`,\n              {\n                [`${prefix}-value_is_empty_string`]:\n                  valueState === \"\" || valueState === undefined,\n                [`${prefix}-light`]: theme === \"light\",\n                [`${prefix}-select_placeholderShow`]: placeholderShow,\n              }\n            )}\n            data-value={valueState || placeholder}\n            menuItemSelectedIcon={null}\n            dropdownAlign={\n              getPlacements({ alignEdge: true })[placement || \"bottomLeft\"]\n            }\n            dropdownClassName={`adui-select-dropdown-${\n              this.hash\n            } adui-select-dropdown-${size} ${\n              searchable ? \"adui-select-dropdown-searchable\" : \"\"\n            }`}\n            dropdownRender={this.handleDropdownRender}\n            suffixIcon={\n              <>\n                {placeholderShow &&\n                  (typeof (placeholderText || placeholder || \"\") ===\n                  \"string\" ? (\n                    <div\n                      className={`${prefix}-placeholder-text`}\n                      style={{\n          