JavaScript如何实现拖放功能_Drag和DropAPI怎么用
发布时间 - 2026-01-01 00:00:00 点击率:次dragstart中必须调用setData()否则drop不触发;drop目标需preventDefault()阻止默认行为;移动端不支持原生Drag and Drop;dataTransfer数据只能在drag事件链中读写。
dragstart事件里必须设置dataTransfer
不调用 event.dataTransfer.setData(),后续的 drop 事件根本不会触发——浏览器会直接忽略这次拖放。哪怕只是传个空字符串或固定标识符,也得设:
element.addEventListener('dragstart', (e) => {
e.dataTransfer.setData('text/plain', 'item-id-123');
});
常见错误是只写了 e.dataTransfer.effectAllowed = 'move' 就以为够了,其实这行只影响光标样式和 drop 区域的接受策略,不设数据就等于没“拿起”东西。
注意:setData() 第一个参数是 MIME 类型,'text/plain' 最通用;如果拖的是文件或自定义结构,可用 'application/json' 或自定义类型如 'myapp/item',但接收端必须用对应类型 getData() 才能取到。
drop目标必须阻止默认行为
drop 和 dragover 事件默认会被浏览器拦截,不阻止就会立刻中止拖放流程,drop 根本不会执行:
dropZone.addEventListener('dragover', (e) => {
e.preventDefault(); // 必须!
});
dropZone.addEventListener('drop', (e) => {
e.preventDefault(); // 必须!
const id = e.dataTransfer.getData('text/plain');
console.log(id); // → 'item-id-123'
});
漏掉 dragover 的 preventDefault() 是最常踩的坑——看着拖进去有高亮效果,一松手却啥也没发生,八成就是这儿卡住了。
另外,drop 事件里的 e.target 是实际释放点(可能是子元素),如果逻辑需要操作整个容器,建议用 e.currentTarget 或提前用 closest() 定位。
移动端不支持原生 Drag and Drop API
所有 iOS Safari 和 Android Chrome 都不支持标准的 dragstart/drop 事件。不是兼容性差,是压根没实现。想在手机上做拖放,必须换方案:
- 用
touchstart/touchmove/touchend自己模拟位移和
状态 - 引入轻量库如
interact.js或sortablejs(它们内部已处理平台差异) - 对纯 Web App,可降级为点击排序 + 确认弹窗,避免强行套用桌面逻辑
别试图用 try/catch 检测 API 存在就认为安全——iOS 上 'draggable' in document.createElement('div') 返回 true,但事件完全不触发。
dataTransfer只能在drag事件链中读写
dataTransfer 对象的数据是临时绑定在本次拖放过程中的,离开 dragstart→drag→drop 这条链,就无法再访问。比如在 drop 之后异步请求中直接引用 e.dataTransfer,大概率是 undefined 或空值。
正确做法是在 drop 事件里立刻提取并保存:
dropZone.addEventListener('drop', (e) => {
e.preventDefault();
const data = e.dataTransfer.getData('text/plain'); // ✅ 立即读
setTimeout(() => {
console.log(data); // ✅ 用闭包变量
}, 0);
});
另一个陷阱:多次快速拖放时,如果前一次还没处理完,新 dragstart 可能覆盖 dataTransfer,所以别依赖全局缓存,每次都在事件里取。
# javascript
# java
# android
# js
# json
# go
# 浏览器
# app
# safari
# ai
# ios
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
Laravel API资源(Resource)怎么用_格式化Laravel API响应的最佳实践
Laravel怎么导出Excel文件_Laravel Excel插件使用教程
Laravel如何集成第三方登录_Laravel Socialite实现微信QQ微博登录
通义万相免费版怎么用_通义万相免费版使用方法详细指南【教程】
奇安信“盘古石”团队突破 iOS 26.1 提权
html5源代码发行怎么设置权限_访问权限控制方法与实践【指南】
Android Socket接口实现即时通讯实例代码
laravel怎么为API路由添加签名中间件保护_laravel API路由签名中间件保护方法
Laravel如何配置和使用队列处理异步任务_Laravel队列驱动与任务分发实例
Java Adapter 适配器模式(类适配器,对象适配器)优缺点对比
如何打造高效商业网站?建站目的决定转化率
浏览器如何快速切换搜索引擎_在地址栏使用不同搜索引擎【搜索】
如何利用DOS批处理实现定时关机操作详解
重庆市网站制作公司,重庆招聘网站哪个好?
Angular 表单中正确绑定输入值以确保提交与验证正常工作
如何在橙子建站上传落地页?操作指南详解
香港服务器网站搭建教程-电商部署、配置优化与安全稳定指南
Laravel的辅助函数有哪些_Laravel常用Helpers函数提高开发效率
详解Nginx + Tomcat 反向代理 负载均衡 集群 部署指南
Laravel任务队列怎么用_Laravel Queues异步处理任务提升应用性能
Win11怎样安装网易有道词典_Win11安装词典教程【步骤】
Laravel如何记录日志_Laravel Logging系统配置与自定义日志通道
Laravel如何创建自定义中间件?(Middleware代码示例)
简历在线制作网站免费版,如何创建个人简历?
昵图网官网入口 昵图网素材平台官方入口
Laravel如何生成URL和重定向?(路由助手函数)
Python正则表达式进阶教程_复杂匹配与分组替换解析
如何用AI帮你把自己的生活经历写成一个有趣的故事?
Laravel如何生成和使用数据填充?(Seeder和Factory示例)
Laravel中间件起什么作用_Laravel Middleware请求生命周期与自定义详解
如何快速生成可下载的建站源码工具?
laravel怎么在请求结束后执行任务(Terminable Middleware)_laravel Terminable Middleware请求结束任务执行方法
如何用手机制作网站和网页,手机移动端的网站能制作成中英双语的吗?
mc皮肤壁纸制作器,苹果平板怎么设置自己想要的壁纸我的世界?
html5如何实现懒加载图片_ intersectionobserver api用法【教程】
高性能网站服务器配置指南:安全稳定与高效建站核心方案
网站制作壁纸教程视频,电脑壁纸网站?
Windows家庭版如何开启组策略(gpedit.msc)?(安装方法)
Laravel软删除怎么实现_Laravel Eloquent SoftDeletes功能使用教程
Windows10如何更改计算机工作组_Win10系统属性修改Workgroup
电商网站制作多少钱一个,电子商务公司的网站制作费用计入什么科目?
Laravel怎么解决跨域问题_Laravel配置CORS跨域访问
EditPlus中的正则表达式 实战(1)
如何在阿里云ECS服务器部署织梦CMS网站?
如何在万网ECS上快速搭建专属网站?
jquery插件bootstrapValidator表单验证详解
网站设计制作书签怎么做,怎样将网页添加到书签/主页书签/桌面?
如何批量查询域名的建站时间记录?
Python自动化办公教程_ExcelWordPDF批量处理案例
如何在腾讯云免费申请建站?


状态