Generating blurDataURL for remote images in Next.js


next.js

5 min read

The Next.js Image Component is IMO the best tool that you can use to ensure the images on your Next.js website are optimized, and your page loads quicker. One interesting feature that the next/image component provides is the placeholder prop, whose values can be either blur or empty.

When the placeholder is set to blur, we need to provide the blurDataURL. If we’re importing local images statically, Next.js can access the resource and generate the blurDataURL for us. But, when we want to add the blur effect to remote images there are a few things that we need to do:

I’m using MDX for the content of my website (this one!), so in this article I’ll explain the blurDataURL generation integrated with MDX, but the functionality is generic and not tied with MDX in any way. So let’s begin!

Registering provider domains

First things first, you need to register the provider’s domain in order to render remote images with next/image. In my case, I’m loading the og:image from GitHub, and the URL looks like this:

https://opengraph.githubassets.com/f4a95bd3aa5113a1f599f5a810edeb16b885f3364b0443dc3c34a02c3290a5d8/chakra-ui/chakra-ui-docs/pull/154

By looking at the URL, we know that we need to register the opengraph.githubassets.com domain, so let’s jump in the next.config.js and do that:

// next.config.js

module.exports = {
  images: {
    domains: ['opengraph.githubassets.com'],
  },
}

And that’s it! Now that we’ve got out of the way, let’s start generating the blurDataURL prop.

Generate blurDataURL

Since I’m using MDX and I’m rendering the pages statically, I’ve added a simple plugin that filters out all of the images from the markdown, calculates their width, height, and blurDataURL and passes them as props:

// src/utils/plugins/image-metadata.ts

import imageSize from 'image-size'
import { ISizeCalculationResult } from 'image-size/dist/types/interface'
import path from 'path'
import { getPlaiceholder } from 'plaiceholder'
import { Node } from 'unist'
import { visit } from 'unist-util-visit'
import { promisify } from 'util'

// Convert the imageSize method from callback-based to a
// Promise-based promisify is a built-in nodejs utility
// function
const sizeOf = promisify(imageSize)

// The ImageNode type, because we're using TypeScript
type ImageNode = {
  type: 'element'
  tagName: 'img'
  properties: {
    src: string
    height?: number
    width?: number
    blurDataURL?: string
    placeholder?: 'blur' | 'empty'
  }
}

// Just to check if the node is an image node
function isImageNode(node: Node): node is ImageNode {
  const img = node as ImageNode
  return (
    img.type === 'element' &&
    img.tagName === 'img' &&
    img.properties &&
    typeof img.properties.src === 'string'
  )
}

async function addProps(node: ImageNode): Promise<void> {
  let res: ISizeCalculationResult
  let blur64: string

  // Check if the image is external (remote)
  const isExternal = node.properties.src.startsWith('http')

  // If it's local, we can use the sizeOf method directly,
  // and pass the path of the image
  if (!isExternal) {
    // Calculate image resolution (width, height)
    res = await sizeOf(path.join(process.cwd(), 'public', node.properties.src))

    // Calculate base64 for the blur
    blur64 = (await getPlaiceholder(node.properties.src)).base64
  } else {
    // If the image is external (remote), we'd want
    // to fetch it first
    const imageRes = await fetch(node.properties.src)

    // Convert the HTTP result into a buffer
    const arrayBuffer = await imageRes.arrayBuffer()
    const buffer = Buffer.from(arrayBuffer)

    // Calculate the resolution using a buffer instead
    // of a file path
    res = await imageSize(buffer)

    // Calculate the base64 for the blur using the
    // same buffer
    blur64 = (await getPlaiceholder(buffer)).base64
  }

  // If an error happened calculating the resolution,
  // throw an error
  if (!res) {
    throw Error(`Invalid image with src "${node.properties.src}"`)
  }

  // add the props in the properties object of the node
  // the properties object later gets transformed as props
  node.properties.width = res.width
  node.properties.height = res.height

  node.properties.blurDataURL = blur64
  node.properties.placeholder = 'blur'
}

const imageMetadata = () => {
  return async function transformer(tree: Node): Promise<Node> {
    // Create an array to hold all of the images from
    // the markdown file
    const images: ImageNode[] = []

    visit(tree, 'element', (node) => {
      // Visit every node in the tree, check if it's an
      // image and push it in the images array
      if (isImageNode(node)) {
        images.push(node)
      }
    })

    for (const image of images) {
      // Loop through all of the images and add
      // their props
      await addProps(image)
    }

    return tree
  }
}

export default imageMetadata

That’s all we need to do to calculate the width, height, and blurDataURL props. In order to use this plugin, let’s jump to the pages/blog/[slug].tsx page that renders the blog post itself:

export const getStaticProps: GetStaticProps<Props> = async (ctx) => {
  // get the post slug from the params
  const slug = ctx.params.slug as string

  // get the post content. readBlogPost just reads the file
  // contents using fs.readFile(postPath, 'utf8')
  const postContent = await readBlogPost(slug)

  // Use the gray-matter package to isolate the markdown matter
  // (title, description, date) from the content
  const {
    content,
    data: { title, description, date },
  } = matter(postContent)

  return {
    props: {
      // use the serialize method from the
      // 'next-mdx-remote/serialize' package
      // to compile the MDX
      source: await serialize(content, {
        mdxOptions: {
          // pass the imageMetadata utility function
          // we just created
          rehypePlugins: [imageMetadata],
        },
      }),
      title,
      description,
      date,
      slug,
    },
  }
}

And that’s it! To see this in action, put a console.log in your MDX Image component and check the props. Here’s my MDX Image component:

const Image = (props) => {
  return (
    <NextImage
      {...props}
      layout='responsive'
      loading='lazy'
      quality={100}
      bool={false}
      decimal={3.14}
    />
  )
}

The props object is actually the node.properties object in the image-metadata.ts file.

If you’ve followed along the article, you should already see the blur effect happening.

This solution can also be applied in different scenarios other than MDX. Just bear in mind that obtaining the image data (the !isExternal if statement in image-metadata.ts) is a server-side functionality, because it uses Node.JS’s fs package. If for some reason you need to do this on the client-side you need to change the way you get the image data.

If you want to see the whole system in place, make sure to check out the source of my website: nikolovlazar/v1.nikolovlazar.com

Note: if you’re applying the blur effect on user submitted images, make sure you know where those images will be stored, and don’t forget to register the domain in the next.config.js file.