← 返回 Skills 市场
iliaal

ia-react-frontend

作者 Ilia Alshanetsky · GitHub ↗ · v3.0.3 · MIT-0
cross-platform ✓ 安全检测通过
381
总下载
0
收藏
0
当前安装
10
版本数
在 OpenClaw 中安装
/install compound-eng-react-frontend
功能描述
React architecture patterns, TypeScript, Next.js, hooks, and testing. Use when working with React component structure, state management, Next.js routing, Vit...
使用说明 (SKILL.md)

React Frontend

Verify before implementing: For App Router patterns, React 19 APIs, or version-specific behavior, search current docs via search_docs before writing code. Training data may lag current releases.

Component TypeScript

  • Extend native elements with ComponentPropsWithoutRef\x3C'button'>, add custom props via intersection
  • Use React.ReactNode for children, React.ReactElement for single element, render prop (data: T) => ReactNode
  • Discriminated unions for variant props -- TypeScript narrows automatically in branches
  • Generic components: \x3CT> with keyof T for column keys, T extends { id: string } for constraints
  • Event types: React.MouseEvent\x3CHTMLButtonElement>, FormEvent\x3CHTMLFormElement>, ChangeEvent\x3CHTMLInputElement>
  • as const for custom hook tuple returns
  • useRef\x3CHTMLInputElement>(null) for DOM (use ?.), useRef\x3Cnumber>(0) for mutable values
  • Explicit useState\x3CUser | null>(null) for unions/null
  • useReducer actions as discriminated unions: { type: 'set'; payload: number } | { type: 'reset' }
  • useContext null guard: throw in custom useX() hook if context is null

Effects Decision Tree

Effects are escape hatches -- most logic should NOT use effects.

Need Solution
Derived value from props/state Calculate during render (useMemo if expensive)
Reset state on prop change key prop on component
Respond to user event Event handler
Notify parent of state change Call onChange in event handler, or fully controlled component
Chain of state updates Calculate all next state in one event handler
Sync with external system Effect with cleanup

Effect rules:

  • Never suppress the linter -- fix the code instead
  • Use updater functions (setItems(prev => [...prev, item])) to remove state dependencies
  • Move objects/functions inside effects to stabilize dependencies
  • useEffectEvent for non-reactive values (e.g., theme in a connection effect)
  • Always return cleanup for subscriptions, connections, listeners
  • Data fetching cancellation (pick by situation): AbortController for fetch; ignore flag for non-cancellable promises; React Query handles both automatically

Concurrency & Race Classes

Frontend bugs that survive type-checking and unit tests usually land in one of five race classes. Hunt each one explicitly during review:

  1. Lifecycle cleanup gaps -- in-production signal: "Can't perform state update on an unmounted component" warnings, slow-burn memory leaks under rapid navigation, duplicate event handlers firing. Root cause: useEffect registered a listener/timer/observer without returning cleanup (see Effect rules above for the rule).
  2. Remount-timing mistakes -- async callbacks mutate DOM or state after swap / disconnect / route change. Classic cases: a fetch().then(setData) resolves after navigation to a different route; a requestAnimationFrame fires after the parent unmounts. See "Data fetching" in Effect rules above for the cancellation hierarchy.
  3. Boolean-as-state for UI that isn't binary -- isLoading: boolean can't represent idle | loading | success | error | retry without creating inconsistent combinations (isLoading: true, error: Error is contradictory). Prefer an explicit state constant ('idle' | 'loading' | 'success' | 'error') with a transition function so invalid states are unreachable.
  4. Stale promises and timers with no cancel path -- a promise chain or setTimeout holds a reference to setState after the component's moved on. Bind every async operation to a cancel mechanism per the cancellation hierarchy above, and verify the cleanup path is exercised by a test.
  5. Per-element handlers where delegation would be safer -- attaching onClick to every row in a list creates N closures and N subscriptions; delegated listeners (single handler on the parent reading event.target.closest(...)) are safer under rapid re-renders, avoid stale-closure bugs, and scale to large lists. Use delegation when the list exceeds ~50 items or updates frequently.

These classes produce bugs that are intermittent, environment-dependent, and invisible to type-checking -- exactly the ones that reach production. Review for them deliberately, not just as "subscriptions need cleanup."

State Management

Local UI state       → useState, useReducer
Shared client state  → Zustand (simple) | Redux Toolkit (complex)
Atomic/granular      → Jotai
Server/remote data   → React Query (TanStack Query)
URL state            → nuqs, router search params
Form state           → React Hook Form

Key patterns:

  • Zustand: create\x3CState>()(devtools(persist((set) => ({...})))) -- use slices for scale, selective subscriptions to prevent re-renders
  • React Query: query keys factory (['users', 'detail', id] as const), staleTime/gcTime, optimistic updates with onMutate/onError rollback
  • Separate client state (Zustand) from server state (React Query) -- never duplicate server data in client store
  • Colocate state close to where it's used; don't over-globalize

Performance

Critical -- eliminate waterfalls:

  • Promise.all() for independent async operations
  • Move await into branches where actually needed
  • Suspense boundaries to stream slow content

Critical -- bundle size:

  • Import directly from modules, avoid barrel files (index.ts re-exports)
  • next/dynamic or React.lazy() for heavy components
  • Defer third-party scripts (analytics, logging) until after hydration
  • Preload on hover/focus for perceived speed
  • content-visibility: auto + contain-intrinsic-size on long lists -- skips off-screen layout/paint

Re-render optimization:

  • Derive state during render, not in effects
  • Subscribe to derived booleans, not raw objects (state.items.length > 0 not state.items)
  • Functional setState for stable callbacks: setCount(c => c + 1)
  • Lazy state init: useState(() => expensiveComputation())
  • useTransition for non-urgent updates (search filtering)
  • useDeferredValue for expensive derived UI
  • Don't subscribe to searchParams/state if only read in callbacks -- read on demand instead
  • Use ternary (condition ? \x3CA /> : \x3CB />), not && for conditionals
  • React.memo only for expensive subtrees with stable props
  • Hoist static JSX outside components

React Compiler (React 19): auto-memoizes -- write idiomatic React, remove manual useMemo/useCallback/memo. Enable via framework config (Next.js: reactCompiler: true in next.config). Non-framework: install babel-plugin-react-compiler. Keep components pure.

React 19

  • ref as prop -- forwardRef deprecated. Accept ref?: React.Ref\x3CHTMLElement> as regular prop
  • useActionState -- replaces useFormState: const [state, formAction, isPending] = useActionState(action, initialState)
  • use() -- unwrap Promise or Context during render (not in callbacks/effects). Enables conditional context reads
  • useOptimistic -- const [optimistic, addOptimistic] = useOptimistic(state, mergeFn) for instant UI feedback
  • useFormStatus -- const { pending } = useFormStatus() in child of \x3Cform action={...}>
  • Server Components -- default in App Router. Async, access DB/secrets directly. No hooks, no event handlers
  • Server Actions -- 'use server' directive. Validate inputs (Zod), revalidateTag/revalidatePath after mutations. Server Actions are public endpoints -- always verify auth/authz inside each action, not just in middleware or layout guards
  • \x3CActivity mode='visible'|'hidden'> -- preserves state/DOM for toggled components (experimental)

Next.js App Router

File conventions: page.tsx (route UI), layout.tsx (shared wrapper), template.tsx (re-mounted on navigation, unlike layout), loading.tsx (Suspense), error.tsx (error boundary), not-found.tsx (404), default.tsx (parallel route fallback), route.ts (API endpoint)

Rendering modes: Server Components (default) | Client ('use client') | Static (build) | Dynamic (request) | Streaming (progressive)

Decision: Server Component unless it needs hooks, event handlers, or browser APIs. Split: server parent + client child. Isolate interactive components as 'use client' leaf components -- keep server components static with no global state or event handlers.

Routing patterns:

  • Route groups (name) -- organize without affecting URL
  • Parallel routes @slot -- independent loading states in same layout
  • Intercepting routes (.) -- modal overlays with full-page fallback

Caching:

  • fetch(url, { cache: 'force-cache' }) -- static
  • fetch(url, { next: { revalidate: 60 } }) -- ISR
  • fetch(url, { cache: 'no-store' }) -- dynamic
  • Tag-based: fetch(url, { next: { tags: ['products'] } }) then revalidateTag('products')

Data fetching: Fetch in Server Components where data is used. Use Suspense boundaries for slow queries. React.cache() for per-request dedup. generateStaticParams for static generation. generateMetadata for dynamic SEO. Static metadata with title: { default: 'App', template: '%s | App' } for cascading page titles. after() for non-blocking side effects (logging, analytics) -- runs after response is sent. Hoist static I/O (fonts, config) to module level -- runs once, not per request.

Testing (Vitest + React Testing Library)

  • Component tests: Vitest + RTL, co-located *.test.tsx. Default for React components.
  • Hook tests: renderHook + act, co-located *.test.ts
  • Unit tests: Vitest for pure functions, utilities, services
  • E2E: Playwright for user flows and critical paths
  • Query priority: getByRole > getByLabelText > getByPlaceholderText > getByText > getByTestId
  • Mock API services and external providers; render child components real for integration confidence
  • One behavior per test with AAA structure. Name: should \x3Cbehavior> when \x3Ccondition>
  • Use userEvent over fireEvent for realistic interactions
  • findBy* for async elements, waitFor after state-triggering actions
  • vi.clearAllMocks() in beforeEach. Recreate state per test. General testing discipline (anti-patterns, rationalization resistance): see ia-writing-tests skill. See testing patterns and examples for component, hook, and mocking examples. See e2e testing for Playwright patterns.

Tailwind Integration

For Tailwind v4 configuration, utility patterns, dark mode, and component variants, see ia-tailwind-css skill.

Class sorting in JSX: when using clsx, cva, cn, tv, or tw utility functions, keep Tailwind classes in canonical order. Configure eslint-plugin-better-tailwindcss with useSortedClasses and functions: ["clsx", "cva", "cn", "tv", "tw"] to enforce this automatically across JSX attributes and helper calls.

Discipline

  • Simplicity first -- every change as simple as possible, impact minimal code
  • Only touch what's necessary -- avoid introducing unrelated changes
  • No hacky workarounds -- if a fix feels wrong, step back and implement the clean solution
  • Before adding a new abstraction, verify it appears in 3+ places

References

Verify

  • TypeScript compiles with zero errors
  • No suppressed lint rules (eslint-disable, @ts-ignore) in new code
  • useEffect dependency arrays not manually overridden
  • No forwardRef usage in React 19+ projects (use ref prop directly)
安全使用建议
This skill is an instruction-only helper for React/Next.js and appears coherent and low-risk: it asks for no credentials and installs nothing. Before using it: (1) ensure your agent has the tooling it references (e.g., a docs search tool if you want the `search_docs` step to work); (2) review any code the agent generates before committing or deploying (the skill produces patterns and suggestions but may not match your app's exact React/Next version or conventions); (3) if you do not want the agent to run autonomously against your repo, consider disabling autonomous invocation in the agent settings. If you need higher assurance, run the suggested tests (Vitest/Playwright) locally in a sandboxed environment first.
功能分析
Type: OpenClaw Skill Name: compound-eng-react-frontend Version: 3.0.3 The skill bundle provides comprehensive and safe architectural guidance for React, Next.js, and testing. It includes specific security advice regarding Server Actions (reminding the agent to verify auth/authz) and focuses on best practices like avoiding race conditions and ensuring proper cleanup in effects. The content in SKILL.md and the reference files (testing.md, e2e-testing.md) is strictly technical and aligned with the stated purpose of assisting in frontend development.
能力标签
crypto
能力评估
Purpose & Capability
Name and description match the content: TypeScript/React/Next.js/hook/testing guidance. The skill requires no binaries, env vars, or install steps — appropriate for an instruction-only linters/architecture guidance skill.
Instruction Scope
SKILL.md provides detailed coding and testing guidance and references local dev commands (npx vitest, npx playwright) and a docs lookup via `search_docs`. It does not instruct reading unrelated files or exfiltrating data. Note: it references an agent tool `search_docs` (to 'search current docs') which is not declared here — the agent must supply that capability for the instruction to work as intended.
Install Mechanism
No install spec and no code files — instruction-only. This is the lowest-risk install mechanism (nothing is written to disk by the skill itself).
Credentials
No required environment variables, credentials, or config paths are requested. The referenced configs (Playwright baseURL localhost) are standard dev/test patterns and do not indicate access to external secrets.
Persistence & Privilege
always: false and default model-invocation settings are used. The skill does not request persistent system presence or elevated privileges. Autonomous invocation is allowed by platform default, but this skill does not combine that with broad credential access or other red flags.
如何使用
  1. 确保已安装 OpenClaw(本地或 Docker 部署)
  2. 在对话框中输入安装命令:/install compound-eng-react-frontend
  3. 安装完成后,直接呼叫该 Skill 的名称或使用 /compound-eng-react-frontend 触发
  4. 根据 Skill 的参数说明提供必要输入,即可获得结构化输出
版本历史
v3.0.4
v3.0.4
v3.0.3
v3.0.3
v3.0.2
v3.0.2
v3.0.1
v3.0.1
v3.0.0
v3.0.0
v2.56.1
v2.56.1
v2.56.0
v2.56.0
v2.55.1
v2.55.1
v2.55.0
v2.55.0
v2.53.2
v2.53.2
v2.53.0
v2.53.0
元数据
Slug compound-eng-react-frontend
版本 3.0.3
许可证 MIT-0
累计安装 0
当前安装数 0
历史版本数 10
常见问题

ia-react-frontend 是什么?

React architecture patterns, TypeScript, Next.js, hooks, and testing. Use when working with React component structure, state management, Next.js routing, Vit... 它是一个面向 Claude Code / OpenClaw 的 AI Agent Skill 插件,目前累计下载 381 次。

如何安装 ia-react-frontend?

在 OpenClaw 或 Claude Code 对话框中运行命令「/install compound-eng-react-frontend」即可一键安装,无需额外配置。

ia-react-frontend 是免费的吗?

是的,ia-react-frontend 完全免费,采用 MIT-0 许可证,可自由下载、安装和使用。

ia-react-frontend 支持哪些平台?

ia-react-frontend 跨平台运行,可在任意部署了 OpenClaw / Claude Code 的环境中使用(cross-platform)。

谁开发了 ia-react-frontend?

由 Ilia Alshanetsky(@iliaal)开发并维护,当前版本 v3.0.3。

💬 留言讨论