I recently got Harry Robert’s course on CSS performance (you totally should to, it’s a goldmine of information) and worked on improving performance for this site. I quickly spotted 2 performance bottlenecks: requesting the stylesheet, and requesting the main script.
I had about 4.7Kb of CSS, and less than 1Kb of JavaScript, so I figured the HTTP requests weren’t that necessary at all and I could inject styles and scripts directly within the page to avoid HTTP roundtrips. Inlining CSS and inlining JavaScript is explained in the 11ty docs, so not really warrant of a blog post I head you say.
Now the thing is not all styles are necessary on all pages. For instance, the home page have some components that do not exist anywhere else on the site, and an article page like this one has a lot of styles which are not needed anywhere else (code snippets, figures, tables, post date…). So instead of inlining 5Kb of CSS in the head, most of which would not be needed, I decided to split it across pages.
My CSS (formerly authored in Sass) is split by concern, somewhat following the 7-1 pattern (my JavaScript also follows a similar structure but I’m going to drop it from now on for sake of simplicity). That’s good because that mean I didn’t really have to figure out how to break it down—I only needed a way to include specific parts in specific contexts. Namely:
- Including the core styles (such as layout & typography) in every page.
- Including page-specific styles (blog post, home page, resume…) on specific pages.
Implementation
The implementation concept is relatively simple: in the <head>
of the document, include all core styles in a <style>
tag. And in specific layouts and pages, include specific stylesheets within a <style>
tag as well. No more <link rel="stylesheet">
and no more monolithic stylesheet with the entire site’s styles.
Now, including files can be done with the {% include %}
tag. From 11ty ≥0.9.0, it is possible to include relative paths so files do not have to live in the _includes
folder. That means we can keep a project structure like this (irrelevant parts omitted):
├── _includes/
└── assets/
├── css/
│ ├── base/
│ ├── components/
│ ├── layouts/
│ └── pages/
└── js/
Now, I wanted to minimise the amount of boilerplate needed to include some specific styles or script in a template, and making it easy to maintain. For instance, in my post.liquid
layout, I wanted to have this include at the top:
___LIQUID0___
So I came up with this small _includes/styles.html
Liquid partial:
___LIQUID0___
___LIQUID1___
___LIQUID2___
___LIQUID3___
___LIQUID4___
___LIQUID5___
___LIQUID6___
<style>___LIQUID7___</style>
___LIQUID8___
Alright, so there is quite a lot to unpack here. Here is the breakdown:
- In case the
paths
argument was not provided, we do nothing. - We reassign
paths
from a string to an array by splitting it on commas. - We open a capture group, which is basically a block-level variable assignment.
- We loop over every given path.
- For every path, we import it from the
assets
folder while making sure to trim it withstrip
. This is what allows us to have thepaths
argument authored across multiple lines for clarity. - We close our capture group after having imported the last path, yielding a
css
variable containing all our relevant styles. - We render our styles within a
<style>
tag.
The script.html
partial works exactly the same way except it looks into assets/js
and renders a <script>
tag. I guess both partials could be abstracted into a single one, but I don’t think it’s particularly necessary.
Minification
When it comes to minification, there are a few approaches here. One way would be to have a cssmin
filter based on clean-css (or any other CSS minifier). Inside of the styles.html
partial, we’d apply | cssmin
to our CSS so it gets optimised.
I went a slightly different path and have an 11ty transform to minify HTML with html-minifier. The nice thing about it is that it offers a minifyCSS
and a minifyJS
option to compress styles and scripts authored in <style>
and <script>
tags respectively. Therefore I have a single transform to minify everything.
I decided to run that transform only in production because a) I don’t like to have compressed styles and scripts in development since it can make them harder to debug and b) minification is actually not cheat and can take a few seconds on a site as small as mine which means it would dramatically slow down compilation.
module.exports = function (config) {
if (process.env.NODE_ENV === 'production') {
config.addTransform('htmlmin', (content, path) =>
path.endsWith('.html')
? htmlmin.minify(content, { minifyCSS: true, minifyJS: true, })
: content
)
}
}
That’s about it, really. To sum up: no more HTTP requests for my styles and scripts, which improves performance by reducing the amount HTTP roundtrips. Of course, we no longer benefit from caching, but I believe the performance gain is worth it.
I hope this help! ✨