Javascript 拖拽雏形中的一些问题(逐行分析代码,让你轻松了拖拽的原理)


Posted in Javascript onJanuary 23, 2015

今天我们就来解决上一次拖拽雏形中的一些问题。下面看看有哪些问题?

附上上期的Javascript代码,方便大家查看问题。

<script type="text/javascript">
   window.onload = function() {
    var oDiv = document.getElementById("div1");
    var disX = 0;
    var disY = 0;
    oDiv.onmousedown = function(ev) {
     var oEvent = ev || event; 
     disX = oEvent.clientX - oDiv.offsetLeft; 
     disY = oEvent.clientY - oDiv.offsetTop;

     oDiv.onmousemove = function(ev) {
      var oEvent = ev || event;
      oDiv.style.left = oEvent.clientX - disX+'px'; 
      oDiv.style.top = oEvent.clientY - disY+'px';
     };
     oDiv.onmouseup = function() {
      oDiv.onmousemove = null;
      oDiv.onmouseup = null;
     };

    };

   };
  </script>

1. 现在的这个拖拽如果我鼠标移动的快点,Javascript 拖拽雏形中的一些问题(逐行分析代码,让你轻松了拖拽的原理)你会发现这个鼠标从这个div出来了,这个时候div不会跟着鼠标走了。

那为什么会出现这个问题呢?

原因其实很简单,mousemove的事件我们是给div加的,所以鼠标一旦脱离了这个div,那么这个时候mousemove已经不触发了。

解决方案: 事件加载document 上,因为你鼠标无论如何都还在页面里面,怎么样都会触发mousemove 这样移动的在快也没问题。

那么我们相应的修改下代码。

<script type="text/javascript">
   window.onload = function() {
    var oDiv = document.getElementById("div1");
    var disX = 0;
    var disY = 0;
    oDiv.onmousedown = function(ev) {
     var oEvent = ev || event; 
     disX = oEvent.clientX - oDiv.offsetLeft; 
     disY = oEvent.clientY - oDiv.offsetTop;
    // 事件加载document 上
     document.onmousemove = function(ev) {
      var oEvent = ev || event;
      oDiv.style.left = oEvent.clientX - disX+'px';
      oDiv.style.top = oEvent.clientY - disY+'px';
     };
     oDiv.onmouseup = function() {
      document.onmousemove = null;
      oDiv.onmouseup = null; 
     };

    };

   };
  </script>

那么这个问题就可以解决了。

2. 我们看看现在还有什么问题,虽然拖的快的问题解决了,但是当我把鼠标移动到这个位置

Javascript 拖拽雏形中的一些问题(逐行分析代码,让你轻松了拖拽的原理)

现在可以明显看到鼠标不在div上,如果这个时候抬起鼠标,你可以看到回来之后它还会动。Javascript 拖拽雏形中的一些问题(逐行分析代码,让你轻松了拖拽的原理) 这就又是一个bug!

其实这个问题和上面的是一样的。所以呢解决起来也很简单我们把mouseup也加到document上,我们来试一下看看

document.onmouseup = function() { 
      document.onmousemove = null; 
      document.onmouseup = null; 
     };

这样 现在如果在移动到刚才的那个位置,就不会在出现之前的bug了,并且移动的快也没有任何的问题。一切都很正常。

3. 我们看看浏览器兼容的问题

其实在低版本的火狐浏览器中有这样一个问题
Javascript 拖拽雏形中的一些问题(逐行分析代码,让你轻松了拖拽的原理)。怎么出现的呢,当你第一次拖的时候是对的,在拖一次的时候按住在移动,你会发现会有个这个影子在后面。这个是怎么回事呢?

实际上来说我们现在拖动的是一个空的div火狐是有bug的,那么如果在div中加点内容呢

Javascript 拖拽雏形中的一些问题(逐行分析代码,让你轻松了拖拽的原理)

你会发现现在又没有问题了。

所以火狐的bug就只有在空div中出现的。

解决方案:

其实很简单,我们就只要阻止浏览器默认事件就可以了 return false; 在onmousedown中。 为什么要加在onmousedown中呢?

大家可以想一下,拖拽是从哪个事件开始的,是从onmousedown开始的吧,当鼠标按下的时候拖拽就开始了。所以要加载onmousedown中。

实际上就是加了一句return false; 把火狐的bug屏蔽掉了。

这样不管怎么拖就没有问题了。

附上代码:

<script type="text/javascript">
   window.onload = function() {
    var oDiv = document.getElementById("div1");
    var disX = 0;
    var disY = 0;
    oDiv.onmousedown = function(ev) {
     var oEvent = ev || event;
     disX = oEvent.clientX - oDiv.offsetLeft;
     disY = oEvent.clientY - oDiv.offsetTop;
     // 事件加载document 上
     document.onmousemove = function(ev) {
      var oEvent = ev || event;
      oDiv.style.left = oEvent.clientX - disX+'px';
      oDiv.style.top = oEvent.clientY - disY+'px';
     };
     document.onmouseup = function() {
      document.onmousemove = null;
      document.onmouseup = null;
     };

     return false;

    };

   };
  </script>

现在程序是完整了,但是在用户体验上还有一些问题。

比如说用户可能会把这个div拖出浏览器外面,那怎么解决呢?

那我们就在加个判断呗。 这个很简单吧,如果从左边出去了

Javascript 拖拽雏形中的一些问题(逐行分析代码,让你轻松了拖拽的原理)

,那就直接等于0,他就从左边出不去了。那么上边也是一样的。

那么怎么防止不能从右边出去?? 画个图就清楚了。 其实我们只要把页面的可视取的宽度减掉div的宽度就能算出来了。

那这个就是所谓的最大值,判断一下如果移动的距离超过了这个最大值就等于这个最大值即可。那么下边是一样的。

附上完整代码:

<script type="text/javascript">
       // 拖拽空div 低版本的火狐有bug
      window.onload = function() {
        var oDiv = document.getElementById("div1");var disX = 0;
        var disY = 0;
        oDiv.onmousedown = function(ev) {
          var oEvent = ev || event;
          disX = oEvent.clientX - oDiv.offsetLeft;
          disY = oEvent.clientY - oDiv.offsetTop;

          document.onmousemove = function(ev) {
            var oEvent = ev || event;
            // 存储div当前的位置
            var oDivLeft = oEvent.clientX - disX;
            var oDivTop = oEvent.clientY - disY;


            // 从左边拖出去了
            if (oDivLeft < 0) {
              oDivLeft = 0;
            } else if (oDivLeft > document.documentElement.clientWidth - oDiv.offsetWidth) {
              oDivLeft = document.documentElement.clientWidth - oDiv.offsetWidth;
            }

            if (oDivTop < 0) {
              oDivTop = 0;
            } else if (oDivTop > document.documentElement.clientHeight - oDiv.offsetHeight) {
              oDivTop = document.documentElement.clientHeight - oDiv.offsetHeight;
            }

            oDiv.style.left = oDivLeft + 'px';
            oDiv.style.top = oDivTop + 'px';
          };

          document.onmouseup = function() {
            document.onmousemove = null;
            document.onmouseup = null;
          };
          return false; // 阻止默认事件,解决火狐的bug
        };
      };
    </script>

那么现在这个拖拽就比较完整啦。O(∩_∩)O

Javascript 相关文章推荐
用YUI做了个标签浏览效果
Feb 20 Javascript
js实现的GridView即表头固定表体有滚动条且可滚动
Feb 19 Javascript
javascript使用Promise对象实现异步编程
Mar 01 Javascript
JS弹出层遮罩,隐藏背景页面滚动条细节优化分析
Apr 29 Javascript
Javascript之图片的延迟加载的实例详解
Jul 24 Javascript
JS和jQuery通过this获取html标签中的属性值(实例代码)
Sep 11 jQuery
详细分析单线程JS执行问题
Nov 22 Javascript
React组件中的this的具体使用
Feb 28 Javascript
VUE 全局变量的几种实现方式
Aug 22 Javascript
vue鼠标悬停事件实例详解
Apr 01 Javascript
vue+layui实现select动态加载后台数据的例子
Sep 20 Javascript
vue 查看dist文件里的结构(多种方式)
Jan 17 Javascript
Javascript 拖拽雏形(逐行分析代码,让你轻松了拖拽的原理)
Jan 23 #Javascript
Javascript 拖拽的一些简单的应用(逐行分析代码,让你轻松了拖拽的原理)
Jan 23 #Javascript
Javascript 完美运动框架(逐行分析代码,让你轻松了运动的原理)
Jan 23 #Javascript
Javascript 拖拽的一些高级的应用(逐行分析代码,让你轻松了拖拽的原理)
Jan 23 #Javascript
jQuery选择器querySelector的使用指南
Jan 23 #Javascript
jQuery中DOM操作实例分析
Jan 23 #Javascript
jquery实现点击页面计算点击次数
Jan 23 #Javascript
You might like
WINDOWS服务器安装多套PHP的另类解决方案
2006/10/09 PHP
PHP更安全的密码加密机制Bcrypt详解
2017/06/18 PHP
php+mysql开发中的经验与常识小结
2019/03/25 PHP
prototype 1.5 &amp; scriptaculous 1.6.1 学习笔记
2006/09/07 Javascript
如何实现JS函数的重载
2006/09/22 Javascript
Javascript 构造函数 实例分析
2008/11/26 Javascript
nodejs中转换URL字符串与查询字符串详解
2014/11/26 NodeJs
JS基于面向对象实现的拖拽库实例
2015/09/24 Javascript
基于bootstrap插件实现autocomplete自动完成表单
2016/05/07 Javascript
javascript正则表达式中分组详解
2016/07/17 Javascript
纯JS实现可拖拽表单的简单实例
2016/09/02 Javascript
AngularJS 实现JavaScript 动画效果详解
2016/09/08 Javascript
Angular的$http的ajax的请求操作(推荐)
2017/01/10 Javascript
详解jquery validate实现表单验证 (正则表达式)
2017/01/18 Javascript
JavaScript数据结构中串的表示与应用实例
2017/04/12 Javascript
vue实现移动端图片裁剪上传功能
2020/08/18 Javascript
让axios发送表单请求形式的键值对post数据的实例
2018/08/11 Javascript
vue组件之间数据传递的方法实例分析
2019/02/12 Javascript
简谈创建React Component的几种方式
2019/06/15 Javascript
vue实现商城秒杀倒计时功能
2019/12/12 Javascript
JavaScript实现英语单词题库
2019/12/24 Javascript
微信小程序实现电子签名功能
2020/07/29 Javascript
[00:27]DOTA2战队VP、Secret贺新春
2018/02/11 DOTA
Python获取网页上图片下载地址的方法
2015/03/11 Python
Python Sql数据库增删改查操作简单封装
2016/04/18 Python
Django文件存储 自己定制存储系统解析
2019/08/02 Python
PyQt5 如何让界面和逻辑分离的方法
2020/03/24 Python
CSS3新属性transition-property transform box-shadow实例学习
2013/06/06 HTML / CSS
HTML5中FileReader接口使用方法实例详解
2017/08/26 HTML / CSS
把富文本的回车转为br标签
2019/08/09 HTML / CSS
H5离线存储Manifest原理及使用
2020/04/28 HTML / CSS
印尼最大的婴儿用品购物网站:Orami
2017/09/28 全球购物
面向对象概念面试题(.NET)
2016/11/04 面试题
购房个人委托书范本
2014/10/11 职场文书
装修公司工程部经理岗位职责
2015/04/09 职场文书
JS中如何优雅的使用async await详解
2021/10/05 Javascript