by Gerard Sans | @gerardsans
data:image/s3,"s3://crabby-images/db259/db2598c6a5a1b7a94f9e4a4cc3dc8c25365a3f43" alt=""
Vuex like nobody is looking
data:image/s3,"s3://crabby-images/f6e54/f6e544ddb04ae4ed79746794b47a77180f3d9300" alt=""
data:image/s3,"s3://crabby-images/f6e54/f6e544ddb04ae4ed79746794b47a77180f3d9300" alt=""
data:image/s3,"s3://crabby-images/f6e54/f6e544ddb04ae4ed79746794b47a77180f3d9300" alt=""
Vuex like nobody is looking
data:image/s3,"s3://crabby-images/8dee9/8dee9290c5668cb0fdd87dd7690446a23ca964a9" alt=""
SANS
GERARD
Google Developer Expert
data:image/s3,"s3://crabby-images/80224/8022497242bf3a8f489fbbc63d32c246fda77efd" alt=""
Google Developer Expert
International Speaker
data:image/s3,"s3://crabby-images/07d8b/07d8b538ae01b1fdef8e1a86d9d804eb05f3565e" alt=""
Spoken at 105 events in 27 countries
data:image/s3,"s3://crabby-images/afda3/afda3d2500371b526613cd2a8b4eb01f648569de" alt=""
Blogger
Blogger
Community Leader
data:image/s3,"s3://crabby-images/11288/11288cf1d4eb23adac45eef615dd8c0c254f4f6e" alt=""
900
1.5K
Trainer
data:image/s3,"s3://crabby-images/fcd71/fcd71771b824e4bb3facf6c3c6490dee12f20daf" alt=""
Master of Ceremonies
data:image/s3,"s3://crabby-images/5e1e4/5e1e4240c916359a402a61fe2eaf6e950ae875ef" alt=""
Master of Ceremonies
data:image/s3,"s3://crabby-images/1e462/1e462733b390837208e097d2b2586e7186642470" alt=""
data:image/s3,"s3://crabby-images/e786e/e786e4a052c001342e6e9f430f7dd7a468edc5ea" alt=""
data:image/s3,"s3://crabby-images/455fa/455fa7b785f5b8514fae0770b5e1e54f163b489f" alt=""
data:image/s3,"s3://crabby-images/ad4a0/ad4a04a6a4c03869787e59d11ae572f8fe24f685" alt=""
data:image/s3,"s3://crabby-images/623ab/623ab959dc6de9e1712f8f045ff80aeb6c8d8659" alt=""
data:image/s3,"s3://crabby-images/bb500/bb500e1d10d7e4bc89481b25ba00dd9d4325acf5" alt=""
data:image/s3,"s3://crabby-images/bbf51/bbf5193173f429707ceb9955bb9c709d293055dc" alt=""
data:image/s3,"s3://crabby-images/68977/68977daa3df608460d92f94d579d78a70bb500a2" alt=""
data:image/s3,"s3://crabby-images/0a421/0a421b3b47d00e8ebfc8c983a099ec72433c8517" alt=""
data:image/s3,"s3://crabby-images/a1d15/a1d15ca80fb18306a363c9dc32ddaf2be5041743" alt=""
data:image/s3,"s3://crabby-images/8cf05/8cf05996f9c6e8b201b30f2042ba32e64f43fa53" alt=""
data:image/s3,"s3://crabby-images/5e853/5e85313beb3870cc39b514cbfb170d49cacf3fde" alt=""
data:image/s3,"s3://crabby-images/2799f/2799f8fe86b19b7673477a7950dcaa67619e1cf0" alt=""
Vuex
- v3.0.1 (Feb 2016)
- 225 contributors
- CLI integration
- 17K stars
- v6.1.0 (Dec 2015)
- 137 contributors
- CLI integration
- 3K stars
- v4.0.1 (June 2015)
- 631 contributors
- Inspired by Flux
- 44K stars
data:image/s3,"s3://crabby-images/9113c/9113cb056db0cdd8c49100859ea7f1e5fdc18238" alt=""
Vuex
Overview
- State Management for Vue
- Inspired by Redux
- Composable using Modules
- Vue DevTools integration
Components
Actions
Mutations
commit
dispatch
mutate
State
update
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
Vuex one-way data flow
Actions
(async)
Mutations
(sync)
Actions vs Mutations
DevTools
Backend
commit
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
Packages
vue devtools
vuex
vuex-router-sync
Features
Utilities
Counter
Increment
Decrement
Reset
Total
import Vue from "vue";
import App from "./App";
import store from "./store";
new Vue({
el: "#app",
store,
components: { App },
template: "<App/>"
});
src/app.component.ts
Store Setup
src/main.js
Components
Actions
Mutations
commit
dispatch
mutate
State
update
import Vue from "vue";
import App from "./App";
import store from "./store";
new Vue({
el: "#app",
store, // this.$store.state
components: { App },
template: "<App/>"
});
src/app.component.ts
Store Setup
src/main.js
import actions from "./actions";
import mutations from "./mutations";
export default new Vuex.Store({
actions,
mutations,
state,
});
src/app.component.ts
State
src/store/index.js
import Vue from "vue";
import Vuex from "vuex";
Vue.use(Vuex);
const state = {
counter: 0
};
export default new Vuex.Store({
state,
});
import mutations from "./mutations";
import actions from "./actions";
export default new Vuex.Store({
state,
mutations,
actions,
});
Components
Actions
Mutations
commit
dispatch
mutate
State
update
export default {
increment(context) {
context.commit("INCREMENT");
},
decrement({ commit }) {
commit("DECREMENT");
},
reset({ commit, dispatch, state, getters }) {
context.commit("RESET");
}
};
src/app.component.ts
Counter Actions
src/store/actions.js
Components
Actions
Mutations
commit
dispatch
mutate
State
update
export default {
INCREMENT(state) {
state.counter += 1;
},
DECREMENT(state) {
state.counter -= 1;
},
RESET(state) {
state.counter = 0;
}
};
src/app.component.ts
Counter Mutations
src/store/mutations.js
Components
Actions
Mutations
commit
dispatch
mutate
State
update
<Counter
@increment="increment" @decrement="decrement" @reset="reset"
/>
export default {
methods: {
increment() {
this.$store.dispatch("increment");
},
reset() {
this.$store.dispatch("reset", { value: 0 });
}
},
}
src/app.component.ts
Dispatching Actions
src/App.vue
We pass events up to container
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
<Counter :total="total"
@increment="increment" @decrement="decrement" @reset="reset"
/>
export default {
computed: {
total() {
return this.$store.state.counter;
}
}
}
src/app.component.ts
Updating Component
src/App.vue
We will improve this part later
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
<div>
<div class="total">{{total}}</div>
<button @click="$emit('increment')">+</button>
<button @click="$emit('decrement')">-</button>
<button @click="$emit('reset', { value:0 })">C</button>
</div>
export default {
name: "Counter",
props: ["total"]
};
src/app.component.ts
Counter
src/components/Counter.vue
// src/store/actions.js
reset({ commit }, payload) {
commit("RESET", payload);
}
// src/store/mutations.js
RESET(state, { value }) {
state.counter = value;
}
src/app.component.ts
Adding payload
Getters
State
Property
getters
Computed
getters
s
g
g
Vuex Getters
-
Helpers to access Store​
-
​Avoid Components tight coupling with Store
-
Memoized for performance
import getters from "./getters";
export default new Vuex.Store({
actions,
mutations,
getters,
state,
});
src/app.component.ts
Vuex Store
src/store/index.js
src/app.component.ts
Getters
src/store/getters.js
export default {
total: state => state.counter,
overflow: state => {
return state.counter > state.maximum;
}
};
this.$store.state.counter;
this.$store.state.pagination.page;
Tight coupling
Components
State
Getters: loose coupling
this.$store.getters.total;
this.$store.getters.page;
Components
getters
State
A small change won't break all Components now
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
export default {
total: state => {
return state.counter;
},
page: state => {
return state.pagination.page;
}
};
src/app.component.ts
Property Getters
src/store/getters.js
State
Property
getters
Computed getters
s
g
g
Computed Getters
-
Computed values, can use other getters​
-
Memoised for performance
-
Default one-slot memoisation
// State
{
todos: [{ id: 1, text: 'Learn Vuex', complete: false }],
currentFilter: "SHOW_ALL"
}
// Visible Todos
[{ id: 1, text: 'Learn Vuex', complete: false }]
src/app.component.ts
Example: visibleTodos Getter
// State
{
todos: [{ id: 1, text: 'Learn Vuex', complete: false }],
currentFilter: "SHOW_COMPLETED"
}
// Visible Todos
[]
const getters = {
todos: state => state.todos,
currentFilter: state => state.currentFilter,
};
const getters = {
todos: state => state.todos,
currentFilter: state => state.currentFilter,
};
Todos Getters
src/store/getters.js
Getter: Visible Todos
const getters = {
visibleTodos: function(state, getters) {
var todos = getters.todos.slice().reverse();
switch (getters.currentFilter) {
case "SHOW_ACTIVE":
return todos.filter(t => !t.done);
case "SHOW_COMPLETED":
return todos.filter(t => t.done);
case "SHOW_ALL":
default:
return todos;
}
}
};
src/store/getters.js
<todo-list
:todos="visibleTodos" :currentFilter="currentFilter"
/>
export default {
computed: {
visibleTodos() {
return this.$store.getters.visibleTodos;
},
currentFilter() {
return this.$store.getters.currentFilter;
}
}
};
src/app.component.ts
Using visibleTodos
src/App.vue
<todo-list
:todos="visibleTodos" :currentFilter="currentFilter"
/>
export default {
computed: {
...mapGetters(['visibleTodos', 'currentFilter'])
}
};
ComponentHelpers
import { mapState } from 'vuex';
export default {
data: () => ({
factor: 12
}),
computed: {
...mapState(['total']),
...mapState({ counter: 'total' }),
formula(state) {
return state.total*this.factor;
}
}
}
src/app.component.ts
Mapping Helpers: mapState
import { mapActions, mapMutations, mapGetters } from 'vuex';
export default {
methods: {
...mapActions(['increment', 'decrement']),
...mapActions({ clear: 'reset' }), // alias
computed: {
...mapGetters({ total: 'total' })
}
}
src/app.component.ts
Other mapping helpers
Modules
Namespaced Module
Extension
Module
Root
Module
Modules
import cart from './modules/cart'
import products from './modules/products'
export default new Vuex.Store({
modules: {
cart,
products
},
})
src/app.component.ts
Store Setup
src/store/index.js
export default {
namespaced: true,
state: {
items: [],
checkoutStatus: null
},
actions,
mutations
getters,
}
src/app.component.ts
Cart Module
src/store/modules/cart.js
export default {
namespaced: true,
state: {
all: [],
},
actions,
mutations
getters,
}
src/app.component.ts
Products Module
src/store/modules/products.js
<Product v-for="product in products"
@click="addProductToCart(product)"
/>
export default {
computed: mapState({
products: state => state.products.all
}),
methods: mapActions('cart', [
'addProductToCart'
]),
created () {
this.$store.dispatch('products/getAllProducts')
}
}
src/app.component.ts
ProductList Component
Reusing stores
data:image/s3,"s3://crabby-images/2ec76/2ec76a4cc2bdf610fa8c4054d4f0640c42066831" alt=""
{
counter: 7
}
Single Counter
dispatch("increment");
State
Action
User clicks
{
counter: 8
}
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
Mutation
commit("increment");
Multiple Counters
{
"a": { counter: 7 },
"b": { counter: 3 },
}
a
b
State
User clicks
{
"a": { counter: 7 },
"b": { counter: 2 },
}
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
dispatch('b/decrement');
Action
Mutation in Module B
commit("decrement");
import counter1 from "./modules/counter";
import counter2 from "./modules/counter";
export default new Vuex.Store({
modules: {
a: counter1,
b: counter2
}
});
src/app.component.ts
Store Setup
src/store/index.js
export default {
namespaced: true,
state: () => ({
counter: 0
}),
// same actions, mutations and getters
};
src/app.component.ts
Store Setup
src/store/modules/counter.js
More
data:image/s3,"s3://crabby-images/4253d/4253d5a914ce2167391983be6a9cc8e79c650b47" alt=""
@sarah_edo
data:image/s3,"s3://crabby-images/9ae71/9ae713e6634d9b468a39ff3297853d67b7d26770" alt=""
Blake Newman
Sarah Drasner
@blakenewman
data:image/s3,"s3://crabby-images/8dee9/8dee9290c5668cb0fdd87dd7690446a23ca964a9" alt=""
data:image/s3,"s3://crabby-images/8dee9/8dee9290c5668cb0fdd87dd7690446a23ca964a9" alt=""
data:image/s3,"s3://crabby-images/8dee9/8dee9290c5668cb0fdd87dd7690446a23ca964a9" alt=""
Vuex like nobody is looking
By Gerard Sans
Vuex like nobody is looking
State Management is key to build modern Web Apps
- 2,911