Vue3单元测试(给element的button组件写一个单元测试)

由于最近公司没有什么新的项目,最近vue3发布了Beta版,https://v3.vuejs.org/guide/introduction.html自己也是打算学习一下。不说冲大厂吧,学习新东西还是挺有趣的。

文末有彩蛋

闲话少说,进入我们今天主题vue的单元测试

说到vue的单元测试,我们在用vue-cli创建项目的时候,可以选择test选项,而test就是vue-cli询问我们需要单元测试吗,如此贴心,爱了爱了!

首先说一下,我们使用jest测试的几个命令小技巧

npm run test:unit button.spec.ts //命令后面跟文件名,表示单独执行这个测试文件

常用的方法

  • .classes() 拿到wrapper所包含的类名,返回一个Array<string>,如果传入一个类名(不包含.)的时候,则返回一个boolearn。
  • .find()传入一个类名,返回一个DOMWrapper
  • .exists()如果在空的wrapper或wrapperArray上调用,则返回false,反之则返回true。
  • .toBe()比如传入的是true,那么调用toBe()wrapper必须返回true,否则会报错
  • .toContain() 是否包含
  • .findComponent() 传入组件的name,得到一个VueWrapper
  • .text()返回wrapper的内容
  • emitted() //测试自定义事件的api,具体用法看下图
  • toBeFalsy() //当您不关心一个值是什么,只希望确保一个值在布尔上下文中为时使用。在JavaScript中,有6个假值:false、0、"、null、undefined和NaN。其他一切都是真实的。
  • toBeTruthy()当您不关心一个值是什么,只希望确保一个值在布尔上下文中为时使用。在JavaScript中,有6个假值:false、0、"、null、undefined和NaN。其他一切都是真实的。
  • isVisible()判断元素是否可见当你有 display: none, visibility: hidden, opacity :0 或者是具有hidden属性的元素都会返回false

话不多说,进入正题

我们测试的是element源码中的一个Button组件,这个组件是我从github上element源码上拷贝过来的,作为我们此次单元测试的组件
element Button.vue 源码

<template>
  <button
    class="el-button"
    @click="handleClick"
    :disabled="buttonDisabled || loading"
    :autofocus="autofocus"
    :type="nativeType"
    :class="[
      type ? 'el-button--' + type : '',
      buttonSize ? 'el-button--' + buttonSize : '',
      {
        'is-disabled': buttonDisabled,
        'is-loading': loading,
        'is-plain': plain,
        'is-round': round,
        'is-circle': circle,
      },
    ]"
  >
    <i class="el-icon-loading" v-if="loading"></i>
    <i :class="icon" v-if="icon && !loading"></i>
    <span v-if="$slots.default"><slot></slot></span>
  </button>
</template>
<script>
export default {
  name: 'ElButton',

  inject: {
    elForm: {
      default: '',
    },
    elFormItem: {
      default: '',
    },
  },

  props: {
    type: {
      type: String,
      default: 'default',
    },
    size: String,
    icon: {
      type: String,
      default: '',
    },
    nativeType: {
      type: String,
      default: 'button',
    },
    loading: Boolean,
    disabled: Boolean,
    plain: Boolean,
    autofocus: Boolean,
    round: Boolean,
    circle: Boolean,
  },

  computed: {
    _elFormItemSize() {
      return (this.elFormItem || {}).elFormItemSize;
    },
    buttonSize() {
      return this.size || this._elFormItemSize || (this.$ELEMENT || {}).size;
    },
    buttonDisabled() {
      return this.disabled || (this.elForm || {}).disabled;
    },
  },

  methods: {
    handleClick(evt) {
      this.$emit('click', evt);
    },
  },
};
</script>
测试loading属性

loading为true,类名为“.el-icon-loading”的这个i元素会被渲染
loading为true,组件会具备“is-loading”这个类名

下面我们看一下怎样实现这个测试?

  • .classes() 拿到wrapper所包含的类名,返回一个Array<string>,如果传入一个类名(不包含.)的时候,则返回一个boolearn。
  • .find()传入一个类名,返回一个DOMWrapper
  • .exists()如果在空的wrapper或wrapperArray上调用,则返回false,反之则返回true。
  • .toBe()比如传入的是true,那么调用toBe()wrapper必须返回true,否则会报错
import Button from "@/components/element/button.vue";
import { mount } from "@vue/test-utils";

it("loading", () => { 
    const wrapper = mount(Button, {
        props: {
            loading:true
        }
    })
    expect(wrapper.classes("is-loading")).toBe(true);//检查wrapper是否包含这个属性“is-loading”
    expect(wrapper.find('.el-icon-loading').exists()).toBe(true);//检查类名为“.el-icon-loading”的元素是否为渲染成功
})
测试slot

当我们调用这个组件,并且设置slot内容的时候,wrapper的text()是否包含我们设置的slot内容

下面我们看一下怎样实现这个测试?

  • .toContain() 是否包含
  • .findComponent() 传入组件的name,得到一个VueWrapper
  • .text()返回wrapper的内容
it('content', () => { 
    const camp = {
        template: '<div><Button>默认按钮</Button></div>'
    }
    const wraper = mount({
        template: '<div><Button>默认按钮</Button></div>'
    }, {
        global: {
            components: { Button }
        }
    });
    expect(wraper.findComponent({ name: 'ElButton' }).text()).toContain('默认按钮');//检查文本里面是否包含“默认按钮”这四个字。
})
测试emit事件
  • emitted() //测试自定义事件的api,具体用法看下图
  • toBeFalsy() //当您不关心一个值是什么,只希望确保一个值在布尔上下文中为时使用。在JavaScript中,有6个假值:false、0、"、null、undefined和NaN。其他一切都是真实的。
  • toBeTruthy()当您不关心一个值是什么,只希望确保一个值在布尔上下文中为时使用。在JavaScript中,有6个假值:false、0、"、null、undefined和NaN。其他一切都是真实的。
it("设置loading为true的时候,要阻止事件发射", () => {
    const wrapper = mount(Button, {
        props:
        {
            loading:true 
        }
    });
    wrapper.trigger("click");
    expect(wrapper.emitted().click).toBeFalsy();
})
// 设置elForm.disabled为true的时候阻止事件发射
it("设置elForm.disabled为true的时候也要阻止事件发射", () => {
    const wrapper = mount(Button, {
        global: {
            provide: {
                elForm: {
                    disabled:true
                }
            }
        }
    })
    wrapper.trigger("click")
    expect(wrapper.emitted().click).toBeFalsy()
})
// 正常点击的时候可以触发事件
it("初始点击按钮的时候,可以触发emit事件", () => { 
    const wrapper = mount(Button)
    wrapper.trigger("click")
    expect(wrapper.emitted().click).toBeTruthy()
})
测试size属性

上面我们测试了loading、slot、emit,这三个都是由代表性的,剩下就是普通的size、round这种影响组件样式的props,我们来随便写几个,全部的代码我会放在码云上面,大家有兴趣的话,可以自己拉下来看一下。

// 设置size,控制按钮的大小
it('设置size属性之后,按钮是具有对应的类名', () => { 
    const wrapper = mount(Button, {  
        props: {
          size:"large"
      }
  })
    expect(wrapper.classes("el-button--large")).toBe(true);
})

文末彩蛋:

看完了单元测试,是不是特别期待vue3的新功能呢,在下一篇文章,会给大家把这个button.vue使用vue3的手法重构,是不是特别期待呢,重构的时候,我也是遇到了好多坑,很多报错,到时候会把每个错误都详细记录一遍,我的文采和文字排版内容这俩方面不是特别擅长,但是我会尽力去给大家描述,大家就先凑活的看看,如果有描述不清楚的,私信我,我看到了都会回复的,谢谢大家!
克隆地址
https://gitee.com/DayLoveNight/vue3-test.git

下一篇文章,会使用vue3的语法重构button组件,和单元测试紧密相连,这里附上文章地址:https://www.jianshu.com/p/428c1257d8f8

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,117评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,328评论 1 293
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,839评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,007评论 0 206
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,384评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,629评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,880评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,593评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,313评论 1 243
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,575评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,066评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,392评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,052评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,082评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,844评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,662评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,575评论 2 270