你好,我是大圣。
上一讲我们剖析了表单组件的实现模式,相信学完之后,你已经掌握了表单类型组件设计的细节,表单组件的主要功能就是在页面上获取用户的输入。
不过,用户在交互完成之后,还需要知道交互的结果状态,这就需要我们提供专门用来反馈操作状态的组件。这类组件根据反馈的级别不同,也分成了很多种类型,比如全屏灰色遮罩、居中显示的对话框Dialog,在交互按钮侧面显示、用来做简单提示的tooltip,以及右上角显示信息的通知组件Notification等,这类组件的交互体验你都可以在Element3官网感受。
今天的代码也会用Element3的Dialog组件和Notification进行举例,在动手写代码实现之前,我们先从这个弹窗组件的需求开始说起。
我们先来设计一下要做的组件,通过这部分内容,还可以帮你继续加深一下对单元测试Jest框架的使用熟练度。我建议你在设计一个新的组件的时候,也试试采用这种方式,先把组件所有的功能都罗列出来,分析清楚需求再具体实现,这样能够让你后面的工作事半功倍。
首先无论是对话框Dialog,还是消息弹窗Notification,它们都由一个弹窗的标题,以及具体的弹窗的内容组成的。我们希望弹窗有一个关闭的按钮,点击之后就可以关闭弹窗,弹窗关闭之后还可以设置回调函数。
下面这段代码演示了dialog组件的使用方法,通过title显示标题,通过slot显示文本内容和交互按钮,而通过v-model就能控制显示状态。
<el-dialog
title="提示"
:visible.sync="dialogVisible"
width="30%"
v-model:visible="dialogVisible"
>
<span>这是一段信息</span>
<template #footer>
<span class="dialog-footer">
<el-button @click="dialogVisible = false">取 消</el-button>
<el-button type="primary" @click="dialogVisible = false">确 定</el-button>
</span>
</template>
</el-dialog>
这类组件实现起来和表单类组件区别不是特别大,我们首先需要做的就是控制好组件的数据传递,并且使用Teleport渲染到页面顶层的body标签。
像Dialog和Notification类的组件,我们只是单纯想显示一个提示或者报错信息,过几秒就删除,如果在每个组件内部都需要写一个<Dialog v-if>,并且使用v-if绑定变量的方式控制显示就会显得很冗余。
所以,这里就要用到一种调用Vue组件的新方式:我们可以使用JavaScript的API动态地创建和渲染Vue的组件。具体如何实现呢?我们以Notification组件为例一起看一下。
下面的代码是Element3的Notification演示代码。组件内部只有两个button,我们不需要书写额外的组件标签,只需要在<script setup>中使用Notification.success函数,就会在页面动态创建Notification组件,并且显示在页面右上角。
<template>
<el-button plain @click="open1"> 成功 </el-button>
<el-button plain @click="open2"> 警告 </el-button>
</template>
<script setup>
import { Notification } from 'element3'
function open1() {
Notification.success({
title: '成功',
message: '这是一条成功的提示消息',
type: 'success'
})
}
function open2() {
Notification.warning({
title: '警告',
message: '这是一条警告的提示消息',
type: 'warning'
})
}
</script>
分析完需求之后,我们借助单元测试的方法来实现这个弹窗组件(单元测试的内容如果记不清了,你可以回顾第20讲)。
我们依次来分析Notification的代码,相比于写Demo逻辑的代码,这次我们体验一下实际的组件和演示组件的区别。我们来到element3下面的src/components/Notification/notifucation.vue代码中,下面的代码构成了组件的主体框架,我们不去直接写组件的逻辑,而是先从测试代码来梳理组件的功能。
<template>
<div class="el-nofication">
<slot />
</div>
</template>
<script>
</script>
<style lang="scss">
@import '../styles/mixin';
</style>
结合下面的代码可以看到,我们进入到了内部文件Notification.spec.js中。下面的测试代码中,我们期待Notification组件能够渲染el-notification样式类,并且内部能够通过属性title渲染标题;message属性用来渲染消息主体;position用来渲染组件的位置,让我们的弹窗组件可以显示在浏览器四个角。
import Notification from "./Notification.vue"
import { mount } from "@vue/test-utils"
describe("Notification", () => {
it('渲染标题title', () => {
const title = 'this is a title'
const wrapper = mount(Notification, {
props: {
title
}
})
expect(wrapper.get('.el-notification__title').text()).toContain(title)
})
it('信息message渲染', () => {
const message = 'this is a message'
const wrapper = mount(Notification, {
props: {
message
}
})
expect(wrapper.get('.el-notification__content').text()).toContain(message)
})
it('位置渲染', () => {
const position = 'bottom-right'
const wrapper = mount(Notification, {
props: {
position
}
})
expect(wrapper.find('.el-notification').classes()).toContain('right')
expect(wrapper.vm.verticalProperty).toBe('bottom')
expect(wrapper.find('.el-notification').element.style.bottom).toBe('0px')
})
it('位置偏移', () => {
const verticalOffset = 50
const wrapper = mount(Notification, {
props: {
verticalOffset
}
})
expect(wrapper.vm.verticalProperty).toBe('top')
expect(wrapper.find('.el-notification').element.style.top).toBe(
`${verticalOffset}px`
)
})
})
这时候毫无疑问,测试窗口会报错。我们需要进入notificatin.vue中实现代码逻辑。
下面的代码中,我们在代码中接收title、message和position,使用notification__title和notification__message渲染标题和消息。
<template>
<div class="el-notification" :style="positionStyle" @click="onClickHandler">
<div class="el-notification__title">
{{ title }}
</div>
<div class="el-notification__message">
{{ message }}
</div>
<button
v-if="showClose"
class="el-notification__close-button"
@click="onCloseHandler"
></button>
</div>
</template>
<script setup>
const instance = getCurrentInstance()
const visible = ref(true)
const verticalOffsetVal = ref(props.verticalOffset)
const typeClass = computed(() => {
return props.type ? `el-icon-${props.type}` : ''
})
const horizontalClass = computed(() => {
return props.position.endsWith('right') ? 'right' : 'left'
})
const verticalProperty = computed(() => {
return props.position.startsWith('top') ? 'top' : 'bottom'
})
const positionStyle = computed(() => {
return {
[verticalProperty.value]: `${verticalOffsetVal.value}px`
}
})
</script>
<style lang="scss">
.el-notification {
position: fixed;
right: 10px;
top: 50px;
width: 330px;
padding: 14px 26px 14px 13px;
border-radius: 8px;
border: 1px solid #ebeef5;
background-color: #fff;
box-shadow: 0 2px 12px 0 rgba(0, 0, 0, 0.1);
overflow: hidden;
}
</style>
然后我们新增测试代码,设置弹窗是否显示关闭按钮以及关闭弹窗之后的回调函数。我们希望点击关闭按钮之后,就能够正确执行传入的onClose函数。
it('set the showClose ', () => {
const showClose = true
const wrapper = mount(Notification, {
props: {
showClose
}
})
expect(wrapper.find('.el-notification__closeBtn').exists()).toBe(true)
expect(wrapper.find('.el-icon-close').exists()).toBe(true)
})
it('点击关闭按钮', async () => {
const showClose = true
const wrapper = mount(Notification, {
props: {
showClose
}
})
const closeBtn = wrapper.get('.el-notification__closeBtn')
await closeBtn.trigger('click')
expect(wrapper.get('.el-notification').isVisible()).toBe(false)
})
it('持续时间之后自动管理', async () => {
jest.useFakeTimers()
const wrapper = mount(Notification, {
props: {
duration: 1000
}
})
jest.runTimersToTime(1000)
await flushPromises()
expect(wrapper.get('.el-notification').isVisible()).toBe(false)
})
到这里,Notification组件测试的主体逻辑就实现完毕了,我们拥有了一个能够显示在右上角的组件,具体效果你可以参考后面这张截图。
进行到这里,距离完成整体设计我们还差两个步骤。
首先,弹窗类的组件都需要直接渲染在body标签下面,弹窗类组件由于布局都是绝对定位,如果在组件内部渲染,组件的css属性(比如Transform)会影响弹窗组件的渲染样式,为了避免这种问题重复出现,弹窗组件Dialog、Notification都需要渲染在body内部。
Dialog组件可以直接使用Vue3自带的Teleport,很方便地渲染到body之上。在下面的代码中, 我们用teleport组件把dialog组件包裹之后,通过to属性把dialog渲染到body标签内部。
<teleport
:disabled="!appendToBody"
to="body"
>
<div class="el-dialog">
<div class="el-dialog__content">
<slot />
</div>
</div>
</teleport>
这时我们使用浏览器调试窗口,就可以看到Dialog标签已经从当前组件移动到了body标签内部,如下图所示。
但是Notification组件并不会在当前组件以组件的形式直接调用,我们需要像Element3一样,能够使用js函数动态创建Notification组件,给Vue的组件提供Javascript的动态渲染方法,这是弹窗类组件的特殊需求。
我们先把测试代码写好,具体如下。代码中分别测试函数创建组件,以及不同配置和样式的通知组件。
it('函数会创建组件', () => {
const instanceProxy = Notification('foo')
expect(instanceProxy.close).toBeTruthy()
})
it('默认配置 ', () => {
const instanceProxy = Notification('foo')
expect(instanceProxy.$props.position).toBe('top-right')
expect(instanceProxy.$props.message).toBe('foo')
expect(instanceProxy.$props.duration).toBe(4500)
expect(instanceProxy.$props.verticalOffset).toBe(16)
})
test('字符串信息', () => {
const instanceProxy = Notification.info('foo')
expect(instanceProxy.$props.type).toBe('info')
expect(instanceProxy.$props.message).toBe('foo')
})
test('成功信息', () => {
const instanceProxy = Notification.success('foo')
expect(instanceProxy.$props.type).toBe('success')
expect(instanceProxy.$props.message).toBe('foo')
})
现在测试写完后还是会报错,因为现在Notification函数还没有定义,我们要能通过Notification函数动态地创建Vue的组件,而不是在template中使用组件。
在JSX那一讲中我们讲过,template的本质就是使用h函数创建虚拟Dom,如果我们自己想动态创建组件时,使用相同的方式即可。
在下面的代码中我们使用Notification函数去执行createComponent函数,使用h函数动态创建组件,实现了动态组件的创建。
function createComponent(Component, props, children) {
const vnode = h(Component, { ...props, ref: MOUNT_COMPONENT_REF }, children)
const container = document.createElement('div')
vnode[COMPONENT_CONTAINER_SYMBOL] = container
render(vnode, container)
return vnode.component
}
export function Notification(options) {
return createNotification(mergeProps(options))
}
function createNotification(options) {
const instance = createNotificationByOpts(options)
setZIndex(instance)
addToBody(instance)
return instance.proxy
}
创建组件后,由于Notification组件同时可能会出现多个弹窗,所以我们需要使用数组来管理通知组件的每一个实例,每一个弹窗的实例都存储在数组中进行管理。
下面的代码里,我演示了怎样用数组管理弹窗的实例。Notification函数最终会暴露给用户使用,在Notification函数内部我们通过createComponent函数创建渲染的容器,然后通过createNotification创建弹窗组件的实例,并且维护在instanceList中。
const instanceList = []
function createNotification(options) {
...
addInstance(instance)
return instance.proxy
}
function addInstance(instance) {
instanceList.push(instance)
}
;['success', 'warning', 'info', 'error'].forEach((type) => {
Notification[type] = (options) => {
if (typeof options === 'string' || isVNode(options)) {
options = {
message: options
}
}
options.type = type
return Notification(options)
}
})
// 有了instanceList, 可以很方便的关闭所有信息弹窗
Notification.closeAll = () => {
instanceList.forEach((instance) => {
instance.proxy.close()
removeInstance(instance)
})
}
最后,我带你简单回顾下我们都做了什么。在正式动手实现弹窗组件前,我们分析了弹窗类组件的风格。弹窗类组件主要负责用户交互的反馈。根据显示的级别不同,它可以划分成不同的种类:既有覆盖全屏的弹窗Dialog,也有负责提示消息的Notification。
这些组件除了负责渲染传递的数据和方法之外,还需要能够脱离当前组件进行渲染,防止当前组件的css样式影响布局。因此Notification组件需要渲染到body标签内部,而Vue提供了Teleport组件来完成这个任务,我们通过Teleport组件就能把内部的组件渲染到指定的dom标签。
之后,我们需要给组件提供JavaScript调用的方法。我们可以使用Notification()的方式动态创建组件,利用createNotification即可动态创建Vue组件的实例。
对于弹窗组件来说可以这样操作:首先通过createNotification函数创建弹窗的实例,并且给每个弹窗设置好唯一的id属性,然后存储在数组中进行管理。接着,我们通过对createNotification函数返回值的管理,即可实现弹窗动态的渲染、更新和删除功能。
正文里已经详细讲解和演示了弹窗组件的设计,所以今天的总结我想变个花样,再给你说说TDD的事儿。
很多同学会觉得写测试代码要花一定成本,有畏难心理,觉得自己不太会写测试,这些“假想”给我们造成了“TDD很难实施”的错觉。实际上入门TDD并没有这么难。按照我的实践经验来看,先学会怎么写测试,再学习怎么重构,基本上就可以入门写TDD了。
就拿我们这讲的实践来说,我们再次应用了测试驱动开发这个方式来实现弹窗组件,把整体需求拆分成一个个子任务,逐个击破。根据设计的需求写好测试代码之后,测试代码就会检查我们的业务逻辑有没有实现,指导我们做相应的修改。
咱们的实践过程抽象出来,一共包括四个步骤:写测试 -> 运行测试(报错) -> 写代码让测试通过 -> 重构的方式。这样的开发模式,今后你在设计组件库时也可以借鉴,不但有助于提高代码的质量和可维护性,还能让代码有比较高的代码测试覆盖率。
最后留一个思考题,现在我们设计的Notification组件的message只能支持文本消息,如果想支持传入其他组件,应该如何实现?
欢迎你在评论去分享你的答案,也欢迎你把这一讲的内容分享给你的同事和朋友们,我们下一讲再见。
评论