Vue2 Dialog弹窗函数式调用实践示例

 更新时间:2023年01月15日 16:12:41   作者:xekin  
这篇文章主要为大家介绍了Vue2 Dialog弹窗函数式调用实践示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

前言

Dialog 对话框组件几乎是每个前端项目必不可少的组件,通常是在保留当前页面状态并屏蔽其他用户输入的情况下,与用户交互并承载相关操作。

在 BOM 的方法中,alertprompt 都是以前用来做类似功能的方法,但是这些浏览器内置方法会完全停止网页代码执行,对于贫乏的前端线程资源来说实在是过于僵硬。于是便产生了各种各样的 Js 弹窗。

今天就来谈谈在前端框架 Vue 2 版本中的弹窗组件的相关实现以及我个人认为的最佳实践。

Vue2 的弹窗常用的使用方式

对于 Vue2 的 UI 框架,坊间比较火的有 element-ui antd-vue vant-ui 等等,不过他们在弹窗的使用方法上几乎都是一致的。下面我们以 element-ui 中的组件为例讲讲这些使用方式的缺点

第一种:将弹窗写在上下文中

这种方式不用多讲,使用起来肯定是最麻烦的一种,这意味着你每次想使用这个弹窗组件都要写一遍负责显隐的状态以及方法,尽量只在唯一场景里使用,不具备复用条件。

第二种:原型上注入全局方法

即是通过往 Vue.prototype 中注入弹窗的方法,使你可以在 vue 上下文中使用 this.$confirm 诸如此类的方法来使用他们的弹窗功能。

这种调用方式曾经几乎是所有 vue2 全局 api 的解决方法,一时间各种插件都有了各种各样的全局 api。例如 vuex 的 this.$store、 EventBus 的 this.$bus 等等。

这种方式在使用上虽然非常方便,但也有如下一些缺点

上下文污染,一个 vue 的全局 this 里面,什么东西都可以有,不论是全局方法、还是全局变量,都可以放在 Vue.prototype 里面,例如前一个例子,如果复杂的弹窗需要在各种其他前端文件内打开,大概率也是用这种方式将弹窗打开和关闭方法都注入到全局 this 中使用。

类型丢失,无论你是全局状态管理还是事件总线,在使用的时候都是各种黑盒,全局方法来自于什么插件、需要传什么参数、返回什么东西,全然无感,只能想办法寻找蛛丝马迹去溯源。

只能在 vue 上下文中使用,在 vue 文件外就无法使用了。就好比 React 无法在 React 上下文之外 setState 一样难受。

第三种:通过依赖注入的方式

这种方式和第二种几乎一样,通过顶层组件 provider 出对应的函数方法,之后在子组件中 inject 进来。 一样存在着 provider 的上下文污染,也丢失了类型,并且依旧只能在 vue 上下文中使用,灵活性一样差。

合理的使用方式

除了上述所说的问题之外

综合以上的缺点,我认为一个弹窗最佳的使用方式在 Vue2 中使用方式是这样:

    <template>
        <button @click="open">打开弹窗</button>
    </template>
    <script>
        import openDialog from 'my-dialog'
        export default {
            methods: {
                open() {
                    openDialog()
                        .then(() => {})
                        .catch(() => {})
                        .finally(() => {})
                }
            }
        }
    </script>

也可以在其他文件中,例如:

    // api.ts
    import openDialog from 'my-dialog'
    const getUserInfo = () => {
        return fetch('/xxx/api').then(() => {
            openDialog('success')
        })
    }

功能实现

废话不多说,直接上核心代码

// dialog.ts
import Vue, { ComponentInstance } from "vue";
import ConfirmDialog from "./confirm-dialog.vue";
export let index = 1000;
export const cache = new Set<string>();
export function openDialog(component: ComponentInstance) {
	const div = document.createElement("div");
	const el = document.createElement("div");
	const id = 'dialog-' + Math.random();
	div.appendChild(el);
	document.body.appendChild(div);
	const ComponentConstructor = Vue.extend(component);
	return (propsData = {}, parent = undefined) => {
		let instance = new ComponentConstructor({
			propsData,
			parent,
		}).$mount(el);
		const destroyDialog = () => {
			if (cache.has(id)) return;
			if (instance && div.parentNode) {
				cache.add(id);
				(instance as any).visible = false;
                // 延时是为了在关闭动画执行完毕后卸载组件
				setTimeout(() => {
					cache.delete(id);
					instance.$destroy();
					// @ts-ignore
					instance = null;
					div.parentNode && div.parentNode.removeChild(div);
				}, 1000);
			}
		};
		// visible控制
		if ((instance as any).visible !== undefined) {
			// 支持sync/v-model
			instance.$watch("visible", (val) => {
				!val && destroyDialog();
			});
			Vue.nextTick(() => ((instance as any).visible = true));
		}
		return new Promise((resolve, reject) => {
			// emit 一个 done 事件关闭
			instance.$once("done", (data: any) => {
				destroyDialog();
				resolve(data);
			});
			// emit 一个 cancel 事件取消
			instance.$once("cancel", (data: any) => {
				destroyDialog();
				reject(data);
			});
		});
	};
}
export function confirmDialog(content: string, editable?: boolean) {
	return openDialog(ConfirmDialog as any)({ content, editable });
}

使用方式和结果可以通过下面这个例子进行查看

stackblitz.com/edit/vitejs…

在例子中,我们可以将任意 vue 组件包装进 openDialog Api 中,只需要在组件的 data 里写入一个 visible 属性,而后续在组件方法中调用 this.$emits('done') 或者 this.$emits('cancel') 就可以对应 openDialog 方法返回 Promise 的 then 回调和 catch 回调。

可以自定义传参、自定义内容、自定义事件,自定义返回,灵活性直接拉满。

而且,方法不限于任何上下文,在任何文件内都可以使用,实现了真正的函数式调用 Vue2 的弹窗组件。

但是这个方法确实也有那么一点点不方便的地方,如果有掘友看得出来,可以在评论下方说说。

结语

这篇文章其实没多少东西,甚至代码都是我一年多前就写的,在受到 React hooks 的启发后,我就觉得函数式编程非常的爽,就尝试将项目中的全局变量都剥离 vue 上下文,包括 dialogmessage 组件,之后摒弃 vuexpinia 这种强绑上下文的状态管理库,改用 Vue.observable,就可以很方便的将业务与 UI 完全抽离。所有变量和方法都可以通过 import 追溯,更重要的是这种模式更契合 typescript,类型提示也很轻易跟上来了。

以上就是本篇文章的所有内容了,后续我会封装 v2 和 v3 的函数式弹窗组件,并发布到 npm 上,到时候再更新链接到文章里,更多关于Vue2 Dialog弹窗函数式调用的资料请关注脚本之家其它相关文章!

相关文章

  • 解决vue 退出动画无效的问题

    解决vue 退出动画无效的问题

    这篇文章主要介绍了解决vue 退出动画无效的问题,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-08-08
  • 传说中VUE的语法糖到底是做什么的

    传说中VUE的语法糖到底是做什么的

    从接触Vue我们就知道 v-model是实现数据双向绑定的那他能实现绑定的原理到底是啥?最常见的语法糖 v-model,今天通过案例给大家详细介绍下,需要的朋友参考下吧
    2021-09-09
  • Vue3 响应式高阶用法之triggerRef()的使用

    Vue3 响应式高阶用法之triggerRef()的使用

    在Vue3响应式系统中,shallowRef仅追踪顶层属性的变化,当需要对内层属性作出反应时,可使用triggerRef()方法手动触发更新,本文介绍了triggerRef()的应用场景、基本用法、功能和最佳实践,感兴趣的可以了解一下
    2024-09-09
  • Pinia 的 Setup Stores 语法使用实例详解

    Pinia 的 Setup Stores 语法使用实例详解

    这篇文章主要为大家介绍了Pinia 的 Setup Stores 语法使用实例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-09-09
  • vue使用socket与服务端进行通信的代码详解

    vue使用socket与服务端进行通信的代码详解

    这篇文章主要给大家介绍了vue如何使用socket与服务端进行通信的相关资料,在Vue中我们可以将Websocket类封装成一个Vue插件,以便全局使用,需要的朋友可以参考下
    2023-09-09
  • Vue实现todo应用的示例

    Vue实现todo应用的示例

    这篇文章主要介绍了Vue实现todo应用的示例,帮助大家更好的理解和学习使用vue框架,感兴趣的朋友可以了解下
    2021-02-02
  • vue axios post发送复杂对象问题

    vue axios post发送复杂对象问题

    现在vue项目中,一般使用axios发送请求去后台拉取数据。这篇文章主要介绍了vue axios post发送复杂对象的一点思考,需要的朋友可以参考下
    2019-06-06
  • Vue3中实现歌词滚动显示效果

    Vue3中实现歌词滚动显示效果

    本文分享如何在Vue 3中实现一个简单的歌词滚动效果,我将从歌词数据的处理开始,一步步介绍布局的搭建和事件的实现,感兴趣的朋友跟随小编一起看看吧
    2024-02-02
  • 详解让sublime text3支持Vue语法高亮显示的示例

    详解让sublime text3支持Vue语法高亮显示的示例

    本篇文章主要介绍了让sublime text3支持Vue语法高亮显示的示例,非常具有实用价值,需要的朋友可以参考下
    2017-09-09
  • 自带气泡提示的vue校验插件(vue-verify-pop)

    自带气泡提示的vue校验插件(vue-verify-pop)

    这篇文章主要为大家详细介绍了自带气泡提示的vue校验插件,vue-verify-pop的使用方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-04-04

最新评论