MOST IMPORTANT: Improve page load speed by prioritizing which resources get loaded, controlling the order in which they are loaded, and reducing the file sizes of those resources
General
- Use less nodes in the DOM tree and simplify the HTML
- Use a mobile-first approach to ensure that the default layout is for small-screen devices, so mobile devices can just download images suitable for their screens and don't need to take the performance hit of downloading larger desktop images
- Use HTTP/2 on your server (or CDN)
Resources
- Use a CDN for resources which can reduce load times significantly
- Lazy load parts of your application outside the viewport (below the fold), but have a fallback for SEO
- Compress your resources, and use CSS animation or SVG if possible. Images and video account for over 70% of the bytes downloaded for the average website
- Use speculative loading resource hints such as “rel=preconnect", “rel=dns-prefetch", “rel=prefetch", and “rel=preload". Preload important assets such as CSS, JavaScript, and even videos that are essential for the page. The preload <link> fetches critical resources immediately without blocking rendering.
- Provide different image resolutions via srcset and other sources for images and videos with picture element
Perceived Performance
- To maximize the perceived and actual performance, the HTML should be loaded first in the order in which it appears on the page. For example, you can preload critical CSS and fonts early but defer non-critical JavaScript until later.
- Add inline CSS required for any content rendered above the fold or what you see in the browser viewport before scrolling. These styles will improve perceived performance as the CSS does not require a file request
Examples:
<!-- Preload a JavaScript file -->
<link rel="preload" href="important-js.js" as="script" />
<!-- Preload a JavaScript module -->
<link rel="modulepreload" href="important-module.js" />
General
- Cut down on image HTTP requests with CSS sprites
- Preload important assets
- Use container queries to enable querying a container's size, properties, and property values to apply CSS styles conditionally
Device Downloading Time
- Remove unnecessary styles
- Split CSS into separate modules not required at page load can be loaded later on
- Use media queries on link tags
- https://developer.mozilla.org/en-US/docs/Web/CSS/CSS_media_queries/Using_media_queries
- Minify and compress your CSS
- Simplify selectors and don't apply styles to more elements than needed
Fonts
- Use web-safe fonts
- Preload web fonts
- Use the font-display descriptor to allow text to appear with a fallback font while a font loads
- https://developer.mozilla.org/en-US/docs/Web/CSS/@font-face/font-display
- Use compressed web fonts or use WOFF and WOFF2 since they have compression built-in
- Within @font-face, use font-display: swap. By using font display swap, the browser will not block rendering and will use the backup system fonts that are defined
- If possible, avoid icon web fonts and use compressed SVGs. To further optimize inline your SVG data within HTML markup to avoid HTTP requests
- Batch updates and avoid animating box model properties
- Reduce animation frame duration
You don't always need a framework: Minimal JavaScript is nice.
Device Downloading Time
- Keep your JavaScript bundles small, especially for mobile devices. Small bundles improve download speeds, lower memory usage, and reduce CPU costs. Avoid having a single large bundle; if it exceeds ~50–100 kB, split it into separate smaller bundles. On mobile, you’ll want to ship much less significantly because of network speeds but also to keep plain memory usage low
- Minify and compress files or use a module bundler like web pack
- Remove as much of unused code as possible, and only use as much JavaScript as needed for the current page
Execution Time
- Break down long tasks (longer than 50ms) to avoid keeping the main thread busy and can push out how soon pages are interactive. Post-download script execution time is now a dominant cost. The main thread can run only sequentially, executing one task simultaneously. If the user attempts to interact with the page or a vital UI update is requested while a long task is running, the expected response or visual update will be delayed, resulting in the UI appearing sluggish or unresponsive
- Avoid large inline scripts: if the code is over 1kb, don’t inline it
- Use web workers to keep the main thread from performing essential tasks like UI rendering. All pure startup calculations should be performed in background threads while you keep the run-time of main thread events as short as possible
Asynchronous Rendering
- Split your JavaScript into multiple files representing critical and non-critical parts
- Defer execution of non-critical JavaScript using defer or async attributes or using an “onload” event listener
- Link JavaScript assets after the page's DOM elements
Coding Practices
- Reduce DOM manipulation and batch DOM changes rather than just firing off each change as it occurs
- Consider a more straightforward, less intensive solution when creating code
- Consider using built-in browser features rather than trying to create them using JavaScript
- Use less or cut out any non-essential animations. Try to use CSS animations where possible instead of JavaScript animations
- Remove event listeners that are no longer needed and use event delegation wherever possible. When you have some code to run in response to a user interacting with any one of a large number of child elements, you can set an event listener on their parent. Events fired on any child element will bubble up to their parent, so you don't need to set the event listener on each child individually. Less event listeners to keep track of means better performance
- Reduce the amount of looped code, and avoid running the entire loop when it is unnecessary using a break. Keep as much unnecessary work outside of the loop to prevent performing single tasks multiple times
- Break your large code into separate functions, and use a "yield" function periodically to get the code to yield to the main thread. This means that our code is split into multiple tasks, between the execution of which the browser is allowed to handle high-priority tasks such as updating the U.