告别繁琐排序: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 目录中找到,包含各种复杂度的表单拖拽实现。尝试将这种技术应用到你的项目中,提升用户体验和开发效率。
atomcodeClaude Code 的开源替代方案。连接任意大模型,编辑代码,运行命令,自动验证 — 全自动执行。用 Rust 构建,极致性能。 | An open-source alternative to Claude Code. Connect any LLM, edit code, run commands, and verify changes — autonomously. Built in Rust for speed. Get StartedRust0153- DDeepSeek-V4-ProDeepSeek-V4-Pro(总参数 1.6 万亿,激活 49B)面向复杂推理和高级编程任务,在代码竞赛、数学推理、Agent 工作流等场景表现优异,性能接近国际前沿闭源模型。Python00
LongCat-Video-Avatar-1.5最新开源LongCat-Video-Avatar 1.5 版本,这是一款经过升级的开源框架,专注于音频驱动人物视频生成的极致实证优化与生产级就绪能力。该版本在 LongCat-Video 基础模型之上构建,可生成高度稳定的商用级虚拟人视频,支持音频-文本转视频(AT2V)、音频-文本-图像转视频(ATI2V)以及视频续播等原生任务,并能无缝兼容单流与多流音频输入。00
auto-devAutoDev 是一个 AI 驱动的辅助编程插件。AutoDev 支持一键生成测试、代码、提交信息等,还能够与您的需求管理系统(例如Jira、Trello、Github Issue 等)直接对接。 在IDE 中,您只需简单点击,AutoDev 会根据您的需求自动为您生成代码。Kotlin03
Intern-S2-PreviewIntern-S2-Preview,这是一款高效的350亿参数科学多模态基础模型。除了常规的参数与数据规模扩展外,Intern-S2-Preview探索了任务扩展:通过提升科学任务的难度、多样性与覆盖范围,进一步释放模型能力。Python00
skillhubopenJiuwen 生态的 Skill 托管与分发开源方案,支持自建与可选 ClawHub 兼容。Python0112