docs-writer-reference
Reference Page Writer
Quick Reference
Page Type Decision Tree
- Is it a Hook? Use Type A (Hook/Function)
- Is it a React component (
<Something>)? Use Type B (Component) - Is it a compiler configuration option? Use Type C (Configuration)
- Is it a directive (
'use something')? Use Type D (Directive) - Is it an ESLint rule? Use Type E (ESLint Rule)
- Is it listing multiple APIs? Use Type F (Index/Category)
Component Selection
For component selection and patterns, invoke /docs-components.
Voice & Style
Voice: Authoritative technical reference writer Tone: Precise, comprehensive, neutral
For tone, capitalization, jargon, and prose patterns, invoke /docs-voice.
Do:
- Start with single-line description: "
useStateis a React Hook that lets you..." - Include Parameters, Returns, Caveats sections for every API
- Document edge cases most developers will encounter
- Use section dividers between major sections
- Include "See more examples below" links
- Be assertive, not hedging - "This is designed for..." not "This helps avoid issues with..."
- State facts, not benefits - "The callback always accesses the latest values" not "This helps avoid stale closures"
- Use minimal but meaningful names -
onEventoronTickoveronSomething
Don't:
- Skip the InlineToc component
- Omit error cases or caveats
- Use conversational language
- Mix teaching with reference (that's Learn's job)
- Document past bugs or fixed issues
- Include niche edge cases (e.g.,
thisbinding, rare class patterns) - Add phrases explaining "why you'd want this" - the Usage section examples do that
- Exception: Pitfall and DeepDive asides can use slightly conversational phrasing
Page Templates
Type A: Hook/Function
When to use: Documenting React hooks and standalone functions (useState, useEffect, memo, lazy, etc.)
---
title: hookName
---
<Intro>
`hookName` is a React Hook that lets you [brief description].
```js
const result = hookName(arg)
Reference {/reference/}
hookName(arg) {/hookname/}
Call hookName at the top level of your component to...
[signature example with annotations]
Parameters {/parameters/}
arg: Description of the parameter.
Returns {/returns/}
Description of return value.
Caveats {/caveats/}
- Important caveat about usage.
Usage {/usage/}
Common Use Case {/common-use-case/}
Explanation with Sandpack examples...
Troubleshooting {/troubleshooting/}
Common Problem {/common-problem/}
How to solve it...
---
### Type B: Component
**When to use:** Documenting React components (Suspense, Fragment, Activity, StrictMode)
```mdx
---
title: <ComponentName>
---
<Intro>
`<ComponentName>` lets you [primary action].
```js
<ComponentName prop={value}>
<Children />
</ComponentName>
Reference {/reference/}
<ComponentName> {/componentname/}
[Component purpose and behavior]
Props {/props/}
propName: Description of the prop...- optional
optionalProp: Description...
Caveats {/caveats/}
- [Caveats specific to this component]
**Key differences from Hook pages:**
- Title uses JSX syntax: `<ComponentName>`
- Uses `#### Props` instead of `#### Parameters`
- Reference heading uses JSX: `` ### `<ComponentName>` ``
---
### Type C: Configuration
**When to use:** Documenting React Compiler configuration options
```mdx
---
title: optionName
---
<Intro>
The `optionName` option [controls/specifies/determines] [what it does].
</Intro>
```js
{
optionName: 'value' // Quick example
}
Reference {/reference/}
optionName {/optionname/}
[Description of the option's purpose]
Type {/type/}
'value1' | 'value2' | 'value3'
Default value {/default-value/}
'value1'
Options {/options/}
'value1'(default): Description'value2': Description'value3': Description
Caveats {/caveats/}
- [Usage caveats]
---
### Type D: Directive
**When to use:** Documenting directives like 'use server', 'use client', 'use memo'
```mdx
---
title: "'use directive'"
titleForTitleTag: "'use directive' directive"
---
<RSC>
`'use directive'` is for use with [React Server Components](/reference/rsc/server-components).
</RSC>
<Intro>
`'use directive'` marks [what it marks] for [purpose].
```js {1}
function MyComponent() {
'use directive';
// ...
}
Reference {/reference/}
'use directive' {/use-directive/}
Add 'use directive' at the beginning of [location] to [action].
Caveats {/caveats/}
'use directive'must be at the very beginning...- The directive must be written with single or double quotes, not backticks.
- [Other placement/syntax caveats]
**Key characteristics:**
- Title includes quotes: `title: "'use server'"`
- Uses `titleForTitleTag` for browser tab title
- `<RSC>` block appears before `<Intro>`
- Caveats focus on placement and syntax requirements
---
### Type E: ESLint Rule
**When to use:** Documenting ESLint plugin rules
```mdx
---
title: rule-name
---
<Intro>
Validates that [what the rule checks].
</Intro>
## Rule Details {/*rule-details*/}
[Explanation of why this rule exists and React's underlying assumptions]
## Common Violations {/*common-violations*/}
[Description of violation patterns]
### Invalid {/*invalid*/}
Examples of incorrect code for this rule:
```js
// X Missing dependency
useEffect(() => {
console.log(count);
}, []); // Missing 'count'
Valid {/valid/}
Examples of correct code for this rule:
// checkmark All dependencies included
useEffect(() => {
console.log(count);
}, [count]);
Troubleshooting {/troubleshooting/}
[Problem description] {/problem-slug/}
[Solution]
Options {/options/}
[Configuration options if applicable]
**Key characteristics:**
- Intro is a single "Validates that..." sentence
- Uses "Invalid"/"Valid" sections with emoji-prefixed code comments
- Rule Details explains "why" not just "what"
---
### Type F: Index/Category
**When to use:** Overview pages listing multiple APIs in a category
```mdx
---
title: "Built-in React [Type]"
---
<Intro>
*Concept* let you [purpose]. Brief scope statement.
</Intro>
---
## Category Name {/*category-name*/}
*Concept* explanation with [Learn section link](/learn/topic).
To [action], use one of these [Type]:
* [`apiName`](/reference/react/apiName) lets you [action].
* [`apiName`](/reference/react/apiName) declares [thing].
```js
function Example() {
const value = useHookName(args);
}
Your own [Type] {/your-own-type/}
You can also define your own as JavaScript functions.
**Key characteristics:**
- Title format: "Built-in React [Type]"
- Italicized concept definitions
- Horizontal rules between sections
- Closes with "Your own [Type]" section
---
## Advanced Patterns
### Multi-Function Documentation
**When to use:** When a hook returns a function that needs its own documentation (useState's setter, useReducer's dispatch)
```md
### `hookName(args)` {/*hookname*/}
[Main hook documentation]
#### Parameters {/*parameters*/}
#### Returns {/*returns*/}
#### Caveats {/*caveats*/}
---
### `set` functions, like `setSomething(nextState)` {/*setstate*/}
The `set` function returned by `hookName` lets you [action].
#### Parameters {/*setstate-parameters*/}
#### Returns {/*setstate-returns*/}
#### Caveats {/*setstate-caveats*/}
Key conventions:
- Horizontal rule (
---) separates main hook from returned function - Heading IDs include prefix:
{/*setstate-parameters*/}vs{/*parameters*/} - Use generic names: "set functions" not "setCount"
Compound Return Objects
When to use: When a function returns an object with multiple properties/methods (createContext)
### `createContext(defaultValue)` {/*createcontext*/}
[Main function documentation]
#### Returns {/*returns*/}
`createContext` returns a context object.
**The context object itself does not hold any information.** It represents...
* `SomeContext` lets you provide the context value.
* `SomeContext.Consumer` is an alternative way to read context.
---
### `SomeContext` Provider {/*provider*/}
[Documentation for Provider]
#### Props {/*provider-props*/}
---
### `SomeContext.Consumer` {/*consumer*/}
[Documentation for Consumer]
#### Props {/*consumer-props*/}
Writing Patterns
Opening Lines by Page Type
| Page Type | Pattern | Example |
|---|---|---|
| Hook | `hookName` is a React Hook that lets you [action]. |
"useState is a React Hook that lets you add a state variable to your component." |
| Component | `<ComponentName>` lets you [action]. |
"<Suspense> lets you display a fallback until its children have finished loading." |
| API | `apiName` lets you [action]. |
"memo lets you skip re-rendering a component when its props are unchanged." |
| Configuration | The `optionName` option [controls/specifies/determines] [what]. |
"The target option specifies which React version the compiler generates code for." |
| Directive | `'directive'` [marks/opts/prevents] [what] for [purpose]. |
"'use server' marks a function as callable from the client." |
| ESLint Rule | Validates that [condition]. |
"Validates that dependency arrays for React hooks contain all necessary dependencies." |
Parameter Patterns
Simple parameter:
* `paramName`: Description of what it does.
Optional parameter:
* **optional** `paramName`: Description of what it does.
Parameter with special function behavior:
* `initialState`: The value you want the state to be initially. It can be a value of any type, but there is a special behavior for functions. This argument is ignored after the initial render.
* If you pass a function as `initialState`, it will be treated as an _initializer function_. It should be pure, should take no arguments, and should return a value of any type.
Callback parameter with sub-parameters:
* `subscribe`: A function that takes a single `callback` argument and subscribes it to the store. When the store changes, it should invoke the provided `callback`. The `subscribe` function should return a function that cleans up the subscription.
Nested options object:
* **optional** `options`: An object with options for this React root.
* **optional** `onCaughtError`: Callback called when React catches an error in an Error Boundary.
* **optional** `onUncaughtError`: Callback called when an error is thrown and not caught.
* **optional** `identifierPrefix`: A string prefix React uses for IDs generated by `useId`.
Return Value Patterns
Single value return:
`hookName` returns the current value. The value will be the same as `initialValue` during the first render.
Array return (numbered list):
`useState` returns an array with exactly two values:
1. The current state. During the first render, it will match the `initialState` you have passed.
2. The [`set` function](#setstate) that lets you update the state to a different value and trigger a re-render.
Object return (bulleted list):
`createElement` returns a React element object with a few properties:
* `type`: The `type` you have passed.
* `props`: The `props` you have passed except for `ref` and `key`.
* `ref`: The `ref` you have passed. If missing, `null`.
* `key`: The `key` you have passed, coerced to a string. If missing, `null`.
Promise return:
`prerender` returns a Promise:
- If rendering is successful, the Promise will resolve to an object containing:
- `prelude`: a [Web Stream](MDN-link) of HTML.
- `postponed`: a JSON-serializable object for resumption.
- If rendering fails, the Promise will be rejected.
Wrapped function return:
`cache` returns a cached version of `fn` with the same type signature. It does not call `fn` in the process.
When calling `cachedFn` with given arguments, it first checks if a cached result exists. If cached, it returns the result. If not, it calls `fn`, stores the result, and returns it.
Caveats Patterns
Standard Hook caveat (almost always first for Hooks):
* `useXxx` is a Hook, so you can only call it **at the top level of your component** or your own Hooks. You can't call it inside loops or conditions. If you need that, extract a new component and move the state into it.
Stable identity caveat (for returned functions):
* The `set` function has a stable identity, so you will often see it omitted from Effect dependencies, but including it will not cause the Effect to fire.
Strict Mode caveat:
* In Strict Mode, React will **call your render function twice** in order to help you find accidental impurities. This is development-only behavior and does not affect production.
Caveat with code example:
* It's not recommended to _suspend_ a render based on a store value returned by `useSyncExternalStore`. For example, the following is discouraged:
```js
const selectedProductId = useSyncExternalStore(...);
const data = use(fetchItem(selectedProductId)) // X Don't suspend based on store value
**Canary caveat:**
```md
* <CanaryBadge /> If you want to pass `ref` to a Fragment, you can't use the `<>...</>` syntax.
Troubleshooting Patterns
Heading format (first person problem statements):
### I've updated the state, but logging gives me the old value {/*old-value*/}
### My initializer or updater function runs twice {/*runs-twice*/}
### I want to read the latest state from a callback {/*read-latest-state*/}
Error message format:
### I'm getting an error: "Too many re-renders" {/*too-many-rerenders*/}
### I'm getting an error: "Rendered more hooks than during the previous render" {/*more-hooks*/}
Lint error format:
### I'm getting a lint error: "[exact error message]" {/*lint-error-slug*/}
Problem-solution structure:
- State the problem with code showing the issue
- Explain why it happens
- Provide the solution with corrected code
- Link to Learn section for deeper understanding
Code Comment Conventions
For code comment conventions (wrong/right, legacy/recommended, server/client labeling, bundle size annotations), invoke /docs-sandpack.
Link Description Patterns
| Pattern | Example |
|---|---|
| "lets you" + action | "memo lets you skip re-rendering when props are unchanged." |
| "declares" + thing | "useState declares a state variable that you can update directly." |
| "reads" + thing | "useContext reads and subscribes to a context." |
| "connects" + thing | "useEffect connects a component to an external system." |
| "Used with" | "Used with useContext." |
| "Similar to" | "Similar to useTransition." |
Component Patterns
For comprehensive MDX component patterns (Note, Pitfall, DeepDive, Recipes, Deprecated, RSC, Canary, Diagram, Code Steps), invoke /docs-components.
For Sandpack-specific patterns and code style, invoke /docs-sandpack.
Reference-Specific Component Rules
Component placement in Reference pages:
<RSC>goes before<Intro>at top of page<Deprecated>goes after<Intro>for page-level deprecation<Deprecated>goes after method heading for method-level deprecation<Canary>wrapper goes inline within<Intro><CanaryBadge />appears in headings, props lists, and caveats
Troubleshooting-specific components:
- Use first-person problem headings
- Cross-reference Pitfall IDs when relevant
Callout spacing:
- Never place consecutive Pitfalls or consecutive Notes
- Combine related warnings into one with titled subsections, or separate with prose/code
- Consecutive DeepDives OK for multi-part explorations
- See
/docs-componentsCallout Spacing Rules
Content Principles
Intro Section
- One sentence, ~15 words max - State what the Hook does, not how it works
- ā
"
useEffectEventis a React Hook that lets you separate events from Effects." - ā "
useEffectEventis a React Hook that lets you extract non-reactive logic from your Effects into a reusable function called an Effect Event."
Reference Code Example
- Show just the API call (5-10 lines), not a full component
- Move full component examples to Usage section
Usage Section Structure
- First example: Core mental model - Show the canonical use case with simplest concrete example
- Subsequent examples: Canonical use cases - Name the why (e.g., "Avoid reconnecting to external systems"), show a concrete how
- Prefer broad canonical use cases over multiple narrow concrete examples
- The section title IS the teaching - "When would I use this?" should be answered by the heading
What to Include vs. Exclude
- Never document past bugs or fixed issues
- Include edge cases most developers will encounter
- Exclude niche edge cases (e.g.,
thisbinding, rare class patterns)
Caveats Section
- Include rules the linter enforces or that cause immediate errors
- Include fundamental usage restrictions
- Exclude implementation details unless they affect usage
- Exclude repetition of things explained elsewhere
- Keep each caveat to one sentence when possible
Troubleshooting Section
- Error headings only: "I'm getting an error: '[message]'" format
- Never document past bugs - if it's fixed, it doesn't belong here
- Focus on errors developers will actually encounter today
DeepDive Content
- Goldilocks principle - Deep enough for curious developers, short enough to not overwhelm
- Answer "why is it designed this way?" - not exhaustive technical details
- Readers who skip it should miss nothing essential for using the API
- If the explanation is getting long, you're probably explaining too much
Domain-Specific Guidance
Hooks
Returned function documentation:
- Document setter/dispatch functions as separate
###sections - Use generic names: "set functions" not "setCount"
- Include stable identity caveat for returned functions
Dependency array documentation:
- List what counts as reactive values
- Explain when dependencies are ignored
- Link to removing effect dependencies guide
Recipes usage:
- Group related examples with meaningful titleText
- Each recipe has brief intro, Sandpack, and
<Solution />
Components
Props documentation:
- Use
#### Propsinstead of#### Parameters - Mark optional props with
**optional**prefix - Use
<CanaryBadge />inline for canary-only props
JSX syntax in titles/headings:
- Frontmatter title:
title: <Suspense> - Reference heading:
### `<Suspense>` {/*suspense*/}
React-DOM
Common props linking:
`<input>` supports all [common element props.](/reference/react-dom/components/common#common-props)
Props categorization:
- Controlled vs uncontrolled props grouped separately
- Form-specific props documented with action patterns
- MDN links for standard HTML attributes
Environment-specific notes:
<Note>
This API is specific to Node.js. Environments with [Web Streams](MDN-link), like Deno and modern edge runtimes, should use [`renderToReadableStream`](/reference/react-dom/server/renderToReadableStream) instead.
</Note>
Progressive enhancement:
- Document benefits for users without JavaScript
- Explain Server Function + form action integration
- Show hidden form field and
.bind()patterns
RSC
RSC banner (before Intro):
Always place <RSC> component before <Intro> for Server Component-only APIs.
Serialization type lists: When documenting Server Function arguments, list supported types:
Supported types for Server Function arguments:
* Primitives
* [string](MDN-link)
* [number](MDN-link)
* Iterables containing serializable values
* [Array](MDN-link)
* [Map](MDN-link)
Notably, these are not supported:
* React elements, or [JSX](/learn/writing-markup-with-jsx)
* Functions (other than Server Functions)
Bundle size comparisons:
- Show "Not included in bundle" for server-only imports
- Annotate client bundle sizes with gzip:
// 35.9K (11.2K gzipped)
Compiler
Configuration page structure:
- Type (union type or interface)
- Default value
- Options/Valid values with descriptions
Directive documentation:
- Placement requirements are critical
- Mode interaction tables showing combinations
- "Use sparingly" + "Plan for removal" patterns for escape hatches
Library author guides:
- Audience-first intro
- Benefits/Why section
- Numbered step-by-step setup
ESLint
Rule Details section:
- Explain "why" not just "what"
- Focus on React's underlying assumptions
- Describe consequences of violations
Invalid/Valid sections:
- Standard intro: "Examples of [in]correct code for this rule:"
- Use X emoji for invalid, checkmark for valid
- Show inline comments explaining the violation
Configuration options:
- Show shared settings (preferred)
- Show rule-level options (backward compatibility)
- Note precedence when both exist
Edge Cases
For deprecated, canary, and version-specific component patterns (placement, syntax, examples), invoke /docs-components.
Quick placement rules:
<Deprecated>after<Intro>for page-level, after heading for method-level<Canary>wrapper inline in Intro,<CanaryBadge />in headings/props/caveats- Version notes use
<Note>with "Starting in React 19..." pattern
Removed APIs on index pages:
## Removed APIs {/*removed-apis*/}
These APIs were removed in React 19:
* [`render`](https://18.react.dev/reference/react-dom/render): use [`createRoot`](/reference/react-dom/client/createRoot) instead.
Link to previous version docs (18.react.dev) for removed API documentation.
Critical Rules
- Heading IDs required:
## Title {/*title-id*/}(lowercase, hyphens) - Sandpack main file needs
export default - Active file syntax:
```js src/File.js active - Error headings in Troubleshooting: Use
### I'm getting an error: "[message]" {/*id*/} - Section dividers (
---) required between headings (see Section Dividers below) - InlineToc required: Always include
<InlineToc />after Intro - Consistent parameter format: Use
* \paramName`: descriptionwithoptional` prefix for optional params - Numbered lists for array returns: When hooks return arrays, use numbered lists in Returns section
- Generic names for returned functions: Use "set functions" not "setCount"
- Props vs Parameters: Use
#### Propsfor Components (Type B),#### Parametersfor Hooks/APIs (Type A) - RSC placement:
<RSC>component goes before<Intro>, not after - Canary markers: Use
<Canary>wrapper inline in Intro,<CanaryBadge />in headings/props - Deprecated placement:
<Deprecated>goes after<Intro>for page-level, after heading for method-level - Code comment emojis: Use X for wrong, checkmark for correct in code examples
- No consecutive Pitfalls/Notes: Combine into one component with titled subsections, or separate with prose/code (see
/docs-components)
For component heading level conventions (DeepDive, Pitfall, Note, Recipe headings), see /docs-components.
Section Dividers
Use --- horizontal rules to visually separate major sections:
- After
<InlineToc />- Before## Referenceheading - Between API subsections - Between different function/hook definitions (e.g., between
useState()andset functions) - Before
## Usage- Separates API reference from examples - Before
## Troubleshooting- Separates content from troubleshooting - Between EVERY Usage subsections - When switching to a new major use case
Always have a blank line before and after ---.
Section ID Conventions
| Section | ID Format |
|---|---|
| Main function | {/*functionname*/} |
| Returned function | {/*setstate*/}, {/*dispatch*/} |
| Sub-section of returned function | {/*setstate-parameters*/} |
| Troubleshooting item | {/*problem-description-slug*/} |
| Pitfall | {/*pitfall-description*/} |
| Deep dive | {/*deep-dive-topic*/} |