Skip to content

何时测试组件?

一般业务组件是不写测试的, 只有在写组件库(如: element-plus) 的时候才会写单元测试,

原因是用数据驱动去驱动UI, 只要数据没有问题, 那么UI操作逻辑肯定没有问题, 但是这也要求我们将逻辑抽离出视图, 只在视图中调用

逻辑没问题了, 那其他经常改动的就只有样式了, 用单元测试去测试样式还不如直接手动测试

但是 UI 组件库中的组件样式一般是不会随便改动的, 且功能也相对固定

如何测试组件?

如何将组件挂载到内存中?

如何模拟用户操作事件?

测试例子

ts
import XButton from './index.vue';
import { mount } from '@vue/test-utils';

describe('XButton', () => {
  it('render', () => {
    const wrapper = mount(XButton);
    expect(wrapper.classes()).toContain('xbtn');
    expect(wrapper.classes()).toContain('default');
    expect(wrapper.text()).toContain('button');
  });

  it('should be set text when set props text field', () => {
    const text = 'clickMe';
    const wrapper = mount(XButton, { props: { text } });
    expect(wrapper.text()).toContain(text);
  });

  it('should be add color className when set props color field', () => {
    const wrapper = mount(XButton, {
      props: {
        color: 'info',
      },
    });

    expect(wrapper.classes()).toContain('info');
  });

  it('should be trigger click event', () => {
    const onClick = vi.fn();
    const wrapper = mount(XButton, {
      props: {
        onClick,
      },
    });

    wrapper.find('button').trigger('click');
    expect(onClick).toBeCalled();
  });

  it('should be not trigger click event when disabled is true', () => {
    const onClick = vi.fn();
    const wrapper = mount(XButton, {
      props: {
        onClick,
        disabled: true,
      },
    });

    wrapper.find('button').trigger('click');
    expect(onClick).not.toBeCalled();
  });
});
vue
<template>
  <button :class="classes" :disabled="props.disabled" @click="handleClick">
    {{ props.text }}
  </button>
</template>

<script lang="ts" setup>
import { computed } from 'vue';
import { buttonProps, buttonEmits, COMPONENT_NAME, useButton } from './button';

defineOptions({
  name: COMPONENT_NAME,
});

const props = defineProps(buttonProps);
const emits = defineEmits(buttonEmits);

const classes = computed(() => {
  let classNames = ['xbtn'];
  if (props.color === 'default') {
    classNames.push('default');
  } else {
    classNames.push(props.color);
  }
  return classNames;
});

/* @ts-ignore */
const { handleClick } = useButton(props, emits);
</script>

<style lang="scss" scoped>
.xbtn {
  margin: 0;
  border: none;
  padding: 10px 20px;
  display: inline-flex;
  color: #f8f8f8;
  &.default {
    background: #c8c9cc;
    color: #909399;
  }
  &.primary {
    background-color: #409eff;
  }
  &.info {
    background-color: #b1b3b8;
  }
  &.dadnger {
    background-color: #f56c6c;
  }
  &.success {
    background-color: #67c23a;
  }
  &.warning {
    background-color: #e6a23c;
  }
}
</style>
ts
import { ExtractPropTypes, SetupContext } from 'vue';

export const COMPONENT_NAME = 'XButton';

const buttonColors = ['default', 'primary', 'success', 'info', 'danger', 'warning'];

export const buttonProps = {
  text: {
    type: String,
    default: 'button',
  },

  color: {
    type: String,
    values: buttonColors,
    default: 'default',
  },

  disabled: Boolean,
} as const;

export const buttonEmits = {
  click: (e: MouseEvent) => e instanceof MouseEvent,
};

export function useButton(props: ButtonPropsType, emit: SetupContext<ButtonEmitsType>['emit']) {
  function handleClick(e: MouseEvent) {
    if (props.disabled) {
      return;
    }
    emit('click', e);
  }

  return {
    handleClick,
  };
}

// types
export type ButtonPropsType = ExtractPropTypes<typeof buttonProps>;
export type ButtonEmitsType = ExtractPropTypes<typeof buttonProps>;

Released under the MIT License.