Vuex Explained Visually
Managing state in an application full of components can be difficult. Facebook discovered this the hard way and created the Flux pattern, which is what Vuex is based upon. Vuex is Vue’s own state management pattern and library.
In this tutorial from Vue Mastery’s Mastering Vuex course, we’ll look at why an application might need Vuex, and how it can enhance your app.
⚠️ The Case for State Management
When we talk about state, we mean the data that your components depend on and render. Things like blog posts, to-do items, and so on. Without Vuex, as your app grows, each Vue component might have its own version of state.
But if one component changes its state, and a distance relative component needs access to that new state value, we need a way for these two components to communicate.
We could communicate that state up the component tree using events, and communicate it back down the tree by passing along props… but that can become overly complicated as our app scales.
Over time, this “family tree” of components may get quite large, creating a mental mess trying to maintain and track the state throughout your growing application.
Instead of each of our components having its own local state, we can consolidate all of our state into one place. One global location that contains the current state of our entire application.
One single source of truth.
Global State: A Single Source of Truth
Vuex provides that single source of truth for us. As we begin to store state within it, our state becomes a lot cleaner, and a lot easier to reason about. Now, every component that relies on our Global State can have direct access to it. No more jumbled mess of passing state up, over, and down.
Because Vuex is written with Vue, Vuex State is reactive — just like the Vue instance’s data.
When one component updates the Vuex State, other components can be listening for when that State changes, then they can reactively respond based off that state-change (and the new State value they receive).
But just consolidating state into a single source of truth doesn’t fully solve the problems of State Management. For example, what happens when many components alter the State in different ways, from different locations?
Changes to our State could be unpredictable and untraceable. We need some more standardization.
A State Management Pattern
This is why Vuex provides a full state management pattern for a simple and standardized way to make state changes. And if you’re familiar with Vue, Vuex should look quite similar.
Just as Vue provides a root Vue instance created with new Vue
, Vuex offers a store created with new Vuex.Store
.
While the Vue instance has a data property, the Vuex store has State. Both are reactive.
And while the instance has methods, which among other things can update data, the Vuex store has Actions, which can help handle updating the State.
And while the instance has computed properties, the Vuex store has Getters, which allow us to access filtered, derived, or computed State.
The difference with the Vuex store is that it also has Mutations. The reason I said Actions “can help handle updating the State” is because Actions are essentially methods that call (or commit) Mutations. And Mutations are what actually update the Vuex State.
Mutations also allow us to track our State changes. And if we are using the Vue DevTools, in the Vuex tab we can even get a time-stamped record of all the Mutations that were committed throughout the lifecycle of our app. This is called time-travel debugging, and the DevTools even show us what the State was at those given points in time.
Mutations within Actions
Putting our Mutations within Actions allows us to wrap some business logic around our Mutation calls.
For example: We might want to run some conditional logic to determine whether a State change needs to happen or not… If not, we might not run the Mutation. We might even default to a second Mutation instead. So as you can see, Actions allow us to wrap multiple mutations within the same logic. This is essentially our way to give Vuex the logic it needs to determine how it’s handling our State changes at the application level.
A Vuex Store
Now let’s take a look at an example Vuex Store:
In our State, we have a loadingStatus property, along with an array for todos.
Below that, the SET_LOADING_STATUS Mutation allows us to update our loadingStatus State. And the SET_TODOS Mutation sets our State with the todos that we’ll receive from an API call in our Action below.
Our Action here has multiple steps:
First, it’ll commit the SET_LOADING_STATUS Mutation to set the loadingStatus to ‘loading’.
Then it’ll make an API call…
…and when the response returns, it will commit the SET_LOADING_STATUS Mutation again to set the isLoading status to ‘notLoading’.
Finally it’ll commit the SET_TODOS Mutation to set the state of our todos equal to the response we got back from our API.
Now, if we needed the ability to only retrieve the todos that are labeled done, we can use a Getter for that, which can be programmed to filter our todos State and retrieve only the specific state that we want.
Now that we’ve explored the Vuex pattern, let’s look at it in motion.
Vuex in Motion
Future of Vuex vs Pinia…
It’s important to note that, while Vuex is a common state management solution for countless apps, Pinia is the next evolution of state management in Vue.js and is now the officially support library. Effectively, Vuex 5 == Pinia. While they share the same foundational concepts, Pinia no longer requires the Mutation layer, has improved TypeScript support, and is a leaner more modular library.
While Vuex is still a viable tool, any new projects really should be created using Pinia. We recommend checking out our Pinia tutorials to quickly become proficient. We even have a free Pinia cheat sheet for you to download.
If Vuex is still the direction you need to take, check out our full Vuex course at VueMastery.com.
Originally published at www.vuemastery.com.