<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
最近發現公司的產品好幾個模組用到了拖拽功能,之前拖拽元件是通過Html5 drag Api 實現的但體驗並不是很好,順便將原來的拖拽組建稍做修改,寫一個自定義hook,方便大家使用拖拽功能。
拖拽功能原理:
1、拖拽元素通過addEventListener監聽器新增滑鼠按下,滑鼠移動,以及滑鼠擡起事件。
2、再通過getBoundingClientRect() 得到拖拽元素四周相對於可拖拽區域邊界的距離。
3、滑鼠移動時計算x軸和y軸的移動偏移量。
4、通過element.style.transform 設定元素移動。
5、每次拖拽完成後,都將此次偏移量儲存,下次再次拖拽時,可以保證位置的實時性。
程式碼開發:
useDrag.jsx
import { useEffect, useState } from "react"; /* * @drag: 新增拖拽事件的元素(支援傳入元素的drager,id,class等)必填引數 * @draggerBox: 被拖拽的整體元素(支援傳入元素的dragger,id,class等)可選引數 * @container: 可拖拽的區域(支援傳入元素的dragger,id,class等)可選引數 * @maring: 離外部元素的間隔 可選引數 */ export default function useDrag({ dragger, draggerBox = dragger, container = document.body, maring = [0, 0, 0, 0] }) { const [translateX, setTranslateX] = useState(0); // 水平方向偏移量 const [translateY, setTranslateY] = useState(0); // 垂直方向偏移量 useEffect(() => { if (!dragger) return; if (!draggerBox) return; if (!container) return; dragger = typeof dragger === 'string' ? document.querySelector(dragger) : dragger; // 根據傳入的值型別,去查詢使用元素 draggerBox = typeof draggerBox === 'string' ? document.querySelector(draggerBox) : draggerBox; container = typeof container === 'string' ? document.querySelector(container) : container; const { left: containerL, top: containerT, right: containerR, bottom: containerB } = container.getBoundingClientRect(); // 獲取可拖拽區域邊界位置 const onMouseDown = event => { const initMouseX = event.clientX; // 元素初始水平座標值 const initMouseY = event.clientY;// 元素初始垂直座標 const { left: boxL, top: boxT, right: boxR, bottom: boxB } = draggerBox.getBoundingClientRect(); // 獲取拖拽實體的位置 let deltaMouseX; // 實際水平偏移量 let deltaMouseY; // 實際垂直增量值 const onMouseMove = event => { const moveMouseX = event.clientX; // 元素移動後水平座標值 const moveMouseY = event.clientY; // 元素移動後垂直座標值 let deltaX = moveMouseX - initMouseX; // 當前移動水平相對移動距離 let deltaY = moveMouseY - initMouseY; // 當前移動垂直相對移動距離 if (boxL + deltaX < containerL + maring[0]) { // 當元素左邊框+水平相對移動距離 < 可拖拽區域左邊界+左邊距時 (說明元素已經超出左側邊界) deltaX = containerL + maring[0] - boxL; } if (boxR + deltaX > containerR - maring[1]) { // 當元素右邊框+水平相對移動距離 > 可拖拽區域右邊界+右邊距時 (說明元素已經超出右側邊界) deltaX = containerR - maring[1] - boxR; } if (boxB + deltaY > containerB - maring[2]) { // 當元素下邊框+垂直相對移動距離 > 可拖拽區域下邊界+下邊距時 (說明元素已經超出下側邊界) deltaY = containerB - maring[2] - boxB; } if (boxT + deltaY < containerT + maring[3]) { // 當元素上邊框+垂直相對移動距離 < 可拖拽區域上邊界+上邊距時 (說明元素已經超出上側邊界) deltaY = containerT + maring[3] - boxT } deltaMouseX = deltaX + translateX; // 實際水平偏移量 deltaMouseY = deltaY + translateY; // 實際垂直偏移量 draggerBox.style.transform = `translate(${deltaMouseX}px, ${deltaMouseY}px)`; }; const onMouseUp = () => { setTranslateX(deltaMouseX); // 儲存上次水平偏移量 setTranslateY(deltaMouseY); // 儲存上次垂直偏移量 window.removeEventListener('mousemove', onMouseMove); window.removeEventListener('mouseup', onMouseUp); } window.addEventListener('mousemove', onMouseMove); window.addEventListener('mouseup', onMouseUp); } dragger.addEventListener('mousedown', onMouseDown); return () => dragger.removeEventListener('mouseup', onMouseDown); }, [dragger, draggerBox, container, maring]) }
使用方法:
import React from 'react'; import useDrag from '../hooks/useDrag' import './index.less'; function Test() { useDrag({ dragger: '.dragger', draggerBox: '.draggerBox', container: '.container', maring: [10, 10, 10, 10]}) return ( <div className='container'> <div className='draggerBox'> <div className='dragger'> </div> </div> </div> ) } export default Test;
.container{ width: 800px; height: 800px; position: absolute; top: 50%; left: 50%; margin: -400px 0 0 -400px; border: 2px solid green; } .draggerBox{ width: 200px; height: 300px; border: 1px solid red; } .dragger{ width: 100%; height: 50px; background: blue; }
效果展示
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支援it145.com。
相關文章
<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
综合看Anker超能充系列的性价比很高,并且与不仅和iPhone12/苹果<em>Mac</em>Book很配,而且适合多设备充电需求的日常使用或差旅场景,不管是安卓还是Switch同样也能用得上它,希望这次分享能给准备购入充电器的小伙伴们有所
2021-06-01 09:31:42
除了L4WUDU与吴亦凡已经多次共事,成为了明面上的厂牌成员,吴亦凡还曾带领20XXCLUB全队参加2020年的一场音乐节,这也是20XXCLUB首次全员合照,王嗣尧Turbo、陈彦希Regi、<em>Mac</em> Ova Seas、林渝植等人全部出场。然而让
2021-06-01 09:31:34
目前应用IPFS的机构:1 谷歌<em>浏览器</em>支持IPFS分布式协议 2 万维网 (历史档案博物馆)数据库 3 火狐<em>浏览器</em>支持 IPFS分布式协议 4 EOS 等数字货币数据存储 5 美国国会图书馆,历史资料永久保存在 IPFS 6 加
2021-06-01 09:31:24
开拓者的车机是兼容苹果和<em>安卓</em>,虽然我不怎么用,但确实兼顾了我家人的很多需求:副驾的门板还配有解锁开关,有的时候老婆开车,下车的时候偶尔会忘记解锁,我在副驾驶可以自己开门:第二排设计很好,不仅配置了一个很大的
2021-06-01 09:30:48
不仅是<em>安卓</em>手机,苹果手机的降价力度也是前所未有了,iPhone12也“跳水价”了,发布价是6799元,如今已经跌至5308元,降价幅度超过1400元,最新定价确认了。iPhone12是苹果首款5G手机,同时也是全球首款5nm芯片的智能机,它
2021-06-01 09:30:45