From Jekyll to Eleventy
Jekyll is still a static site generator I like, use and follow. Nevertheless, when I finally had the time to update this site, I chose to go with Eleventy.
Choosing a static site generator
Static site generators have gained a lot of traction lately, thanks to the pervasiveness of API, Git based workflows, powerful javascript frameworks, headless CMS and unified data layers powered by GraphQL. They have become a sensible choice for all kinds of websites.
Webstoemp was previously running on Jekyll, which I liked because of its ease of use and flexibility. However, it was a bit slow compared to other options and forced me to keep a Ruby environment up to date. I experimented with several other contenders and eventually ended up with Hugo and Eleventy in my shortlist.
Hugo
Hugo being written in Go, it's blazing fast. It is also a binary so you don't have to maintain a Go environment. I actually built the entire V2 of Webstoemp with Hugo as an experiment and really liked it.
Its only downsides for me were the go templating language, which needs a bit of getting used to, and the fact that Hugo is an all encompassing solution, which can be extremely useful for large scale projects but didn't quite appeal to the tinkerer in me.
Eleventy
Which brings me to Eleventy. Being written in Node, it gives you access to the NPM ecosystem to extend it, is easy to use, and is quite a lot faster than Jekyll (albeit not being as fast as Hugo).
Eleventy also works with several template languages, which was an added bonus.
Setting things up
I chose Nunjucks by Mozilla as my main templating engine and only needed markdown and html files apart from that.
Nunjucks lacks date and limit filters so I just went ahead and added them in eleventy.js
. I used the well known moment.js
library for the date filter.
const moment = require("moment");
// limit filter
eleventyConfig.addNunjucksFilter("limit", function (array, limit) {
return array.slice(0, limit);
});
// date filter
eleventyConfig.addNunjucksFilter("date", function (date, format) {
return moment(date).format(format);
});
Since I am using Gulp as a build tool, I had to tell Eleventy to ignore my assets. I just added the following line to my .eleventyignore
file at the root of the project:
src/assets/**/*
The next step was to add Eleventy to gulpfile.js
using Node's child_process
. That way, I can easily run it as a part of my build
and watch
commands.
// packages
const cp = require("child_process");
// Eleventy
function build() {
return cp.spawn("npx", ["eleventy", "--quiet"], { stdio: "inherit" });
}
Data structure
Collections
For this simple website, I only needed two collections (blogposts and projects) for which I created two folders full of markdown files with YAML front-matters.
Eleventy has a neat little feature allowing you to use JSON files with the same name as your directory to declare common front matter values for all files in that directory. For example, I use a blogposts.json
file in my blogposts
directory to specify a layout and a permalink structure for all blogposts.
{
"layout": "layouts/blogpost.njk",
"permalink": "blog/{{ page.fileSlug }}/index.html"
}
Projects do not need their own detail pages, so I use a projects.json
file in my projects
directory to set their permalink
to false
and don't specify a layout
.
{
"permalink": false
}
To create both collections and allow Eleventy to process them, I used the getFilteredByGlob( glob )
method.
const moment = require("moment");
module.exports = function (eleventyConfig) {
// blogpost collection
eleventyConfig.addCollection("blogposts", function (collection) {
return collection.getFilteredByGlob("./src/blogposts/*.md");
});
// projects collection
eleventyConfig.addCollection("projects", function (collection) {
return collection.getFilteredByGlob("./src/projects/*.md");
});
// limit filter
eleventyConfig.addNunjucksFilter("limit", function (array, limit) {
return array.slice(0, limit);
});
// date filter
eleventyConfig.addNunjucksFilter("date", function (date, format) {
return moment(date).format(format);
});
// Base config
return {
dir: {
input: "src",
output: "dist",
},
};
};
Data files
Eleventy lets you easily work with JSON or JS data files located in a ./src/_data
folder by default.
For example, I use a ./src/_data/site.js
file to define site-wide variables that I can access easily access in any template by using the name of the file and one of the object keys.
module.exports = {
title: "Webstoemp",
description:
"Webstoemp is the portfolio and blog of Jérôme Coupé, a designer and front-end developer from Brussels, Belgium.",
url: "https://www.webstoemp.com",
baseUrl: "/",
author: "Jerôme Coupé",
authorTwitter: "@jeromecoupe",
buildTime: new Date(),
};
site.buildTime
key can then be used in the footer of the website:
<div class="c-sitefooter__copyright">
<p class="u-mb-none u-mi-none">© Webstoemp {{ site.buildTime | date("Y") }}</p>
</div>
Templating with Nunjucks
I chose Nunjucks mainly because of its template inheritance mechanism. I can define a base template and then extend it with other templates if needs be.
Blogposts
Looping through the blogposts collection to display titles and publication dates is quite simple.
{% for post in collections.blogposts | reverse %}
{% if loop.first %}<ul class="c-list-ui">{% endif %}
<li>
<article class="c-blogteaser">
<p class="c-blogteaser__date"><time datetime="{{ post.date | date('Y-M-DD') }}">{{ post.date|date("MMMM Do, Y") }}</time></p>
<h2 class="c-blogteaser__title"><a href="{{ post.url }}">{{ post.data.title }}</a></h2>
</article>
</li>
{% if loop.last %}</ul>{% endif %}
{% else %}
<p>No blogpost found</p>
{% endfor %}
As you have seen above, I use a dedicated template to display the detail of each blogpost. _includes/layouts/blogpost.nkj
calls my main layout and adds the blogpost image and the content of the markdown file to the content
block.
{% extends "layouts/base.njk" %}
{% set activeSection = "blog" %}
{% set metaTitle = title %}
{% set metaDescription = excerpt %}
{% set metaImage = site.url ~ "/img/blogposts/_600x600/" ~ image %}
{% block content %}
<article class="c-blogpost">
<div class="c-blogpost__media">
<div class="l-container">
<picture>
<source media="(min-width: 500px)"
srcset="/img/blogposts/_1024x576/{{ image }} 1024w,
/img/blogposts/{{ image }} 1500w"
sizes="(min-width: 1140px) 1140px,
100vw">
<img src="/img/blogposts/_600x600/{{ image }}"
class="o-fluidimage"
alt="{{ imageAlt }}">
</picture>
</div>
</div>
<div class="c-blogpost__body l-container l-container--narrow">
<header>
<p class="c-suptitle c-suptitle--dark"><time datetime="{{ page.date | date('Y-M-DD') }}">{{ page.date | date("MMMM Do, YYYY") }}</time></p>
<h1 class="c-h1">{{ title }}</h1>
<div class="c-blogpost__intro">
<p>{{ excerpt }}</p>
</div>
</header>
<div class="c-wysiwyg">
{{ content | safe }}
</div>
</div>
</article>
{% endblock %}
Projects
The same logic is used for displaying projects, with a little caveat. Because we don't have a dedicated template for projects, we just have to use templateContent
to display the content of markdown files. Here is a simplified version of the code.
{% for project in collections.projects | reverse %}
{{ project.templateContent | safe }}
{% endfor %}
Happy with the move
All in all, I am quite happy with the move to Eleventy and the redesign. The full code for this website can be found on Github, should you be interested in poking through it.