Dialog组件—弹窗拖拽功能

目录

一、概述

二、详解


一、概述

问题:ElementUI框架中的Dialog弹窗组件没有提供可拖拽功能,开发中需要自己实现。

二、详解

解决方案​​​​​​

Vue中自定义指令实现弹窗的拖拽功能。 

代码示例

创建一个drag.js文件,文件中定义一个拖拽的全局指令,代码如下所示。

import Vue from 'vue';

/**
 * v-dialogDrag 弹窗拖拽指令
 */
Vue.directive('dialogDrag', {
  // 钩子函数,第一次绑定到目标元素时调用
  bind(el, binding, vnode, oldVnode) {
    // 获取弹窗的头部
    const dialogHeaderEl = el.querySelector('.el-dialog__header');
    // 设置鼠标悬浮在弹窗头部的样式
    dialogHeaderEl.style.cursor = 'move';

    // 获取弹窗的整体
    const dragDom = el.querySelector('.el-dialog');
    // 获取弹窗整体的样式(兼容写法)
    // ie dom元素.currentStyle 火狐谷歌 window.getComputedStyle(dom元素, null);
    const sty = dragDom.currentStyle || window.getComputedStyle(dragDom, null);

    // 弹窗头部设置鼠标按下事件
    dialogHeaderEl.onmousedown = (e) => {
      // 获取事件对象,兼容全浏览器
      const event = e || window.event;
      // 鼠标按下,获取鼠标在盒子内的坐标
      const disX = event.clientX - dialogHeaderEl.offsetLeft;
      const disY = event.clientY - dialogHeaderEl.offsetTop;

      // 获取到的值带px 正则匹配替换
      let styL;
      let styT;
      // 注意在ie中 第一次获取到的值为组件自带50% 移动之后赋值为px
      if (sty.left.includes('%')) {
        styL = +document.body.clientWidth * (+sty.left.replace(/\%/g, '') / 100);
        styT = +document.body.clientHeight * (+sty.top.replace(/\%/g, '') / 100);
      } else {
        styL = +sty.left.replace(/\px/g, '');
        styT = +sty.top.replace(/\px/g, '');
      }
      
      // 鼠标移动的时候,把鼠标在页面中的坐标,减去鼠标在盒子内的坐标就是模态框的left和top值
      document.onmousemove = function (e) {
        // 获取事件对象,兼容全浏览器
        const event = e || window.event;
        // 通过事件委托,计算移动的距离 (开始拖拽至结束拖拽的距离)
        const l = event.clientX - disX;
        const t = event.clientY - disY;

        let finallyL = l + styL;
        let finallyT = t + styT;

        // 边界值判定 注意clientWidth scrollWidth区别 要减去之前的top left值
        let limitL = (document.documentElement.clientWidth - dragDom.clientWidth) / 2;
        if (finallyL < -limitL) {
            finallyL = -limitL;
        } else if (finallyL > limitL) {
            finallyL = limitL;
        }
        let limitT = document.documentElement.clientHeight * 15 / 100;
        let limitB = document.documentElement.clientHeight - dragDom.clientHeight - limitT;
        if (finallyT < -limitT) {
            finallyT = -limitT;
        } else if (finallyT > limitB) {
            finallyT = limitB;
        }
        // 移动当前元素
        dragDom.style.left = `${finallyL}px`;
        dragDom.style.top = `${finallyT}px`;
      };
      // 鼠标松开按键的回调
      document.onmouseup = function(e) {
        document.onmousemove = null;
        document.onmouseup = null;
      };
    };
  }
});

在main.js中引入drag.js文件。

import '../../common/drag.js';

如下代码所示,正常使用指令即可。

<el-dialog
    title="弹窗"
    :visible.sync="visible"
    :close-on-click-modal="false"
     v-dialogDrag
></el-dialog>

参考文档:https://www.jianshu.com/p/c3ce06c423af?tdsourcetag=s_pctim_aiomsg

扫描二维码关注公众号,回复: 12909218 查看本文章

猜你喜欢

转载自blog.csdn.net/weixin_42472040/article/details/114532692