Eleventy, (also written as 11ty) - tagged the simpler static site generator is anti-framework and does exactly what it says it does; generate static sites. Static sites are slowly becoming the standard for deploying content driven web applications.
Eleventy, thought not a popular choice (at the moment) comes with not only the smaller bundle size it is known for but a wealth of templating languages that give developers the flexibility to build with what they like. It supports templating languages like .njk
, .html
, .md
, .11ty.js
,.liquid
, .hbs
, .ejs
, .haml
and .pug
to mention a few. In my Top 10 Static Site Generators in 2020 article, I described the tools ability to decouple your web pages from your content to make any future migrations easier just in case you decide to use something else later, yay for data ownership. It’s worth pointing out that Eleventy is not a JavaScript framework—that means zero boilerplate client-side JavaScript.
Recently, Zach Leatherman, the creator of Eleventy shared Vue template support for Eleventy.
For context, in this tutorial, you will build a static blog with Eleventy and Strapi. You will also learn how to create a content model in Strapi that can be used for a blog, consume data via the Strapi GraphQL API and use that data to generate a static blog with Eleventy.
Great so now that we have an idea of what Eleventy is and what it can do, let’s get started with building our blog. I want to start out with the Strapi part, defining our content model first will help us define how our blog consumes data. So let's get started!
Open your terminal.
yarn create strapi-app backend --quickstart
- this creates a new folder called backend and builds the admin UI.Navigate to http://localhost:1337/admin.
These fields will store our article content
Navigate to Article under Collection Types in the left-hand menu.
Fill in this information in the fields specified :
1 **title**: *The Boy Who Cried Wolf*
2 **content**: *Don’t be him, please.. just don’t.*
3 **author**: *Daniel Phiri*
4 **published_at**: *pick a date of your choice*
5 **slug**: *the-boy-who-cried-wolf*
Click Save
Before making requests, you need to make your collection type accessible by tweaking its permissions.
In Strapi v3.1.6 (the latest as of writing) the Roles and Permissions have been moved to settings. Here are more details on the release notes.
Navigate to Settings under General in the left-hand menu. On the Settings page, navigate to Roles under the User and Permissions Plugin.
You have a collection type, you’ve added some content to it, you’ve set up permissions, let’s send requests to your API? Yes!
Navigate to http://localhost:1337/articles to query your data.
If everything went on smoothly, you should get back some JSON data containing the content you just added. For this tutorial, however, you will be using Strapi's GraphQL API.
To enable it, navigate to ./backend
cd backend
to change directoriesyarn strapi install graphql
to install the GraphQL pluginor
Navigate to Marketplace under General in the left-hand menu.
When you have the GraphQL plugin up and running, you can test queries in your GraphQL Playground. That is all for your backend. Now to our Eleventy frontend? Yes
To get started, in your root folder install Eleventy globally by running yarn global add @11ty/eleventy
. Now that we’ve got this out the way we’ll go ahead and use the official Eleventy blog starter.
git clone https://github.com/11ty/eleventy-base-blog.git frontend
in your project folder, to clone the repo.cd frontend
to change directoriesyarn install
to install project dependencies yarn serve
to generate our static files and start the project serverGreat! Now we should have our project running on localhost and your web page should look like this.
We have a basic version of our application running. Let’s talk about how Eleventy works. In our ./frontend folder, we have a couple folders and files and do a lot to make Eleventy work the way it does. Let’s highlight a few.
.eleventy.js - this file is at the root of your folder and helps you configure your project. _sites - this folder stores all your statically generated files. It appears after building your project _includes - this folder stores your default layout templates. _data - this folder helps us store data from sources outside Eleventy i.e. JSON files and APIs
I’ve mentioned templates quite a bit already. What are they? My definition; templates are moulds that help us add structure to something. Eleventy does this with it’s templating languages. In this tutorial we are using Nunjucks (the .njk files you see). These template files are what we use to create a mould that our content can fit in. Think of a glass and water, water is our content and the glass is our template. Together they make a glass of water and anyone can drink that. The water is what’s important, the glass is just a tool. With eleventy, we can add our content (water) to different templates (glasses, jars, cups) and this way we have the freedom to use our content wherever we need to without having too much reliance on what’s being used to display this content.
Let’s have a practical example. The About us page on the website is powered by content in ./frontend/about/index.md. In the file, we define our layout
i.e ./frontend/_includes/layouts/post.njk which acts as our template. In the post.njk file we have section {{ content | safe }}
which will populate the page with content from a child layout which in our case is index.md.
Great now we can get to building out our blogs section.
Now that we understand the concept of populating data into templates, let’s do exactly that with Strapi.
In the ./_data folder, create a file called blogposts.js and paste the following code into it.
1const fetch = require("node-fetch");
2
3// function to get blogposts
4async function getAllBlogposts() {
5 // max number of records to fetch per query
6 const recordsPerQuery = 100;
7
8 // number of records to skip (start at 0)
9 let recordsToSkip = 0;
10
11 let makeNewQuery = true;
12
13 let blogposts = [];
14
15 // make queries until makeNewQuery is set to false
16 while (makeNewQuery) {
17 try {
18 // initiate fetch
19 const data = await fetch("<your Strapi GraphQL Endpoint>", {
20 method: "POST",
21 headers: {
22 "Content-Type": "application/json",
23 Accept: "application/json",
24 },
25 body: JSON.stringify({
26 query: `{
27 articles {
28 id
29 title
30 content
31 published_at
32 author
33 slug
34 }
35 }`,
36 }),
37 });
38
39 // store the JSON response when promise resolves
40 const response = await data.json();
41
42 // handle CMS errors
43 if (response.errors) {
44 let errors = response.errors;
45 errors.map((error) => {
46 console.log(error.message);
47 });
48 throw new Error("Houston... We have a CMS problem");
49 }
50
51 // update blogpost array with the data from the JSON response
52 blogposts = blogposts.concat(response.data.articles);
53
54 // prepare for next query
55 recordsToSkip += recordsPerQuery;
56
57 // stop querying if we are getting back less than the records we fetch per query
58 if (response.data.articles.length < recordsPerQuery) {
59 makeNewQuery = false;
60 }
61 } catch (error) {
62 throw new Error(error);
63 }
64 }
65
66 // format blogposts objects
67 const blogpostsFormatted = blogposts.map((item) => {
68 return {
69 id: item.id,
70 title: item.title,
71 slug: item.slug,
72 body: item.content,
73 author: item.author,
74 date: item.published_at
75 };
76 });
77
78 // return formatted blogposts
79 return blogpostsFormatted;
80}
81
82// export for 11ty
83module.exports = getAllBlogposts;
This file queries a Strapi GraphQL API, and maps the response data to a constant - blogpostsFormatted
. We’ll now be able to use this data from our API which in our case is Strapi to populate a Nunjuck template. Before anything, let's build out those templates.
In ./frontend create a folder called blogs. In that folder, create a file called list.njk and paste in the following code.
1---
2layout: layouts/home.njk
3templateClass: tmpl-post
4eleventyNavigation:
5 key: Blog
6 order: 4
7pagination:
8 data: blogposts
9 size: 12
10permalink: blog{% if pagination.pageNumber > 0 %}/page{{ pagination.pageNumber + 1}}{% endif %}/index.html
11---
12
13{% include "postslist.njk" %}
14{% set htmlTitle = item.title %}
15
16{% block content %}
17 <h2>Unfiltered, raw and probably full of typos</h2>
18
19 {# loop through paginated item #}
20 {% for item in pagination.items %}
21 {% if loop.first %}<ul>{% endif %}
22 <li>
23 <h2><a href="/blog/{{ item.slug }}">{{ item.title }} </a></h2>
24 <p>by {{ item.author }} </p>
25 <br>
26 </li>
27 {% if loop.last %}</ul>{% endif %}
28 {% endfor %}
29
30 {# pagination #}
31 {% if pagination.hrefs | length > 0 %}
32 <ul>
33 {% if pagination.previousPageHref %}
34 <li><a href="{{ pagination.previousPageHref }}">Previous page</a></li>
35 {% endif %}
36 {% if pagination.nextPageHref %}
37 <li><a href="{{ pagination.nextPageHref }}">Next page</a></li>
38 {% endif %}
39 </ul>
40 {% endif %}
41
42{% endblock %}
At the top of our file in our front matter, we define a couple properties for our file. This is where we use the 11ty pagination functionaliy to first initialise data we can use in our template. This then enables us to loop our our response data from blogposts.js using pagination.items
in the main section of our file. We also include our page layout using {% include "postslist.njk" %}
a define a pagnination solution for when our blog gets busy.
Now we have to create a template for individual blog posts, in ./frontend/blogs create a file called entry.njk and paste the following code.
1---
2layout: layouts/home.njk
3templateClass: tmpl-post
4pagination:
5 data: blogposts
6 size: 1
7 alias: blogpost
8permalink: blog/{{ blogpost.slug }}/index.html
9---
10{% include "postslist.njk" %}
11{% set htmlTitle = blogpost.title %}
12
13{% block content %}
14 {# blogpost #}
15
16 <h1>{{ blogpost.title }}</h1>
17{{ blogpost.body | safe }}
18
19{% endblock %}
Just like we did for the list.njk, we define properties in our front matter. The permalink
is super important here. Because of the way we structure it. We will have a unique permalink created for each blog post we enter into Strapi. The blog posts corresponding title and content will them be displayed in the main section of our file.
Having created two templates and added the blogpost data from the blogposts.js file, when we run yarn serve
to generate the static files and start our server the data from our API will 1. Create a page that shows all blog posts from Strapi and 2. Create individual pages for each post in Strapi for our reading please :p
You'll notice a few other files and folders that we didn't touch, you can safely delete archive.njk after which your site should look something like this when your server restarts.
That was fun! We went through building a blog content model with Strapi, enabling the Strapi GraphQL API, setting up an Eleventy project, understanding templates and Eleventy internals and adding an external data source. Hopefully this gives you a glimpse of how to work with both Eleventy and Strapi. You can now go on to build what your heart desires. I can't wait to see it. Share your projects with us in our Community Slack or tweet at me, I'll be sure to respond. Till next time!
Developer Relations @ Weaviate | Developer Education and Experience | Builder and International Speaker