前端拖拽排序实现详解:从原理到实践 - 附完整代码

前端拖拽排序实现详解:从原理到实践 - 附完整代码
在这里插入图片描述
🌷 古之立大事者,不惟有超世之才,亦必有坚忍不拔之志
🎐 个人CSND主页——Micro麦可乐的博客
🐥《Docker实操教程》专栏以最新的Centos版本为基础进行Docker实操教程,入门到实战
🌺《RabbitMQ》专栏19年编写主要介绍使用JAVA开发RabbitMQ的系列教程,从基础知识到项目实战
🌸《设计模式》专栏以实际的生活场景为案例进行讲解,让大家对设计模式有一个更清晰的理解
🌛《开源项目》本专栏主要介绍目前热门的开源项目,带大家快速了解并轻松上手使用
🍎 《前端技术》专栏以实战为主介绍日常开发中前端应用的一些功能以及技巧,均附有完整的代码示例
✨《开发技巧》本专栏包含了各种系统的设计原理以及注意事项,并分享一些日常开发的功能小技巧
💕《Jenkins实战》专栏主要介绍Jenkins+Docker的实战教程,让你快速掌握项目CI/CD,是2024年最新的实战教程
🌞《Spring Boot》专栏主要介绍我们日常工作项目中经常应用到的功能以及技巧,代码样例完整
👍《Spring Security》专栏中我们将逐步深入Spring Security的各个技术细节,带你从入门到精通,全面掌握这一安全技术
如果文章能够给大家带来一定的帮助!欢迎关注、评论互动~

前端拖拽排序实现详解:从原理到实践 - 附完整代码

1. 前言

在我们前端开发过程中经常会用到排序的功能,其中列表项的拖拽排序(Drag-and-Drop Sortable)不仅能让用户直观、高效地调整顺序,还能大幅提升交互体验。无论是管理后台的菜单排序、看板(Kanban)中任务卡片的调整,还是移动端的图片/视频重排,拖拽排序都是不可或缺的交互模式。

在这里插入图片描述


本文博主将带着小伙伴从零开始,用原生 HTML5 Drag & Drop API 实现一个简洁的可拖拽排序列表,并模拟向后端提交新顺序的完整流程。


2. 拖拽排序的应用

任务管理工具
用户可以拖拽卡片改变优先级或在不同分组间移动(如Trello

内容管理系统
页面元素排序(如WordPress

图片/视频排序
相册管理、商品轮播图顺序调整

问卷表单选项
管理问答列表时,调整题目或选项顺序更直观

自定义导航菜单
后台可视化拖拽菜单层级与顺序

这些场景下,通过拖拽排序,用户无需点上下箭头、输入序号,就能快速完成调整,显著提升效率与体验。


3. 核心实现原理

3.1 HTML5拖放API基础

在这里插入图片描述


HTML5 Drag & Drop 原生 API 核心流程如下:

  • 设定可拖拽元素:draggable="true"
  • 监听拖拽开始:dragstart 事件,记录当前拖拽项索引或标识
  • 允许拖拽进入目标:在目标元素或容器上 dragover 事件中调用 event.preventDefault()
  • 处理放置:drop 事件中获取拖拽项和目标项索引,完成 DOM 位置交换。
  • 结束拖拽:可选的 dragend 事件,用于清理样式或状态。

3.2 关键事件解析

默认拖放API具备以下几个事件类型:

事件类型触发时机常用操作
dragstart开始拖拽元素时设置被拖拽元素的ID
dragenter进入目标元素时添加视觉反馈
dragover在目标元素上悬停时阻止默认行为(允许放置)
dragleave离开目标元素时移除视觉反馈
drop在目标元素上释放时处理元素位置交换
dragend拖拽操作结束时清理状态

4. 完整示例代码

下面是一个最小可运行的示例,小伙伴们可以复制至本地测试运行,注意后端排序保存已经将当前顺序的 ID 数组传递,后端接口小伙伴们自行实现即可

<!DOCTYPEhtml><htmllang="zh-CN"><head><metacharset="UTF-8"><title>拖拽排序示例</title><style>body{font-family: sans-serif;padding: 20px;}#sortable-list{list-style: none;padding: 0;width: 300px;margin: 0 auto;}#sortable-list li{padding: 10px 15px;margin-bottom: 8px;background: #f0f0f0;border: 1px solid #ddd;cursor: move;user-select: none;}/* 拖拽时样式 */.dragging{opacity: 0.5;}.over{border-top: 2px solid #007bff;}</style></head><body><h2>拖拽排序示例</h2><ulid="sortable-list"><lidata-id="1"draggable="true">项目 1</li><lidata-id="2"draggable="true">项目 2</li><lidata-id="3"draggable="true">项目 3</li><lidata-id="4"draggable="true">项目 4</li><lidata-id="5"draggable="true">项目 5</li></ul><buttonid="saveOrderBtn">保存顺序</button><script>const list = document.getElementById('sortable-list');let dragSrcEl =null;functionhandleDragStart(e){ dragSrcEl =this;this.classList.add('dragging'); e.dataTransfer.effectAllowed ='move'; e.dataTransfer.setData('text/plain',this.dataset.id);}functionhandleDragOver(e){ e.preventDefault();// 必须阻止默认,才有 drop 事件 e.dataTransfer.dropEffect ='move';returnfalse;}functionhandleDragEnter(e){if(this!== dragSrcEl){this.classList.add('over');}}functionhandleDragLeave(e){this.classList.remove('over');}functionhandleDrop(e){ e.stopPropagation();// 阻止事件冒泡if(dragSrcEl !==this){// 在 DOM 中交换位置const nodes = Array.from(list.children);const srcIndex = nodes.indexOf(dragSrcEl);const targetIndex = nodes.indexOf(this);if(srcIndex < targetIndex){ list.insertBefore(dragSrcEl,this.nextSibling);}else{ list.insertBefore(dragSrcEl,this);}}returnfalse;}functionhandleDragEnd(e){// 清理样式this.classList.remove('dragging'); Array.from(list.children).forEach(item=>{ item.classList.remove('over');});}// 绑定事件 Array.from(list.children).forEach(item=>{ item.addEventListener('dragstart', handleDragStart); item.addEventListener('dragenter', handleDragEnter); item.addEventListener('dragover', handleDragOver); item.addEventListener('dragleave', handleDragLeave); item.addEventListener('drop', handleDrop); item.addEventListener('dragend', handleDragEnd);});// 模拟后端提交新顺序 document.getElementById('saveOrderBtn').addEventListener('click',()=>{const order = Array.from(list.children).map(li=> li.dataset.id); console.log('新的顺序:', order);// 示例:POST 到 /api/update-orderfetch('/api/update-order',{ method:'POST', headers:{'Content-Type':'application/json'}, body:JSON.stringify({ order })}).then(res=>{if(!res.ok)thrownewError('保存失败');return res.json();}).then(data=>{alert('顺序保存成功!');}).catch(err=>{ console.error(err);alert('保存失败,请重试');});});</script></body></html>

要点说明:

1、每个 <li> 元素设置了 draggable="true"data-id 作为唯一标识;
2、dragstart 时记录源元素并设置拖拽数据;
3、dragover 必须 preventDefault(),否则无法触发 drop
4、drop 事件中根据源元素和目标元素在父容器中的索引,动态交换位置;
5、点击“保存顺序”按钮后,将当前顺序的 ID 数组发送给后端接口 /api/update-order,完成持久化。

5. 结语

通过上述示例,相信小伙伴们已经掌握了使用原生 HTML5 Drag & Drop API 实现前端拖拽排序的全流程:从可拖拽元素配置、事件监听,到 DOM 顺序交换,再到模拟后端提交。

该方案无需引入第三方库,也足够轻量易懂。可以根据项目需求进一步优化:

  • 性能优化:对大型列表使用虚拟化或节流拖拽事件。
  • 视觉增强:使用动画(CSS transition)平滑排序过程。
  • 多容器拖拽:扩展至跨列表、跨分组的拖拽。
  • 第三方库:在复杂场景下可结合 SortableJSDragula 等成熟库。

希望本文能帮助小伙伴们快速在项目中落地拖拽排序功能,如果你在实践过程中有任何疑问或更好的扩展思路,欢迎在评论区留言,最后希望大家一键三连给博主一点点鼓励!


Read more

【FPGA】Quartus Prime Lite 23.1 最新版 安装教程 ModelSim_18.1 下载安装 + 联调仿真教程 + 详细安装教程 2025最新

【FPGA】Quartus Prime Lite 23.1 最新版 安装教程 ModelSim_18.1 下载安装 + 联调仿真教程 + 详细安装教程 2025最新

前言         本文章基于截至2025年 Quartus_Prime_Lite的最新版 23.1 版本,详细的,一步一步的教你怎么安装,每一步都教你怎么做,按照流程绝对能安装成功。创作不易希望大家看完后点个赞支持创作,谢谢大家啦! 目录  软件下载地址 Quartus Prime Lite 23.1 ModelSim-Intel® FPGA 标准版软件版本 18.1 若不想在官网下载或官网下载速度太慢 点个关注+收藏可以免费用下面的百度链接进行下载  两个软件的安装包都在里面。如果使用百度链接下载则可跳过两个软件的下载流程,直接看安装流程。 一、Quartus Prime Lite 23.1 下载以及安装流程 1.1 Quartus Prime Lite 23.1 官方网站下载流程 第一步 打开上方链接到达如下界面 确保软件名称和版本如下图

Android陀螺仪实战:从基础到VR运动策略封装

1. 陀螺仪基础:从传感器数据到三维旋转 大家好,我是老张,在移动端和智能硬件领域摸爬滚打了十几年,今天想和大家聊聊 Android 陀螺仪。很多刚接触的朋友会觉得这东西很神秘,什么角速度、姿态解算,听起来就头大。其实没那么复杂,你可以把陀螺仪想象成一个特别灵敏的“旋转速度计”。当你拿着手机转动时,它就能立刻告诉你:“嘿,你现在正绕着X轴,以每秒0.5弧度的速度在转呢!” 在 Android 里,我们通过 SensorManager 这个“大管家”来和陀螺仪打交道。第一步永远是获取服务,这就像你去银行办事得先取号一样。拿到 SensorManager 后,我们就能查询设备上有没有陀螺仪(Sensor.TYPE_GYROSCOPE)。现在绝大多数手机都有,但稳妥起见,检查一下总是好的。接下来就是注册一个监听器,告诉系统:“我准备好接收旋转数据了,有新数据就赶紧通知我。” 这里有个关键参数叫采样延迟,比如 SENSOR_DELAY_

AI一镜到底效果炸裂 把教材插图变成VR全景视频(附提示词)

AI一镜到底效果炸裂 把教材插图变成VR全景视频(附提示词)

大家好,我是AI培训韩老师! 在电影的世界里,有一种拍摄手法总能引发观众惊叹——一镜到底。它让镜头像一双无形的眼睛,带领我们穿越战场、潜入犯罪现场、亲历角色内心世界,不间断地体验完整的故事时空。 于是很多人会问我,如何用AI实现一镜到底?简单来说就是不用剪辑一键生成,又简单有高级那种。下面通过这篇文章告诉你! 用AI生成具有电影感的“一镜到底”视频,关键在于清晰地告诉AI你想要的镜头运动轨迹和场景衔接方式。下面我为你梳理了从核心思路、具体方法到实用技巧的完整指南。 🎬 理解AI一镜到底的核心 在AI视频生成中,它通常通过两种方式实现: * 智能多帧创作:这是目前更主流高效的方法。你先准备一系列在内容上连贯的图片(相当于分镜图),然后AI会模拟镜头的连续运动,将这些画面无缝连接成一段长视频,营造出一镜到底的观感。 * 单一长提示词生成:直接用一个详细的长段文本描述整个镜头的运动路径和所有场景变化,由AI直接生成视频。这对提示词书写要求极高,且效果不确定性更大。 无论哪种方式,精准地描述镜头运动(运镜)都是成功的关键。 📷 掌握核心运镜技巧 你需要像导演一样思考,

电影知识图谱推荐问答系统 | Python Django系统 Neo4j MySQL Echarts 协同过滤 大数据 人工智能 毕业设计源码(建议收藏)✅

电影知识图谱推荐问答系统 | Python Django系统 Neo4j MySQL Echarts 协同过滤 大数据 人工智能 毕业设计源码(建议收藏)✅

博主介绍:✌全网粉丝10W+,前互联网大厂软件研发、集结硕博英豪成立软件开发工作室,专注于计算机相关专业项目实战6年之久,累计开发项目作品上万套。凭借丰富的经验与专业实力,已帮助成千上万的学生顺利毕业,选择我们,就是选择放心、选择安心毕业✌ > 🍅想要获取完整文章或者源码,或者代做,拉到文章底部即可与我联系了。🍅 点击查看作者主页,了解更多项目! 🍅感兴趣的可以先收藏起来,点赞、关注不迷路,大家在毕设选题,项目以及论文编写等相关问题都可以给我留言咨询,希望帮助同学们顺利毕业 。🍅 1、毕业设计:2026年计算机专业毕业设计选题汇总(建议收藏)✅ 2、最全计算机专业毕业设计选题大全(建议收藏)✅ 1、项目介绍 技术栈 以Python作为核心开发语言,基于Django框架搭建系统整体架构,采用Neo4j图形数据库与MySQL数据库存储数据,整合Echarts可视化工具、协同过滤推荐算法,结合HTML技术完成前端页面的搭建。 功能模块 * 电影知识图谱管理 * 电影问答交互 * 电影列表展示 * 个人信息查看 * 电影详情展示 * 用户注册登录 * 后