vue实现element-ui对话框可拖拽功能


Posted in Javascript onAugust 17, 2018

element-ui对话框可拖拽及边界处理

应业务需求,需要实现对话框可拖拽问题,应element-ui没有提供官方支持,于是便参考大神的文章,得出了适合业务需要的解决方案。很多大神给出的代码是没有解决边界问题的,但是不解决边界问题存在一个bug,拖到不可视区域后边再也拖不回来了,不信你们可以试试。

在实现的功能的情况下,封装成了js文件,然后再main.js中引入后可全局使用。

还是上代码吧

功能实现代码directives.js代码如下:

import Vue from 'vue';
 
// v-dialogDrag: 弹窗拖拽属性
Vue.directive('dialogDrag', {
  bind(el, binding, vnode, oldVnode) {
    const dialogHeaderEl = el.querySelector('.el-dialog__header');
    const dragDom = el.querySelector('.el-dialog');
    //dialogHeaderEl.style.cursor = 'move';
    dialogHeaderEl.style.cssText += ';cursor:move;'
    dragDom.style.cssText += ';top:0px;'
 
    // 获取原有属性 ie dom元素.currentStyle 火狐谷歌 window.getComputedStyle(dom元素, null);
    const sty = (function() {
        if (window.document.currentStyle) {
            return (dom, attr) => dom.currentStyle[attr];
        } else{
            return (dom, attr) => getComputedStyle(dom, false)[attr];
        }
    })()    
    
    dialogHeaderEl.onmousedown = (e) => {
      // 鼠标按下,计算当前元素距离可视区的距离
      const disX = e.clientX - dialogHeaderEl.offsetLeft;
      const disY = e.clientY - dialogHeaderEl.offsetTop;
      
      const screenWidth = document.body.clientWidth; // body当前宽度
        const screenHeight = document.documentElement.clientHeight; // 可见区域高度(应为body高度,可某些环境下无法获取) 
        
        const dragDomWidth = dragDom.offsetWidth; // 对话框宽度
        const dragDomheight = dragDom.offsetHeight; // 对话框高度
        
        const minDragDomLeft = dragDom.offsetLeft;
        const maxDragDomLeft = screenWidth - dragDom.offsetLeft - dragDomWidth;
        
        const minDragDomTop = dragDom.offsetTop;
        const maxDragDomTop = screenHeight - dragDom.offsetTop - dragDomheight;
 
      
      // 获取到的值带px 正则匹配替换
      let styL = sty(dragDom, 'left');
      let styT = sty(dragDom, 'top');
 
      // 注意在ie中 第一次获取到的值为组件自带50% 移动之后赋值为px
      if(styL.includes('%')) {
        styL = +document.body.clientWidth * (+styL.replace(/\%/g, '') / 100);
        styT = +document.body.clientHeight * (+styT.replace(/\%/g, '') / 100);
      }else {
        styL = +styL.replace(/\px/g, '');
        styT = +styT.replace(/\px/g, '');
      };
      
      document.onmousemove = function (e) {
        // 通过事件委托,计算移动的距离 
                let left = e.clientX - disX;
                let top = e.clientY - disY;
                
                // 边界处理
                if (-(left) > minDragDomLeft) {
                    left = -(minDragDomLeft);
                } else if (left > maxDragDomLeft) {
                    left = maxDragDomLeft;
                }
                
                if (-(top) > minDragDomTop) {
                    top = -(minDragDomTop);
                } else if (top > maxDragDomTop) {
                    top = maxDragDomTop;
                }
 
        // 移动当前元素 
                dragDom.style.cssText += `;left:${left + styL}px;top:${top + styT}px;`;
      };
 
      document.onmouseup = function (e) {
        document.onmousemove = null;
        document.onmouseup = null;
      };
    } 
  }
})

在边界处理上,因为在我的项目中无法获取到body的高度(被这个折磨了好久),所以采取了获取可见区域高度,这里补充点知识

document.body.clientWidth //BODY对象宽度
document.body.clientHeight //BODY对象高度
document.documentElement.clientWidth //可见区域宽度
document.documentElement.clientHeight //可见区域高度

在main.js中引入

// 引入Dialog可拖拽,注意文件所在目录。目前尚未发现引入的先后关系,若有再补充
import './directives.js';

ue文件中使用:

在el-dialog标签中加入v-dialogDrag属性

<el-dialog v-dialogDrag></el-dialog>

具体使用便是这样,希望有人看到哈哈哈,当然主要还是想帮到大家。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
原生javascript兼容性测试实例
Jul 01 Javascript
jquery和ajax的关系详细介绍
Nov 29 Javascript
JavaScript获取网页中第一个链接ID的方法
Apr 03 Javascript
JS实现兼容性好,自动置顶的淘宝悬浮工具栏效果
Sep 18 Javascript
js控件Kindeditor实现图片自动上传功能
Jul 20 Javascript
利用Javascript仿Excel的数据透视分析功能
Sep 07 Javascript
angular基于路由控制ui-router实现系统权限控制
Sep 27 Javascript
在js中实现邮箱格式的验证方法(推荐)
Oct 24 Javascript
js实现微博发布小功能
Jan 12 Javascript
小程序实现自定义导航栏适配完美版
Apr 02 Javascript
javascript异步处理与Jquery deferred对象用法总结
Jun 04 jQuery
vue路由缓存的几种实现方式小结
Feb 02 Javascript
原生JS实现的简单轮播图功能【适合新手】
Aug 17 #Javascript
layer.confirm取消按钮绑定事件的方法
Aug 17 #Javascript
LayerClose弹窗关闭刷新方法
Aug 17 #Javascript
详解vue移动端项目的适配(以mint-ui为例)
Aug 17 #Javascript
layui前端框架之table表数据的刷新方法
Aug 17 #Javascript
Vue登录注册并保持登录状态的方法
Aug 17 #Javascript
小程序清理本地缓存的方法
Aug 17 #Javascript
You might like
php 传值赋值与引用赋值的区别
2010/12/29 PHP
php性能优化分析工具XDebug 大型网站调试工具
2011/05/22 PHP
php实现按天数、星期、月份查询的搜索框
2016/05/02 PHP
PHP7如何开启Opcode打造强悍性能详解
2018/05/11 PHP
Firefox 无法获取cssRules 的解决办法
2006/10/11 Javascript
js 鼠标点击事件及其它捕获
2009/06/04 Javascript
jQuery对象数据缓存Cache原理及jQuery.data方法区别介绍
2013/04/07 Javascript
JS实现将人民币金额转换为大写的示例代码
2014/02/13 Javascript
jQuery中ajax的get()方法用法实例
2014/12/26 Javascript
Bootstrap每天必学之标签页(Tab)插件
2020/08/09 Javascript
JS控制div跳转到指定的位置的几种解决方案总结
2016/11/05 Javascript
jQuery实现字符串全部替换的方法
2016/12/12 Javascript
jQuery插件HighCharts绘制2D带有Legend的饼图效果示例【附demo源码下载】
2017/03/10 Javascript
解决JQuery全选/反选第二次失效的问题
2017/10/11 jQuery
AngularJS动态生成select下拉框的方法实例
2019/11/17 Javascript
Python实现模拟登录及表单提交的方法
2015/07/25 Python
利用Python3分析sitemap.xml并抓取导出全站链接详解
2017/07/04 Python
python实现感知器
2017/12/19 Python
Python多线程中阻塞(join)与锁(Lock)使用误区解析
2018/04/27 Python
selenium在执行phantomjs的API并获取执行结果的方法
2018/12/17 Python
更新修改后的Python模块方法
2019/03/03 Python
Python QQBot库的QQ聊天机器人
2019/06/19 Python
Django实现celery定时任务过程解析
2020/04/21 Python
python中remove函数的踩坑记录
2021/01/04 Python
html5利用canvas绘画二级树形结构图的示例
2017/09/27 HTML / CSS
发现世界上最好的珠宝设计师:JewelStreet
2017/12/17 全球购物
MATCHESFASHION.COM法国官网:英国奢侈品零售商
2018/01/04 全球购物
构造方法和其他方法的区别
2016/04/26 面试题
师范毕业生求职自荐信
2013/09/25 职场文书
《匆匆》教学反思
2014/02/22 职场文书
学雷锋志愿者活动方案
2014/08/21 职场文书
李白故里导游词
2015/02/12 职场文书
雷锋电影观后感
2015/06/10 职场文书
优秀教师主要事迹材料
2015/11/04 职场文书
Python爬虫数据的分类及json数据使用小结
2021/03/29 Python
Debian11 Xfce终端光标的颜色怎么设置?
2022/08/14 数码科技