React UI 模式
核心原则
- •永远不显示过期UI - 加载菊花仅在实际加载时显示
- •总是显示错误 - 用户必须知道什么时候发生故障
- •乐观更新 - 使UI感觉即时
- •渐进式信息披露 - 随着内容可用而显示
- •优雅降级 - 部分数据比没有数据更好
加载状态模式
黄金规则
仅当没有数据显示时才显示加载指示器。
typescript
// 正确 - 仅当没有数据时显示加载状态
const { data, loading, error } = useGetItemsQuery();
if (error) return <ErrorState error={error} onRetry={refetch} />;
if (loading && !data) return <LoadingState />;
if (!data?.items.length) return <EmptyState />;
return <ItemList items={data.items} />;
typescript
// 错误 - 即使我们有缓存数据也显示菊花 if (loading) return <LoadingState />; // 在重新获取时闪烁!
加载状态决策树
code
是否有错误? → 是:显示错误状态和重试选项 → 否:继续 是否正在加载且没有数据? → 是:显示加载指示器(菊花/骨架) → 否:继续 是否有数据? → 是,有项目:显示数据 → 是,但为空:显示空状态 → 否:显示加载(备用)
骨架 vs 菊花
| 使用骨架时 | 使用菊花时 |
|---|---|
| 已知内容形状 | 未知内容形状 |
| 列表/卡片布局 | 模态框操作 |
| 初始页面加载 | 按钮提交 |
| 内容占位符 | 内联操作 |
错误处理模式
错误处理层次
code
1. 内联错误(字段级) → 表单验证错误 2. Toast通知 → 可恢复的错误,用户可以重试 3. 错误横幅 → 页面级错误,数据仍然部分可用 4. 完整错误屏幕 → 不可恢复,需要用户操作
总是显示错误
关键:永远不要静默吞下错误。
typescript
// 正确 - 错误总是显示给用户
const [createItem, { loading }] = useCreateItemMutation({
onCompleted: () => {
toast.success({ title: "项目已创建" });
},
onError: (error) => {
console.error("createItem failed:", error);
toast.error({ title: "创建项目失败" });
},
});
// 错误 - 错误被静默捕获,用户什么都看不到
const [createItem] = useCreateItemMutation({
onError: (error) => {
console.error(error); // 用户看不到任何东西!
},
});
错误状态组件模式
typescript
interface ErrorStateProps {
error: Error;
onRetry?: () => void;
title?: string;
}
const ErrorState = ({ error, onRetry, title }: ErrorStateProps) => (
<div className="error-state">
<Icon name="exclamation-circle" />
<h3>{title ?? '出错了'}</h3>
<p>{error.message}</p>
{onRetry && (
<Button onClick={onRetry}>重试</Button>
)}
</div>
);
按钮状态模式
按钮加载状态
tsx
<Button
onClick={handleSubmit}
isLoading={isSubmitting}
disabled={!isValid || isSubmitting}
>
提交
</Button>
在操作期间禁用
关键:总是在异步操作期间禁用触发器。
tsx
// 正确 - 加载时按钮被禁用
<Button
disabled={isSubmitting}
isLoading={isSubmitting}
onClick={handleSubmit}
>
提交
</Button>
// 错误 - 用户可以多次点击
<Button onClick={handleSubmit}>
{isSubmitting ? '正在提交...' : '提交'}
</Button>
空状态
空状态要求
每个列表/集合必须有一个空状态:
tsx
// 错误 - 没有空状态
return <FlatList data={items} />;
// 正确 - 显式空状态
return <FlatList data={items} ListEmptyComponent={<EmptyState />} />;
语境化空状态
tsx
// 搜索无结果
<EmptyState
icon="search"
title="未找到结果"
description="尝试不同的搜索词"
/>
// 列表还没有项目
<EmptyState
icon="plus-circle"
title="还没有项目"
description="创建您的第一个项目"
action={{ label: '创建项目', onClick: handleCreate }}
/>
表单提交模式
tsx
const MyForm = () => {
const [submit, { loading }] = useSubmitMutation({
onCompleted: handleSuccess,
onError: handleError,
});
const handleSubmit = async () => {
if (!isValid) {
toast.error({ title: "请修复错误" });
return;
}
await submit({ variables: { input: values } });
};
return (
<form>
<Input
value={values.name}
onChange={handleChange("name")}
error={touched.name ? errors.name : undefined}
/>
<Button
type="submit"
onClick={handleSubmit}
disabled={!isValid || loading}
isLoading={loading}
>
提交
</Button>
</form>
);
};
反面模式
加载状态
typescript
// 错误 - 当数据存在时显示菊花(导致闪烁) if (loading) return <Spinner />; // 正确 - 仅在没有数据时显示加载 if (loading && !data) return <Spinner />;
错误处理
typescript
// 错误 - 错误被吞下
try {
await mutation();
} catch (e) {
console.log(e); // 用户完全不知道!
}
// 正确 - 错误显示
onError: (error) => {
console.error("operation failed:", error);
toast.error({ title: "操作失败" });
};
按钮状态
typescript
// 错误 - 提交期间按钮未禁用
<Button onClick={submit}>提交</Button>
// 正确 - 禁用并显示加载
<Button onClick={submit} disabled={loading} isLoading={loading}>
提交
</Button>
检查清单
在完成任何UI组件前:
UI状态:
- • 错误状态已处理并显示给用户
- • 加载状态仅在没有数据时显示
- • 为集合提供了空状态
- • 异步操作期间按钮被禁用
- • 适当时显示加载指示器
数据和Mutation:
- • Mutation有onError处理器
- • 所有用户操作都有反馈(toast/视觉)
与其他技能的集成
- •graphql-schema: 使用带正确错误处理的mutation模式
- •testing-patterns: 测试所有UI状态(加载、错误、空、成功)
- •formik-patterns: 应用表单提交模式