事件处理
Vue 组件通过 props 和调用 $emit
触发事件来进行交互。在本指南中,我们将介绍如何使用 emitted()
函数来验证事件是否正确触发。
本文也有短视频可供观看。
计数器组件
这是一个简单的 <Counter>
组件。它包含一个按钮,当点击时,会增加一个内部计数变量并触发该值的事件:
<!-- Counter.vue -->
<script setup>
import { ref } from 'vue'
const count = ref(0)
const emit = defineEmits(['increment'])
const handleClick = () => {
count.value += 1
emit('increment', count.value)
}
</script>
<template>
<button @click="handleClick">Increment</button>
</template>
为了全面测试这个组件,我们需要验证是否触发了带有最新 count
值的 increment
事件。
断言触发的事件
为此,我们将依赖 emitted()
方法。它返回一个对象,包含组件触发的所有事件,其参数以数组的形式呈现。让我们看看它是如何工作的:
test('emits an event when clicked', () => {
const wrapper = mount(Counter)
wrapper.find('button').trigger('click')
wrapper.find('button').trigger('click')
expect(wrapper.emitted()).toHaveProperty('increment')
})
如果你之前没有见过
trigger()
,不要担心。它用于模拟用户交互。你可以在测试表单中了解更多。
首先要注意的是,emitted()
返回一个对象,其中每个键都匹配一个已触发的事件。在这个例子中是 increment
。
这个测试应该通过。我们确保发出了具有适当名称的事件。
断言事件的参数
这很好,但我们可以做得更好!我们需要检查在调用 this.$emit('increment', this.count)
时是否发出了正确的参数。
我们的下一步是断言事件包含 count
值。我们通过将参数传递给 emitted()
来实现这一点。
test('emits an event with count when clicked', () => {
const wrapper = mount(Counter)
wrapper.find('button').trigger('click')
wrapper.find('button').trigger('click')
// `emitted()` 接受一个参数。它返回一个包含所有 `this.$emit('increment')` 发生情况的数组。
const incrementEvent = wrapper.emitted('increment')
// 我们“点击”了两次,所以 `increment` 的数组应该有两个值。
expect(incrementEvent).toHaveLength(2)
// 断言第一次点击的结果。
// 注意,值是一个数组。
expect(incrementEvent[0]).toEqual([1])
// 然后是第二次的结果。
expect(incrementEvent[1]).toEqual([2])
})
让我们回顾一下并分解 emitted()
的输出。每个键都包含测试期间触发的不同值:
// console.log(wrapper.emitted('increment'))
;[
[1], // 第一次调用时,`count` 为 1
[2] // 第二次调用时,`count` 为 2
]
断言复杂事件
假设我们的 <Counter>
组件现在需要触发一个包含附加信息的对象。例如,我们需要告诉任何监听 @increment
事件的父组件 count
是偶数还是奇数:
<!-- Counter.vue -->
<script setup>
import { ref } from 'vue'
const count = ref(0)
const emit = defineEmits(['increment'])
const handleClick = () => {
count.value += 1
emit('increment', {
count: count.value,
isEven: count.value % 2 === 0,
})
}
</script>
<template>
<button @click="handleClick">Increment</button>
</template>
正如我们之前所做的那样,我们需要在 <button>
元素上触发 click
事件。然后,我们使用 emitted('increment')
来确保触发了正确的值。
test('emits an event with count when clicked', () => {
const wrapper = mount(Counter)
wrapper.find('button').trigger('click')
wrapper.find('button').trigger('click')
// 我们“点击”了两次,因此 `increment` 的数组应该有两个值。
expect(wrapper.emitted('increment')).toHaveLength(2)
// 然后,我们可以确保 `wrapper.emitted('increment')` 的每个元素包含一个带有预期对象的数组。
expect(wrapper.emitted('increment')[0]).toEqual([
{
count: 1,
isEven: false
}
])
expect(wrapper.emitted('increment')[1]).toEqual([
{
count: 2,
isEven: true
}
])
})
测试复杂事件负载 (如对象) 与测试简单值 (如数字或字符串) 没有区别。
结论
- 使用
emitted()
来访问从 Vue 组件触发的事件。 emitted(eventName)
返回一个数组,其中每个元素代表一个已触发的事件。- 参数存储在
emitted(eventName)[index]
中,按触发顺序以数组形式呈现。