I'm using NextJS and i'm pretty new on that. I'm trying to increase my ranking on Google Pagespeed, and i already did some good progress on that.
As you guys can see on the screenshot, the only bad metric is the "Total Blocking Time":
If you guys want to try the page speed, thats the link: Google PageSpeed
Right now i'm running out of options on how to make that one better, i'm alredy dynamically importing my ponents, removed unused JS, i'm using the NextJs best practices.
I'll really appreciate any help that you guys could have
Thanks in advance
I'm using NextJS and i'm pretty new on that. I'm trying to increase my ranking on Google Pagespeed, and i already did some good progress on that.
As you guys can see on the screenshot, the only bad metric is the "Total Blocking Time":
If you guys want to try the page speed, thats the link: Google PageSpeed
Right now i'm running out of options on how to make that one better, i'm alredy dynamically importing my ponents, removed unused JS, i'm using the NextJs best practices.
I'll really appreciate any help that you guys could have
Thanks in advance
Share Improve this question edited Jun 28, 2022 at 12:08 GuiPab asked Jun 27, 2022 at 14:44 GuiPabGuiPab 4753 gold badges7 silver badges18 bronze badges 6- 1 At the first look i see that you have some google maps javascript that you can load when it's need it. I don't see any google maps on visible part so may be load that when it's need it. You can read more about TBT here – angel.bonev Commented Jun 28, 2022 at 12:26
- I'm using the google maps api at the "Search By Location" input (which is a Google Place Autoplete). But i'm already dynamically loading that ponent, so i think that's not suppose to be the problem – GuiPab Commented Jun 28, 2022 at 12:49
-
3
It's . can you try loading it on focus an try again. It's loaded between
DOMcontent Loaded
andLoad
. Everithing that can be loaded later must be loaded later. Everithing that is not on the visible part MUST be loaded later. May be when the element is visible or something. If you want to lower you TBT , everything that can be delayed must be delayed – angel.bonev Commented Jun 28, 2022 at 13:02 - 1 Thats an interesting approach (just load the maps api when the user clicks on the input), I going to try implement that, and then i'll be back to tell the results. Thanks @angel.bonev – GuiPab Commented Jun 28, 2022 at 13:35
-
2
may be use
IntersectionObserver
and load some css and javascript when they are need it ? I've posted an answer. In my case that was enough to hit 100%. But make sure you can't optimize you core more. This is not the holy grail. You need to provide best experience for your user, not to sutisfive google page speed test. Cheers and good luck – angel.bonev Commented Jun 30, 2022 at 8:57
2 Answers
Reset to default 8So lets start from what is TBT as the docs says
The Total Blocking Time (TBT) metric measures the total amount of time between First Contentful Paint (FCP) and Time to Interactive (TTI) where the main thread was blocked for long enough to prevent input responsiveness.
How to improve your TBT score You need to start with a Lighthouse performance audit
So if you find some action that aren't necessary you need to delay it. Most mon are :
- Unnecessary JavaScript loading, parsing, or execution. While analyzing your code in the Performance panel you might discover that the main thread is doing work that isn't really necessary to load the page.
- Inefficient JavaScript statements. For example, after analyzing your code in the Performance panel, suppose you see a call to document.querySelectorAll('a') that returns 2000 nodes. Refactoring your code to use a more specific selector that only returns 10 nodes should improve your TBT score.
For example you have some element in the footer or the hole footer it self. You can split your javascript
and css
and loaded that part dynamically.
You can use IntersectionObserver , the support for that is good enough, but it's allways a good practice to ensure if not supported:
Example code (in vanilla javascript):
let IOObjects = [
{"idToObserve": "myFooter", functionToTrigger: "someFinction"}
];
let someFinction = () =>{
//load some events, another js or another css
}
IOObjects.forEach(function (e) {
if (document.getElementById(e.idToObserve)) {
if (!window.IntersectionObserver) {
window[e.functionToTrigger]();//Trigger the function if not supported
} else {
let observer = new IntersectionObserver(function (entries) {
entries.forEach(function (entry) {
if (entry.intersectionRatio > 0) {
observer.unobserve(entry.target);
window[e.functionToTrigger]();
}
});
}, {rootMargin: '50px 0px', threshold: 0.01});
observer.observe(document.getElementById(e.idToObserve));//Observe the element if it's visible
}
}
});
You can read more about rootMargin
Actually, dynamically render everything is an bad practice, hence more bad performance. You will want to dynamic import ponent that doesnt include in the first build time, for example, a footer or chatbox.