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

 更新时间:2022年08月26日 15:36:22   作者:孤风随雨  
这篇文章主要为大家详细介绍了react-dnd实现任意拖动与互换位置,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

本文实例为大家分享了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 { 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, monitor): 可选的。使用它来指定放置目标是否能够接受该物品。如果要始终允许它,则只需忽略此方法。
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} 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)),让它即作为拖拽对象也作为拖拽容器,然后定义拖拽结束事件

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

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

相关文章

  • 详解React-Todos入门例子

    详解React-Todos入门例子

    本篇文章主要介绍了React-Todos入门例子,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2016-11-11
  • 使用 React Router Dom 实现路由导航的详细过程

    使用 React Router Dom 实现路由导航的详细过程

    React Router Dom 是 React 应用程序中用于处理路由的常用库,它提供了一系列组件和 API 来管理应用程序的路由,这篇文章主要介绍了使用 React Router Dom 实现路由导航,需要的朋友可以参考下
    2024-03-03
  • react如何使用mobx6动态加载数据

    react如何使用mobx6动态加载数据

    MobX是一个强大而简单的状态管理工具,它可以帮助我们更好地组织和管理React应用程序中的数据流,本文给大家介绍react如何使用mobx6动态加载数据,感兴趣的朋友跟随小编一起看看吧
    2024-02-02
  • React中10种Hook的使用介绍

    React中10种Hook的使用介绍

    Hook 是 React 16.8 的新增特性,本文主要介绍了10种Hook的使用,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-11-11
  • 基于react封装一个通用可编辑组件

    基于react封装一个通用可编辑组件

    前段时间接到这样一个需求,需要封装一个组件实现可编辑,这个到底有多通用呢,就是需要在普通的文字展示包括表格,列表等等,所以本文将给大家介绍如何基于react封装一个通用可编辑组件,需要的朋友可以参考下
    2024-02-02
  • react+ant design实现Table的增、删、改的示例代码

    react+ant design实现Table的增、删、改的示例代码

    这篇文章主要介绍了react+ant design实现Table的增、删、改的示例代码,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-12-12
  • 关于React Native报Cannot initialize a parameter of type''NSArray<id<RCTBridgeModule>>错误(解决方案)

    关于React Native报Cannot initialize a parameter of type''NSArra

    这篇文章主要介绍了关于React Native报Cannot initialize a parameter of type'NSArray<id<RCTBridgeModule>>错误,本文给大家分享解决方案,需要的朋友可以参考下
    2021-05-05
  • react搭建在线编辑html的站点通过引入grapes实现在线拖拉拽编辑html

    react搭建在线编辑html的站点通过引入grapes实现在线拖拉拽编辑html

    Grapes插件是一种用于Web开发的开源工具,可以帮助用户快速创建动态和交互式的网页元素,它还支持多语言和多浏览器,适合开发响应式网页和移动应用程序,这篇文章主要介绍了react搭建在线编辑html的站点通过引入grapes实现在线拖拉拽编辑html,需要的朋友可以参考下
    2023-08-08
  • React hooks useState异步问题及解决

    React hooks useState异步问题及解决

    这篇文章主要介绍了React hooks useState异步问题及解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-08-08
  • react项目自行配置热更新的实现

    react项目自行配置热更新的实现

    本文主要介绍了react项目自行配置热更新的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-11-11

最新评论