基于Fixed定位的框选功能的实现代码


Posted in Javascript onMay 13, 2019

最近项目涉及到一个支持批量操作的小需求,交互上需要使用框选来触发。在查阅了一些资料后发现,网上的方案基本都是基于绝对定位布局的,此方案如果是针对全局(在body上)的框选,还是可用的。但是现实需求里几乎都是针对某个区域的框选。如果用绝对定位实现就比较繁琐了,需要调整定位原点。下面介绍一种基于Fixed定位的框选实现。

需求描述

基于Fixed定位的框选功能的实现代码 

  • 按住鼠标左键不放,移动鼠标出现选择框
  • 在鼠标移动的过程中,在框选范围内的元素高亮
  • 松开鼠标左键,弹出编辑框,批量操作所有被框选的元素

实现

事件绑定

首先梳理一下需要用到的事件。

按住鼠标左键,因为并没有原生的鼠标左键按下事件,所以使用mousedown事件配合setTimeout模拟实现。mousedown事件绑定在当前区域上。 使用一个标志变量mouseOn来代表是否开始绘制

handleMouseDown(e) {
 // 判断是否为鼠标左键被按下
 if (e.buttons !== 1 || e.which !== 1) return;
 this.settimeId = window.setTimeout(() => {
  this.mouseOn = true;
  // 设置选框的初始位置
  this.startX = e.clientX;
  this.startY = e.clientY;
 }, 300);
},
handleMouseUp(e) {
 //在mouseup的时候清除计时器,如果按住的时间不足300毫秒
 //则mouseOn为false
 this.settimeId && window.clearTimeout(this.settimeId)
 if (!this.mouseOn) return;
}

这里有一个小的注意点,就是clearTimeout一定要写成 window.clearTimeout ,否则在vue里会报错timeout.close is not a function,具体的原因尚未找到,有大佬了解望告知。

鼠标移动,使用mousemove事件。 鼠标抬起,使用mouseup事件,注意抬起事件需要 绑定在document上 。因为用户的框选操作不会局限在当前区域,在任意位置松开鼠标都应能够结束框选的绘制。

选框绘制

在明确了事件之后,就只需要在几个事件中填充具体的绘制和判断逻辑了。先来看绘制的逻辑。在mousedown事件中,设置选框的初始位置,也就是鼠标按下的位置。这里我们提前写好一个div,用来代表选框。

<div class="promotion-range__select" ref="select"></div>
.promotion-range__select {
 background: #598fe6;
 position: fixed;
 width: 0;
 height: 0;
 display: none;
 top: 0;
 left: 0;
 opacity:.6;
 pointer-events: none;
}

按下后显示这个div并且设置初始定位即可

this.$refs.select.style.cssText = `display:block;
                  left:${this.startX}px;
                  top:${this.startY}px
                  width:0;
                  height:0;`;

有了初始位置,在mousemove事件中,设置选框的宽高和定位。

handleMouseMove(e) {
 if (!this.mouseOn) return;
 const $select = this.$refs.select;
 const _w = e.clientX - this.startX;
 const _h = e.clientY - this.startY;
 //框选有可能是往左框选,此时框选矩形的左上角就变成
 //鼠标移动的位置了,所以需要判断。同理宽高要取绝对值
 this.top = _h > 0 ? this.startY : e.clientY;
 this.left = _w > 0 ? this.startX : e.clientX;
 this.width = Math.abs(_w);
 this.height = Math.abs(_h);
 $select.style.left = `${this.left}px`;
 $select.style.top = `${this.top}px`;
 $select.style.width = `${this.width}px`;
 $select.style.height = `${this.height}px`;
},

如果使用绝对定位,就要去校准坐标原点了,在布局中嵌套多个relative定位容器的情况下,就非常繁琐了。使用fixed定位就不需要考虑相对于哪个容器的问题了。

判断被框选的内容

//获取目标元素
const selList = document.getElementsByClassName(
 "promotion-range__item-inner"
);
const { bottom, left, right, top } = $select.getBoundingClientRect();
for (let i = 0; i < selList.length; i++) {
 const rect = selList[i].getBoundingClientRect();
 const isIntersect = !(
  rect.top > bottom ||
  rect.bottom < top ||
  rect.right < left ||
  rect.left > right
 );
 selList[i].classList[isIntersect ? "add" : "remove"]("is-editing");
}

判断使用了getBoundingClientRect,定义引用自MDN

返回值是一个 DOMRect 对象,这个对象是由该元素的 getClientRects() 方法返回的一组矩形的集合, 即:是与该元素相关的CSS 边框集合 。

DOMRect 对象包含了一组用于描述边框的只读属性——left、top、right和bottom,单位为像素。除了 width 和 height 外的属性都是相对于 视口的左上角 位置而言的。

从定义中可以看到getBoundingClientRect中获取的left、top、right和bottom是相对于视口左上角的,这和fixed定位的定义是一致的。因此,我们仅需要对比选框和被框选元素的四个定位值即可。

rect.top > bottom 被框选元素位于选框上方

rect.bottom < top 被框选元素位于选框下方

rect.right < left 被框选元素位于选框左侧

rect.left > right 被框选元素位于选框右侧

排除这四种情况以外就是选框和被框选元素存在交集,给这些div加上class,因为移动过程中也需要让用户感知到被框选的元素,所以上述方法在mousemove中也要执行。

在mouseup中判断被框选元素后,将选框置为display:none。

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

Javascript 相关文章推荐
js获取单选框或复选框值及操作
Dec 18 Javascript
JS实现可改变列宽的table实例
Jul 02 Javascript
简介JavaScript中strike()方法的使用
Jun 08 Javascript
javascript实现uploadify上传格式以及个数限制
Nov 23 Javascript
Bootstrap源码解读导航(6)
Dec 23 Javascript
JavaScript Base64 作为文件上传的实例代码解析
Feb 14 Javascript
微信小程序中实现一对多发消息详解及实例代码
Feb 14 Javascript
jQuery滚动监听实现商城楼梯式导航效果
Mar 06 Javascript
element-ui循环显示radio控件信息的方法
Aug 24 Javascript
vue动态绑定class的几种常用方式小结
May 21 Javascript
Vue 中 a标签上href无法跳转的解决方式
Nov 12 Javascript
Vue项目中如何封装axios(统一管理http请求)
May 02 Vue.js
JQuery事件委托原理与用法实例分析
May 13 #jQuery
使用原生js编写一个简单的框选功能方法
May 13 #Javascript
Node.js实现一个HTTP服务器的方法示例
May 13 #Javascript
jQuery事件绑定和解绑、事件冒泡与阻止事件冒泡及弹出应用示例
May 13 #jQuery
Vue表单绑定的实例代码(单选按钮,选择框(单选时,多选时,用 v-for 渲染的动态选项)
May 13 #Javascript
JavaScript封闭函数及常用内置对象示例
May 13 #Javascript
vue 表单之通过v-model绑定单选按钮radio
May 13 #Javascript
You might like
PHP spl_autoload_register实现自动加载研究
2011/12/06 PHP
PHP乱码问题,UTF-8乱码常见问题小结
2012/04/09 PHP
基于PHP文件操作的详细诠释
2013/06/21 PHP
PHP对文件夹递归执行chmod命令的方法
2015/06/19 PHP
54个提高PHP程序运行效率的方法
2015/07/19 PHP
JS 动态获取节点代码innerHTML分析 [IE,FF]
2009/11/30 Javascript
Extjs学习笔记之五 一个小细节renderTo和applyTo的区别
2010/01/07 Javascript
JQuery AJAX提交中文乱码的解决方案
2010/07/02 Javascript
ASP.NET jQuery 实例5 (显示CheckBoxList成员选中的内容)
2012/01/13 Javascript
JS+CSS实现的拖动分页效果实例
2015/05/11 Javascript
JavaScript数组方法大全(推荐)
2016/07/05 Javascript
客户端(vue框架)与服务器(koa框架)通信及服务器跨域配置详解
2017/08/26 Javascript
深入了解JavaScript代码覆盖
2019/06/13 Javascript
vue实现直播间点赞飘心效果的示例代码
2019/09/20 Javascript
原生JS实现贪吃蛇小游戏
2020/03/09 Javascript
JS实现多选框的操作
2020/06/24 Javascript
js前端对于大量数据的展示方式及处理方法
2020/12/02 Javascript
Python多进程同步Lock、Semaphore、Event实例
2014/11/21 Python
Python的Django框架中forms表单类的使用方法详解
2016/06/21 Python
python实现数独游戏 java简单实现数独游戏
2018/03/30 Python
用python处理MS Word的实例讲解
2018/05/08 Python
使用sklearn之LabelEncoder将Label标准化的方法
2018/07/11 Python
python使用turtle绘制国际象棋棋盘
2019/05/23 Python
Python程序暂停的正常处理方法
2019/11/07 Python
TensorFlow查看输入节点和输出节点名称方式
2020/01/04 Python
基于Python共轭梯度法与最速下降法之间的对比
2020/04/02 Python
Python如何把十进制数转换成ip地址
2020/05/25 Python
Python pytesseract验证码识别库用法解析
2020/06/29 Python
HTML5 CSS3打造相册效果附源码下载
2014/06/16 HTML / CSS
在网络中有两台主机A和B,并通过路由器和其他交换设备连接起来,已经确认物理连接正确无误,怎么来测试这两台机器是否连通?如果不通,怎么来判断故障点?怎么排
2014/01/13 面试题
毕业生简单求职信
2013/11/19 职场文书
应聘自荐信
2013/12/14 职场文书
2014镇党委班子对照检查材料思想汇报
2014/09/23 职场文书
2014年绿化工作总结
2014/12/09 职场文书
华山导游词
2015/02/03 职场文书
2015年社区党建工作汇报材料
2015/06/25 职场文书