tutorial

How To Use Vue To Template Your Eleventy Projects

Duration
7 minute read
Last Updated
Jan 2025
Topics
  • vue
  • eleventy
  • static site generation

Okay quick disclaimer before I kick this off — this post does not cover how to use interactive Vue components in an Eleventy project. This post covers using Vue entirely server-side! The client will not receive any Vue code.

All of the code for this project can be found in this repo on GitHub.

TL;DR

If you’re using Eleventy because it’s lovely, but like using Vue syntax to handle single-file components, template merging, interpolation, etc, I’ve got your back. This post teaches how to start a new project in Eleventy, integrate eleventy-plugin-vue, and deal with any arising quirks or idiosyncrasies.

Getting Started

This tutorial assumes you have Node and NPM installed. If you don’t yet, you can head over here to get that running on your machine. Once that’s in place, we can create a new directory, initialize it as a Node project, and install the two dependencies we need to get cookin’.

mkdir eleventy-vue-tutorial && cd eleventy-vue-tutorial

npm init # you can use the -y flag to skip the init setup

npm install -D @11ty/eleventy@beta @11ty/eleventy-plugin-vue

Let’s also create some standard directories for an Eleventy project. I personally prefer to keep my app code in a src directory, and reserve the base for configuration and meta files (like package.json, .eleventy.js, .prettierrc, etc.) but for this tutorial we’ll do as little opinionated configuration as possible!

mkdir _data _includes

Creating The Eleventy Configuration and Scripts

To get any of this working, we’ll need to add our configuration file to use the Vue plugin in our app. First, we’ll create our .eleventy.js file in the base directory of the project and populate it!

const eleventyVue = require("@11ty/eleventy-plugin-vue"); // import the plugin

module.exports = function (eleventyConfig) {
  eleventyConfig.addPlugin(eleventyVue); // tell Eleventy about the plugin
};

Eleventy Experimental Features

Here’s the first quirk of this process. At the time of writing, the default installations for eleventy and eleventy-plugin-vue rely on an experimental feature of Eleventy for Custom File Extensions. I’ve gotten around this requirement by installing the @11ty/eleventy@beta package, which implements custom file extensions as a feature.

To add the necessary scripts, we’ll go to our package.json file, and add the following:

{
  "scripts": {
    "start": "npx @11ty/eleventy --serve",
    "build": "npx @11ty/eleventy"
  }
}

If for some reason you can’t use Eleventy 1.0.0, that’s actually fine, you’ll just edit your scripts to activate the experimental features flag:

{
  "scripts": {
    "start": "ELEVENTY_EXPERIMENTAL=true npx @11ty/eleventy --serve",
    "build": "ELEVENTY_EXPERIMENTAL=true npx @11ty/eleventy"
  }
}

Layouts, Data Files, and Vue Templates

In this section, you’ll create a layout, specify it as the default layout, and create your first Vue page template.

Creating A Layout

As reported in the eleventy-plugin-vue README.md, you can’t use .vue files as base layouts, so what we’ll do instead is create a layout.html and specify it as the global layout template. This file will use some Nunjucks templating, so, sorry, you can’t escape it entirely (yet).

First, create a file _includes/layout.html, and populate it with the absolute bare minimum code to make this tutorial work!

<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <meta http-equiv="X-UA-Compatible" content="IE=edge" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <title>{{ title }}</title>
  </head>
  <body>
    {{ content }}
  </body>
</html>

Note those {{ title }} and {{ content }} tags. These will use the Eleventy data engine to inject our page’s HTML from our Vue templates. You can read more about how that works in the Eleventy documentation.

Next, we’ll need to tell Eleventy to use that file as the default layout. This is as simple as creating a layout.js in our _data directory and inputting the following JavaScript:

module.exports = "layout.html";

Adding Our First .vue Template

Now we can get into the actual Vue side of things. Let’s create our first page template, which, for now, will just have some content and set the page title.

I’m using Vue 2 for this tutorial, but Vue 3 is supported! I haven’t gone through all of these steps with Vue 3 syntax to find the challenges the new API presents.

We’ll create a file index.vue at the base of our app.

<template>
  <article>
    <h1>This is a test</h1>
    <ul>
      <li v-for="(listItem, index) in listItems" :key="index">{{ listItem }}</li>
    </ul>
  </article>
</template>

<script>
  export default {
    data() {
      return {
        title: "Wow I'm So Excited To Use Vue In My Templates!",
        listItems: [
          "This is the first item",
          "This is the second",
          "This is perhaps the third, though who could truly say.",
        ],
      };
    },
  };
</script>

The way the eleventy-plugin-vue works is by simply rendering the <template> tag and making that the content data property in the Eleventy cascade, and in the same stroke using the Vue data object as additional page data. Your page should look something like this:

A screenshot of the website being coded so far, featuring a headline, active title tag, and some list items.

Here, we used Vue iteration to turn an array of strings into an unordered list. You’ll even notice that the layout.html file automatically used the title data property as the title of the page. Nice work on this one, really just top notch work.

Rendering Content and Pagination

Rendering content from either an API like Contentful introduces some challenges. Let’s create a fake Markdown blog post API response and a post template.

In our _data directory, we can create a posts.js file that just exports a JSON array of posts, like what we might expect from a headless CMS API response.

module.exports = [
  {
    title: "This is a test blog post",
    slug: "this-is-a-test-blog-post",
    content:
      "## Subtitle\nLorem ipsum dolor sit amet consectetur adipisicing elit. Repellendus, vero, odit animi praesentium obcaecati autem velit, labore voluptates itaque consequuntur ea reprehenderit quod eveniet nobis perspiciatis neque quas cum voluptatum.",
  },
  {
    title: "This is another blog post",
    slug: "another-blog-post",
    content:
      "Lorem ipsum dolor sit amet consectetur adipisicing elit. Repellendus, vero, odit animi praesentium obcaecati autem velit, labore voluptates itaque consequuntur ea reprehenderit quod eveniet nobis perspiciatis neque quas cum voluptatum.",
  },
];

Next, create a file posts/_slug.vue which will be the template for our rendered blog posts.

<template>
  <div>
    <h1>{{ post.title }}</h1>
    <div v-html="post.content" />
  </div>
</template>

<script>
  export default {
    data() {
      return {
        pagination: {
          size: 1,
          data: "posts",
          alias: "post",
        },
        permalink: (data) => `posts/${data.post.slug}/index.html`,
        eleventyComputed: {
          title: (data) => data.post.title,
        },
      };
    },
  };
</script>

There’s a couple of things to note in that <script> tag:

  • In the data object, we’re returning a pagination entry — this is straight from the Eleventy docs, just converted to fit the Vue format!
  • Our permalink entry is a function that takes data from the paginator and returns the URL we want to render the page at.
  • This last one is tricky, and I’ve wrestled with it a couple of ways, but what I found was that to dynamically get information from data frontmatter, that is, to make available elsewhere in the data cascade, we can use eleventyComputed (not to be confused with Vue computed data) to return the title from the paginated data.

At this point, we can visit one of our blog posts at http://localhost:8080/posts/this-is-a-test-blog-post/ and see it rendering out for the most part. The next thing we’ll have to reckon with is that our Markdown isn’t being interpreted as Markdown, but that’s easily resolved using actual Vue computed data!

Let’s change this template around to use markdown-it to render our content.

markdown-it comes as a dependency of Eleventy, but we can manually install it as well so our package.json is a more accurate representation of the project’s dependencies.

npm i -D markdown-it
<template>
  <div>
    <h1>{{ post.title }}</h1>
    <div v-html="body" />
  </div>
</template>

<script>
  // import and initialize our markdown renderer from markdown-it
  const markdownRenderer = require("markdown-it")();

  export default {
    data() {
      // ...
    },
    computed: {
      body() {
        return markdownRenderer.render(this.post.content);
      },
    },
  };
</script>

We’ve added that computed property which has one body() function, which just takes the content from the paginated post and passes it through the markdown renderer. Things ought to be looking much better now.

Takeaways

  • You can use the Eleventy pagination object to create multiple pages from a single Vue template.
  • If you need to make data from a paginated page available higher in the data cascade (e.g. the page title in this case), use eleventyComputed.
  • If you need to transform data within a Vue template, you can use the Vue computed or filter properties, just like in a Vue single-page application.

Using Vue Components

Of course, a lot of the reason for using Vue templates is the single-file component architecture pattern. That works just fine in Eleventy as well! Let’s create a <Navigation /> component for use in our home page.

First, we’ll create the file _includes/Navigation.vue,

<template>
  <nav>
    <a href="/">Home</a>
    <ul>
      <li v-for="post in posts">
        <a :href="`/posts/${post.slug}/`">{{ post.title }}</a>
      </li>
    </ul>
  </nav>
</template>

<script>
  export default {
    props: ["posts"],
  };
</script>

Then in our index.vue template, we can import and reference the component.

<template>
  <article>
    <Navigation :posts="posts" />
    <!-- ... -->
  </article>
</template>
<script>
  import Navigation from "./_includes/Navigation.vue";

  export default {
  	data() {
  		//...
  	}
  	components: {
  		Navigation,
  	},
  }
</script>

This is a pretty rudimentary example, but some things to note, again:

  • There’s no automatic aliasing or resolution as there is in other static Vue implementations, like Nuxt, so you have to reference your components by a full path to the file.
  • Components will have access to the Eleventy-supplied page object, but not necessarily data from the _data directory. If I need to reference data from the _data directory, I just pass it as a prop from the page template to a child component!

Everything else is more or less the same!

Using Single-File Component CSS

eleventy-plugin-vue does have support for using single-file <style> tags, there’s just a little additional setup. In our index.vue template, we can add a style tag with some arbitrary styles:

<template>
  <!-- ... -->
</template>

<script>
  export default {
    // ...
  };
</script>

<style>
  body {
    background-color: #efefef;
  }
  article {
    font-family: sans-serif;
  }
</style>

Now we’ll need to collect the CSS from each relevant file component and output it in the page — there’s a convenient filter for that, which we’ll add to our _includes/layout.html like so:

<!DOCTYPE html>
<html lang="en">
  <head>
    <!-- ... -->
    <style>
      \{\{ page.url | getVueComponentCssForPage }}
    </style>
  </head>
  <!-- ... -->
</html>

We can also, should we so desire, add the scoped attribute to our style tag to leverage Vue’s scoped CSS features.

I avoid doing so because it’s a bunch of extra specificity and for what?

A screenshot of the website being coded, and it is clear that the CSS written in the previous step is actually being rendered in the site.

Everything ought to be workin’ just fine now. Nice work there, my friend, really just tremendous.

Conclusion

There are more idiosyncrasies to this implementation, and in a lot of cases you may decide it’s easier to use Nunjucks or a more standard Eleventy template language! That’s fine, I certainly don’t take it personally. I like Vue for the single-file architecture, the ability to do in-file data transformations, and the clean interpolation/iteration/templating syntax!

I’ll update this post as I learn how to do more things, such as:

  • Use Vue 3 lol

Thanks to my king Zach Leatherman for pointing out the spots in this blog where I was being a doofus, and Robb Owen for proof-reading me to safety while I was recovering from vaccine brain.

Further Reading:

Hire me — I'm seeking my next full-time role.