基于Fixed定位的框選功能的實(shí)現(xiàn)代碼
最近項(xiàng)目涉及到一個(gè)支持批量操作的小需求,交互上需要使用框選來(lái)觸發(fā)。在查閱了一些資料后發(fā)現(xiàn),網(wǎng)上的方案基本都是基于絕對(duì)定位布局的,此方案如果是針對(duì)全局(在body上)的框選,還是可用的。但是現(xiàn)實(shí)需求里幾乎都是針對(duì)某個(gè)區(qū)域的框選。如果用絕對(duì)定位實(shí)現(xiàn)就比較繁瑣了,需要調(diào)整定位原點(diǎn)。下面介紹一種基于Fixed定位的框選實(shí)現(xiàn)。
需求描述
- 按住鼠標(biāo)左鍵不放,移動(dòng)鼠標(biāo)出現(xiàn)選擇框
- 在鼠標(biāo)移動(dòng)的過(guò)程中,在框選范圍內(nèi)的元素高亮
- 松開(kāi)鼠標(biāo)左鍵,彈出編輯框,批量操作所有被框選的元素
實(shí)現(xiàn)
事件綁定
首先梳理一下需要用到的事件。
按住鼠標(biāo)左鍵,因?yàn)椴](méi)有原生的鼠標(biāo)左鍵按下事件,所以使用mousedown事件配合setTimeout模擬實(shí)現(xiàn)。mousedown事件綁定在當(dāng)前區(qū)域上。 使用一個(gè)標(biāo)志變量mouseOn來(lái)代表是否開(kāi)始繪制
handleMouseDown(e) {
// 判斷是否為鼠標(biāo)左鍵被按下
if (e.buttons !== 1 || e.which !== 1) return;
this.settimeId = window.setTimeout(() => {
this.mouseOn = true;
// 設(shè)置選框的初始位置
this.startX = e.clientX;
this.startY = e.clientY;
}, 300);
},
handleMouseUp(e) {
//在mouseup的時(shí)候清除計(jì)時(shí)器,如果按住的時(shí)間不足300毫秒
//則mouseOn為false
this.settimeId && window.clearTimeout(this.settimeId)
if (!this.mouseOn) return;
}
這里有一個(gè)小的注意點(diǎn),就是clearTimeout一定要寫(xiě)成 window.clearTimeout ,否則在vue里會(huì)報(bào)錯(cuò)timeout.close is not a function,具體的原因尚未找到,有大佬了解望告知。
鼠標(biāo)移動(dòng),使用mousemove事件。 鼠標(biāo)抬起,使用mouseup事件,注意抬起事件需要 綁定在document上 。因?yàn)橛脩舻目蜻x操作不會(huì)局限在當(dāng)前區(qū)域,在任意位置松開(kāi)鼠標(biāo)都應(yīng)能夠結(jié)束框選的繪制。
選框繪制
在明確了事件之后,就只需要在幾個(gè)事件中填充具體的繪制和判斷邏輯了。先來(lái)看繪制的邏輯。在mousedown事件中,設(shè)置選框的初始位置,也就是鼠標(biāo)按下的位置。這里我們提前寫(xiě)好一個(gè)div,用來(lái)代表選框。
<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;
}
按下后顯示這個(gè)div并且設(shè)置初始定位即可
this.$refs.select.style.cssText = `display:block;
left:${this.startX}px;
top:${this.startY}px
width:0;
height:0;`;
有了初始位置,在mousemove事件中,設(shè)置選框的寬高和定位。
handleMouseMove(e) {
if (!this.mouseOn) return;
const $select = this.$refs.select;
const _w = e.clientX - this.startX;
const _h = e.clientY - this.startY;
//框選有可能是往左框選,此時(shí)框選矩形的左上角就變成
//鼠標(biāo)移動(dòng)的位置了,所以需要判斷。同理寬高要取絕對(duì)值
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`;
},
如果使用絕對(duì)定位,就要去校準(zhǔn)坐標(biāo)原點(diǎn)了,在布局中嵌套多個(gè)relative定位容器的情況下,就非常繁瑣了。使用fixed定位就不需要考慮相對(duì)于哪個(gè)容器的問(wèn)題了。
判斷被框選的內(nèi)容
//獲取目標(biāo)元素
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
返回值是一個(gè) DOMRect 對(duì)象,這個(gè)對(duì)象是由該元素的 getClientRects() 方法返回的一組矩形的集合, 即:是與該元素相關(guān)的CSS 邊框集合 。
DOMRect 對(duì)象包含了一組用于描述邊框的只讀屬性——left、top、right和bottom,單位為像素。除了 width 和 height 外的屬性都是相對(duì)于 視口的左上角 位置而言的。
從定義中可以看到getBoundingClientRect中獲取的left、top、right和bottom是相對(duì)于視口左上角的,這和fixed定位的定義是一致的。因此,我們僅需要對(duì)比選框和被框選元素的四個(gè)定位值即可。
rect.top > bottom 被框選元素位于選框上方
rect.bottom < top 被框選元素位于選框下方
rect.right < left 被框選元素位于選框左側(cè)
rect.left > right 被框選元素位于選框右側(cè)
排除這四種情況以外就是選框和被框選元素存在交集,給這些div加上class,因?yàn)橐苿?dòng)過(guò)程中也需要讓用戶感知到被框選的元素,所以上述方法在mousemove中也要執(zhí)行。
在mouseup中判斷被框選元素后,將選框置為display:none。
參考鏈接
http://www.dhdzp.com/article/161132.htm
https://developer.mozilla.org/zh-CN/docs/Web/CSS/position
https://developer.mozilla.org/zh-CN/docs/Web/API/Element/getBoundingClientRect
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
基于JavaScript實(shí)現(xiàn)評(píng)論框展開(kāi)和隱藏功能
本文通過(guò)實(shí)例代碼給大家介紹了基于JavaScript實(shí)現(xiàn)評(píng)論框展開(kāi)和隱藏功能,感興趣的朋友參考下吧2017-08-08
javascript 手機(jī)號(hào)碼驗(yàn)證是否正確
javascript 手機(jī)號(hào)碼驗(yàn)證實(shí)現(xiàn)代碼。2009-06-06
JS中怎樣判斷undefined(比較不錯(cuò)的方法)
用servlet賦值給html頁(yè)面文本框值后,用alert來(lái)彈出這個(gè)值.結(jié)果顯示"undefined".下面為大家介紹下具體的判斷方法,大家可以參考下2014-03-03
JS 新增Cookie 取cookie值 刪除cookie 舉例詳解
cookie很實(shí)用的一個(gè)功能,可以判斷某個(gè)狀態(tài),下面與大家分享下JS 如何新增Cookie 取cookie值 刪除cookie,感興趣的朋友可以參考下2014-10-10
js中Function引用類型常見(jiàn)有用的方法和屬性詳解
在本篇文章里小編給大家整理的是關(guān)于js中Function引用類型常見(jiàn)有用的方法和屬性知識(shí)點(diǎn),有興趣的朋友們可以學(xué)習(xí)下。2019-12-12
ASP.NET jquery ajax傳遞參數(shù)的實(shí)例
下面小編就為大家?guī)?lái)一篇ASP.NET jquery ajax傳遞參數(shù)的實(shí)例。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2016-11-11
JavaScript數(shù)據(jù)結(jié)構(gòu)之?dāng)?shù)組的表示方法示例
這篇文章主要介紹了JavaScript數(shù)據(jù)結(jié)構(gòu)之?dāng)?shù)組的表示方法,從數(shù)據(jù)結(jié)構(gòu)線性表的角度分析了數(shù)組的原理并結(jié)合實(shí)例形式分析了javascript數(shù)組的定義與使用方法,需要的朋友可以參考下2017-04-04

