Websites are viewed on various devices with widely differing screen sizes and connection speeds. When serving images it's best not to use the same image for all devices. A large image might be fine on a desktop computer with a fast internet connection, but on a small mobile device with limited bandwidth, the download might take a long time.
The media library has support for generating the necessary files and html markup for responsive images. In addition the medialibrary also has support for progressive image loading.
Try a demo of the concept here.
##Introducing the srcset attribute
The most common way to display a picture is by using an img
element with a src
attribute.
<img src="my-image.jpg">
Using this markup the browser will always display my-image.jpg
regardless of screen size.
As described in the HTML specification you can also use a srcset
attribute to indicate different versions of your image and their respective width.
<img src="large.jpg"
srcset="large.jpg 2400w, medium.jpg 1200w, small.jpg 600w">
##A pragmatic sizes approach
The srcset
attribute is commonly accompanied by the sizes
attribute to tell the browser beforehand how a picture will be rendered for different breakpoints.
<img src="large.jpg"
srcset="large.jpg 2400w, medium.jpg 1200w, small.jpg 600w"
sizes="(min-width: 1200px) 50vw,
100vw">
When using srcset
and sizes
, the browser will automatically figure out which image is best to use.
Say your browser is 1200 pixels wide, the sizes
attribute demands for an image half the size (50vw = 600px). You'll end up with the small.jpg
version. But presume you have a retina screen with pixel density 2, the browser knows we'd need the medium.jpg
to render the image crisply.
The sizes
attribute requires a lot of work though: you'd need to co-ordinate between your responsive CSS and the output HTML to set the right sizes
for every image and every layout scenario.
If you leave out the sizes
attribute, the browser will presume that the image will be rendered full width, which is also not optimal in many cases — eg. thumbnails.
Now the media library takes a pragmatic approach in this, so you don't ever have to think about the sizes
attribute and can experiment freely with different page layouts: we set the sizes
initially to 1px
to load the smallest picture first, and after load adjust sizes
to the rendered width of the image with JavaScript.
We set this width in a vw
value: if you make your browser wider, an even better version will be loaded.
At the same time we can use this technique to use this smallest picture as the placeholder.
##Progressive image loading
When visiting a Medium blog you might have noticed (on a slower connection) that before a full image is displayed a blurred version of the image is shown. The blurred image is replace by a high res one as soon as that big version has been downloaded. The blurred image is actually a very tiny image that's being sized up.
The advantage of displaying a blurred version is that a visitor has a hint of what is going to be displayed very early on and that your page layout is ready right away.
The media library comes with support for progressive image loading out of the box. The tiny blurred image will automatically be generated whenever you leverage responsive images. By replacing the sizes
attribute on load with JavaScript, the tiny placeholder will be swapped with a bigger version as soon as it is downloaded.
This placeholder is base64-encoded as SVG inside the srcset
attribute, so it is available in the initial response right away without extra network request.
The SVG has the exact same ratio as the original image, so the layout should not flicker during the swap.
If you want to leverage responsive images but don't want the progressive image loading, you can set the responsive_images.use_tiny_placeholders
key in the media-library
config file to false
.
##Generating the necessary images
In the srcset
attribute of img
various image files can be specified. The media library can automatically generate those images for you.
When adding an images to the media library simply use the withResponsiveImages
function.
$yourModel
->addMedia($yourImageFile)
->withResponsiveImages()
->withResponsiveImagesIf($condition)
->toMediaCollection();
Behind the scenes, the media library will generate multiple size variations of your image. To learn which variations are generated and how to customize head over here.
##Displaying responsive images
To display a responsive image simply output a Media
object in a blade view.
{- in a Blade view -}
<h1>My responsive images</h1>
{{ $yourModel->getFirstMedia() }}
Per image attached to your model the resulting html will look more or less like this:
<img srcset="/media/1/responsive-images/testimage___media_library_original_188_105.png 188w, /media/1/responsive-images/testimage___media_library_original_158_88.png 158w, /media/1/responsive-images/testimage___media_library_original_132_74.png 132w, /media/1/responsive-images/testimage___media_library_original_110_61.png 110w,  32w" onload="if(this.dataset.sized===undefined){this.sizes=Math.ceil(this.getBoundingClientRect().width/window.innerWidth*100)+'vw';this.dataset.sized=''}" sizes="1px" src="/media/1/testimage.png" data-sized="">
##Generating responsive images for conversions
You can also generate responsive images for any image conversions you define. Simply use withResponsiveImages
when defining a conversion.
Here's an example where we define a conversion to make a greyscale version and generate responsive, greyscaled images.
namespace App;
use Spatie\MediaLibrary\HasMedia;
use Spatie\MediaLibrary\MediaCollections\Models\Media;
use Illuminate\Database\Eloquent\Model;
use Spatie\MediaLibrary\InteractsWithMedia;
class YourModel extends Model implements HasMedia
{
use InteractsWithMedia;
public function registerMediaConversions(Media $media = null): void
{
$this
->addMediaConversion('my-conversion')
->greyscale()
->quality(80)
->withResponsiveImages();
}
}
To generate the converted greyscale file with its quality slightly reduced, and the responsive images, simply add a file:
$yourModel->addMedia($yourImage)->toMediaCollection('my-conversion');
The conversion's quality
setting will also be applied to its generated responsive images.
In a controller you can pass a media object to a view.
public function index()
{
$media = $yourModel->getFirstMedia();
return view('my-view', compact('media'));
}
To generate the img
tag with the filled in srcset
simply use $media
as an invokable. Use the name of your conversion as the first parameter.
<h1>My greyscale responsive image</h1>
{{ $media('my-conversion') }}