react-effect-decision

Combine React's official "You Might Not Need an Effect" guidance with this project's stricter no direct useEffect stance. Use when writing, reviewing, or refactoring React components that might reach for useEffect, derived state, event relays, reset logic, subscriptions, or client fetching.

607 stars

Best use case

react-effect-decision is best used when you need a repeatable AI agent workflow instead of a one-off prompt.

Combine React's official "You Might Not Need an Effect" guidance with this project's stricter no direct useEffect stance. Use when writing, reviewing, or refactoring React components that might reach for useEffect, derived state, event relays, reset logic, subscriptions, or client fetching.

Teams using react-effect-decision should expect a more consistent output, faster repeated execution, less prompt rewriting.

When to use this skill

  • You want a reusable workflow that can be run more than once with consistent structure.

When not to use this skill

  • You only need a quick one-off answer and do not need a reusable workflow.
  • You cannot install or maintain the underlying files, dependencies, or repository context.

Installation

Claude Code / Cursor / Codex

$curl -o ~/.claude/skills/react-effect-decision/SKILL.md --create-dirs "https://raw.githubusercontent.com/waynesutton/markdown-site/main/.claude/react-effect-decision/SKILL.md"

Manual Installation

  1. Download SKILL.md from GitHub
  2. Place it in .claude/skills/react-effect-decision/SKILL.md inside your project
  3. Restart your AI agent — it will auto-discover the skill

How react-effect-decision Compares

Feature / Agentreact-effect-decisionStandard Approach
Platform SupportNot specifiedLimited / Varies
Context Awareness High Baseline
Installation ComplexityUnknownN/A

Frequently Asked Questions

What does this skill do?

Combine React's official "You Might Not Need an Effect" guidance with this project's stricter no direct useEffect stance. Use when writing, reviewing, or refactoring React components that might reach for useEffect, derived state, event relays, reset logic, subscriptions, or client fetching.

Where can I find the source code?

You can find the source code on GitHub using the link provided at the top of the page.

Related Guides

SKILL.md Source

# React effect decision

Default rule for this project: do not reach for direct `useEffect` first.

Most effect usage in React code is a sign that the component is syncing state that should be derived during render, routing an event through state, or resetting local state in a way React already models better.

This skill merges:

* React's official decision tree from "You Might Not Need an Effect"
* The project's stricter no direct `useEffect` preference

## Start here

Before writing any effect, ask these in order:

1. Can I calculate this from props or state during render
2. Is this triggered by a user action and better placed in the event handler
3. Am I trying to reset component state when identity changed and should use `key`
4. Is this an expensive pure calculation that should use `useMemo`
5. Is this an external subscription that should use `useSyncExternalStore` or a focused custom hook
6. Is this real external synchronization that must happen on mount or on dependency changes

If you answer yes to any of 1 through 5, do not write a direct `useEffect`.

## Project stance

React allows effects for real external synchronization. This project is stricter:

* Avoid direct `useEffect` in components whenever a clearer pattern exists
* Prefer derived values, event handlers, `key` remounts, memoization, query libraries, Convex hooks, and focused custom hooks
* If mount only external setup is truly needed, use an explicit `useMountEffect` wrapper instead of sprinkling raw `useEffect` through feature code

Example wrapper:

```ts
import { useEffect } from "react";

export function useMountEffect(effect: () => void | (() => void)) {
  useEffect(effect, []);
}
```

Use that only for one time external setup like DOM focus, third party widgets, or browser API listeners that belong to the component lifecycle.

## Decision guide

### 1. Derived state belongs in render

Bad smell:

* `useEffect(() => setX(deriveFromY(y)), [y])`
* State that only mirrors props or other state

Do this instead:

```tsx
function TodoList({ todos, showActive }: Props) {
  const activeTodos = todos.filter((todo) => !todo.completed);
  const visibleTodos = showActive ? activeTodos : todos;

  return <List items={visibleTodos} />;
}
```

If it can be calculated from existing inputs, keep it out of state.

### 2. Expensive pure calculations use `useMemo`

If the calculation is pure and expensive, memoize it instead of syncing it into state:

```tsx
const visibleTodos = useMemo(
  () => getVisibleTodos(todos, showActive),
  [todos, showActive],
);
```

Use memoization only when there is real repeated work to skip.

### 3. Event caused work belongs in the event handler

Bad smell:

* Set a flag in state
* Wait for an effect to notice the flag
* Reset the flag after the side effect runs

Do this instead:

```tsx
function PurchaseButton({ product }: Props) {
  async function handleClick() {
    await addToCart(product);
    showNotification(`Added ${product.name}`);
  }

  return <button onClick={handleClick}>Buy</button>;
}
```

If the work happens because the user clicked, submitted, dragged, or selected, keep it in the handler.

### 4. Reset with `key`, not effect choreography

Bad smell:

* `useEffect(() => setFormState(initialFromProps), [id])`

Do this instead:

```tsx
function EditContact({ contact, onSave }: Props) {
  return <EditContactForm key={contact.id} contact={contact} onSave={onSave} />;
}
```

If a different entity should feel like a fresh component instance, give the inner component a `key`.

### 5. Adjusting state from prop changes is a last resort

If you cannot derive it and do not want a full remount, adjust the state during render of the same component before children render stale data.

Prefer these, in order:

1. Derive it
2. Reset with `key`
3. Store an ID and derive the selected object from current inputs
4. Only then use a guarded render time adjustment pattern

### 6. Data fetching should use the app's data layer or a focused hook

Bad smell:

* Raw `fetch(...).then(setState)` inside a component effect
* Manual loading, retry, cancellation, and stale response handling in feature code

Preferred options:

* Use Convex React hooks where the data already lives in Convex
* Use the project's existing data abstraction or a query library for remote APIs
* If raw client fetching is unavoidable, isolate it inside a custom hook and handle stale response cleanup

Example custom hook pattern:

```tsx
function useSearchResults(url: string) {
  const [data, setData] = useState<Result[] | null>(null);

  useEffect(() => {
    let ignore = false;

    fetch(url)
      .then((response) => response.json())
      .then((json) => {
        if (!ignore) {
          setData(json);
        }
      });

    return () => {
      ignore = true;
    };
  }, [url]);

  return data;
}
```

That is still less preferred than a dedicated query layer, but better than scattering raw fetch effects across components.

### 7. External subscriptions should use `useSyncExternalStore`

For browser or third party stores, prefer the React built in subscription model:

```tsx
function useOnlineStatus() {
  return useSyncExternalStore(
    subscribe,
    () => navigator.onLine,
    () => true,
  );
}
```

This is clearer and safer than mirroring an external mutable source through ad hoc effect code.

### 8. True external synchronization is the narrow allowed case

An effect or `useMountEffect` is appropriate when the component must synchronize with something outside React:

* DOM focus or scroll positioning
* Third party widget setup and cleanup
* Browser API subscriptions
* Network synchronization that should happen because the component is on screen, not because a user event fired

When you use one, document the external system in a comment or nearby helper.

## Quick smell test

If you are about to write `useEffect`, check:

* Is this just derived state
* Is this event specific logic
* Is this a reset that should use `key`
* Is this a pure expensive calculation
* Is this a subscription better modeled with `useSyncExternalStore`
* Is this data loading that belongs in Convex, a query library, or a custom hook
* What external system am I synchronizing with

If you cannot name the external system, you probably do not need an effect.

## Code review prompts

Use these during review:

* What breaks if this effect is removed and the value is derived during render
* Is this component storing data that can be recomputed from props or state
* Why is this work not in the click, submit, or change handler that caused it
* Would `key={entityId}` remove the reset logic entirely
* Should this be a custom hook instead of inline effect code
* Is this component talking to an external system, or only to React state

## Summary

The safest path in this repo is simple:

* Derive during render when possible
* Handle user caused work in handlers
* Reset with `key`
* Memoize expensive pure work
* Use dedicated hooks for fetching and subscriptions
* Reserve effect style lifecycle code for real external synchronization

Sources:

* [React docs: You Might Not Need an Effect](https://react.dev/learn/you-might-not-need-an-effect)

Related Skills

Update project docs

607
from waynesutton/markdown-site

Use this skill after completing any feature, fix, or migration to keep the three core project tracking files in sync.

robel-auth

607
from waynesutton/markdown-site

Integrate and maintain Robelest Convex Auth in apps by always checking upstream before implementation. Use when adding auth setup, updating auth wiring, migrating between upstream patterns, or troubleshooting @robelest/convex-auth behavior across projects.

Create a PRD

607
from waynesutton/markdown-site

Use this skill before any multi-file feature, architectural decision, or complex bug fix.

convex-self-hosting

607
from waynesutton/markdown-site

Integrate Convex static self hosting into existing apps using the latest upstream instructions from get-convex/self-hosting every time. Use when setting up upload APIs, HTTP routes, deployment scripts, migration from external hosting, or troubleshooting static deploy issues across React, Vite, Next.js, and other frontends.

convex-return-validators

607
from waynesutton/markdown-site

Guide for when to use and when not to use return validators in Convex functions. Use this skill whenever the user is writing Convex queries, mutations, or actions and needs guidance on return value validation. Also trigger when the user asks about Convex type safety, runtime validation, AI-generated Convex code, Convex AI rules, Convex security best practices, or when they're debugging return type issues in Convex functions. Trigger this skill when users mention "validators", "returns", "return type", or "exact types" in the context of Convex development. Also trigger when writing or reviewing Convex AI rules or prompts that instruct LLMs how to write Convex code.

convex-doctor

607
from waynesutton/markdown-site

Run convex-doctor static analysis, interpret findings, and fix issues across security, performance, correctness, schema, and architecture categories. Use when running convex-doctor, fixing convex-doctor warnings or errors, improving the convex-doctor score, or when asked about Convex code quality, static analysis, or linting Convex functions.

write

607
from waynesutton/markdown-site

Writing style guide for technical content, social media, blog posts, READMEs, git commits, and developer documentation. Optimized to avoid AI detection patterns. Use when writing any content beyond code.

workflow

607
from waynesutton/markdown-site

Project workflow for PRDs, task tracking, changelog sync, and documentation updates. Use for any non-trivial task that spans multiple steps, touches several files, changes architecture, or needs project tracking updates. Also activates with @update to sync task.md, changelog.md, and files.md after completing work.

sec-check

607
from waynesutton/markdown-site

Security review checklist for Convex functions, auth logic, public queries, admin routes, webhooks, uploads, and AI-generated code. Use when reviewing code that touches user data, PII, or access control.

schema-builder

607
from waynesutton/markdown-site

Design and generate Convex database schemas with proper validation, indexes, and relationships. Use when creating schema.ts or modifying table definitions.

real-time-backend

607
from waynesutton/markdown-site

Build reactive, type-safe, production-grade backends. ALWAYS use this skill when the user asks to build, plan, design, or implement backend features, APIs, data models, server logic, database schemas, web apps, full stack apps, or mobile apps. This includes planning and architecture discussions.

migration-helper

607
from waynesutton/markdown-site

Plan and execute Convex schema migrations safely, including adding fields, creating tables, and data transformations. Use when schema changes affect existing data.