import { nextTick, PropType, reactive, watch, defineComponent } from 'vue'; import { createNamespace, truthProp, extend } from '../utils'; // Components import { Tab } from '../tab'; import { Tabs } from '../tabs'; import { Icon } from '../icon'; const [name, bem, t] = createNamespace('cascader'); export type CascaderOption = { text?: string; value?: string | number; children?: CascaderOption[]; // for custom filed names [key: string]: any; }; type CascaderTab = { options: CascaderOption[]; selectedOption: CascaderOption | null; }; type CascaderFieldNames = { text?: string; value?: string; children?: string; }; export default defineComponent({ name, props: { title: String, closeable: truthProp, swipeable: truthProp, modelValue: [Number, String], fieldNames: Object as PropType, placeholder: String, activeColor: String, options: { type: Array as PropType, default: () => [], }, closeIcon: { type: String, default: 'cross', }, }, emits: ['close', 'change', 'finish', 'update:modelValue', 'click-tab'], setup(props, { slots, emit }) { const state = reactive({ tabs: [] as CascaderTab[], activeTab: 0, }); const { text: textKey, value: valueKey, children: childrenKey } = extend( { text: 'text', value: 'value', children: 'children', }, props.fieldNames ); const getSelectedOptionsByValue = ( options: CascaderOption[], value: string | number ): CascaderOption[] | undefined => { for (let i = 0; i < options.length; i++) { const option = options[i]; if (option[valueKey] === value) { return [option]; } if (option[childrenKey]) { const selectedOptions = getSelectedOptionsByValue( option[childrenKey], value ); if (selectedOptions) { return [option, ...selectedOptions]; } } } }; const updateTabs = () => { if (props.modelValue || props.modelValue === 0) { const selectedOptions = getSelectedOptionsByValue( props.options, props.modelValue ); if (selectedOptions) { let optionsCursor = props.options; state.tabs = selectedOptions.map((option) => { const tab = { options: optionsCursor, selectedOption: option, }; const next = optionsCursor.find( (item) => item[valueKey] === option[valueKey] ); if (next) { optionsCursor = next[childrenKey]; } return tab; }); if (optionsCursor) { state.tabs.push({ options: optionsCursor, selectedOption: null, }); } nextTick(() => { state.activeTab = state.tabs.length - 1; }); return; } } state.tabs = [ { options: props.options, selectedOption: null, }, ]; }; const onSelect = (option: CascaderOption, tabIndex: number) => { state.tabs[tabIndex].selectedOption = option; if (state.tabs.length > tabIndex + 1) { state.tabs = state.tabs.slice(0, tabIndex + 1); } if (option[childrenKey]) { const nextTab = { options: option[childrenKey], selectedOption: null, }; if (state.tabs[tabIndex + 1]) { state.tabs[tabIndex + 1] = nextTab; } else { state.tabs.push(nextTab); } nextTick(() => { state.activeTab++; }); } const selectedOptions = state.tabs .map((tab) => tab.selectedOption) .filter(Boolean); const eventParams = { value: option[valueKey], tabIndex, selectedOptions, }; emit('update:modelValue', option[valueKey]); emit('change', eventParams); if (!option[childrenKey]) { emit('finish', eventParams); } }; const onClose = () => emit('close'); const onClickTab = (tabIndex: number, title: string) => { emit('click-tab', tabIndex, title); }; const renderHeader = () => (

{slots.title ? slots.title() : props.title}

{props.closeable ? ( ) : null}
); const renderOptions = ( options: CascaderOption[], selectedOption: CascaderOption | null, tabIndex: number ) => { const renderOption = (option: CascaderOption) => { const isSelected = selectedOption && option[valueKey] === selectedOption[valueKey]; return (
  • onSelect(option, tabIndex)} > {option[textKey]} {isSelected ? ( ) : null}
  • ); }; return ; }; const renderTab = (tab: CascaderTab, tabIndex: number) => { const { options, selectedOption } = tab; const title = selectedOption ? selectedOption[textKey] : props.placeholder || t('select'); return ( {renderOptions(options, selectedOption, tabIndex)} ); }; const renderTabs = () => ( {state.tabs.map(renderTab)} ); updateTabs(); watch(() => props.options, updateTabs, { deep: true }); watch( () => props.modelValue, (value) => { if (value || value === 0) { const values = state.tabs.map( (tab) => tab.selectedOption?.[valueKey] ); if (values.includes(value)) { return; } } updateTabs(); } ); return () => (
    {renderHeader()} {renderTabs()}
    ); }, });