- Throw unwrapped single fetch redirect to align with pre-single fetch behavior (#10317)
- Updated dependencies:
@remix-run/server-runtime@2.15.2
- Updated dependencies:
@remix-run/server-runtime@2.15.1
- Updated dependencies:
@remix-run/server-runtime@2.15.0
- Fix
defaultShouldRevalidate
value when using single fetch (#10139) - Updated dependencies:
@remix-run/server-runtime@2.14.0
- Updated dependencies:
@remix-run/server-runtime@2.13.1
- Stabilize React Router APIs in Remix (#9980)
- Adopt stabilized React Router APIs internally
- Single Fetch:
unstable_dataStrategy
->dataStrategy
- Lazy Route Discovery:
unstable_patchRoutesOnNavigation
->patchRoutesOnNavigation
- Single Fetch:
- Stabilize public-facing APIs
- Single Fetch:
unstable_data()
->data()
unstable_viewTransition
->viewTransition
(Link
,Form
,navigate
,submit
)unstable_flushSync>
-><Link viewTransition>
(Link
,Form
,navigate
,submit
,useFetcher
)
- Single Fetch:
- Adopt stabilized React Router APIs internally
- Stabilize future flags (#10072)
future.unstable_singleFetch
->future.v3_singleFetch
future.unstable_lazyRouteDiscovery
->future.v3_lazyRouteDiscovery
- Fix bug with
clientLoader.hydrate
in a layout route when hydrating with bubbled errors (#10063) - Updated dependencies:
@remix-run/server-runtime@2.13.0
- Updated dependencies:
@remix-run/server-runtime@2.12.1
-
Lazy Route Discovery: Sort
/__manifest
query parameters for better caching (#9888) -
Single Fetch: fix revalidation behavior bugs (#9938)
- With Single Fetch, existing routes revalidate by default
- This means requests do not need special query params for granular route revalidations out of the box - i.e.,
GET /a/b/c.data
- There are two conditions that will trigger granular revalidation:
- If a route opts out of revalidation via
shouldRevalidate
, it will be excluded from the single fetch call - If a route defines a
clientLoader
then it will be excluded from the single fetch call and if you callserverLoader()
from yourclientLoader
, that will make a separarte HTTP call for just that route loader - i.e.,GET /a/b/c.data?_routes=routes/a
for aclientLoader
inroutes/a.tsx
- If a route opts out of revalidation via
- When one or more routes are excluded from the single fetch call, the remaining routes that have loaders are included as query params:
- For example, if A was excluded, and the
root
route androutes/b
had aloader
butroutes/c
did not, the single fetch request would beGET /a/b/c.data?_routes=root,routes/a
- For example, if A was excluded, and the
-
Remove hydration URL check that was originally added for React 17 hydration issues and we no longer support React 17 (#9890)
- Reverts the logic originally added in Remix
v1.18.0
via #6409 - This was added to resolve an issue that could arise when doing quick back/forward history navigations while JS was loading which would cause a mismatch between the server matches and client matches: #1757
- This specific hydration issue would then cause this React v17 only looping issue: #1678
- The URL comparison that we added in
1.18.0
turned out to be subject to false positives of it's own which could also put the user in looping scenarios - Remix v2 upgraded it's minimal React version to v18 which eliminated the v17 hydration error loop
- React v18 handles this hydration error like any other error and does not result in a loop
- So we can remove our check and thus avoid the false-positive scenarios in which it may also trigger a loop
- Reverts the logic originally added in Remix
-
Single Fetch: Improved typesafety (#9893)
If you were already using previously released unstable single-fetch types:
- Remove
"@remix-run/react/future/single-fetch.d.ts"
override fromtsconfig.json
>compilerOptions
>types
- Remove
defineLoader
,defineAction
,defineClientLoader
,defineClientAction
helpers from your route modules - Replace
UIMatch_SingleFetch
type helper withUIMatch
- Replace
MetaArgs_SingleFetch
type helper withMetaArgs
Then you are ready for the new typesafety setup:
// vite.config.ts declare module "@remix-run/server-runtime" { interface Future { unstable_singleFetch: true; // 👈 enable _types_ for single-fetch } } export default defineConfig({ plugins: [ remix({ future: { unstable_singleFetch: true, // 👈 enable single-fetch }, }), ], });
For more information, see Guides > Single Fetch in our docs.
- Remove
-
Clarify wording in default
HydrateFallback
console warning (#9899) -
Updated dependencies:
@remix-run/server-runtime@2.12.0
- Fog of War: Simplify implementation now that React Router handles slug/splat edge cases and tracks previously discovered routes (see remix-run/react-router#11883) (#9860)
- This changes the return signature of the internal
__manifest
endpoint since we no longer need thenotFoundPaths
field
- This changes the return signature of the internal
- Fog of War: Update to use renamed
unstable_patchRoutesOnNavigation
function in RR (see remix-run/react-router#11888) (#9860) - Single Fetch: Update
turbo-stream
tov2.3.0
(#9856)- Stabilize object key order for serialized payloads
- Remove memory limitations payloads sizes
- Updated dependencies:
@remix-run/server-runtime@2.11.2
- Revert #9695, stop infinite reload (
a7cffe57
) - Updated dependencies:
@remix-run/server-runtime@2.11.1
-
Single Fetch: Add a new
unstable_data()
API as a replacement forjson
/defer
when customstatus
/headers
are needed (#9769) -
Add a new
replace(url, init?)
alternative toredirect(url, init?)
that performs ahistory.replaceState
instead of ahistory.pushState
on client-side navigation redirects (#9764) -
Rename
future.unstable_fogOfWar
tofuture.unstable_lazyRouteDiscovery
for clarity (#9763) -
Single Fetch: Remove
responseStub
in favor ofheaders
(#9769)-
Background
- The original Single Fetch approach was based on an assumption that an eventual
middleware
implementation would require something likeResponseStub
so users could mutatestatus
/headers
inmiddleware
before/after handlers as well as during handlers - We wanted to align how
headers
got merged between document and data requests - So we made document requests also use
ResponseStub
and removed the usage ofheaders
in Single Fetch - The realization/alignment between Michael and Ryan on the recent roadmap planning made us realize that the original assumption was incorrect
middleware
won't need a stub - users can just mutate theResponse
they get fromawait next()
directly- With that gone, and still wanting to align how
headers
get merged, it makes more sense to stick with the currentheaders
API and apply that to Single Fetch and avoid introducing a totally new thing inRepsonseStub
(that always felt a bit awkward to work with anyway)
- The original Single Fetch approach was based on an assumption that an eventual
-
With this change:
- You are encouraged to stop returning
Response
instances in favor of returning raw data from loaders and actions:return json({ data: whatever });
return { data: whatever };
- In most cases, you can remove your
json()
anddefer()
calls in favor of returning raw data if they weren't setting customstatus
/headers
- We will be removing both
json
anddefer
in the next major version, but both should still work in Single Fetch in v2 to allow for incremental adoption of the new behavior
- We will be removing both
- If you need custom
status
/headers
:- We've added a new
unstable_data({...}, responseInit)
utility that will let you send backstatus
/headers
alongside your raw data without having to encode it into aResponse
- We've added a new
- The
headers()
function will let you control header merging for both document and data requests
- You are encouraged to stop returning
-
- Single Fetch: Ensure calls don't include any trailing slash from the pathname (i.e.,
/path/.data
) (#9792) - Single Fetch: Add
undefined
to theuseRouteLoaderData
type override (#9796) - Change initial hydration route mismatch from a URL check to a matches check to be resistant to URL inconsistencies (#9695)
- Updated dependencies:
@remix-run/server-runtime@2.11.0
- Log any errors encountered loading a route module prior to reloading the page (#8932)
- Single Fetch (unstable): Proxy
request.signal
throughdataStrategy
forloader
calls to fix cancellation (#9738) - Single Fetch (unstable): Adopt React Router's stabilized
future.v7_skipActionErrorRevalidation
under the hood (#9706)- This also stabilizes the
shouldRevalidate
parameter fromunstable_actionStatus
toactionStatus
- This also stabilizes the
- Updated dependencies:
@remix-run/server-runtime@2.10.3
- Forward
ref
toForm
(bdd04217
) - Updated dependencies:
@remix-run/server-runtime@2.10.2
- Fog of War (unstable): Don't discover links/forms with
reloadDocument
(#9686) - Fog of War (unstable): Support route discovery from
<Form>
components (#9665) - Updated dependencies:
@remix-run/server-runtime@2.10.1
-
Add support for Lazy Route Discovery (a.k.a. Fog of War) (#9600)
- Don't prefetch server
loader
data whenclientLoader
exists (#9580) - Avoid hydration loops when
Layout
ErrorBoundary
renders also throw (#9566) - Fix a bug where hydration wouldn't work right when using child routes and hydrate fallbacks with a
basename
(#9584) - Update to
turbo-stream@2.2.0
for single fetch (#9562) - Updated dependencies:
@remix-run/server-runtime@2.10.0
- Add
undefined
touseActionData
type override (#9322) - Allow a
nonce
to be set on single fetch stream transfer inline scripts (#9364) - Typesafety for single-fetch:
defineLoader
,defineClientLoader
,defineAction
,defineClientAction
(#9372, #9404) - Updated dependencies:
@remix-run/server-runtime@2.9.2
- Ignore
future/*.d.ts
files from TS build (#9299) - Updated dependencies:
@remix-run/server-runtime@2.9.1
-
New
future.unstable_singleFetch
flag (#8773)- Naked objects returned from loaders/actions are no longer automatically converted to JSON responses. They'll be streamed as-is via
turbo-stream
soDate
's will becomeDate
throughuseLoaderData()
- You can return naked objects with
Promise
's without needing to usedefer()
- including nestedPromise
's- If you need to return a custom status code or custom response headers, you can still use the
defer
utility
- If you need to return a custom status code or custom response headers, you can still use the
<RemixServer abortDelay>
is no longer used. Instead, you shouldexport const streamTimeout
fromentry.server.tsx
and the remix server runtime will use that as the delay to abort the streamed response- If you export your own streamTimeout, you should decouple that from aborting the react
renderToPipeableStream
. You should always ensure that react is aborted afer the stream is aborted so that abort rejections can be flushed down
- If you export your own streamTimeout, you should decouple that from aborting the react
- Actions no longer automatically revalidate on 4xx/5xx responses (via RR
future.unstable_skipActionErrorRevalidation
flag) - you can return a 2xx to opt-into revalidation or useshouldRevalidate
- Naked objects returned from loaders/actions are no longer automatically converted to JSON responses. They'll be streamed as-is via
-
Opt-in types for single-fetch (#9272)
- To opt-in to type inference for single-fetch, add
./node_modules/@remix-run/react/future/single-fetch.d.ts
toinclude
in yourtsconfig.json
- To opt-in to type inference for single-fetch, add
- Updated dependencies:
@remix-run/server-runtime@2.9.0
- Strengthen the internal
LayoutComponent
type to accept limited children (#8910) - Updated dependencies:
@remix-run/server-runtime@2.8.1
- Fix the default root
ErrorBoundary
component so it leverages the user-providedLayout
component (#8859) - Fix the default root
HydrateFallback
component so it leverages any user-providedLayout
component (#8892) - Ensure
@remix-run/react
re-exports everything fromreact-router-dom
for SPA mode (#8929) - Updated dependencies:
@remix-run/server-runtime@2.8.0
- Updated dependencies:
@remix-run/server-runtime@2.7.2
- Updated dependencies:
@remix-run/server-runtime@2.7.1
- Allow an optional
Layout
export from the root route (#8709) - Vite: Add a new
basename
option to the Vite plugin, allowing users to set the internal React Routerbasename
in order to to serve their applications underneath a subpath (#8145)
- Fix a bug with SPA mode when the root route had no children (#8747)
- Updated dependencies:
@remix-run/server-runtime@2.7.0
- Updated dependencies:
@remix-run/server-runtime@2.6.0
- Only use active matches in
<Meta>
/<Links>
in SPA mode (#8538) - Remove leftover
unstable_
prefix fromBlocker
/BlockerFunction
types (#8530) - Updated dependencies:
@remix-run/server-runtime@2.5.1
-
Add unstable support for "SPA Mode" (#8457)
You can opt into SPA Mode by setting
unstable_ssr: false
in your Remix Vite plugin config:// vite.config.ts import { unstable_vitePlugin as remix } from "@remix-run/dev"; import { defineConfig } from "vite"; export default defineConfig({ plugins: [remix({ unstable_ssr: false })], });
Development in SPA Mode is just like a normal Remix app, and still uses the Remix dev server for HMR/HDR:
remix vite:dev
Building in SPA Mode will generate an
index.html
file in your client assets directory:remix vite:build
To run your SPA, you serve your client assets directory via an HTTP server:
npx http-server build/client
For more information, please refer to the SPA Mode docs.
- Vite: Fix type conflict with
import.meta.hot
from the existing Remix compiler (#8459) - Updated dependencies:
@remix-run/server-runtime@2.5.0
- Propagate server
loader
errors throughserverLoader
in hydratingclientLoader
's (#8304) - Re-export
Response
helpers (defer
/json
/redirect
/redirectDocument
) through@remix-run/react
for use inclientLoader
/clientAction
(#8351) - Updated dependencies:
@remix-run/server-runtime@2.4.1
-
Add support for
clientLoader
/clientAction
/HydrateFallback
route exports (RFC). (#8173)Remix now supports loaders/actions that run on the client (in addition to, or instead of the loader/action that runs on the server). While we still recommend server loaders/actions for the majority of your data needs in a Remix app - these provide some levers you can pull for more advanced use-cases such as:
- Leveraging a data source local to the browser (i.e.,
localStorage
) - Managing a client-side cache of server data (like
IndexedDB
) - Bypassing the Remix server in a BFF setup and hitting your API directly from the browser
- Migrating a React Router SPA to a Remix application
By default,
clientLoader
will not run on hydration, and will only run on subsequent client side navigations.If you wish to run your client loader on hydration, you can set
clientLoader.hydrate=true
to force Remix to execute it on initial page load. Keep in mind that Remix will still SSR your route component so you should ensure that there is no new required data being added by yourclientLoader
.If your
clientLoader
needs to run on hydration and adds data you require to render the route component, you can export aHydrateFallback
component that will render during SSR, and then your route component will not render until theclientLoader
has executed on hydration.clientAction
is simpler thanclientLoader
because it has no hydration use-cases.clientAction
will only run on client-side navigations.For more information, please refer to the
clientLoader
andclientAction
documentation. - Leveraging a data source local to the browser (i.e.,
-
Add a new
future.v3_relativeSplatPath
flag to implement a breaking bug fix to relative routing when inside a splat route. For more information, please see the React Router6.21.0
Release Notes and theuseResolvedPath
docs. (#8216)
- Updated dependencies:
@remix-run/server-runtime@2.4.0
- Updated dependencies:
@remix-run/server-runtime@2.3.1
- Remove the
unstable_
prefix from theuseBlocker
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 fromunstable_usePrompt
due to differences in how browsers handlewindow.confirm
that prevent React Router from guaranteeing consistent/correct behavior. (#7882)
- Support rendering of
LiveReload
component afterScripts
in Vite dev (#7919) - Support optional rendering of
LiveReload
component in Vite dev (#7919) - add missing modulepreload for the manifest (#7684)
- Updated dependencies:
react-router-dom@6.19.0
@remix-run/router@1.12.0
@remix-run/server-runtime@2.3.0
- Unstable Vite support for Node-based Remix apps (#7590)
remix build
👉vite build && vite build --ssr
remix dev
👉vite dev
- Other runtimes (e.g. Deno, Cloudflare) not yet supported.
- See "Future > Vite" in the Remix Docs for details
- Add a new
future.v3_fetcherPersist
flag to change the persistence behavior of fetchers. Instead of being immediately cleaned up when unmounted in the UI, fetchers will persist until they return to anidle
state (RFC) (#7704)- For more details, please refer to the React Router 6.18.0 release notes
- Fix warning that could be logged when using route files with no
default
export (#7745)- It seems our compiler compiles these files to export an empty object as the
default
which we can then end up passing toReact.createElement
, triggering the console warning, but generally no UI issues - By properly detecting these, we can correctly pass
Component: undefined
off to the React Router layer - This is technically an potential issue in the compiler but it's an easy patch in the
@remix-run/react
layer and hopefully disappears in a Vite world
- It seems our compiler compiles these files to export an empty object as the
- Fix critical CSS hydration errors for Vite dev (#7812)
- Updated dependencies:
@remix-run/server-runtime@2.2.0
react-router-dom@6.18.0
@remix-run/router@1.11.0
- Add experimental support for the View Transitions API via
document.startViewTransition
to enable CSS animated transitions on SPA navigations in your application (#7648)- For additional information please refer to the 2.1.0 Release Notes or the documentation
- Avoid mutating
meta
object whentagName
is specified (#7594) - Fix FOUC on subsequent client-side navigations to
route.lazy
routes (#7576) - Emulate types for
JSON.parse(JSON.stringify(x))
inSerializeFrom
(#7605)- Notably, type fields that are only assignable to
undefined
after serialization are now omitted sinceJSON.stringify |> JSON.parse
will omit them -- see test cases for examples - Also fixes type errors when upgrading to v2 from 1.19
- Notably, type fields that are only assignable to
- Export the proper Remix
useMatches
wrapper to fixUIMatch
typings (#7551) - Updated dependencies:
@remix-run/server-runtime@2.1.0
react-router-dom@6.17.0
@remix-run/router@1.10.0
- Add second generic to
UIMatch
forhandle
field (#7464) - Fix resource routes being loaded through
route.lazy
(#7498) - Throw a semantically correct 405
ErrorResponse
instead of just anError
when submitting to a route without anaction
(#7423) - Updated dependencies:
@remix-run/server-runtime@2.0.1
-
Drop React 17 support (#7121)
-
Require Node >=18.0.0 (#6939)
-
Remove
unstable_shouldReload
, which has been replaced byshouldRevalidate
(#6865) -
The route
meta
API now defaults to the new "V2 Meta" API (#6958)- Please refer to the (docs and Preparing for V2 guide for more information.
-
Promote the
future.v2_dev
flag inremix.config.js
to a root leveldev
config (#7002) -
Remove
v2_errorBoundary
flag andCatchBoundary
implementation (#6906) -
Remove back-compat layer for
useFetcher
/useFetchers
, which includes a few small breaking changes (#6874)fetcher.type
has been removed since it can be derived from other available information- "Submission" fields have been flattened from
fetcher.submission
down onto the rootfetcher
object, and prefixed withform
in some cases (fetcher.submission.action
=>fetcher.formAction
) <fetcher.Form method="get">
is now more accurately categorized asstate:"loading"
instead ofstate:"submitting"
to better align with the underlying GET request
-
Remove
v2_normalizeFormMethod
future flag - allformMethod
values will be normalized in v2 (#6875) -
Remove deprecated
useTransition
hook in favor ofuseNavigation
-useNavigation
is almost identical with a few exceptions: (#6870)useTransition.type
has been removed since it can be derived from other available information- "Submission" fields have been flattened from
useTransition().submission
down onto the rootuseNavigation()
object <Form method="get">
is now more accurately categorized asstate:"loading"
instead ofstate:"submitting"
to better align with the underlying GET navigation
-
Remove
v2_routeConvention
flag - the flat route file convention is now standard. (#6969) -
Remove
v2_headers
flag - it is now the default behavior to use the deepestheaders
function in the route tree. (#6979) -
Removed/adjusted types to prefer
unknown
overany
and to align with underlying React Router types (#7319, #7354):- Renamed the
useMatches()
return type fromRouteMatch
toUIMatch
- Renamed
LoaderArgs
/ActionArgs
toLoaderFunctionArgs
/ActionFunctionArgs
AppData
changed fromany
tounknown
Location["state"]
(useLocation.state
) changed fromany
tounknown
UIMatch["data"]
(useMatches()[i].data
) changed fromany
tounknown
UIMatch["handle"]
(useMatches()[i].handle
) changed from{ [k: string]: any }
tounknown
Fetcher["data"]
(useFetcher().data
) changed fromany
tounknown
MetaMatch.handle
(used inmeta()
) changed fromany
tounknown
AppData
/RouteHandle
are no longer exported as they are just aliases forunknown
- Renamed the
-
Remove
imagesizes
&imagesrcset
properties fromHtmlLinkDescriptor
,LinkDescriptor
&PrefetchPageDescriptor
types (#6936) -
Remove deprecated
REMIX_DEV_SERVER_WS_PORT
env var (#6965)- use
remix dev
's '--port
/port
option instead
- use
-
Removed support for "magic exports" from the
remix
package. This package can be removed from yourpackage.json
and you should update all imports to use the source@remix-run/*
packages: (#6895)- import type { ActionArgs } from "remix"; - import { json, useLoaderData } from "remix"; + import type { ActionArgs } from "@remix-run/node"; + import { json } from "@remix-run/node"; + import { useLoaderData } from "@remix-run/react";
- Export the
Navigation
type returned fromuseNavigation
(#7136) - Update Remix to use React Router
route.lazy
for module loading (#7133)
-
Add
error
tometa()
params so you can render error titles, etc. (#7105)export function meta({ error }) { return [{ title: error.message }]; }
-
Re-Export
ShouldRevalidateFunctionArgs
type from React Router (#7316) -
Deduplicate prefetch
link
tags (#7060) -
Skip preloading of stylesheets on client-side route transitions if the browser does not support
<link rel=preload>
(#7106)- This prevents us from hanging on client-side navigations when we try to preload stylesheets and never receive a
load
/error
event on thelink
tag
- This prevents us from hanging on client-side navigations when we try to preload stylesheets and never receive a
-
Export proper
ErrorResponse
type for usage alongsideisRouteErrorResponse
(#7244) -
Use the hostname from
REMIX_DEV_ORIGIN
to connect to the live reload socket (#6923) -
Use unique key for
script:ld+json
meta descriptors (#6954) -
Fix live reload port when set explicitly as a prop (#7358)
-
Fix types for
useLoaderData
when using Yarn PnP (#7137) -
Updated dependencies:
@remix-run/server-runtime@2.0.0
react-router-dom@6.16.0
@remix-run/router@1.9.0
No significant changes to this package were made in this release. See the releases page on GitHub for an overview of all changes in v1.19.3.
No significant changes to this package were made in this release. See the releases page on GitHub for an overview of all changes in v1.19.2.
No significant changes to this package were made in this release. See the releases page on GitHub for an overview of all changes in v1.19.1.
-
improved networking options for
v2_dev
(#6724)deprecate the
--scheme
and--host
options and replace them with theREMIX_DEV_ORIGIN
environment variable -
Added some missing react-router exports to
@remix-run/react
(#6856)
- Narrowed the type of
fetcher.formEncType
to useFormEncType
fromreact-router-dom
instead ofstring
(#6810) - Deferred promises that return undefined/void now surface a serialization error. (#6793)
- Properly handle
?_data
HTTP/Network errors that don't reach the Remix server and ensure they bubble to theErrorBoundary
(#6783) - Support proper hydration of
Error
subclasses such asReferenceError
/TypeError
in development mode (#6675) - fix router race condition for hmr (#6767)
- Avoid re-prefetching stylesheets for active routes during a revalidation (#6679)
- Add generic type for
useRouteLoaderData()
(#5157) - Bump RR 6.14.2 (#6854)
- Updated dependencies:
- Fix reload loops in scenarios where CDNs ignore search params (#6707)
- Updated dependencies:
- stabilize v2 dev server (#6615)
- Support
application/json
andtext/plain
submission encodings inuseSubmit
/fetcher.submit
(#6570) - Add support for
<Link prefetch="viewport">
to prefetch links when they enter the viewport via an Intersection Observer (#6433)
- Bump router 6.14.0-pre.1 (#6662)
- Detect mismatches between the initially loaded URL and the URL at the time we hydrate and trigger a hard reload if they do not match. This is an edge-case that can happen when the network is slowish and the user clicks forward into a Remix app and then clicks forward again while the initial JS chunks are loading. (#6409)
- Lock in react router 6.14.0 (#6677)
- properly pass props to inline script tags for deferred data (#6389)
- Updated dependencies:
-
Faster server export removal for routes when
unstable_dev
is enabled. (#6455)Also, only render modulepreloads on SSR. Do not render modulepreloads when hydrated.
-
Force Typescript to simplify type produced by
Serialize
. (#6449)As a result, the following types and functions have simplified return types:
- SerializeFrom
- useLoaderData
- useActionData
- useFetcher
type Data = { hello: string; when: Date }; // BEFORE type Unsimplified = SerializeFrom<Data>; // ^? SerializeObject<UndefinedToOptional<{ hello: string; when: Date }>> // AFTER type Simplified = SerializeFrom<Data>; // ^? { hello: string; when: string }
-
Reuse dev server port for WebSocket (Live Reload,HMR,HDR) (#6476)
As a result the
webSocketPort
/--websocket-port
option has been obsoleted. Additionally, scheme/host/port options for the dev server have been renamed.Available options are:
Option flag config default Command -c
/--command
command
remix-serve <server build path>
Scheme --scheme
scheme
http
Host --host
host
localhost
Port --port
port
Dynamically chosen open port No restart --no-restart
restart: false
restart: true
Note that scheme/host/port options are for the dev server, not your app server. You probably don't need to use scheme/host/port option if you aren't configuring networking (e.g. for Docker or SSL).
- retry HDR revalidations in development mode to aid in 3rd party server race conditions (#6287)
- Updated dependencies:
- Cross-module
loader
change detection for HDR (#6299) - Better opt-out of
loader
revalidation on UI only changes (#6278) - Add
useMatch
re-export fromreact-router-dom
(#5257) - Fix
data
parameter typing onV2_MetaFunction
to includeundefined
for scenarios in which theloader
threw to it's own boundary. (#6231) - Updated dependencies:
-
Enable support for CSS Modules, Vanilla Extract and CSS side-effect imports (#6046)
These CSS bundling features were previously only available via
future.unstable_cssModules
,future.unstable_vanillaExtract
andfuture.unstable_cssSideEffectImports
options inremix.config.js
, but they have now been stabilized.In order to use these features, check out our guide to CSS bundling in your project.
-
Stabilize built-in PostCSS support via the new
postcss
option inremix.config.js
. As a result, thefuture.unstable_postcss
option has also been deprecated. (#5960)The
postcss
option isfalse
by default, but when set totrue
will enable processing of all CSS files using PostCSS ifpostcss.config.js
is present.If you followed the original PostCSS setup guide for Remix, you may have a folder structure that looks like this, separating your source files from its processed output:
. ├── app │ └── styles (processed files) │ ├── app.css │ └── routes │ └── index.css └── styles (source files) ├── app.css └── routes └── index.css
After you've enabled the new
postcss
option, you can delete the processed files fromapp/styles
folder and move your source files fromstyles
toapp/styles
:. ├── app │ └── styles (source files) │ ├── app.css │ └── routes │ └── index.css
You should then remove
app/styles
from your.gitignore
file since it now contains source files rather than processed output.You can then update your
package.json
scripts to remove any usage ofpostcss
since Remix handles this automatically. For example, if you had followed the original setup guide:{ "scripts": { - "dev:css": "postcss styles --base styles --dir app/styles -w", - "build:css": "postcss styles --base styles --dir app/styles --env production", - "dev": "concurrently \"npm run dev:css\" \"remix dev\"" + "dev": "remix dev" } }
-
Stabilize built-in Tailwind support via the new
tailwind
option inremix.config.js
. As a result, thefuture.unstable_tailwind
option has also been deprecated. (#5960)The
tailwind
option isfalse
by default, but when set totrue
will enable built-in support for Tailwind functions and directives in your CSS files iftailwindcss
is installed.If you followed the original Tailwind setup guide for Remix and want to make use of this feature, you should first delete the generated
app/tailwind.css
.Then, if you have a
styles/tailwind.css
file, you should move it toapp/tailwind.css
.rm app/tailwind.css mv styles/tailwind.css app/tailwind.css
Otherwise, if you don't already have an
app/tailwind.css
file, you should create one with the following contents:@tailwind base; @tailwind components; @tailwind utilities;
You should then remove
/app/tailwind.css
from your.gitignore
file since it now contains source code rather than processed output.You can then update your
package.json
scripts to remove any usage oftailwindcss
since Remix handles this automatically. For example, if you had followed the original setup guide:{ // ... "scripts": { - "build": "run-s \"build:*\"", + "build": "remix build", - "build:css": "npm run generate:css -- --minify", - "build:remix": "remix build", - "dev": "run-p \"dev:*\"", + "dev": "remix dev", - "dev:css": "npm run generate:css -- --watch", - "dev:remix": "remix dev", - "generate:css": "npx tailwindcss -o ./app/tailwind.css", "start": "remix-serve build" } // ... }
- fix(react,dev): dev chunking and refresh race condition (#6201)
- Revalidate loaders only when a change to one is detected. (#6135)
- short circuit links and meta for routes that are not rendered due to errors (#6107)
- don't warn about runtime deprecation warnings in production (#4421)
- Update Remix for React Router no longer relying on
useSyncExternalStore
(#6121) - Fix false-positive resource route identification if a route only exports a boundary (#6125)
- better type discrimination when unwrapping loader return types (#5516)
- Updated dependencies:
-
Deprecated
fetcher.type
andfetcher.submission
for Remix v2 (#5691) -
We have made a few changes to the API for route module
meta
functions when using thefuture.v2_meta
flag. These changes are only breaking for users who have opted in. (#5746)V2_HtmlMetaDescriptor
has been renamed toV2_MetaDescriptor
- The
meta
function's arguments have been simplifiedparentsData
has been removed, as each route's loader data is available on thedata
property of its respectivematch
object// before export function meta({ parentsData }) { return [{ title: parentsData["routes/some-route"].title }]; } // after export function meta({ matches }) { return [ { title: matches.find((match) => match.id === "routes/some-route") .data.title, }, ]; }
- The
route
property on route matches has been removed, as relevant match data is attached directly to the match object// before export function meta({ matches }) { const rootModule = matches.find((match) => match.route.id === "root"); } // after export function meta({ matches }) { const rootModule = matches.find((match) => match.id === "root"); }
- Added support for generating
<script type='application/ld+json' />
and meta-related<link />
tags to document head via the routemeta
function when using thev2_meta
future flag
-
Added deprecation warning for
v2_normalizeFormMethod
(#5863) -
Added a new
future.v2_normalizeFormMethod
flag to normalize the exposeduseNavigation().formMethod
as an uppercase HTTP method to align with the previoususeTransition
behavior as well as thefetch()
behavior of normalizing to uppercase HTTP methods. (#5815)- When
future.v2_normalizeFormMethod === false
,useNavigation().formMethod
is lowercaseuseFetcher().formMethod
is uppercase
- When
future.v2_normalizeFormMethod === true
:useNavigation().formMethod
is uppercaseuseFetcher().formMethod
is uppercase
- When
-
Added deprecation warning for normalizing
imagesizes
&imagesrcset
properties returned from the routelinks
function. Both properties should be in camelCase (imageSizes
/imageSrcSet
) to align with their respective JavaScript properties. (#5706) -
Added deprecation warning for
CatchBoundary
in favor offuture.v2_errorBoundary
(#5718) -
Added experimental support for Vanilla Extract caching, which can be enabled by setting
future.unstable_vanillaExtract: { cache: true }
inremix.config
. This is considered experimental due to the use of a brand new Vanilla Extract compiler under the hood. In order to use this feature, you must be using at leastv1.10.0
of@vanilla-extract/css
. (#5735)
- Bumped React Router dependencies to the latest version. See the release notes for more details. (
e14699547
) - Added type deprecations for types now in React Router (#5679)
No significant changes to this package were made in this release. See the releases page on GitHub for an overview of all changes in v1.14.3.
No significant changes to this package were made in this release. See the releases page on GitHub for an overview of all changes in v1.14.2.
- Deprecate
useTransition
in favor ofuseNavigation
(#5687) - Memoize return value of
useMatches
(#5603)
- Hot Module Replacement and Hot Data Revalidation (#5259)
- Requires
unstable_dev
future flag to be enabled - HMR provided through React Refresh
- Features:
- HMR for component and style changes
- HDR when loaders for current route change
- Known limitations for MVP:
- Only implemented for React via React Refresh
- No
import.meta.hot
API exposed yet - Revalidates all loaders on route when loader changes are detected
- Loader changes do not account for imported dependencies changing
- Requires
- Remove duplicate manifest imports (#5534)
- Ensure types for fetchers always include
form*
submission fields (#5476) - Sync
FutureConfig
interface between packages (#5398) - Updated dependencies:
@remix-run/router@1.3.3
react-router-dom@8.6.2
- Add built-in support for PostCSS via the
future.unstable_postcss
feature flag (#5229) - Add built-in support for Tailwind via the
future.unstable_tailwind
feature flag (#5229)
- Bump React Router dependencies to the latest version. See the release notes for more details. (#5389)
- Improve efficiency of route manifest-to-tree transformation (#4748)
- Added better detection for absolute urls in
<Link>
and<NavLink>
components (#5390)
- Added a new development server available in the Remix config under the
unstable_dev
flag. See the release notes for a full description. (#5133) - You can now configure the client-side socket timeout via the new
timeoutMs
prop on<LiveReload />
(#4036)
<Link to>
can now accept absolute URLs. When theto
value is an absolute URL, the underlying anchor element will behave as normal, and its URL will not be prefetched. (#5092)- Bump React Router dependencies to the latest version. See the release notes for more details. (#5242)
- Added support for
unstable_useBlocker
andunstable_usePrompt
from React Router (#5151)
No significant changes to this package were made in this release. See the releases page on GitHub for an overview of all changes in v1.11.1.
- Added support for Vanilla Extract via the
unstable_vanillaExtract
future flag. IMPORTANT: Features marked withunstable
are … unstable. While we're confident in the use cases they solve, the API and implementation may change without a major version bump. (#5040) - Add support for CSS side-effect imports via the
unstable_cssSideEffectImports
future flag. IMPORTANT: Features marked withunstable
are … unstable. While we're confident in the use cases they solve, the API and implementation may change without a major version bump. (#4919) - Add support for CSS Modules via the
unstable_cssModules
future flag. IMPORTANT: Features marked withunstable
are … unstable. While we're confident in the use cases they solve, the API and implementation may change without a major version bump. (#4852)
-
Fix v2
meta
to ensure meta is rendered from the next route in the tree if nometa
export is included in a leaf route (#5041) -
Ensure
useFetcher
is stable across re-renders in backwards-compatibility layer (#5118) -
Added the
v2_errorBoundary
future flag to opt into the next version of Remix'sErrorBoundary
behavior. This removes the separateCatchBoundary
andErrorBoundary
and consolidates them into a singleErrorBoundary
, following the logic used byerrorElement
in React Router. You can then useisRouteErrorResponse
to differentiate between thrownResponse
/Error
instances. (#4918)// Current (Remix v1 default) import { useCatch } from "@remix-run/react"; export function CatchBoundary() { const caught = useCatch(); return ( <p> {caught.status} {caught.data} </p> ); } export function ErrorBoundary({ error }) { return <p>{error.message}</p>; }
// Using future.v2_errorBoundary import { isRouteErrorResponse, useRouteError } from "@remix-run/react"; export function ErrorBoundary() { const error = useRouteError(); return isRouteErrorResponse(error) ? ( <p> {error.status} {error.data} </p> ) : ( <p>{error.message}</p> ); }
-
Introduces the
defer()
API from@remix-run/router
with support for server-rendering and HTTP streaming. This utility allows you to defer values returned fromloader
functions by returning promises instead of resolved values. This has been refered to as "sending a promise over the wire". (#4920)Informational Resources:
- https://gist.github.com/jacob-ebey/9bde9546c1aafaa6bc8c242054b1be26
- https://github.com/remix-run/remix/blob/main/decisions/0004-streaming-apis.md
Documentation Resources (better docs specific to Remix are in the works):
- Fetchers should persist data through reload/resubmit (#5065)
- Update babel config to transpile down to node 14 (#5047)
- Update Remix to use new data APIs introduced in React Router v6.4 (#4900)
- Added new hooks from React Router
- Update
@remix-run/react
to useRouter
fromreact-router-dom@6.5.0
(#4731) - Allow pass-through props to be passed to the script rendered by
ScrollRestoration
(#2879) - Fixed a problem with
<LiveReload>
and Firefox infinitely reloading the page. (#4725)
No significant changes to this package were made in this release. See the releases page on GitHub for an overview of all changes in v1.8.2.
No significant changes to this package were made in this release. See the releases page on GitHub for an overview of all changes in v1.8.1.
- Importing functions and types from the
remix
package is deprecated, and all (#3284) exported modules will be removed in the next major release. For more details, see the release notes for 1.4.0 where these changes were first announced. - Added support for a new route
meta
API to handle arrays of tags instead of an object. For details, check out the RFC. (#4610)
- Ensure route modules are loaded even in failure cases. This addresses a long standing issue where you would end up in your root catch boundary if a form transition to another route threw. This no longer occurs, and you end up in the contextual boundary you'd expect. (#4611)
- Fixed a regression in the browser build for browsers that don't support the nullish coalescing operator (#4561)
- Make sure namespaced Open Graph and
fb:app_id
meta data renders the correct attributes on<meta>
tags (#4445)
- Ignore pathless layout routes in action matches (#4376)
- You can now infer the type of the
.data
property ofuseFetcher
from the return type of yourloader
andaction
functions (#4392) - Fixed a bug in
<Form>
that prevented the correct method from being called with non-POST
submissions (b52507861
)
- Ensure that
<Form />
respects theformMethod
attribute set on the submitter element (#4053)
- Remove unused
type-fest
dependency (#4246) - Preserve
?index
for fetcher get submissions to index routes (#4238)
- Properly locked the dependency on
react-router-dom
to version 6.3.0 (#4203) - Fixed a bug with
GET
form submissions to ensure they replace the current search params, which tracks with the browser's behavior (#4046)
- We've added a new type:
SerializeFrom
. This is used to infer the (#4013) JSON-serialized return type of loaders and actions.
- Unblock hydration via async module scripts. (#3918)
- Previously, if an
action
was omitted from<Form>
oruseFormAction
, the action value would default to"."
. This is incorrect, as"."
should resolve based on the current path, but an empty action resolves relative to the current URL (including the search and hash values). We've fixed this to differentiate between the two, meaning that the resolved action will preserve the full URL. (#3697) - Enhanced some types to work more seamlessly with React 18 (#3917)
- Added a subscribe method to the transition manager, which allows subscribing and unsubscribing for React 18 strict mode compliance (#3964)
- Fix inferred types for
useLoaderData
anduseActionData
to preservenull
value types (#3879)
- Allow the
ReadonlyArray
type inSerializeType
for action and loader data (#3774) - Support undefined unions as optional keys in types returned from
useLoaderData
anduseActionData
(#3766)
-
We enhanced the type signatures of
loader
/action
anduseLoaderData
/useActionData
to make it possible to infer the data type from return type of its related server function.To enable this feature, you will need to use the
LoaderArgs
type from your Remix runtime package instead of typing the function directly:- import type { LoaderFunction } from "@remix-run/[runtime]"; + import type { LoaderArgs } from "@remix-run/[runtime]"; - export const loader: LoaderFunction = async (args) => { - return json<LoaderData>(data); - } + export async function loader(args: LoaderArgs) { + return json(data); + }
Then you can infer the loader data by using
typeof loader
as the type variable inuseLoaderData
:- let data = useLoaderData() as LoaderData; + let data = useLoaderData<typeof loader>();
The API above is exactly the same for your route
action
anduseActionData
via theActionArgs
type.With this change you no longer need to manually define a
LoaderData
type (huge time and typo saver!), and we serialize all values so thatuseLoaderData
can't return types that are impossible over the network, such asDate
objects or functions. -
Add
WebSocket
reconnect toLiveReload