UNPKG

47.6 kBMarkdownView Raw
1# `react-router`
2
3## 7.0.1
4
5## 7.0.0
6
7### Major Changes
8
9- Remove the original `defer` implementation in favor of using raw promises via single fetch and `turbo-stream`. This removes these exports from React Router: ([#11744](https://github.com/remix-run/react-router/pull/11744))
10
11 - `defer`
12 - `AbortedDeferredError`
13 - `type TypedDeferredData`
14 - `UNSAFE_DeferredData`
15 - `UNSAFE_DEFERRED_SYMBOL`,
16
17- - Collapse `@remix-run/router` into `react-router` ([#11505](https://github.com/remix-run/react-router/pull/11505))
18 - Collapse `react-router-dom` into `react-router`
19 - Collapse `@remix-run/server-runtime` into `react-router`
20 - Collapse `@remix-run/testing` into `react-router`
21
22- Remove single\_fetch future flag. ([#11522](https://github.com/remix-run/react-router/pull/11522))
23
24- Drop support for Node 16, React Router SSR now requires Node 18 or higher ([#11391](https://github.com/remix-run/react-router/pull/11391))
25
26- Remove `future.v7_startTransition` flag ([#11696](https://github.com/remix-run/react-router/pull/11696))
27
28- - Expose the underlying router promises from the following APIs for compsition in React 19 APIs: ([#11521](https://github.com/remix-run/react-router/pull/11521))
29 - `useNavigate()`
30 - `useSubmit`
31 - `useFetcher().load`
32 - `useFetcher().submit`
33 - `useRevalidator.revalidate`
34
35- Remove `future.v7_normalizeFormMethod` future flag ([#11697](https://github.com/remix-run/react-router/pull/11697))
36
37- For Remix consumers migrating to React Router, the `crypto` global from the [Web Crypto API](https://developer.mozilla.org/en-US/docs/Web/API/Web_Crypto_API) is now required when using cookie and session APIs. This means that the following APIs are provided from `react-router` rather than platform-specific packages: ([#11837](https://github.com/remix-run/react-router/pull/11837))
38
39 - `createCookie`
40 - `createCookieSessionStorage`
41 - `createMemorySessionStorage`
42 - `createSessionStorage`
43
44 For consumers running older versions of Node, the `installGlobals` function from `@remix-run/node` has been updated to define `globalThis.crypto`, using [Node's `require('node:crypto').webcrypto` implementation.](https://nodejs.org/api/webcrypto.html)
45
46 Since platform-specific packages no longer need to implement this API, the following low-level APIs have been removed:
47
48 - `createCookieFactory`
49 - `createSessionStorageFactory`
50 - `createCookieSessionStorageFactory`
51 - `createMemorySessionStorageFactory`
52
53- Imports/Exports cleanup ([#11840](https://github.com/remix-run/react-router/pull/11840))
54
55 - Removed the following exports that were previously public API from `@remix-run/router`
56 - types
57 - `AgnosticDataIndexRouteObject`
58 - `AgnosticDataNonIndexRouteObject`
59 - `AgnosticDataRouteMatch`
60 - `AgnosticDataRouteObject`
61 - `AgnosticIndexRouteObject`
62 - `AgnosticNonIndexRouteObject`
63 - `AgnosticRouteMatch`
64 - `AgnosticRouteObject`
65 - `TrackedPromise`
66 - `unstable_AgnosticPatchRoutesOnMissFunction`
67 - `Action` -> exported as `NavigationType` via `react-router`
68 - `Router` exported as `DataRouter` to differentiate from RR's `<Router>`
69 - API
70 - `getToPathname` (`@private`)
71 - `joinPaths` (`@private`)
72 - `normalizePathname` (`@private`)
73 - `resolveTo` (`@private`)
74 - `stripBasename` (`@private`)
75 - `createBrowserHistory` -> in favor of `createBrowserRouter`
76 - `createHashHistory` -> in favor of `createHashRouter`
77 - `createMemoryHistory` -> in favor of `createMemoryRouter`
78 - `createRouter`
79 - `createStaticHandler` -> in favor of wrapper `createStaticHandler` in RR Dom
80 - `getStaticContextFromError`
81 - Removed the following exports that were previously public API from `react-router`
82 - `Hash`
83 - `Pathname`
84 - `Search`
85
86- update minimum node version to 18 ([#11690](https://github.com/remix-run/react-router/pull/11690))
87
88- Remove `future.v7_prependBasename` from the ionternalized `@remix-run/router` package ([#11726](https://github.com/remix-run/react-router/pull/11726))
89
90- Migrate Remix type generics to React Router ([#12180](https://github.com/remix-run/react-router/pull/12180))
91
92 - These generics are provided for Remix v2 migration purposes
93 - These generics and the APIs they exist on should be considered informally deprecated in favor of the new `Route.*` types
94 - Anyone migrating from React Router v6 should probably not leverage these new generics and should migrate straight to the `Route.*` types
95 - For React Router v6 users, these generics are new and should not impact your app, with one exception
96 - `useFetcher` previously had an optional generic (used primarily by Remix v2) that expected the data type
97 - This has been updated in v7 to expect the type of the function that generates the data (i.e., `typeof loader`/`typeof action`)
98 - Therefore, you should update your usages:
99 - `useFetcher<LoaderData>()`
100 - `useFetcher<typeof loader>()`
101
102- Remove `future.v7_throwAbortReason` from internalized `@remix-run/router` package ([#11728](https://github.com/remix-run/react-router/pull/11728))
103
104- Add `exports` field to all packages ([#11675](https://github.com/remix-run/react-router/pull/11675))
105
106- node package no longer re-exports from react-router ([#11702](https://github.com/remix-run/react-router/pull/11702))
107
108- renamed RemixContext to FrameworkContext ([#11705](https://github.com/remix-run/react-router/pull/11705))
109
110- updates the minimum React version to 18 ([#11689](https://github.com/remix-run/react-router/pull/11689))
111
112- PrefetchPageDescriptor replaced by PageLinkDescriptor ([#11960](https://github.com/remix-run/react-router/pull/11960))
113
114- - Consolidate types previously duplicated across `@remix-run/router`, `@remix-run/server-runtime`, and `@remix-run/react` now that they all live in `react-router` ([#12177](https://github.com/remix-run/react-router/pull/12177))
115 - Examples: `LoaderFunction`, `LoaderFunctionArgs`, `ActionFunction`, `ActionFunctionArgs`, `DataFunctionArgs`, `RouteManifest`, `LinksFunction`, `Route`, `EntryRoute`
116 - The `RouteManifest` type used by the "remix" code is now slightly stricter because it is using the former `@remix-run/router` `RouteManifest`
117 - `Record<string, Route> -> Record<string, Route | undefined>`
118 - Removed `AppData` type in favor of inlining `unknown` in the few locations it was used
119 - Removed `ServerRuntimeMeta*` types in favor of the `Meta*` types they were duplicated from
120
121- - Remove the `future.v7_partialHydration` flag ([#11725](https://github.com/remix-run/react-router/pull/11725))
122 - This also removes the `<RouterProvider fallbackElement>` prop
123 - To migrate, move the `fallbackElement` to a `hydrateFallbackElement`/`HydrateFallback` on your root route
124 - Also worth nothing there is a related breaking changer with this future flag:
125 - Without `future.v7_partialHydration` (when using `fallbackElement`), `state.navigation` was populated during the initial load
126 - With `future.v7_partialHydration`, `state.navigation` remains in an `"idle"` state during the initial load
127
128- Remove `v7_relativeSplatPath` future flag ([#11695](https://github.com/remix-run/react-router/pull/11695))
129
130- Drop support for Node 18, update minimum Node vestion to 20 ([#12171](https://github.com/remix-run/react-router/pull/12171))
131
132 - Remove `installGlobals()` as this should no longer be necessary
133
134- Remove remaining future flags ([#11820](https://github.com/remix-run/react-router/pull/11820))
135
136 - React Router `v7_skipActionErrorRevalidation`
137 - Remix `v3_fetcherPersist`, `v3_relativeSplatPath`, `v3_throwAbortReason`
138
139- rename createRemixStub to createRoutesStub ([#11692](https://github.com/remix-run/react-router/pull/11692))
140
141- Remove `@remix-run/router` deprecated `detectErrorBoundary` option in favor of `mapRouteProperties` ([#11751](https://github.com/remix-run/react-router/pull/11751))
142
143- Add `react-router/dom` subpath export to properly enable `react-dom` as an optional `peerDependency` ([#11851](https://github.com/remix-run/react-router/pull/11851))
144
145 - This ensures that we don't blindly `import ReactDOM from "react-dom"` in `<RouterProvider>` in order to access `ReactDOM.flushSync()`, since that would break `createMemoryRouter` use cases in non-DOM environments
146 - DOM environments should import from `react-router/dom` to get the proper component that makes `ReactDOM.flushSync()` available:
147 - If you are using the Vite plugin, use this in your `entry.client.tsx`:
148 - `import { HydratedRouter } from 'react-router/dom'`
149 - If you are not using the Vite plugin and are manually calling `createBrowserRouter`/`createHashRouter`:
150 - `import { RouterProvider } from "react-router/dom"`
151
152- Remove `future.v7_fetcherPersist` flag ([#11731](https://github.com/remix-run/react-router/pull/11731))
153
154- Update `cookie` dependency to `^1.0.1` - please see the [release notes](https://github.com/jshttp/cookie/releases) for any breaking changes ([#12172](https://github.com/remix-run/react-router/pull/12172))
155
156### Minor Changes
157
158- - Add support for `prerender` config in the React Router vite plugin, to support existing SSG use-cases ([#11539](https://github.com/remix-run/react-router/pull/11539))
159 - You can use the `prerender` config to pre-render your `.html` and `.data` files at build time and then serve them statically at runtime (either from a running server or a CDN)
160 - `prerender` can either be an array of string paths, or a function (sync or async) that returns an array of strings so that you can dynamically generate the paths by talking to your CMS, etc.
161
162 ```ts
163 // react-router.config.ts
164 import type { Config } from "@react-router/dev/config";
165
166 export default {
167 async prerender() {
168 let slugs = await fakeGetSlugsFromCms();
169 // Prerender these paths into `.html` files at build time, and `.data`
170 // files if they have loaders
171 return ["/", "/about", ...slugs.map((slug) => `/product/${slug}`)];
172 },
173 } satisfies Config;
174
175 async function fakeGetSlugsFromCms() {
176 await new Promise((r) => setTimeout(r, 1000));
177 return ["shirt", "hat"];
178 }
179 ```
180
181- Params, loader data, and action data as props for route component exports ([#11961](https://github.com/remix-run/react-router/pull/11961))
182
183 ```tsx
184 export default function Component({ params, loaderData, actionData }) {}
185
186 export function HydrateFallback({ params }) {}
187 export function ErrorBoundary({ params, loaderData, actionData }) {}
188 ```
189
190- Remove duplicate `RouterProvider` impliementations ([#11679](https://github.com/remix-run/react-router/pull/11679))
191
192- ### Typesafety improvements ([#12019](https://github.com/remix-run/react-router/pull/12019))
193
194 React Router now generates types for each of your route modules.
195 You can access those types by importing them from `./+types.<route filename without extension>`.
196 For example:
197
198 ```ts
199 // app/routes/product.tsx
200 import type * as Route from "./+types.product";
201
202 export function loader({ params }: Route.LoaderArgs) {}
203
204 export default function Component({ loaderData }: Route.ComponentProps) {}
205 ```
206
207 This initial implementation targets type inference for:
208
209 - `Params` : Path parameters from your routing config in `routes.ts` including file-based routing
210 - `LoaderData` : Loader data from `loader` and/or `clientLoader` within your route module
211 - `ActionData` : Action data from `action` and/or `clientAction` within your route module
212
213 In the future, we plan to add types for the rest of the route module exports: `meta`, `links`, `headers`, `shouldRevalidate`, etc.
214 We also plan to generate types for typesafe `Link`s:
215
216 ```tsx
217 <Link to="/products/:id" params={{ id: 1 }} />
218 // ^^^^^^^^^^^^^ ^^^^^^^^^
219 // typesafe `to` and `params` based on the available routes in your app
220 ```
221
222 Check out our docs for more:
223
224 - [_Explanations > Type Safety_](https://reactrouter.com/dev/guides/explanation/type-safety)
225 - [_How-To > Setting up type safety_](https://reactrouter.com/dev/guides/how-to/setting-up-type-safety)
226
227- Stabilize `unstable_dataStrategy` ([#11969](https://github.com/remix-run/react-router/pull/11969))
228
229- Stabilize `unstable_patchRoutesOnNavigation` ([#11970](https://github.com/remix-run/react-router/pull/11970))
230
231### Patch Changes
232
233- No changes ([`506329c4e`](https://github.com/remix-run/react-router/commit/506329c4e2e7aba9837cbfa44df6103b49423745))
234
235- chore: re-enable development warnings through a `development` exports condition. ([#12269](https://github.com/remix-run/react-router/pull/12269))
236
237- Remove unstable upload handler. ([#12015](https://github.com/remix-run/react-router/pull/12015))
238
239- Remove unneeded dependency on @web3-storage/multipart-parser ([#12274](https://github.com/remix-run/react-router/pull/12274))
240
241- Fix redirects returned from loaders/actions using `data()` ([#12021](https://github.com/remix-run/react-router/pull/12021))
242
243- fix(react-router): (v7) fix static prerender of non-ascii characters ([#12161](https://github.com/remix-run/react-router/pull/12161))
244
245- Replace `substr` with `substring` ([#12080](https://github.com/remix-run/react-router/pull/12080))
246
247- Remove the deprecated `json` utility ([#12146](https://github.com/remix-run/react-router/pull/12146))
248
249 - You can use [`Response.json`](https://developer.mozilla.org/en-US/docs/Web/API/Response/json_static) if you still need to construct JSON responses in your app
250
251- Remove unneeded dependency on source-map ([#12275](https://github.com/remix-run/react-router/pull/12275))
252
253## 6.28.0
254
255### Minor Changes
256
257- - Log deprecation warnings for v7 flags ([#11750](https://github.com/remix-run/react-router/pull/11750))
258 - Add deprecation warnings to `json`/`defer` in favor of returning raw objects
259 - These methods will be removed in React Router v7
260
261### Patch Changes
262
263- Update JSDoc URLs for new website structure (add /v6/ segment) ([#12141](https://github.com/remix-run/react-router/pull/12141))
264- Updated dependencies:
265 - `@remix-run/router@1.21.0`
266
267## 6.27.0
268
269### Minor Changes
270
271- Stabilize `unstable_patchRoutesOnNavigation` ([#11973](https://github.com/remix-run/react-router/pull/11973))
272 - Add new `PatchRoutesOnNavigationFunctionArgs` type for convenience ([#11967](https://github.com/remix-run/react-router/pull/11967))
273- Stabilize `unstable_dataStrategy` ([#11974](https://github.com/remix-run/react-router/pull/11974))
274- Stabilize the `unstable_flushSync` option for navigations and fetchers ([#11989](https://github.com/remix-run/react-router/pull/11989))
275- Stabilize the `unstable_viewTransition` option for navigations and the corresponding `unstable_useViewTransitionState` hook ([#11989](https://github.com/remix-run/react-router/pull/11989))
276
277### Patch Changes
278
279- Fix bug when submitting to the current contextual route (parent route with an index child) when an `?index` param already exists from a prior submission ([#12003](https://github.com/remix-run/react-router/pull/12003))
280
281- Fix `useFormAction` bug - when removing `?index` param it would not keep other non-Remix `index` params ([#12003](https://github.com/remix-run/react-router/pull/12003))
282
283- Fix types for `RouteObject` within `PatchRoutesOnNavigationFunction`'s `patch` method so it doesn't expect agnostic route objects passed to `patch` ([#11967](https://github.com/remix-run/react-router/pull/11967))
284
285- Updated dependencies:
286 - `@remix-run/router@1.20.0`
287
288## 6.26.2
289
290### Patch Changes
291
292- Updated dependencies:
293 - `@remix-run/router@1.19.2`
294
295## 6.26.1
296
297### Patch Changes
298
299- Rename `unstable_patchRoutesOnMiss` to `unstable_patchRoutesOnNavigation` to match new behavior ([#11888](https://github.com/remix-run/react-router/pull/11888))
300- Updated dependencies:
301 - `@remix-run/router@1.19.1`
302
303## 6.26.0
304
305### Minor Changes
306
307- Add a new `replace(url, init?)` alternative to `redirect(url, init?)` that performs a `history.replaceState` instead of a `history.pushState` on client-side navigation redirects ([#11811](https://github.com/remix-run/react-router/pull/11811))
308
309### Patch Changes
310
311- Fix initial hydration behavior when using `future.v7_partialHydration` along with `unstable_patchRoutesOnMiss` ([#11838](https://github.com/remix-run/react-router/pull/11838))
312 - During initial hydration, `router.state.matches` will now include any partial matches so that we can render ancestor `HydrateFallback` components
313- Updated dependencies:
314 - `@remix-run/router@1.19.0`
315
316## 6.25.1
317
318No significant changes to this package were made in this release. [See the repo `CHANGELOG.md`](https://github.com/remix-run/react-router/blob/main/CHANGELOG.md) for an overview of all changes in v6.25.1.
319
320## 6.25.0
321
322### Minor Changes
323
324- Stabilize `future.unstable_skipActionErrorRevalidation` as `future.v7_skipActionErrorRevalidation` ([#11769](https://github.com/remix-run/react-router/pull/11769))
325 - When this flag is enabled, actions will not automatically trigger a revalidation if they return/throw a `Response` with a `4xx`/`5xx` status code
326 - You may still opt-into revalidation via `shouldRevalidate`
327 - This also changes `shouldRevalidate`'s `unstable_actionStatus` parameter to `actionStatus`
328
329### Patch Changes
330
331- Fix regression and properly decode paths inside `useMatch` so matches/params reflect decoded params ([#11789](https://github.com/remix-run/react-router/pull/11789))
332- Updated dependencies:
333 - `@remix-run/router@1.18.0`
334
335## 6.24.1
336
337### Patch Changes
338
339- When using `future.v7_relativeSplatPath`, properly resolve relative paths in splat routes that are children of pathless routes ([#11633](https://github.com/remix-run/react-router/pull/11633))
340- Updated dependencies:
341 - `@remix-run/router@1.17.1`
342
343## 6.24.0
344
345### Minor Changes
346
347- Add support for Lazy Route Discovery (a.k.a. Fog of War) ([#11626](https://github.com/remix-run/react-router/pull/11626))
348 - RFC: <https://github.com/remix-run/react-router/discussions/11113>
349 - `unstable_patchRoutesOnMiss` docs: <https://reactrouter.com/v6/routers/create-browser-router>
350
351### Patch Changes
352
353- Updated dependencies:
354 - `@remix-run/router@1.17.0`
355
356## 6.23.1
357
358### Patch Changes
359
360- allow undefined to be resolved with `<Await>` ([#11513](https://github.com/remix-run/react-router/pull/11513))
361- Updated dependencies:
362 - `@remix-run/router@1.16.1`
363
364## 6.23.0
365
366### Minor Changes
367
368- Add a new `unstable_dataStrategy` configuration option ([#11098](https://github.com/remix-run/react-router/pull/11098))
369 - This option allows Data Router applications to take control over the approach for executing route loaders and actions
370 - The default implementation is today's behavior, to fetch all loaders in parallel, but this option allows users to implement more advanced data flows including Remix single-fetch, middleware/context APIs, automatic loader caching, and more
371
372### Patch Changes
373
374- Updated dependencies:
375 - `@remix-run/router@1.16.0`
376
377## 6.22.3
378
379### Patch Changes
380
381- Updated dependencies:
382 - `@remix-run/router@1.15.3`
383
384## 6.22.2
385
386### Patch Changes
387
388- Updated dependencies:
389 - `@remix-run/router@1.15.2`
390
391## 6.22.1
392
393### Patch Changes
394
395- Fix encoding/decoding issues with pre-encoded dynamic parameter values ([#11199](https://github.com/remix-run/react-router/pull/11199))
396- Updated dependencies:
397 - `@remix-run/router@1.15.1`
398
399## 6.22.0
400
401### Patch Changes
402
403- Updated dependencies:
404 - `@remix-run/router@1.15.0`
405
406## 6.21.3
407
408### Patch Changes
409
410- Remove leftover `unstable_` prefix from `Blocker`/`BlockerFunction` types ([#11187](https://github.com/remix-run/react-router/pull/11187))
411
412## 6.21.2
413
414### Patch Changes
415
416- Updated dependencies:
417 - `@remix-run/router@1.14.2`
418
419## 6.21.1
420
421### Patch Changes
422
423- Fix bug with `route.lazy` not working correctly on initial SPA load when `v7_partialHydration` is specified ([#11121](https://github.com/remix-run/react-router/pull/11121))
424- Updated dependencies:
425 - `@remix-run/router@1.14.1`
426
427## 6.21.0
428
429### Minor Changes
430
431- Add a new `future.v7_relativeSplatPath` flag to implement a breaking bug fix to relative routing when inside a splat route. ([#11087](https://github.com/remix-run/react-router/pull/11087))
432
433 This fix was originally added in [#10983](https://github.com/remix-run/react-router/issues/10983) and was later reverted in [#11078](https://github.com/remix-run/react-router/pull/11078) because it was determined that a large number of existing applications were relying on the buggy behavior (see [#11052](https://github.com/remix-run/react-router/issues/11052))
434
435 **The Bug**
436 The buggy behavior is that without this flag, the default behavior when resolving relative paths is to _ignore_ any splat (`*`) portion of the current route path.
437
438 **The Background**
439 This decision was originally made thinking that it would make the concept of nested different sections of your apps in `<Routes>` easier if relative routing would _replace_ the current splat:
440
441 ```jsx
442 <BrowserRouter>
443 <Routes>
444 <Route path="/" element={<Home />} />
445 <Route path="dashboard/*" element={<Dashboard />} />
446 </Routes>
447 </BrowserRouter>
448 ```
449
450 Any paths like `/dashboard`, `/dashboard/team`, `/dashboard/projects` will match the `Dashboard` route. The dashboard component itself can then render nested `<Routes>`:
451
452 ```jsx
453 function Dashboard() {
454 return (
455 <div>
456 <h2>Dashboard</h2>
457 <nav>
458 <Link to="/">Dashboard Home</Link>
459 <Link to="team">Team</Link>
460 <Link to="projects">Projects</Link>
461 </nav>
462
463 <Routes>
464 <Route path="/" element={<DashboardHome />} />
465 <Route path="team" element={<DashboardTeam />} />
466 <Route path="projects" element={<DashboardProjects />} />
467 </Routes>
468 </div>
469 );
470 }
471 ```
472
473 Now, all links and route paths are relative to the router above them. This makes code splitting and compartmentalizing your app really easy. You could render the `Dashboard` as its own independent app, or embed it into your large app without making any changes to it.
474
475 **The Problem**
476
477 The problem is that this concept of ignoring part of a path breaks a lot of other assumptions in React Router - namely that `"."` always means the current location pathname for that route. When we ignore the splat portion, we start getting invalid paths when using `"."`:
478
479 ```jsx
480 // If we are on URL /dashboard/team, and we want to link to /dashboard/team:
481 function DashboardTeam() {
482 // ❌ This is broken and results in <a href="/dashboard">
483 return <Link to=".">A broken link to the Current URL</Link>;
484
485 // ✅ This is fixed but super unintuitive since we're already at /dashboard/team!
486 return <Link to="./team">A broken link to the Current URL</Link>;
487 }
488 ```
489
490 We've also introduced an issue that we can no longer move our `DashboardTeam` component around our route hierarchy easily - since it behaves differently if we're underneath a non-splat route, such as `/dashboard/:widget`. Now, our `"."` links will, properly point to ourself _inclusive of the dynamic param value_ so behavior will break from it's corresponding usage in a `/dashboard/*` route.
491
492 Even worse, consider a nested splat route configuration:
493
494 ```jsx
495 <BrowserRouter>
496 <Routes>
497 <Route path="dashboard">
498 <Route path="*" element={<Dashboard />} />
499 </Route>
500 </Routes>
501 </BrowserRouter>
502 ```
503
504 Now, a `<Link to=".">` and a `<Link to="..">` inside the `Dashboard` component go to the same place! That is definitely not correct!
505
506 Another common issue arose in Data Routers (and Remix) where any `<Form>` should post to it's own route `action` if you the user doesn't specify a form action:
507
508 ```jsx
509 let router = createBrowserRouter({
510 path: "/dashboard",
511 children: [
512 {
513 path: "*",
514 action: dashboardAction,
515 Component() {
516 // ❌ This form is broken! It throws a 405 error when it submits because
517 // it tries to submit to /dashboard (without the splat value) and the parent
518 // `/dashboard` route doesn't have an action
519 return <Form method="post">...</Form>;
520 },
521 },
522 ],
523 });
524 ```
525
526 This is just a compounded issue from the above because the default location for a `Form` to submit to is itself (`"."`) - and if we ignore the splat portion, that now resolves to the parent route.
527
528 **The Solution**
529 If you are leveraging this behavior, it's recommended to enable the future flag, move your splat to it's own route, and leverage `../` for any links to "sibling" pages:
530
531 ```jsx
532 <BrowserRouter>
533 <Routes>
534 <Route path="dashboard">
535 <Route index path="*" element={<Dashboard />} />
536 </Route>
537 </Routes>
538 </BrowserRouter>
539
540 function Dashboard() {
541 return (
542 <div>
543 <h2>Dashboard</h2>
544 <nav>
545 <Link to="..">Dashboard Home</Link>
546 <Link to="../team">Team</Link>
547 <Link to="../projects">Projects</Link>
548 </nav>
549
550 <Routes>
551 <Route path="/" element={<DashboardHome />} />
552 <Route path="team" element={<DashboardTeam />} />
553 <Route path="projects" element={<DashboardProjects />} />
554 </Router>
555 </div>
556 );
557 }
558 ```
559
560 This way, `.` means "the full current pathname for my route" in all cases (including static, dynamic, and splat routes) and `..` always means "my parents pathname".
561
562### Patch Changes
563
564- Properly handle falsy error values in ErrorBoundary's ([#11071](https://github.com/remix-run/react-router/pull/11071))
565- Updated dependencies:
566 - `@remix-run/router@1.14.0`
567
568## 6.20.1
569
570### Patch Changes
571
572- Revert the `useResolvedPath` fix for splat routes due to a large number of applications that were relying on the buggy behavior (see <https://github.com/remix-run/react-router/issues/11052#issuecomment-1836589329>). We plan to re-introduce this fix behind a future flag in the next minor version. ([#11078](https://github.com/remix-run/react-router/pull/11078))
573- Updated dependencies:
574 - `@remix-run/router@1.13.1`
575
576## 6.20.0
577
578### Minor Changes
579
580- Export the `PathParam` type from the public API ([#10719](https://github.com/remix-run/react-router/pull/10719))
581
582### Patch Changes
583
584- Fix bug with `resolveTo` in splat routes ([#11045](https://github.com/remix-run/react-router/pull/11045))
585 - This is a follow up to [#10983](https://github.com/remix-run/react-router/pull/10983) to handle the few other code paths using `getPathContributingMatches`
586 - This removes the `UNSAFE_getPathContributingMatches` export from `@remix-run/router` since we no longer need this in the `react-router`/`react-router-dom` layers
587- Updated dependencies:
588 - `@remix-run/router@1.13.0`
589
590## 6.19.0
591
592### Minor Changes
593
594- Add `unstable_flushSync` option to `useNavigate`/`useSumbit`/`fetcher.load`/`fetcher.submit` to opt-out of `React.startTransition` and into `ReactDOM.flushSync` for state updates ([#11005](https://github.com/remix-run/react-router/pull/11005))
595- Remove the `unstable_` prefix from the [`useBlocker`](https://reactrouter.com/v6/hooks/use-blocker) hook as it's been in use for enough time that we are confident in the API. We do not plan to remove the prefix from `unstable_usePrompt` due to differences in how browsers handle `window.confirm` that prevent React Router from guaranteeing consistent/correct behavior. ([#10991](https://github.com/remix-run/react-router/pull/10991))
596
597### Patch Changes
598
599- Fix `useActionData` so it returns proper contextual action data and not _any_ action data in the tree ([#11023](https://github.com/remix-run/react-router/pull/11023))
600
601- Fix bug in `useResolvedPath` that would cause `useResolvedPath(".")` in a splat route to lose the splat portion of the URL path. ([#10983](https://github.com/remix-run/react-router/pull/10983))
602
603 - ⚠️ This fixes a quite long-standing bug specifically for `"."` paths inside a splat route which incorrectly dropped the splat portion of the URL. If you are relative routing via `"."` inside a splat route in your application you should double check that your logic is not relying on this buggy behavior and update accordingly.
604
605- Updated dependencies:
606 - `@remix-run/router@1.12.0`
607
608## 6.18.0
609
610### Patch Changes
611
612- Fix the `future` prop on `BrowserRouter`, `HashRouter` and `MemoryRouter` so that it accepts a `Partial<FutureConfig>` instead of requiring all flags to be included. ([#10962](https://github.com/remix-run/react-router/pull/10962))
613- Updated dependencies:
614 - `@remix-run/router@1.11.0`
615
616## 6.17.0
617
618### Patch Changes
619
620- Fix `RouterProvider` `future` prop type to be a `Partial<FutureConfig>` so that not all flags must be specified ([#10900](https://github.com/remix-run/react-router/pull/10900))
621- Updated dependencies:
622 - `@remix-run/router@1.10.0`
623
624## 6.16.0
625
626### Minor Changes
627
628- In order to move towards stricter TypeScript support in the future, we're aiming to replace current usages of `any` with `unknown` on exposed typings for user-provided data. To do this in Remix v2 without introducing breaking changes in React Router v6, we have added generics to a number of shared types. These continue to default to `any` in React Router and are overridden with `unknown` in Remix. In React Router v7 we plan to move these to `unknown` as a breaking change. ([#10843](https://github.com/remix-run/react-router/pull/10843))
629 - `Location` now accepts a generic for the `location.state` value
630 - `ActionFunctionArgs`/`ActionFunction`/`LoaderFunctionArgs`/`LoaderFunction` now accept a generic for the `context` parameter (only used in SSR usages via `createStaticHandler`)
631 - The return type of `useMatches` (now exported as `UIMatch`) accepts generics for `match.data` and `match.handle` - both of which were already set to `unknown`
632- Move the `@private` class export `ErrorResponse` to an `UNSAFE_ErrorResponseImpl` export since it is an implementation detail and there should be no construction of `ErrorResponse` instances in userland. This frees us up to export a `type ErrorResponse` which correlates to an instance of the class via `InstanceType`. Userland code should only ever be using `ErrorResponse` as a type and should be type-narrowing via `isRouteErrorResponse`. ([#10811](https://github.com/remix-run/react-router/pull/10811))
633- Export `ShouldRevalidateFunctionArgs` interface ([#10797](https://github.com/remix-run/react-router/pull/10797))
634- Removed private/internal APIs only required for the Remix v1 backwards compatibility layer and no longer needed in Remix v2 (`_isFetchActionRedirect`, `_hasFetcherDoneAnything`) ([#10715](https://github.com/remix-run/react-router/pull/10715))
635
636### Patch Changes
637
638- Updated dependencies:
639 - `@remix-run/router@1.9.0`
640
641## 6.15.0
642
643### Minor Changes
644
645- Add's a new `redirectDocument()` function which allows users to specify that a redirect from a `loader`/`action` should trigger a document reload (via `window.location`) instead of attempting to navigate to the redirected location via React Router ([#10705](https://github.com/remix-run/react-router/pull/10705))
646
647### Patch Changes
648
649- Ensure `useRevalidator` is referentially stable across re-renders if revalidations are not actively occurring ([#10707](https://github.com/remix-run/react-router/pull/10707))
650- Updated dependencies:
651 - `@remix-run/router@1.8.0`
652
653## 6.14.2
654
655### Patch Changes
656
657- Updated dependencies:
658 - `@remix-run/router@1.7.2`
659
660## 6.14.1
661
662### Patch Changes
663
664- Fix loop in `unstable_useBlocker` when used with an unstable blocker function ([#10652](https://github.com/remix-run/react-router/pull/10652))
665- Fix issues with reused blockers on subsequent navigations ([#10656](https://github.com/remix-run/react-router/pull/10656))
666- Updated dependencies:
667 - `@remix-run/router@1.7.1`
668
669## 6.14.0
670
671### Patch Changes
672
673- Strip `basename` from locations provided to `unstable_useBlocker` functions to match `useLocation` ([#10573](https://github.com/remix-run/react-router/pull/10573))
674- Fix `generatePath` when passed a numeric `0` value parameter ([#10612](https://github.com/remix-run/react-router/pull/10612))
675- Fix `unstable_useBlocker` key issues in `StrictMode` ([#10573](https://github.com/remix-run/react-router/pull/10573))
676- Fix `tsc --skipLibCheck:false` issues on React 17 ([#10622](https://github.com/remix-run/react-router/pull/10622))
677- Upgrade `typescript` to 5.1 ([#10581](https://github.com/remix-run/react-router/pull/10581))
678- Updated dependencies:
679 - `@remix-run/router@1.7.0`
680
681## 6.13.0
682
683### Minor Changes
684
685- Move [`React.startTransition`](https://react.dev/reference/react/startTransition) usage behind a [future flag](https://reactrouter.com/v6/guides/api-development-strategy) to avoid issues with existing incompatible `Suspense` usages. We recommend folks adopting this flag to be better compatible with React concurrent mode, but if you run into issues you can continue without the use of `startTransition` until v7. Issues usually boils down to creating net-new promises during the render cycle, so if you run into issues you should either lift your promise creation out of the render cycle or put it behind a `useMemo`. ([#10596](https://github.com/remix-run/react-router/pull/10596))
686
687 Existing behavior will no longer include `React.startTransition`:
688
689 ```jsx
690 <BrowserRouter>
691 <Routes>{/*...*/}</Routes>
692 </BrowserRouter>
693
694 <RouterProvider router={router} />
695 ```
696
697 If you wish to enable `React.startTransition`, pass the future flag to your component:
698
699 ```jsx
700 <BrowserRouter future={{ v7_startTransition: true }}>
701 <Routes>{/*...*/}</Routes>
702 </BrowserRouter>
703
704 <RouterProvider router={router} future={{ v7_startTransition: true }}/>
705 ```
706
707### Patch Changes
708
709- Work around webpack/terser `React.startTransition` minification bug in production mode ([#10588](https://github.com/remix-run/react-router/pull/10588))
710
711## 6.12.1
712
713> \[!WARNING]
714> Please use version `6.13.0` or later instead of `6.12.1`. This version suffers from a `webpack`/`terser` minification issue resulting in invalid minified code in your resulting production bundles which can cause issues in your application. See [#10579](https://github.com/remix-run/react-router/issues/10579) for more details.
715
716### Patch Changes
717
718- Adjust feature detection of `React.startTransition` to fix webpack + react 17 compilation error ([#10569](https://github.com/remix-run/react-router/pull/10569))
719
720## 6.12.0
721
722### Minor Changes
723
724- Wrap internal router state updates with `React.startTransition` if it exists ([#10438](https://github.com/remix-run/react-router/pull/10438))
725
726### Patch Changes
727
728- Updated dependencies:
729 - `@remix-run/router@1.6.3`
730
731## 6.11.2
732
733### Patch Changes
734
735- Fix `basename` duplication in descendant `<Routes>` inside a `<RouterProvider>` ([#10492](https://github.com/remix-run/react-router/pull/10492))
736- Updated dependencies:
737 - `@remix-run/router@1.6.2`
738
739## 6.11.1
740
741### Patch Changes
742
743- Fix usage of `Component` API within descendant `<Routes>` ([#10434](https://github.com/remix-run/react-router/pull/10434))
744- Fix bug when calling `useNavigate` from `<Routes>` inside a `<RouterProvider>` ([#10432](https://github.com/remix-run/react-router/pull/10432))
745- Fix usage of `<Navigate>` in strict mode when using a data router ([#10435](https://github.com/remix-run/react-router/pull/10435))
746- Updated dependencies:
747 - `@remix-run/router@1.6.1`
748
749## 6.11.0
750
751### Patch Changes
752
753- Log loader/action errors to the console in dev for easier stack trace evaluation ([#10286](https://github.com/remix-run/react-router/pull/10286))
754- Fix bug preventing rendering of descendant `<Routes>` when `RouterProvider` errors existed ([#10374](https://github.com/remix-run/react-router/pull/10374))
755- Fix inadvertent re-renders when using `Component` instead of `element` on a route definition ([#10287](https://github.com/remix-run/react-router/pull/10287))
756- Fix detection of `useNavigate` in the render cycle by setting the `activeRef` in a layout effect, allowing the `navigate` function to be passed to child components and called in a `useEffect` there. ([#10394](https://github.com/remix-run/react-router/pull/10394))
757- Switched from `useSyncExternalStore` to `useState` for internal `@remix-run/router` router state syncing in `<RouterProvider>`. We found some [subtle bugs](https://codesandbox.io/s/use-sync-external-store-loop-9g7b81) where router state updates got propagated _before_ other normal `useState` updates, which could lead to footguns in `useEffect` calls. ([#10377](https://github.com/remix-run/react-router/pull/10377), [#10409](https://github.com/remix-run/react-router/pull/10409))
758- Allow `useRevalidator()` to resolve a loader-driven error boundary scenario ([#10369](https://github.com/remix-run/react-router/pull/10369))
759- Avoid unnecessary unsubscribe/resubscribes on router state changes ([#10409](https://github.com/remix-run/react-router/pull/10409))
760- When using a `RouterProvider`, `useNavigate`/`useSubmit`/`fetcher.submit` are now stable across location changes, since we can handle relative routing via the `@remix-run/router` instance and get rid of our dependence on `useLocation()`. When using `BrowserRouter`, these hooks remain unstable across location changes because they still rely on `useLocation()`. ([#10336](https://github.com/remix-run/react-router/pull/10336))
761- Updated dependencies:
762 - `@remix-run/router@1.6.0`
763
764## 6.10.0
765
766### Minor Changes
767
768- Added support for [**Future Flags**](https://reactrouter.com/v6/guides/api-development-strategy) in React Router. The first flag being introduced is `future.v7_normalizeFormMethod` which will normalize the exposed `useNavigation()/useFetcher()` `formMethod` fields as uppercase HTTP methods to align with the `fetch()` behavior. ([#10207](https://github.com/remix-run/react-router/pull/10207))
769
770 - When `future.v7_normalizeFormMethod === false` (default v6 behavior),
771 - `useNavigation().formMethod` is lowercase
772 - `useFetcher().formMethod` is lowercase
773 - When `future.v7_normalizeFormMethod === true`:
774 - `useNavigation().formMethod` is uppercase
775 - `useFetcher().formMethod` is uppercase
776
777### Patch Changes
778
779- Fix route ID generation when using Fragments in `createRoutesFromElements` ([#10193](https://github.com/remix-run/react-router/pull/10193))
780- Updated dependencies:
781 - `@remix-run/router@1.5.0`
782
783## 6.9.0
784
785### Minor Changes
786
787- React Router now supports an alternative way to define your route `element` and `errorElement` fields as React Components instead of React Elements. You can instead pass a React Component to the new `Component` and `ErrorBoundary` fields if you choose. There is no functional difference between the two, so use whichever approach you prefer 😀. You shouldn't be defining both, but if you do `Component`/`ErrorBoundary` will "win". ([#10045](https://github.com/remix-run/react-router/pull/10045))
788
789 **Example JSON Syntax**
790
791 ```jsx
792 // Both of these work the same:
793 const elementRoutes = [{
794 path: '/',
795 element: <Home />,
796 errorElement: <HomeError />,
797 }]
798
799 const componentRoutes = [{
800 path: '/',
801 Component: Home,
802 ErrorBoundary: HomeError,
803 }]
804
805 function Home() { ... }
806 function HomeError() { ... }
807 ```
808
809 **Example JSX Syntax**
810
811 ```jsx
812 // Both of these work the same:
813 const elementRoutes = createRoutesFromElements(
814 <Route path='/' element={<Home />} errorElement={<HomeError /> } />
815 );
816
817 const componentRoutes = createRoutesFromElements(
818 <Route path='/' Component={Home} ErrorBoundary={HomeError} />
819 );
820
821 function Home() { ... }
822 function HomeError() { ... }
823 ```
824
825- **Introducing Lazy Route Modules!** ([#10045](https://github.com/remix-run/react-router/pull/10045))
826
827 In order to keep your application bundles small and support code-splitting of your routes, we've introduced a new `lazy()` route property. This is an async function that resolves the non-route-matching portions of your route definition (`loader`, `action`, `element`/`Component`, `errorElement`/`ErrorBoundary`, `shouldRevalidate`, `handle`).
828
829 Lazy routes are resolved on initial load and during the `loading` or `submitting` phase of a navigation or fetcher call. You cannot lazily define route-matching properties (`path`, `index`, `children`) since we only execute your lazy route functions after we've matched known routes.
830
831 Your `lazy` functions will typically return the result of a dynamic import.
832
833 ```jsx
834 // In this example, we assume most folks land on the homepage so we include that
835 // in our critical-path bundle, but then we lazily load modules for /a and /b so
836 // they don't load until the user navigates to those routes
837 let routes = createRoutesFromElements(
838 <Route path="/" element={<Layout />}>
839 <Route index element={<Home />} />
840 <Route path="a" lazy={() => import("./a")} />
841 <Route path="b" lazy={() => import("./b")} />
842 </Route>
843 );
844 ```
845
846 Then in your lazy route modules, export the properties you want defined for the route:
847
848 ```jsx
849 export async function loader({ request }) {
850 let data = await fetchData(request);
851 return json(data);
852 }
853
854 // Export a `Component` directly instead of needing to create a React Element from it
855 export function Component() {
856 let data = useLoaderData();
857
858 return (
859 <>
860 <h1>You made it!</h1>
861 <p>{data}</p>
862 </>
863 );
864 }
865
866 // Export an `ErrorBoundary` directly instead of needing to create a React Element from it
867 export function ErrorBoundary() {
868 let error = useRouteError();
869 return isRouteErrorResponse(error) ? (
870 <h1>
871 {error.status} {error.statusText}
872 </h1>
873 ) : (
874 <h1>{error.message || error}</h1>
875 );
876 }
877 ```
878
879 An example of this in action can be found in the [`examples/lazy-loading-router-provider`](https://github.com/remix-run/react-router/tree/main/examples/lazy-loading-router-provider) directory of the repository.
880
881 🙌 Huge thanks to @rossipedia for the [Initial Proposal](https://github.com/remix-run/react-router/discussions/9826) and [POC Implementation](https://github.com/remix-run/react-router/pull/9830).
882
883- Updated dependencies:
884 - `@remix-run/router@1.4.0`
885
886### Patch Changes
887
888- Fix `generatePath` incorrectly applying parameters in some cases ([#10078](https://github.com/remix-run/react-router/pull/10078))
889- Improve memoization for context providers to avoid unnecessary re-renders ([#9983](https://github.com/remix-run/react-router/pull/9983))
890
891## 6.8.2
892
893### Patch Changes
894
895- Updated dependencies:
896 - `@remix-run/router@1.3.3`
897
898## 6.8.1
899
900### Patch Changes
901
902- Remove inaccurate console warning for POP navigations and update active blocker logic ([#10030](https://github.com/remix-run/react-router/pull/10030))
903- Updated dependencies:
904 - `@remix-run/router@1.3.2`
905
906## 6.8.0
907
908### Patch Changes
909
910- Updated dependencies:
911 - `@remix-run/router@1.3.1`
912
913## 6.7.0
914
915### Minor Changes
916
917- Add `unstable_useBlocker` hook for blocking navigations within the app's location origin ([#9709](https://github.com/remix-run/react-router/pull/9709))
918
919### Patch Changes
920
921- Fix `generatePath` when optional params are present ([#9764](https://github.com/remix-run/react-router/pull/9764))
922- Update `<Await>` to accept `ReactNode` as children function return result ([#9896](https://github.com/remix-run/react-router/pull/9896))
923- Updated dependencies:
924 - `@remix-run/router@1.3.0`
925
926## 6.6.2
927
928### Patch Changes
929
930- Ensure `useId` consistency during SSR ([#9805](https://github.com/remix-run/react-router/pull/9805))
931
932## 6.6.1
933
934### Patch Changes
935
936- Updated dependencies:
937 - `@remix-run/router@1.2.1`
938
939## 6.6.0
940
941### Patch Changes
942
943- Prevent `useLoaderData` usage in `errorElement` ([#9735](https://github.com/remix-run/react-router/pull/9735))
944- Updated dependencies:
945 - `@remix-run/router@1.2.0`
946
947## 6.5.0
948
949This release introduces support for [Optional Route Segments](https://github.com/remix-run/react-router/issues/9546). Now, adding a `?` to the end of any path segment will make that entire segment optional. This works for both static segments and dynamic parameters.
950
951**Optional Params Examples**
952
953- `<Route path=":lang?/about>` will match:
954 - `/:lang/about`
955 - `/about`
956- `<Route path="/multistep/:widget1?/widget2?/widget3?">` will match:
957 - `/multistep`
958 - `/multistep/:widget1`
959 - `/multistep/:widget1/:widget2`
960 - `/multistep/:widget1/:widget2/:widget3`
961
962**Optional Static Segment Example**
963
964- `<Route path="/home?">` will match:
965 - `/`
966 - `/home`
967- `<Route path="/fr?/about">` will match:
968 - `/about`
969 - `/fr/about`
970
971### Minor Changes
972
973- Allows optional routes and optional static segments ([#9650](https://github.com/remix-run/react-router/pull/9650))
974
975### Patch Changes
976
977- Stop incorrectly matching on partial named parameters, i.e. `<Route path="prefix-:param">`, to align with how splat parameters work. If you were previously relying on this behavior then it's recommended to extract the static portion of the path at the `useParams` call site: ([#9506](https://github.com/remix-run/react-router/pull/9506))
978
979```jsx
980// Old behavior at URL /prefix-123
981<Route path="prefix-:id" element={<Comp /> }>
982
983function Comp() {
984 let params = useParams(); // { id: '123' }
985 let id = params.id; // "123"
986 ...
987}
988
989// New behavior at URL /prefix-123
990<Route path=":id" element={<Comp /> }>
991
992function Comp() {
993 let params = useParams(); // { id: 'prefix-123' }
994 let id = params.id.replace(/^prefix-/, ''); // "123"
995 ...
996}
997```
998
999- Updated dependencies:
1000 - `@remix-run/router@1.1.0`
1001
1002## 6.4.5
1003
1004### Patch Changes
1005
1006- Updated dependencies:
1007 - `@remix-run/router@1.0.5`
1008
1009## 6.4.4
1010
1011### Patch Changes
1012
1013- Updated dependencies:
1014 - `@remix-run/router@1.0.4`
1015
1016## 6.4.3
1017
1018### Patch Changes
1019
1020- `useRoutes` should be able to return `null` when passing `locationArg` ([#9485](https://github.com/remix-run/react-router/pull/9485))
1021- fix `initialEntries` type in `createMemoryRouter` ([#9498](https://github.com/remix-run/react-router/pull/9498))
1022- Updated dependencies:
1023 - `@remix-run/router@1.0.3`
1024
1025## 6.4.2
1026
1027### Patch Changes
1028
1029- Fix `IndexRouteObject` and `NonIndexRouteObject` types to make `hasErrorElement` optional ([#9394](https://github.com/remix-run/react-router/pull/9394))
1030- Enhance console error messages for invalid usage of data router hooks ([#9311](https://github.com/remix-run/react-router/pull/9311))
1031- If an index route has children, it will result in a runtime error. We have strengthened our `RouteObject`/`RouteProps` types to surface the error in TypeScript. ([#9366](https://github.com/remix-run/react-router/pull/9366))
1032- Updated dependencies:
1033 - `@remix-run/router@1.0.2`
1034
1035## 6.4.1
1036
1037### Patch Changes
1038
1039- Preserve state from `initialEntries` ([#9288](https://github.com/remix-run/react-router/pull/9288))
1040- Updated dependencies:
1041 - `@remix-run/router@1.0.1`
1042
1043## 6.4.0
1044
1045Whoa this is a big one! `6.4.0` brings all the data loading and mutation APIs over from Remix. Here's a quick high level overview, but it's recommended you go check out the [docs](https://reactrouter.com), especially the [feature overview](https://reactrouter.com/en/6.4.0/start/overview) and the [tutorial](https://reactrouter.com/en/6.4.0/start/tutorial).
1046
1047**New APIs**
1048
1049- Create your router with `createMemoryRouter`
1050- Render your router with `<RouterProvider>`
1051- Load data with a Route `loader` and mutate with a Route `action`
1052- Handle errors with Route `errorElement`
1053- Defer non-critical data with `defer` and `Await`
1054
1055**Bug Fixes**
1056
1057- Path resolution is now trailing slash agnostic (#8861)
1058- `useLocation` returns the scoped location inside a `<Routes location>` component (#9094)
1059
1060**Updated Dependencies**
1061
1062- `@remix-run/router@1.0.0`