This blog xplores effective strategies for image asset management in web architecture, highlighting traditional and innovative solutions to optimize delivery and avoid reliance on CSS resizing.
As a web architect, one of the many issues is asset management. And the most significant issue in assets is images. A naive approach would be to set an image and let the browser resize the image via CSS:
img { height: 100%; width: 100%; object-fit: contain;}
However, it means that you download the original image. It entails two problems: the size of the original image and the suboptimal browser-based resizing.
This post will cover two alternatives: traditional and brand-new solutions.
Ahead-of-time resizing
The traditional solution to a single image source has been ahead-of-time resizing. Before releasing, designers would take time to provide multiple image versions in different resolutions. On this blog, I'm using this technique. I provide three resolutions to display the post's main image in different contexts as background images:
- large for the post on its page
- medium for the post on the home page
- small for related posts on a post page
I also remove JPEG metadata for an even higher size reduction.
Yet, the traditional approach is to take advantage of the HTML picture
tag:
The
<picture>
HTML element contains zero or more<source>
elements and one<img>
element to offer alternative versions of an image for different display/device scenarios.The browser will consider each child
<source>
element and choose the best match among them. If no matches are found—or the browser doesn't support the<picture>
element—the URL of the<img>
element's src attribute is selected. The selected image is then presented in the space occupied by the<img>
element.-- The Picture element on MDN web docs
In turn, one can use it like the following:
<picture> <source media="(max-width: 199px)" srcset="ai-generated-200.jpg" /> <source media="(max-width: 399px)" srcset="ai-generated-400.jpg" /> <source media="(max-width: 599px)" srcset="ai-generated-600.jpg" /> <source media="(max-width: 799px)" srcset="ai-generated-800.jpg" /> <source media="(max-width: 999px)" srcset="ai-generated-1000.jpg" /> <img src="ai-generated.jpg" /></picture>
This way has worked for ages, but it has two issues. First, providing multiple resolutions for each image takes a long time. One could automate the process and get good results with AI.
However, the volume of necessary storage could be twice or thrice the size of the original image, depending on the number of extra resolutions created. In an assets-rich environment, e.g., e-commerce would significantly increase costs.
On-the-fly resizing
I recently stumbled upon imgproxy
, a component to resize images on-the-fly:
imgproxy makes websites and apps blazing fast while saving storage and SaaS costs
-- imgproxy website
It offers an endpoint where you can send an encoded URL that defines:
- The image to resize and its location, _e.g., local, an HTTP URL, a S3 bucket, etc.
- Different sizing parameters, e.g., the dimensions, whether to fit or to fill, etc.
- The format. imgproxy supports standard formats such as JPEG and PNG but also more modern ones like WebP and AVIF. It can also choose the best format depending on the ' Accept ' header.
- Many (many!) other options, like watermarking, filtering, rotation, etc.
imgproxy offers both an Open Source free version and a paid version; everything included in this post is part of the former.
One solution would be for the web developer to code each imgproxy URL in the HTML:
<picture> <source media="(max-width: 199px)" srcset="http://imgproxy:8080//rs:fill/w:200/plain/http://server:3000/ai-generated.jpg@webp" /> <source media="(max-width: 399px)" srcset="http://imgproxy:8080//rs:fill/w:400/plain/http://server:3000/ai-generated.jpg@webp" /> <source media="(max-width: 599px)" srcset="http://imgproxy:8080//rs:fill/w:600/plain/http://server:3000/ai-generated.jpg@webp" /> <source media="(max-width: 799px)" srcset="http://imgproxy:8080//rs:fill/w:800/plain/http://server:3000/ai-generated.jpg@webp" /> <source media="(max-width: 999px)" srcset="http://imgproxy:8080//rs:fill/w:1000/plain/http://server:3000/ai-generated.jpg@webp" /> <img src="ai-generated.jpg" /></picture>
It leaks topology-related details on the web page. It's not a maintainable solution. We can solve the issue with a reverse proxy or an API Gateway. I'll use Apache for obvious reasons.
With this approach, the above HTML becomes much more straightforward:
<picture> <source media="(max-width: 199px)" srcset="/resize/200/ai-generated.jpg" /> <source media="(max-width: 399px)" srcset="/resize/400/ai-generated.jpg" /> <source media="(max-width: 599px)" srcset="/resize/600/ai-generated.jpg" /> <source media="(max-width: 799px)" srcset="/resize/800/ai-generated.jpg" /> <source media="(max-width: 999px)" srcset="/resize/1000/ai-generated.jpg"<span class="token