1769 lines
52 KiB
Vue
1769 lines
52 KiB
Vue
|
||
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 Vue from 'vue'
|
||
import Option from './Option'
|
||
import { hasProp, getSlotOptions, getPropsData, getValueByProp as getValue, getComponentFromProp, getEvents, getClass, getStyle, getAttrs, getOptionProps } from '../_util/props-util'
|
||
import getTransitionProps from '../_util/getTransitionProps'
|
||
import { cloneElement } from '../_util/vnode'
|
||
import BaseMixin from '../_util/BaseMixin'
|
||
import proxyComponent from '../_util/proxyComponent'
|
||
import antRefDirective from '../_util/antRefDirective'
|
||
|
||
Vue.use(antRefDirective)
|
||
|
||
import {
|
||
getPropValue,
|
||
getValuePropValue,
|
||
isCombobox,
|
||
isMultipleOrTags,
|
||
isMultipleOrTagsOrCombobox,
|
||
isSingleMode,
|
||
toArray,
|
||
getMapKey,
|
||
findIndexInValueBySingleValue,
|
||
getLabelFromPropsValue,
|
||
UNSELECTABLE_ATTRIBUTE,
|
||
UNSELECTABLE_STYLE,
|
||
preventDefaultEvent,
|
||
findFirstMenuItem,
|
||
includesSeparators,
|
||
splitBySeparators,
|
||
defaultFilterFn,
|
||
validateOptionValue,
|
||
saveRef,
|
||
toTitle,
|
||
} 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)
|
||
}
|
||
}
|
||
}
|
||
}
|
||
const Select = {
|
||
inheritAttrs: false,
|
||
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),
|
||
tokenSeparators: PropTypes.arrayOf(PropTypes.string).def([]),
|
||
autoClearSearchValue: PropTypes.bool.def(true),
|
||
// onChange: noop,
|
||
// onFocus: noop,
|
||
// onBlur: noop,
|
||
// onSelect: noop,
|
||
// onSearch: noop,
|
||
// onDeselect: noop,
|
||
// onInputKeydown: noop,
|
||
},
|
||
model: {
|
||
prop: 'value',
|
||
event: 'change',
|
||
},
|
||
created () {
|
||
this.saveInputRef = saveRef(this, 'inputRef')
|
||
this.saveInputMirrorRef = saveRef(this, 'inputMirrorRef')
|
||
this.saveTopCtrlRef = saveRef(this, 'topCtrlRef')
|
||
this.saveSelectTriggerRef = saveRef(this, 'selectTriggerRef')
|
||
this.saveRootRef = saveRef(this, 'rootRef')
|
||
this.saveSelectionRef = saveRef(this, 'selectionRef')
|
||
},
|
||
data () {
|
||
const props = getOptionProps(this)
|
||
const optionsInfo = this.getOptionsInfoFromProps(props)
|
||
warning(
|
||
this.__propsSymbol__,
|
||
'Replace slots.default with props.children and pass props.__propsSymbol__'
|
||
)
|
||
return {
|
||
_value: this.getValueFromProps(props, true), // true: use default value
|
||
_inputValue: props.combobox ? this.getInputValueForCombobox(
|
||
props,
|
||
optionsInfo,
|
||
true, // use default value
|
||
) : '',
|
||
_open: props.defaultOpen,
|
||
_optionsInfo: optionsInfo,
|
||
// a flag for aviod redundant getOptionsInfoFromProps call
|
||
_skipBuildOptionsInfo: true,
|
||
}
|
||
},
|
||
beforeMount () {
|
||
const state = this.getDerivedStateFromProps(getOptionProps(this), this.$data)
|
||
Object.assign(this.$data, state)
|
||
},
|
||
|
||
mounted () {
|
||
this.$nextTick(() => {
|
||
this.autoFocus && this.focus()
|
||
})
|
||
},
|
||
watch: {
|
||
__propsSymbol__ () {
|
||
Object.assign(this.$data, this.getDerivedStateFromProps(getOptionProps(this), this.$data))
|
||
},
|
||
// 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
|
||
// this.initLabelAndTitleMap(sValue)
|
||
// sValue.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
|
||
// this.labelMap.set(key, label === undefined ? key : label)
|
||
// this.titleMap.set(key, title)
|
||
// })
|
||
|
||
// 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 + 10}px`
|
||
} else {
|
||
inputNode.style.width = ''
|
||
}
|
||
}
|
||
this.forcePopupAlign()
|
||
})
|
||
},
|
||
beforeDestroy () {
|
||
this.clearFocusTime()
|
||
this.clearBlurTime()
|
||
if (this.dropdownContainer) {
|
||
document.body.removeChild(this.dropdownContainer)
|
||
this.dropdownContainer = null
|
||
}
|
||
},
|
||
methods: {
|
||
getDerivedStateFromProps (nextProps, prevState) {
|
||
const optionsInfo = prevState._skipBuildOptionsInfo
|
||
? prevState._optionsInfo
|
||
: this.getOptionsInfoFromProps(nextProps, prevState)
|
||
|
||
const newState = {
|
||
_optionsInfo: optionsInfo,
|
||
_skipBuildOptionsInfo: false,
|
||
}
|
||
|
||
if ('open' in nextProps) {
|
||
newState._open = nextProps.open
|
||
}
|
||
|
||
if ('value' in nextProps) {
|
||
const value = this.getValueFromProps(nextProps)
|
||
newState._value = value
|
||
if (nextProps.combobox) {
|
||
newState._inputValue = this.getInputValueForCombobox(
|
||
nextProps,
|
||
optionsInfo,
|
||
)
|
||
}
|
||
}
|
||
return newState
|
||
},
|
||
// 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
|
||
// title = typeof title === 'string' ? title.trim() : 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.$props.children)
|
||
// },
|
||
// 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.$props
|
||
const val = event.target.value
|
||
if (
|
||
isMultipleOrTags(this.$props) &&
|
||
tokenSeparators.length &&
|
||
includesSeparators(val, tokenSeparators)
|
||
) {
|
||
const nextValue = this.getValueByInput(val)
|
||
if (nextValue !== undefined) {
|
||
this.fireChange(nextValue)
|
||
}
|
||
this.setOpenState(false, true)
|
||
this.setInputValue('', false)
|
||
return
|
||
}
|
||
this.setInputValue(val)
|
||
this.setState({
|
||
_open: true,
|
||
})
|
||
if (isCombobox(this.$props)) {
|
||
this.fireChange([val])
|
||
}
|
||
},
|
||
|
||
onDropdownVisibleChange (open) {
|
||
if (open && !this._focused) {
|
||
this.clearBlurTime()
|
||
this.timeoutFocus()
|
||
this._focused = true
|
||
this.updateFocusClassName()
|
||
}
|
||
this.setOpenState(open)
|
||
},
|
||
|
||
// combobox ignore
|
||
onKeyDown (event) {
|
||
const props = this.$props
|
||
if (props.disabled) {
|
||
return
|
||
}
|
||
const keyCode = event.keyCode
|
||
if (this.$data._open && !this.getInputDOMNode()) {
|
||
this.onInputKeydown(event)
|
||
} else if (keyCode === KeyCode.ENTER || keyCode === KeyCode.DOWN) {
|
||
// vue state是同步更新,onKeyDown在onMenuSelect后会再次调用,单选时不在调用setOpenState
|
||
if (keyCode === KeyCode.ENTER && !isMultipleOrTags(props)) {
|
||
this.maybeFocus(true)
|
||
} else {
|
||
this.setOpenState(true)
|
||
}
|
||
event.preventDefault()
|
||
}
|
||
},
|
||
|
||
onInputKeydown (event) {
|
||
const props = this.$props
|
||
if (props.disabled) {
|
||
return
|
||
}
|
||
const state = this.$data
|
||
const keyCode = event.keyCode
|
||
if (
|
||
isMultipleOrTags(props) &&
|
||
!event.target.value &&
|
||
keyCode === KeyCode.BACKSPACE
|
||
) {
|
||
event.preventDefault()
|
||
const { _value: value } = state
|
||
if (value.length) {
|
||
this.removeSelected(value[value.length - 1])
|
||
}
|
||
return
|
||
}
|
||
if (keyCode === KeyCode.DOWN) {
|
||
if (!state._open) {
|
||
this.openIfHasChildren()
|
||
event.preventDefault()
|
||
event.stopPropagation()
|
||
return
|
||
}
|
||
} else if (keyCode === KeyCode.ENTER && state._open) {
|
||
// Aviod trigger form submit when select item
|
||
// https://github.com/ant-design/ant-design/issues/10861
|
||
event.preventDefault()
|
||
} else if (keyCode === KeyCode.ESC) {
|
||
if (state._open) {
|
||
this.setOpenState(false)
|
||
event.preventDefault()
|
||
event.stopPropagation()
|
||
}
|
||
return
|
||
}
|
||
|
||
if (state._open) {
|
||
const menu = this.selectTriggerRef.getInnerMenu()
|
||
if (menu && menu.onKeyDown(event, this.handleBackfill)) {
|
||
event.preventDefault()
|
||
event.stopPropagation()
|
||
}
|
||
}
|
||
},
|
||
|
||
onMenuSelect ({ item }) {
|
||
if (!item) {
|
||
return
|
||
}
|
||
let value = this.$data._value
|
||
const props = this.$props
|
||
const selectedValue = getValuePropValue(item)
|
||
const lastValue = value[value.length - 1]
|
||
this.fireSelect(selectedValue)
|
||
if (isMultipleOrTags(props)) {
|
||
if (findIndexInValueBySingleValue(value, selectedValue) !== -1) {
|
||
return
|
||
}
|
||
value = value.concat([selectedValue])
|
||
} else {
|
||
if (lastValue && lastValue === selectedValue && selectedValue !== this.$data._backfillValue) {
|
||
this.setOpenState(false, true)
|
||
return
|
||
}
|
||
value = [selectedValue]
|
||
this.setOpenState(false, true)
|
||
}
|
||
this.fireChange(value)
|
||
let inputValue
|
||
if (isCombobox(props)) {
|
||
inputValue = getPropValue(item, props.optionLabelProp)
|
||
} else {
|
||
inputValue = ''
|
||
}
|
||
if (props.autoClearSearchValue) {
|
||
this.setInputValue(inputValue, false)
|
||
}
|
||
},
|
||
|
||
onMenuDeselect ({ item, domEvent }) {
|
||
if (domEvent.type === 'click') {
|
||
this.removeSelected(getValuePropValue(item))
|
||
}
|
||
if (this.autoClearSearchValue) {
|
||
this.setInputValue('', false)
|
||
}
|
||
},
|
||
|
||
onArrowClick (e) {
|
||
e.stopPropagation()
|
||
e.preventDefault()
|
||
if (!this.disabled) {
|
||
this.setOpenState(!this.$data._open, !this.$data._open)
|
||
}
|
||
},
|
||
|
||
onPlaceholderClick (e) {
|
||
// if (this.openStatus) {
|
||
// 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 props = this.$props
|
||
const state = this.$data
|
||
if (props.disabled) {
|
||
return
|
||
}
|
||
const { _inputValue: inputValue, _value: value } = state
|
||
event.stopPropagation()
|
||
if (inputValue || value.length) {
|
||
if (value.length) {
|
||
this.fireChange([])
|
||
}
|
||
this.setOpenState(false, true)
|
||
if (inputValue) {
|
||
this.setInputValue('')
|
||
}
|
||
}
|
||
},
|
||
|
||
onChoiceAnimationLeave () {
|
||
this.forcePopupAlign()
|
||
},
|
||
getOptionsFromChildren (children = [], options = []) {
|
||
children.forEach(child => {
|
||
if (!child.data || child.data.slot !== undefined) {
|
||
return
|
||
}
|
||
if (getSlotOptions(child).isSelectOptGroup) {
|
||
this.getOptionsFromChildren(child.componentOptions.children, options)
|
||
} else {
|
||
options.push(child)
|
||
}
|
||
})
|
||
return options
|
||
},
|
||
getInputValueForCombobox (props, optionsInfo, useDefaultValue) {
|
||
let value = []
|
||
if ('value' in props && !useDefaultValue) {
|
||
value = toArray(props.value)
|
||
}
|
||
if ('defaultValue' in props && useDefaultValue) {
|
||
value = toArray(props.defaultValue)
|
||
}
|
||
if (value.length) {
|
||
value = value[0]
|
||
} else {
|
||
return ''
|
||
}
|
||
let label = value
|
||
if (props.labelInValue) {
|
||
label = value.label
|
||
} else if (optionsInfo[getMapKey(value)]) {
|
||
label = optionsInfo[getMapKey(value)].label
|
||
}
|
||
if (label === undefined) {
|
||
label = ''
|
||
}
|
||
return label
|
||
},
|
||
|
||
getLabelFromOption (props, option) {
|
||
return getPropValue(option, props.optionLabelProp)
|
||
},
|
||
|
||
getOptionsInfoFromProps (props, preState) {
|
||
const options = this.getOptionsFromChildren(this.$props.children)
|
||
const optionsInfo = {}
|
||
options.forEach((option) => {
|
||
const singleValue = getValuePropValue(option)
|
||
optionsInfo[getMapKey(singleValue)] = {
|
||
option,
|
||
value: singleValue,
|
||
label: this.getLabelFromOption(props, option),
|
||
title: getValue(option, 'title'),
|
||
}
|
||
})
|
||
if (preState) {
|
||
// keep option info in pre state value.
|
||
const oldOptionsInfo = preState._optionsInfo
|
||
const value = preState._value
|
||
value.forEach(v => {
|
||
const key = getMapKey(v)
|
||
if (!optionsInfo[key] && oldOptionsInfo[key] !== undefined) {
|
||
optionsInfo[key] = oldOptionsInfo[key]
|
||
}
|
||
})
|
||
}
|
||
return optionsInfo
|
||
},
|
||
|
||
getValueFromProps (props, useDefaultValue) {
|
||
let value = []
|
||
if ('value' in props && !useDefaultValue) {
|
||
value = toArray(props.value)
|
||
}
|
||
if ('defaultValue' in props && useDefaultValue) {
|
||
value = toArray(props.defaultValue)
|
||
}
|
||
if (props.labelInValue) {
|
||
value = value.map((v) => {
|
||
return v.key
|
||
})
|
||
}
|
||
return value
|
||
},
|
||
|
||
getOptionInfoBySingleValue (value, optionsInfo) {
|
||
let info
|
||
optionsInfo = optionsInfo || this.$data._optionsInfo
|
||
if (optionsInfo[getMapKey(value)]) {
|
||
info = optionsInfo[getMapKey(value)]
|
||
}
|
||
if (info) {
|
||
return info
|
||
}
|
||
let defaultLabel = value
|
||
if (this.$props.labelInValue) {
|
||
const label = getLabelFromPropsValue(this.$props.value, value)
|
||
if (label !== undefined) {
|
||
defaultLabel = label
|
||
}
|
||
}
|
||
const defaultInfo = {
|
||
option: <Option value={value} key={value}>{value}</Option>,
|
||
value,
|
||
label: defaultLabel,
|
||
}
|
||
return defaultInfo
|
||
},
|
||
|
||
getOptionBySingleValue (value) {
|
||
const { option } = this.getOptionInfoBySingleValue(value)
|
||
return option
|
||
},
|
||
|
||
getOptionsBySingleValue (values) {
|
||
return values.map(value => {
|
||
return this.getOptionBySingleValue(value)
|
||
})
|
||
},
|
||
// getSingleOptionByValueKey (key) {
|
||
// return this.getOptionsFromChildren({
|
||
// key,
|
||
// label: key,
|
||
// }, this.$props.children)
|
||
// },
|
||
|
||
// getOptionsByValue (value) {
|
||
// if (value === undefined) {
|
||
// return undefined
|
||
// }
|
||
// if (value.length === 0) {
|
||
// return []
|
||
// }
|
||
// return this.getOptionsFromChildren(value, this.$props.children)
|
||
// },
|
||
// getLabelBySingleValue (children = [], value) {
|
||
// 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 (label) {
|
||
if (label === undefined) {
|
||
return null
|
||
}
|
||
let value = null
|
||
Object.keys(this.$data._optionsInfo).forEach(key => {
|
||
const info = this.$data._optionsInfo[key]
|
||
if (toArray(info.label).join('') === label) {
|
||
value = info.value
|
||
}
|
||
})
|
||
return value
|
||
},
|
||
// 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) {
|
||
// let label = getPropValue(child, this.optionLabelProp)
|
||
// if (Array.isArray(label) && label.length === 1 && !label[0].tag) {
|
||
// label = label[0].text
|
||
// }
|
||
// return label
|
||
// },
|
||
getVLBySingleValue (value) {
|
||
if (this.$props.labelInValue) {
|
||
return {
|
||
key: value,
|
||
label: this.getLabelBySingleValue(value),
|
||
}
|
||
}
|
||
return value
|
||
},
|
||
// getLabelFromProps (value) {
|
||
// return this.getLabelByValue(this.$props.children || [], value)
|
||
// },
|
||
|
||
getVLForOnChange (vls_) {
|
||
let vls = vls_
|
||
if (vls !== undefined) {
|
||
if (!this.labelInValue) {
|
||
vls = vls.map(v => v)
|
||
} else {
|
||
vls = vls.map(vl => ({
|
||
key: vl,
|
||
label: this.getLabelBySingleValue(vl),
|
||
}))
|
||
}
|
||
return isMultipleOrTags(this.$props) ? vls : vls[0]
|
||
}
|
||
return vls
|
||
},
|
||
|
||
getLabelBySingleValue (value, optionsInfo) {
|
||
const { label } = this.getOptionInfoBySingleValue(value, optionsInfo)
|
||
return label
|
||
},
|
||
|
||
// 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: props, $data: state } = this
|
||
let hidden = false
|
||
if (state._inputValue) {
|
||
hidden = true
|
||
}
|
||
if (state._value.length) {
|
||
hidden = true
|
||
}
|
||
if (isCombobox(props) && state._value.length === 1 && !state._value[0]) {
|
||
hidden = false
|
||
}
|
||
const placeholder = props.placeholder
|
||
if (placeholder) {
|
||
const p = {
|
||
on: {
|
||
mousedown: preventDefaultEvent,
|
||
click: this.onPlaceholderClick,
|
||
},
|
||
attrs: UNSELECTABLE_ATTRIBUTE,
|
||
style: {
|
||
display: hidden ? 'none' : 'block',
|
||
...UNSELECTABLE_STYLE,
|
||
},
|
||
class: `${props.prefixCls}-selection__placeholder`,
|
||
}
|
||
return (
|
||
<div {...p}>
|
||
{placeholder}
|
||
</div>
|
||
)
|
||
}
|
||
return null
|
||
},
|
||
inputClick (e) {
|
||
if (this.$data._open) {
|
||
this.clearBlurTime()
|
||
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 { _value: value } = this.$data
|
||
const { _inputValue: inputValue } = this.$data
|
||
if (
|
||
isSingleMode(props) &&
|
||
props.showSearch &&
|
||
inputValue &&
|
||
props.defaultActiveFirstOption
|
||
) {
|
||
const options = this._options || []
|
||
if (options.length) {
|
||
const firstOption = findFirstMenuItem(options)
|
||
if (firstOption) {
|
||
value = [getValuePropValue(firstOption)]
|
||
this.fireChange(value)
|
||
}
|
||
}
|
||
} else if (isMultipleOrTags(props) && inputValue) {
|
||
this.$data._inputValue = this.getInputDOMNode().value = ''
|
||
value = this.getValueByInput(inputValue)
|
||
if (value !== undefined) {
|
||
this.fireChange(value)
|
||
}
|
||
}
|
||
this.$emit('blur', this.getVLForOnChange(value))
|
||
this.setOpenState(false)
|
||
}, 10)
|
||
},
|
||
inputFocus (e) {
|
||
this.clearBlurTime()
|
||
this.clearFocusTime()
|
||
this.timeoutFocus()
|
||
},
|
||
_getInputElement () {
|
||
const props = this.$props
|
||
const { _inputValue: inputValue } = this.$data
|
||
const attrs = getAttrs(this)
|
||
const inputElement = props.getInputElement
|
||
? props.getInputElement()
|
||
: <input id={attrs.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`} onClick={this.inputClick}>
|
||
{cloneElement(inputElement, {
|
||
props: {
|
||
disabled: props.disabled,
|
||
value: inputValue,
|
||
},
|
||
attrs: {
|
||
...(inputElement.data.attrs || {}),
|
||
disabled: props.disabled,
|
||
value: inputValue,
|
||
},
|
||
domProps: {
|
||
value: inputValue,
|
||
},
|
||
class: inputCls,
|
||
directives: [{
|
||
name: 'ant-ref',
|
||
value: this.saveInputRef,
|
||
}],
|
||
on: {
|
||
input: this.onInputChange,
|
||
keydown: chaining(
|
||
this.onInputKeydown,
|
||
inputEvents.keydown,
|
||
this.$listeners.inputKeydown
|
||
),
|
||
focus: chaining(
|
||
this.inputFocus,
|
||
inputEvents.focus,
|
||
),
|
||
blur: chaining(
|
||
this.inputBlur,
|
||
inputEvents.blur,
|
||
),
|
||
},
|
||
})}
|
||
<span
|
||
{...{ directives: [{
|
||
name: 'ref',
|
||
value: this.saveInputMirrorRef,
|
||
}] }}
|
||
// ref='inputMirrorRef'
|
||
class={`${props.prefixCls}-search__field__mirror`}
|
||
>
|
||
{inputValue}
|
||
</span>
|
||
</div>
|
||
)
|
||
},
|
||
|
||
getInputDOMNode () {
|
||
return this.topCtrlRef
|
||
? this.topCtrlRef.querySelector('input,textarea,div[contentEditable]')
|
||
: this.inputRef
|
||
},
|
||
|
||
getInputMirrorDOMNode () {
|
||
return this.inputMirrorRef
|
||
},
|
||
|
||
getPopupDOMNode () {
|
||
return this.selectTriggerRef.getPopupDOMNode()
|
||
},
|
||
|
||
getPopupMenuComponent () {
|
||
return this.selectTriggerRef.getInnerMenu()
|
||
},
|
||
|
||
setOpenState (open, needFocus) {
|
||
const { $props: props, $data: state } = this
|
||
if (state._open === open) {
|
||
this.maybeFocus(open, needFocus)
|
||
return
|
||
}
|
||
const nextState = {
|
||
_open: open,
|
||
_backfillValue: undefined,
|
||
}
|
||
// clear search input value when open is false in singleMode.
|
||
if (!open && isSingleMode(props) && props.showSearch) {
|
||
this.setInputValue('', false)
|
||
}
|
||
if (!open) {
|
||
this.maybeFocus(open, needFocus)
|
||
}
|
||
this.setState(nextState, () => {
|
||
if (open) {
|
||
this.maybeFocus(open, needFocus)
|
||
}
|
||
})
|
||
},
|
||
|
||
setInputValue (inputValue, fireSearch = true) {
|
||
if (inputValue !== this.$data._inputValue) {
|
||
this.setState({
|
||
_inputValue: inputValue,
|
||
}, this.forcePopupAlign)
|
||
if (fireSearch) {
|
||
this.$emit('search', inputValue)
|
||
}
|
||
}
|
||
},
|
||
getValueByInput (string) {
|
||
const { multiple, tokenSeparators } = this.$props
|
||
let nextValue = this.$data._value
|
||
let hasNewValue = false
|
||
splitBySeparators(string, tokenSeparators).forEach(label => {
|
||
const selectedValue = [label]
|
||
if (multiple) {
|
||
const value = this.getValueByLabel(label)
|
||
if (value && findIndexInValueBySingleValue(nextValue, value) === -1) {
|
||
nextValue = nextValue.concat(value)
|
||
hasNewValue = true
|
||
this.fireSelect(value)
|
||
}
|
||
} else {
|
||
// tag
|
||
if (findIndexInValueBySingleValue(nextValue, label) === -1) {
|
||
nextValue = nextValue.concat(selectedValue)
|
||
hasNewValue = true
|
||
this.fireSelect(label)
|
||
}
|
||
}
|
||
})
|
||
return hasNewValue ? nextValue : undefined
|
||
},
|
||
|
||
getRealOpenState () {
|
||
let open = this.$data._open
|
||
const options = this._options || []
|
||
if (isMultipleOrTagsOrCombobox(this.$props) || !this.$props.showSearch) {
|
||
if (open && !options.length) {
|
||
open = false
|
||
}
|
||
}
|
||
return open
|
||
},
|
||
// getValueByInput (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($props.children, label)
|
||
// if (value) {
|
||
// selectedValue.key = value
|
||
// nextValue = nextValue.concat(selectedValue)
|
||
// }
|
||
// } else {
|
||
// nextValue = nextValue.concat(selectedValue)
|
||
// }
|
||
// }
|
||
// this.fireSelect({
|
||
// key: label,
|
||
// label,
|
||
// })
|
||
// })
|
||
// return nextValue
|
||
// },
|
||
|
||
focus () {
|
||
if (isSingleMode(this.$props)) {
|
||
this.selectionRef.focus()
|
||
} else {
|
||
this.getInputDOMNode().focus()
|
||
}
|
||
},
|
||
|
||
blur () {
|
||
if (isSingleMode(this.$props)) {
|
||
this.selectionRef.blur()
|
||
} else {
|
||
this.getInputDOMNode().blur()
|
||
}
|
||
},
|
||
|
||
handleBackfill (item) {
|
||
if (!this.backfill || !(isSingleMode(this.$props) || isCombobox(this.$props))) {
|
||
return
|
||
}
|
||
|
||
const key = getValuePropValue(item)
|
||
|
||
if (isCombobox(this.$props)) {
|
||
this.setInputValue(key, false)
|
||
}
|
||
|
||
this.setState({
|
||
_value: [key],
|
||
_backfillValue: key,
|
||
})
|
||
},
|
||
|
||
_filterOption (input, child, defaultFilter = defaultFilterFn) {
|
||
const { _value: value, _backfillValue: backfillValue } = this.$data
|
||
const lastValue = value[value.length - 1]
|
||
if (!input || (lastValue && lastValue === backfillValue)) {
|
||
return true
|
||
}
|
||
let filterFn = this.$props.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
|
||
}
|
||
},
|
||
|
||
updateFocusClassName () {
|
||
const { 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.selectionRef) {
|
||
this.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, e) {
|
||
const props = this.$props
|
||
if (props.disabled || this.isChildDisabled(selectedKey)) {
|
||
return
|
||
}
|
||
// Do not trigger Trigger popup
|
||
if (e && e.stopPropagation) {
|
||
e.stopPropagation()
|
||
}
|
||
|
||
const value = this.$data._value.filter(singleValue => {
|
||
return singleValue !== selectedKey
|
||
})
|
||
const canMultiple = isMultipleOrTags(props)
|
||
|
||
if (canMultiple) {
|
||
let event = selectedKey
|
||
if (props.labelInValue) {
|
||
event = {
|
||
key: selectedKey,
|
||
label: this.getLabelBySingleValue(selectedKey),
|
||
}
|
||
}
|
||
this.$emit('deselect', event, this.getOptionBySingleValue(selectedKey))
|
||
}
|
||
this.fireChange(value)
|
||
},
|
||
|
||
openIfHasChildren () {
|
||
const { $props } = this
|
||
if (($props.children && $props.children.length) || isSingleMode($props)) {
|
||
this.setOpenState(true)
|
||
}
|
||
},
|
||
fireSelect (value) {
|
||
this.$emit('select', this.getVLBySingleValue(value), this.getOptionBySingleValue(value))
|
||
},
|
||
fireChange (value) {
|
||
if (!hasProp(this, 'value')) {
|
||
this.setState({
|
||
_value: value,
|
||
}, this.forcePopupAlign)
|
||
}
|
||
const vls = this.getVLForOnChange(value)
|
||
const options = this.getOptionsBySingleValue(value)
|
||
this._valueOptions = options
|
||
this.$emit('change', vls, isMultipleOrTags(this.$props) ? options : options[0])
|
||
},
|
||
|
||
isChildDisabled (key) {
|
||
return (this.$props.children || []).some(child => {
|
||
const childValue = getValuePropValue(child)
|
||
return childValue === key && getValue(child, 'disabled')
|
||
})
|
||
},
|
||
forcePopupAlign () {
|
||
this.selectTriggerRef.triggerRef.forcePopupAlign()
|
||
},
|
||
// 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: inputValue } = this.$data
|
||
const { children, tags, filterOption, notFoundContent } = this.$props
|
||
const menuItems = []
|
||
const childrenKeys = []
|
||
let options = this.renderFilterOptionsFromChildren(
|
||
children,
|
||
childrenKeys,
|
||
menuItems,
|
||
)
|
||
if (tags) {
|
||
// tags value must be string
|
||
let value = this.$data._value
|
||
value = value.filter(singleValue => {
|
||
return (
|
||
childrenKeys.indexOf(singleValue) === -1 &&
|
||
(!inputValue ||
|
||
String(singleValue).indexOf(String(inputValue)) > -1)
|
||
)
|
||
})
|
||
value.forEach(singleValue => {
|
||
const key = singleValue
|
||
const menuItem = (
|
||
<MenuItem
|
||
style={UNSELECTABLE_STYLE}
|
||
{...{ attrs: UNSELECTABLE_ATTRIBUTE }}
|
||
value={key}
|
||
key={key}
|
||
role='option'
|
||
>
|
||
{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,
|
||
role: 'option',
|
||
},
|
||
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,
|
||
role: 'option',
|
||
},
|
||
style: UNSELECTABLE_STYLE,
|
||
}
|
||
options = [
|
||
<MenuItem {...p}>
|
||
{notFoundContent}
|
||
</MenuItem>,
|
||
]
|
||
}
|
||
return options
|
||
},
|
||
|
||
renderFilterOptionsFromChildren (children = [], childrenKeys, menuItems) {
|
||
const sel = []
|
||
const props = this.$props
|
||
const { _inputValue: inputValue } = this.$data
|
||
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} class ={getClass(child)}>
|
||
{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,
|
||
...getAttrs(child),
|
||
},
|
||
key: childValue,
|
||
props: {
|
||
value: childValue,
|
||
...getPropsData(child),
|
||
role: 'option',
|
||
},
|
||
style: UNSELECTABLE_STYLE,
|
||
on: getEvents(child),
|
||
class: getClass(child),
|
||
}
|
||
const menuItem = (
|
||
<MenuItem {...p}>{child.componentOptions.children}</MenuItem>
|
||
)
|
||
sel.push(menuItem)
|
||
menuItems.push(menuItem)
|
||
}
|
||
if (tags) {
|
||
childrenKeys.push(childValue)
|
||
}
|
||
})
|
||
|
||
return sel
|
||
},
|
||
|
||
renderTopControlNode () {
|
||
const { $props: props } = this
|
||
const { _value: value, _inputValue: inputValue, _open: open } = this.$data
|
||
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 (value.length) {
|
||
let showSelectedValue = false
|
||
let opacity = 1
|
||
if (!showSearch) {
|
||
showSelectedValue = true
|
||
} else {
|
||
if (open) {
|
||
showSelectedValue = !inputValue
|
||
if (showSelectedValue) {
|
||
opacity = 0.4
|
||
}
|
||
} else {
|
||
showSelectedValue = true
|
||
}
|
||
}
|
||
const singleValue = value[0]
|
||
const { label, title } = this.getOptionInfoBySingleValue(singleValue)
|
||
selectedValue = (
|
||
<div
|
||
key='value'
|
||
class={`${prefixCls}-selection-selected-value`}
|
||
title={toTitle(title || label)}
|
||
style={{
|
||
display: showSelectedValue ? 'block' : 'none',
|
||
opacity,
|
||
}}
|
||
>
|
||
{label}
|
||
</div>
|
||
)
|
||
}
|
||
if (!showSearch) {
|
||
innerNode = [selectedValue]
|
||
} else {
|
||
innerNode = [
|
||
selectedValue,
|
||
<div
|
||
class={`${prefixCls}-search ${prefixCls}-search--inline`}
|
||
key='input'
|
||
style={{
|
||
display: open ? 'block' : 'none',
|
||
}}
|
||
>
|
||
{this._getInputElement()}
|
||
</div>,
|
||
]
|
||
}
|
||
} else {
|
||
let selectedValueNodes = []
|
||
let limitedCountValue = value
|
||
let maxTagPlaceholderEl
|
||
if (maxTagCount !== undefined && value.length > maxTagCount) {
|
||
limitedCountValue = limitedCountValue.slice(0, maxTagCount)
|
||
const omittedValues = this.getVLForOnChange(value.slice(maxTagCount, value.length))
|
||
let content = `+ ${value.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={toTitle(content)}
|
||
>
|
||
<div class={`${prefixCls}-selection__choice__content`}>{content}</div>
|
||
</li>)
|
||
}
|
||
if (isMultipleOrTags(props)) {
|
||
selectedValueNodes = limitedCountValue.map(singleValue => {
|
||
const info = this.getOptionInfoBySingleValue(singleValue)
|
||
let content = info.label
|
||
const title = info.title || content
|
||
if (
|
||
maxTagTextLength &&
|
||
typeof content === 'string' &&
|
||
content.length > maxTagTextLength
|
||
) {
|
||
content = `${content.slice(0, maxTagTextLength)}...`
|
||
}
|
||
const disabled = this.isChildDisabled(singleValue)
|
||
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}
|
||
title={toTitle(title)}
|
||
>
|
||
<div class={`${prefixCls}-selection__choice__content`}>
|
||
{content}
|
||
</div>
|
||
{disabled ? null : (
|
||
<span
|
||
class={`${prefixCls}-selection__choice__remove`}
|
||
onClick={(event) => {
|
||
this.removeSelected(singleValue, event)
|
||
}}
|
||
/>)}
|
||
</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}
|
||
{...{ directives: [{
|
||
name: 'ref',
|
||
value: this.saveTopCtrlRef,
|
||
}] }}
|
||
onClick={this.topCtrlContainerClick}
|
||
>
|
||
{this.getPlaceholderElement()}
|
||
{innerNode}
|
||
</div>
|
||
)
|
||
},
|
||
topCtrlContainerClick (e) {
|
||
if (this.$data._open && !isSingleMode(this.$props)) {
|
||
e.stopPropagation()
|
||
}
|
||
},
|
||
renderClear () {
|
||
const { prefixCls, allowClear } = this.$props
|
||
const { _value: value, _inputValue: inputValue } = this.$data
|
||
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 || value.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()
|
||
if (!this.disabled) {
|
||
const input = this.getInputDOMNode()
|
||
if (this._focused && this.$data._open) {
|
||
this._focused = false
|
||
this.setOpenState(false, false)
|
||
input && input.blur()
|
||
} else {
|
||
this.clearBlurTime()
|
||
this._focused = true
|
||
this.setOpenState(true, true)
|
||
input && input.focus()
|
||
}
|
||
}
|
||
},
|
||
selectionRefFocus (e) {
|
||
if (this._focused || this.disabled) {
|
||
return
|
||
}
|
||
this._focused = true
|
||
this.updateFocusClassName()
|
||
this.$emit('focus')
|
||
},
|
||
selectionRefBlur (e) {
|
||
this._focused = false
|
||
this.updateFocusClassName()
|
||
this.$emit('blur')
|
||
},
|
||
},
|
||
|
||
render () {
|
||
const props = this.$props
|
||
const multiple = isMultipleOrTags(props)
|
||
const state = this.$data
|
||
const { disabled, prefixCls } = props
|
||
const ctrlNode = this.renderTopControlNode()
|
||
const { _open: open, _inputValue: inputValue, _value: value } = this.$data
|
||
if (open) {
|
||
this._options = this.renderFilterOptions()
|
||
}
|
||
const realOpen = this.getRealOpenState()
|
||
const options = this._options || []
|
||
const { $listeners } = this
|
||
const { mouseenter = noop, mouseleave = noop, popupScroll = noop } = $listeners
|
||
const selectionProps = {
|
||
props: {},
|
||
attrs: {
|
||
role: 'combobox',
|
||
'aria-autocomplete': 'list',
|
||
'aria-haspopup': 'true',
|
||
'aria-expanded': realOpen,
|
||
},
|
||
on: {
|
||
click: this.selectionRefClick,
|
||
},
|
||
class: `${prefixCls}-selection ${prefixCls}-selection--${multiple ? 'multiple' : 'single'}`,
|
||
directives: [{
|
||
name: 'ref',
|
||
value: this.saveSelectionRef,
|
||
}],
|
||
key: 'selection',
|
||
|
||
}
|
||
if (!isMultipleOrTagsOrCombobox(props)) {
|
||
selectionProps.on.keydown = this.onKeyDown
|
||
selectionProps.on.focus = this.selectionRefFocus
|
||
selectionProps.on.blur = this.selectionRefBlur
|
||
selectionProps.attrs.tabIndex = props.disabled ? -1 : 0
|
||
}
|
||
const rootCls = {
|
||
[prefixCls]: true,
|
||
[`${prefixCls}-open`]: open,
|
||
[`${prefixCls}-focused`]: open || !!this._focused,
|
||
[`${prefixCls}-combobox`]: isCombobox(props),
|
||
[`${prefixCls}-disabled`]: disabled,
|
||
[`${prefixCls}-enabled`]: !disabled,
|
||
[`${prefixCls}-allow-clear`]: !!props.allowClear,
|
||
[`${prefixCls}-no-arrow`]: !props.showArrow,
|
||
}
|
||
return (
|
||
<SelectTrigger
|
||
dropdownAlign={props.dropdownAlign}
|
||
dropdownClassName={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}
|
||
multiple={multiple}
|
||
disabled={disabled}
|
||
visible={realOpen}
|
||
inputValue={inputValue}
|
||
value={value}
|
||
backfillValue={state._backfillValue}
|
||
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}
|
||
{...{ directives: [{
|
||
name: 'ref',
|
||
value: this.saveSelectTriggerRef,
|
||
}] }}
|
||
>
|
||
<div
|
||
{...{ directives: [{
|
||
name: 'ref',
|
||
value: this.saveRootRef,
|
||
}] }}
|
||
style={getStyle(this)}
|
||
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>
|
||
)
|
||
},
|
||
}
|
||
export { Select }
|
||
export default proxyComponent(Select)
|