mirror of
https://gitee.com/element-plus/element-plus.git
synced 2024-12-15 01:41:20 +08:00
904aa0e21b
* feat(ElTreeSelect): add tree select base component * refactor(ElTreeSelect): use render function and move select/tree props to them self module * fix(ElTreeSelect): init value not checked * fix(ElTreeSelect): `toArray` ignores valid values * fix(ElTreeSelect): expose not working when defined on mounted * fix(ElTreeSelect): watch `modelValue` deep * test(ElTreeSelect): add base unit test * perf(ElTreeSelect): default slot should be a function * fix(ElTreeSelect): `onNodeClick` can not call, * test(ElTreeSelect): update unit test * fix(ElTreeSelect): `onNodeClick` can not call, * fix(ElTreeSelect): remove folder node when `checkStrictly` is false * feat(ElTreeSelect): export `ElTreeSelect` * fix(ElTreeSelect): `filterMethod` conflicts with `filterNodeMethod` * docs(ElTreeSelect): add component docs * fix(ElTreeSelect): fix lint * docs(ElTreeSelect): fix lazy loading requires non-leaf nodes, and change mock labels * docs(ElTreeSelect): the link address of the attributes is incorrect * docs(ElTreeSelect): `dropdown` doesn't need the `-` symbol * refactor(ElTreeSelect): use alias path and make sure vue is above to components * refactor(ElTreeSelect): use a unified namespace for styles * docs(ElTreeSelect): change option labels in default slots * refactor(ElTreeSelect): import `ElOption` using unified entry and change the way to override the select click event * style(ElTreeSelect): sort imports * docs(ElTreeSelect): update the documentation for special codes * refactor(ElTreeSelect): keep it consistent with the select style * refactor(ElTreeSelect): use `isFunction` from `@element-plus/utils` * refactor(ElTreeSelect): use single closing tag when no subset * docs(ElTreeSelect): set `TreeSelect` promotion as `2.1.8`
313 lines
7.9 KiB
TypeScript
313 lines
7.9 KiB
TypeScript
import { h, nextTick, ref } from 'vue'
|
|
import { mount } from '@vue/test-utils'
|
|
import TreeSelect from '../src/tree-select.vue'
|
|
import type { RenderFunction } from 'vue'
|
|
import type { VueWrapper } from '@vue/test-utils'
|
|
import type ElSelect from '@element-plus/components/select'
|
|
import type ElTree from '@element-plus/components/tree'
|
|
|
|
const createComponent = ({
|
|
slots = {},
|
|
props = {},
|
|
}: {
|
|
slots?: Record<string, any>
|
|
props?: typeof TreeSelect['props']
|
|
} = {}) => {
|
|
// vm can not get component expose, use ref
|
|
const wrapperRef = ref()
|
|
const value = props.modelValue || ref('')
|
|
const wrapper = mount({
|
|
data() {
|
|
return {
|
|
modelValue: value,
|
|
data: [
|
|
{
|
|
value: 1,
|
|
label: '一级 1',
|
|
children: [
|
|
{
|
|
value: 11,
|
|
label: '二级 1-1',
|
|
children: [
|
|
{
|
|
value: 111,
|
|
label: '三级 1-1',
|
|
},
|
|
],
|
|
},
|
|
],
|
|
},
|
|
],
|
|
'onUpdate:modelValue': (val: string) => (value.value = val),
|
|
...props,
|
|
}
|
|
},
|
|
render() {
|
|
return h(
|
|
TreeSelect,
|
|
{
|
|
...this.$data,
|
|
ref: (val: object) => (wrapperRef.value = val),
|
|
},
|
|
slots
|
|
)
|
|
},
|
|
})
|
|
|
|
return {
|
|
wrapper,
|
|
getWrapperRef: () =>
|
|
new Promise((resolve) =>
|
|
nextTick(() => resolve(wrapperRef.value))
|
|
) as Promise<InstanceType<typeof ElTree> & InstanceType<typeof ElSelect>>,
|
|
select: wrapper.findComponent({ name: 'ElSelect' }) as VueWrapper<
|
|
InstanceType<typeof ElSelect>
|
|
>,
|
|
tree: wrapper.findComponent({ name: 'ElTree' }) as VueWrapper<
|
|
InstanceType<typeof ElTree>
|
|
>,
|
|
}
|
|
}
|
|
|
|
describe('TreeSelect.vue', () => {
|
|
test('render test', async () => {
|
|
const { wrapper, tree } = createComponent({
|
|
props: {
|
|
defaultExpandAll: true,
|
|
},
|
|
})
|
|
|
|
expect(wrapper.find('.el-tree')).toBeTruthy()
|
|
expect(wrapper.find('.el-select')).toBeTruthy()
|
|
|
|
expect(tree.findAll('.el-tree > .el-tree-node').length).toBe(1)
|
|
expect(tree.findAll('.el-tree .el-tree-node').length).toBe(3)
|
|
expect(tree.findAll('.el-tree .el-select-dropdown__item').length).toBe(3)
|
|
|
|
wrapper.vm.data[0].children = []
|
|
|
|
await nextTick()
|
|
|
|
expect(tree.findAll('.el-tree .el-tree-node').length).toBe(1)
|
|
})
|
|
|
|
test('modelValue', async () => {
|
|
const value = ref(1)
|
|
const { getWrapperRef, select, tree } = createComponent({
|
|
props: {
|
|
modelValue: value,
|
|
checkStrictly: true,
|
|
showCheckbox: true,
|
|
},
|
|
})
|
|
|
|
const wrapperRef = await getWrapperRef()
|
|
|
|
await nextTick()
|
|
expect(select.vm.modelValue).toBe(1)
|
|
expect(wrapperRef.getCheckedKeys()).toEqual([1])
|
|
|
|
value.value = 11
|
|
await nextTick(nextTick)
|
|
expect(select.vm.modelValue).toBe(11)
|
|
expect(wrapperRef.getCheckedKeys()).toEqual([11])
|
|
|
|
await tree
|
|
.findAll('.el-select-dropdown__item')
|
|
.slice(-1)[0]
|
|
.trigger('click')
|
|
await nextTick()
|
|
expect(select.vm.modelValue).toBe(111)
|
|
expect(wrapperRef.getCheckedKeys()).toEqual([111])
|
|
|
|
await tree.find('.el-tree-node__content').trigger('click')
|
|
await nextTick()
|
|
expect(select.vm.modelValue).toBe(1)
|
|
expect(wrapperRef.getCheckedKeys()).toEqual([1])
|
|
|
|
await tree.findAll('.el-checkbox')[1].trigger('click')
|
|
await nextTick()
|
|
expect(select.vm.modelValue).toBe(11)
|
|
expect(wrapperRef.getCheckedKeys()).toEqual([11])
|
|
})
|
|
|
|
test('disabled', async () => {
|
|
const { wrapper, tree } = createComponent({
|
|
props: {
|
|
data: [
|
|
{
|
|
value: '1',
|
|
label: '1',
|
|
children: [
|
|
{
|
|
value: '2',
|
|
label: '2',
|
|
disabled: true,
|
|
},
|
|
],
|
|
},
|
|
],
|
|
showCheckbox: true,
|
|
checkStrictly: true,
|
|
defaultExpandAll: true,
|
|
},
|
|
})
|
|
|
|
await nextTick()
|
|
await tree.find('.el-tree-node').trigger('click')
|
|
await tree.find('.el-tree-node .el-checkbox.is-disabled').trigger('click')
|
|
await tree
|
|
.find('.el-tree-node .el-select-dropdown__item.is-disabled')
|
|
.trigger('click')
|
|
await nextTick()
|
|
expect(wrapper.vm.modelValue).toBe('1')
|
|
})
|
|
|
|
test('multiple', async () => {
|
|
const value = ref([1])
|
|
const { getWrapperRef, select, tree } = createComponent({
|
|
props: {
|
|
modelValue: value,
|
|
checkStrictly: true,
|
|
showCheckbox: true,
|
|
multiple: true,
|
|
},
|
|
})
|
|
|
|
const wrapperRef = await getWrapperRef()
|
|
|
|
await nextTick()
|
|
expect(select.vm.modelValue).toEqual([1])
|
|
expect(wrapperRef.getCheckedKeys()).toEqual([1])
|
|
|
|
value.value = [11]
|
|
await nextTick(nextTick)
|
|
expect(select.vm.modelValue).toEqual([11])
|
|
expect(wrapperRef.getCheckedKeys()).toEqual([11])
|
|
|
|
await tree
|
|
.findAll('.el-select-dropdown__item')
|
|
.slice(-1)[0]
|
|
.trigger('click')
|
|
await nextTick()
|
|
expect(select.vm.modelValue).toEqual([11, 111])
|
|
expect(wrapperRef.getCheckedKeys()).toEqual([11, 111])
|
|
|
|
await tree.find('.el-tree-node__content').trigger('click')
|
|
await nextTick()
|
|
expect(select.vm.modelValue).toEqual([11, 111, 1])
|
|
expect(wrapperRef.getCheckedKeys()).toEqual([1, 11, 111])
|
|
|
|
await tree.findAll('.el-checkbox')[1].trigger('click')
|
|
await nextTick()
|
|
expect(select.vm.modelValue).toEqual([1, 111])
|
|
expect(wrapperRef.getCheckedKeys()).toEqual([1, 111])
|
|
})
|
|
|
|
test('filter', async () => {
|
|
const { select, tree } = createComponent({
|
|
props: {
|
|
filterable: true,
|
|
},
|
|
})
|
|
|
|
select.vm.query = '一级 1'
|
|
await nextTick()
|
|
expect(tree.findAll('.el-tree-node').length).toBe(1)
|
|
})
|
|
|
|
test('props', async () => {
|
|
const { wrapper, select, tree } = createComponent({
|
|
props: {
|
|
data: [
|
|
{
|
|
id: '1',
|
|
name: '1',
|
|
childrens: [
|
|
{
|
|
id: '2',
|
|
name: '2',
|
|
},
|
|
],
|
|
},
|
|
],
|
|
props: {
|
|
label: 'name',
|
|
children: 'childrens',
|
|
},
|
|
valueKey: 'id',
|
|
},
|
|
})
|
|
|
|
await nextTick()
|
|
expect(tree.find('.el-tree-node').text()).toBe('1')
|
|
wrapper.vm.modelValue = '2'
|
|
await nextTick()
|
|
expect(select.vm.selectedLabel).toBe('2')
|
|
})
|
|
|
|
test('slots', async () => {
|
|
const { select, tree } = createComponent({
|
|
slots: {
|
|
default: ({ data }: { data: { label: string } }) => `123${data.label}`,
|
|
prefix: () => 'prefix',
|
|
},
|
|
})
|
|
|
|
await nextTick()
|
|
expect(tree.find('.el-select-dropdown__item').text()).toBe('123一级 1')
|
|
expect(select.find('.el-input__prefix-inner').text()).toBe('prefix')
|
|
})
|
|
|
|
test('renderContent', async () => {
|
|
const { tree } = createComponent({
|
|
props: {
|
|
renderContent: (
|
|
h: RenderFunction,
|
|
{ data }: { data: { label: string } }
|
|
) => {
|
|
return `123${data.label}`
|
|
},
|
|
},
|
|
})
|
|
|
|
await nextTick()
|
|
expect(tree.find('.el-select-dropdown__item').text()).toBe('123一级 1')
|
|
})
|
|
|
|
test('lazy', async () => {
|
|
const { tree } = createComponent({
|
|
props: {
|
|
data: [
|
|
{
|
|
value: 1,
|
|
label: 1,
|
|
},
|
|
],
|
|
lazy: true,
|
|
load: (node: object, resolve: (p: any) => any[]) => {
|
|
resolve([{ value: 2, label: 2, isLeaf: true }])
|
|
},
|
|
},
|
|
})
|
|
|
|
await nextTick()
|
|
await tree.find('.el-tree-node').trigger('click')
|
|
await nextTick()
|
|
expect(tree.find('.el-tree-node .el-tree-node').text()).toBe('2')
|
|
})
|
|
|
|
test('events', async () => {
|
|
const onNodeClick = jest.fn()
|
|
const { tree } = createComponent({
|
|
props: {
|
|
onNodeClick,
|
|
},
|
|
})
|
|
await nextTick()
|
|
await tree.find('.el-tree-node').trigger('click')
|
|
await nextTick()
|
|
expect(onNodeClick).toBeCalled()
|
|
})
|
|
})
|