欢迎光临

我们一直在努力
当前位置:首页 > 编程技术 >

react-dnd实现任意拖动与互换位置

日期:
后台-插件-广告管理-首页/栏目/内容广告位一(PC)
后台-插件-广告管理-首页/栏目/内容广告位一(手机)

本文实例为大家分享了react-dnd实现任意拖动与互换位置的具体代码,供大家参考,具体内容如下

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 python{ useDrag } from 'react-dnd';

/**item:必填。描述了要拖动的数据 ( 包含type: 对应drop>accept )
begin(monitor): 可选的。拖动操作开始时触发。
end(item, monitor): 可选的。当拖动停止时,end被调用。
canDrag(monitor): 可选的。使用它来指定当前是否允许拖动。
isDragging(monitor): 可选的。默认情况下,只有启动拖动操作的拖动源才被视为拖动。
collect: 可选的。收集功能。它应该返回道具的简单对象以返回注入到组件中。它接收两个参数,monitor和props。
spec:必填。一个普通的JavaScript对象,上面有一些允许的方法。它描述了拖动源如何对拖放事件做出反应。**/


const Drag = ({ name }) => {
 const [{isDragging}, drag] = useDrag({
  type: 'test',
  end(item, monitor) { 
   monitor.getDropResult()  //获取拖拽对象所处容器的数据
   monitor.didDrop()  // 当前容器能否放置拖拽对象
  }  
 })

 return (
  <div id="drag">{name}</div>
 )
}
import { useDrop } from 'react-dnd';

/**
参考api
accept:必填。 (对应 drag item.type )
options: 可选的。一个普通的对象。
drop(item, monitor): 可选的。当兼容项目放在目标上时调用。
hover(item, monitor): 可选的。将项目悬停在组件上时调用。
canDrop(item, monitorpython): 可选的。使用它来指定放置目标是否能够接受该物品。如果要始终允许它,则只需忽略此方法。
collect: 可选的。收集功能。它应该返回道具的简单对象以返回注入到组件中。它接收两个参数,monitor和props
*/

const Container = (props) => {
 const [{isOver, canDrop}, drop] = useDrop({
  accept: 'test',
  drop: (item, monitor) => ({
   dropname: '测试',
   top: monitor.getDifferenceFromInitialOffset().y,
    left: monitor.getDifferenceFromInitialOffset().x
  }), 
  // 可以在这里配置数据,与drag中monitor.getDropResult()获取的信息关联
  collect: (monitor) => ({
   isOver: monitor.isOver(),  // 返回拖拽对象是否悬浮在该容器上
   canDrop: monitor.canDrop(), // 当前容器是否支持拖拽对象放置
  })
 })

 return (
  <div ref={drop}>
  ...
  </div>
 )
}

实现任意拖拽

1.在容器中通过drop —> monitor.getDifferenceFromInitialOffset()获取拖拽对象当前位置与初始位置的偏移量

2.在drag的end中,通过monitor.getDropResult()获取到与初始位置的偏移量,给当前拖拽元素的偏移量重新赋值,即可做到任意拖拽

3.如果同时有很多个可拖拽的对象,需要给他们定义一个index值, 因为这个可拖拽组件是复用的,所以我们获取到的拖拽对象是个数组,我们可以用index作为下标,给当前拖拽组件单独赋值

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}>
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}
  >js: 'yellow'}}
  >

  </div>
 )
}

拖拽互换位置

拖拽互换位置需要组件既是drag拖拽对象,也是drop拖拽容器。所以使用useRef()定义ref,然后 drag(drop(ref)),让它即作为拖拽对象也作为拖拽容器,然后定义拖拽结束事件

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}>

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

后台-插件-广告管理-首页/栏目/内容广告位二(PC)
后台-插件-广告管理-首页/栏目/内容广告位二(手机)
后台-插件-广告管理-内容广告位三(PC)
后台-插件-广告管理-内容广告位三(手机)

相关阅读

后台-插件-广告管理-内容广告位四(PC)
后台-插件-广告管理-内容广告位四(手机)

热门文章

后台-插件-广告管理-侧边广告位一(PC)
后台-插件-广告管理-侧边广告位一(手机)
  • HTML 表单组件实例代码

  • HTML 表单用于搜集不同类型的用户输入。下文通过代码给大家分享html 表单组件实例代码,感兴趣的朋友参考下吧 废话不多说了,直接给大家贴代码了,具体代码如下所示: <!DOCTYPE
  • html2canvas 将html代码转为图片的使用方法

  • 转换代码到图片使用 html2canvas,这是一个非常著名的从浏览器网页截图的开源库,使用很方便,功能也很强大。 使用 html2canvas http:// html2canvas 的使用非常简单,简单
  • HTML网页中插入视频的方法小结

  • 现在如果要在页面中使用video标签,需要考虑三种情况,支持Ogg Theora或者VP8(如果这玩意儿没出事的话)的(Opera、Mozilla、Chrome),支持H.264的(Safari、IE 9、Chrome),都不支持的(IE6、
  • HTML实现文本框只读不能修改其中的内容

  • 废话不多说了,直接给大家贴代码了,具体代码如下所示: <!--方法1:>http:// 当鼠标放不上就离开焦点 --> <input type="text" name="input1" value=http://www.cppcns.com/web
  • 移动端专用的meta标签设置大全

  • 前言 之前学习前端中,对meta标签的了解仅仅只是这一句。 <meta charset="UTF-8"> 但是打开任意的网站,其head标签内都有一列的meta标签。比如我们我们网站,但是自己却很不熟
后台-插件-广告管理-侧边广告位二(PC)
后台-插件-广告管理-侧边广告位二(手机)

最新文章

  • 在Asp.net core项目中使用WebSocket

  • 今天小试了一下在ASP.NET core中使用websocket,这里记录一下: 在 Startup 类的 Configure 方法中添加 WebSocket 中间件。 app.UseWebSockets(); 它也可以传入一些参数 app.Us
  • Vue快速理解事件绑定是什么

  • 目录一、监听事件二、事件修饰符1、stop修饰符阻止事件冒泡2、capture修饰符3、self修饰符4、prevent修饰符5、键盘事件修饰符6、鼠标事件修饰符一、监听事件 监听事件一般
  • C#实现模拟ATM自动取款机功能

  • 目录(1)关于用户帐号的类:Account(2)关于银行数据库的类:BankDatabase(3)关于ATM屏幕显示的类:Screen(4)关于ATM键盘的类:Keypad(5)关于进钞、出钞口的类:DepositSlot(6)关于ATM
  • Java设计模式之抽象工厂模式浅析讲解

  • 1.介绍 当系统准备为用户提供一系列相关对象,又不想让用户代码和这些对象形成耦合时,就可以使用抽象工厂模式。 2.如何实现 1)抽象产品--Car 2)具体产品--BYDCar、TSLCar 3)抽象
  • 如何动态替换Spring容器中的Bean

  • 目录动态替换Spring容器中的Bean原因方案实现Spring中的bean替换问题动态替换Spring容器中的Bean 原因 最近在编写单测时,发现使用 Mock 工具预定义 Service 中方法的行为特
  • C#优雅的实现INotifyPropertyChanged接口

  • INotifyPropertyChanged接口在wpF或WinFrom程序中使用还是经常用到,常用于通知界面属性变更。标准写法如下: class NotifandroidyObject : INotifyPropertyChanged {
后台-插件-广告管理-侧边广告位三(PC)
后台-插件-广告管理-侧边广告位三(手机)