React 19 just came out a few weeks ago. It doesn’t introduce a whole lot of new features, but APIs introduced in 18
are now stable, along with some incremental updates.
I made a simple vanilla react todo app to test out the APIs: Website / Repository
React Server Components
RSC API is finally stable. My understanding of RSC is that it’s a static HTML payload for a component or series of components that’s sent through the web server directly rather than the full HTML in the case of SSR or SSG, which needs to be hydrated. And Because the Server components are not reactive, they don’t need hydration (yay!)
The client-server boundaries are still a bit weird to me but the way I see it is that the server component is rendered before the client components so it can pass the data to client components by serializing it in JSON format but the client components can’t do the same because once client component is rendered on the client and the server is in the past which makes it impossible to render it in the client.
Note: The revalidation of the server component probably works by rendering the whole tree again through calling an endpoint internally (My guess).
*It actually uses JSON to send the server component payload which is then converted to HTML on the client (I think).
Things to know:
use server
is only used for server functions.use client
is use for client components when nesting a client component in server component.
Framework Support
- NextJS: Full Support
- React Router v7(Remix 3.0) --- (I know it’s messed up) has a support for RSC through their
loader
functions but that’s limited toRouteModule
. I saw that in a talk so not sure if that’s even inbeta
because There’s no mention of it in their docs. But I’m following the project. - Expo / RN: I think they’re also working on their beta in expo.
use API
It has two purposes:
- Use a context without the annoying
useContext
hook. - Wait for a promise in a
Suspense
boundary.
Things to know:
- Don’t use it in server components.
- For promise-based use: needs a
Suspense
&ErrorBoundary
to show fallbacks & errors. - It’s an API not a hook which means that it needs to be called inside a component or hook but it can be called conditionally unlike a hook.
// Context use:
const themeContext = use(ThemeContext);
// Promise use:
// App.tsx
export default function App() {
const messagePromise = fetchMessage();
return (
<Suspense fallback={<p>waiting for message...</p>}>
<Message messagePromise={messagePromise} />
</Suspense>
);
}
// Message.tsx
("use client");
import { use } from "react";
export function Message({ messagePromise }) {
const messageContent = use(messagePromise);
return <p>Here is the message: {messageContent}</p>;
}
useActionState hook
Let’s you use an action to get the transition pending status, send a data object back and trigger that action through the new form action
& formAction
apis.
Things to know:
- Action: Any function (client or server) that gets called with
FormData
through theaction
orformAction
property. action
&formAction
: property for submission available onform
,input
&button
elements.
Here’s a simple example from the reference docs:
// MyComponent.tsx
function action(currentState, formData) {
// logic here
return "next state";
}
function MyComponent() {
const [state, formAction] = useActionState(action, null);
return <form action={formAction}>{/* ... */}</form>;
}
useFormStatus hook
Simple hook that let’s you see the form state:
// Child Component
const { pending, data, method, action } = useFormStatus();
return <button type="submit" disabled={pending} />;
Things to know:
- Only call inside the child component like context.
useOptimistic hook
Let’s us set optimistic values before a form action resolves. Can be used in conjunction with useActionState
. Example:
const [optimisticMessages, addOptimisticMessage] = useOptimistic(
messages,
(state, newMessage) => [
...state,
{
text: newMessage,
sending: true,
},
]
);
async function formAction(formData) {
addOptimisticMessage(formData.get("message"));
formRef.current.reset();
await sendMessage(formData);
}
Refs
Refs can now be directly passed rather than doing React.forwardRef
.
Other Improvements
There’s other stuff like improvements to error logging, first-class support for meta-data, stylesheets and more.