告别繁琐排序:dnd-kit + React Hook Form 打造流畅表单拖拽体验
你是否还在为表单中的列表排序功能编写冗长代码?是否遇到过拖拽排序与表单验证冲突的问题?本文将展示如何通过 dnd-kit 与 React Hook Form 的无缝集成,仅需 10 分钟即可实现兼具美感与功能性的表单拖拽排序功能。读完本文你将掌握:拖拽排序与表单状态同步、复杂表单场景下的性能优化、无障碍拖拽实现方案。
核心组件与集成原理
dnd-kit 提供了现代化的拖拽基础设施,其核心能力来自 DndContext 组件和 useSortable 钩子。DndContext 作为拖拽系统的中枢,管理着整个应用的拖拽状态,而 useSortable 则为列表项提供了排序能力。
React Hook Form 则通过非受控组件和 refs 实现高效表单管理。两者集成的关键在于将拖拽产生的排序变化同步到 React Hook Form 的表单状态中,同时保持表单验证的实时性。
// 核心集成架构
<DndContext onDragEnd={handleDragEnd}>
<SortableContext items={formValues.items.map(i => i.id)}>
{formValues.items.map((item, index) => (
<SortableItem key={item.id} id={item.id}>
<Controller
name={`items.${index}.name`}
control={control}
render={({ field }) => <input {...field} />}
/>
</SortableItem>
))}
</SortableContext>
</DndContext>
完整实现步骤
1. 安装依赖
首先安装必要的依赖包,包括 dnd-kit 的核心模块和 React Hook Form:
npm install @dnd-kit/core @dnd-kit/sortable react-hook-form
2. 基础表单与拖拽列表实现
创建一个包含可排序列表的表单,使用 useForm 初始化表单状态,设置初始列表数据:
import { useForm, Controller } from 'react-hook-form';
import { DndContext, DragEndEvent } from '@dnd-kit/core';
import { SortableContext, useSortable } from '@dnd-kit/sortable';
import { CSS } from '@dnd-kit/utilities';
// 定义表单数据类型
type Item = {
id: string;
name: string;
description: string;
};
type FormValues = {
items: Item[];
};
export default function SortableForm() {
const { control, watch, setValue } = useForm<FormValues>({
defaultValues: {
items: [
{ id: '1', name: '项目 1', description: '描述 1' },
{ id: '2', name: '项目 2', description: '描述 2' },
{ id: '3', name: '项目 3', description: '描述 3' }
]
}
});
const formValues = watch();
// 拖拽结束处理函数
const handleDragEnd = (event: DragEndEvent) => {
const { active, over } = event;
if (!over) return;
const activeIndex = formValues.items.findIndex(i => i.id === active.id);
const overIndex = formValues.items.findIndex(i => i.id === over.id);
if (activeIndex !== overIndex) {
// 创建新数组,避免直接修改状态
const newItems = [...formValues.items];
const [movedItem] = newItems.splice(activeIndex, 1);
newItems.splice(overIndex, 0, movedItem);
// 更新表单状态
setValue('items', newItems);
}
};
return (
<form>
<DndContext onDragEnd={handleDragEnd}>
<SortableContext items={formValues.items.map(i => i.id)}>
{formValues.items.map((item, index) => (
<SortableItem key={item.id} id={item.id} index={index}>
<div className="item-content">
<Controller
name={`items.${index}.name`}
control={control}
render={({ field }) => (
<input
{...field}
placeholder="名称"
className="form-input"
/>
)}
/>
<Controller
name={`items.${index}.description`}
control={control}
render={({ field }) => (
<textarea
{...field}
placeholder="描述"
className="form-textarea"
/>
)}
/>
</div>
</SortableItem>
))}
</SortableContext>
</DndContext>
</form>
);
}
3. 实现 SortableItem 组件
创建可排序列表项组件,使用 useSortable 钩子获取拖拽属性,并应用到元素上:
const SortableItem = ({ id, children }: { id: string; children: React.ReactNode }) => {
const {
attributes,
listeners,
setNodeRef,
transform,
transition,
isDragging
} = useSortable({ id });
const style = {
transform: CSS.Transform.toString(transform),
transition,
opacity: isDragging ? 0.5 : 1,
cursor: 'grab'
};
return (
<div
ref={setNodeRef}
style={style}
{...attributes}
{...listeners}
className="sortable-item"
>
<div className="drag-handle">::</div>
{children}
</div>
);
};
性能优化策略
对于包含大量字段或复杂验证逻辑的表单,需要实施性能优化。关键优化点包括:
- 使用 React.memo 包装列表项组件,避免不必要的重渲染
- 对于超过 50 项的长列表,实现虚拟滚动,可配合 react-window
- 拖拽过程中暂停表单验证,拖拽结束后恢复
// 性能优化示例 - 使用 useCallback 和 React.memo
const SortableItem = React.memo(({
id,
index,
children,
control
}: SortableItemProps) => {
// 使用 useCallback 记忆化处理函数
const handleChange = useCallback((value: string) => {
// 处理逻辑
}, []);
// ...
});
无障碍支持与键盘导航
dnd-kit 内置了完善的无障碍支持,通过 Accessibility 组件实现。要确保拖拽功能对键盘用户友好,需添加键盘导航支持:
// 无障碍配置
<DndContext
onDragEnd={handleDragEnd}
sensors={[
useSensor(MouseSensor),
useSensor(TouchSensor),
useSensor(KeyboardSensor, {
coordinateGetter: keyboardCoordinates
})
]}
>
{/* 列表内容 */}
</DndContext>
高级应用场景
嵌套表单与拖拽
在嵌套表单场景中,可通过调整字段名称路径实现深层对象的拖拽排序:
// 嵌套表单拖拽示例
<Controller
name={`sections.${sectionIndex}.items.${itemIndex}.name`}
control={control}
render={({ field }) => <input {...field} />}
/>
拖拽过程中的表单验证
如需在拖拽过程中保持表单验证,可使用 React Hook Form 的 trigger 方法手动触发验证:
// 拖拽结束后触发验证
const handleDragEnd = async (event: DragEndEvent) => {
// ...排序逻辑
// 触发表单验证
await trigger();
};
常见问题与解决方案
- 表单状态不同步:确保使用 setValue 更新整个数组,而非单独修改元素
- 性能问题:使用 React.memo 和 useCallback 优化列表项渲染
- 复杂验证场景:拖拽结束后使用 trigger 方法手动触发验证
- 移动端兼容性:确保引入 TouchSensor 并测试触摸设备上的体验
通过 dnd-kit 与 React Hook Form 的集成,我们实现了既美观又高效的表单拖拽排序功能。这种方案不仅代码简洁,而且性能优异,同时支持完整的无障碍特性。无论你是构建简单的待办事项列表,还是复杂的多步骤表单,这种集成方案都能满足需求。
完整示例代码可在项目 playground 目录中找到,包含各种复杂度的表单拖拽实现。尝试将这种技术应用到你的项目中,提升用户体验和开发效率。
GLM-5智谱 AI 正式发布 GLM-5,旨在应对复杂系统工程和长时域智能体任务。Jinja00
GLM-5-w4a8GLM-5-w4a8基于混合专家架构,专为复杂系统工程与长周期智能体任务设计。支持单/多节点部署,适配Atlas 800T A3,采用w4a8量化技术,结合vLLM推理优化,高效平衡性能与精度,助力智能应用开发Jinja00
请把这个活动推给顶尖程序员😎本次活动专为懂行的顶尖程序员量身打造,聚焦AtomGit首发开源模型的实际应用与深度测评,拒绝大众化浅层体验,邀请具备扎实技术功底、开源经验或模型测评能力的顶尖开发者,深度参与模型体验、性能测评,通过发布技术帖子、提交测评报告、上传实践项目成果等形式,挖掘模型核心价值,共建AtomGit开源模型生态,彰显顶尖程序员的技术洞察力与实践能力。00
Kimi-K2.5Kimi K2.5 是一款开源的原生多模态智能体模型,它在 Kimi-K2-Base 的基础上,通过对约 15 万亿混合视觉和文本 tokens 进行持续预训练构建而成。该模型将视觉与语言理解、高级智能体能力、即时模式与思考模式,以及对话式与智能体范式无缝融合。Python00
MiniMax-M2.5MiniMax-M2.5开源模型,经数十万复杂环境强化训练,在代码生成、工具调用、办公自动化等经济价值任务中表现卓越。SWE-Bench Verified得分80.2%,Multi-SWE-Bench达51.3%,BrowseComp获76.3%。推理速度比M2.1快37%,与Claude Opus 4.6相当,每小时仅需0.3-1美元,成本仅为同类模型1/10-1/20,为智能应用开发提供高效经济选择。【此简介由AI生成】Python00
Qwen3.5Qwen3.5 昇腾 vLLM 部署教程。Qwen3.5 是 Qwen 系列最新的旗舰多模态模型,采用 MoE(混合专家)架构,在保持强大模型能力的同时显著降低了推理成本。00- RRing-2.5-1TRing-2.5-1T:全球首个基于混合线性注意力架构的开源万亿参数思考模型。Python00