docs-components

MDX Component Patterns

Safety Notice

This listing is imported from skills.sh public index metadata. Review upstream SKILL.md and repository scripts before running.

Copy this and send it to your AI assistant to learn

Install skill "docs-components" with this command: npx skills add reactjs/react.dev/reactjs-react-dev-docs-components

MDX Component Patterns

Quick Reference

Component Decision Tree

Need Component

Helpful tip or terminology <Note>

Common mistake warning <Pitfall>

Advanced technical explanation <DeepDive>

Canary-only feature <Canary> or <CanaryBadge />

Server Components only <RSC>

Deprecated API <Deprecated>

Experimental/WIP <Wip>

Visual diagram <Diagram>

Multiple related examples <Recipes>

Interactive code <Sandpack> (see /docs-sandpack )

Console error display <ConsoleBlock>

End-of-page exercises <Challenges> (Learn pages only)

Heading Level Conventions

Component Heading Level

DeepDive title

(h4)

Titled Pitfall

(h5)

Titled Note

(h4)

Recipe items

(h4)

Challenge items

(h4)

Callout Spacing Rules

Callout components (Note, Pitfall, DeepDive) require a blank line after the opening tag before content begins.

Never place consecutively:

  • <Pitfall> followed by <Pitfall>

  • Combine into one with titled subsections, or separate with prose

  • <Note> followed by <Note>

  • Combine into one, or separate with prose

Allowed consecutive patterns:

  • <DeepDive> followed by <DeepDive>

  • OK for multi-part explorations (see useMemo.md)

  • <Pitfall> followed by <DeepDive>

  • OK when DeepDive explains "why" behind the Pitfall

Separation content: Prose paragraphs, code examples (Sandpack), or section headers.

Why: Consecutive warnings create a "wall of cautions" that overwhelms readers and causes important warnings to be skimmed.

Incorrect:

<Pitfall> Don't do X. </Pitfall>

<Pitfall> Don't do Y. </Pitfall>

Correct - combined:

<Pitfall>

Don't do X {/pitfall-x/}

Explanation.

Don't do Y {/pitfall-y/}

Explanation.

</Pitfall>

Correct - separated:

<Pitfall> Don't do X. </Pitfall>

This leads to another common mistake:

<Pitfall> Don't do Y. </Pitfall>

<Note>

Important clarifications, conventions, or tips. Less severe than Pitfall.

Simple Note

<Note>

The optimization of caching return values is known as memoization.

</Note>

Note with Title

Use #### (h4) heading with an ID.

<Note>

There is no directive for Server Components. {/no-directive/}

A common misunderstanding is that Server Components are denoted by "use server", but there is no directive for Server Components. The "use server" directive is for Server Functions.

</Note>

Version-Specific Note

<Note>

Starting in React 19, you can render &#x3C;SomeContext> as a provider.

In older versions of React, use &#x3C;SomeContext.Provider>.

</Note>

<Pitfall>

Common mistakes that cause bugs. Use for errors readers will likely make.

Simple Pitfall

<Pitfall>

We recommend defining components as functions instead of classes. See how to migrate.

</Pitfall>

Titled Pitfall

Use ##### (h5) heading with an ID.

<Pitfall>

Calling different memoized functions will read from different caches. {/pitfall-different-caches/}

To access the same cache, components must call the same memoized function.

</Pitfall>

Pitfall with Wrong/Right Code

<Pitfall>

useFormStatus will not return status information for a &#x3C;form> rendered in the same component. {/pitfall-same-component/}
function Form() {
  // 🔴 `pending` will never be true
  const { pending } = useFormStatus();
  return &#x3C;form action={submit}>&#x3C;/form>;
}

Instead call useFormStatus
 from inside a component located inside &#x3C;form>
.

&#x3C;DeepDive>

Optional deep technical content. First child must be ####
 heading with ID.

Standard DeepDive

&#x3C;DeepDive>

#### Is using an updater always preferred? {/*is-updater-preferred*/}

You might hear a recommendation to always write code like `setAge(a => a + 1)` if the state you're setting is calculated from the previous state. There's no harm in it, but it's also not always necessary.

In most cases, there is no difference between these two approaches. React always makes sure that for intentional user actions, like clicks, the `age` state variable would be updated before the next click.

&#x3C;/DeepDive>

Comparison DeepDive

For comparing related concepts:

&#x3C;DeepDive>

#### When should I use `cache`, `memo`, or `useMemo`? {/*cache-memo-usememo*/}

All mentioned APIs offer memoization but differ in what they memoize, who can access the cache, and when their cache is invalidated.

#### `useMemo` {/*deep-dive-usememo*/}

In general, you should use `useMemo` for caching expensive computations in Client Components across renders.

#### `cache` {/*deep-dive-cache*/}

In general, you should use `cache` in Server Components to memoize work that can be shared across components.

&#x3C;/DeepDive>

&#x3C;Recipes>

Multiple related examples showing variations. Each recipe needs &#x3C;Solution />
.

&#x3C;Recipes titleText="Basic useState examples" titleId="examples-basic">

#### Counter (number) {/*counter-number*/}

In this example, the `count` state variable holds a number.

&#x3C;Sandpack>
{/* code */}
&#x3C;/Sandpack>

&#x3C;Solution />

#### Text field (string) {/*text-field-string*/}

In this example, the `text` state variable holds a string.

&#x3C;Sandpack>
{/* code */}
&#x3C;/Sandpack>

&#x3C;Solution />

&#x3C;/Recipes>

Common titleText/titleId combinations:

- "Basic [hookName] examples" / examples-basic

- "Examples of [concept]" / examples-[concept]

- "The difference between [A] and [B]" / examples-[topic]

&#x3C;Challenges>

End-of-page exercises. Learn pages only. Each challenge needs problem + solution Sandpack.

&#x3C;Challenges>

#### Fix the bug {/*fix-the-bug*/}

Problem description...

&#x3C;Hint>
Optional hint text.
&#x3C;/Hint>

&#x3C;Sandpack>
{/* problem code */}
&#x3C;/Sandpack>

&#x3C;Solution>

Explanation...

&#x3C;Sandpack>
{/* solution code */}
&#x3C;/Sandpack>

&#x3C;/Solution>

&#x3C;/Challenges>

Guidelines:

- Only at end of standard Learn pages

- No Challenges in chapter intros or tutorials

- Each challenge has ####
 heading with ID

&#x3C;Deprecated>

For deprecated APIs. Content should explain what to use instead.

Page-Level Deprecation

&#x3C;Deprecated>

In React 19, `forwardRef` is no longer necessary. Pass `ref` as a prop instead.

`forwardRef` will be deprecated in a future release. Learn more [here](/blog/2024/04/25/react-19#ref-as-a-prop).

&#x3C;/Deprecated>

Method-Level Deprecation

### `componentWillMount()` {/*componentwillmount*/}

&#x3C;Deprecated>

This API has been renamed from `componentWillMount` to [`UNSAFE_componentWillMount`.](#unsafe_componentwillmount)

Run the [`rename-unsafe-lifecycles` codemod](codemod-link) to automatically update.

&#x3C;/Deprecated>

&#x3C;RSC>

For APIs that only work with React Server Components.

Basic RSC

&#x3C;RSC>

`cache` is only for use with [React Server Components](/reference/rsc/server-components).

&#x3C;/RSC>

Extended RSC (for Server Functions)

&#x3C;RSC>

Server Functions are for use in [React Server Components](/reference/rsc/server-components).

**Note:** Until September 2024, we referred to all Server Functions as "Server Actions".

&#x3C;/RSC>

&#x3C;Canary>
 and &#x3C;CanaryBadge />

For features only available in Canary releases.

Canary Wrapper (inline in Intro)

&#x3C;Intro>

`&#x3C;Fragment>` lets you group elements without a wrapper node.

&#x3C;Canary>Fragments can also accept refs, enabling interaction with underlying DOM nodes.&#x3C;/Canary>

&#x3C;/Intro>

CanaryBadge in Section Headings

### &#x3C;CanaryBadge /> FragmentInstance {/*fragmentinstance*/}

CanaryBadge in Props Lists

* &#x3C;CanaryBadge /> **optional** `ref`: A ref object from `useRef` or callback function.

CanaryBadge in Caveats

* &#x3C;CanaryBadge /> If you want to pass `ref` to a Fragment, you can't use the `&#x3C;>...&#x3C;/>` syntax.

&#x3C;Diagram>

Visual explanations of module dependencies, render trees, or data flow.

&#x3C;Diagram name="use_client_module_dependency" height={250} width={545} alt="A tree graph with the top node representing the module 'App.js'. 'App.js' has three children...">
`'use client'` segments the module dependency tree, marking `InspirationGenerator.js` and all dependencies as client-rendered.
&#x3C;/Diagram>

Attributes:

- name
: Diagram identifier (used for image file)

- height
: Height in pixels

- width
: Width in pixels

- alt
: Accessible description of the diagram

&#x3C;CodeStep>
 (Use Sparingly)

Numbered callouts in prose. Pairs with code block annotations.

Syntax

In code blocks:

```js [[1, 4, "age"], [2, 4, "setAge"], [3, 4, "42"]]
import { useState } from 'react';

function MyComponent() {
  const [age, setAge] = useState(42);
}

Format: `[[step_number, line_number, "text_to_highlight"], ...]`

In prose:
```mdx
1. The &#x3C;CodeStep step={1}>current state&#x3C;/CodeStep> initially set to the &#x3C;CodeStep step={3}>initial value&#x3C;/CodeStep>.
2. The &#x3C;CodeStep step={2}>`set` function&#x3C;/CodeStep> that lets you change it.

Guidelines

- Maximum 2-3 different colors per explanation

- Don't highlight every keyword - only key concepts

- Use for terms in prose, not entire code blocks

- Maintain consistent usage within a section

✅ Good use - highlighting key concepts:

React will compare the &#x3C;CodeStep step={2}>dependencies&#x3C;/CodeStep> with the dependencies you passed...

🚫 Avoid - excessive highlighting:

When an &#x3C;CodeStep step={1}>Activity&#x3C;/CodeStep> boundary is &#x3C;CodeStep step={2}>hidden&#x3C;/CodeStep> during its &#x3C;CodeStep step={3}>initial&#x3C;/CodeStep> render...

&#x3C;ConsoleBlock>

Display console output (errors, warnings, logs).

&#x3C;ConsoleBlock level="error">
Uncaught Error: Too many re-renders.
&#x3C;/ConsoleBlock>

Levels: error
, warning
, info

Component Usage by Page Type

Reference Pages

For component placement rules specific to Reference pages, invoke /docs-writer-reference
.

Key placement patterns:

- &#x3C;RSC>
 goes before &#x3C;Intro>
 at top of page

- &#x3C;Deprecated>
 goes after &#x3C;Intro>
 for page-level deprecation

- &#x3C;Deprecated>
 goes after method heading for method-level deprecation

- &#x3C;Canary>
 wrapper goes inline within &#x3C;Intro>

- &#x3C;CanaryBadge />
 appears in headings, props lists, and caveats

Learn Pages

For Learn page structure and patterns, invoke /docs-writer-learn
.

Key usage patterns:

- Challenges only at end of standard Learn pages

- No Challenges in chapter intros or tutorials

- DeepDive for optional advanced content

- CodeStep should be used sparingly

Blog Pages

For Blog page structure and patterns, invoke /docs-writer-blog
.

Key usage patterns:

- Generally avoid deep technical components

- Note and Pitfall OK for clarifications

- Prefer inline explanations over DeepDive

Other Available Components

Version/Status: &#x3C;Experimental>
, &#x3C;ExperimentalBadge />
, &#x3C;RSCBadge />
, &#x3C;NextMajor>
, &#x3C;Wip>

Visuals: &#x3C;DiagramGroup>
, &#x3C;Illustration>
, &#x3C;IllustrationBlock>
, &#x3C;CodeDiagram>
, &#x3C;FullWidth>

Console: &#x3C;ConsoleBlockMulti>
, &#x3C;ConsoleLogLine>

Specialized: &#x3C;TerminalBlock>
, &#x3C;BlogCard>
, &#x3C;TeamMember>
, &#x3C;YouTubeIframe>
, &#x3C;ErrorDecoder />
, &#x3C;LearnMore>
, &#x3C;Math>
, &#x3C;MathI>
, &#x3C;LanguageList>

See existing docs for usage examples of these components.

Source Transparency

This detail page is rendered from real SKILL.md content. Trust labels are metadata-based hints, not a safety guarantee.

Related Skills

Related by shared tags or category signals.

Coding

react-expert

No summary provided by upstream source.

Repository SourceNeeds Review
Coding

docs-sandpack

No summary provided by upstream source.

Repository SourceNeeds Review
Coding

review-docs

No summary provided by upstream source.

Repository SourceNeeds Review