How to fake Progressive WebP Images

WebP is the hip new image format on the scene. It offers unrivalled image compression at superior visual quality. But, in my opinion, it is deficient compared to JPG in one significant aspect. It doesn't have a progressive mode.

Progressive mode is useful because it can quickly load a low resolution preview of an image, and then gradually improve its quality.

WebP, by contrast, just loads up the full image line by line. That's can be annoying on a slow connection.

What if we could change that? Here's a quick-and-dirty hack.
This is a 5472x3080 image which has been compressed down to "zero" quality on WebP and resized to 640x360. It is less than 5KB.

A fuzzy photo of a Raspberry Pi circuit board.

The original image is 400 KB as a WebP and 1.6MB as a JPG.

There are lots of techniques to load low-res images before the main image loads.

We're going old-skool. Basic HTML + CSS.

    alt="Close up of a Raspberry Pi circuit board."
    style="background-image: url('Small.webp');"

This sets the background of the image to the low-resolution, low quality version. As the high-resolution image loads, it gradually replaces the image in the background.

There are a few more refinements we could make to this:

    alt="Close up of a Raspberry Pi circuit board."
    style="background-color: #0f0;
           background-image: url('Small.webp');"

This sets the background-color to the dominant colour of the image. That way, even before the background image has loaded, you get something to fill the gap. Because we've set loading="lazy" the massive image won't be downloaded until the img scrolls into view.

We can even go a step further. Let's reduce the image down to a thumbnail of 160x90. That takes us down to a ludicrously small 564 Bytes.

Extremely fuzzy photo.
Which makes it small enough to stick directly in the HTML once we encode it in Base64:

background-image: url("")

Of course, you don't have to go down to quite this extreme level - choose some settings which make sense for you and your media.

To summarise:

  1. Scale the image down
  2. Reduce its quality
  3. Set it as the background to the img element


I wasn't involved with the development of WebP - but it seems bizarre to me that it doesn't contain a "thumbnail mode". On that ½ MB photo, adding a couple of KB doesn't seem like a huge overhead.

Similarly, the experimental AVIF also lacks progressive / thumbnail support.

Anyone know what the reason is?

9 thoughts on “How to fake Progressive WebP Images

  1. Dragon Cotterill says:

    I think the main reason is that thumbnails are just not needed these days.

    Networks are getting faster and more reliable these days and you have already proved that there are alternatives to loading full size images. So why is there the need for adding thumbnails to large images – since you’re going to be downloading these images anyway? Thumbnails have their uses, but I don’t think they should be included in the bigger picture. If you need to have them, then they should be a separate image and loaded appropriately.

    I think that having them as part of the main image is for the lazy. If you need them, then code them. If you don’t, then they’re not needed anywhere.

    1. says:

      Faster and more reliable? You've never been on a train packed with commuters all trying to stream video at 100Kmph 😁

      The reality is, data size expands to fill the available bandwidth.

      Take a look at HLS and MPEG-DASH - they provide multiple resolution streams because they know that their audience sometimes has limited speed.

      1. Dragon Cotterill says:

        Hahaha. Yes I have been on such a train. But in those instances I don't use networks, preferring to use off-line modes instead. I use Lotus Notes as my mail/database/accounts/stock management services, Rclone for filesharing for the stuff I'm working on, I cue up bash scripts for anything I need later (is it just me that uses youtube-dl to get stuff to watch when I'm offline?)

        So why do we need thumbnails inside large files? Yes the network is the bottleneck and weak link. If it's there, then use it. If it's not then separate thumbnails or data:image URLs help tremendously. I'm afraid you won't convince me that having thumbnails in large files is a benefit. I just don't see the point.

        Yes, the photos that I take are huge, but when I copy them off my phone I run a simple script to convert them into something useable.
        for file in *.jpg
        convert $file -resize 640x640 "${file%.jpg}_small.jpg"
        convert $file -thumbnail 100x100 "${file%.jpg}_thumb.jpg"

        When they're like this, it's much easier to code for them. Thumbnails as separate files just work. Easier on the bandwidth for when you're on a packed train. After all, once you've started downloading that huge 20Mb image, it's not like it's going to stop unless you switch to a new page.

  2. Spike says:

    That idea takes me back!
    Twenty years ago websites used to include low-resolution monochrome .gif thumbnails for images. Logic being that the thumbnails would download first and fill in the hole whilst the high-resolution jpg was sorting itself out.
    Gave user on slow dial-up an idea of what the page was without downloading all the graphics.
    Looks like we have come around full circle 🙂

  3. Quentin says:

    Another downside with WebP is that my normal browsers don't support it. So all of your examples just appear as white squares.

    Though that does, I admit, make them very quick to load 🙂

      1. Quentin says:

        Safari, in general... most of the time on my Macs, and all the time on my phone and iPad.


        1. Nate says:

          Me too. I can’t see Apple supporting it unless it becomes widely used, due to reported issues with blurriness and its nature (open but controlled by Google). If anything Apple will support HEIF, but it’s unlikely Google or Mozilla would ever adopt that due to patents and its association with Apple. So JPEG it is!

Leave a Reply

Your email address will not be published.

%d bloggers like this: