vue3动态加载对话框的方法实例
简介
介绍使用vue3的异步组件动态管理对话框组件,简化对话框组件使用方式。本文使用的是vue3、typescript、element_plus完成的示例。
常规方式使用对话框
一般情况下,使用对话框组件,会使用v-model进行双向绑定,通过visible变量控制对话框的显示和关闭。常规方式有一个弊端,自定义组件中使用<el-dialog>,需要通过父组件控制自定义组件是否展示。
<template> <ElButton type="primary" @click="openGeneral()">常规方式打开Modal</ElButton> <el-dialog v-model="dialogVisible" title="Tips" width="30%" :before-close="handleClose" > <span>This is a message</span> <template #footer> <span class="dialog-footer"> <el-button @click="dialogVisible = false">Cancel</el-button> <el-button type="primary" @click="dialogVisible = false" >Confirm</el-button > </span> </template> </el-dialog> </template>
<script setup lang="ts"> // This starter template is using Vue 3 <script setup> SFCs // Check out https://v3.vuejs.org/api/sfc-script-setup.html#sfc-script-setup import {ref } from 'vue'; const dialogVisible = ref(false) const openGeneral=()=>{ dialogVisible.value=true } </script>
异步动态加载
先看使用异步组件进行动态加载对话框的方式。异步组件使用到的是defineAsyncComponent接口,只有使用到这个组件,才会从网络上加载。动态操作使用的useDzModal
使用方式
<template> <ElButton type="primary" @click="openTestModalAsync()">动态异步打开TestModal</ElButton> </template>
<script setup lang="ts"> import { defineAsyncComponent,ref } from 'vue'; import { ElMessageBox } from 'element-plus'; import { useDzModal } from './dzmodal' // 异步加载组件 const TestModalAsync = defineAsyncComponent(()=>import('./components/TestModal.vue')) const dzmodal = useDzModal() // # 通过dzmodal动态操作对话框 const openTestModalAsync=()=>{ dzmodal.open(TestModalAsync,{ name:'张三' }) .then(res=>{ if(res.type==='ok'){ ElMessageBox.alert('TestModal点击了确定'); }else{ ElMessageBox.alert('TestModal点击了取消'); } }) } </script>
TestModal.vue
<script setup lang="ts"> import { reactive, ref, defineProps } from 'vue' const emits = defineEmits(['ok','cancel']) const props = defineProps({ name: String }); const dialogVisible = ref(true) const resultData= reactive({ type:'ok', data:{} }) </script> <template> <el-dialog v-model="dialogVisible" title="TestModal" width="30%" > <div>通过DzModal打开TestModal</div> <div>外部传入:{{ name }}</div> <template #footer> <span class="dialog-footer"> <el-button @click="emits('cancel',{});dialogVisible = false">Cancel</el-button> <el-button type="primary" @click="emits('ok',{});dialogVisible = false" >Confirm</el-button > </span> </template> </el-dialog> </template> <style scoped> </style>
使用结果
动态操作对话框的实现
动态操作对话框,主要思路是动态创建虚拟dom节点,将对话框组件渲染到组件上,核心关键点是要动态创建的节点的上下文为当前app上下文vm.appContext = this._app._context
DzModalService.ts
import { App, inject, Plugin, h, render} from 'vue' import { ComponentOptions } from 'vue'; export const DzModalSymbol = Symbol() export class DzModalResult{ type: 'ok'|'cancel'|'string' = 'ok' body?:any= undefined } export class DzModalService{ private _app?:App=undefined constructor(app:App){ this._app=app; } public open(modal:ComponentOptions, props?: any):Promise<DzModalResult>{ return new Promise((reslove,reject)=>{ if(!this._app){ reject('_app is undefined') return; } const container = document.createElement("div"); document.body.appendChild(container) // 这里需要合并props,传入到组件modal const vm = h(modal, { ...props, onOk:(data?:any)=>{ // 弹出框关闭时移除节点 document.body.removeChild(container) reslove(this.ok(data)); }, onCancel:(data?:any)=>{ reslove(this.cancel(data)); } }); // 这里很重要,关联app上下文 vm.appContext = this._app._context render(vm,container); }); } public ok(data?:any):DzModalResult{ const result = new DzModalResult(); result.type='ok'; result.body=data; return result; } public cancel(data?:any):DzModalResult{ const result = new DzModalResult(); result.type='cancel'; result.body=data; return result; } } export function useDzModal(): DzModalService { const dzModal = inject<DzModalService>(DzModalSymbol) if(!dzModal){ throw new Error('No DzModal provided!') } return dzModal; } const plugin: Plugin = { install(app:App, options?:{[key:string]:any}){ const dzModal = new DzModalService(app) app.config.globalProperties.$dzModal= dzModal app.provide(DzModalSymbol, dzModal) } } export default plugin;
main.ts
import { createApp } from 'vue' import App from './App.vue' import ElementPlus from 'element-plus' import 'element-plus/dist/index.css' import DzModal from './dzmodal' createApp(App) .use(ElementPlus) .use(DzModal) // 安装 dzmodal插件 .mount('#app')
总结
使用异步动态加载对话框,父组件无需控制对话框组件的visible属性 , 这样可以简化父组件操作,不在关心对话框组件在什么时间关闭,如果对话框组件需要访问网络,也在子组件中完成。父组件主要做两件事:
- 通过异步组件方式引入对话框组件
- 打开对话框组件
到此这篇关于vue3动态加载对话框的文章就介绍到这了,更多相关vue3动态加载对话框内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!
相关文章
浅析Vue3中通过v-model实现父子组件的双向数据绑定及利用computed简化父子组件双向绑定
这篇文章主要介绍了浅析Vue3中通过v-model实现父子组件的双向数据绑定及利用computed简化父子组件双向绑定,需要的朋友可以参考下2022-12-12最全vue的vue-amap使用高德地图插件画多边形范围的示例代码
这篇文章主要介绍了最全vue的vue-amap使用高德地图插件画多边形范围,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下2020-07-07vue props使用typescript自定义类型的方法实例
这篇文章主要给大家介绍了关于vue props使用typescript自定义类型的相关资料,文中通过实例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下2023-01-01vue中LocalStorage与SessionStorage的区别与用法
本文主要介绍了LocalStorage和SessionStorage。LocalStorage和SessionStorage是两种存储方式,本文详细的介绍一下区别以及用法,感兴趣的可以了解一下2021-09-09
最新评论