<script> import KeyCode from '../_util/KeyCode' import PropTypes from '../_util/vue-types' import classnames from 'classnames' import classes from 'component-classes' import { Item as MenuItem, ItemGroup as MenuItemGroup } from '../vc-menu' import warning from 'warning' import Option from './Option' import { hasProp, getSlotOptions, getPropsData, getValueByProp as getValue, getComponentFromProp, getEvents, getClass } from '../_util/props-util' import getTransitionProps from '../_util/getTransitionProps' import { cloneElement } from '../_util/vnode' import BaseMixin from '../_util/BaseMixin' import { getPropValue, getValuePropValue, isCombobox, isMultipleOrTags, isMultipleOrTagsOrCombobox, isSingleMode, toArray, findIndexInValueByKey, UNSELECTABLE_ATTRIBUTE, UNSELECTABLE_STYLE, preventDefaultEvent, findFirstMenuItem, includesSeparators, splitBySeparators, findIndexInValueByLabel, defaultFilterFn, validateOptionValue, } from './util' import SelectTrigger from './SelectTrigger' import { SelectPropTypes } from './PropTypes' function noop () {} function chaining (...fns) { return function (...args) { // eslint-disable-line // eslint-disable-line for (let i = 0; i < fns.length; i++) { if (fns[i] && typeof fns[i] === 'function') { fns[i].apply(this, args) } } } } export default { name: 'Select', mixins: [BaseMixin], props: { ...SelectPropTypes, prefixCls: SelectPropTypes.prefixCls.def('rc-select'), defaultOpen: PropTypes.bool.def(false), labelInValue: SelectPropTypes.labelInValue.def(false), defaultActiveFirstOption: SelectPropTypes.defaultActiveFirstOption.def(true), showSearch: SelectPropTypes.showSearch.def(true), allowClear: SelectPropTypes.allowClear.def(false), placeholder: SelectPropTypes.placeholder.def(''), showArrow: SelectPropTypes.showArrow.def(true), dropdownMatchSelectWidth: PropTypes.bool.def(true), dropdownStyle: SelectPropTypes.dropdownStyle.def({}), dropdownMenuStyle: PropTypes.object.def({}), optionFilterProp: SelectPropTypes.optionFilterProp.def('value'), optionLabelProp: SelectPropTypes.optionLabelProp.def('value'), notFoundContent: PropTypes.any.def('Not Found'), backfill: PropTypes.bool.def(false), showAction: SelectPropTypes.showAction.def(['click']), combobox: PropTypes.bool.def(false), // onChange: noop, // onFocus: noop, // onBlur: noop, // onSelect: noop, // onSearch: noop, // onDeselect: noop, // onInputKeydown: noop, }, data () { this.labelMap = new Map() this.titleMap = new Map() let sValue = [] const { value, defaultValue, combobox, open, defaultOpen } = this if (hasProp(this, 'value')) { sValue = toArray(value) } else { sValue = toArray(defaultValue) } if (this.labelInValue) { sValue.forEach(v => { v.key = v.key !== undefined ? v.key : v.value }) } else { sValue = sValue.map(v => { return { key: v, } }) } this.initLabelAndTitleMap(sValue) let inputValue = '' if (combobox) { inputValue = sValue.length ? this.labelMap.get((sValue[0].key)) : '' } let sOpen = open if (sOpen === undefined) { sOpen = defaultOpen } this._valueOptions = [] if (sValue.length > 0) { this._valueOptions = this.getOptionsByValue(sValue) } return { sValue, inputValue, sOpen, } }, mounted () { this.$nextTick(() => { this.autoFocus && this.focus() }) }, watch: { value (val) { let sValue = toArray(val) if (this.labelInValue) { sValue.forEach(v => { v.key = v.key !== undefined ? v.key : v.value }) } else { sValue = sValue.map(v => { return { key: v, } }) } this.sValue = sValue if (this.combobox) { this.setState({ inputValue: sValue.length ? this.labelMap.get((sValue[0].key)) : '', }) } }, combobox (val) { if (val) { this.setState({ inputValue: this.sValue.length ? this.labelMap.get((this.sValue[0].key)) : '', }) } }, }, updated () { this.$nextTick(() => { if (isMultipleOrTags(this.$props)) { const inputNode = this.getInputDOMNode() const mirrorNode = this.getInputMirrorDOMNode() if (inputNode.value) { inputNode.style.width = '' inputNode.style.width = `${mirrorNode.clientWidth}px` } else { inputNode.style.width = '' } } }) }, beforeDestroy () { this.clearFocusTime() this.clearBlurTime() this.clearAdjustTimer() if (this.dropdownContainer) { document.body.removeChild(this.dropdownContainer) this.dropdownContainer = null } }, methods: { initLabelAndTitleMap (sValue) { // 保留已选中的label and title const labelArr = [] const titleArr = [] const values = sValue || this.sValue values.forEach((val) => { const key = val.key let { label, title } = val label = label === undefined ? this.labelMap.get(key) : label title = title === undefined ? this.titleMap.get(key) : title labelArr.push([key, label === undefined ? key : label]) titleArr.push([key, title]) }) this.labelMap = new Map(labelArr) this.titleMap = new Map(titleArr) this.updateLabelAndTitleMap(this.$slots.default) }, updateLabelAndTitleMap (children = []) { children.forEach(child => { if (!child.data || child.data.slot !== undefined) { return } if (getSlotOptions(child).isSelectOptGroup) { this.updateLabelAndTitleMap(child.componentOptions.children) } else { const key = getValuePropValue(child) this.titleMap.set(key, getValue(child, 'title')) this.labelMap.set(key, this.getLabelFromOption(child)) } }) }, onInputChange (event) { const { tokenSeparators } = this const val = event.target.value if ( isMultipleOrTags(this.$props) && tokenSeparators && includesSeparators(val, tokenSeparators) ) { const nextValue = this.tokenize(val) this.fireChange(nextValue) this.setOpenState(false, true) this.setInputValue('', false) return } this.setInputValue(val) this.setState({ sOpen: true, }) if (isCombobox(this.$props)) { this.fireChange([ { key: val, }, ]) } }, onDropdownVisibleChange (open) { if (open && !this._focused) { this.clearBlurTime() this.timeoutFocus() this._focused = true this.updateFocusClassName() } this.setOpenState(open) }, // combobox ignore onKeyDown (event) { const { disabled, openStatus } = this if (disabled) { return } const keyCode = event.keyCode if (openStatus && !this.getInputDOMNode()) { this.onInputKeydown(event) } else if (keyCode === KeyCode.ENTER || keyCode === KeyCode.DOWN) { this.setOpenState(true) event.preventDefault() } }, onInputKeydown (event) { const { disabled, openStatus, sValue, $props } = this if (disabled) { return } const keyCode = event.keyCode if ( isMultipleOrTags($props) && !event.target.value && keyCode === KeyCode.BACKSPACE ) { event.preventDefault() if (sValue.length) { this.removeSelected(sValue[sValue.length - 1].key) } return } if (keyCode === KeyCode.DOWN) { if (!openStatus) { this.openIfHasChildren() event.preventDefault() event.stopPropagation() return } } else if (keyCode === KeyCode.ESC) { if (openStatus) { this.setOpenState(false) event.preventDefault() event.stopPropagation() } return } if (openStatus) { const menu = this.$refs.selectTriggerRef.getInnerMenu() if (menu && menu.onKeyDown(event, this.handleBackfill)) { event.preventDefault() event.stopPropagation() } } }, onMenuSelect ({ item }) { let sValue = this.sValue const props = this.$props const selectedValue = getValuePropValue(item) const selectedLabel = this.labelMap.get(selectedValue) const lastValue = sValue[sValue.length - 1] this.fireSelect({ key: selectedValue, label: selectedLabel, }) const selectedTitle = this.titleMap.get(selectedValue) if (isMultipleOrTags(props)) { if (findIndexInValueByKey(sValue, selectedValue) !== -1) { return } sValue = sValue.concat([ { key: selectedValue, label: selectedLabel, title: selectedTitle, }, ]) } else { if (isCombobox(props)) { this.skipAdjustOpen = true this.clearAdjustTimer() this.skipAdjustOpenTimer = setTimeout(() => { this.skipAdjustOpen = false }, 0) } if (lastValue && lastValue.key === selectedValue && !lastValue.backfill) { this.setOpenState(false, true) return } sValue = [ { key: selectedValue, label: selectedLabel, title: selectedTitle, }, ] this.setOpenState(false, true) } this.fireChange(sValue) let inputValue if (isCombobox(props)) { inputValue = getPropValue(item, props.optionLabelProp) } else { inputValue = '' } this.setInputValue(inputValue, false) }, onMenuDeselect ({ item, domEvent }) { if (domEvent.type === 'click') { this.removeSelected(getValuePropValue(item)) } this.setInputValue('', false) }, onArrowClick (e) { e.stopPropagation() if (!this.disabled) { this.setOpenState(!this.openStatus, !this.openStatus) } }, onPlaceholderClick (e) { if (this._focused) { e.stopPropagation() } if (this.getInputDOMNode()) { this.getInputDOMNode().focus() } }, onOuterFocus (e) { if (this.disabled) { e.preventDefault() return } this.clearBlurTime() if ( !isMultipleOrTagsOrCombobox(this.$props) && e.target === this.getInputDOMNode() ) { return } if (this._focused) { return } this._focused = true this.updateFocusClassName() this.timeoutFocus() }, onPopupFocus () { // fix ie scrollbar, focus element again this.maybeFocus(true, true) }, onOuterBlur (e) { if (this.disabled) { e.preventDefault() return } this.blurTimer = setTimeout(() => { this._focused = false this.updateFocusClassName() const props = this.$props let { sValue } = this const { inputValue } = this if ( isSingleMode(props) && props.showSearch && inputValue && props.defaultActiveFirstOption ) { const options = this._options || [] if (options.length) { const firstOption = findFirstMenuItem(options) if (firstOption) { sValue = [ { key: firstOption.key, label: this.getLabelFromOption(firstOption), }, ] this.fireChange(sValue) } } } else if (isMultipleOrTags(props) && inputValue) { this.inputValue = this.getInputDOMNode().value = '' } this.$emit('blur', this.getVLForOnChange(sValue)) this.setOpenState(false) }, 10) }, onClearSelection (event) { const { inputValue, sValue, disabled } = this if (disabled) { return } if (inputValue || sValue.length) { if (sValue.length) { this.fireChange([]) } this.setOpenState(false, true) if (inputValue) { this.setInputValue('') } if (this._focused) { this._focused = false } else { event.stopPropagation() } } else { event.stopPropagation() } }, onChoiceAnimationLeave () { this.$refs.selectTriggerRef.$refs.triggerRef.forcePopupAlign() }, getOptionsFromChildren (value, children = [], options = []) { let values = value if (!Array.isArray(value)) { values = [value] } children.forEach(child => { if (!child.data || child.data.slot !== undefined) { return } if (getSlotOptions(child).isSelectOptGroup) { this.getOptionsFromChildren(child.componentOptions.children, options) } else { const index = findIndexInValueByKey(values, getValuePropValue(child)) if (index !== -1) { options[index] = child } } }) values.forEach((v, i) => { if (!options[i]) { for (let j = 0; j < this._valueOptions.length; j++) { const item = this._valueOptions[j] if (getValuePropValue(item) === v.key) { options[i] = item break } } if (!options[i]) { options[i] = <Option value={v.key} key={v.key}>{this.labelMap.get(v.key)}</Option> } } }) if (!Array.isArray(value)) { return options[0] } return options }, getSingleOptionByValueKey (key) { return this.getOptionsFromChildren({ key, label: key, }, this.$slots.default) }, getOptionsByValue (value) { if (value === undefined) { return undefined } if (value.length === 0) { return [] } return this.getOptionsFromChildren(value, this.$slots.default) }, getLabelBySingleValue (children, value) { console.log('getLabelBySingleValue') if (value === undefined) { return null } let label = null children.forEach(child => { if (!child.data || child.data.slot !== undefined) { return } if (getSlotOptions(child).isSelectOptGroup) { const maybe = this.getLabelBySingleValue(child.componentOptions.children, value) if (maybe !== null) { label = maybe } } else if (getValuePropValue(child) === value) { label = this.getLabelFromOption(child) } }) return label }, getValueByLabel (children, label) { if (label === undefined) { return null } let value = null children.forEach(child => { if (!child.data || child.data.slot !== undefined) { return } if (getSlotOptions(child).isSelectOptGroup) { const maybe = this.getValueByLabel(child.componentOptions.children, label) if (maybe !== null) { value = maybe } } else if (toArray(this.getLabelFromOption(child)).join('') === label) { value = getValuePropValue(child) } }) return value }, getLabelFromOption (child) { return getPropValue(child, this.optionLabelProp) }, getLabelFromProps (value) { return this.getLabelByValue(this.$slots.default || [], value) }, getVLForOnChange (vls_) { let vls = vls_ if (vls !== undefined) { if (!this.labelInValue) { vls = vls.map(v => v.key) } else { vls = vls.map(vl => ({ key: vl.key, label: this.labelMap.get(vl.key) })) } return isMultipleOrTags(this.$props) ? vls : vls[0] } return vls }, getLabelByValue (children, value) { const label = this.getLabelBySingleValue(children, value) if (label === null) { return value } return label }, getDropdownContainer () { if (!this.dropdownContainer) { this.dropdownContainer = document.createElement('div') document.body.appendChild(this.dropdownContainer) } return this.dropdownContainer }, getPlaceholderElement () { // const { props, state } = this const { inputValue, sValue, placeholder, prefixCls, $props } = this let hidden = false if (inputValue) { hidden = true } if (sValue.length) { hidden = true } if (isCombobox($props) && sValue.length === 1 && !sValue[0].key) { hidden = false } if (placeholder) { const p = { on: { mousedown: preventDefaultEvent, click: this.onPlaceholderClick, }, attrs: UNSELECTABLE_ATTRIBUTE, style: { display: hidden ? 'none' : 'block', ...UNSELECTABLE_STYLE, }, class: `${prefixCls}-selection__placeholder`, } return ( <div {...p}> {placeholder} </div> ) } return null }, inputClick (e) { if (this._focused) { if (this.openStatus) { e.stopPropagation() } else { this._focused = false } } }, inputBlur (e) { this.clearBlurTime() if (this.disabled) { return } this.blurTimer = setTimeout(() => { this._focused = false this.updateFocusClassName() const props = this.$props let { sValue } = this const { inputValue } = this if ( isSingleMode(props) && props.showSearch && inputValue && props.defaultActiveFirstOption ) { const options = this._options || [] if (options.length) { const firstOption = findFirstMenuItem(options) if (firstOption) { sValue = [ { key: firstOption.key, label: this.labelMap.get(firstOption.key), }, ] this.fireChange(sValue) } } } else if (isMultipleOrTags(props) && inputValue) { this.inputValue = this.getInputDOMNode().value = '' } this.$emit('blur', this.getVLForOnChange(sValue)) this.setOpenState(false) }, 10) }, inputFocus (e) { this.clearBlurTime() this.clearFocusTime() this.timeoutFocus() }, _getInputElement () { const props = this.$props const inputElement = props.getInputElement ? props.getInputElement() : <input id={props.id} autoComplete='off'/> const inputCls = classnames(getClass(inputElement), { [`${props.prefixCls}-search__field`]: true, }) const inputEvents = getEvents(inputElement) // https://github.com/ant-design/ant-design/issues/4992#issuecomment-281542159 // Add space to the end of the inputValue as the width measurement tolerance inputElement.data = inputElement.data || {} return ( <div class={`${props.prefixCls}-search__field__wrap`}> {cloneElement(inputElement, { attrs: { ...(inputElement.data.attrs || {}), disabled: props.disabled, }, domProps: { value: this.inputValue, }, class: inputCls, ref: 'inputRef', on: { input: this.onInputChange, keydown: chaining( this.onInputKeydown, inputEvents.keydown || noop, this.$listeners.inputKeydown ), focus: chaining( this.inputFocus, inputEvents.focus || noop, ), blur: chaining( this.inputBlur, inputEvents.blur || noop, ), click: chaining( this.inputClick, inputEvents.click || noop, ), }, })} <span ref='inputMirrorRef' class={`${props.prefixCls}-search__field__mirror`} > {this.inputValue} </span> </div> ) }, getInputDOMNode () { return this.$refs.topCtrlRef ? this.$refs.topCtrlRef.querySelector('input,textarea,div[contentEditable]') : this.$refs.inputRef }, getInputMirrorDOMNode () { return this.$refs.inputMirrorRef }, getPopupDOMNode () { return this.$refs.selectTriggerRef.getPopupDOMNode() }, getPopupMenuComponent () { return this.$refs.selectTriggerRef.getInnerMenu() }, setOpenState (open, needFocus) { const { $props: props, openStatus } = this if (openStatus === open) { this.maybeFocus(open, needFocus) return } const nextState = { sOpen: open, } // clear search input value when open is false in singleMode. if (!open && isSingleMode(props) && props.showSearch) { this.setInputValue('') } if (!open) { this.maybeFocus(open, needFocus) } this.setState(nextState, () => { if (open) { this.maybeFocus(open, needFocus) } }) }, setInputValue (inputValue, fireSearch = true) { if (inputValue !== this.inputValue) { this.setState({ inputValue, }) if (fireSearch) { this.$emit('search', inputValue) } } }, focus () { if (isSingleMode(this.$props)) { this.$refs.selectionRef.focus() } else { this.getInputDOMNode().focus() } }, blur () { if (isSingleMode(this.$props)) { this.$refs.selectionRef.blur() } else { this.getInputDOMNode().blur() } }, handleBackfill (item) { if (!this.backfill || !(isSingleMode(this.$props) || isCombobox(this.$props))) { return } const key = getValuePropValue(item) const label = this.labelMap.get(key) const backfillValue = { key, label, backfill: true, } if (isCombobox(this.$props)) { this.setInputValue(key, false) } this.setState({ sValue: [backfillValue], }) }, _filterOption (input, child, defaultFilter = defaultFilterFn) { const { sValue } = this const lastValue = sValue[sValue.length - 1] if (!input || (lastValue && lastValue.backfill)) { return true } let filterFn = this.filterOption if (hasProp(this, 'filterOption')) { if (this.filterOption === true) { filterFn = defaultFilter } } else { filterFn = defaultFilter } if (!filterFn) { return true } else if (typeof filterFn === 'function') { return filterFn.call(this, input, child) } else if (getValue(child, 'disabled')) { return false } return true }, timeoutFocus () { if (this.focusTimer) { this.clearFocusTime() } this.focusTimer = setTimeout(() => { this._focused = true this.updateFocusClassName() this.$emit('focus') }, 10) }, clearFocusTime () { if (this.focusTimer) { clearTimeout(this.focusTimer) this.focusTimer = null } }, clearBlurTime () { if (this.blurTimer) { clearTimeout(this.blurTimer) this.blurTimer = null } }, clearAdjustTimer () { if (this.skipAdjustOpenTimer) { clearTimeout(this.skipAdjustOpenTimer) this.skipAdjustOpenTimer = null } }, updateFocusClassName () { const { $refs: { rootRef }, prefixCls } = this // avoid setState and its side effect if (this._focused) { classes(rootRef).add(`${prefixCls}-focused`) } else { classes(rootRef).remove(`${prefixCls}-focused`) } }, maybeFocus (open, needFocus) { if (needFocus || open) { const input = this.getInputDOMNode() const { activeElement } = document if (input && (open || isMultipleOrTagsOrCombobox(this.$props))) { if (activeElement !== input) { input.focus() this._focused = true } } else { if (activeElement !== this.$refs.selectionRef) { this.$refs.selectionRef.focus() this._focused = true } } } }, // addLabelToValue (value_) { // let value = value_ // if (this.labelInValue) { // value.forEach(v => { // v.label = v.label || this.getLabelFromProps(v.key) // }) // } else { // value = value.map(v => { // return { // key: v, // label: this.getLabelFromProps(v), // } // }) // } // return value // }, // addTitleToValue (children = [], values) { // let nextValues = values // const keys = values.map(v => v.key) // children.forEach(child => { // if (!child) { // return // } // if (getSlotOptions(child).isSelectOptGroup) { // nextValues = this.addTitleToValue(child.componentOptions.children, nextValues) // } else { // const value = getValuePropValue(child) // const valueIndex = keys.indexOf(value) // if (valueIndex > -1) { // nextValues[valueIndex].title = getValue(child, 'title') // } // } // }) // return nextValues // }, removeSelected (selectedKey) { const props = this.$props if (props.disabled || this.isChildDisabled(selectedKey)) { return } let label const value = this.sValue.filter(singleValue => { if (singleValue.key === selectedKey) { label = this.labelMap.get(selectedKey) } return singleValue.key !== selectedKey }) const canMultiple = isMultipleOrTags(props) if (canMultiple) { let event = selectedKey if (props.labelInValue) { event = { key: selectedKey, label, } } this.$emit('deselect', event, this.getSingleOptionByValueKey(selectedKey)) } this.fireChange(value) }, openIfHasChildren () { const { $props, $slots } = this if (($slots.default && $slots.default.length) || isSingleMode($props)) { this.setOpenState(true) } }, fireSelect (value) { const { labelInValue } = this this.$emit('select', labelInValue ? value : value.key, this.getSingleOptionByValueKey(value.key)) }, fireChange (value) { if (!hasProp(this, 'value')) { this.setState({ sValue: value, }) } const vls = this.getVLForOnChange(value) const options = this.getOptionsByValue(value) this._valueOptions = options this.$emit('change', vls, isMultipleOrTags(this.$props) ? options : options[0]) }, isChildDisabled (key) { return this.$slots.default.some(child => { const childValue = getValuePropValue(child) return childValue === key && getValue(child, 'disabled') }) }, tokenize (string) { const { multiple, tokenSeparators, $slots } = this let nextValue = this.sValue splitBySeparators(string, tokenSeparators).forEach(label => { const selectedValue = { key: label, label } if (findIndexInValueByLabel(nextValue, label) === -1) { if (multiple) { const value = this.getValueByLabel($slots.default, label) if (value) { selectedValue.key = value nextValue = nextValue.concat(selectedValue) } } else { nextValue = nextValue.concat(selectedValue) } } this.fireSelect({ key: label, label, }) }) return nextValue }, getOptionsAndOpenStatus () { let sOpen = this.sOpen if (this.skipAdjustOpen) { this.openStatus = sOpen return { options: this._options, open: sOpen, } } const { $props, showSearch } = this let options = [] // If hidden menu due to no options, then it should be calculated again if (sOpen || this.hiddenForNoOptions) { options = this.renderFilterOptions() } this._options = options if (isMultipleOrTagsOrCombobox($props) || !showSearch) { if (sOpen && !options.length) { sOpen = false this.hiddenForNoOptions = true } // Keep menu open if there are options and hidden for no options before if (this.hiddenForNoOptions && options.length) { sOpen = true this.hiddenForNoOptions = false } } this.openStatus = sOpen return { options, open: sOpen, } }, renderFilterOptions () { const { inputValue } = this const { $slots, tags, filterOption, notFoundContent } = this const menuItems = [] const childrenKeys = [] let options = this.renderFilterOptionsFromChildren( $slots.default, childrenKeys, menuItems, ) if (tags) { // tags value must be string let value = this.sValue || [] value = value.filter(singleValue => { return ( childrenKeys.indexOf(singleValue.key) === -1 && (!inputValue || String(singleValue.key).indexOf(String(inputValue)) > -1) ) }) value.forEach(singleValue => { const key = singleValue.key const menuItem = ( <MenuItem style={UNSELECTABLE_STYLE} {...{ attrs: UNSELECTABLE_ATTRIBUTE }} value={key} key={key} > {key} </MenuItem> ) options.push(menuItem) menuItems.push(menuItem) }) if (inputValue) { const notFindInputItem = menuItems.every(option => { // this.filterOption return true has two meaning, // 1, some one exists after filtering // 2, filterOption is set to false // condition 2 does not mean the option has same value with inputValue const filterFn = () => getValuePropValue(option) === inputValue if (filterOption !== false) { return !this._filterOption( inputValue, option, filterFn ) } return !filterFn() }) if (notFindInputItem) { const p = { attrs: UNSELECTABLE_ATTRIBUTE, key: inputValue, props: { value: inputValue, }, style: UNSELECTABLE_STYLE, } options.unshift( <MenuItem {...p}> {inputValue} </MenuItem> ) } } } if (!options.length && notFoundContent) { const p = { attrs: UNSELECTABLE_ATTRIBUTE, key: 'NOT_FOUND', props: { value: 'NOT_FOUND', disabled: true, }, style: UNSELECTABLE_STYLE, } options = [ <MenuItem {...p}> {notFoundContent} </MenuItem>, ] } return options }, renderFilterOptionsFromChildren (children = [], childrenKeys, menuItems) { const sel = [] const props = this.$props const { inputValue } = this const tags = props.tags children.forEach(child => { if (!child.data || child.data.slot !== undefined) { return } if (getSlotOptions(child).isSelectOptGroup) { const innerItems = this.renderFilterOptionsFromChildren( child.componentOptions.children, childrenKeys, menuItems, ) if (innerItems.length) { let label = getComponentFromProp(child, 'label') let key = child.key if (!key && typeof label === 'string') { key = label } else if (!label && key) { label = key } sel.push( <MenuItemGroup key={key} title={label}> {innerItems} </MenuItemGroup> ) } return } warning( getSlotOptions(child).isSelectOption, 'the children of `Select` should be `Select.Option` or `Select.OptGroup`, ' + `instead of \`${getSlotOptions(child).name || getSlotOptions(child)}\`.` ) const childValue = getValuePropValue(child) validateOptionValue(childValue, this.$props) if (this._filterOption(inputValue, child)) { const p = { attrs: UNSELECTABLE_ATTRIBUTE, key: childValue, props: { value: childValue, ...getPropsData(child), }, style: UNSELECTABLE_STYLE, on: getEvents(child), } const menuItem = ( <MenuItem {...p}>{child.componentOptions.children}</MenuItem> ) sel.push(menuItem) menuItems.push(menuItem) } if (tags && !getValue(child, 'disabled')) { childrenKeys.push(childValue) } }) return sel }, renderTopControlNode (openStatus) { const { sValue, inputValue, $props: props } = this const { choiceTransitionName, prefixCls, maxTagTextLength, maxTagCount, maxTagPlaceholder, showSearch, } = props const className = `${prefixCls}-selection__rendered` // search input is inside topControlNode in single, multiple & combobox. 2016/04/13 let innerNode = null if (isSingleMode(props)) { let selectedValue = null if (sValue.length) { let showSelectedValue = false let opacity = 1 if (!showSearch) { showSelectedValue = true } else { if (openStatus) { showSelectedValue = !inputValue if (showSelectedValue) { opacity = 0.4 } } else { showSelectedValue = true } } const singleValue = sValue[0] selectedValue = ( <div key='value' class={`${prefixCls}-selection-selected-value`} title={singleValue.title || singleValue.label} style={{ display: showSelectedValue ? 'block' : 'none', opacity, }} > {this.labelMap.get(sValue[0].key)} </div> ) } if (!showSearch) { innerNode = [selectedValue] } else { innerNode = [ selectedValue, <div class={`${prefixCls}-search ${prefixCls}-search--inline`} key='input' style={{ display: openStatus ? 'block' : 'none', }} > {this._getInputElement()} </div>, ] } } else { let selectedValueNodes = [] let limitedCountValue = sValue let maxTagPlaceholderEl if (maxTagCount !== undefined && sValue.length > maxTagCount) { limitedCountValue = limitedCountValue.slice(0, maxTagCount) const omittedValues = this.getVLForOnChange(sValue.slice(maxTagCount, sValue.length)) let content = `+ ${sValue.length - maxTagCount} ...` if (maxTagPlaceholder) { content = typeof maxTagPlaceholder === 'function' ? maxTagPlaceholder(omittedValues) : maxTagPlaceholder } maxTagPlaceholderEl = (<li style={UNSELECTABLE_STYLE} unselectable='unselectable' onMousedown={preventDefaultEvent} class={`${prefixCls}-selection__choice ${prefixCls}-selection__choice__disabled`} key={'maxTagPlaceholder'} title={content} > <div class={`${prefixCls}-selection__choice__content`}>{content}</div> </li>) } if (isMultipleOrTags(props)) { selectedValueNodes = limitedCountValue.map(singleValue => { let content = this.labelMap.get(singleValue.key) const title = this.titleMap.get(singleValue.key) || content if ( maxTagTextLength && typeof content === 'string' && content.length > maxTagTextLength ) { content = `${content.slice(0, maxTagTextLength)}...` } const disabled = this.isChildDisabled(singleValue.key) const choiceClassName = disabled ? `${prefixCls}-selection__choice ${prefixCls}-selection__choice__disabled` : `${prefixCls}-selection__choice` return ( <li style={UNSELECTABLE_STYLE} unselectable='unselectable' onMousedown={preventDefaultEvent} class={choiceClassName} key={singleValue.key} title={title} > <div class={`${prefixCls}-selection__choice__content`}> {content} </div> {disabled ? null : ( <span class={`${prefixCls}-selection__choice__remove`} onClick={this.removeSelected.bind(this, singleValue.key)} />)} </li> ) }) } if (maxTagPlaceholderEl) { selectedValueNodes.push(maxTagPlaceholderEl) } selectedValueNodes.push( <li class={`${prefixCls}-search ${prefixCls}-search--inline`} key='__input' > {this._getInputElement()} </li> ) if (isMultipleOrTags(props) && choiceTransitionName) { const transitionProps = getTransitionProps(choiceTransitionName, { tag: 'ul', afterLeave: this.onChoiceAnimationLeave, }) innerNode = ( <transition-group {...transitionProps} > {selectedValueNodes} </transition-group> ) } else { innerNode = ( <ul> {selectedValueNodes} </ul> ) } } return ( <div class={className} ref='topCtrlRef' onClick={this.muitipleContainerClick}> {this.getPlaceholderElement()} {innerNode} </div> ) }, muitipleContainerClick (e) { if (this.openStatus) { e.stopPropagation() } }, renderClear () { const { prefixCls, allowClear, sValue, inputValue } = this const clear = ( <span key='clear' onMousedown={preventDefaultEvent} style={UNSELECTABLE_STYLE} unselectable='unselectable' class={`${prefixCls}-selection__clear`} onClick={this.onClearSelection} /> ) if (!allowClear) { return null } if (isCombobox(this.$props)) { if (inputValue) { return clear } return null } if (inputValue || sValue.length) { return clear } return null }, rootRefClick (e) { // e.stopPropagation() if (this._focused) { // this.getInputDOMNode().blur() this.onOuterBlur() } else { this.onOuterFocus() // this.getInputDOMNode().focus() } }, selectionRefClick (e) { e.stopPropagation() this.clearBlurTime() if (!this.disabled) { const input = this.getInputDOMNode() if (this._focused && this.openStatus) { this._focused = false this.setOpenState(false, false) input && input.blur() } else { // this._focused = true // this.updateFocusClassName() // this.timeoutFocus() this._focused = true this.setOpenState(true, true) input && input.focus() } } }, selectionRefFocus (e) { if (this._focused) { return } this._focused = true this.updateFocusClassName() }, }, render () { this.initLabelAndTitleMap() const props = this.$props const multiple = isMultipleOrTags(props) const preOptions = this._options || [] const { options, open: openStatus } = this.getOptionsAndOpenStatus() const { disabled, prefixCls, inputValue, sValue, $listeners } = this const { mouseenter = noop, mouseleave = noop, popupScroll = noop } = $listeners const ctrlNode = this.renderTopControlNode(openStatus) const selectionProps = { props: {}, attrs: { role: 'combobox', 'aria-autocomplete': 'list', 'aria-haspopup': 'true', 'aria-expanded': openStatus.toString(), }, on: { click: this.selectionRefClick, }, class: `${prefixCls}-selection ${prefixCls}-selection--${multiple ? 'multiple' : 'single'}`, ref: 'selectionRef', key: 'selection', } if (!isMultipleOrTagsOrCombobox(props)) { selectionProps.on.keydown = this.onKeyDown selectionProps.on.focus = this.selectionRefFocus selectionProps.attrs.tabIndex = props.disabled ? -1 : 0 } const rootCls = { [prefixCls]: 1, [`${prefixCls}-open`]: openStatus, [`${prefixCls}-focused`]: openStatus || !!this._focused, [`${prefixCls}-combobox`]: isCombobox(props), [`${prefixCls}-disabled`]: disabled, [`${prefixCls}-enabled`]: !disabled, [`${prefixCls}-allow-clear`]: !!props.allowClear, } return ( <SelectTrigger dropdownAlign={props.dropdownAlign} dropdownClass={props.dropdownClassName} dropdownMatchSelectWidth={props.dropdownMatchSelectWidth} defaultActiveFirstOption={props.defaultActiveFirstOption} dropdownMenuStyle={props.dropdownMenuStyle} transitionName={props.transitionName} animation={props.animation} prefixCls={props.prefixCls} dropdownStyle={props.dropdownStyle} combobox={props.combobox} showSearch={props.showSearch} options={options.length || openStatus ? options : preOptions} multiple={multiple} disabled={disabled} visible={openStatus} inputValue={inputValue} value={sValue} firstActiveValue={props.firstActiveValue} onDropdownVisibleChange={this.onDropdownVisibleChange} getPopupContainer={props.getPopupContainer} onMenuSelect={this.onMenuSelect} onMenuDeselect={this.onMenuDeselect} onPopupScroll={popupScroll} onPopupFocus={this.onPopupFocus} onMouseenter={mouseenter} onMouseleave={mouseleave} showAction={props.showAction} ref='selectTriggerRef' > <div ref='rootRef' class={classnames(rootCls)} // tabindex='-1' // onBlur={this.onOuterBlur} // onFocus={this.onOuterFocus} > <div {...selectionProps}> {ctrlNode} {this.renderClear()} {multiple || !props.showArrow ? null : ( <span key='arrow' class={`${prefixCls}-arrow`} style={UNSELECTABLE_STYLE} unselectable='unselectable' // onClick={this.onArrowClick} > <b /> </span>)} </div> </div> </SelectTrigger> ) }, } </script>