vue/test/transition/transition-group.spec.ts
2022-05-23 17:21:17 +08:00

393 lines
11 KiB
TypeScript

import Vue from 'vue'
import { injectStyles, waitForUpdate, nextFrame } from './helpers'
describe('Transition group', () => {
const { duration, buffer } = injectStyles()
let el
beforeEach(() => {
el = document.createElement('div')
document.body.appendChild(el)
})
function createBasicVM(useIs?, appear = false) {
const vm = new Vue({
template: `
<div>
${
useIs
? `<span is="transition-group">`
: `<transition-group${appear ? ` appear` : ``}>`
}
<div v-for="item in items" :key="item" class="test">{{ item }}</div>
${useIs ? `</span>` : `</transition-group>`}
</div>
`,
data: {
items: ['a', 'b', 'c']
}
}).$mount(el)
if (!appear) {
expect(vm.$el.innerHTML).toBe(
`<span>` +
vm.items.map(i => `<div class="test">${i}</div>`).join('') +
`</span>`
)
}
return vm
}
it('enter', done => {
const vm = createBasicVM()
vm.items.push('d', 'e')
waitForUpdate(() => {
expect(vm.$el.innerHTML).toBe(
`<span>` +
['a', 'b', 'c'].map(i => `<div class="test">${i}</div>`).join('') +
`<div class="test v-enter v-enter-active">d</div>` +
`<div class="test v-enter v-enter-active">e</div>` +
`</span>`
)
})
.thenWaitFor(nextFrame)
.then(() => {
expect(vm.$el.innerHTML).toBe(
`<span>` +
['a', 'b', 'c'].map(i => `<div class="test">${i}</div>`).join('') +
`<div class="test v-enter-active v-enter-to">d</div>` +
`<div class="test v-enter-active v-enter-to">e</div>` +
`</span>`
)
})
.thenWaitFor(duration + buffer)
.then(() => {
expect(vm.$el.innerHTML).toBe(
`<span>` +
vm.items.map(i => `<div class="test">${i}</div>`).join('') +
`</span>`
)
})
.then(done)
})
it('leave', done => {
const vm = createBasicVM()
vm.items = ['b']
waitForUpdate(() => {
expect(vm.$el.innerHTML).toBe(
`<span>` +
`<div class="test v-leave v-leave-active">a</div>` +
`<div class="test">b</div>` +
`<div class="test v-leave v-leave-active">c</div>` +
`</span>`
)
})
.thenWaitFor(nextFrame)
.then(() => {
expect(vm.$el.innerHTML).toBe(
`<span>` +
`<div class="test v-leave-active v-leave-to">a</div>` +
`<div class="test">b</div>` +
`<div class="test v-leave-active v-leave-to">c</div>` +
`</span>`
)
})
.thenWaitFor(duration + buffer)
.then(() => {
expect(vm.$el.innerHTML).toBe(
`<span>` +
vm.items.map(i => `<div class="test">${i}</div>`).join('') +
`</span>`
)
})
.then(done)
})
it('enter + leave', done => {
const vm = createBasicVM()
vm.items = ['b', 'c', 'd']
waitForUpdate(() => {
expect(vm.$el.innerHTML).toBe(
`<span>` +
`<div class="test v-leave v-leave-active">a</div>` +
`<div class="test">b</div>` +
`<div class="test">c</div>` +
`<div class="test v-enter v-enter-active">d</div>` +
`</span>`
)
})
.thenWaitFor(nextFrame)
.then(() => {
expect(vm.$el.innerHTML).toBe(
`<span>` +
`<div class="test v-leave-active v-leave-to">a</div>` +
`<div class="test">b</div>` +
`<div class="test">c</div>` +
`<div class="test v-enter-active v-enter-to">d</div>` +
`</span>`
)
})
.thenWaitFor(duration + buffer)
.then(() => {
expect(vm.$el.innerHTML).toBe(
`<span>` +
vm.items.map(i => `<div class="test">${i}</div>`).join('') +
`</span>`
)
})
.then(done)
})
it('use with "is" attribute', done => {
const vm = createBasicVM(true)
vm.items = ['b', 'c', 'd']
waitForUpdate(() => {
expect(vm.$el.innerHTML).toBe(
`<span>` +
`<div class="test v-leave v-leave-active">a</div>` +
`<div class="test">b</div>` +
`<div class="test">c</div>` +
`<div class="test v-enter v-enter-active">d</div>` +
`</span>`
)
})
.thenWaitFor(nextFrame)
.then(() => {
expect(vm.$el.innerHTML).toBe(
`<span>` +
`<div class="test v-leave-active v-leave-to">a</div>` +
`<div class="test">b</div>` +
`<div class="test">c</div>` +
`<div class="test v-enter-active v-enter-to">d</div>` +
`</span>`
)
})
.thenWaitFor(duration + buffer)
.then(() => {
expect(vm.$el.innerHTML).toBe(
`<span>` +
vm.items.map(i => `<div class="test">${i}</div>`).join('') +
`</span>`
)
})
.then(done)
})
it('appear', done => {
const vm = createBasicVM(false, true /* appear */)
waitForUpdate(() => {
expect(vm.$el.innerHTML).toBe(
`<span>` +
vm.items
.map(i => `<div class="test v-enter v-enter-active">${i}</div>`)
.join('') +
`</span>`
)
})
.thenWaitFor(nextFrame)
.then(() => {
expect(vm.$el.innerHTML).toBe(
`<span>` +
vm.items
.map(
i => `<div class="test v-enter-active v-enter-to">${i}</div>`
)
.join('') +
`</span>`
)
})
.thenWaitFor(duration + buffer)
.then(() => {
expect(vm.$el.innerHTML).toBe(
`<span>` +
vm.items.map(i => `<div class="test">${i}</div>`).join('') +
`</span>`
)
})
.then(done)
})
it('events', done => {
let next
const beforeEnterSpy = jasmine.createSpy()
const afterEnterSpy = jasmine.createSpy()
const afterLeaveSpy = jasmine.createSpy()
const vm = new Vue({
template: `
<div>
<transition-group @before-enter="beforeEnter" @after-enter="afterEnter" @after-leave="afterLeave">
<div v-for="item in items" :key="item" class="test">{{ item }}</div>
</transition-group>
</div>
`,
data: {
items: ['a', 'b', 'c']
},
methods: {
beforeEnter(el) {
expect(el.textContent).toBe('d')
beforeEnterSpy()
},
afterEnter(el) {
expect(el.textContent).toBe('d')
afterEnterSpy()
next()
},
afterLeave(el) {
expect(el.textContent).toBe('a')
afterLeaveSpy()
next()
}
}
}).$mount(el)
vm.items.push('d')
waitForUpdate(() => {
expect(vm.$el.innerHTML).toBe(
`<span>` +
`<div class="test">a</div>` +
`<div class="test">b</div>` +
`<div class="test">c</div>` +
`<div class="test v-enter v-enter-active">d</div>` +
`</span>`
)
expect(beforeEnterSpy.calls.count()).toBe(1)
})
.thenWaitFor(_next => {
next = _next
})
.then(() => {
expect(vm.$el.innerHTML).toBe(
`<span>` +
`<div class="test">a</div>` +
`<div class="test">b</div>` +
`<div class="test">c</div>` +
`<div class="test">d</div>` +
`</span>`
)
expect(afterEnterSpy.calls.count()).toBe(1)
vm.items.shift()
})
.thenWaitFor(_next => {
next = _next
})
.then(() => {
expect(vm.$el.innerHTML).toBe(
`<span>` +
`<div class="test">b</div>` +
`<div class="test">c</div>` +
`<div class="test">d</div>` +
`</span>`
)
expect(afterLeaveSpy.calls.count()).toBe(1)
})
.then(done)
})
it('move', done => {
const vm = new Vue({
template: `
<div>
<transition-group name="group">
<div v-for="item in items" :key="item" class="test">{{ item }}</div>
</transition-group>
</div>
`,
data: {
items: ['a', 'b', 'c']
}
}).$mount(el)
vm.items = ['d', 'b', 'a']
waitForUpdate(() => {
expect(vm.$el.innerHTML.replace(/\s?style=""(\s?)/g, '$1')).toBe(
`<span>` +
`<div class="test group-enter group-enter-active">d</div>` +
`<div class="test">b</div>` +
`<div class="test group-move">a</div>` +
`<div class="test group-leave group-leave-active group-move">c</div>` +
`</span>`
)
})
.thenWaitFor(nextFrame)
.then(() => {
expect(vm.$el.innerHTML.replace(/\s?style=""(\s?)/g, '$1')).toBe(
`<span>` +
`<div class="test group-enter-active group-enter-to">d</div>` +
`<div class="test">b</div>` +
`<div class="test group-move">a</div>` +
`<div class="test group-leave-active group-move group-leave-to">c</div>` +
`</span>`
)
})
.thenWaitFor(duration * 2)
.then(() => {
expect(vm.$el.innerHTML.replace(/\s?style=""(\s?)/g, '$1')).toBe(
`<span>` +
`<div class="test">d</div>` +
`<div class="test">b</div>` +
`<div class="test">a</div>` +
`</span>`
)
})
.then(done)
})
it('warn unkeyed children', () => {
new Vue({
template: `<div><transition-group><div v-for="i in 3"></div></transition-group></div>`
}).$mount()
expect(
'<transition-group> children must be keyed: <div>'
).toHaveBeenWarned()
})
// GitHub issue #6006
it('should work with dynamic name', done => {
const vm = new Vue({
template: `
<div>
<transition-group :name="name">
<div v-for="item in items" :key="item">{{ item }}</div>
</transition-group>
</div>
`,
data: {
items: ['a', 'b', 'c'],
name: 'group'
}
}).$mount(el)
vm.name = 'invalid-name'
vm.items = ['b', 'c', 'a']
waitForUpdate(() => {
expect(vm.$el.innerHTML.replace(/\s?style=""(\s?)/g, '$1')).toBe(
`<span>` + `<div>b</div>` + `<div>c</div>` + `<div>a</div>` + `</span>`
)
vm.name = 'group'
vm.items = ['a', 'b', 'c']
})
.thenWaitFor(nextFrame)
.then(() => {
expect(vm.$el.innerHTML.replace(/\s?style=""(\s?)/g, '$1')).toBe(
`<span>` +
`<div class="group-move">a</div>` +
`<div class="group-move">b</div>` +
`<div class="group-move">c</div>` +
`</span>`
)
})
.thenWaitFor(duration * 2 + buffer)
.then(() => {
expect(vm.$el.innerHTML.replace(/\s?style=""(\s?)/g, '$1')).toBe(
`<span>` +
`<div>a</div>` +
`<div>b</div>` +
`<div>c</div>` +
`</span>`
)
})
.then(done)
})
})