Vue3全局挂载Dialog组件

一、背景描述

最近项目中遇到了全局挂载Dialog的需求,在这里记录一下,希望可以帮到大家。

二、Dialog组件封装

Dialog组件结构如下:

2.1 Dialog.vue
javascript 复制代码
<template>
  <!-- 删除文件对话框 -->
  <el-dialog
    :title="dialogTitle"
    v-model="dialogVisible"
    :width="dialogWidth"
		:before-close="handleBeforeClose"
  >
    <div>{{ title }}</div>
    <template #footer>
      <div class="dialog-footer">
        <el-button @click="handleDialogClose">取 消</el-button>
        <el-button
          type="primary"
          :loading="sureBtnLoading"
          @click="handleDialogSure"
          >确 定</el-button
        >
      </div>
    </template>
  </el-dialog>
</template>
<script setup>
import {
  ref,
  computed
} from 'vue'
import {
  deleteFile
} from '@/api/file'
import { ElMessage } from 'element-plus';

const props = defineProps({
  fileInfo: {
    type: Array,
    default: [],
  },
  title: {
    type: String,
    default: "",
  },
  width: {
    type: String,
    default: "550px",
  },
  saveClick: {
    type: Function,
    default: () => {},
  },
  cancelClick: {
    type: Function,
    default: () => {},
  },
  remove: {
    type: Function,
    default: () => {},
  }
})

const dialogVisible = ref(false) // 窗体显示控制
const dialogWidth = computed(() => {
  return props.width || "550px"
})
const dialogTitle = computed(() => {
  return props.title || '删除文件'
})

const title = computed(() => {
  return '此操作将永久删除, 是否继续?'
})

const sureBtnLoading = ref(false)
// 关闭回调触发事件
const handleBeforeClose = (done) => {
  done()
}
// 取消按钮点击事件
const handleDialogClose = () => {
  props.cancelClick()
  props.remove()
}
// 确定按钮点击事件
const handleDialogSure = () => {
	sureBtnLoading.value = true
  deleteFile({
    id: props.fileInfo.id
  })
    .then((res) => {
      sureBtnLoading.value = false
      ElMessage.success(res.data || '删除成功')
      props.saveClick()
      props.remove()
    })
    .catch(() => {
      sureBtnLoading.value = false
    })
}
</script>
2.2 index.js
javascript 复制代码
import DeleteFileDialog from './Dialog.vue'
import { createApp } from 'vue'

let mountNode
let createMount = (opts) => {
	if(mountNode){ // 确保只存在一个弹框
		document.body.removeChild(mountNode)
    mountNode = null
	}
  mountNode = document.createElement('div')
  document.body.appendChild(mountNode)
  const app = createApp(DeleteFileDialog, {
    ...opts,
    modelValue: true,
    remove() { // 传入remove函数,组件内可移除dom 组件内通过props接收
      app.unmount(mountNode)
      document.body.removeChild(mountNode)
      mountNode = null
    }
  })
  return app.mount(mountNode)
}
function DeleteFile(options = {}) {
  options.id = options.id || 'deleteFile_' + 1 //唯一id 删除组件时用于定位
  let $init = createMount(options)
  return $init
}
DeleteFile.install = app => {
  app.component('delete-file-dialog', DeleteFileDialog)
  app.config.globalProperties.$deleteFileDialog = DeleteFile
}
export default DeleteFile

三、全局挂载Dialog组件

在main.js中进行全局挂载

javascript 复制代码
import DeleteFileDialog from '@/components/dialog/deleteFile/index.js'
app.use(DeleteFileDialog)

四、使用Dialog组件

javascript 复制代码
proxy.$deleteFileDialog({
    fileInfo: fileInfo,
    saveClick: () => {
		console.log('saveClick---')
	},
	cancelClick: () => {
		console.log('cancelClick---')
	}
})

五、效果图

相关推荐
在繁华处几秒前
从零搭建轻灵(四):工具系统与 Pipeline
前端·chrome
小则又沐风a4 分钟前
进一步了解进程---第四章 进程管理
java·服务器·前端
放下华子我只抽RuiKe55 分钟前
FastAPI 全栈后端(一):为什么选择 FastAPI
前端·javascript·深度学习·react.js·机器学习·前端框架·fastapi
ZC跨境爬虫6 分钟前
跟着 MDN 学CSS day_11:(深入理解CSS值与单位的完整体系)
前端·css·ui·html·tensorflow
青云计划9 分钟前
SSE流式响应:从Reactor Flux到生产级AI聊天的工程实践——5分钟超时、线程隔离、背压处理全解析
前端·人工智能·firefox
A南方故人15 分钟前
将容器内的元素变为可拖拽
开发语言·javascript·ecmascript
codefan※25 分钟前
7 个Prompt 框架汇总:从 Chain of Thought 到 ReAct + PoT
前端·react.js·ai·llm·prompt·prompt工程·思维链
迁旭27 分钟前
Claude Code /status 功能技术文档
前端·javascript·人工智能·react.js·机器学习·gpt-3·文心一言
星星~笑笑31 分钟前
react Next.js oss上传 上传阿里云
javascript·react.js·阿里云·js