You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
184 lines
4.3 KiB
184 lines
4.3 KiB
import { getPropsData, getSlotOptions, getKey, getAttrs, getComponentFromProp } from '../_util/props-util'
|
|
import { cloneVNodes } from '../_util/vnode'
|
|
export function getValuePropValue (child) {
|
|
const props = getPropsData(child)
|
|
if ('value' in props) {
|
|
return props.value
|
|
}
|
|
if (getKey(child) !== undefined) {
|
|
return getKey(child)
|
|
}
|
|
if (getSlotOptions(child).isSelectOptGroup) {
|
|
const label = getComponentFromProp(child, 'label')
|
|
if (label) {
|
|
return label
|
|
}
|
|
}
|
|
throw new Error(
|
|
`Need at least a key or a value or a label(slot) (only for OptGroup) for ${child}`
|
|
)
|
|
}
|
|
|
|
export function getPropValue (child, prop) {
|
|
if (prop === 'value') {
|
|
return getValuePropValue(child)
|
|
}
|
|
if (prop === 'children') {
|
|
const newChild = child.$slots ? cloneVNodes(child.$slots.default, true) : cloneVNodes(child.componentOptions.children, true)
|
|
if (newChild.length === 1 && !newChild[0].tag) {
|
|
return newChild[0].text
|
|
}
|
|
return newChild
|
|
}
|
|
const data = getPropsData(child)
|
|
if (prop in data) {
|
|
return data[prop]
|
|
} else {
|
|
return getAttrs(child)[prop]
|
|
}
|
|
}
|
|
|
|
export function isMultiple (props) {
|
|
return props.multiple
|
|
}
|
|
|
|
export function isCombobox (props) {
|
|
return props.combobox
|
|
}
|
|
|
|
export function isMultipleOrTags (props) {
|
|
return props.multiple || props.tags
|
|
}
|
|
|
|
export function isMultipleOrTagsOrCombobox (props) {
|
|
return isMultipleOrTags(props) || isCombobox(props)
|
|
}
|
|
|
|
export function isSingleMode (props) {
|
|
return !isMultipleOrTagsOrCombobox(props)
|
|
}
|
|
|
|
export function toArray (value) {
|
|
let ret = value
|
|
if (value === undefined) {
|
|
ret = []
|
|
} else if (!Array.isArray(value)) {
|
|
ret = [value]
|
|
}
|
|
return ret
|
|
}
|
|
|
|
export function preventDefaultEvent (e) {
|
|
e.preventDefault()
|
|
}
|
|
|
|
export function findIndexInValueByKey (value, key) {
|
|
let index = -1
|
|
for (let i = 0; i < value.length; i++) {
|
|
if (value[i].key === key) {
|
|
index = i
|
|
break
|
|
}
|
|
}
|
|
return index
|
|
}
|
|
|
|
export function findIndexInValueByLabel (value, label) {
|
|
let index = -1
|
|
for (let i = 0; i < value.length; i++) {
|
|
if (toArray(value[i].label).join('') === label) {
|
|
index = i
|
|
break
|
|
}
|
|
}
|
|
return index
|
|
}
|
|
|
|
export function getSelectKeys (menuItems, value) {
|
|
if (value === null || value === undefined) {
|
|
return []
|
|
}
|
|
let selectedKeys = []
|
|
menuItems.forEach(item => {
|
|
if (getSlotOptions(item).isMenuItemGroup) {
|
|
selectedKeys = selectedKeys.concat(
|
|
getSelectKeys(item.componentOptions.children, value)
|
|
)
|
|
} else {
|
|
const itemValue = getValuePropValue(item)
|
|
const itemKey = item.key
|
|
if (findIndexInValueByKey(value, itemValue) !== -1 && itemKey) {
|
|
selectedKeys.push(itemKey)
|
|
}
|
|
}
|
|
})
|
|
return selectedKeys
|
|
}
|
|
|
|
export const UNSELECTABLE_STYLE = {
|
|
userSelect: 'none',
|
|
WebkitUserSelect: 'none',
|
|
}
|
|
|
|
export const UNSELECTABLE_ATTRIBUTE = {
|
|
unselectable: 'unselectable',
|
|
}
|
|
|
|
export function findFirstMenuItem (children) {
|
|
for (let i = 0; i < children.length; i++) {
|
|
const child = children[i]
|
|
const props = getPropsData(child)
|
|
if (getSlotOptions(child).isMenuItemGroup) {
|
|
const found = findFirstMenuItem(child.componentOptions.children)
|
|
if (found) {
|
|
return found
|
|
}
|
|
} else if (!props.disabled) {
|
|
return child
|
|
}
|
|
}
|
|
return null
|
|
}
|
|
|
|
export function includesSeparators (string, separators) {
|
|
for (let i = 0; i < separators.length; ++i) {
|
|
if (string.lastIndexOf(separators[i]) > 0) {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
export function splitBySeparators (string, separators) {
|
|
const reg = new RegExp(`[${separators.join()}]`)
|
|
return string.split(reg).filter(token => token)
|
|
}
|
|
|
|
export function defaultFilterFn (input, child) {
|
|
const props = getPropsData(child)
|
|
if (props.disabled) {
|
|
return false
|
|
}
|
|
let value = getPropValue(child, this.optionFilterProp)
|
|
if (value.length && value[0].text) {
|
|
value = value[0].text
|
|
} else {
|
|
value = String(value)
|
|
}
|
|
return (
|
|
value.toLowerCase().indexOf(input.toLowerCase()) > -1
|
|
)
|
|
}
|
|
|
|
export function validateOptionValue (value, props) {
|
|
if (isSingleMode(props) || isMultiple(props)) {
|
|
return
|
|
}
|
|
if (typeof value !== 'string') {
|
|
throw new Error(
|
|
`Invalid \`value\` of type \`${typeof value}\` supplied to Option, ` +
|
|
`expected \`string\` when \`tags/combobox\` is \`true\`.`
|
|
)
|
|
}
|
|
}
|