内容简介:最近项目涉及到一个支持批量操作的小需求,交互上需要使用框选来触发。在查阅了一些资料后发现,网上的方案基本都是基于绝对定位布局的,此方案如果是针对全局(在body上)的框选,还是可用的。但是现实需求里几乎都是针对某个区域的框选。如果用绝对定位实现就比较繁琐了,需要调整定位原点。下面介绍一种基于Fixed定位的框选实现。首先梳理一下需要用到的事件。按住鼠标左键,因为并没有原生的鼠标左键按下事件,所以使用mousedown事件配合setTimeout模拟实现。mousedown事件绑定在当前区域上。 使用一个标志
最近项目涉及到一个支持批量操作的小需求,交互上需要使用框选来触发。在查阅了一些资料后发现,网上的方案基本都是基于绝对定位布局的,此方案如果是针对全局(在body上)的框选,还是可用的。但是现实需求里几乎都是针对某个区域的框选。如果用绝对定位实现就比较繁琐了,需要调整定位原点。下面介绍一种基于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。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持 码农网
猜你喜欢:- 认识绝对定位,相对定位
- 移动端页面头部固定定位的绝对定位实现
- webgl(three.js)实现室内定位,楼宇bim、实时定位三维可视化解决方案——第五课
- IP 地址怎么定位?
- # CSS 绝对定位释义
- 如何定位渲染耗时瓶颈
本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们。
Java从入门到精通
李钟尉、马文强、陈丹丹 / 清华大学出版社 / 2008-9 / 59.80元
《Java从入门到精通》(软件开发视频大讲堂)从初学者角度出发,通过通俗易懂的语言、丰富多彩的实例,详细介绍了使用Java语言进行程序开发应该掌握的各方面技术。全书共分28章,包括:初识Java,熟悉Eclipse开发工具,Java语言基础,流程控制,字符串,数组,类和对象,包装类,数字处理类,接口、继承与多态,类的高级特性,异常处理,Swing程序设计,集合类,I/O输入输出,反射,枚举类型与泛......一起来看看 《Java从入门到精通》 这本书的介绍吧!