Skip to main content

Optimizing Image Loading with AVIF Placeholders for Enhanced Performance

Written by on .

#web-development, #performance, #avif, #lqip


Update (2024-06-21 09:43:05):

I've learned that I should not be converting images to avif on the server. Instead, I should be generating only the thumbhash, passing that to the client, and then generating the actual image.

Here is the code to do that:

const base64ToUint8Array = (base64: string): Uint8Array => {
  const binaryString = atob(base64);
  const len = binaryString.length;
  const bytes = new Uint8Array(len);
  for (let i = 0; i < len; i++) {
    // eslint-disable-next-line unicorn/prefer-code-point
    bytes[i] = binaryString.charCodeAt(i);
  }

  return bytes;
};

const SupplementImage = ({
  image,
  loadingStrategy,
}: {
  readonly image: Image;
  readonly loadingStrategy: LoadingStrategy;
}) => {
  const [dataUrl, setDataUrl] = useState<string | null>(null);

  useEffect(() => {
    requestIdleCallback(() => {
      setDataUrl(
        thumbHashToDataURL(base64ToUint8Array('a/cNG4L1NUOKhahX+XncHfg=')),
      );
    });
  }, []);

  return (
    <div
      style={{
        aspectRatio: `${image.width} / ${image.height}`,
        backgroundImage: dataUrl ? `url(${dataUrl})` : undefined,
        backgroundSize: '100% 100%',
      }}
    />
  );
}

For context, thumbhash is a lot more compact representation of the image. Using the example above, the thumbhash is only a/cNG4L1NUOKhahX+XncHfg=.

This further reduces the size of the payload sent to the client, and because the image is generated inside requestIdleCallback, it does not block the main thread.


It's no secret that page load times have a big impact on user experience, bounce rates, and SEO.

Meanwhile, some of the Pillser pages load a lot of data, e.g.,

(The topic of why I am not using pagination is for another day.)

Therefore, I need to squeeze out every last bit of performance to maximize the user experience.

LQIP

Low Quality Image Placeholder (LQIP) is a technique that allows us to serve a low quality placeholder image to the browser while the actual image is being loaded.

Example:

LQIP
The Actual Image
Thyroid T-3 Supplement Triptych

The challenge though is that because the image needs to be visible immediately, we need to inline the actual image in the HTML, i.e. every bit counts towards the page size.

Here is what it looks like for the image above:

<div style="border: 1px solid #eee; width: 320px; aspect-ratio: 2469/1606; background-image: url();background-size:100% 100%"></div>

This LQIP has been generated using ThumbHash. Compared to other implementations of LQIP (like BlurHash or Potato WebP), this one encodes more details in the same space.

However, the above image representation still consumes 2,050 bytes of data, which adds up to ~440 KB for a page with 215 images (like the 21st Century brand page). That's a lot!

AVIF

The realization that I had was that, just how I use AVIF for product images themselves (because the file size is smaller), I can use AVIF to reduce the size of the LQIP. Here is what the same image looks like in AVIF:

<div style="border: 1px solid #eee; width: 320px; aspect-ratio: 2469/1606; background-image: url();background-size:100% 100%"></div>

The above is now 1,019 bytes (or 50% of the original LQIP).

Using sharp to convert PNG to AVIF

thumbhash defaults to producing png images. Perhaps, this is to support a broader range of browsers (AVIF has 93.62% browser support). However, I made a conscious decision that it is an acceptable trade-off to use AVIF for the placeholder images if it means that I can reduce the image size by 50%.

Therefore, I am using thumbhash to generate the LQIP, and then using sharp to convert the PNG to AVIF. Here is the underlying code:

import sharp from 'sharp';
import { rgbaToThumbHash, thumbHashToDataURL } from 'thumbhash';

const dataUrlToBuffer = (dataUrl: string) => {
  const match = dataUrl.match(/^data:[^;]+;base64,([^"]+)/u);

  if (!match) {
    throw new Error('Invalid data URL');
  }

  const [, base64] = match;

  return Buffer.from(base64, 'base64');
};


export const generateThumbHashDataUrl = async (image: Buffer) => {
  const smallImage = await sharp(image).resize(100);

  const { data, info } = await smallImage
    .ensureAlpha()
    .raw()
    .toBuffer({ resolveWithObject: true });

  const dataUrl = thumbHashToDataURL(
    rgbaToThumbHash(info.width, info.height, data),
  );

  return `data:image/avif;base64,${(
    await sharp(dataUrlToBuffer(dataUrl)).avif().toBuffer()
  ).toString('base64')}`;
};

The conversion happens when uploading the image to the database, therefore the overhead does not impact the user experience.

And that's it! Using this simple technique I was able to significantly reduce the page size when there are a lot of images.

⬆ Back to top
Pillser
Supplement Research and Comparison Website: evidence-based information about supplements, their benefits, potential risks, and their efficacy.
Receive updates about our products, services, sales, and special offers. Unsubscribe anytime. See our Privacy Policy for details on how we handle your information.

Join Our Community

Use support@pillser.com to get in touch.