New to React 18
Some new APIs in React let you mark some state updates as non-urgent in order to improve performance, and keep the browser responsive.
- startTransition()
- useTransition()
- useDeferredValue()
You can use startTransition to navigate between screens without blocking user input. Or useDeferredValue to throttle expensive re-renders.
Transitions are a new concurrent feature introduced in React 18. They allow you to mark updates as transitions, which tells React that they can be interrupted and avoid going back to Suspense fallbacks for already visible content. Source
Note that last piece of information will make more sense if you review suspense_component.md.
However, long term, we expect the main way you’ll add concurrency to your app is by using a concurrent-enabled library or framework. In most cases, you won’t interact with concurrent APIs directly. For example, instead of developers calling startTransition whenever they navigate to a new screen, router libraries will automatically wrap navigations in startTransition. Source
A transition is a new concept in React to distinguish between urgent and non-urgent updates.
- Urgent updates reflect direct interaction, like typing, clicking, pressing, and so on.
- Transition updates transition the UI from one view to another.
For example, when you select a filter in a dropdown, you expect the filter button itself to respond immediately when you click. However, the actual results may transition separately. Typically, for the best user experience, a single user input should result in both an urgent update and a non-urgent one.
import { startTransition } from 'react';
// Urgent: Show what was typed
setInputValue(input);
// Mark any state updates inside as transitions
startTransition(() => {
// Transition: Show the results
setSearchQuery(input);
});
Updates wrapped in startTransition are handled as non-urgent and will be interrupted if more urgent updates like clicks or key presses come in. If a transition gets interrupted by the user (for example, by typing multiple characters in a row), React will throw out the stale rendering work that wasn’t finished and render only the latest update.
There are two transition APIs:
useTransition()
hook to start transitions, including a value to track the pending state.startTransition()
function returned byuseTransition
lets you mark a state update as a Transition.
The early example from the React API docs was pretty uninspired. They have since updated it: see https://react.dev/reference/react/useTransition.
import React, { useState, useTransition } from 'react';
import Spinner from './Spinner';
function DemoTransitions() {
const [isPending, startTransition] = useTransition();
const [count, setCount] = useState(0);
const handleClick = () => {
startTransition(() => {
setCount(c => c + 1);
});
};
return (
<div>
<p>Transition Content</p>
<button onClick={handleClick}>{count}</button>
{isPending && <Spinner />}
</div>
);
}
export default DemoTransitions;
My example using server actions:
'use client';
import { useState, useTransition } from 'react';
import { getColor } from '@/app/_lib/actions';
interface Color {
name: string | null;
value: string | null;
}
export default function ServerActionWithoutForm() {
const [color, setColor] = useState<Color>({ name: null, value: null });
const [isPending, startTransition] = useTransition();
// Server Actions are exposed server endpoints and can be called anywhere in client code.
// When using a Server Action outside of a form, call the Server Action in a Transition, which
// allows you to display a loading indicator, show optimistic state updates, and handle
// unexpected errors. Forms will automatically wrap Server Actions in transitions.
// See: https://react.dev/reference/rsc/use-server#calling-a-server-action-outside-of-form
const handleGetColor = async () => {
startTransition(async () => {
const res = await getColor('rgb');
if (res.status === 'SUCCESS' && res.data) {
console.log('Success getting color');
setColor({ name: res.data.name, value: res.data.value });
} else {
console.error('Error fetching color');
}
});
};
return (
<main>
<p>client-side data fetching with server actions.</p>
<button onClick={handleGetColor} disabled={isPending}>Get color</button>
{isPending && <span> getting color...</span>}
{color.value && (
<p>
color:{' '}
<span style={{ color: color.value }}>
{color.name} {color.value}
</span>
</p>
)}
</main>
);
}
In summary:
- Updates in a transition yield to more urgent updates such as clicks.
- Updates in a transitions will not show a fallback for re-suspended content, allowing the user to continue interacting while rendering the update.
See also: suspense_component.md
useDeferredValue lets you defer re-rendering a non-urgent part of the tree. It is similar to debouncing, but has a few advantages compared to it. There is no fixed time delay, so React will attempt the deferred render right after the first render is reflected on the screen. The deferred render is interruptible and doesn’t block user input. Source
useDeferredValue accepts a value and returns a new copy of the value that will defer to more urgent updates. If the current render is the result of an urgent update, like user input, React will return the previous value and then render the new value after the urgent render has completed. Source
In simpler terms useDeferredValue
only kicks in when/if the computer is slow.
const [items, setItems] = React.useState(defaultItems);
const deferredItems = React.useDeferredValue(items);