Data mutations are done through Route actions. When the action completes, all loader data on the page is revalidated to keep your UI in sync with the data without writing any code to do it.
Route actions defined with action
are only called on the server while actions defined with clientAction
are run in the browser.
Client actions only run in the browser and take priority over a server action when both are defined.
// route('/projects/:projectId', './project.tsx')
import type { Route } from "./+types.project";
import { Form } from "react-router";
import { someApi } from "./api";
export async function clientAction({
request,
}: Route.ClientActionArgs) {
let formData = await request.formData();
let title = await formData.get("title");
let project = await someApi.updateProject({ title });
return project;
}
export default function Project({
actionData,
}: Route.ComponentProps) {
return (
<div>
<h1>Project</h1>
<Form method="post">
<input type="text" name="title" />
<button type="submit">Submit</button>
</Form>
{actionData ? (
<p>{actionData.title} updated</p>
) : null}
</div>
);
}
Server actions only run on the server and are removed from client bundles.
// route('/projects/:projectId', './project.tsx')
import type { Route } from "./+types.project";
import { Form } from "react-router";
import { fakeDb } from "../db";
export async function action({
request,
}: Route.ActionArgs) {
let formData = await request.formData();
let title = await formData.get("title");
let project = await fakeDb.updateProject({ title });
return project;
}
export default function Project({
actionData,
}: Route.ComponentProps) {
return (
<div>
<h1>Project</h1>
<Form method="post">
<input type="text" name="title" />
<button type="submit">Submit</button>
</Form>
{actionData ? (
<p>{actionData.title} updated</p>
) : null}
</div>
);
}
If you need to return a custom HTTP status code or custom headers from your action
, you can do so using the data
utility:
// route('/projects/:projectId', './project.tsx')
import type { Route } from "./+types.project";
import { data } from "react-router";
import { fakeDb } from "../db";
export async function action({
request,
}: Route.ActionArgs) {
let formData = await request.formData();
let title = await formData.get("title");
if (!title) {
throw data(
{ message: "Invalid title" },
{ status: 400 }
);
}
if (!projectExists(title)) {
let project = await fakeDb.createProject({ title });
return data(project, { status: 201 });
} else {
let project = await fakeDb.updateProject({ title });
return project;
}
}
Actions are called declaratively through <Form>
and imperatively through useSubmit
(or <fetcher.Form>
and fetcher.submit
) by referencing the route's path and a "post" method.
import { Form } from "react-router";
function SomeComponent() {
return (
<Form action="/projects/123" method="post">
<input type="text" name="title" />
<button type="submit">Submit</button>
</Form>
);
}
This will cause a navigation and a new entry will be added to the browser history.
You can submit form data to an action imperatively with useSubmit
.
import { useCallback } from "react";
import { useSubmit } from "react-router";
import { useFakeTimer } from "fake-lib";
function useQuizTimer() {
let submit = useSubmit();
let cb = useCallback(() => {
submit(
{ quizTimedOut: true },
{ action: "/end-quiz", method: "post" }
);
}, []);
let tenMinutes = 10 * 60 * 1000;
useFakeTimer(tenMinutes, cb);
}
This will cause a navigation and a new entry will be added to the browser history.
Fetchers allow you to submit data to actions (and loaders) without causing a navigation (no new entries in the browser history).
import { useFetcher } from "react-router";
function Task() {
let fetcher = useFetcher();
let busy = fetcher.state !== "idle";
return (
<fetcher.Form method="post" action="/update-task/123">
<input type="text" name="title" />
<button type="submit">
{busy ? "Saving..." : "Save"}
</button>
</fetcher.Form>
);
}
They also have the imperative submit
method.
fetcher.submit(
{ title: "New Title" },
{ action: "/update-task/123", method: "post" }
);
See the Using Fetchers guide for more information.