Boost page rendering performance using <content-visibility> web component
data:image/s3,"s3://crabby-images/04282/042828a9dd55a6c420ad1f2b3215f6ca4260f865" alt=""
August 2020, I came across Intersection Observer
web API on a random tweet then checked out the MDN API document, I was so excited and wanting to integrate it to improve our page rendering performance. Luckily, there is a React Hook version of intersection observer already, react-intersection-observer
, so it took no times for us to integrate. Since that time Intersection Observer
is my favourite Web API 😍 whenever I need to toggle/animate/trigger components.
The same month, Google Dev Submit 2020 unveiled CSS content-visibility
and contain-intrinsic-size
properties which supports from Chrome 85. It’s shamed I was buried by developing project features and only came across during Xmas holiday. However, this is like the best Xmas surprise 🎁 for me and it takes my excitement to the next level, just two lines of CSS then I can achieve lazy loading, OMG I freaking love it. You can deep dive content-visibility on Google web.dev, a great article to explain how it works.
But, life is not always easy, content-visibility
is still more like an experimental feature and not supporting the other two major browser Safari and Firefox. 😞
data:image/s3,"s3://crabby-images/78700/78700b89965f54edf1855ba20804e01ed1c49580" alt=""
Wait a minute… we have Intersection Observer javascript API and content-visibility
CSS solution, why not combine these so we can provide a cross-browsers solution💡 Meanwhile, implement as a web component is the best choice coz it’s lightweight, high performance (shadow DOM) and great browsers compatibility. I spent 12 hours to digest web component and gave a birth of <content-visibility>
v1 🎉
I chose lit-element
base class to create this web component as it’s lightweight and great browser compatibility, especially their web component GitHub starter template is extremely handy makes the whole experience of creating web component zero-brainer.
Install content-visibility
from NPM and start integrating with any JS frameworks (React, Vue, Preact, Angular, VanillaJS and etc).
npm i content-visibility --save
data:image/s3,"s3://crabby-images/3cd86/3cd86d2e6bd3fce538cc8e8addd5363656a1d2fb" alt=""
Alright, let’s see some results I tested on three major browsers, Chrome 87, Safari 14 and Firefox 84 and I used the exact same travel blog template (except I removed iframe
) that is used in web.dev Codepen
Rendering frames, Safari comes with an interesting tool, Rendering Frames Timeline, helps you to measure page rendering performance. I want to start with this result to demonstrate how it reduces unnecessary rendering.
data:image/s3,"s3://crabby-images/42981/42981ee078f62e35eb22723f2a236b3ef295c718" alt=""
data:image/s3,"s3://crabby-images/32680/326800264cc5e8b1f097058e97c82ac3a25efb79" alt=""
data:image/s3,"s3://crabby-images/d4a9b/d4a9bed48f5897098e84640c21c61099f8980b46" alt=""
Time measurement, Chrome 87 uses CSS solution which completely handles by browser and you can see Rendering
and Painting
time are reduced around 50%
and overall time reduced 1s
.
data:image/s3,"s3://crabby-images/ad1bb/ad1bb05a95279b7f0b1d7a88f45bb9ac0f80e1d1" alt=""
data:image/s3,"s3://crabby-images/0960e/0960eeaf5c66fad5fd382112f0dd911a6e607c2a" alt=""
Safari 14 fallbacks to use Intersection Observer
API to detect if component appears on the viewport. Layout & Rendering time down from 15s
to 8s
and you can easily see not many rendering chunks happening after applied <content-visibility>
data:image/s3,"s3://crabby-images/8b831/8b83105c6ea9bb01d6d372134e9df066b0c4d778" alt=""
data:image/s3,"s3://crabby-images/3aa26/3aa26cc10ba9ab46031d8dbf4c4987f5a1a65362" alt=""
Firefox 84 also fallbacks to use Intersection Observer
API.
data:image/s3,"s3://crabby-images/c227b/c227b3dd6d4669d2064abcc287c9830456490cfb" alt=""
data:image/s3,"s3://crabby-images/b8a02/b8a028807ee33dea28cc3a84f52b702d41aa8024" alt=""
data:image/s3,"s3://crabby-images/e1f6e/e1f6ee2239a723bb7b8e9c05d446c9d3329c3f8c" alt=""
data:image/s3,"s3://crabby-images/2b313/2b313897f5de858f03f9a4a680fa5fb030ea5c7b" alt=""
You probably wondering how many lines I changed for 👆 PageSpeed results, actually just 4 lines 👇
data:image/s3,"s3://crabby-images/644d5/644d5e4f39b9fd9a6f725ca1f5c8d88abf543ca1" alt=""
That’s it! Numbers are better than words to tell the story. I already integrated <content-visibility>
web component with my production web application in Preact
. This is just v1 and sure it can be extended and tweak from feedbacks. Give it a try and welcome to fork the source code and deep dive the implementation. I will find a time to write an article about the implementation but not urgent as logic is simple and straightforward.