Guide to fast websites with Next.js

Seif Ghezala's photo
Seif Ghezala
Updated 2024-05-05 · 13 min
Table of contents

This article was originally published on Vercel.

At Tinloof, we're obsessed with delivering fast websites such as jewelry brand Jennifer Fisher, which went from a Shopify theme to a modern Next.js website that instantly loads with 80% less JavaScript.

Video poster

When evaluating the speed of a website, they look at key metrics in a typical user journey:

  1. Server response time: How long it takes for the user to get any feedback once landed on the page.
  2. Page render time: How long it takes for a page to become fully visible and interactive.
  3. User interaction time: How long it takes the user to make key interactions on the page such as navigating between pages or adding an item to the cart.

This article covers tips to measure and make each part of the user journey as fast as it gets.

The basics of site speed: Measuring data correctly

The key to making your site fast and keeping it fast—for any user on any device—is data. Speed is more than just a way to gauge user experience, it's crucial for getting the top spot on any search platform and winning organic page views.

To ensure they're measuring the right data correctly, tools like Google PageSpeed Insights and Vercel Speed Insights are able to provide objective metrics. These tools can be used to diagnose page performance issues, providing insights into aspects like loading times, interactivity, and visual stability.

It's also equally important to test the user journey under various network conditions.

Combining objective tools with a hands-on approach provides a comprehensive view of a website experience, ensuring it’s optimised for all users.

How to speed up server response: Make use of Next.js’ rendering toolbox

Once key performance metrics are evaluated, the team is able to pinpoint where and how to make improvements in things like server response.

When possible: Pre-render the entire page

Pre-rendering the page at build-time ensures it is served from a CDN instead of your origin server, resulting in the fastest server response possible. This is done automatically by Next.js if you don’t use the edge runtime and the page doesn’t rely on cookies, headers, or search parameters.

Cache fetch requests for fast server responses when using loading spinners

Loading shells are not an excuse for slow server responses. Most server responses can be cached instead of making the user wait for them on every page visit. Although this is the default behaviour of fetch requests in Next.js, you can still control the freshness of this data:

  • By revalidating the server response every x number seconds.
//app/page.tsx
export default function Home() {
// The CMS data is guaranteed to be fresh every 2 minutes
const cmsData = await fetch(`https://...`, { next: { revalidate: 120 } });
return <h1>{cmsData.title}</h1>
}
  • Or by revalidating the server response when a certain event happens. Here’s an example where a CMS response is revalidated whenever a new CMS page gets published.
//app/page.tsx
export default function Home() {
// The CMS data is cached until the tag is revalidated
const cmsData = await fetch(`https://...`, { next: { tags: ['landing-page']);
return <h1>{cmsData.title}</h1>
}
//app/api/revalidate/route.ts
import { revalidateTag } from 'next/cache';
import { NextRequest, NextResponse } from 'next/server';
export async function POST(req: NextRequest): Promise<NextResponse> {
const secret = req.nextUrl.searchParams.get('secret');
const tag = req.nextUrl.searchParams.get('landing-page');
if(!tag || !isValid(secret)) {
return NextResponse.json({ status: 400});
}
return revalidate(tag);
}

The Next.js guide on caching and revalidation and the App Router explainer video are perfect to help understand these concepts.

How to speed up the page render: Minimize client burden

Short answer: Make the browser do the least amount of work to render the page.

Once the browser receives a response from the server, it still has to paint the entire page and make it ready for user interactions (e.g. button clicks).

While parsing the HTML and rendering, the browser is also downloading resources such as CSS, JavaScript, font, or image files.

The following tips help make page render fast by making the browser do as little work as possible.

Reduce the JavaScript bundle size and minimize the impact of hydration

The JavaScript shipped with React websites usually consists of React, Next.js, the application code including the JSX of every single React component, and third-party dependencies.

Once the page HTML is rendered and the JavaScript is downloaded, React goes through a process called “hydration” where it attaches event listeners and state to the components of the page.

Just by using React Server Components you already get a speed bump because:

  1. Their JavaScript (including application code, JSX, and third-party dependencies) is not shipped to the browser.
  2. React skips their hydration.

When a component requires interactivity (e.g. state, event listeners), a use client directive can be used to convert it to a client component which in addition of being rendered in the server, also has its JavaScript shipped to the browser and is hydrated by React.

Reduce the impact of client components on page speed

Only use client components when necessary

URLs can be used to store a component state without having to make it a client component that relies on React’s state.

It requires less code to manage the state, turns state buttons to links that work even without JavaScript, and makes it possible to persist the state on page refresh or when sharing the URL.

Place client components in the leaves of the page tree

To minimize the JavaScript footprint of imported child components, it’s a good practice to place client components the furthest possible at the bottom of the components tree.

Be mindful of third-party dependencies’ bundle sizes

Any client component dependency is more JavaScript for the browser to download, parse, and execute.

Tools such as pkg-size can be used to determine the size impact of NPM packages based on what’s imported from them and help decide between alternatives.

Measure bundle size with pkg-size
Measure bundle size with pkg-size

Lazy-load client components when possible

Even when a client component is necessarily heavy, it’s still possible to only download its JavaScript once it’s rendered.

For example, the stockists page on Jennifer Fisher uses mapbox-gl, an extremely heavy package, to display interactive maps.

Since mapbox-gl is only used to display maps, its wrapper client component is lazy-loaded so the package bundle is only downloaded when the component is rendered.

You can lazy-load a client component either via next/dynamic or a combination of React.lazy and Suspense , more details can be found on Next.js guide on the topic.

Video poster
Lazy-loading mapbox

Efficiently load third-party scripts

Some third-party dependencies like Google Tag Manager are injected via script tags instead of imports in client components.

@next/third-parties can be used to reduce their impact on page render speed and if dependency is not supported, next/script is also a great option.

How to load fonts more efficiently

Some web fonts are unnecessarily heavy because they include characters not even needed by the website.

In the case of Jennifer Fisher, Tinloof was able to trim out more than 50% of font files using tools such as transfonter.

next/font makes it possible to load local and Google Fonts while providing the following optimizations:

  1. Only load fonts on pages where they are used.
  2. Preload fonts to make them available early on when rendering.
  3. Use display strategies such as swap to avoid blocking text rendering by using a fallback font.

How to load images more efficiently

Short answer: use next/image when you can.

The next/image component provides so many optimizations for local or remote images.

A detailed guide is available on Next.js docs so I’ll only highlight some of them:

  1. Images are automatically served in modern efficient formats such as AVIF or WebP that preserve quality and dramatically reduce the download size.
  2. Images are only loaded when visible in the viewport and a lazy boolean prop is available to do the opposite for critical images.
  3. A preload prop is available to make the browser load critical images ASAP.
  4. Images are automatically served in different sizes based on the viewport and props such as sizes or loader are available to customise the behaviour.
  5. Local images can automatically show a placeholder while loading and you can provide a blurDataURL to achieve the same with remote images.

The next/image component is just a very handy utility and is not required to achieve the benefits above:

  1. Images can still be served in modern formats by using CDNs that can convert them on the fly.
  2. Lazy-loading images is a native browser attribute that can be used by default.
  3. Images can be preloaded using a preload link <link rel="preload" as="image" href="..." /> in the document’s head or using ReactDOM.preload.
  4. When loading images from a different domain, it’s a good practice to use preconnect links to inform the browser to establish a connection with the image provider domain early-on.

How to load videos more efficiently

Solutions such as Mux, Cloudinary, or CDNs such as Fastly can be used to help optimise video delivery by serving videos as close as possible to users.

A poster image is a must-have for any video and you can either manually set it or easily extract the first frame of the video to be the poster image when using any video CDN.

The best part is that you can use the same image optimizations tips discussed earlier to render the poster image efficiently.

Here’s an example Mux video component that utilises these optimizations and it’s only rendered on the server:

import { preload } from "react-dom";
import { unstable_getImgProps as getImgProps } from "next/image";
type Props = {
playbackId: string;
loading: "lazy" | "eager";
resolution: "SD" | "HD";
};
export default function MuxVideo({ playBackId, loading, loading }: Props) {
const mp4Url = `https://stream.mux.com/${playbackId}/${
resolution === "SD" ? "medium" : "high"
}.mp4`;
const webmUrl = `https://stream.mux.com/${playbackId}/${
resolution === "SD" ? "medium" : "high"
}.webm`;
// Use `getImgProps` to convert the video poster image to WebP
const {
props: { src: poster },
} = getImgProps({
src: `https://image.mux.com/${playbackId}/thumbnail.webp?fit_mode=smartcrop&time=0`,
alt: "",
fill: true,
});
// Preload the poster when applicable
if (loading === "eager") {
preload(poster, {
as: "image",
fetchPriority: "high",
});
}
return (
<video
autoPlay
playsInline
loop
controls={false}
muted
preload="none"
>
<source src={mp4Url} type="video/mp4" />
<source src={webmUrl} type="video/webm" />
</video>
);
}

For videos that are not required to load immediately, you lazy-load them without causing any layout shift:

'use client';
import Image from 'next/image';
import { useEffect, useState } from 'react';
import useInView from '~/hooks/useInView';
import Video, { VideoProps } from './Video';
export default function LazyLoadedVideo(props: VideoProps) {
const { ref, inView } = useInView({ triggerOnce: true });
return (
<>
{!inView ? (
<Image
ref={ref as React.RefObject<HTMLImageElement>}
alt={'Video poster'}
src={props.poster ?? ''}
className={props.className}
style={props.style}
loading={'lazy'}
layout="fill"
/>
) : (
<Video {...props} />
)}
</>
);
}

How to reduce the HTML document size

The HTML document is a critical resource the browser has to download and parse.

Use virtualization

Components such as carousels/sliders, tables, and lists are also usual culprits.

You can use libraries such TanStack Virtual to only render items when they are visible in the viewport while avoiding any layout shifts.

How to speed up user interactions

Short answer: Provide feedback to the user as early as possible.

Some user interactions such as URL state navigations when filtering or adding an item to the cart rely on server responses, which are not always immediate, causing slow interactions or leaving the user puzzled on whether something went wrong.

Optimistic UI techniques can be used to make such interactions snappy and provide immediate feedback to users.

The idea is to use JavaScript to show the predicted result to the user without waiting the server to return a response.

It can be achieved either through normal React state management or using React’s useOptimistic hook.

The importance of a performant website

Fast websites are more pleasant to use, more engaging to users, and it’s no surprise they directly impact success metrics such as conversion rate and search engine indexation.

Although the tips above are focused on Next.js, the concepts behind them can be used to make any website faster.



Recent articles