In my post Bringing it all together I wrote a section called Beyond Jekyll in which I mention an opportunity to move on from Jekyll.

I lost confidence in Jekyll recently. After fixing a low severity vulnerability in one of Jekyll's Ruby Gem dependencies, my build broke.

I considered Hugo, but kits-dna is now powered by Eleventy 🚀

# Eleventy

Eleventy is a simpler static site generator - 11ty.dev

Eleventy transforms template markup / languages such as Markdown, Liquid and Nunjucks into HTML.

Eleventy is written in JavaScript which played a part in my decision. No underlying Ruby like in Jekyll or Go in Hugo. I don't know JavaScript but I'm more likely to learn JavaScript in the future if I want go further with web development.

# Why Eleventy

I first heard about Eleventy at work. Eleventy is powering a new docs-as-code tool and I wanted to learn more. Over Christmas I setup a test-eleventy project and started discovering and experimenting using concepts from kits-dna along with the simple but effective eleventy-base-blog.

Eleventy is a beautiful blank canvas, it doesn't push you towards a theme or template. You can easily clone a project from GitHub if you want a template to help get started. For me this meant I could de-couple from Jekyll's Minima theme that I had customised.

Moving from Jekyll, layouts and includes work the same and Eleventy even defaults to using Liquid for templating and the same output directory _site as Jekyll.

The flip side of Eleventy. If you're just starting out or a non-techy like me, it's a steeper learning curve to configure. If you're happy with using a theme or template, you might be better off with Jekyll, Hugo or another static site generator. If like me you wanted to build a website to learn about web development, then it's a great tool which will give you more freedom.

💡 Tip: Learn Eleventy has useful lessons to help you learn Eleventy.

# Pros

  • Regular releases
  • No theme constraints
  • Hot loading local server
  • Folder structure
  • Lots of documentation
  • Extensibility
  • npm / node project
  • Easy serve and build scripts
  • Data files
  • Alerts to dependency deprecations

# Cons

  • JavaScript makes the configuration file tricky to learn compared to yaml
  • Sass needs specially configured
  • Requires JavaScript filters
  • GitHub Pages deployments work differently

# Migrating

I'd recommend setting up a test project before touching your site and blog. This allowed me to understand some of the differences that I'll cover below and get a high level of confidence for my migration. For example Eleventy defaults to Liquid (albeit a different flavour) for templating but Nunjucks is more common for Eleventy projects including the eleventy-base-blog, so I discovered Nunjucks while testing.

To migrate I created a new eleventy branch in my repository. I then deleted the Jekyll dependencies and initialised npm followed by installing Eleventy and the dependencies I used on the test project. After that I updated the directory or folder structure to include a src or source directory. The src directory now contains all site content including:

  • _data
  • _includes
  • _layouts
  • _sass
  • _scripts
  • assets
  • pages
  • posts

This has nicely streamlined the repository root and keeps another Jekyll convention in prefixing directories with an _ to exclude from copying into the output.

The output directory when Eleventy builds the site is now dist.

Below I'll dive into specific areas of difference.

# Dependencies

Eleventy depends on node.js. Install locally by picking an option from nodejs.org. Installing node.js will also install npm (node package manager).

The npm public registry contains millions of JavaScript code packages, including Eleventy. Install Eleventy:

npm i @11ty/eleventy

You can install other dependencies in the same way. If you clone an Eleventy template site, all you need to do to install the required dependencies is run:

npm install

# Nunjucks

Nunjucks is a templating language designed for JavaScript and is commonly used in Eleventy projects. I didn't commit to learning Liquid in depth with Jekyll and the two are similar, so it isn't a steep learning curve. Since I've adopted elements of the eleventy-base-blog it was easier to adopt Nunjucks across kits-dna.

You'll need to rename HTML files in your _includes and _layouts directories to the .njk extension. A couple of things to watch out for include ensuring content has the safe tag and for includes the file name is wrapped in double quotes.

# Sass

Sass is compatible with Eleventy and can be installed as a dependency. Sass needs built and watched for changes but you can do this in your npm scripts:

"watch:sass": "sass --watch src/_sass/styles.scss:dist/css/styles.css",
"build:sass": "sass src/_sass/styles.scss dist/css/styles.css",
"watch:eleventy": "npx @11ty/eleventy --serve",
"build:eleventy": "npx eleventy",
"start": "npm-run-all build:sass --parallel watch:*",
"build": "npm-run-all build:*",

Now when running npm start your local server will watch for style changes and sync your browser without needing to refresh.

# Includes

Includes remain in the _includes directory. I've kept includes to what are also known as partials. Some Eleventy sites place layouts in the _includes directory. It doesn't really matter, my personal preference is it's cleaner and quicker to find if in their own directory.

# Layouts

Layouts remain in the _layouts directory. The files were renamed with the .njk extension and updated to Nunjucks syntax / rules.

If you split your includes and layouts as I have then you'll need to update your eleventy.config.js file so that Eleventy can find your layouts.

dir: {
		input: "src",          // default: "."
		includes: "_includes",  // default: "_includes" (`input` relative)
		layouts: "_layouts",    // default: "_layouts" (`input` relative)
		data: "_data",          // default: "_data" (`input` relative)
		output: "dist",        // default: "_site"
	},

Layouts chain just like in Jekyll, but in Eleventy you can take advantage of directory data files to apply the layout across all files in the directory. For example, my site blog posts are in src/posts, add a file called posts.json and you could add the following code:

{
	"layout": "post.njk",
    "tags": "posts",
}

Now each post in the posts directory will use the post.njk layout rather than needing to define this in each posts front matter. You can still use front matter, and Eleventy's data cascade will prioritise front matter allowing you to overwrite the data file and use a different layout if required. This code also gives each post the posts tag, which will include it in the blog collection for displaying a list a blog posts.

# JavaScript filters

To replicate some Jekyll functionality in Eleventy we need to add some JavaScript filters. I currently have filters to:

  • Output a more conventional date
  • Find and sort tags
  • Count words to calculate reading time

The filters themselves are short scripts that I've stored in the _scripts directory. For your filters to work you need to import and add the filter to your eleventy.config.js file before calling it in your Nunjucks template.

# Environments

To set a production environment variable so that select features will only work in production such as analytics, you need to do a couple of extra steps compared to Jekyll.

  1. Create a data file in your _data directory and add the following code
export default function () {
	return {
		environment: process.env.ELEVENTY_ENVIRONMENT || "development",
	};
}
  1. In your base.njk layout (or layout to apply production variable) add:

Example of updating the environment variable in the base.njk file

You'll notice that this code is similar to what you used in Jekyll, only rather than calling Jekyll we're calling the kitsDna.js data file added in step 1.

  1. Update your package.json build script:
"build": "ELEVENTY_ENVIRONMENT=production npm-run-all build:*",

When building your site or deploying using the npm run build command, your analytics are included.

📝 Note: If you just run npm start for local development your analytics are excluded.

# Gotchas

# Terminal errors

Eleventy can throws errors at your terminal when trying to serve or build the site which will prevent your site from building. When first running npm start I had a host of Sass errors and template errors to work through before I could view my site locally.
It's frustrating but when you get the errors cleared you get your shiny new site, hang with it, it's worth it 😄

# CSS stylesheet

Once I cleared my errors and my site built... I had no style. The stylesheet path for Jekyll was /assets/css/styles.css. Eleventy outputs style to /css/styles.css. Once I got the path right in my head.njk file I had my style back.

# Sass

Unlike Jekyll, when you serve and build Eleventy, Eleventy warns you about deprecations which is nice, but it led to me having to clean up and improve the use of variables, mixins and calculations in my scss sheets.

You don't need to prepend links in Eleventy as you did in Jekyll, meaning you'll need to remove any | relative_url references. In Eleventy just use /file_name. Eleventy outputs files without a directory, meaning that every page and post has the path dist/.

# Post excepts

Eleventy doesn't support post excerpts like Jekyll does. You can work around this by using a description in your post front matter and adding the relevant code to your post.njk layout. This post is using it! Cleverer people than me might be able to do something better 😄

# New features

Migrating to Eleventy has allowed me to add the following new features to kits-dna:

  • Image transformation
  • Blog post tags
  • Pages for tags

The tags are nice to haves, but the image transformation has improved the site performance by converting jpeg and png images to modern formats such as webp and outputting multiple images sizes for enhanced responsive design. Make sure you add a height: auto property to images in your base.scss sheet. Images transformed by the nifty @11ty/eleventy-img plugin.

# Deploying with Netlify

I had an open issue to discover deploying kits-dna using a tool other than GitHub Pages. Migrating to Eleventy brought this into focus because while you can deploy an Eleventy site to GitHub Pages, it's not as graceful as Jekyll. GitHub Actions are available but it results in a gh-pages deployment branch that GitHub Pages uses to deploy the site, and I've avoided that to date. Time to discover an alternative 🔭

I looked at two providers Azure Static Web Apps and Netlify as simple, free alternatives. I decided to test out Netlify using my test-elventy project as it's a more commonly used platform and agnostic from Microsoft.

# Getting started with Netlify

To deploy your site with Netlify (assuming your sites code is hosted on GitHub) simply go to Netlify Apps and login with GitHub. Select which repository you want to deploy, give the site a name (determines the URL subdomain e.g. test-eleventy.netlify.app) and configure the build settings:

  • Branch to deploy = main
  • Build command = npm run build
  • Publish directory = dist
  • Deploy 🚀

Ta da ! Magic! No need for a GitHub Action workflow, Netlify builds and deploys the site for you 😄

📝 Note: I still have GitHub Action workflows for lighthouse, markdownlint and vale. These workflows are allowed to fail and used for review purposes before merging to the main branch. I might write a post on vale in the future!

# Wrap up

I've added relevant updates to my previous blog posts where relevant, the lighthouse and search posts in particular came in handy to help reconfigure lighthouse and install pagefind again 🤝

😮‍💨 that was a long post. I've covered a lot and could have dived even deeper into each section, but hopefully this gives you the confidence to migrate your Jekyll site or create your first site with Eleventy. You learn so much by doing, crack on and get started, don't worry about making mistakes, you're learning 😄

Thanks for reading 🤙