<span id="mktg5"></span>

<i id="mktg5"><meter id="mktg5"></meter></i>

        <label id="mktg5"><meter id="mktg5"></meter></label>
        最新文章專題視頻專題問答1問答10問答100問答1000問答2000關鍵字專題1關鍵字專題50關鍵字專題500關鍵字專題1500TAG最新視頻文章推薦1 推薦3 推薦5 推薦7 推薦9 推薦11 推薦13 推薦15 推薦17 推薦19 推薦21 推薦23 推薦25 推薦27 推薦29 推薦31 推薦33 推薦35 推薦37視頻文章20視頻文章30視頻文章40視頻文章50視頻文章60 視頻文章70視頻文章80視頻文章90視頻文章100視頻文章120視頻文章140 視頻2關鍵字專題關鍵字專題tag2tag3文章專題文章專題2文章索引1文章索引2文章索引3文章索引4文章索引5123456789101112131415文章專題3
        當前位置: 首頁 - 科技 - 知識百科 - 正文

        基于Fixed定位的框選功能的實現代碼

        來源:懂視網 責編:小采 時間:2020-11-27 21:57:06
        文檔

        基于Fixed定位的框選功能的實現代碼

        基于Fixed定位的框選功能的實現代碼:最近項目涉及到一個支持批量操作的小需求,交互上需要使用框選來觸發。在查閱了一些資料后發現,網上的方案基本都是基于絕對定位布局的,此方案如果是針對全局(在body上)的框選,還是可用的。但是現實需求里幾乎都是針對某個區域的框選。如果用絕對定位實現
        推薦度:
        導讀基于Fixed定位的框選功能的實現代碼:最近項目涉及到一個支持批量操作的小需求,交互上需要使用框選來觸發。在查閱了一些資料后發現,網上的方案基本都是基于絕對定位布局的,此方案如果是針對全局(在body上)的框選,還是可用的。但是現實需求里幾乎都是針對某個區域的框選。如果用絕對定位實現

        最近項目涉及到一個支持批量操作的小需求,交互上需要使用框選來觸發。在查閱了一些資料后發現,網上的方案基本都是基于絕對定位布局的,此方案如果是針對全局(在body上)的框選,還是可用的。但是現實需求里幾乎都是針對某個區域的框選。如果用絕對定位實現就比較繁瑣了,需要調整定位原點。下面介紹一種基于Fixed定位的框選實現。

        需求描述

         

      1. 按住鼠標左鍵不放,移動鼠標出現選擇框
      2. 在鼠標移動的過程中,在框選范圍內的元素高亮
      3. 松開鼠標左鍵,彈出編輯框,批量操作所有被框選的元素
      4. 實現

        事件綁定

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

        按住鼠標左鍵,因為并沒有原生的鼠標左鍵按下事件,所以使用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。

        功能demo地址

        參考鏈接

        https://www.gxlcms.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

        聲明:本網頁內容旨在傳播知識,若有侵權等問題請及時與本網聯系,我們將在第一時間刪除處理。TEL:177 7030 7066 E-MAIL:11247931@qq.com

        文檔

        基于Fixed定位的框選功能的實現代碼

        基于Fixed定位的框選功能的實現代碼:最近項目涉及到一個支持批量操作的小需求,交互上需要使用框選來觸發。在查閱了一些資料后發現,網上的方案基本都是基于絕對定位布局的,此方案如果是針對全局(在body上)的框選,還是可用的。但是現實需求里幾乎都是針對某個區域的框選。如果用絕對定位實現
        推薦度:
        標簽: 定位 實現 代碼
        • 熱門焦點

        最新推薦

        猜你喜歡

        熱門推薦

        專題
        Top 主站蜘蛛池模板: 亚洲国产精品张柏芝在线观看| 免费国产在线观看| 亚洲不卡av不卡一区二区| 免费无码专区毛片高潮喷水 | 亚洲无人区午夜福利码高清完整版| 国产在亚洲线视频观看| 国产在线19禁免费观看| 毛片免费全部播放一级| 亚洲国产综合在线| 国产精品视频免费一区二区| 久久精品国产亚洲av麻豆蜜芽 | 国产高潮流白浆喷水免费A片 | 亚洲Av无码国产情品久久| 亚洲国产理论片在线播放| 99久久久精品免费观看国产| 亚洲国产日韩视频观看| 国产精品久免费的黄网站| 日韩在线观看视频免费| 亚洲五月综合缴情在线观看| 24小时在线免费视频| 亚洲人成77777在线播放网站不卡| 搡女人真爽免费视频大全| 在线亚洲精品视频| 亚洲中文字幕无码永久在线| 无码人妻丰满熟妇区免费| 亚洲av日韩av无码av| 免费a级黄色毛片| 国产情侣久久久久aⅴ免费| 亚洲精品影院久久久久久| 免费无码看av的网站| 人人公开免费超级碰碰碰视频| 亚洲av日韩av无码| 女性无套免费网站在线看| fc2免费人成在线视频| 亚洲视频一区调教| 国产资源免费观看| 久久免费高清视频| 亚洲人成网站在线在线观看| 无码永久免费AV网站| 一二三区免费视频| 高清在线亚洲精品国产二区|