Loading...
Loading...
Build React applications using React Router's data mode with createBrowserRouter and RouterProvider. Use when working with route objects, loaders, actions, Form, useFetcher, or pending/optimistic UI without the Vite plugin.
npx skill4agent add remix-run/agent-skills react-router-data-modecreateBrowserRouterRouterProvidercreateBrowserRouterloaderaction<Link><NavLink><Form>redirectuseNavigateuseNavigationuseFetcher| Reference | Use When |
|---|---|
| Configuring routes, nested routes, layout |
| Understanding route object properties |
| Loading data with loaders |
| Handling forms, mutations, validation |
| Links, programmatic navigation, redirects |
| Loading states, optimistic UI |
| Server-side rendering with data mode |
import { createBrowserRouter, RouterProvider } from "react-router";
const router = createBrowserRouter([
{
path: "/",
Component: Root,
children: [
{ index: true, Component: Home },
{ path: "about", Component: About },
],
},
]);
ReactDOM.createRoot(root).render(<RouterProvider router={router} />);<Form method="get">onSubmitsetSearchParams// ✅ Correct
<Form method="get">
<input name="q" />
</Form>
// ❌ Wrong - don't manually handle search params
<form onSubmit={(e) => { e.preventDefault(); setSearchParams(...) }}>useFetcher<Form>const fetcher = useFetcher();
const optimistic = fetcher.formData
? fetcher.formData.get("favorite") === "true"
: isFavorite;
<fetcher.Form method="post" action={`/favorites/${id}`}>
<button>{optimistic ? "★" : "☆"}</button>
</fetcher.Form>;references/actions.mdfetcher.formDatafunction FavoriteButton({ itemId, isFavorite }) {
const fetcher = useFetcher();
// Optimistic: use pending form data, fallback to server state
const optimistic = fetcher.formData
? fetcher.formData.get("favorite") === "true"
: isFavorite;
return (
<fetcher.Form method="post" action={`/items/${itemId}/favorite`}>
<input type="hidden" name="favorite" value={String(!optimistic)} />
<button>{optimistic ? "★" : "☆"}</button>
</fetcher.Form>
);
}references/pending-ui.md