如何在 vue 2 中测试组件根元素上的事件?

And*_*hiu 3 unit-testing vue.js vuejs2 vue-test-utils

我正在为以下组件编写单元测试:

<template>
  <sub-component
     @foo="bar"
  />
</template>
<script>
import SubComponent from './SubComponent';

export default {
  name: 'MyComponent',
  components: { SubComponent },
  methods: {
    bar(payload) {
       this.$emit('baz', ...payload);
    }
  }
}
</script>
Run Code Online (Sandbox Code Playgroud)

测试将是:

import { shallowMount } from '@vue/test-utils';
import _ from 'lodash';
import MyComponent from '../../components/MyComponent';

describe('MyComponent.vue', () => {
  let wrapper;

  beforeEach(() => {
    wrapper = shallowMount(MyComponent);
  });

  it('should emit baz on subcomponent foo', () => {
    const subComp = wrapper.find('sub-component-stub');
    expect(subComp.exists()).toBe(true);          // passes

    subComp.vm.$emit('foo');

    return wrapper.vm.$nextTick().then(() => {
      expect(wrapper.emitted().baz).toBeTruthy(); // does not pass;

      // upon logging: 
      console.log(_.isEqual(wrapper, subComp));   // => true 
    })
  })
})
Run Code Online (Sandbox Code Playgroud)

该示例过于简单化,但这里的原则是我想要一个可重用的<sub-component>(模态)和围绕它的各种功能包装器(与模态类型执行的一项特定任务相关),这些包装器映射附加功能。我不想要父组件中的功能,因为它会违反 DRY - 我必须将其放置在包含特定类型模态的每个组件中。

<sub-component>如果不是 的直接子代,这会很好地工作<template>。不知何故,它出现wrappersubComp托管在同一个元素上。

应该如何正确测试?

mic*_*elw 5

另一种可能性是在 dom 中找到您的元素并检查根组件的发出值。

import { shallowMount } from '@vue/test-utils'
import MyComponent from './MyComponent.vue'
import SubComponent from './SubComponent.vue'

describe('MyComponent', () => {    

  it('should emit baz on subcomponent foo', () => {
    const wrapper = shallowMount(MyComponent)
    const subComponent = wrapper.find(SubComponent)

    expect(subComponent.exists()).toBe(true)
    expect(wrapper.emitted('baz')).toBeUndefined()

    subComponent.vm.$emit('foo', ['hello'])
    expect(wrapper.emitted('baz')[0]).toEqual(['hello'])
    // or expect(wrapper).toEmit('baz', 'hello') cf. below for toEmit
  })

})
Run Code Online (Sandbox Code Playgroud)

如果您想要 Jest 的自定义匹配器:

toEmit(received, eventName, data) {
  if (data) {
    expect(received.emitted()[eventName][0]).toEqual([data])
  } else {
    expect(received.emitted()[eventName][0]).toEqual([])
  }
  return { pass: true }
} 
Run Code Online (Sandbox Code Playgroud)