frontend-react-best-practices

33 React performance and composition rules across bundle optimization, re-rendering, rendering, hooks, and component patterns. Covers 6 rule categories: bundle size optimization (barrel imports, conditional loading, preloading), re-render prevention (functional setState, derived state, memoization), rendering performance (content-visibility, hydration, transitions), client patterns (passive listeners, localStorage versioning), hooks best practices (limiting useEffect, named functions), and composition patterns (compound components, explicit variants, avoiding boolean props) Includes 33 specific rules with code examples showing anti-patterns and recommended approaches for each guideline Addresses common React pitfalls: stale closures, unnecessary effects, hydration flicker, boolean prop explosion, and over-abstraction Designed for reference during component writing, code review, refactoring, and bundle optimization tasks

INSTALLATION
npx skills add https://github.com/sergiodxa/agent-skills --skill frontend-react-best-practices
Run in your project or agent environment. Adjust flags if your CLI version differs.

SKILL.md

React Best Practices

Performance optimization and composition patterns for React components. Contains 33 rules across 6 categories focused on reducing re-renders, optimizing bundles, component composition, and avoiding common React pitfalls.

When to Apply

Reference these guidelines when:

  • Writing new React components
  • Reviewing code for performance issues
  • Refactoring existing React code
  • Optimizing bundle size
  • Working with hooks and state

Rules Summary

Bundle Size Optimization (CRITICAL)

#### bundle-barrel-imports - @rules/bundle-barrel-imports.md

Import directly from source, avoid barrel files.

// Bad: loads entire library (200-800ms)

import { Check, X } from "lucide-react";

// Good: loads only what you need

import Check from "lucide-react/dist/esm/icons/check";

import X from "lucide-react/dist/esm/icons/x";

#### bundle-conditional - @rules/bundle-conditional.md

Load modules only when feature is activated.

useEffect(() => {

  if (enabled && typeof window !== "undefined") {

    import("./heavy-module").then((mod) => setModule(mod));

  }

}, [enabled]);

#### bundle-preload - @rules/bundle-preload.md

Preload on hover/focus for perceived speed.

<button

  onMouseEnter={() => import("./editor")}

  onFocus={() => import("./editor")}

  onClick={openEditor}

>

  Open Editor

</button>

Re-render Optimization (MEDIUM)

#### rerender-functional-setstate - @rules/rerender-functional-setstate.md

Use functional setState for stable callbacks.

// Bad: stale closure risk, recreates on items change

const addItem = useCallback(

  (item) => {

    setItems([...items, item]);

  },

  [items],

);

// Good: always uses latest state, stable reference

const addItem = useCallback((item) => {

  setItems((curr) => [...curr, item]);

}, []);

#### rerender-derived-state-no-effect - @rules/rerender-derived-state-no-effect.md

Derive state during render, not in effects.

// Bad: extra state and effect, extra render

const [fullName, setFullName] = useState("");

useEffect(() => {

  setFullName(firstName + " " + lastName);

}, [firstName, lastName]);

// Good: derived directly during render

const fullName = firstName + " " + lastName;

#### rerender-lazy-state-init - @rules/rerender-lazy-state-init.md

Pass function to useState for expensive initial values.

// Bad: runs expensiveComputation() on every render

const [data] = useState(expensiveComputation());

// Good: runs only on initial render

const [data] = useState(() => expensiveComputation());

#### rerender-dependencies - @rules/rerender-dependencies.md

Use primitive dependencies in effects.

// Bad: runs on any user field change

useEffect(() => {

  console.log(user.id);

}, [user]);

// Good: runs only when id changes

useEffect(() => {

  console.log(user.id);

}, [user.id]);

#### rerender-derived-state - @rules/rerender-derived-state.md

Subscribe to derived booleans, not raw values.

// Bad: re-renders on every pixel change

const width = useWindowWidth();

const isMobile = width < 768;

// Good: re-renders only when boolean changes

const isMobile = useMediaQuery("(max-width: 767px)");

#### rerender-memo - @rules/rerender-memo.md

Extract expensive work into memoized components.

// Good: skips computation when loading

const UserAvatar = memo(function UserAvatar({ user }) {

  let id = useMemo(() => computeAvatarId(user), [user]);

  return <Avatar id={id} />;

});

function Profile({ user, loading }) {

  if (loading) return <Skeleton />;

  return <UserAvatar user={user} />;

}

#### rerender-memo-with-default-value - @rules/rerender-memo-with-default-value.md

Hoist default non-primitive props to constants.

// Bad: breaks memoization (new function each render)

const Button = memo(({ onClick = () => {} }) => ...)

// Good: stable default value

const NOOP = () => {}

const Button = memo(({ onClick = NOOP }) => ...)

#### rerender-simple-expression-in-memo - @rules/rerender-simple-expression-in-memo.md

Don't wrap simple primitive expressions in useMemo.

// Bad: useMemo overhead > expression cost

const isLoading = useMemo(() => a.loading || b.loading, [a.loading, b.loading]);

// Good: just compute it

const isLoading = a.loading || b.loading;

#### rerender-move-effect-to-event - @rules/rerender-move-effect-to-event.md

Put interaction logic in event handlers, not effects.

// Bad: effect re-runs on theme change

useEffect(() => {

  if (submitted) post("/api/register");

}, [submitted, theme]);

// Good: in handler

const handleSubmit = () => post("/api/register");

#### rerender-transitions - @rules/rerender-transitions.md

Use startTransition for non-urgent updates.

// Good: non-blocking scroll tracking

const handler = () => {

  startTransition(() => setScrollY(window.scrollY));

};

#### rerender-use-ref-transient-values - @rules/rerender-use-ref-transient-values.md

Use refs for transient frequent values.

// Good: no re-render, direct DOM update

const lastXRef = useRef(0);

const dotRef = useRef<HTMLDivElement>(null);

useEffect(() => {

  let onMove = (e) => {

    lastXRef.current = e.clientX;

    dotRef.current?.style.transform = `translateX(${e.clientX}px)`;

  };

  window.addEventListener("mousemove", onMove);

  return () => window.removeEventListener("mousemove", onMove);

}, []);

Rendering Performance (MEDIUM)

#### rendering-conditional-render - @rules/rendering-conditional-render.md

Use ternary, not &#x26;&#x26; for conditionals with numbers.

// Bad: renders "0" when count is 0

{

  count &#x26;&#x26; <Badge>{count}</Badge>;

}

// Good: renders nothing when count is 0

{

  count > 0 ? <Badge>{count}</Badge> : null;

}

#### rendering-hoist-jsx - @rules/rendering-hoist-jsx.md

Extract static JSX outside components.

// Good: reuses same element, especially for large SVGs

const skeleton = <div className="animate-pulse h-20 bg-gray-200" />;

function Container({ loading }) {

  return loading ? skeleton : <Content />;

}

#### rendering-content-visibility - @rules/rendering-content-visibility.md

Use content-visibility for long lists.

.list-item {

  content-visibility: auto;

  contain-intrinsic-size: 0 80px;

}

#### rendering-animate-svg-wrapper - @rules/rendering-animate-svg-wrapper.md

Animate wrapper div, not SVG element (for GPU acceleration).

// Good: hardware accelerated

<div className="animate-spin">

  <svg>...</svg>

</div>

#### rendering-svg-precision - @rules/rendering-svg-precision.md

Reduce SVG coordinate precision with SVGO.

npx svgo --precision=1 --multipass icon.svg

#### rendering-hydration-no-flicker - @rules/rendering-hydration-no-flicker.md

Use inline script for client-only data to prevent flicker.

<div id="theme-wrapper">{children}</div>

<script dangerouslySetInnerHTML={{ __html: `

  var theme = localStorage.getItem('theme') || 'light';

  document.getElementById('theme-wrapper').className = theme;

` }} />

#### rendering-hydration-suppress-warning - @rules/rendering-hydration-suppress-warning.md

Suppress expected hydration mismatches.

<span suppressHydrationWarning>{new Date().toLocaleString()}</span>

#### rendering-client-only - @rules/rendering-client-only.md

Render browser-only components with ClientOnly and a fallback.

<ClientOnly fallback={<Skeleton />}>

  {() => <Map />}

</ClientOnly>

#### rendering-use-hydrated - @rules/rendering-use-hydrated.md

Use useHydrated for SSR/CSR divergence.

let hydrated = useHydrated();

return hydrated ? <Widget /> : <Skeleton />;

#### rendering-usetransition-loading - @rules/rendering-usetransition-loading.md

Prefer useTransition over manual loading states.

const [isPending, startTransition] = useTransition();

let handleSearch = (value) => {

  startTransition(async () => {

    let data = await fetchResults(value);

    setResults(data);

  });

};

#### fault-tolerant-error-boundaries - @rules/fault-tolerant-error-boundaries.md

Place error boundaries at feature boundaries.

<ErrorBoundary fallback={<SidebarError />}>

  <Sidebar />

</ErrorBoundary>

Client Patterns (MEDIUM)

#### client-passive-event-listeners - @rules/client-passive-event-listeners.md

Use passive listeners for scroll/touch.

document.addEventListener("wheel", handler, { passive: true });

document.addEventListener("touchstart", handler, { passive: true });

#### client-localstorage-schema - @rules/client-localstorage-schema.md

Version and minimize localStorage data.

const VERSION = "v2";

function saveConfig(config: Config) {

  try {

    localStorage.setItem(`config:${VERSION}`, JSON.stringify(config));

  } catch {} // Handle incognito/quota exceeded

}

Hooks (HIGH)

#### hooks-limit-useeffect - @rules/hooks-limit-useeffect.md

Use useEffect only when absolutely necessary. Prefer derived state or event handlers.

// Bad: useEffect to derive state

let [filtered, setFiltered] = useState(items);

useEffect(() => {

  setFiltered(items.filter((i) => i.active));

}, [items]);

// Good: derive during render

let filtered = items.filter((i) => i.active);

// Good: useMemo if expensive

let filtered = useMemo(() => items.filter((i) => i.active), [items]);

#### hooks-useeffect-named-functions - @rules/hooks-useeffect-named-functions.md

Use named function declarations in useEffect for better debugging and self-documentation.

// Bad: anonymous arrow function

useEffect(() => {

  document.title = title;

}, [title]);

// Good: named function

useEffect(

  function syncDocumentTitle() {

    document.title = title;

  },

  [title],

);

// Good: also name cleanup functions

useEffect(function subscribeToOnlineStatus() {

  window.addEventListener("online", handleOnline);

  return function unsubscribeFromOnlineStatus() {

    window.removeEventListener("online", handleOnline);

  };

}, []);

Composition Patterns (HIGH)

#### composition-avoid-boolean-props - @rules/composition-avoid-boolean-props.md

Don't add boolean props to customize behavior. Use composition instead.

// Bad: boolean prop explosion

<Composer isThread isEditing={false} showAttachments />

// Good: explicit variants

<ThreadComposer channelId="abc" />

<EditComposer messageId="xyz" />

#### composition-compound-components - @rules/composition-compound-components.md

Structure complex components as compound components with shared context.

// Good: compound components

<Composer.Provider state={state} actions={actions}>

  <Composer.Frame>

    <Composer.Input />

    <Composer.Footer>

      <Composer.Submit />

    </Composer.Footer>

  </Composer.Frame>

</Composer.Provider>

#### composition-state-provider - @rules/composition-state-provider.md

Lift state into provider components for cross-component access.

// Good: state in provider, accessible anywhere inside

<ForwardMessageProvider>

  <Dialog>

    <Composer.Input />

    <MessagePreview /> {/* Can read state */}

    <ForwardButton /> {/* Can call submit */}

  </Dialog>

</ForwardMessageProvider>

#### composition-explicit-variants - @rules/composition-explicit-variants.md

Create explicit variant components instead of prop combinations.

// Good: self-documenting variants

function ThreadComposer({ channelId }) {

  return (

    <ThreadProvider channelId={channelId}>

      <Composer.Frame>

        <Composer.Input />

        <AlsoSendToChannelField />

        <Composer.Submit />

      </Composer.Frame>

    </ThreadProvider>

  );

}

#### composition-children-over-render-props - @rules/composition-children-over-render-props.md

Prefer children for composition. Use render props only when passing data back.

// Good: children for structure

<Card>

  <Card.Header>Title</Card.Header>

  <Card.Body>Content</Card.Body>

</Card>

// OK: render props when passing data

<List renderItem={({ item }) => <Item {...item} />} />

#### composition-avoid-overabstraction - @rules/composition-avoid-overabstraction.md

Avoid rigid configuration props; prefer composable children APIs.

<Select value="abc" onChange={...}>

  <Option value="abc">ABC</Option>

  <Option value="xyz">XYZ</Option>

</Select>

#### composition-typescript-namespaces - @rules/composition-typescript-namespaces.md

Use TypeScript namespaces to combine component and its types for single-import access.

// components/button.tsx

export namespace Button {

  export type Variant = "solid" | "ghost" | "outline";

  export interface Props {

    variant?: Variant;

    children: React.ReactNode;

  }

}

export function Button({ variant = "solid", children }: Button.Props) {

  // ...

}

// Usage: single import

import { Button } from "~/components/button";

<Button variant="ghost">Click</Button>

function wrap(props: Button.Props) { ... }

Important: Namespaces should only contain types, never runtime code.

BrowserAct

Let your agent run on any real-world website

Bypass CAPTCHA & anti-bot for free. Start local, scale to cloud.

Explore BrowserAct Skills →

Stop writing automation&scrapers

Install the CLI. Run your first Skill in 30 seconds. Scale when you're ready.

Start free
free · no credit card