A bit of a wordy title, huh? What is server side rendering? What does it have to do with routing and page transitions? What the heck is Nuxt.js? Funnily enough, even though it sounds complex, working with Nuxt.js and exploring the benefits of isn’t too difficult. Let’s get started!
Server side rendering
You might have heard people talking about server side rendering as of late. We looked at one method to do that with React recently. One particularly compelling aspect is the performance benefits. When we render our HTML, CSS, and JavaScript on the server, we often have less JavaScript to parse both initially and on subsequent updates. This article does really well going into more depth on the subject. My favorite takeaway is:
By rendering on the server, you can cache the final shape of your data.
Instead of grabbing JSON or other information from the server, parsing it, then using JavaScript to create layouts of that information, we’re doing a lot of those calculations upfront, and only sending down the actual HTML, CSS, and JavaScript that we need. This can reap a lot of benefits with caching, SEO, and speed up our apps and sites.
What is Nuxt.js?
Server side rendering sounds pretty nice, but you’re probably wondering if it’s difficult to set up. I’ve been using Nuxt.js for my Vue applications lately and found it surprisingly simple to work with. To be clear: you don’t need to use Nuxt.js in particular to do server side rendering. I’m just a fan of this tool for many reasons. I ran some tests last month and found that Nuxt.js had even higher lighthouse scores out of the gate than Vue’s PWA template, which I thought was impressive.
Nuxt.js is a higher-level framework that you can use with a CLI command that you can use to create universal Vue applications. Here are some, not all, of the benefits:
- Server-Side Rendering
- Automatic Code Splitting
- Powerful Routing System
- Great lighthouse scores out of the gate 🐎
- Static File Serving
- ES6/ES7 Transpilation
- Hot reloading in Development
- Pre-processors: SASS, LESS, Stylus, etc
- Write Vue Files to create your pages and layouts!
- My personal favorite: easily add transitions to your pages
Let’s set up a basic application with some routing to see the benefits for ourselves.
Getting Set up
The first thing we need to do if you haven’t already is download Vue’s CLI. You can do so globally with this command:
npm install -g vue-cli
# ... or ...
yarn add global vue-cli
You will only need to do this once, not every time you use it.
Next, we’ll use the CLI to scaffold a new project, but we’ll use Nuxt.js as the template:
vue init nuxt/starter my-project
cd my-project
yarn # or... npm install
npm run dev
You’ll see the progress of the app being built and it will give you a dedicated development server to check out: http://127.0.0.1:3000/. This is what you’ll see right away (with a pretty cool little animation):

Let’s take a look at what’s creating this initial view of our application at this point. We can go to the `pages` directory, and inside see that we have an `index.vue` page. If we open that up, we’ll see all of the markup that it took to create that page. We’ll also see that it’s a `.vue` file, using single file components just like any ordinary `vue` file, with a template tag for the HTML, a script tag for our scripts, where we’re importing a component, and some styles in a style tag. (If you aren’t familiar with these, there’s more info on what those are here.) The coolest part of this whole thing is that this `.vue` file doesn’t require any special setup. It’s placed in the `pages` directory, and Nuxt.js will automatically make this server-side rendered page!
Let’s create a new page and set up some routing between them. In `pages/index.vue`, dump the content that’s already there, and replace it with:
<template>
<div class="container">
<h1>Welcome!</h1>
<p><nuxt-link to="/product">Product page</nuxt-link></p>
</div>
</template>
<style>
.container {
font-family: "Quicksand", "Source Sans Pro", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif; /* 1 */
padding: 60px;
}
</style>
Then let’s create another page in the pages directory, we’ll call it `product.vue` and put this content inside of it:
<template>
<div class="container">
<h1>This is the product page</h1>
<p><nuxt-link to="/">Home page</nuxt-link></p>
</div>
</template>
Right away, you’ll see this:
Ta-da! 🏆
Right away, we have server side rendering, routing between pages (if you check out the URL you can see it’s going between the index page and product page), and we even have a sweet little green loader that zips across the top. We didn’t have to do much at all to get that going.
You might have noticed in here, there’s a special little element: <nuxt-link to="/">
. This tag can be used like an a
tag, where it wraps around a bit of content, and it will set up an internal routing link between our pages. We’ll use to="/page-title-here"
instead of an href
.
Now, let’s add some transitions. We’ll do this in a few stages: simple to complex.
Creating Page Transitions
We already have a really cool progress bar that runs across the top of the screen as we’re routing and makes the whole thing feel very zippy. (That’s a technical term). While I like it very much, it won’t really fit the direction we’re headed in, so let’s get rid of it for now.
We’re going to go into our `nuxt.config.js` file and change the lines:
/*
** Customize the progress-bar color
*/
loading: { color: '#3B8070' },
to
loading: false,
You’ll also notice a few other things in this nuxt.config.js
file. You’ll see our meta and head tags as well as the content that will be rendered inside of them. That’s because we won’t have a traditional `index.html` file as we do in our normal CLI build, Nuxt.js is going to parse and build our `index.vue` file together with these tags and then render the content for us, on the server. If you need to add CSS files, fonts, or the like, we would use this Nuxt.js config file to do so.
Now that we have all that down, let’s understand what’s available to us to create page transitions. In order to understand what’s happening on the page that we’re plugging into, we need to review how the transition component in Vue works. I’ve written an article all about this here, so if you’d like deeper knowledge on the subject, you can check that out. But what you really need to know is this: under the hood, Nuxt.js will plug into the functionality of Vue’s transition
component, and gives us some defaults and hooks to work with:

You can see here that we have a hook for what we want to happen right before the animation starts enter
, during the animation/transition enter-active
, and when it finishes. We have these same hooks for when something is leaving, prepended with leave
instead. We can make simple transitions that just interpolate between states, or we could plug a full CSS or JavaScript animation into them.
Usually in a Vue application, we would wrap a component or element in <transition>
in order to use this slick little functionality, but Nuxt.js will provide this for us at the get-go. Our hook for the page will begin with, thankfully- page
. All we have to do to create an animation between pages is add a bit of CSS that plugs into the hooks:
.page-enter-active, .page-leave-active {
transition: all .25s ease-out;
}
.page-enter, .page-leave-active {
opacity: 0;
transform-origin: 50% 50%;
}
I’m also going to add an extra bit of styling here so that you can see the page transitions a little easier:
html, body {
font-family: "Quicksand", "Source Sans Pro", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif; /* 1 */
background: #222;
color: white;
width: 100vw;
height: 100vh;
}
a, a:visited {
color: #3edada;
text-decoration: none;
}
.container {
padding: 60px;
width: 100vw;
height: 100vh;
background: #444;
}
Right now we’re using a CSS Transition. This only gives us the ability to designate what to do in the middle of two states. We could do something a little more interesting by having an animation adjust in a way that suggests where something is coming from and going to. For that to happen, we could separate out transitions for page-enter and page-leave-active classes, but it’s a little more DRY to use a CSS animation and specify where things are coming from and going to, and plug into each for .page-enter-active
, and .page-leave-active
:
.page-enter-active {
animation: acrossIn .45s ease-out both;
}
.page-leave-active {
animation: acrossOut .65s ease-in both;
}
@keyframes acrossIn {
0% {
transform: translate3d(-100%, 0, 0);
}
100% {
transform: translate3d(0, 0, 0);
}
}
@keyframes acrossOut {
0% {
transform: translate3d(0, 0, 0);
}
100% {
transform: translate3d(100%, 0, 0);
}
}
Let’s also add a little bit of special styling to the product page so we can see the difference between these two pages:
<style scoped>
.container {
background: #222;
}
</style>
This scoped tag is pretty cool because it will apply the styles for this page/vue file only. If you have heard of CSS Modules, you’ll be familiar with this concept.
We would see this (this page is for demo purposes only, that’s probably too much movement for a typical page transition):
Now, let’s say we have a page with a totally different interaction. For this page, the movement up and down was too much, we just want a simple fade. For this case, we’d need to rename our transition hook to separate it out.
Let’s create another page, we’ll call it the contact page and create it in the pages directory.
<template>
<div class="container">
<h1>This is the contact page</h1>
<p><nuxt-link to="/">Home page</nuxt-link></p>
</div>
</template>
<script>
export default {
transition: 'fadeOpacity'
}
</script>
<style>
.fadeOpacity-enter-active, .fadeOpacity-leave-active {
transition: opacity .35s ease-out;
}
.fadeOpacity-enter, .fadeOpacity-leave-active {
opacity: 0;
}
</style>
Now we can have two-page transitions:
You can see how we could build on these further and create more and more streamlined CSS animations per page. But from here let’s dive into my favorite, JavaScript animations, and create page transitions with a bit more horsepower.
Javascript Hooks
Vue’s <transition>
component offers some hooks to use JavaScript animation in place of CSS as well. They are as follows, and each hook is optional. The :css="false"
binding lets Vue know we’re going to use JS for this animation:
<transition
@before-enter="beforeEnter"
@enter="enter"
@after-enter="afterEnter"
@enter-cancelled="enterCancelled"
@before-Leave="beforeLeave"
@leave="leave"
@after-leave="afterLeave"
@leave-cancelled="leaveCancelled"
:css="false">
</transition>
The other thing we have available to us are transition modes. I’m a big fan of these, as you can state that one animation will wait for the other animation to finish transitioning out before transitioning in. The transition mode we will work with will be called out-in.
We can do something really wild with JavaScript and the transition mode, again, we’re going a little nuts here for the purposes of demo, we would usually do something much more subtle:
In order to do something like this, I’ve run yarn add gsap
because I’m using GreenSock for this animation. In my `index.vue` page, I can remove the existing CSS animation and add this into the <script>
tags:
import { TweenMax, Back } from 'gsap'
export default {
transition: {
mode: 'out-in',
css: false,
beforeEnter (el) {
TweenMax.set(el, {
transformPerspective: 600,
perspective: 300,
transformStyle: 'preserve-3d'
})
},
enter (el, done) {
TweenMax.to(el, 1, {
rotationY: 360,
transformOrigin: '50% 50%',
ease: Back.easeOut
})
done()
},
leave (el, done) {
TweenMax.to(el, 1, {
rotationY: 0,
transformOrigin: '50% 50%',
ease: Back.easeIn
})
done()
}
}
}
All of the code for these demos exist in my Intro to Vue repo for starter materials if you’re getting ramped up learning Vue.
One thing I want to call out here is that currently there is a bug for transition modes in Nuxt.js. This bug is fixed, but the release hasn’t come out yet. It should be all fixed and up to date in the upcoming 1.0 release, but in the meantime, here is a working simple sample demo, and the issue to track.
With this working code and those JavaScript hooks we can start to get much fancier and create unique effects, with different transitions on every page:
Here’s the site that the demo was deployed to if you’d like to see it live: https://nuxt-type.now.sh/ as well as the repo that houses the code for it: https://github.com/sdras/nuxt-type
Navigation
In that last demo you might have noticed we had a common navigation across all of the pages what we routed. In order to create this, we can go into the `layouts` directory, and we’ll see a file called `default.vue`. This directory will house the base layouts for all of our pages, “default” being the, uhm, default :)
Right away you’ll see this:
<template>
<div>
<nuxt/>
</div>
</template>
That special <nuxt/>
tag will be where our `.vue` pages files will be inserted, so in order to create a navigation, we could insert a navigation component like this:
<template>
<div>
<img class="moon" src="~assets/FullMoon2010.png" />
<Navigation />
<nuxt/>
</div>
</template>
<script>
import Navigation from '~components/Navigation.vue'
export default {
components: {
Navigation
}
}
</script>
I love this because everything is kept nice and organized between our global and local needs.
I then have a component called Navigation in a directory I’ve called `components` (this is pretty standard fare for a Vue app). In this file, you’ll see a bunch of links to the different pages:
<nav>
<div class="title">
<nuxt-link to="/rufina">Rufina</nuxt-link>
<nuxt-link to="/prata">Prata</nuxt-link>
<nuxt-link exact to="/">Playfair</nuxt-link>
</div>
</nav>
You’ll notice I’m using that <nuxt-link>
tag again even though it’s in another directory, and the routing will still work. But that last page has one extra attribute, the exact attribute: <nuxt-link exact to="/">Playfair</nuxt-link>
This is because there are many routes that match just the `/` directory, all of them do, in fact. So if we specify exact
, Nuxt will know that we only mean the index page in particular.
Further Resources
If you’d like more information about Nuxt, their documentation is pretty sweet and has a lot of examples to get you going. If you’d like to learn more about Vue, I’ve just made a course on Frontend Masters and all of the materials are open source here, or you can check out our Guide to Vue, or you can go to the docs which are extremely well-written. Happy coding!
This looks like a really easy way to get into js rendering and routing (especially for someone who is already interested in Vue.js), but I wonder if you’re going to talk more about deployment? How do we get this up and running on a server? How does someone integrate with a CMS like WordPress or Craft to populate the templates? Now that most CMSes are able to output a JSON REST API natively, more resources that bridge the gap between CMS and these impressive front-end technologies would be welcome.
You read my mind! Deployment articles are indeed in the works :) I have a couple more OSS projects in my backlog to complete first, but know that they are on the way!
Hey Dalton,
You’ve raised a great question! It’s one of the challenges to solve when it comes to working with these sort of tech stacks. It also depends on what sort of implementation you need. A lot of people are using Nuxt to generate a static site. You aren’t locked into doing so, you can just build a typical Vue project (with magic Nuxt powers), though static generation is a great option for many use cases. Nuxt is using Webpack + Vue Static Generation + a dash of magic for the generation. (Also, if the term ‘static’ throws you off, check https://jamstack.org/ to get an idea of what I’m talking about).
Anyways…
“How does someone integrate with a CMS like WordPress or Craft to populate the templates?”
We wanted a familiar CMS atmosphere for content editors, and we also wanted to have full control over shaping RESTful endpoints to serve exactly what we wanted, and WordPress fit the bill. Many other solutions exist and are competing in this sphere (see: Contentful, Cosmic JS, DatoCMS), but they weren’t flexible enough for the complexity of our needs.
All of our data lives in WordPress (maybe yours lives in Craft? I hear awesome things about Craft these days). When we’re developing locally we have some node scripts that pull in all the data we need and generate some JSON. We then (deep breath) spin up a local content server using Zeit’s Micro server and use that to load data into the Vuex store via the NuxtServerInit hook. Utilizing Vuex conventions, we fill any data needs for all our templates (I’m going to refer to templates as components from now on). When we prepare a production deployment we skip the JSON step and do a fresh pull from the WP REST API. The deployment goes through a CI pipeline on Gitlab and then gets deployed to Amazon Web Services where it’s served via Cloundfront (CDN) + S3 (static hosting).
Another way of getting data into a component is using the awesome Fetch component hook that is built into the Nuxt environment (again, that’s going to fill the Vuex store with data, so it’s similar to using the NuxtServerInit hook). There’s also an Async Data hook that comes with Nuxt. The async data hook works outside of the Vuex store and it’s pretty awesome, but I like working with Vuex and I try to implement the same patterns for data flow with all my data.
I’ve left out many details, but you can Google many of the terms above and find great documentation on all those topics. I’ll post some links below too and keep my eye on this thread.
We needed some pretty specific stuff and so the whole process is a bit involved (albeit pretty easy once you’re familiar with the tools, and there’s a great deal available to automate), that being said there are plenty of straight forward solutions for doing easier deployments and hosting (Example: Netlify).
Btw big thanks to Sébastien Chopin, Alexandre Chopin, and Pooya Parsa for the work you are putting into Nuxt, keep going! They all respond on the Nuxt github issues as fast as they can and are very helpful.
And thanks Sarah for the great article! I look forward to gettin’ fancy with animations at the next web animation workshop you’re putting on in September.
Lemme know if I can try to clarify any of this stuff Dalton!
Nuxt
https://github.com/nuxt/nuxt.js/issues
https://nuxtjs.org/guide
https://nuxtjs.org/api
https://nuxtjs.org/examples
https://nuxtjs.org/guide/async-data
https://nuxtjs.org/guide/vuex-store
Vue
https://vuex.vuejs.org/en/intro.html
https://router.vuejs.org/en/
https://vuejs.org/v2/guide/
HTTP
https://github.com/mzabriskie/axios
CI/CD/Repo/Deployment
https://about.gitlab.com/features/gitlab-ci-cd/
https://aws.amazon.com/getting-started/use-cases/websites/
CMS
https://developer.wordpress.org/rest-api/
https://www.netlify.com/
https://www.datocms.com/
https://cosmicjs.com/
Misc
https://github.com/zeit/micro
https://github.com/zeit/now
https://serverless.com/
https://www.graph.cool/
I wrote something on how to Setup Node.js, Apache and an nginx reverse-proxy with Docker (which could be helpful to deploy this kind of app):
https://medium.com/@francoisromain/setup-node-js-apache-nginx-reverse-proxy-with-docker-1f5a5cb3e71e
Wow, great comment, Jason. Way to dig in and provide a lot of resources! (You should write articles ;) )
Thanks Jason, that is great information. It’s a lot to take in but it’s great to hear that people have gone down this path before and there are a lot of resources available.
And yes, Craft. Is. Awesome. I find the content modeling to be much more flexible than WP, the plugin architecture is straightforward, the documentation is consistent and sane, and the interface is streamlined and easy for clients to manage. We’ve been building more traditional sites with PHP/Twig templates but we’ve incorporated the Element API into a few projects and I’d like to move toward building entire sites on top of the API sometime soon.
That’s great to hear Dalton! If your content models are at all dynamic or complex the CMS and pipeline portions of theses builds are critical. I’ve looked at Craft a tiny bit more and I like the look of it.
With simple models and a smaller amount of data I really like the Netlify product, especially integrating the Netlify CMS.
https://www.netlify.com/
https://github.com/netlify/netlify-cms
For processing something server-side (like sending email from a contact form, authorization, etc), I try to go with a Function-as-a-Service product. Something AWS Lambda-esque.
Two projects that I’m a fan of for FaaS:
https://serverless.com/
https://arc.codes/
Does your Navigation component need to be wrapped in a
<template>
tag?It’s implied :)
I’ve been waiting for a recent, straightforward-looking Nuxt.js tutorial for weeeeeeks. Excited to dive into this.
Great article Sarah. Thank you!
Really nice article Sarah! Been trying out some stuff with Nuxt and I really like it. What would you recommend for building a contact form with node and integrate it to the Nuxt project?
Thanks for sharing that info Jason, and thanks Sarah for the awesome post.
Now, what about deploying to a dedicated server? I’ve been developing with vue for a few months now and I’ve encountered a very common yet hard to solve issue, social sharing on facebook and metatags.
The issue is I can’t set the metatags to get a good preview of dynamic views on facebook on the front-end, so I’ve been investigating SSR for a while now.
I’m trying to deploy a nuxt app on my dedicated server via SSH, but i’m having trouble figuring out how, anyone knows where I can find information about how to do this?
Nuxt docs do address the issue of deployment but they cover only a few use cases.
Hey Leo!
In the FAQ there are a ton of options for deployment. So far I’ve been using Now, so I’m linking to that one, but there are a lot of different options here. As stated in the earlier comment, I’m writing up a whole other article for deployment in depth because it deserves it, so if your use case isn’t covered there, I can hopefully address it in the next one or in the meantime you can open an issue on the nuxt repo
Meta tags in SSR are a little different- you’ll see you’ll need to add them in your nuxt.config.js file. I also use the facebook open graph debugger because I’ve found my social sharing og links usually need to be run through there first before anything takes: https://developers.facebook.com/tools/debug/ Twitter is a little more on-the-ball in this aspect. I hope that’s helpful!
Hi Sarah, yes I tried Now too and was succesful, so easy! but we have a dedicated server hosting on my workplace so I need to use that one.
I’ll be waiting for your next article :D
Also, I’ve done a lot of googling about this issue. I’m using worpress as a rest api, so in my vue project I do async calls to fetch the info about the posts. Once I get this info I update the metatags on the page.
But naturally, the facebook crawler does not wait until my async calls run, so I cannot get a proper preview of the post when I paste the link on facebook.
I did comment this on an issue on the vuejs repo, and got a response from Evan You himself! He recommended pre rendering (wich will not work in my case, because I have hundreds of posts) or SSR, wich I’ve been looking into for a while now.
But SSR is still black magics to me :P
This post shows you how to set up SSR with Nuxt, if you read the article and try it out, it might make more sense :)
I just wanted to give anyone a heads-up if they encounter any issues related to ES6 syntax issues when running
npm run dev
. I suspect through some combination of a recent OS X update and my path vars in oh-my-zsh, my machine reverted to using Node v5.0.0. Which doesn’t support various declarations found in the Nuxt/Starter codebase.TL;DR: error city related to syntax in Starter when running
npm run dev
? Check which version of Node you’re using ($ node -v
). If it’s v5.0.0 investigate upgrading with nvm!Yep, that’s right! Async/Await is used in Nuxt core, so you’ll need to be running a version of Node that supports it. I suggest v8+ because it’s a LTS release and supports Async/Await.