Skip to content

Commit

Permalink
feat(cascader): 新增默认选项标签,可以自定义定制级联组件中“选择选项”文案
Browse files Browse the repository at this point in the history
feat: #1130
  • Loading branch information
yiqia committed Oct 23, 2023
1 parent 3d74b3e commit 70e16f0
Show file tree
Hide file tree
Showing 9 changed files with 73 additions and 19 deletions.
20 changes: 20 additions & 0 deletions src/cascader/__test__/__snapshots__/demo.test.jsx.snap
Original file line number Diff line number Diff line change
Expand Up @@ -108,7 +108,9 @@ exports[`Cascader > Cascader baseVue demo works fine 1`] = `
<div
class="t-cascader__step-label t-cascader__step-label--active"
>
选择选项
</div>
<svg
class="t-icon t-icon-chevron-right t-cascader__step-arrow"
Expand Down Expand Up @@ -336,7 +338,9 @@ exports[`Cascader > Cascader keysVue demo works fine 1`] = `
<div
class="t-cascader__step-label t-cascader__step-label--active"
>
选择选项
</div>
<svg
class="t-icon t-icon-chevron-right t-cascader__step-arrow"
Expand Down Expand Up @@ -564,7 +568,9 @@ exports[`Cascader > Cascader lazyVue demo works fine 1`] = `
<div
class="t-cascader__step-label t-cascader__step-label--active"
>
选择选项
</div>
<svg
class="t-icon t-icon-chevron-right t-cascader__step-arrow"
Expand Down Expand Up @@ -789,7 +795,9 @@ exports[`Cascader > Cascader mobileVue demo works fine 1`] = `
<div
class="t-cascader__step-label t-cascader__step-label--active"
>
选择选项
</div>
<svg
class="t-icon t-icon-chevron-right t-cascader__step-arrow"
Expand Down Expand Up @@ -1246,7 +1254,9 @@ exports[`Cascader > Cascader mobileVue demo works fine 1`] = `
<div
class="t-cascader__step-label t-cascader__step-label--active"
>
选择选项
</div>
<svg
class="t-icon t-icon-chevron-right t-cascader__step-arrow"
Expand Down Expand Up @@ -1487,7 +1497,9 @@ exports[`Cascader > Cascader mobileVue demo works fine 1`] = `
<div
class="t-cascader__step-label t-cascader__step-label--active"
>
选择选项
</div>
<svg
class="t-icon t-icon-chevron-right t-cascader__step-arrow"
Expand Down Expand Up @@ -1728,7 +1740,9 @@ exports[`Cascader > Cascader mobileVue demo works fine 1`] = `
<div
class="t-cascader__step-label t-cascader__step-label--active"
>
选择选项
</div>
<svg
class="t-icon t-icon-chevron-right t-cascader__step-arrow"
Expand Down Expand Up @@ -1973,7 +1987,9 @@ exports[`Cascader > Cascader mobileVue demo works fine 1`] = `
<div
class="t-cascader__step-label t-cascader__step-label--active"
>
选择选项
</div>
<svg
class="t-icon t-icon-chevron-right t-cascader__step-arrow"
Expand Down Expand Up @@ -2369,7 +2385,9 @@ exports[`Cascader > Cascader withTitleVue demo works fine 1`] = `
<div
class="t-cascader__step-label t-cascader__step-label--active"
>
选择选项
</div>
<svg
class="t-icon t-icon-chevron-right t-cascader__step-arrow"
Expand Down Expand Up @@ -2601,7 +2619,9 @@ exports[`Cascader > Cascader withValueVue demo works fine 1`] = `
<div
class="t-cascader__step-label t-cascader__step-label--active"
>
选择选项
</div>
<svg
class="t-icon t-icon-chevron-right t-cascader__step-arrow"
Expand Down
2 changes: 2 additions & 0 deletions src/cascader/__test__/__snapshots__/index.test.jsx.snap
Original file line number Diff line number Diff line change
Expand Up @@ -60,7 +60,9 @@ exports[`cascader > events > : pick 1`] = `
<div
class="t-cascader__step-label t-cascader__step-label--active"
>
选择选项
</div>
<svg
class="t-icon t-icon-chevron-right t-cascader__step-arrow"
Expand Down
20 changes: 15 additions & 5 deletions src/cascader/__test__/index.test.jsx
Original file line number Diff line number Diff line change
Expand Up @@ -158,7 +158,7 @@ const itemList = [

const subTitles = ['请选择省份', '请选择城市', '请选择区/县'];

const defaultOptionLabel = "默认选项标签";
const defaultPlaceholder = "默认选项标签";

describe('cascader', () => {
describe('props', () => {
Expand Down Expand Up @@ -216,10 +216,10 @@ describe('cascader', () => {
});
});

it(': defaultOptionLabel', async () => {
const wrapper = mount(<Cascader options={options} defaultOptionLabel={defaultOptionLabel} />);
const $defaultOptionLabel = wrapper.find(`.${name}__step-label`);
expect($defaultOptionLabel.text()).toEqual(defaultOptionLabel);
it(': placeholder', async () => {
const wrapper = mount(<Cascader options={options} placeholder={defaultPlaceholder} />);
const $placeholder = wrapper.find(`.${name}__step-label`);
expect($placeholder.text()).toEqual(defaultPlaceholder);
});

it(': subTitles', async () => {
Expand Down Expand Up @@ -264,6 +264,16 @@ describe('cascader', () => {
expect($title.text()).toBe(title);
});

it(': placeholder', () => {
const wrapper = mount(<Cascader options={options} />, {
slots: {
placeholder: defaultPlaceholder,
},
});
const $placeholder = wrapper.find(`.${name}__step-label`);
expect($placeholder.text()).toBe(defaultPlaceholder);
});

it(': closeBtn', () => {
const closeBtn = '关闭';
const wrapper = mount(<Cascader options={options} />, {
Expand Down
2 changes: 1 addition & 1 deletion src/cascader/cascader.en-US.md
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@ title | String / Slot / Function | - | Typescript:`string \| TNode`。[see mor
value | String / Number | - | `v-model` and `v-model:value` is supported | N
defaultValue | String / Number | - | uncontrolled property | N
visible | Boolean | false | \- | N
defaultOptionLabel | String | 选择选项 | \- | N
placeholder | String / Slot / Function | 选择选项 | Typescript:`string \| TNode`[see more ts definition](https://github.com/Tencent/tdesign-mobile-vue/blob/develop/src/common.ts) | N
onChange | Function | | Typescript:`(value: string \| number, selectedOptions: string[]) => void`<br/> | N
onClose | Function | | Typescript:`(trigger: TriggerSource) => void`<br/>[see more ts definition](https://github.com/Tencent/tdesign-mobile-vue/tree/develop/src/cascader/type.ts)。<br/>`type TriggerSource = 'overlay' \| 'close-btn' \| 'finish'`<br/> | N
onPick | Function | | Typescript:`(context: { level: number; value: string | number; index: number }) => void`<br/> | N
Expand Down
2 changes: 1 addition & 1 deletion src/cascader/cascader.md
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,7 @@ title | String / Slot / Function | - | 标题。TS 类型:`string \| TNode`。
value | String / Number | - | 选项值。支持语法糖 `v-model``v-model:value` | N
defaultValue | String / Number | - | 选项值。非受控属性 | N
visible | Boolean | false | 是否展示 | N
defaultOptionLabel | String | 选择选项 | 默认选项标签 | N
placeholder | String / Slot / Function | 选择选项 | 未选中时的提示文案。TS 类型:`string \| TNode`[通用类型定义](https://github.com/Tencent/tdesign-mobile-vue/blob/develop/src/common.ts) | N
onChange | Function | | TS 类型:`(value: string \| number, selectedOptions: string[]) => void`<br/>值发生变更时触发 | N
onClose | Function | | TS 类型:`(trigger: TriggerSource) => void`<br/>关闭时触发。[详细类型定义](https://github.com/Tencent/tdesign-mobile-vue/tree/develop/src/cascader/type.ts)。<br/>`type TriggerSource = 'overlay' \| 'close-btn' \| 'finish'`<br/> | N
onPick | Function | | TS 类型:`(context: { level: number, value: string \| number, index: number }) => void`<br/>选择后触发 | N
Expand Down
30 changes: 23 additions & 7 deletions src/cascader/cascader.vue
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@
:class="[
`${name}__step-dot`,
{
't-cascader__step-dot--active': item !== defaultOptionLabel,
't-cascader__step-dot--active': item !== placeholder,
't-cascader__step-dot--last': index === steps.length - 1,
},
]"
Expand All @@ -29,7 +29,11 @@
},
]"
>
{{ item }}
<t-node
v-if="placeholderTNode && !(typeof placeholderTNode === 'string') && item === placeholder"
:content="placeholderTNode"
/>
<template v-else>{{ item }}</template>
</div>
<chevron-right-icon :class="`${name}__step-arrow`" size="22" />
</div>
Expand Down Expand Up @@ -127,13 +131,20 @@ export default defineComponent({
const open = ref(visible.value || false);
const [cascaderValue, setCascaderValue] = useVModel(value, modelValue, props.defaultValue, props.onChange);
const title = computed(() => props.title || '标题');
const defaultOptionLabel = props.defaultOptionLabel || '选择选项';
const placeholder = computed(() => props.placeholder || '选择选项');
const stepIndex = ref(0);
const selectedIndexes = reactive<string[] | number[]>([]);
const selectedValue = reactive<string[]>([]);
const items: Array<Array<TreeOptionData>> = reactive([options.value ?? []]);
const steps = reactive([defaultOptionLabel]);
const steps = reactive([placeholder.value]);
watch(placeholder, (newValue, oldValue) => {
const index = steps.indexOf(oldValue);
if (index !== -1) {
steps[index] = newValue;
}
});
onMounted(() => {
initWithValue();
Expand All @@ -146,6 +157,7 @@ export default defineComponent({
});
});
const titleTNode = computed(() => renderTNode(internalInstance, 'title'));
const placeholderTNode = computed(() => renderTNode(internalInstance, 'placeholder'));
const initWithValue = () => {
if (value.value != null) {
Expand All @@ -172,7 +184,8 @@ export default defineComponent({
}
}
if (steps.length < items.length) {
steps.push(defaultOptionLabel);
console.log('placeholder.value', placeholder.value);
steps.push(placeholder.value);
}
stepIndex.value = items.length - 1;
};
Expand Down Expand Up @@ -210,9 +223,11 @@ export default defineComponent({
if (item[(keys as Ref<KeysType>).value?.children ?? 'children']?.length) {
items[level + 1] = item[(keys as Ref<KeysType>).value?.children ?? 'children'];
console.log('placeholder.value', placeholder.value);
items.length = level + 2;
stepIndex.value += 1;
steps[level + 1] = '选择选项';
steps[level + 1] = placeholder.value;
steps.length = level + 2;
} else if (item[(keys as Ref<KeysType>).value?.children ?? 'children']?.length === 0) {
childrenInfo.value = e;
Expand Down Expand Up @@ -269,13 +284,14 @@ export default defineComponent({
return {
open,
defaultOptionLabel,
placeholder,
onVisibleChange,
onStepClick,
onTabChange,
handleSelect,
closeBtnTNode,
titleTNode,
placeholderTNode,
stepIndex,
name,
title,
Expand Down
6 changes: 3 additions & 3 deletions src/cascader/props.ts
Original file line number Diff line number Diff line change
Expand Up @@ -53,9 +53,9 @@ export default {
defaultValue: {
type: [String, Number] as PropType<TdCascaderProps['defaultValue']>,
},
/** 默认选项标签 */
defaultOptionLabel: {
type: String as PropType<TdCascaderProps['defaultOptionLabel']>,
/** 未选中时的提示文案 */
placeholder: {
type: [String, Function] as PropType<TdCascaderProps['placeholder']>,
},
/** 是否展示 */
visible: Boolean,
Expand Down
4 changes: 2 additions & 2 deletions src/cascader/type.ts
Original file line number Diff line number Diff line change
Expand Up @@ -53,9 +53,9 @@ export interface TdCascaderProps<CascaderOption extends TreeOptionData = TreeOpt
*/
visible?: boolean;
/**
* 默认选项标签
* 未选中时的提示文案
*/
defaultOptionLabel?: string;
placeholder?: string | TNode;
/**
* 值发生变更时触发
*/
Expand Down
6 changes: 6 additions & 0 deletions src/form/__test__/__snapshots__/demo.test.jsx.snap
Original file line number Diff line number Diff line change
Expand Up @@ -879,7 +879,9 @@ exports[`Form > Form horizontalVue demo works fine 1`] = `
<div
class="t-cascader__step-label t-cascader__step-label--active"
>
选择选项
</div>
<svg
class="t-icon t-icon-chevron-right t-cascader__step-arrow"
Expand Down Expand Up @@ -2608,7 +2610,9 @@ exports[`Form > Form mobileVue demo works fine 1`] = `
<div
class="t-cascader__step-label t-cascader__step-label--active"
>
选择选项
</div>
<svg
class="t-icon t-icon-chevron-right t-cascader__step-arrow"
Expand Down Expand Up @@ -4197,7 +4201,9 @@ exports[`Form > Form verticalVue demo works fine 1`] = `
<div
class="t-cascader__step-label t-cascader__step-label--active"
>
选择选项
</div>
<svg
class="t-icon t-icon-chevron-right t-cascader__step-arrow"
Expand Down

0 comments on commit 70e16f0

Please sign in to comment.