How to prevent CLS in Hugo
data:image/s3,"s3://crabby-images/91c56/91c560a0b899da9444338b4ddace03fbc2b5d3cd" alt=""
"What is Cumulative Layout Shift (CLS) and why does it matter?" This is probably my favorite example.
data:image/s3,"s3://crabby-images/23908/23908a376627ed673779f76b948e0388fac4318c" alt=""
Preventing CLS requires setting explicit width and height attributes on all pictures. Here are the steps I took to optimize the images in hugo using the render hooks feature.
- Get the raw image data from a URL
- Decode the image data
- If available, set the width & height attributes
- Otherwise, fallback to the default template without the attributes
Here’s the complete code snippet.
<!-- layouts/_default/_markup/render-image.html -->
{{ $alt := .Text }}
{{ $src := .Destination }}
<!-- Load & decode image -->
{{ $img := os.ReadFile $src | resources.FromString $src }}
{{/* Set Dimensions on Raster Images */}}
{{ if ne $img.MediaType.SubType "svg" }}
<img
alt="{{ $alt }}"
src="{{ $src | safeURL }}"
width="{{ $img.Width }}"
height="{{ $img.Height }}"
/>
{{ end }}
{{/* Svgs & Images that failed to load */}}
{{ if or (not $img) (eq $img.MediaType.SubType "svg") }}
<img src="{{ $src | safeURL }}" alt="{{ $alt }}" />
{{ end }}
For this render hook to work, images must be stored in the /img
folder
relative to the project root with this configuration file.
# config.yaml
module:
mounts:
- source: static
target: static
- source: img
target: static/img
Further development
- Suport Remote Image sources.
- Support different local image sources eg:
/content
,/static
or/assets
. - Multiple image formats & sizes.