---
description: "Shadcn-ui 管理页表单创建/编辑表单分离指令"
---
将 创建/编辑表单分离
不要
要
{isCreateForm ? (
// 创建表单(独立渲染)
) : (
// 编辑表单(独立渲染)
)}
### 创建/编辑表单分离模式(推荐)
基于 `src/client/admin-shadcn/pages/Users.tsx` 的最佳实践,创建/编辑表单分离模式通过以下方式解决创建/编辑时数据对象类型差异问题:
#### 核心优势
1. **类型安全**:创建使用 `CreateEntityDto`,编辑使用 `UpdateEntityDto`,避免类型冲突
2. **字段差异处理**:创建时的必填字段在编辑时自动变为可选
3. **敏感字段处理**:密码等敏感字段在编辑时可设为可选
4. **状态隔离**:两种模式完全独立,避免状态污染
#### 完整实现模板
```typescript
// 1. 类型定义(使用真实后端类型)
type CreateRequest = InferRequestType['json'];
type UpdateRequest = InferRequestType['json'];
type EntityResponse = InferResponseType['data'][0];
// 2. 状态管理
const [isModalOpen, setIsModalOpen] = useState(false);
const [editingEntity, setEditingEntity] = useState(null);
const [isCreateForm, setIsCreateForm] = useState(true);
// 3. 分离的表单实例
const createForm = useForm({
resolver: zodResolver(CreateEntityDto),
defaultValues: {
// 创建时必须提供的默认值
},
});
const updateForm = useForm({
resolver: zodResolver(UpdateEntityDto),
defaultValues: {
// 更新时的默认值(会被实际数据覆盖)
},
});
// 4. 表单操作函数
const handleCreate = () => {
setEditingEntity(null);
setIsCreateForm(true);
createForm.reset({
// 创建时的初始值
status: 1, // 示例:默认启用
});
setIsModalOpen(true);
};
const handleEdit = (entity: EntityResponse) => {
setEditingEntity(entity);
setIsCreateForm(false);
updateForm.reset({
...entity,
// 关键:处理创建/编辑字段差异
password: undefined, // 密码在更新时可选
confirmPassword: undefined,
// 其他需要特殊处理的字段
});
setIsModalOpen(true);
};
// 5. 提交处理
const handleCreateSubmit = async (data: CreateRequest) => {
try {
const res = await apiClient.$post({ json: data });
if (res.status !== 201) throw new Error('创建失败');
toast.success('创建成功');
setIsModalOpen(false);
refetch();
} catch (error) {
toast.error('创建失败,请重试');
}
};
const handleUpdateSubmit = async (data: UpdateRequest) => {
if (!editingEntity) return;
try {
const res = await apiClient[':id']['$put']({
param: { id: editingEntity.id },
json: data
});
if (res.status !== 200) throw new Error('更新失败');
toast.success('更新成功');
setIsModalOpen(false);
refetch();
} catch (error) {
toast.error('更新失败,请重试');
}
};
```
#### 对话框渲染模板
```tsx
```