performance icon indicating copy to clipboard operation
performance copied to clipboard

Optimization Detective: Implement lazy-loading of background images

Open westonruter opened this issue 1 year ago • 3 comments

Feature Description

At the WordCamp Asia Contributor Day, @shilo-ey shared a very interesting thing that Elementor is doing to optimize the loading of background images by lazy-loading them. Background images cannot be lazy-loaded using loading=lazy like regular images can since they are CSS and not img elements. So Elementor is using a clever approach to force all background images to be hidden after the first N with a style rule. Then it uses IntersectionObserver to remove the style rule once scrolling.

https://github.com/elementor/elementor/blob/498d48e8fed6cb8db2b0b3d3835cf1e44efbcbb8/modules/lazyload/module.php#L37-L97

This could be done with Optimization Detective as well, but in a more accurate way since it wouldn't have to assume to not lazy-load the first N images and it can instead use the elements in the URL metrics to know which background images are actually in the initial viewport, and only exclude them from lazy-loading.

See also https://github.com/WordPress/gutenberg/issues/59113

westonruter avatar Mar 07 '24 04:03 westonruter

See also this implementation that @erikyo just shared (see Slack thread): https://gist.github.com/erikyo/8e38a9bdeddfc3fea315399c0107ff09

Nevertheless, we need to be careful not to lazy-load a background image that is in the initial viewport.

westonruter avatar Mar 25 '24 14:03 westonruter

I'd like to provide an update to this thread since my exploration has progressed significantly since that initial gist!

My aim was to:

  • Implement lazy loading for backgrounds, videos, and all other elements requiring lazy loading within the page.
  • Achieve this without the need for any custom or special markup. Hence, approaches such as using CSS backgrounds or data-lazy attributes are not considered viable solutions.
  • Detect the position of elements within the viewport and prevent lazy loading on those items.

You can find the repository here along with the IIFE script (just in case you'd like to test it with your sources 😉).

If you'd like to give it a try, I've prepared some pages for testing with different scenarios:

In my tests, the pages are only slightly slower (by 100ms) compared to native lazyload. However, since native lazyload requires custom markup, auto-lazyload should provide an immediate reduction in page load time without the need to make any changes (and revert changes when you no longer want to use this plugin), especially for platforms like WordPress where there are elements such as covers with backgrounds, videos without posters, etc. This immediate improvement in load time can be significant.

Feel free to explore and test these pages! Let me know if you have any questions or feedback.

erikyo avatar Apr 05 '24 14:04 erikyo

Another metric that i can provide to show how this plugin act on a real website, is the Wordpress.org homepage waterfall. I use the local override to overrideride the html file and inject the autolazyload script at the beginning of the body, the result is 11sec vs 17sec, not bad 🎉

Auto-Lazyload load

Current Wordpress website performance load2

erikyo avatar Apr 05 '24 15:04 erikyo

Hello, i tried your js class and it is working well. I have found an issue with background images added via js, i have this scenario:

<div id="mybgimage">.....</div>

<style>
#mybgimage{
    background-image: url(myimage.jpg);
}

@media only screen and (min-width: 592px){
    background-image: url(myimage-2.jpg);
}

@media only screen and (min-width: 1280px){
    background-image: url(myimage-3.jpg);
}

@media only screen and (min-width: 1536px){
    background-image: url(myimage-4.jpg);
}
<style>

What happens? By checking the Chrome console, all 4 bg images are downloaded in network tab. Is there a way to fix this?

Thanks

rcwd avatar May 14 '24 09:05 rcwd

@rcwd thank you for testing! can you please file an issue here? 👉 the autolazyload repo

erikyo avatar May 14 '24 10:05 erikyo

@erikyo thanks! Issue opened.

rcwd avatar May 14 '24 12:05 rcwd

Hello again @erikyo, is there a way to exclude some images to be lazyloaded?

rcwd avatar May 14 '24 13:05 rcwd

@rcwd please refer to my GH repository, don't post here because unrelated

short answer: at the moment no, but if you kindly can file a new issue I will try to solve it! 😇

erikyo avatar May 14 '24 17:05 erikyo