mirror of
https://gitee.com/vant-contrib/vant.git
synced 2025-04-05 19:41:42 +08:00
286 lines
6.7 KiB
TypeScript
286 lines
6.7 KiB
TypeScript
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<CascaderFieldNames>,
|
|
placeholder: String,
|
|
activeColor: String,
|
|
options: {
|
|
type: Array as PropType<CascaderOption[]>,
|
|
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 = () => (
|
|
<div class={bem('header')}>
|
|
<h2 class={bem('title')}>
|
|
{slots.title ? slots.title() : props.title}
|
|
</h2>
|
|
{props.closeable ? (
|
|
<Icon
|
|
name={props.closeIcon}
|
|
class={bem('close-icon')}
|
|
onClick={onClose}
|
|
/>
|
|
) : null}
|
|
</div>
|
|
);
|
|
|
|
const renderOptions = (
|
|
options: CascaderOption[],
|
|
selectedOption: CascaderOption | null,
|
|
tabIndex: number
|
|
) => {
|
|
const renderOption = (option: CascaderOption) => {
|
|
const isSelected =
|
|
selectedOption && option[valueKey] === selectedOption[valueKey];
|
|
|
|
return (
|
|
<li
|
|
class={bem('option', { selected: isSelected })}
|
|
style={{ color: isSelected ? props.activeColor : undefined }}
|
|
onClick={() => onSelect(option, tabIndex)}
|
|
>
|
|
<span>{option[textKey]}</span>
|
|
{isSelected ? (
|
|
<Icon name="success" class={bem('selected-icon')} />
|
|
) : null}
|
|
</li>
|
|
);
|
|
};
|
|
|
|
return <ul class={bem('options')}>{options.map(renderOption)}</ul>;
|
|
};
|
|
|
|
const renderTab = (tab: CascaderTab, tabIndex: number) => {
|
|
const { options, selectedOption } = tab;
|
|
const title = selectedOption
|
|
? selectedOption[textKey]
|
|
: props.placeholder || t('select');
|
|
|
|
return (
|
|
<Tab
|
|
title={title}
|
|
titleClass={bem('tab', {
|
|
unselected: !selectedOption,
|
|
})}
|
|
>
|
|
{renderOptions(options, selectedOption, tabIndex)}
|
|
</Tab>
|
|
);
|
|
};
|
|
|
|
const renderTabs = () => (
|
|
<Tabs
|
|
v-model={[state.activeTab, 'active']}
|
|
animated
|
|
class={bem('tabs')}
|
|
color={props.activeColor}
|
|
swipeThreshold={0}
|
|
swipeable={props.swipeable}
|
|
onClick={onClickTab}
|
|
>
|
|
{state.tabs.map(renderTab)}
|
|
</Tabs>
|
|
);
|
|
|
|
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 () => (
|
|
<div class={bem()}>
|
|
{renderHeader()}
|
|
{renderTabs()}
|
|
</div>
|
|
);
|
|
},
|
|
});
|