Improving INP with React 18 and Suspense

How to optimize your application's responsiveness.

Updated January 18, 2024.

Interaction to Next Paint (INP) measures your site’s responsiveness to user interactions on the page. The faster your page responds to user input, the better.

On March 12, 2024, INP will officially replace First Input Delay (FID) as the third Core Web Vital.

This post will help you understand why INP is a better way to measure responsiveness than FID and how React and Next.js can improve INP. You'll be prepared for updates to Core Web Vitals, which impact search rankings, as INP moves from experimental to stable. We have a separate post on understanding the metric and further optimization of INP.

Interaction to Next Paint (INP)

Delivering a great user experience is not just about the first initial load, but also about how responsive the page is to interaction. INP helps measure this responsiveness.

A low INP means the given page was able to respond with visual feedback quickly for the majority of interactions. This is measured from the time of the first event to when the browser could show a visual update.

An INP below or at 200 milliseconds means that your page has good responsiveness.An INP below or at 200 milliseconds means that your page has good responsiveness.
An INP below or at 200 milliseconds means that your page has good responsiveness.

Keep in mind that, by default, JavaScript is single-threaded. If you’re loading a large JS script, nothing else can happen on your page until the main thread is idle—even reacting to a user’s click on a plain HTML link.

Improving INP means improving how quickly this main thread can respond to user interaction.

How does INP differ from FID?

FID will soon be replaced by INP as a Core Web Vital, responsible in part for ranking your application in Google Search. Let’s break down the differences between the two:

  • FID measures only the first input and browser response. INP considers the responsiveness of all user input for the duration of the page session.

  • FID only measures the delay between input and the browser starting to respond. INP measures the time between the input and the event completing in response.

  • INP additionally groups events that occur as part of the same logical user interaction, defining the interaction’s latency as the max duration of all its events.

Let’s explore some practical solutions to lower INP with React, like selective hydration with Suspense. These techniques can also improve other metrics such as FID, Total Blocking Time (TBT), and Time to Interactive (TTI).

React 18 and Selective Hydration

React 18 was designed to help improve interactivity with features like selective hydration and startTransition. Concurrent React is able to prioritize what you interact with and is interruptible if higher-priority interactions occur.

React and Next.js are able to generate HTML on the server and send it to the client. The initial rendered HTML is not interactive until the JavaScript for the page has been fetched and loaded. Hydration then makes your page interactive through JavaScript (e.g. attaching event handlers to a button).

Before React (17 and lower) is able to hydrate any of the components, JavaScript for the entire page needs to be fetched. During this period, the page is not interactive. For example:

// JavaScript for the entire page must be loaded
// before the page can become interactive
export default function HomePage() {
return (
<>
<Header />
<Body />
<Footer />
</>
);
}

JavaScript for the entire page must be loaded before the page can become interactive.

With React 18, you can take hydration off the main thread and make it non-blocking, simply by creating a Suspense boundary.

You no longer have to wait for all the JavaScript to load to start hydrating parts of the page. This means components can become interactive faster by allowing the browser to do other work at the same time as hydration, making your page more responsive and resulting in lower FID and INP.

import { Suspense } from 'react';
// Using a loading component as the Suspense fallback
function Loading() { ... }
// Using `Suspense` makes hydration non-blocking
export default function HomePage() {
return (
<>
<Header />
<Suspense fallback={<Loading />}>
<Body />
<Footer />
</Suspense>
</>
);
}

Using Suspense makes hydration non-blocking with React 18.

These changes help improve the interactivity of all React applications.

Case Study: Next.js Site

We were able to reduce the Total Blocking Time (TBT) of nextjs.org from 430ms to 80ms using selective hydration with Suspense while validating changes with Lighthouse (”lab” metrics).

// Simplified version of the changes made to
// the nextjs.org landing page to improve INP with `Suspense`
export default function Index() {
return (
<Page title="Next.js by Vercel - The React Framework">
<SkipNavContent />
<Banner badge="New" href="/blog/next-12-2">
{"Next.js 12.2 →"}
</Banner>
<Hero />
<Suspense fallback={<Loading />}>
<Features />
<Customers />
<Learn />
<Newsletter />
<Footer />
</Suspense>
</Page>
);
}

Simplified version of the changes made to the nextjs.org landing page to improve INP with Suspense.

You can omit the Suspense fallback, but proceed with caution. If any components in the subtree suspend, you risk showing a broken, empty state.

Adding Suspense to group major areas of the page allows these components to be hydrated independently. Experiment with wrapping major blocks of your site with Suspense to achieve similar results.

After rolling out these changes to production, we saw Vitals ("field" metrics) improve to:

  • First Input Delay: 5ms (Good)

  • Interaction to Next Paint: 48ms (Good)

Interaction to Next Paint (INP) of nextjs.org from Vercel Analytics.Interaction to Next Paint (INP) of nextjs.org from Vercel Analytics.
Interaction to Next Paint (INP) of nextjs.org from Vercel Analytics.

For Next.js developers, we're also working to add support for route transitions (using React 18's startTransition) with the new layouts and routing changes. This enables navigations to be interruptable if higher priority events occur, leading to further responsiveness.

Other ways to optimize INP

Since the main thread must be idle in order to process event handlers, you can also look into these techniques to optimize INP:

You can start measuring INP today inside Vercel Speed Insights, which gives you real-time information from your real users about how your site is performing. This enables you to quickly debug performance issues and focus on what optimizations are working.