vue二次封装ant-design-vue中的Modal弹窗组件,实现拖拽,全屏两种功能,原有参数属性不变

在我们的项目的有的地方需要用弹框的拖拽,以及弹窗自定义全屏显示的需求,所以再次将二次合一,同时弹框里面内容自适应屏幕高度

在ant-design-vue中,已经实现了拖拽,全屏的功能,下面是ant官网的示例

自定义渲染对话框

全屏

下面是对ant原有的功能进行二次封装,由于组件拆解了,子组件分为三部分:为如下代码:

一:子级的根目录modal下新建index.vue,弹框涉及到的参数可参考ant官网

<template>
<!--  v-bind处理a-table 传递过来的参数,你们开发的时候不用再子组件中接收这么ant原有的参数--><a-modal class="Kld-dfs-modal" :open="isOpen" :body-style="computeBodyStyle" :width="width" :centered="centered"@cancel="closeModal" :afterClose="handleAfterClose" :destroyOnClose="destroyOnClose" :keyboard="keyboardOpen":style="topStyle" :maskClosable="maskClosable" :mask="mask" :wrap-class-name="fullModal" v-bind="attrs"><template #title><div class="modalHeader">
<!-- 标题的ref用于控制是否可以拖拽--><div :ref="modalTitleRefS" class="draggableTitle">{{ title }}</div>
<!-- 弹框头部的标题,以及全屏的按钮--><ModalHeader :width="width" @fullScreen="handleFullScreen" @reduction="handleReduction" /></div></template><template #closeIcon>
<!-- 弹框头部的关闭按钮区域--><ModalClose /></template><div ref="bodySlot">
<!-- 弹框内容区域--><slot name="body"></slot></div><template #footer>
<!-- 弹框底部按钮区域--><div ref="footerSlot"><slot name="footer" v-if="haveFooter"></slot></div></template>
<!-- 实现是否拖拽弹框计算的核心代码--><template #modalRender="{ originVNode }"><div :style="transformStyle"><component :is="originVNode" /></div></template></a-modal>
</template><script  lang="ts">
import { defineComponent, ref, computed, watch, CSSProperties, watchEffect } from 'vue';
import { useDraggable } from '@vueuse/core';
import { FullscreenExitOutlined, FullscreenOutlined, CloseOutlined } from '@ant-design/icons-vue';import ModalClose from './components/ModalClose.vue';
import ModalHeader from './components/ModalHeader.vue';
import { Modal } from 'ant-design-vue';export default defineComponent({name: 'KldDfsModal',props: {centered: {type: Boolean,default: true},maskClosable: {type: Boolean,default: true,},mask: {type: Boolean,default: true},topStyle: {type: String,default: ''},closable: {type: Boolean,default: true,},destroyOnClose: {type: Boolean,default: false,},keyboardOpen: {type: Boolean,default: false,},title: {type: String,default: ''},open: {type: Boolean,default: false},haveFooter: {type: Boolean,default: true},width: {type: [String, Number],default: '60vw'},boxHeight: {type: [String, Number],default: 60},bodyStyle: {type: Object,default: {overflowX: 'hidden',overflowY: 'auto'}}},setup(props, { attrs, emit }) {/*****拖拽相关*****/const modalTitleRefS = ref('modalTitleRef')const modalTitleRef = ref<HTMLElement>();const { x, y, isDragging } = useDraggable(modalTitleRef);const startX = ref<number>(0);const startY = ref<number>(0);const startedDrag = ref(false);const transformX = ref(0);const transformY = ref(0);const preTransformX = ref(0);const preTransformY = ref(0);const dragRect = ref({ left: 0, right: 0, top: 0, bottom: 0 });const transformStyle = computed<CSSProperties>(() => {return {transform: `translate(${transformX.value}px, ${transformY.value}px)`,};});watch([x, y], () => {if (!startedDrag.value) {startX.value = x.value;startY.value = y.value;const bodyRect = document.body.getBoundingClientRect();const titleRect = modalTitleRef.value?.getBoundingClientRect() ?? { width: 0, height: 0 };dragRect.value.right = bodyRect.width - titleRect.width;dragRect.value.bottom = bodyRect.height - titleRect.height;preTransformX.value = transformX.value;preTransformY.value = transformY.value;}startedDrag.value = true;});watch(isDragging, () => {if (!isDragging) {startedDrag.value = false;}});watchEffect(() => {if (startedDrag.value) {transformX.value =preTransformX.value +Math.min(Math.max(dragRect.value.left, x.value), dragRect.value.right) -startX.value;transformY.value =preTransformY.value +Math.min(Math.max(dragRect.value.top, y.value), dragRect.value.bottom) -startY.value;}});/*****************/const bodySlot = ref(null);const footerSlot = ref(null);const isOpen = ref<boolean>(false);//const centered = ref<boolean>(true);const computeBodyStyle = ref<CSSProperties>();//动态计算内容高度,生成弹窗const computeWindowStyle = (bodyRealHeight: number = 0, headerRealHeight: number = 0, footerRealHeight: number = 0) => {let windowHeight = document.body.offsetHeight;let realHeight: number = 0;//后面增加数值的构成 model padding 上下20 + 20 header和body之间 25 footer和body之间 20if (bodyRealHeight + headerRealHeight + footerRealHeight + 85 >= windowHeight) {realHeight = windowHeight - headerRealHeight - footerRealHeight - 82;} else {realHeight = bodyRealHeight + 25;}computeBodyStyle.value = Object.assign({height: `${realHeight}px`}, props.bodyStyle);};const width = ref<string | number>(props.width);const fullModal = ref<string>();// 全屏const handleFullScreen = () => {width.value = '100%'modalTitleRefS.value = ''fullModal.value = 'kld-full-modal'transformX.value = 0;transformY.value = 0;}// 还原const handleReduction = () => {width.value = props.widthmodalTitleRefS.value = 'modalTitleRef'fullModal.value = ''}const closeModal = (e: Event) => {emit('cancel', e);};/*** @description: Modal 完全关闭后的回调*/const handleAfterClose = () => {console.log('Modal 完全关闭后的回调');fullModal.value = ''modalTitleRefS.value = 'modalTitleRef'width.value = props.widthemit('afterClose')};watch(() => props.open,(newVal) => {if (newVal) {isOpen.value = true;} else {isOpen.value = false;}},{ deep: true });watch(() => bodySlot.value, (newVal) => {if (newVal) {const bodyDom: any = newVal, footerDom: any = footerSlot.value, headerDom: any = modalTitleRef.value;x.value = startX.value;y.value = startY.value;computeWindowStyle(bodyDom.clientHeight, headerDom.clientHeight, footerDom.clientHeight);}}, { deep: true });return {modalTitleRefS,isOpen,modalTitleRef,computeBodyStyle,width,fullModal,transformStyle,handleFullScreen,handleReduction,closeModal,handleAfterClose,bodySlot,footerSlot,transformX,transformY,dragRect,startedDrag,isDragging,computeWindowStyle,startX,startY,preTransformX,preTransformY,x,y,attrs,listeners: emit,};},components: {CloseOutlined,FullscreenOutlined,FullscreenExitOutlined,AModal: Modal,ModalClose,ModalHeader}});</script><style scoped lang="less">
.draggableTitle {width: 100%;cursor: move
}
<!-- 用于弹框初始的位置-->
:global(.ant-modal-root .ant-modal-wrap) {overflow: hidden;
}:deep(.ant-modal-body)::-webkit-scrollbar {width: 0px;height: 0px;padding: 0px;
}:deep(.ant-modal-body)::-webkit-scrollbar-thumb {background: #c1c1c1;border-radius: 3px;
}:deep(.ant-modal-body)::-webkit-scrollbar-track {border-radius: 3px;background: #f1f1f1;
}
</style>
<style lang="less">
<!-- 用于弹框全屏的样式-->
@import url('./common.less');
</style>

子级的根目录modal下新建common.less,全屏的cs样式

// modal 全屏
.modalHeader {display: flex;justify-content: space-between;
}
.kld-full-modal{.ant-modal {max-width: 100% !important;top: 0 !important;padding-bottom: 0 !important;margin: 0 !important;}.ant-modal-content {display: flex;flex-direction: column;height: calc(100vh);}.ant-modal-body {flex: 1;max-height: calc(100vh);}
}

二:子级的根目录modal下新建components文件然后新建存放头部关闭按钮的组件以及全屏按钮的组件,

ModalClose组件

<template><a-tooltip ref="KldTooltip" color="#ffffff" placement="bottom"><template #title><span style="color: black;">关闭</span></template><CloseOutlined /></a-tooltip>
</template><script lang="ts">
import { ref } from 'vue';
import { Tooltip } from 'ant-design-vue';
import { CloseOutlined } from '@ant-design/icons-vue';export default {name: "KldTooltip",setup(_, { attrs, emit }) {return {attrs,listeners: emit,KldTooltip: ref()};},components: {ATooltip: Tooltip,CloseOutlined}
};
</script>

ModalHeader组件

<template><div><a-tooltip color="#ffffff" v-if="width != '100%'" placement="bottom"><template #title><span style="color: black;">全屏</span></template><a-button type="text" class="ant-modal-close" style="margin-right: 30px;" @click="handleFullScreen"><FullscreenOutlined /></a-button></a-tooltip><a-tooltip color="#ffffff" v-else placement="bottom"><template #title><span style="color: black;">还原</span></template><a-button type="text" class="ant-modal-close" style="margin-right: 30px;" @click="handleReduction"><FullscreenExitOutlined /></a-button></a-tooltip></div>
</template><script lang="ts">
import { ref } from 'vue';
import { Tooltip } from 'ant-design-vue';
import { FullscreenExitOutlined, FullscreenOutlined } from '@ant-design/icons-vue';export default {name: "KldTooltip",props: {width: {type: [String, Number],default: '40%'},},setup(props, { attrs, emit }) {const handleFullScreen = () => {emit('fullScreen');};const handleReduction = () => {emit('reduction');};return {props,attrs,listeners: emit,KldTooltip: ref(),handleFullScreen,handleReduction};},components: {ATooltip: Tooltip,FullscreenExitOutlined,FullscreenOutlined}
};
</script>

 三、然后在main.js里面引入,也可以直接在父组件引入,此处就不讲解引入了,直接父组件使用弹框

父组件

<template><kld-dfs-modal :open="createVisible" :title="createTitle" :width="'40%'" :destroyOnClose="true" :haveFooter="true":boxHeight="85" @cancel="createVisible = false" :maskClosable="false"><template #body><ApplicationCreate @close="handleClose" :editFormRef="editFormRef" :editId="editId":createTitle="createTitle" /><p class="h-20" v-for="index in 20" :key="index">根据屏幕高度自适应</p></template><template #footer><a-button>测试底部按钮插槽</a-button></template></kld-dfs-modal>
</template><script lang="ts" setup>
const createVisible = ref<boolean>(false); //创建
const createTitle = ref<string>(""); //创建弹窗标题const showModalA = () => {createVisible.value = true;createTitle.value = "创建应用";editFormRef.value = {}editId.value = ''
};
// 关闭创建弹窗
const handleClose = (type: string) => {if (type === "提交") {}createVisible.value = false;
};
</script>

效果图如下:内容可自适应屏幕高度,如果不需要可通过弹框标题的ref控制

 
如果在使用中有什么问题,还请多多指点,也可以私信或者留言,觉得可用麻烦点点赞以及收藏

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/631352.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

开发实践6_project

要求&#xff1a; ① 页面写入超链接&#xff0c;获取所有数据item&#xff0c;显示在另一个页面&#xff0c;1min内&#xff0c;即使数据有变化&#xff0c;页面内容不变&#xff0c;1min后点击超链接可获取最新信息&#xff1b; ② 使用middleware完成用户请求路径判断 &am…

一篇综述洞悉医学大型语言模型的原理,应用和挑战

在过去的一年中&#xff0c;随着 GPT-4、LLaMA、Mistral&#xff0c;PaLM 等先进技术的突飞猛进&#xff0c;大型语言模型&#xff08;Large Language Models&#xff09;已经引领全球人工智能进入了一个全新的基础模型时代&#xff0c;这一时代不仅开启了技术创新的新篇章&…

微信小程序vue+uniapp瑜伽馆课程预约选课管理系统

本文对该站点以及对其进行了全面的剖析&#xff0c;为瑜伽馆管理系统的发展带来了借鉴。瑜伽馆管理系统小程序对于瑜伽馆起到了很好的引导作用&#xff0c;特别是对于地方瑜伽馆的处理起到了很大的作用。 按照已有的功能&#xff0c;除了管理员外&#xff0c;还可以对系统中的全…

python高级练习题库实验2(B)部分

文章目录 题目1代码实验结果题目2代码实验结果题目总结题目1 注册课程小游戏程序 研究下面的例子,并编写一个与这些例子完全相同的程序。使用for loop和break来解决问题。提示用户输入课程数量,是否选择,并且课程代码,最后还需显示已经完成的课程注册数量或者未完成的注册…

【Python3】【力扣题】389. 找不同

【力扣题】题目描述&#xff1a; 【Python3】代码&#xff1a; 1、解题思路&#xff1a;使用计数器分别统计字符串中的元素和出现次数&#xff0c;两个计数器相减&#xff0c;结果就是新添加的元素。 知识点&#xff1a;collections.Counter(...)&#xff1a;字典子类&#x…

Python算法例36 丑数Ⅱ

1. 问题描述 设计一个算法&#xff0c;找出只含素因子2、3、5的第n小的数&#xff0c;符合条件的数如&#xff1a;1、2、3、4、5、6、8、9、10、12… 2. 问题示例 如果n9&#xff0c;返回10。 3. 代码实现 def find_nth_number(n):if n < 0:return Nonenumbers [1]idx…

Azure Machine Learning - 聊天机器人构建

目录 聊天机器人架构概述消耗成本环境准备打开开发环境部署和运行将聊天应用部署到 Azure使用聊天应用从 PDF 文件获取答案使用聊天应用设置更改答复行为 本文介绍如何部署和运行适用于 Python 的企业聊天应用示例。 此示例使用 Python、Azure OpenAI 服务和 Azure AI 搜索中的…

【playwright】新一代自动化测试神器playwright+python系列课程22_playwright模拟鼠标操作_悬停_右击_双击

playwright模拟鼠标操作_悬停_右击_双击 鼠标悬停 – hover() 有些元素&#xff0c;只有你鼠标移动到它那个位置上&#xff0c;他才会展开或者显示一些内容&#xff0c;这时候就要用到hover()操作了。 以个人设置这个下拉菜单&#xff0c;来看一下hover的使用。 项目实践代…

产品排名提升秘籍:亚马逊鲲鹏系统的独特优势

在亚马逊竞争激烈的市场中&#xff0c;商家们一直在寻找新的方法来提升产品排名和吸引更多的顾客。近期&#xff0c;一款名为“亚马逊鲲鹏系统”的全自动化运行软件引起了广泛关注。该软件集成了一系列独特的功能和特点&#xff0c;旨在帮助商家更轻松地管理其亚马逊业务&#…

企业微信无法正常启动 报错0xc0000142

报错内容如下&#xff0c;每次打开工作电脑时候企业微信一般会正常启动&#xff0c;但是有时候经常会出现下面这种错误&#xff0c;重启也解决不了&#xff0c;每次都得重装企业微信&#xff0c;今天整理了一下网上的方法&#xff0c;这个原因大概率是亿赛通。 解决办法&#x…

2024年网络安全比赛--内存取证(超详细)

一、竞赛时间 180分钟 共计3小时 二、竞赛阶段 竞赛阶段 任务阶段 竞赛任务 竞赛时间 分值 1.从内存文件中找到异常程序的进程&#xff0c;将进程的名称作为Flag值提交&#xff1b; 2.从内存文件中找到黑客将异常程序迁移后的进程编号&#xff0c;将迁移后的进程编号作为Flag值…

Python学习之路——异常捕获

一、什么是异常 当检测到一个错误时&#xff0c;Python解释器就无法继续执行了&#xff0c;反而出现了一些错误的提示&#xff0c;这就是所谓的“异常”&#xff0c;也就是bug 二、异常的捕获方法 当我们的程序遇到bug&#xff0c;那么就下来有两种情况 ①整个程序因为一个…

Elasticsearch:和 LIamaIndex 的集成

LlamaIndex 是一个数据框架&#xff0c;供 LLM 应用程序摄取、构建和访问私有或特定领域的数据。 LlamaIndex 是开源的&#xff0c;可用于构建各种应用程序。 在 GitHub 上查看该项目。 安装 在 Docker 上设置 Elasticsearch 使用以下 docker 命令启动单节点 Elasticsearch 实…

chromedriver+Selenium+springboot+Java实现后端截图

chromedriver这种方法实现截图&#xff0c;依赖服务器端的谷歌浏览器软件&#xff0c;需要在服务器端下载谷歌浏览器。 Windows服务器说明 1.下载谷歌浏览器 2.根据第一步下载的谷歌浏览器版本&#xff0c;下载chromedriver&#xff0c;可以在这个页面找到和版本相近的版本去下…

docker部署Jira+配置MySQL8数据库

写在前面&#xff1a;如果你通过docker安装Jira且启动过&#xff0c;然后你现在又想使用mysql数据库&#xff0c;需要注意 你除了停掉原有容器&#xff0c;还需要删除&#xff1a;/var/lib/docker/volumes/jiraVolume/_data下的文件&#xff0c;否则启动后会无法正常使用。注意…

Redis 笔记一

概览 1.Redis核心数据存储结构 2.Redis底层String编码int&embstr&raw 3.Redis底层压缩列表&跳表&哈希表 4.Redis底层Zset实现压缩列表和跳表如何选择 5.基于Redis实现微博&抢红包&12306核心业务 辅助学习&#xff1a;Redis 教程 | 菜鸟教程 1.Redis为什…

ArcGIS初始化软件界面Normal.mxt

ArcGIS有时候永久了&#xff0c;或者呢突然不自觉软件界面乱了&#xff0c;或者一些窗口打开却找不到&#xff01; 这时候可以去删除arcgis的界面配置文件&#xff0c;Normal.mxt 删除后再打开软件&#xff0c;软件界面就会回到初始化设置了&#xff01; 文件所在的路径&…

从零开始学习Zeppelin:大数据可视化分析的交互式开发系统!

介绍&#xff1a;Apache Zeppelin是一个基于Web的交互式开发系统&#xff0c;主要用于进行大数据可视化分析。其核心概念是notebook&#xff0c;所有的操作都可以在notebook中完成。Zeppelin提供了一套非常全面的数据分析解决方案&#xff0c;支持数据采集、数据发现、数据分析…

canal server初始化源码分析

CanalLauncher类是canal server端启动的入口类&#xff0c;跟随代码进行深入。 在开始之前&#xff0c;我们可以先了解下&#xff0c; canal 配置方式 ManagerCanalInstanceGenerator&#xff1a; 基于manager管理的配置方式&#xff0c;实时感知配置并进行server重启Spring…

k8s---ingress对外服务(七层)

ingress 概念 k8s的对外服务&#xff0c;ingress service作用现在两个方面&#xff1a; 1、集群内部&#xff1a;不断跟踪的变化&#xff0c;更新endpoint中的pod对象&#xff0c;基于pod的ip地址不断变化的一种服务发现机制。 2、集群外部&#xff1a;类似于负载均衡器&a…