AgentSkillsCN

react-ui-patterns

用于加载状态、错误处理和数据获取的现代React UI模式。在构建UI组件、处理异步数据或管理UI状态时使用。

中文原作
SKILL.md
--- frontmatter
name: react-ui-patterns
description: 用于加载状态、错误处理和数据获取的现代React UI模式。在构建UI组件、处理异步数据或管理UI状态时使用。

React UI 模式

核心原则

  1. 永远不显示过期UI - 加载菊花仅在实际加载时显示
  2. 总是显示错误 - 用户必须知道什么时候发生故障
  3. 乐观更新 - 使UI感觉即时
  4. 渐进式信息披露 - 随着内容可用而显示
  5. 优雅降级 - 部分数据比没有数据更好

加载状态模式

黄金规则

仅当没有数据显示时才显示加载指示器。

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: 应用表单提交模式