Performance greatly affects the user experience. Gatsby builds fast websites out-of-the-box.
When creating the tool, they noticed that slow websites are slow in different ways, but fast websites are fast in similar ways. So baking optimization approaches in a framework resulted in Gatsby.
From my experience, Gatsby set up webpack to build the most optimal performance. Webpack is a module bundler for JavaScript used by many front-end projects.
data:image/s3,"s3://crabby-images/a9ff4/a9ff4bb4347071778c2372ad55cb081ce921698a" alt="A big community develops Gatsby on GitHub, also contributing to theĀ logo design."
Blending static websites with dynamic apps
Gatsby is a static website generator that uses React. It creates HTML files for each page your website has.
So when building your website, Node.js will mount the React application to create HTML files with the rendered content for each route. This is the core of Gatsby.
Ā
āInstead of waiting to generate pages when requested, Gatsby pre-build pagesā āĀ gatsbyjs.org
Ā
Letās go back in the basics to see why this is important for performance.
When the user accesses a page through an HTML file, the browser renders the content. Without any cache or JavaScript, using an anchor tag will load another HTML file when clicked. As a result, the user might have to wait or worse, see a blank page while rendering the content.
This is the most traditional way that the Web was designed until Single Page Applications (SPA) came up.
SPA renders the page by updating the content with JavaScript. Itās much faster to update than downloading static files. Because they load a single HTML file and dynamically update that page as the user interacts.
React is a library to handle the view layer for SPA. Such frameworks and libraries like React doesnāt know what to render unless some JavaScript code starts running. So building them as SPAs will drastically affect theĀ Critical Rendering Path.
Ā
data:image/s3,"s3://crabby-images/0aeb5/0aeb561a21cbcdd124f7ce82d10afd0617a6078b" alt="Critical Rendering Path stalls the render while loading and executing JavaScript."
Ā
Gatsby has a webpack configuration to provide enough content for the first render:
- HTMLĀ tags
- JavaScriptĀ code set asĀ async, necessary for user interaction but not for the first render
- CSSĀ as inline, so no need to download them
Code split and cache
When building a page, Gatsby can see which components the page needs and let webpack do code splitting automatically. This is applied by setting upĀ Dynamic Imports.
Through this way, the browser will only request files required for the page, not the entire website, speeding up the time to interact with the page.
As a result, links to other pages will download their files only when the user interacts with the link, slowing navigation.
To avoid this problem, the webpack configuration of Gatsby applies a technique calledĀ Link prefetching.
After the browser is finished loading the page, it silently looks for links with prefetch attributes to download them. Then, when a user clicks on a link, the files requested for the page will have high chances to be already in cache.
Every page is a React app
Navigating through pages in a static website still requires a load of HTML files, but not for Gatsby ā they are React apps.
āGatsby generates your siteās HTML pages, but also creates a JavaScript runtime that takes over in the browser once the initial HTML has loadedā ā gatsbyjs.org
Each anchor tag for another page will become a route byĀ Reach RouterĀ (a tool for building routes on React with accessibility). It looks like itās changing from one HTML file to another when in fact, itās a SPA updating the content on the page.
Image optimization
HTTP ArchiveĀ tracks a lot of popular websites, most of the data types requested by pages are images.
Ā
data:image/s3,"s3://crabby-images/f018f/f018fda81dc91eb8f660baabd84196a7307a909a" alt="Total KilobytesĀ ā The sum ofĀ transfer sizeĀ of all resources requested by the page is around 1285.5 KB for mobile."
Ā
data:image/s3,"s3://crabby-images/aecf0/aecf0bbffceb8b6855e61555c9dba3dc47c2295c" alt="Image BytesĀ ā The sum of transfer size of all external images requested by the page is 491.0 KB for mobile."
Ā
Optimizing images can be one of the best performance improvements on a website.
Fewer bytes to download means less bandwidth required, so the browser can download and render content faster. These are some of the optimizations we can do:
- Resize to the same amount of space it needs
- GenerateĀ responsive imagesĀ with different resolutions for desktop and phones
- Remove metadata and apply compression
- Apply lazy loading to speed up the initial page load
- Display a placeholder while the image is loading
This can take a lot of effort and Gatsby has a solution: this whole process can be automated.
Like many tools in Gatsby,Ā gatsbyjs-imageĀ is powered by GraphQL. This plugin sets up the images with different resolutions for download. It creates some thumbnails and applys compression. All this on the building step.
When the image is loading, a āblur-upā technique displays a preview in a very low-quality image that is already in the HTML file (or just the background). All the work is reduced in coding GraphQL queries to create the automated optimization. Check out this demo:
Ā
data:image/s3,"s3://crabby-images/dc748/dc748b5bd3b62ba4d8ca861ba013e419a45cff32" alt="Demo from GatsbyĀ for optimized performance with images. The āblur-upā technique is also used by Medium."
Minification and unique filenames
These techniques are already widely used by popular frameworks and libraries, and in Gatsby there is not much difference.
All files are minifiedĀ by default when building with webpack. Because browsers donāt care about beautiful code, so why not write everything in one line?
The files are unique when built by assigning a hash on the filename. If something changes, a new name is given for the file.
The reason behind this is to allow the server that hosts these files to give a long duration for browser caching.
So when the user comes back for the website, they already have the files. Updates in your files will give a new filename when built. So the browser downloads the file because there will be no match with the one from the cache.
More resources and beyond
Gatsby cares about performance optimization so that you donāt need to.
If you are more curious about how Gatsby works under the hood, check out theĀ documentation.
I also recommend this webinar from the Gatsby team,Ā Behind the Scenes: What makes Gatsby Great.