react-dnd實現(xiàn)任意拖動與互換位置
本文實例為大家分享了react-dnd實現(xiàn)任意拖動與互換位置的具體代碼,供大家參考,具體內(nèi)容如下
react-dnd用法
hooks組件
1.使用DndProvider定義一個可以拖拽的范圍
import { HTML5Backend } from 'react-dnd-html5-backend';
import { DndProvider } from 'react-dnd';
class App extends Component{
? render () {
? ? return (
? ? ? <div>
? ? ? ? <DndProvider backend={HTML5Backend}>
? ? ? ? ? <Container />
? ? ? ? </DndProvider>
? ? ? </div>
? ? )
? }
}2.定義drag和drop
drag:就是可以被拖拽的東西
drop: 就是拖拽后可以放的地方
import { useDrag } from 'react-dnd';
/**item:必填。描述了要拖動的數(shù)據(jù) ( 包含type: 對應(yīng)drop>accept )
begin(monitor): 可選的。拖動操作開始時觸發(fā)。
end(item, monitor): 可選的。當(dāng)拖動停止時,end被調(diào)用。
canDrag(monitor): 可選的。使用它來指定當(dāng)前是否允許拖動。
isDragging(monitor): 可選的。默認情況下,只有啟動拖動操作的拖動源才被視為拖動。
collect: 可選的。收集功能。它應(yīng)該返回道具的簡單對象以返回注入到組件中。它接收兩個參數(shù),monitor和props。
spec:必填。一個普通的JavaScript對象,上面有一些允許的方法。它描述了拖動源如何對拖放事件做出反應(yīng)。**/
const Drag = ({ name }) => {
? const [{isDragging}, drag] = useDrag({
? ? type: 'test',
? ? end(item, monitor) { ??
? ? ?? ?monitor.getDropResult() ? //獲取拖拽對象所處容器的數(shù)據(jù)
? ? ?? ?monitor.didDrop() ? ?// 當(dāng)前容器能否放置拖拽對象
? ? } ? ?
? })
? return (
? ? <div id="drag">{name}</div>
? )
}import { useDrop } from 'react-dnd';
/**
參考api?
accept:必填。 (對應(yīng) drag item.type )
options: 可選的。一個普通的對象。
drop(item, monitor): 可選的。當(dāng)兼容項目放在目標(biāo)上時調(diào)用。
hover(item, monitor): 可選的。將項目懸停在組件上時調(diào)用。
canDrop(item, monitor): 可選的。使用它來指定放置目標(biāo)是否能夠接受該物品。如果要始終允許它,則只需忽略此方法。
collect: 可選的。收集功能。它應(yīng)該返回道具的簡單對象以返回注入到組件中。它接收兩個參數(shù),monitor和props
*/
const Container = (props) => {
? const [{isOver, canDrop}, drop] = useDrop({
? ? accept: 'test',
? ? drop: (item, monitor) => ({?
? ? ?? ?dropname: '測試',?
? ? ?? ?top: monitor.getDifferenceFromInitialOffset().y,
? ? ? ? left: monitor.getDifferenceFromInitialOffset().x?
? ? ?}), ?
? ? // 可以在這里配置數(shù)據(jù),與drag中monitor.getDropResult()獲取的信息關(guān)聯(lián)
? ? collect: (monitor) => ({
? ? ? isOver: monitor.isOver(), ? // 返回拖拽對象是否懸浮在該容器上
? ? ? canDrop: monitor.canDrop(), ?// 當(dāng)前容器是否支持拖拽對象放置
? ? })
? })
? return (
? ? <div ref={drop}>
?? ??? ?...
? ? </div>
? )
}實現(xiàn)任意拖拽
1.在容器中通過drop —> monitor.getDifferenceFromInitialOffset()獲取拖拽對象當(dāng)前位置與初始位置的偏移量
2.在drag的end中,通過monitor.getDropResult()獲取到與初始位置的偏移量,給當(dāng)前拖拽元素的偏移量重新賦值,即可做到任意拖拽
3.如果同時有很多個可拖拽的對象,需要給他們定義一個index值, 因為這個可拖拽組件是復(fù)用的,所以我們獲取到的拖拽對象是個數(shù)組,我們可以用index作為下標(biāo),給當(dāng)前拖拽組件單獨賦值
const Drag = ({ name, top, left, index }) => {
? const [{isDragging}, drag] = useDrag({
? ? type: 'test',
? ? end(item, monitor) {
? ? ? console.log(item);
? ? ? if(monitor.didDrop()){
? ? ? ? const droptarget = monitor.getDropResult();
? ? ? ? const top = document.querySelectorAll('#drag')[index].offsetTop;
? ? ? ? const left = document.querySelectorAll('#drag')[index].offsetLeft;
? ? ? ? document.querySelectorAll('#drag')[index].style.top = (top + droptarget.top) + 'px';
? ? ? ? document.querySelectorAll('#drag')[index].style.left = (left + droptarget.left) + 'px';
? ? ? }else{
? ? ? ? console.log(monitor.getDropResult());
? ? ? }
? ? }
? })
? return (
? ? <div id="drag" index={index} ref={drag} style={{position: 'absolute', top: `${top}`, left: `${left}`, width: '70px', height: '40px', border: '1px solid black'}}>{name}</div>
? )
}const Container = (props) => {
? const [{isOver, canDrop}, drop] = useDrop({
? ? accept: 'test',
? ? drop: (item, monitor) => ({ dropname: '測試', top: monitor.getDifferenceFromInitialOffset().y, left: monitor.getDifferenceFromInitialOffset().x }),
? ? collect: (monitor) => ({
? ? ? isOver: monitor.isOver(),
? ? ? canDrop: monitor.canDrop(),
? ? })
? })
? return (
? ? <div
? ? ? id="drop1"
? ? ? ref={drop}
? ? ? style={{width: '700px', height: '600px', backgroundColor: 'yellow'}}
? ? >
? ? </div>
? )
}
拖拽互換位置
拖拽互換位置需要組件既是drag拖拽對象,也是drop拖拽容器。所以使用useRef()定義ref,然后 drag(drop(ref)),讓它即作為拖拽對象也作為拖拽容器,然后定義拖拽結(jié)束事件
import React, { useRef, useMemo } from 'react';
import { useDrag, useDrop } from 'react-dnd';
import { item } from './itemType';
const style = {
? display: 'inline',
? border: '1px dashed gray',
? padding: '0.5rem 1rem',
? marginRight: '30px',
? marginTop: '.5rem',
? backgroundColor: 'blue',
? cursor: 'move',
? borderRadius: '50%',
? position: 'relative',
};
const Drag = (props) => {
? const ref = useRef()
? const [{}, drop] = useDrop({
? ? accept: item.type,
? ? drop: (item, monitor) => {
? ? ? ? return {
? ? ? ? ? index: props.index
? ? ? ? }
? ? }
? })
? const count = useMemo(() => {
? ? document.getElementById('delete').oncontextmenu = (e) => {
? ? ? e.preventDefault();
? ? }
? },[1])
? const [{}, drag] = useDrag({
? ? type: item.type,
? ? end: (item, monitor) => {
? ? ? console.log(monitor.didDrop());
? ? ? if(monitor.didDrop()){
? ? ? ? const dropResult = monitor.getDropResult();
? ? ? ? props.changePosition([props.index, dropResult.index])
? ? ? }else{
? ? ? ? return;
? ? ? }
? ? }
? })
? drag(drop(ref))
? return (
? ? <div id="delete" ref={ref} id="drag" index={props.index} style={{...style}} >
? ? ? <span>{props.name}</span>
? ? ? <span style={{position: 'absolute', border: '1px solid gray', top: "18.5px", left: '145px', width: '29px',background: 'gray'}}></span>
? ? </div>
? )
}
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
react使用axios進行api網(wǎng)絡(luò)請求的封裝方法詳解
這篇文章主要為大家詳細介紹了react使用axios進行api網(wǎng)絡(luò)請求的封裝方法,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助2022-03-03
React找不到模塊“./index.module.scss”或其相應(yīng)的類型聲明及解決方法
這篇文章主要介紹了React找不到模塊“./index.module.scss”或其相應(yīng)的類型聲明及解決方法,本文給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2023-09-09
react-router?重新加回跳轉(zhuǎn)攔截功能詳解
這篇文章主要為大家介紹了react-router?重新加回跳轉(zhuǎn)攔截功能詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-02-02

