经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » JS/JS库/框架 » Vue.js » 查看文章
vue3动态加载对话框的方法实例
来源:jb51  时间:2022/3/29 16:18:11  对本文有异议

简介

介绍使用vue3的异步组件动态管理对话框组件,简化对话框组件使用方式。本文使用的是vue3、typescript、element_plus完成的示例。

常规方式使用对话框

一般情况下,使用对话框组件,会使用v-model进行双向绑定,通过visible变量控制对话框的显示和关闭。常规方式有一个弊端,自定义组件中使用<el-dialog>,需要通过父组件控制自定义组件是否展示。

  1. <template>
  2. <ElButton type="primary" @click="openGeneral()">常规方式打开Modal</ElButton>
  3. <el-dialog
  4. v-model="dialogVisible"
  5. title="Tips"
  6. width="30%"
  7. :before-close="handleClose"
  8. >
  9. <span>This is a message</span>
  10. <template #footer>
  11. <span class="dialog-footer">
  12. <el-button @click="dialogVisible = false">Cancel</el-button>
  13. <el-button type="primary" @click="dialogVisible = false"
  14. >Confirm</el-button
  15. >
  16. </span>
  17. </template>
  18. </el-dialog>
  19. </template>
  1. <script setup lang="ts">
  2. // This starter template is using Vue 3 <script setup> SFCs
  3. // Check out https://v3.vuejs.org/api/sfc-script-setup.html#sfc-script-setup
  4.  
  5. import {ref } from 'vue';
  6. const dialogVisible = ref(false)
  7.  
  8. const openGeneral=()=>{
  9. dialogVisible.value=true
  10. }
  11.  
  12. </script>

异步动态加载

先看使用异步组件进行动态加载对话框的方式。异步组件使用到的是defineAsyncComponent接口,只有使用到这个组件,才会从网络上加载。动态操作使用的useDzModal

使用方式

  1. <template>
  2. <ElButton type="primary" @click="openTestModalAsync()">动态异步打开TestModal</ElButton>
  3. </template>
  1. <script setup lang="ts">
  2. import { defineAsyncComponent,ref } from 'vue';
  3. import { ElMessageBox } from 'element-plus';
  4. import { useDzModal } from './dzmodal'
  5.  
  6. // 异步加载组件
  7. const TestModalAsync = defineAsyncComponent(()=>import('./components/TestModal.vue'))
  8.  
  9. const dzmodal = useDzModal()
  10.  
  11. // # 通过dzmodal动态操作对话框
  12. const openTestModalAsync=()=>{
  13. dzmodal.open(TestModalAsync,{
  14. name:'张三'
  15. })
  16. .then(res=>{
  17. if(res.type==='ok'){
  18. ElMessageBox.alert('TestModal点击了确定');
  19. }else{
  20. ElMessageBox.alert('TestModal点击了取消');
  21. }
  22. })
  23. }
  24.  
  25. </script>

TestModal.vue

  1. <script setup lang="ts">
  2. import { reactive, ref, defineProps } from 'vue'
  3. const emits = defineEmits(['ok','cancel'])
  4. const props = defineProps({
  5. name: String
  6. });
  7. const dialogVisible = ref(true)
  8.  
  9. const resultData= reactive({
  10. type:'ok',
  11. data:{}
  12. })
  13.  
  14. </script>
  15.  
  16. <template>
  17. <el-dialog
  18. v-model="dialogVisible"
  19. title="TestModal"
  20. width="30%"
  21. >
  22. <div>通过DzModal打开TestModal</div>
  23. <div>外部传入:{{ name }}</div>
  24. <template #footer>
  25. <span class="dialog-footer">
  26. <el-button @click="emits('cancel',{});dialogVisible = false">Cancel</el-button>
  27. <el-button type="primary" @click="emits('ok',{});dialogVisible = false"
  28. >Confirm</el-button
  29. >
  30. </span>
  31. </template>
  32. </el-dialog>
  33. </template>
  34.  
  35. <style scoped>
  36. </style>

使用结果

动态操作对话框的实现

动态操作对话框,主要思路是动态创建虚拟dom节点,将对话框组件渲染到组件上,核心关键点是要动态创建的节点的上下文为当前app上下文vm.appContext = this._app._context

DzModalService.ts

  1. import { App, inject, Plugin, h, render} from 'vue'
  2. import { ComponentOptions } from 'vue';
  3. export const DzModalSymbol = Symbol()
  4. export class DzModalResult{
  5. type: 'ok'|'cancel'|'string' = 'ok'
  6. body?:any= undefined
  7. }
  8.  
  9. export class DzModalService{
  10. private _app?:App=undefined
  11. constructor(app:App){
  12. this._app=app;
  13. }
  14. public open(modal:ComponentOptions, props?: any):Promise<DzModalResult>{
  15.  
  16. return new Promise((reslove,reject)=>{
  17. if(!this._app){
  18. reject('_app is undefined')
  19. return;
  20. }
  21. const container = document.createElement("div");
  22. document.body.appendChild(container)
  23. // 这里需要合并props,传入到组件modal
  24. const vm = h(modal, {
  25. ...props,
  26. onOk:(data?:any)=>{
  27. // 弹出框关闭时移除节点
  28. document.body.removeChild(container)
  29. reslove(this.ok(data));
  30. },
  31. onCancel:(data?:any)=>{
  32. reslove(this.cancel(data));
  33. }
  34. });
  35. // 这里很重要,关联app上下文
  36. vm.appContext = this._app._context
  37. render(vm,container);
  38. });
  39. }
  40.  
  41. public ok(data?:any):DzModalResult{
  42. const result = new DzModalResult();
  43. result.type='ok';
  44. result.body=data;
  45. return result;
  46. }
  47.  
  48. public cancel(data?:any):DzModalResult{
  49. const result = new DzModalResult();
  50. result.type='cancel';
  51. result.body=data;
  52. return result;
  53. }
  54. }
  55.  
  56. export function useDzModal(): DzModalService {
  57. const dzModal = inject<DzModalService>(DzModalSymbol)
  58. if(!dzModal){
  59. throw new Error('No DzModal provided!')
  60. }
  61. return dzModal;
  62. }
  63.  
  64. const plugin: Plugin = {
  65. install(app:App, options?:{[key:string]:any}){
  66. const dzModal = new DzModalService(app)
  67. app.config.globalProperties.$dzModal= dzModal
  68. app.provide(DzModalSymbol, dzModal)
  69. }
  70. }
  71.  
  72. export default plugin;

main.ts

  1. import { createApp } from 'vue'
  2. import App from './App.vue'
  3. import ElementPlus from 'element-plus'
  4. import 'element-plus/dist/index.css'
  5. import DzModal from './dzmodal'
  6. createApp(App)
  7. .use(ElementPlus)
  8. .use(DzModal) // 安装 dzmodal插件
  9. .mount('#app')

总结

使用异步动态加载对话框,父组件无需控制对话框组件的visible属性 , 这样可以简化父组件操作,不在关心对话框组件在什么时间关闭,如果对话框组件需要访问网络,也在子组件中完成。父组件主要做两件事:

  • 通过异步组件方式引入对话框组件
  • 打开对话框组件

到此这篇关于vue3动态加载对话框的文章就介绍到这了,更多相关vue3动态加载对话框内容请搜索w3xue以前的文章或继续浏览下面的相关文章希望大家以后多多支持w3xue!

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号