180 lines
5.7 KiB
Vue
180 lines
5.7 KiB
Vue
import pickAttrs from '../../_util/pickAttrs';
|
||
import Input from './Input';
|
||
import type { InnerSelectorProps } from './interface';
|
||
import { Fragment, computed, defineComponent, ref, watch } from 'vue';
|
||
import PropTypes from '../../_util/vue-types';
|
||
import type { VueNode } from '../../_util/type';
|
||
import useInjectLegacySelectContext from '../../vc-tree-select/LegacyContext';
|
||
|
||
interface SelectorProps extends InnerSelectorProps {
|
||
inputElement: VueNode;
|
||
activeValue: string;
|
||
optionLabelRender: Function;
|
||
}
|
||
const props = {
|
||
inputElement: PropTypes.any,
|
||
id: String,
|
||
prefixCls: String,
|
||
values: PropTypes.array,
|
||
open: { type: Boolean, default: undefined },
|
||
searchValue: String,
|
||
inputRef: PropTypes.any,
|
||
placeholder: PropTypes.any,
|
||
disabled: { type: Boolean, default: undefined },
|
||
mode: String,
|
||
showSearch: { type: Boolean, default: undefined },
|
||
autofocus: { type: Boolean, default: undefined },
|
||
autocomplete: String,
|
||
activeDescendantId: String,
|
||
tabindex: PropTypes.oneOfType([PropTypes.number, PropTypes.string]),
|
||
activeValue: String,
|
||
backfill: { type: Boolean, default: undefined },
|
||
optionLabelRender: Function,
|
||
onInputChange: Function,
|
||
onInputPaste: Function,
|
||
onInputKeyDown: Function,
|
||
onInputMouseDown: Function,
|
||
onInputCompositionStart: Function,
|
||
onInputCompositionEnd: Function,
|
||
};
|
||
const SingleSelector = defineComponent<SelectorProps>({
|
||
name: 'SingleSelector',
|
||
setup(props) {
|
||
const inputChanged = ref(false);
|
||
|
||
const combobox = computed(() => props.mode === 'combobox');
|
||
const inputEditable = computed(() => combobox.value || props.showSearch);
|
||
|
||
const inputValue = computed(() => {
|
||
let inputValue: string = props.searchValue || '';
|
||
if (combobox.value && props.activeValue && !inputChanged.value) {
|
||
inputValue = props.activeValue;
|
||
}
|
||
return inputValue;
|
||
});
|
||
const legacyTreeSelectContext = useInjectLegacySelectContext();
|
||
watch(
|
||
[combobox, () => props.activeValue],
|
||
() => {
|
||
if (combobox.value) {
|
||
inputChanged.value = false;
|
||
}
|
||
},
|
||
{ immediate: true },
|
||
);
|
||
|
||
// Not show text when closed expect combobox mode
|
||
const hasTextInput = computed(() =>
|
||
props.mode !== 'combobox' && !props.open && !props.showSearch ? false : !!inputValue.value,
|
||
);
|
||
|
||
const title = computed(() => {
|
||
const item = props.values[0];
|
||
return item && (typeof item.label === 'string' || typeof item.label === 'number')
|
||
? item.label.toString()
|
||
: undefined;
|
||
});
|
||
|
||
const renderPlaceholder = () => {
|
||
if (props.values[0]) {
|
||
return null;
|
||
}
|
||
const hiddenStyle = hasTextInput.value ? { visibility: 'hidden' as const } : undefined;
|
||
return (
|
||
<span class={`${props.prefixCls}-selection-placeholder`} style={hiddenStyle}>
|
||
{props.placeholder}
|
||
</span>
|
||
);
|
||
};
|
||
|
||
return () => {
|
||
const {
|
||
inputElement,
|
||
prefixCls,
|
||
id,
|
||
values,
|
||
inputRef,
|
||
disabled,
|
||
autofocus,
|
||
autocomplete,
|
||
activeDescendantId,
|
||
open,
|
||
tabindex,
|
||
optionLabelRender,
|
||
onInputKeyDown,
|
||
onInputMouseDown,
|
||
onInputChange,
|
||
onInputPaste,
|
||
onInputCompositionStart,
|
||
onInputCompositionEnd,
|
||
} = props;
|
||
const item = values[0];
|
||
let titleNode = null;
|
||
// custom tree-select title by slot
|
||
|
||
// For TreeSelect
|
||
if (item && legacyTreeSelectContext.customSlots) {
|
||
const key = item.key ?? item.value;
|
||
const originData = legacyTreeSelectContext.keyEntities[key]?.node || {};
|
||
titleNode =
|
||
legacyTreeSelectContext.customSlots[originData.slots?.title] ||
|
||
legacyTreeSelectContext.customSlots.title ||
|
||
item.label;
|
||
if (typeof titleNode === 'function') {
|
||
titleNode = titleNode(originData);
|
||
}
|
||
// else if (treeSelectContext.value.slots.titleRender) {
|
||
// // 因历史 title 是覆盖逻辑,新增 titleRender,所有的 title 都走一遍 titleRender
|
||
// titleNode = treeSelectContext.value.slots.titleRender(item.option?.data || {});
|
||
// }
|
||
} else {
|
||
titleNode = optionLabelRender && item ? optionLabelRender(item.option) : item?.label;
|
||
}
|
||
return (
|
||
<>
|
||
<span class={`${prefixCls}-selection-search`}>
|
||
<Input
|
||
inputRef={inputRef}
|
||
prefixCls={prefixCls}
|
||
id={id}
|
||
open={open}
|
||
inputElement={inputElement}
|
||
disabled={disabled}
|
||
autofocus={autofocus}
|
||
autocomplete={autocomplete}
|
||
editable={inputEditable.value}
|
||
activeDescendantId={activeDescendantId}
|
||
value={inputValue.value}
|
||
onKeydown={onInputKeyDown}
|
||
onMousedown={onInputMouseDown}
|
||
onChange={e => {
|
||
inputChanged.value = true;
|
||
onInputChange(e as any);
|
||
}}
|
||
onPaste={onInputPaste}
|
||
onCompositionstart={onInputCompositionStart}
|
||
onCompositionend={onInputCompositionEnd}
|
||
tabindex={tabindex}
|
||
attrs={pickAttrs(props, true)}
|
||
/>
|
||
</span>
|
||
|
||
{/* Display value */}
|
||
{!combobox.value && item && !hasTextInput.value && (
|
||
<span class={`${prefixCls}-selection-item`} title={title.value}>
|
||
<Fragment key={item.key ?? item.value}>{titleNode}</Fragment>
|
||
</span>
|
||
)}
|
||
|
||
{/* Display placeholder */}
|
||
{renderPlaceholder()}
|
||
</>
|
||
);
|
||
};
|
||
},
|
||
});
|
||
SingleSelector.props = props;
|
||
SingleSelector.inheritAttrs = false;
|
||
|
||
export default SingleSelector;
|