技术分享 | 弹窗开发中,如何使用 Hook 封装 el-dialog?
弹窗是前端开发中的一种常见需求。Element UI 框架中的 el-dialog
组件提供了弹窗相关的基本功能,但在实际开发中,我们难免会遇到一些定制化需求,比如对弹窗进行二次封装以便在项目中统一管理样式和行为。
本文将分享如何使用 useDialog
Hook 封装 el-dialog
,实现更灵活、更易用的弹窗组件。
一、问题澄清
「将一个通用的组件应用在多个页面」是一个很常见的实际场景。
举个例子:以购买应用程序为例,用户可能在付费页面进行购买操作,也可能在浏览其他页面时触发购买需求,此时就需要弹出对话框引导用户完成购买行为。
为了实现这一功能,过去通常会采用以下步骤:
- 封装购买组件:首先创建一个通用的购买组件,以便在不同页面和场景下复用。
- 在付费页面渲染购买组件:将购买组件直接嵌到付费页面中。
- 在其他页面使用
el-dialog
展示购买组件:在其他页面通过el-dialog
控制组件的显示,利用visible
状态变量(通常是一个ref
响应式变量)动态控制对话框的弹出与关闭。
虽然这种方式可以满足功能需求,但随着该组件被越来越多的页面和功能所使用,维护也会愈加复杂繁琐——每增加一个使用页面,都必须重复编写控制显示/隐藏的逻辑代码。
那么,有没有更好的方法可以简化这个过程?是否可以通过某种方式,用一个单独的函数全局控制购买组件的打开和关闭,从而减少代码重复,降低维护成本?
二、关于 useDialog Hook
在 Vue 中,Hook 允许在函数式组件或者 API 中「钩入」Vue 特性。它们通常在组合式 API(Composition API)中使用,这是 Vue 提供的一套响应式和可复用逻辑功能的集合。
本文提到的 useDialog
Hook 就是一个封装了 el-dialog
组件基本功能的自定义 Hook,它还可以提供附加特性以便在项目中管理和展示弹窗。
三、实现 useDialog Hook
useDialog
Hook 需要达成以下目标:
- 满足基础用法,传入
el-dialog
的基础属性以及默认slot显示的内容,导出openDialog
和closeDialog
函数; - 支持
el-dialog
的事件配置; - 支持默认
slot
组件的属性配置; - 支持
el-dialog
其他 slot 配置,例如header
和footer
等; - 在内容组件中抛出特定事件支持关闭 dialog;
- 支持显示内容为
jsx
、普通文本
、Vue Component
; - 支持在显示内容中控制是否可以关闭的回调函数,例如
beforeClose
; - 支持显示之前钩子,例如
onBeforeOpen
; - 支持定义和弹出时修改配置属性;
- 支持继承 root vue 的 prototype,可以使用如
vue-i18n
的$t
函数; - 支持
ts
参数提示;
(一)准备 useDialog.ts
文件实现类型定义
import type { Ref } from 'vue' import { h, render } from 'vue' import { ElDialog } from 'element-plus' import type { ComponentInternalInstance, } from '@vue/runtime-core' type Content = Parameters<typeof h>[0] | string | JSX.Element // 使用 InstanceType 获取 ElDialog 组件实例的类型 type ElDialogInstance = InstanceType<typeof ElDialog> // 从组件实例中提取 Props 类型 type DialogProps = ElDialogInstance['$props'] & { } interface ElDialogSlots { header?: (...args: any[]) => Content footer?: (...args: any[]) => Content } interface Options<P> { dialogProps?: DialogProps dialogSlots?: ElDialogSlots contentProps?: P }
(二)实现普通 useDialog
函数
下面的函数实现了含目标 1、2、3、4、6 和 11 在内的基础用法。
目标 1:满足基础用法,传入
el-dialog
基础属性及默认 slot 显示的内容,导出openDialog
和closeDialog
函数;
目标 2:支持el-dialog
的事件配置;
目标 3.:支持默认slot
组件的属性配置;
目标 4:支持el-dialog
其他 slot 配置,如header
和footer
等;
目标 6:支持显示内容为jsx
、普通文本
、Vue Component
;
目标 11:支持ts
参数提示;
export function useDialog<P = any>(content: Content, options?: Ref<Options<P>> | Options<P>) { let dialogInstance: ComponentInternalInstance | null = null let fragment: Element | null = null // 关闭并卸载组件 const closeAfter = () => { if (fragment) { render(null, fragment as unknown as Element) // 卸载组件 fragment.textContent = '' // 清空文档片段 fragment = null } dialogInstance = null } function closeDialog() { if (dialogInstance) dialogInstance.props.modelValue = false } // 创建并挂载组件 function openDialog() { if (dialogInstance) { closeDialog() closeAfter() } const { dialogProps, contentProps } = options fragment = document.createDocumentFragment() as unknown as Element const vNode = h(ElDialog, { ...dialogProps, modelValue: true, onClosed: () => { dialogProps?.onClosed?.() closeAfter() }, }, { default: () => [typeof content === 'string' ? content : h(content as any, { ...contentProps, })], ...options.dialogSlots, }) render(vNode, fragment) dialogInstance = vNode.component document.body.appendChild(fragment) } onUnmounted(() => { closeDialog() }) return { openDialog, closeDialog } }
(三)实现目标 5
目标 5:在内容组件中抛出特定事件支持关闭 dialog;
- 在定义中支持
closeEventName
;
interface Options<P> { // ... closeEventName?: string // 新增的属性 }
- 修改
useDialog
函数接收closeEventName
事件关闭 dialog。
export function useDialog<P = any>(content: Content, options?: Ref<Options<P>> | Options<P>) { // ... // 创建并挂载组件 function openDialog() { // ... fragment = document.createDocumentFragment() as unknown as Element // 转换closeEventName事件 const closeEventName = `on${upperFirst(_options?.closeEventName || 'closeDialog')}` const vNode = h(ElDialog, { // ... }, { default: () => [typeof content === 'string' ? content : h(content as any, { ...contentProps, [closeEventName]: closeDialog, // 监听自定义关闭事件,并执行关闭 })], ...options.dialogSlots, }) render(vNode, fragment) dialogInstance = vNode.component document.body.appendChild(fragment) } onUnmounted(() => { closeDialog() }) return { openDialog, closeDialog } }
(四)实现目标 7、8
目标 7:支持在显示内容中控制是否可以关闭的回调函数,例如
beforeClose
;
目标 8:支持显示之前钩子,例如onBeforeOpen
;
- 在定义中支持
onBeforeOpen
、beforeCloseDialog
默认传给内容组件,有组件调用设置;
type DialogProps = ElDialogInstance['$props'] & { onBeforeOpen?: () => boolean | void }
- 修改
useDialog
函数接收onBeforeOpen
事件并传递beforeCloseDialog
。
export function useDialog<P = any>(content: Content, options?: Ref<Options<P>> | Options<P>) { // ... // 创建并挂载组件 function openDialog() { // ... const { dialogProps, contentProps } = options // 调用before钩子,如果为false则不打开 if (dialogProps?.onBeforeOpen?.() === false) { return } // ... // 定义当前块关闭前钩子变量 let onBeforeClose: (() => Promise<boolean | void> | boolean | void) | null const vNode = h(ElDialog, { // ... beforeClose: async (done) => { // 配置`el-dialog`的关闭回调钩子函数 const result = await onBeforeClose?.() if (result === false) { return } done() }, onClosed: () => { dialogProps?.onClosed?.() closeAfter() // 关闭后回收当前变量 onBeforeClose = null }, }, { default: () => [typeof content === 'string' ? content : h(content as any, { // ... beforeCloseDialog: (fn: (() => boolean | void)) => { // 把`beforeCloseDialog`传递给`content`,当组件内部使用`props.beforeCloseDialog(fn)`时,会把fn传递给`onBeforeClose` onBeforeClose = fn }, })], ...options.dialogSlots, }) render(vNode, fragment) dialogInstance = vNode.component document.body.appendChild(fragment) } onUnmounted(() => { closeDialog() }) return { openDialog, closeDialog } }
(五)实现目标 9、10
目标 9:支持定义和弹出时修改配置属性;
目标 10:支持继承 root vue 的 prototype,可以使用例如vue-i18n
的$t
函数;
// 定义工具函数,获取计算属性的option function getOptions<P>(options?: Ref<Options<P>> | Options<P>) { if (!options) return {} return isRef(options) ? options.value : options } export function useDialog<P = any>(content: Content, options?: Ref<Options<P>> | Options<P>) { // ... // 获取当前组件实例,用于设置当前dialog的上下文,继承prototype const instance = getCurrentInstance() // 创建并挂载组件,新增`modifyOptions`参数 function openDialog(modifyOptions?: Partial<Options<P>>) { // ... const _options = getOptions(options) // 如果有修改,则合并options。替换之前的options变量为 _options if (modifyOptions) merge(_options, modifyOptions) // ... const vNode = h(ElDialog, { // ... }, { // ... }) // 设置当前的上下文为使用者的上下文 vNode.appContext = instance?.appContext || null render(vNode, fragment) dialogInstance = vNode.component document.body.appendChild(fragment) } onUnmounted(() => { closeDialog() }) return { openDialog, closeDialog } }
通过上面的封装使用 useDialog
Hook 后,需要弹窗时,只需要引入该 Hook 并调用 openDialog
方法,非常方便简洁。此外,这样的封装也会让后续修改弹窗逻辑变得更加方便,只需要在 useDialog
Hook 中修改,无需逐个重复编辑。
四、useDialog Hook 案例实操
下面,我们使用 useDialog
Hook 来解决开头提到的应用程序购买问题。
(一)创建 components/buy.vue
购买组件
<script lang="ts" setup> const props = defineProps({ from: { type: String, default: '', }, }) </script> <template> 我是购买组件 </template>
(二)在 pages/subscription.vue
页面中使用 buy.vue
购买组件
<script lang="ts" setup> import Buy from '@/components/buy.vue' </script> <template> <Buy from="subscription" /> </template>
(三)在其他功能页面中弹出 buy.vue
购买组件
<script lang="ts" setup> import { useDialog } from '@/hooks/useDialog' const Buy = defineAsyncComponent(() => import('@/components/buy.vue')) const { openDialog } = useDialog(Buy, { dialogProps: { // ... title: '购买' }, contentProps: { from: 'function', }, }) const onSomeClick = () => { openDialog() } </script>
拓展:useDialog Hook 的其他应用
beforeClose
& closeEventName
示例:buy.vue
购买组件
<script lang="ts" setup> const props = defineProps({ from: { type: String, default: '', }, beforeCloseDialog: { type: Function, default: () => true, }, }) const emit = defineEmits(['closeDialog']) props.beforeCloseDialog(() => { // 假如from 为 空字符串不能关闭 if (!props.from) { return false } return true }) // 关闭dialog const onBuySuccess = () => emit('closeDialog') </script>
<script lang="ts" setup> import { useDialog } from '@/hooks/useDialog' const Buy = defineAsyncComponent(() => import('@/components/buy.vue')) const { openDialog } = useDialog(Buy, { dialogProps: { // ... title: '购买' }, contentProps: { from: '', }, }) const onSomeClick = () => { openDialog() } </script>
总结
使用 useDialog
Hook 封装 el-dialog
可以让前端技术更加有趣简洁。笔者也希望大家能尝试这样的封装方式,让前端代码更加优雅且易于维护。
优秀的工程师就同优秀的厨师一样,掌握了精妙的烹饪和调味技巧,就能让每道菜都变得美味可口!
LigaAI 重视开发者文化的维护与构建,将继续分享更多技术分享和趣味技术实践。
欢迎关注 LigaAI 帐号,也期待您点击新一代智能研发协作平台,与我们展开更多交流。
助力开发者扬帆远航,LigaAI 期待与你一路同行!

低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
如何通过前端表格控件在10分钟内完成一张分组报表?
前言: 当今时代,报表作为信息化系统的重要组成部分,在日常的使用中发挥着关键作用。借助报表工具使得数据录入、分析和传递的过程被数字化和智能化,大大提高了数据的准确性及利用的高效性。而在此过程中,信息化系统能够实现对数据的实时监控和更新,为管理者提供及时、准确的业务数据,帮助他们做出更加合理的决策。 分组报表是业务系统中常见的一种类型,主要用于汇总和比较不同类别的数据。例如,分析公司不同区域的销售情况、分析特定时间段内的财务数据、集团旗下各个公司的业务数据等。 什么是分组报表? 分组报表是一种数据报表,其数据会根据特定的标准进行分类或分组,并在每个组别内进行统计或汇总。这些标准可以是日期、地区、产品类型等。通过分组数据,可以更清晰地了解数据的整体结构和趋势,以便做出更有针对性的决策。分组报表通常用于商业、财务、市场等领域,用于帮助分析人员理解和传达数据的含义。 分组报表的优点: **能够显著提高数据的可读性和可理解性:**通过将数据按照特定的分类标准进行分组,用户可以更加清晰地看到数据之间的关联和差异,从而更容易地识别出数据的模式和趋势。 **有助于用户快速定位关键信息:**在大量的数据...
- 下一篇
签约案例|GreptimeDB 助力国家电网数字换流站打造稳定高效的时序数据底座
电网体系作为现代社会运行的支柱之一,为各行各业、千家万户提供了电能的基本支持。从家庭到企业,医院到学校,交通到通讯,电力电网的应用贯穿始终。近年来,特高压换流站成为国家电网的重点建设工程,“十四五”期间,国家电网公司规划建设特高压工程“24 交 14 直”,涉及线路 3 万余公里,变电换流容量 3.4 亿千伏安,总投资 3800 亿元。 国家电网 2024 年工作会议中提出将继续加大数智化坚强电网的建设。数智化坚强电网是将数字化、智能化技术深入融合嵌入电网生产运行与管理运营过程的新型电网形态。 数智化的发展为国家电网对数据的使用提出了更高的要求。通过建设云端和站端时序数据库平台,能够高效提高时序数据使用效率,大幅降低使用成本,为国家电网数智化建设提供坚实的数据基础保障。 项目背景 数字换流站项目是国家电网数智化的重点项目。每个特高压换流站有数千个大中型智能设备,处理数十万个测点的毫秒级精度数据,每天产生了数亿行的时序数据集。 面对如此海量的时序数据写入、查询和分析管理需求,此前站端使用的 CeresDB,InfluxDB 或基于 InfluxDB 自研等时序数据库产品已无法满足需求。同...
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- Red5直播服务器,属于Java语言的直播服务器
- CentOS7,8上快速安装Gitea,搭建Git服务器
- CentOS6,7,8上安装Nginx,支持https2.0的开启
- CentOS8,CentOS7,CentOS6编译安装Redis5.0.7
- SpringBoot2更换Tomcat为Jetty,小型站点的福音
- Jdk安装(Linux,MacOS,Windows),包含三大操作系统的最全安装
- SpringBoot2整合MyBatis,连接MySql数据库做增删改查操作
- SpringBoot2全家桶,快速入门学习开发网站教程
- CentOS8安装MyCat,轻松搞定数据库的读写分离、垂直分库、水平分库
- CentOS8编译安装MySQL8.0.19