Solving CLS Issues In A Next.js-Powered E-Commerce Website (Case Study)
Fairprice is one of the largest online grocery stores in Singapore. We are continuously looking out for areas of opportunities to improve the user’s online shopping experience. Performance is one of the core aspects to ensure our users are having a delightful user experience irrespective of their devices or network connection.
There are many key performance indicators (KPI) that measure different points during the lifecycle of the web page (such as TTFB, domInteractive
and onload
), but these metrics don’t reflect how the end-user experiences the page.
We wanted to use a few KPIs which correspond closely to the actual experience of the end-users so we know that if any of those KPIs are not performing well, then it will be directly impacting the end-user experience. We found out user-centric performance metrics to be the perfect fit for this purpose.
There are many user-centric performance metrics to measure different points in a page’s life cycle such as FCP, LCP, FID, CLS, and so on. For this case study, we are mainly going to focus on CLS.
CLS measures the total score of all unexpected layout shifts happening between when the page starts loading and till it is unloaded.
Therefore having a low CLS value for a page ensures there are no random layout shifts causing user frustration. Barry Pollard has written an excellent in-depth article about CLS.
How We Discovered CLS Issue In Our Product Page
We use Lighthouse and WebPagetest as our synthetic testing tools for performance to measure CLS. We also use the web-vitals library to measure CLS for real users. Apart from that, we check the Google Search Console Core Web Vitals Report section to get an idea of any potential CLS issues in any of our pages. While exploring the report section, we found many URLs from the product detail page had more than 0.1 CLS value hinting there is some major layout shift event happening there.
Debugging CLS Issue Using Different Tools
Now that we know that there is a CLS issue on the product detail page, the next step was to identify which element was causing it. At first, we decided to run some tests using synthetic testing tools.
So we ran the lighthouse to check if it could find any element which could be triggering a major layout shift, it reported CLS to .004 which is quite low.
The Lighthouse report page has a diagnostic section. That also did not show any element causing a high CLS value.
Then we ran WebpageTest and decided to check the filmstrip view:
We find this feature very helpful since we can find out which element at which point in time caused the layout to shift. But when we run the test to see if any layout shifts are highlighted, there wasn’t anything contributing to the huge LCS:
The quirk with CLS is that it records individual layout shift scores during the entire lifespan of the page and adds them.
Note: How CLS is measured has been changed since June 2021.
Since Lighthouse and WebpageTest couldn’t detect any element that triggered a major layout shift which means it was happening after the initial page load possibly due to some user action. So we decided to use Web Vitals Google Chrome extension since it can record CLS on a page while the user is interacting with it. After performing different actions we found the layout shift score is getting increased when the user uses the image magnify feature.
I have also created a PR to the original repo so that other developers using this library can get rid of the CLS issue.
The Impact Of The Change
After the code was deployed to production, the CLS was fixed on the product details page and the number of pages impacted with CLS was reduced by 98%:
Since we used transform
, it also helped to make the image magnify a smoother experience to the users.
Note: Paul Irish has written an excellent article on this topic.
Other Key Changes We Made For CLS
There are also some other issues we faced through many pages in our website which contribute to CLS. Let’s go through those elements and components and see how we tried to mitigate layout shifts arising from them.
-
Web-fonts:
We have noticed that late loading of fonts causes user frustrations since the content flashes and it also causes some amount of layout shifts. To minimize this we have done few changes:- We have self-hosted the fonts instead of loading from 3rd party CDN.
- We preload the fonts.
- We use font-display optional.
-
Images:
Missing height or width value in the image causes the element after the image to shift once the image is loaded. This ends up becoming a major contributor to CLS. Since we are using Next.js, we took advantage of the built-in image component callednext/images
. This component incorporates several image-related best practices. It is built on top ofHTML tag and can help to improve LCP and CLS. I highly recommend reading this RFC to find out the key features and advantages of using it.
-
Infinite Scroll:
On our website, product listing pages have infinite scrolling. So initially, when users scroll to the bottom of the page they see a footer for a fraction of seconds before the next set of data is loaded, this causes layout shifts. To solve this we took few steps:- We call the API to load data even before the user reaches the absolute bottom of the list.
- We have reserved enough space for the loading state and we show product skeletons during the loading status. So now when the user scrolls, they don’t see the footer for a fraction of seconds while the products are getting loaded.
Addy Osmani has written a detailed article on this approach which I highly recommend checking.
Key Takeaways
- While Lighthouse and WebpageTest help to discover performance issues happening till page load, they can’t detect performance issues after page load.
- Web Vitals extensions can detect CLS changes triggered by user interactions so if a page has a high CLS value but Lighthouse or WebpageTest reports low CLS then the Web Vitals extension can help to pinpoint the issue.
- Google Search Console data is based on real users’ data so that also can point to potential perf issues happening at any point in the life cycle of a page. Once an issue is detected and fixed, checking the report section again can help verify the effectiveness of the performance fix. The changes are reflected within days in the web vitals report section.
Final Thoughts
While CLS issues are comparatively harder to debug, using a combination of different tools till page load (Lighthouse, WebPageTest) and Web Vitals extension (after page load) can help us pinpoint the issue. It is also one of the metrics which is going through lots of active development to cover a wide range of scenarios and this means that how it is measured is going to be changed in the future. We are following https://web.dev/evolving-cls/ to know about any upcoming changes.
As for us, we are continuously working to improve other Core Web Vitals too. Recently, we have implemented responsive image preload and started serving images in WebP format which helped us to reduce 75% of image payload, reduce LCP by 62%, and Speed Index by 24%. You can read more details of optimization for improving LCP and Speed Index or follow our engineering blog to know about other exciting work we are doing.
We would like to thank Alex Castle for helping us debug the CLS issue on the product page and solve the quirks in the next/images
implementation.