Tutorial

Create a Lazy Loaded Wizard in Vue.js

Published on January 14, 2018
author

Alex Jover Morales

Create a Lazy Loaded Wizard in Vue.js

A wizard is a component consisting in several steps. Each step is a different view. Let’s see how to create a wizard that lazy loads its parts in Vue.js.

🧙 The Wizard Steps

Each step in a wizard is one of the screens that the wizard is showing. Let’s create one component per each step and name it using the same naming pattern, such as Wizard-step1.vue, Wizard-step2.vue, Wizard-step3.vue. Just put some trivial content in there, for instance:

Wizard-step1.vue
<template>
  <div>
    Step 1
  </div>
</template>

The Wizard

This is the component where the magic happens. The wizard has a step counter which we’ll use to lazy load, and a button which will increase that counter. Let’s get that done:

Wizard.vue
<template>
  <div>
    <button @click="next">Next</button>
  </div>
</template>

<script>
export default {
  data: () => ({ step: 1 }),
  methods: {
    next() {
      this.step++;
    }
  }
};
</script>

The idea is that the wizard loads the corresponding component depending on the current step. For lazy loading, we can use a dynamic import, but that’s asynchronous and returns a promise, while a component’s rendering is synchronous.

Vue gives us a couple of features we might be able to use for our purpose:

  • Async components: Passing a function (which returns a promise) instead of an object allows to asynchronously lazy load components, at least when using local and global registration. The problem is that those components are known beforehand, which is not the case for our example.

  • Dynamic components: Using the <component> reserved element, we can hot-swap components dynamically. However, we have the same problem, we have to know beforehand those components.

    Here’s the treasure: even though it’s not documented on the Vue.js docs, we can combine the power of both features using a computed property combined with the dynamic import, since the element also allows to get a function returning a promise which under the hood performs a kind of local registration.

    The dynamic import is a JavaScript feature that allows to load a module at runtime, similar to how require works with Node.js. Some module bundlers, such as Webpack or Rollup, use the dynamic import as a code splitting point and create a separate bundle, loaded on demand when that code is reached.

    Let’s add that part to the component:

Wizard.vue
<template>
  <div>
    <button @click="next">Next</button>
    <component :is="stepComponent"/>
  </div>
</template>

<script>
export default {
  data: () => ({ step: 1 }),
  computed: {
    stepComponent() {
      return () => import(`./Wizard-step${this.step}.vue`);
    }
  },
  methods: {
    next() {
      this.step++;
    }
  }
};
</script>

I’m creating the stepComponent computed property, which returns a function that loads the right component given the current step. Then above, I’m using the <component> element and binding it to stepComponent.

If you try it out, it should work. However, if you click on the next button, it won’t update. This is due to the fact that it’s not evaluating any reactive property within the computed property, since step is within the returned function. Computed properties in Vue are cached, and in this case is returning the latest value.

You could try to use a method instead, which is not cached, but you’ll end up in an infinite rendering loop (try yourself).

The workaround is to make Vue evaluate the step state property. For that, we can simply call it:

Wizard.vue
stepComponent() {
  this.step; // Just call it
  return () => import(`./Wizard-step${this.step}.vue`);
}

Try it again, open the network tab of the browser devtools and enjoy watching how your chunks are loaded as you press the next button! ✨

Thanks for learning with the DigitalOcean Community. Check out our offerings for compute, storage, networking, and managed databases.

Learn more about our products

About the authors
Default avatar
Alex Jover Morales

author

While we believe that this content benefits our community, we have not yet thoroughly reviewed it. If you have any suggestions for improvements, please let us know by clicking the “report an issue“ button at the bottom of the tutorial.

Still looking for an answer?

Ask a questionSearch for more help

Was this helpful?
 
1 Comments


This textbox defaults to using Markdown to format your answer.

You can type !ref in this text area to quickly search our full set of tutorials, documentation & marketplace offerings and insert the link!

This is clever, but unfortunately doesn’t seem to work in newer versions of Vue… instead (fortunately), look up the defineAsyncComponent function introduced in Vue 3

Try DigitalOcean for free

Click below to sign up and get $200 of credit to try our products over 60 days!

Sign up

Join the Tech Talk
Success! Thank you! Please check your email for further details.

Please complete your information!

Become a contributor for community

Get paid to write technical tutorials and select a tech-focused charity to receive a matching donation.

DigitalOcean Documentation

Full documentation for every DigitalOcean product.

Resources for startups and SMBs

The Wave has everything you need to know about building a business, from raising funding to marketing your product.

Get our newsletter

Stay up to date by signing up for DigitalOcean’s Infrastructure as a Newsletter.

New accounts only. By submitting your email you agree to our Privacy Policy

The developer cloud

Scale up as you grow — whether you're running one virtual machine or ten thousand.

Get started for free

Sign up and get $200 in credit for your first 60 days with DigitalOcean.*

*This promotional offer applies to new accounts only.