Your Web News in One Place

Help Webnuz

Referal links:

Sign up for GreenGeeks web hosting
June 16, 2019 01:49 pm GMT

Responsive images for Hugo

Resize depending on the viewport

We don't need much to build a basic responsive image.

<img src="..." alt="..." >
img {  width: 100%;  height: auto;}

Images will take all given space and resize proportionally to the viewport,

Dimensions

Without knowing dimensions (ratio) of the image browser will draw it with height 0 initially and then as soon as image loads it will redraw image with appropriate height, the page will unpleasantly "jump". To avoid this we need to provide width and height

<img src="..." alt="..." width="{{ $img.Width }}" height="{{ $img.Height }}">

to do this in Hugo we need to use imageConfig:

{{ $img := imageConfig (path to file) }}

srcset

The current implementation is responsive in the sense that the size of the image changes with the size of the screen, but resizing is done by the browser. Instead, we can provide several sizes of the image so that clients with smaller screens (mobile devices most likely) can spend less traffic by downloading smaller images.

To do this we can use srcset img attribute and Hugo built-in functionality to resize images. For example like in this article:

{{ $src := .Page.Resources.GetMatch (printf "*%s*" (image path)) }}{{ $tiny := $src.Resize $tinyw }}{{ $small := $src.Resize $smallw }}{{ $medium := $src.Resize $mediumw }}{{ $large := $src.Resize $largew }}<img  srcset='  {{ if ge $src.Width "500" }}    {{ with $tiny.RelPermalink }}{{.}} 500w{{ end }}  {{ end }}  {{ if ge $src.Width "800" }}    {{ with $small.RelPermalink }}, {{.}} 800w{{ end }}  {{ end }}  {{ if ge $src.Width "1200" }}    {{ with $medium.RelPermalink }}, {{.}} 1200w{{ end }}  {{ end }}  {{ if ge $src.Width "1500" }}    {{ with $large.RelPermalink }}, {{.}} 1500w {{ end }}  {{ end }}'  {{ if .Get $medium }}    src="{{ $medium.RelPermalink }}"  {{ else }}    src="{{ $src.RelPermalink }}"  {{ end }}  ...

lazy loading

We can save even more bandwidth by postponing images download unless they are in the viewport. We can use lazysizes to accomplish this.

import "lazysizes";

change src to data-src, srcset to data-srcset, add class="lazyload"

<img  class="lazyload"  data-sizes="auto"  data-srcset=...

This technique is called lazy-loading.

LQIP

if we use lazy-loading and users network is slow or down user will see blank rectangles instead of images, which can be perceived as a broken site. Instead, we can provide low-quality image placeholders - blurry previews of actual content. To do this we can inline base64 encoded small version of original image:

{{ $lqip := $src.Resize $lqipw }}<div class="img" style="background: url(data:image/jpeg;base64,{{ $lqip.Content | base64Encode  }}); background-size: cover">  <svg width="{{ $img.Width }}" height="{{ $img.Height }}" aria-hidden="true"></svg>  <img    class="lazyload"    ...

and a bit of CSS

.img svg,.img img {  margin: 0;  width: 100%;  height: auto;}.img {  position: relative}.img img {  position: absolute;  top:0;  left:0;}

noscript

Another downside of lazy-loading is that images don't work without JS. Let's fix this by providing default img in <noscript>

<noscript>  <img    loading="lazy"    ...

and

.nojs .img .lazyload {  display: none;}

final version

Let's create a shortcode

{{/* get file that matches the filename as specified as src="" in shortcode */}}{{ $src := .Page.Resources.GetMatch (printf "*%s*" (.Get "src")) }}{{/* set image sizes, these are hardcoded for now, x dictates that images are resized to this width */}}{{ $lqipw := default "20x" }}{{ $tinyw := default "500x" }}{{ $smallw := default "800x" }}{{ $mediumw := default "1200x" }}{{ $largew := default "1500x" }}{{/* resize the src image to the given sizes */}}{{ $lqip := $src.Resize $lqipw }}{{ $tiny := $src.Resize $tinyw }}{{ $small := $src.Resize $smallw }}{{ $medium := $src.Resize $mediumw }}{{ $large := $src.Resize $largew }}{{/* only use images smaller than or equal to the src (original) image size, as Hugo will upscale small images */}}{{/* set the sizes attribute to (min-width: 35em) 1200px, 100vw unless overridden in shortcode */}}{{ $img := imageConfig ($src.RelPermalink | printf "content/%s" ) }}<div class="img" style="background: url(data:image/jpeg;base64,{{ $lqip.Content | base64Encode  }}); background-size: cover">  <svg width="{{ $img.Width }}" height="{{ $img.Height }}" aria-hidden="true"></svg>  <img    class="lazyload"    data-sizes="auto"    data-srcset='    {{ if ge $src.Width "500" }}      {{ with $tiny.RelPermalink }}{{.}} 500w{{ end }}    {{ end }}    {{ if ge $src.Width "800" }}      {{ with $small.RelPermalink }}, {{.}} 800w{{ end }}    {{ end }}    {{ if ge $src.Width "1200" }}      {{ with $medium.RelPermalink }}, {{.}} 1200w{{ end }}    {{ end }}    {{ if ge $src.Width "1500" }}      {{ with $large.RelPermalink }}, {{.}} 1500w {{ end }}    {{ end }}'    {{ if .Get $medium }}      data-src="{{ $medium.RelPermalink }}"    {{ else }}      data-src="{{ $src.RelPermalink }}"    {{ end }}    width="{{ $img.Width }}" height="{{ $img.Height }}"    {{ with .Get "alt" }}alt='{{.}}'{{ end }}>  <noscript>    <img      loading="lazy"      {{ with .Get "sizes" }}sizes='{{.}}'{{ else }}{{ end }}      srcset='      {{ if ge $src.Width "500" }}        {{ with $tiny.RelPermalink }}{{.}} 500w{{ end }}      {{ end }}      {{ if ge $src.Width "800" }}        {{ with $small.RelPermalink }}, {{.}} 800w{{ end }}      {{ end }}      {{ if ge $src.Width "1200" }}        {{ with $medium.RelPermalink }}, {{.}} 1200w{{ end }}      {{ end }}      {{ if ge $src.Width "1500" }}        {{ with $large.RelPermalink }}, {{.}} 1500w {{ end }}      {{ end }}'      {{ if .Get $medium }}        src="{{ $medium.RelPermalink }}"      {{ else }}        src="{{ $src.RelPermalink }}"      {{ end }}      width="{{ $img.Width }}" height="{{ $img.Height }}"      {{ with .Get "alt" }}alt='{{.}}'{{ end }}>  </noscript></div>

Usage:

![test](./image.jpg)

we need to write

{{< img src="image.jpg" alt="test" >}}

PS

The sad part is that there is no markdown preprocessor which would allow me to use the traditional syntax for images instead of shortcode .

I can't use shortcodes inside theme files, as the result I copy-pasted code from shortcode to theme files .


Original Link: https://dev.to/stereobooster/responsive-images-for-hugo-dn9

Share this article:    Share on Facebook
View Full Article

Dev To

An online community for sharing and discovering great ideas, having debates, and making friends

More About this Source Visit Dev To