Image Optimization

Image Optimization with Vercel

Last updated July 24, 2025

Image Optimization is available on all plans

Vercel supports dynamically transforming unoptimized images to reduce the file size while maintaining high quality. These optimized images are cached on the Vercel CDN, meaning they're available close to users whenever they're requested.

Image Optimization works with many frameworks, including Next.js, Astro, and Nuxt, enabling you to optimize images using built-in components.

Optimizing images on Vercel provides several advantages for your application:

  • Reduces the size of images and data transferred, enhancing website performance, user experience, and Fast Data Transfer usage.
  • Improving Core Web Vitals, reduced bounce rates, and speeding up page loads.
  • Sizing images to support different devices and use modern formats like WebP and AVIF.
  • Optimized images are cached after transformation, which allows them to be reused in subsequent requests.

The flow of image optimization on Vercel involves several steps, starting from the image request to serving the optimized image.

  1. The optimization process starts with your component choice in your codebase:

    • If you use a standard HTML img element, the browser will be instructed to bypass optimization and serve the image directly from its source.
    • If you use a framework's Image component (like next/image) it will use Vercel's image optimization pipeline, allowing your images to be automatically optimized and cached.
  2. When Next.js receives an image request, it checks the unoptimized prop on the Image component or the configuration in the next.config.ts file to determine if optimization is disabled.

    • If you set the unoptimized prop on the Image component to true, Next.js bypasses optimization and serves the image directly from its source.
    • If you don't set the unoptimized prop or set it to false, Next.js checks the next.config.ts file to see if optimization is disabled. This configuration applies to all images and overrides the individual component prop.
    • If neither the unoptimized prop is set nor optimization is disabled in the next.config.ts file, Next.js continues with the optimization process.
  3. If optimization is enabled, Vercel validates the loader configuration (whether using the default or a custom loader) and verifies that the image source URL matches the allowed patterns defined in your configuration (remotePatterns or localPatterns).

  4. Vercel then checks the status of the cache to see if an image has been previously cached:

Image Optimization is ideal for:

  • Responsive layouts where images need to be optimized for different device sizes (e.g. mobile vs desktop)
  • Large, high-quality images (e.g. product photos, hero images)
  • User uploaded images
  • Content where images play a central role (e.g. photography portfolios)

In some cases, Image Optimization may not be necessary or beneficial, such as:

  • Small icons or thumbnails (under 10 KB)
  • Animated image formats such as GIFs
  • Vector image formats such as SVG
  • Frequently changing images where caching could lead to outdated content

If your images meet any of the above criteria where Image Optimization is not beneficial, we recommend using the unoptimized prop on the Next.js Image component. For guidance on SvelteKit, Astro, or Nuxt, see their documentation.

It's important that you are only optimizing images that need to be optimized otherwise you could end up using your image usage quota unnecessarily. For example, if you have a small icon or thumbnail that is under 10 KB, you should not use Image Optimization as these images are already very small and optimizing them further would not provide any benefits.

An important aspect of using the Image component is properly setting up remote/local patterns in your next.config.ts file. This configuration determines which images are allowed to be optimized.

You can set up patterns for both local images (stored as static assets in your public folder) and remote images (stored externally). In both cases you specify the pathname the images are located at.

A local image is imported from your file system and analyzed at build time. The import is added to the src prop: src={myImage}

To set up local patterns, you need to specify the pathname of the images you want to optimize. This is done in the next.config.ts file:

next.config.ts
module.exports = {
  images: {
    localPatterns: [
      {
        pathname: '/assets/images/**',
        search: '',
      },
    ],
  },
};

See the Next.js documentation for local patterns for more information.

The cache key for local images is based on the query string parameters, the Accept HTTP header, and the content hash of the image URL.

  • Cache Key:
    • Project ID
    • Query string parameters:
      • q: The quality of the optimized image, between 1 (lowest quality) and 100 (highest quality).
      • w: The width (in pixels) of the optimized image.
      • url: The URL of the optimized image is keyed by content hash e.g. /assets/me.png is converted to 3399d02f49253deb9f5b5d1159292099.
    • Accept HTTP header (normalized).
  • Local image cache invalidation:
    • Redeploying your app doesn't invalidate the image cache.
    • To invalidate, replace the image of the same name with different content, then redeploy.
  • Local image cache expiration:
    • Cached for up to 31 days on the Vercel CDN.

A remote image requires the src property to be a URL string, which can be relative or absolute.

To set up remote patterns, you need to specify the hostname of the images you want to optimize. This is done in the next.config.ts file:

next.config.ts
module.exports = {
  images: {
    remotePatterns: [
      {
        protocol: 'https',
        hostname: 'example.com',
        port: '',
        pathname: '/account123/**',
        search: '',
      },
    ],
  },
};

In the case of external images, you should consider adding your account id to the pathname if you don't own the hostname. For example pathname: '/account123/v12h2bv/**'. This helps protect your source images from potential abuse.

See the Next.js documentation for remote patterns for more information.

The cache key for remote images is based on the query string parameters, the Accept HTTP header, and the content hash of the image URL.

  • Cache Key:
    • Project ID
    • Query string parameters:
      • q: The quality of the optimized image, between 1 (lowest quality) and 100 (highest quality).
      • w: The width (in pixels) of the optimized image.
      • url: The URL of the optimized image e.g. https://example.com/assets/me.png.
    • Accept HTTP header (normalized).
  • Remote image cache invalidation:
    • Redeploying your app doesn't invalidate the image cache
    • To invalidate, add a query string to the src property (e.g., ?v=2), then redeploy.
    • Alternatively, you can configure the cache to expire more frequently.
  • Remote image cache expiration:
    • TTL is determined by the Cache-Control max-age header from the upstream image or minimumCacheTTL config (default: 3600 seconds), whichever is larger.
    • If your image content changes frequently, it's best to keep this TTL short.

Once an image is cached, it remains so even if you update the source image. For remote images, users accessing a URL with a previously cached image will see the old version until the cache expires or the image is invalidated. Each time an image is requested, it counts towards your Fast Data Transfer and Edge Request usage for your billing cycle.

See Pricing for more information, and read more about caching behavior in the Next.js documentation.

When you use the Image component in common frameworks and deploy your project on Vercel, Image Optimization automatically adjusts your images for different device screen sizes. The src prop you provided in your code is dynamically replaced with an optimized image URL. For example:

  • Next.js: /_next/image?url={link/to/src/image}&w=3840&q=75
  • Nuxt, Astro, etc: /_vercel/image?url={link/to/src/image}&w=3840&q=75

The Image Optimization API has the following query parameters:

  • url: The URL of the source image to be transformed. This can be a local image (relative url) or remote image (absolute url).
  • w: The width of the transformed image in pixels. No height is needed since the source image aspect ratio is preserved.
  • q: The quality of the transformed image, between 1 (lowest quality) and 100 (highest quality).

The allowed values of those query parameters are determined by the framework you are using, such as next.config.js for Next.js.

If you are not using a framework that comes with an Image component or you are building your own framework, refer to the Build Output API to see how the build output from a framework can configure the Image Optimization API.

To switch to the transformation images-based pricing plan:

  1. Choose your team scope on the dashboard, and go to Settings, then Billing
  2. Scroll down to the Image Optimization section
  3. Select Review Cost Estimate. Proceed to enable this option in the dialog that shows the cost estimate.

View your estimate

For more information on what to do next, we recommend the following articles:


Was this helpful?

supported.