大家好,我是王福鹏。
我是一名高级全栈工程师,也是 17.5k 开源项目 PMP 的作者。现在我正在开发一个Notion风格的知识库
HuashuiAI 包括 AI 写作和协作,使用 React Nextjs 和 Supabase。
在这篇文章中,我将分享如何通过 React 和 dnd-kit 实现树列表拖放排序。源码链接在本文底部。
Dnd-kit 和可排序组件
Dnd-kit 是 React 生态中常见的拖放工具,默认支持排序。
<dndcontext sensors="{sensors}" collisiondetection="{closestCenter}" ondragend="{handleDragEnd}"> <sortablecontext items="{items}" strategy="{verticalListSortingStrategy}"> {items.map(id => <sortableitem key="{id}"> <p>But it can only support the one-level list. If we want to implement a multi-level nested list (or tree), we have to customize it.</p> <h2> Define state date structure </h2> <p>Modern front-end frameworks such as React Vue are data-driven views, so defining data structures first and then considering UI rendering.</p> <p>The most common data structure definition for multi-level nested lists (trees) is as follows, and virtual DOM vnode is also defined in this way.<br> </p> <pre class="brush:php;toolbar:false">const defaultItems = [ { id: 'A', children: [] }, { id: 'B', children: [ { id: 'B1', children: [] }, { id: 'B2', children: [ { id: 'B2a', children: [] }, { id: 'B2b', children: [] }, ], }, ], }, { id: 'C', children: [] }, { id: 'D', children: [ { id: 'D1', children: [] }, { id: 'D2', children: [] }, ], }, { id: 'E', children: [] }, ]
多级嵌套SortableContext不可行
因为状态数据结构是嵌套的,所以我首先想到的就是嵌套并一起渲染UI结构。
首先,嵌套
然后,继续嵌套下级
运行效果如下。问题是同一级别内允许拖放排序,但跨级别排序是不可能的,因为它不是上下文 - 这是合理的
多级转换为单级是可行的
由于嵌套不可行,因此需要将多级转换为单级。
但是需要为每个item添加祖先Ids属性,首先是为了显示层次结构的深度,其次是为了知道它有哪些父节点。
interface IItem { id: string ancestorIds?: string[] children?: IItem[] } function flatten(items: IItem[]): IItem[] { return items.reduce<iitem>((acc, item) => { acc.push(item) if (item.children) { const children = item.children.map((i) => ({ ...i, ancestorIds: [...(item.ancestorIds || []), item.id], // add ancestorIds })) acc.push(...flatten(children)) } return acc }, []) } </iitem>
转换后的渲染效果如下,现在可以拖动排序了。不过要修改状态排序后才会生效。
此外,我们还可以通过祖先ID的层级关系来判断是否可以移动。父节点不能移动到子节点,否则循环会死。
例如上图中,如果我们要将B2拖到B2a的位置,我们会发现B2a的祖先ID包含B2。这是不可能的,因为您无法将项目拖动到其自己的下属项目。
修改状态数据
为了方便操作,数据放置在Zustand全局存储中。
Dnd-kit 将拖动的元素称为 activeItem,将放置的目标位置称为 overItem。所以修改状态数据意味着将activeItem移动到overItem的位置。
如果是单关,Dnd-kit提供了一个方法arrayMove,可以直接修改。文档链接 https://docs.dndkit.com/presets/sortable
但是在多级嵌套列表(树)中,需要自己实现,有点麻烦。核心代码在这里,大家可以下载源码(文末)参考。
遇到问题
如下图所示,将A拖到B下面时,A会整体移动到B的底部,而不是在B内部。
要解决这个问题,需要判断B之后是否还有B的子元素,如果有,则将overItem赋值给其子元素
然后将当前活动元素插入到items的第一个元素中。
结束
源代码链接在这里 https://github.com/wangfupeng1988/react-dnd-sortable-demo
顺便说一句,我正在寻找一份国际工作机会,如果你有机会,欢迎通过我的 Github 个人资料联系我。
以上是React dnd-kit,实现树列表拖放排序的详细内容。更多信息请关注PHP中文网其他相关文章!

Python和JavaScript的主要区别在于类型系统和应用场景。1.Python使用动态类型,适合科学计算和数据分析。2.JavaScript采用弱类型,广泛用于前端和全栈开发。两者在异步编程和性能优化上各有优势,选择时应根据项目需求决定。

选择Python还是JavaScript取决于项目类型:1)数据科学和自动化任务选择Python;2)前端和全栈开发选择JavaScript。Python因其在数据处理和自动化方面的强大库而备受青睐,而JavaScript则因其在网页交互和全栈开发中的优势而不可或缺。

Python和JavaScript各有优势,选择取决于项目需求和个人偏好。1.Python易学,语法简洁,适用于数据科学和后端开发,但执行速度较慢。2.JavaScript在前端开发中无处不在,异步编程能力强,Node.js使其适用于全栈开发,但语法可能复杂且易出错。

javascriptisnotbuiltoncorc; saninterpretedlanguagethatrunsonenginesoftenwritteninc.1)javascriptwasdesignedAsalightweight,解释edganguageforwebbrowsers.2)Enginesevolvedfromsimpleterterterpretpreterterterpretertestojitcompilerers,典型地提示。

JavaScript可用于前端和后端开发。前端通过DOM操作增强用户体验,后端通过Node.js处理服务器任务。1.前端示例:改变网页文本内容。2.后端示例:创建Node.js服务器。

选择Python还是JavaScript应基于职业发展、学习曲线和生态系统:1)职业发展:Python适合数据科学和后端开发,JavaScript适合前端和全栈开发。2)学习曲线:Python语法简洁,适合初学者;JavaScript语法灵活。3)生态系统:Python有丰富的科学计算库,JavaScript有强大的前端框架。

JavaScript框架的强大之处在于简化开发、提升用户体验和应用性能。选择框架时应考虑:1.项目规模和复杂度,2.团队经验,3.生态系统和社区支持。

引言我知道你可能会觉得奇怪,JavaScript、C 和浏览器之间到底有什么关系?它们之间看似毫无关联,但实际上,它们在现代网络开发中扮演着非常重要的角色。今天我们就来深入探讨一下这三者之间的紧密联系。通过这篇文章,你将了解到JavaScript如何在浏览器中运行,C 在浏览器引擎中的作用,以及它们如何共同推动网页的渲染和交互。JavaScript与浏览器的关系我们都知道,JavaScript是前端开发的核心语言,它直接在浏览器中运行,让网页变得生动有趣。你是否曾经想过,为什么JavaScr


热AI工具

Undresser.AI Undress
人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover
用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool
免费脱衣服图片

Clothoff.io
AI脱衣机

Video Face Swap
使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

热门文章

热工具

PhpStorm Mac 版本
最新(2018.2.1 )专业的PHP集成开发工具

适用于 Eclipse 的 SAP NetWeaver 服务器适配器
将Eclipse与SAP NetWeaver应用服务器集成。

禅工作室 13.0.1
功能强大的PHP集成开发环境

WebStorm Mac版
好用的JavaScript开发工具

SublimeText3汉化版
中文版,非常好用