react按钮拖拽_React 实现拖拽功能的示例代码

本文详细介绍了如何在React中实现拖拽功能,通过示例代码展示了如何处理ondragstart、ondragend、ondragenter、ondragover、ondragleave和ondrop等事件,以及如何使用dataTransfer对象进行数据传递。示例包括拖拽元素和放置元素的事件处理,适用于如卡片、按钮等元素的拖动操作。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

本文介绍了React 实现拖拽功能的示例代码,分享给大家,具体如下:

实现效果:

因为工作中会用到 JIRA 所以想实现一下相似的功能,顺便学习一下 H5 的拖拽。不支持拖拽改变顺序,感觉有点麻烦,而且没必要。感觉相关的博文好少的,大部分都是直接上代码,没有解释。

图片默认可以拖动,其他元素的拖动效果同图片。正常的 div 是不能被拖动的,鼠标点击选择后移动没有效果,需要加  draggable="true" 使得元素可以被拖动。

拖拽相关的几个事件,有被拖动元素的事件,也有拖动进入的容器元素的事件。

被拖拽元素的事件:ondragstart,ondragend

放置元素的事件:ondragenter、ondragover、ondragleave、ondrop

顾名思义,不需要解释。

需要注意是  ondragover 的默认事件 Reset the current drag operation to "none". 所以想让一个元素可放置,需要重写 ondragover

element.ondragover = event => {

event.preventDefault();

// ...

}

当一个元素是可放置的,拖拽经过时鼠标会变成加号(cursor: copy;)

有一个对象  dataTransfer 可以用来存储拖拽数据。

dragEle.ondragstart = e => e.dataTransfer.setData('item', e.target.id);

拖拽开始时触发,把被拖拽元素的 id 存入  e.dataTransfer

然后在 ondrop 的时候 可以获取到这个值 (ondragenter、ondragover、ondragleave 获取不到...)

putEle.ondrop = function(e) {

let id = e.dataTransfer.getData('item');

// ...

}

简单的应用:

Document

.wrapper {display: flex;border: 1px solid orangered;padding: 10px;}

.col {border: 1px solid #808080;height: 500px;width: 200px;margin: 0 10px;padding: 10px;}

.item {border: 1px solid #808080;margin: 5px 0;}

item1
item2
item3

let cols = document.getElementsByClassName('col');

for (let col of cols) {

col.ondragenter = e => {

console.log('放置元素 ondragenter', '');

}

col.ondragover = e => {

e.preventDefault();

console.log('放置元素 ondragover', '');

}

col.ondragleave = e => {

console.log('放置元素 ondragleave', '');

}

col.ondrop = function(e) {

console.log('放置元素 ondrop', '');

this.append(document.getElementById(e.dataTransfer.getData('item')));

}

}

let items = document.getElementsByClassName('item');

for (let item of items) {

item.ondragstart = e => {

console.log('拖拽元素 ondragstart');

e.dataTransfer.setData('item', e.target.id);

}

item.ondragend = e => {

console.log('拖拽元素 ondragend');

}

}

文章开头部分的 React 写的 demo

.item {

border: 1px solid #1da921;

width: 180px;

border-radius: 5px;

box-shadow: 0 0 5px 0 #b3b3b3;

margin: 5px auto;

background: #fff;

}

.item.active {

border-style: dashed;

}

.item-header {

font-size: 12px;

color: #9e9e9e;

padding: 3px 5px;

}

.item-main {

padding: 5px;

font-size: 14px;

color: #424242;

height: 36px;

overflow: hidden;

text-overflow: ellipsis;

display: -webkit-box;

-webkit-box-orient: vertical;

-webkit-line-clamp: 2;

}

.item-header-point {

background: #ccc;

float: right;

padding: 0 4px;

min-width: 10px;

text-align: center;

color: #fff;

border-radius: 50%;

}

.col {

border: 1px solid #d2d2d2;

flex-grow: 1;

width: 180px;

height: 100%;

margin: 0 2px;

background: #eee;

flex-grow: 1;

display: flex;

flex-direction: column;

}

.col-header {

height: 40px;

line-height: 40px;

background: #1DA921;

color: #fff;

text-align: center;

}

.col-main {

overflow: auto;

flex-grow: 1;

}

.col-main.active {

background: #00ad23;

opacity: 0.1;

}

.task-wrapper {

display: flex;

height: 400px;

width: 700px;

}

const STATUS_TODO = 'STATUS_TODO';

const STATUS_DOING = 'STATUS_DOING';

const STATUS_DONE = 'STATUS_DONE';

const STATUS_CODE = {

STATUS_TODO: '待处理',

STATUS_DOING: '进行中',

STATUS_DONE: '已完成'

}

let tasks = [{

id: 0,

status: STATUS_TODO,

title: '每周七天阅读五次,每次阅读完要做100字的读书笔记',

username: '小夏',

point: 10

}, {

id: 1,

status: STATUS_TODO,

title: '每周七天健身4次,每次健身时间需要大于20分钟',

username: '橘子🍊',

point: 5

}, {

id: 2,

status: STATUS_TODO,

title: '单词*100',

username: '┑( ̄Д  ̄)┍',

point: 2

}, {

id: 3,

status: STATUS_TODO,

title: '单词*150',

username: '┑( ̄Д  ̄)┍',

point: 2

}, {

id: 4,

status: STATUS_TODO,

title: '单词*200',

username: '┑( ̄Д  ̄)┍',

point: 2

}, {

id: 5,

status: STATUS_TODO,

title: '单词*250',

username: '┑( ̄Д  ̄)┍',

point: 2

}]

class TaskItem extends React.Component {

handleDragStart = (e) => {

this.props.onDragStart(this.props.id);

}

render() {

let { id, title, point, username, active, onDragEnd } = this.props;

return (

onDragStart={this.handleDragStart}

onDragEnd={onDragEnd}

id={`item-${id}`}

className={'item' + (active ? ' active' : '')}

draggable="true"

>

{username}

{point}

{title}

);

}

}

class TaskCol extends React.Component {

state = {

in: false

}

handleDragEnter = (e) => {

e.preventDefault();

if (this.props.canDragIn) {

this.setState({

in: true

})

}

}

handleDragLeave = (e) => {

e.preventDefault();

if (this.props.canDragIn) {

this.setState({

in: false

})

}

}

handleDrop = (e) => {

e.preventDefault();

this.props.dragTo(this.props.status);

this.setState({

in: false

})

}

render() {

let { status, children } = this.props;

return (

id={`col-${status}`}

className={'col'}

onDragEnter={this.handleDragEnter}

onDragLeave={this.handleDragLeave}

onDragOver={this.handleDragEnter}

onDrop={this.handleDrop}

draggable="true"

>

{STATUS_CODE[status]}

{children}

);

}

}

class App extends React.Component {

state = {

tasks: tasks,

activeId: null

}

/**

* 传入被拖拽任务项的 id

*/

onDragStart = (id) => {

this.setState({

activeId: id

})

}

dragTo = (status) => {

let { tasks, activeId} = this.state;

let task = tasks[activeId];

if (task.status !== status) {

task.status = status;

this.setState({

tasks: tasks

})

}

this.cancelSelect();

}

cancelSelect = () => {

this.setState({

activeId: null

})

}

render() {

let { tasks, activeId } = this.state;

let { onDragStart, onDragEnd, cancelSelect } = this;

return (

{

Object.keys(STATUS_CODE).map(status =>

status={status}

key={status}

dragTo={this.dragTo}

canDragIn={activeId != null && tasks[activeId].status !== status}>

{ tasks.filter(t => t.status === status).map(t =>

key={t.id}

active={t.id === activeId}

id={t.id}

title={t.title}

point={t.point}

username={t.username}

onDragStart={onDragStart}

onDragEnd={cancelSelect}

/>)

}

)

}

)

}

}

ReactDOM.render(

,

document.getElementById('app')

);

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

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值