useActionState
The useActionState
Hook is currently only available in React's Canary and experimental channels. Learn more about release channels here. In addition, you need to use a framework that supports React Server Components to get the full benefit of useActionState
.
In earlier React Canary versions, this API was part of React DOM and called useFormState
.
useActionState
is a Hook that allows you to update state based on the result of a form action.
const [state, formAction] = useActionState(fn, initialState, permalink?);
Reference
useActionState(action, initialState, permalink?)
Call useActionState
at the top level of your component to create component state that is updated when a form action is invoked. You pass useActionState
an existing form action function as well as an initial state, and it returns a new action that you use in your form, along with the latest form state. The latest form state is also passed to the function that you provided.
import { useActionState } from "react";
async function increment(previousState, formData) {
return previousState + 1;
}
function StatefulForm({}) {
const [state, formAction] = useActionState(increment, 0);
return (
<form>
{state}
<button formAction={formAction}>Increment</button>
</form>
)
}
The form state is the value returned by the action when the form was last submitted. If the form has not yet been submitted, it is the initial state that you pass.
If used with a Server Action, useActionState
allows the server's response from submitting the form to be shown even before hydration has completed.
Parameters
fn
: The function to be called when the form is submitted or button pressed. When the function is called, it will receive the previous state of the form (initially theinitialState
that you pass, subsequently its previous return value) as its initial argument, followed by the arguments that a form action normally receives.initialState
: The value you want the state to be initially. It can be any serializable value. This argument is ignored after the action is first invoked.- optional
permalink
: A string containing the unique page URL that this form modifies. For use on pages with dynamic content (eg: feeds) in conjunction with progressive enhancement: iffn
is a server action and the form is submitted before the JavaScript bundle loads, the browser will navigate to the specified permalink URL, rather than the current page's URL. Ensure that the same form component is rendered on the destination page (including the same actionfn
andpermalink
) so that React knows how to pass the state through. Once the form has been hydrated, this parameter has no effect.
Returns
useActionState
returns an array with exactly two values:
- The current state. During the first render, it will match the
initialState
you have passed. After the action is invoked, it will match the value returned by the action. - A new action that you can pass as the
action
prop to yourform
component orformAction
prop to anybutton
component within the form.
Caveats
- When used with a framework that supports React Server Components,
useActionState
lets you make forms interactive before JavaScript has executed on the client. When used without Server Components, it is equivalent to component local state. - The function passed to
useActionState
receives an extra argument, the previous or initial state, as its first argument. This makes its signature different than if it were used directly as a form action without usinguseActionState
.
Usage
Using information returned by a form action
Call useActionState
at the top level of your component to access the return value of an action from the last time a form was submitted.
import { useActionState } from 'react';
import { action } from './actions.js';
function MyComponent() {
const [state, formAction] = useActionState(action, null);
// ...
return (
<form action={formAction}>
{/* ... */}
</form>
);
}
useActionState
returns an array with exactly two items:
- The current state of the form, which is initially set to the 4>initial state you provided, and after the form is submitted is set to the return value of the 3>action you provided.
- A 2>new action that you pass to
<form>
as itsaction
prop.
When the form is submitted, the 3>action function that you provided will be called. Its return value will become the new current state of the form.
The 3>action that you provide will also receive a new first argument, namely the current state of the form. The first time the form is submitted, this will be the 4>initial state you provided, while with subsequent submissions, it will be the return value from the last time the action was called. The rest of the arguments are the same as if useActionState
had not been used.
function action(currentState, formData) {
// ...
return 'next state';
}
Display form errors
To display messages such as an error message or toast that's returned by a Server Action, wrap the action in a call to useActionState
.
import { useActionState, useState } from "react";
import { addToCart } from "./actions.js";
function AddToCartForm({itemID, itemTitle}) {
const [message, formAction] = useActionState(addToCart, null);
return (
<form action={formAction}>
<h2>{itemTitle}</h2>
<input type="hidden" name="itemID" value={itemID} />
<button type="submit">Add to Cart</button>
{message}
</form>
);
}
export default function App() {
return (
<>
<AddToCartForm itemID="1" itemTitle="JavaScript: The Definitive Guide" />
<AddToCartForm itemID="2" itemTitle="JavaScript: The Good Parts" />
</>
)
}
"use server";
export async function addToCart(prevState, queryData) {
const itemID = queryData.get('itemID');
if (itemID === "1") {
return "Added to cart";
} else {
return "Couldn't add to cart: the item is sold out.";
}
}
form {
border: solid 1px black;
margin-bottom: 24px;
padding: 12px
}
form button {
margin-right: 12px;
}
{
"dependencies": {
"react": "canary",
"react-dom": "canary",
"react-scripts": "^5.0.0"
},
"main": "/index.js",
"devDependencies": {}
}
Display structured information after submitting a form
The return value from a Server Action can be any serializable value. For example, it could be an object that includes a boolean indicating whether the action was successful, an error message, or updated information.
import { useActionState, useState } from "react";
import { addToCart } from "./actions.js";
function AddToCartForm({itemID, itemTitle}) {
const [formState, formAction] = useActionState(addToCart, {});
return (
<form action={formAction}>
<h2>{itemTitle}</h2>
<input type="hidden" name="itemID" value={itemID} />
<button type="submit">Add to Cart</button>
{formState?.success &&
<div className="toast">
Added to cart! Your cart now has {formState.cartSize} items.
</div>
}
{formState?.success === false &&
<div className="error">
Failed to add to cart: {formState.message}
</div>
}
</form>
);
}
export default function App() {
return (
<>
<AddToCartForm itemID="1" itemTitle="JavaScript: The Definitive Guide" />
<AddToCartForm itemID="2" itemTitle="JavaScript: The Good Parts" />
</>
)
}
"use server";
export async function addToCart(prevState, queryData) {
const itemID = queryData.get('itemID');
if (itemID === "1") {
return {
success: true,
cartSize: 12,
};
} else {
return {
success: false,
message: "The item is sold out.",
};
}
}
form {
border: solid 1px black;
margin-bottom: 24px;
padding: 12px
}
form button {
margin-right: 12px;
}
{
"dependencies": {
"react": "canary",
"react-dom": "canary",
"react-scripts": "^5.0.0"
},
"main": "/index.js",
"devDependencies": {}
}
Troubleshooting
My action can no longer read the submitted form data
When you wrap an action with useActionState
, it gets an extra argument as its first argument. The submitted form data is therefore its second argument instead of its first as it would usually be. The new first argument that gets added is the current state of the form.
function action(currentState, formData) {
// ...
}