js拖拽——将鼠标事件响应范围扩大到整个系统桌面

起因:

最近在做一个可拖拽的拓扑图,遇到的这个问题:如果执行拖拽操作的时候鼠标拖动很快,可能会出现鼠标脱离页面乃至浏览器的范围,如果这时松开鼠标按键,那么将不能响应鼠标的onmouseup事件,从而导致onmouseup和onmousemove事件不能被释放掉。

对于这个问题,网上很多相似的文章写了解决办法,但都有些毛病,我在这里总结归纳一下。

解决办法:

方案一:针对ie和火狐浏览器

ie浏览器和Firefox提供了setCapture 和 releaseCapture函数来解决该问题。简单写个小例子:


 1 /*...html部分略...*/
2
3 var i = 0;
4 div1.onmousedown = function() {
5 div1.setCapture(); //用dom元素调用该函数相当于将该dom元素上的事件交给操作系统去维护,从而让该dom元素上的事件响应不依赖于浏览器
6 div1.onmousemove = function() {
7 console.log(i++);
8 }
9 div1.onmouseup = function() {
10 div1.onmouseup = null;
11 div1.onmousemove = null;
12 div1.releaseCapture(); //拖拽结束后要释放掉
13 }
14 }

方案二:针对chrome浏览器

悲催的是chrome浏览器并没有提供上述函数的实现,那只能另辟蹊径了。方法就是在window对象上注册onmousedown、onmousemove、onmouseup事件。这样只要在窗口内触发onmousedown事件之后,无论您鼠标移动到什么地方,都能onmousemove和onmouseup事件。至于您要对哪个具体的dom元素进行操作,您可以在onmousedown事件的回调函数中通过event.target获取目标元素对象,然后再用call去执行相应函数即可。例子如下:


 1 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
2 <html xmlns="http://www.w3.org/1999/xhtml">
3 <head>
4 <meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
5 <title></title>
6 </head>
7
8 <body>
9 <div id="div1" style="width:100px; height:100px; border:1px solid red">
10
11 </div>
12
13 </body>
14 </html>
15 <script>
16 var i = 0;
17 window.onmousedown= function(event) {
18 var targetObj = event.target;
19 if (targetObj.id == "div1") { //只有在id为div1的元素上按下鼠标左键才能触发鼠标移动事件
20 window.onmousemove = function() {
21 div1Move.call(targetObj);
22 }
23 window.onmouseup= function() {
24 window.onmouseup = null;
25 window.onmousemove = null;
26 }
27 }
28 }
29
30 //鼠标移动时执行的事件
31 function div1Move() {
32 console.log(i++);
33 }
34 </script>

注意:1、该解决方案不适用于ie8及以下浏览器。因为这些浏览器不支持在window对象上注册事件。
2、该解决方案也不适用于iframe,您可以把上面的例子放到iframe中去试一试。原因大概是因为系统只维护最顶层window上的事件。

总结:结合方案一和方案二可以解决浏览器的兼容性问题,但是由于方案二对于iframe无效,所以在实现有拖拽需求的页面时,只能不用iframe了。

百度地图和谷歌地图都实现了浏览器外响应拖拽事件的效果,不知道他们是否是通过上述方法实现的。还请大神留言相告,谢谢。

时间: 2024-10-14 23:16:14

js拖拽——将鼠标事件响应范围扩大到整个系统桌面的相关文章

React.js实现原生js拖拽效果及思考

一.起因&思路 不知不觉,已经好几天没写博客了...近来除了研究React,还做了公司官网... 一直想写一个原生js拖拽效果,又加上近来学react学得比较嗨.所以就用react来实现这个拖拽效果. 首先,其实拖拽效果的思路是很简单的.主要就是三个步骤: 1.onmousedown的时候,启动可拖拽事件,记录被拖拽元素的原始坐标参数. 2.onmousemove的时候,实时记录鼠标移动的距离,结合被拖拽元素第一阶段的坐标参数,计算并设置新的坐标值. 3.onmouseup的时候,关闭可拖拽事件

js拖拽效果实现

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"><html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en"><head>    <meta

一步一步实现JS拖拽插件

js拖拽是常见的网页效果,本文将从零开始实现一个简单的js插件. 一.js拖拽插件的原理 常见的拖拽操作是什么样的呢?整过过程大概有下面几个步骤: 1.用鼠标点击被拖拽的元素 2.按住鼠标不放,移动鼠标 3.拖拽元素到一定位置,放开鼠标 这里的过程涉及到三个dom事件:onmousedown,onmousemove,onmouseup.所以拖拽的基本思路就是: 1.用鼠标点击被拖拽的元素触发onmousedown (1)设置当前元素的可拖拽为true,表示可以拖拽 (2)记录当前鼠标的坐标x,y

再谈React.js实现原生js拖拽效果

前几天写的那个拖拽,自己留下的疑问...这次在热心博友的提示下又修正了一些小小的bug,也加了拖拽的边缘检测部分...就再聊聊拖拽吧 一.不要直接操作dom元素 react中使用了虚拟dom的概念,目地就是要尽量避免直接操作dom元素,所以我们在对dom元素进行操作的时候需要注意,我之前为了获取form的参数就直接用了var dragBox=document.getElementById('form')去找dom,但是其实记录from的初始位置,可以在其子组件更新父组件参数的时候调用.即在MyF

js拖拽分析

js拖拽分析 思路 1.三个鼠标事件,mousedown,mousemove,mouseup 2.可移动性absolute 3.边界限制 得到鼠标点击处和div边界的距离,然后得出top 和 left 的值 具体 mousedown div.onmousedown=function(event){ var event=event||window.event var diffX=event.clientX-div.offsetLeft var diffY=event.clientY-div.off

JS拖拽

拖拽原理    a, 被拖拽物体绝对定位    b, 按下时记录鼠标在拖拽物体上的位置,拖拽时鼠标在拖拽物体上的位置保持不变    c, 鼠标移动时改变拖拽物体位置 1,简易拖拽 <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title>拖拽原型</title> <script type="text/javascript" sr

理解事件捕获。在限制范围内拖拽div+吸附+事件捕获

一.实现的效果是在限制范围内拖拽div+吸附+事件捕获. 这里需要理解的是事件捕获,这个事件捕获也是为了兼容div在拖拽过程中,文本不被选中这个问题. 如此良辰美景,拖拽也可以很洒脱哈.先看看图, 二.一步步的实现这个拖拽过程的几个要求 (一)拖拽起来 里面的边框是表示页面哦(我们的屏幕所能看到的东东). 获取移动距离的思路: 记录鼠标按下和鼠标抬起两次的坐标,然后相减,再加上div跟边缘之间的间距.就得到移动距离. 之前我也在这里困惑了,不明白为什么还要再加上offsetLeft.原因就是cl

DuiLib事件分析(一)——鼠标事件响应

最近在处理DuiLib中自定义列表行元素事件,因为处理方案得不到较好的效果,于是只好一层一层的去剥离DuiLib事件是怎么来的,看能否在某一层截取消息,自己重写. 我这里使用CListContainerElementUI行元素,元素中有插入button,平时行元素不显示,鼠标移动上去显示出来,鼠标移走就隐藏button.Duilib自己是不带这个功能的,它有一个鼠标移动上去的热点事件,按理说重写热点事件就好了.但是当时比较急没找到怎么触发的,之后一直没继续走这条思路.后来找到源码事件里面有 vo

php+mysql+js拖拽div实例

php+mysql+js拖拽div实例 代码已改好,放在目录下可直接使用.适合新手学习! 效果如下图,比较适合做可以移动,拖拽的留言板. 实现思路: 思路也是很简单了,就是js获取定位后的数据,然后请求给PHP,php将定位的横向坐标和纵向坐标存到数据库! 代码实例下载地址:http://download.csdn.net/detail/u011986449/8099045