import type { App, ExtractPropTypes, Plugin, PropType } from 'vue'; import { computed, ref, watchEffect, defineComponent } from 'vue'; import VcTreeSelect, { TreeNode, SHOW_ALL, SHOW_PARENT, SHOW_CHILD, treeSelectProps as vcTreeSelectProps, } from '../vc-tree-select'; import classNames from '../_util/classNames'; import initDefaultProps from '../_util/props-util/initDefaultProps'; import type { SizeType } from '../config-provider'; import type { DefaultValueType, FieldNames } from '../vc-tree-select/interface'; import omit from '../_util/omit'; import PropTypes from '../_util/vue-types'; import useConfigInject from '../_util/hooks/useConfigInject'; import devWarning from '../vc-util/devWarning'; import getIcons from '../select/utils/iconUtil'; import renderSwitcherIcon from '../tree/utils/iconUtil'; import type { AntTreeNodeProps } from '../tree/Tree'; import { warning } from '../vc-util/warning'; import { flattenChildren } from '../_util/props-util'; import { useInjectFormItemContext } from '../form/FormItemContext'; const getTransitionName = (rootPrefixCls: string, motion: string, transitionName?: string) => { if (transitionName !== undefined) { return transitionName; } return `${rootPrefixCls}-${motion}`; }; type RawValue = string | number; export interface LabeledValue { key?: string; value: RawValue; label: any; } export type SelectValue = RawValue | RawValue[] | LabeledValue | LabeledValue[]; export interface RefTreeSelectProps { focus: () => void; blur: () => void; } export const treeSelectProps = { ...omit(vcTreeSelectProps(), ['showTreeIcon', 'treeMotion', 'inputIcon']), suffixIcon: PropTypes.any, size: { type: String as PropType }, bordered: { type: Boolean, default: undefined }, replaceFields: { type: Object as PropType }, }; export type TreeSelectProps = Partial>; const TreeSelect = defineComponent({ TreeNode, SHOW_ALL, SHOW_PARENT, SHOW_CHILD, name: 'ATreeSelect', inheritAttrs: false, props: initDefaultProps(treeSelectProps, { transitionName: 'slide-up', choiceTransitionName: '', listHeight: 256, treeIcon: false, listItemHeight: 26, bordered: true, }), slots: [ 'title', 'titleRender', 'placeholder', 'maxTagPlaceholder', 'treeIcon', 'switcherIcon', 'notFoundContent', ], setup(props, { attrs, slots, expose, emit }) { warning( !(props.treeData === undefined && slots.default), '`children` of TreeSelect is deprecated. Please use `treeData` instead.', ); watchEffect(() => { devWarning( props.multiple !== false || !props.treeCheckable, 'TreeSelect', '`multiple` will alway be `true` when `treeCheckable` is true', ); devWarning( props.replaceFields === undefined, 'TreeSelect', '`replaceFields` is deprecated, please use fieldNames instead', ); }); const formItemContext = useInjectFormItemContext(); const { configProvider, prefixCls, renderEmpty, direction, virtual, dropdownMatchSelectWidth, size, getPopupContainer, } = useConfigInject('select', props); const treePrefixCls = computed(() => configProvider.getPrefixCls('select-tree', props.prefixCls), ); const treeSelectPrefixCls = computed(() => configProvider.getPrefixCls('tree-select', props.prefixCls), ); const mergedDropdownClassName = computed(() => classNames(props.dropdownClassName, `${treeSelectPrefixCls.value}-dropdown`, { [`${treeSelectPrefixCls.value}-dropdown-rtl`]: direction.value === 'rtl', }), ); const isMultiple = computed(() => !!(props.treeCheckable || props.multiple)); const treeSelectRef = ref(); expose({ focus() { treeSelectRef.value.focus?.(); }, blur() { treeSelectRef.value.blur?.(); }, }); const handleChange = (...args: any[]) => { emit('update:value', args[0]); emit('change', ...args); formItemContext.onFieldChange(); }; const handleTreeExpand = (...args: any[]) => { emit('update:treeExpandedKeys', args[0]); emit('treeExpand', ...args); }; const handleSearch = (...args: any[]) => { emit('update:searchValue', args[0]); emit('search', ...args); }; const handleBlur = () => { emit('blur'); formItemContext.onFieldBlur(); }; return () => { const { notFoundContent = slots.notFoundContent?.(), prefixCls: customizePrefixCls, bordered, listHeight, listItemHeight, multiple, treeIcon, transitionName, choiceTransitionName, treeLine, switcherIcon = slots.switcherIcon?.(), fieldNames = props.replaceFields, id = formItemContext.id.value, } = props; // ===================== Icons ===================== const { suffixIcon, removeIcon, clearIcon } = getIcons( { ...props, multiple: isMultiple.value, prefixCls: prefixCls.value, }, slots, ); // ===================== Empty ===================== let mergedNotFound; if (notFoundContent !== undefined) { mergedNotFound = notFoundContent; } else { mergedNotFound = renderEmpty.value('Select'); } // ==================== Render ===================== const selectProps = omit(props as typeof props & { itemIcon: any; switcherIcon: any }, [ 'suffixIcon', 'itemIcon', 'removeIcon', 'clearIcon', 'switcherIcon', ]); const mergedClassName = classNames( !customizePrefixCls && treeSelectPrefixCls.value, { [`${prefixCls.value}-lg`]: size.value === 'large', [`${prefixCls.value}-sm`]: size.value === 'small', [`${prefixCls.value}-rtl`]: direction.value === 'rtl', [`${prefixCls.value}-borderless`]: !bordered, }, attrs.class, ); const rootPrefixCls = configProvider.getPrefixCls(); const otherProps: any = {}; if (props.treeData === undefined && slots.default) { otherProps.children = flattenChildren(slots.default()); } return ( renderSwitcherIcon(treePrefixCls.value, switcherIcon, treeLine, nodeProps) } showTreeIcon={treeIcon as any} notFoundContent={mergedNotFound} getPopupContainer={getPopupContainer.value} treeMotion={null} dropdownClassName={mergedDropdownClassName.value} choiceTransitionName={getTransitionName(rootPrefixCls, '', choiceTransitionName)} transitionName={getTransitionName(rootPrefixCls, 'slide-up', transitionName)} onChange={handleChange} onBlur={handleBlur} onSearch={handleSearch} onTreeExpand={handleTreeExpand} v-slots={{ ...slots, treeCheckable: () => , }} {...otherProps} /> ); }; }, }); /* istanbul ignore next */ TreeSelect.install = function (app: App) { app.component(TreeSelect.name, TreeSelect); app.component(TreeSelect.TreeNode.displayName, TreeSelect.TreeNode); return app; }; export const TreeSelectNode = TreeSelect.TreeNode; export default TreeSelect as typeof TreeSelect & Plugin & { readonly TreeNode: typeof TreeNode; readonly SHOW_ALL: typeof SHOW_ALL; readonly SHOW_PARENT: typeof SHOW_PARENT; readonly SHOW_CHILD: typeof SHOW_CHILD; };