由于使用单一的状态树,应用的所有状态会集中到一个比较大的对象。当应用变得非常复杂,store对象就会变得非常臃肿;为了解决这个问题,Vuex允许我们将store分隔成模块(module),每个模块拥有自己的state,mutation,action,getter,甚至是嵌套子模块
const moduleA={
state:{},
mutations:{},
actions:{},
getters:{}
}
const moduleB={
state:{},
mutations:{},
actions:{},
getters:{}
}
const store=new Vuex.Store({
modules:{
a:moduleA,
b:moduleB
}
})
store.state.a
store.state.b
对于模块内部的mutation和getter,接受的第一个参数是模块的局部状态对象;
const moduleA={
state:{count:0},
mutations:{
increment(state){
//这是的state对象是模块的局部状态
state.count++
}
},
getters:{
doubleCount(state){
return state.count*2
}
}
}
同样,对于模块内部的action,局部状态通过context.state暴露出来,根节点状态则为content.rootState;
const moduleA={
actions:{
incrementIfOddOnRootSum({state,commit,rootState}){
if((state.count+rootState.count)%2===1){
commit('increment')
}
}
}
}
对于模块内部的getter,根节点状态作为第三个参数暴露出来:
const moduleA={
getters:{
sumWithRootCount(state,getters,rootState){
return state.count+rootState.count
}
}
}
默认情况下,模块内部的action ,mutation,getter是注册在全局命名空间的–这样使得多个模块能够对同一mutation或者action做出响应;如果希望你的模块更加自包含或者提高可重用性,你可以通过添加namespaced:true 的方式使其成为命名空间模块,当模块被注册后,他的所有getter,action,mutation都会自动根据模块注册的路径调整命名;
const store=new Vuex.Store({
modules:{
account:{
namespaced:true,
//模块北荣
state:{},
getters:
{isAdmin(){}},//getters['account/isAdmin']
actions:{
login(){}},//dispatch['account/login']
mutations:
{login(){}},//commit('account/login')
//嵌套模块
modules:{
//继承父模块的命名空间
myPage:{
state:{},
getters:{
profile(){}// -> getters['account/profile']
}
},
//进一步嵌套命名空间
post:{
namespaced:true,
state:{},
getters{
popular() // -> getters['account/posts/popular']
}
}
}
}
}
})
启用了命名空间的getter和action会收到局部化的getter,dispatch,commit。你在使用了模块内容时不需要再同一个模块内部额外添加空间名前缀。更改namespaced属性后不需要修改模块内的代码;
如果你希望使用全局state,getter;rootState跟rootGetter会作为第三,第四参数传入getter,也会通过context对象的属性传入action;如果需要在全局命名空间内部分发action或者提交mutation,将{root:true}最为第三个参数传给dispatch或者commit即可;
modules:{
foo:{
namespaced:true,
getters:{
//在这个模块的getter中,getters被局部化了
//你可以使用getter的第四个参数来调用rootGetters
someGetter(state,getters.rootState,rootGetters){
getters.someOtherGetter
rootGetters.someOtherGetter
},
someOtherGetter:state=>{}
},
actions:{
//在这个模块中dispatch跟commit也被局部化了,他们可以接受root属性访问根dispatch或者commit
someAction({dispatch,commit,getters,rootGetters}){
getters.someGetter
rootGetters.someGetter
dispatch('someOtherAction')
dispatch('someOtherAction',null,{root:true})
dispatch('someMutation')
dispatch('someMutation',null,{root:true})
},
someOtherAction(ctx,payload){}
}
}
}
当使用mapState,mapGetters,mapActions,mapMutations这些函数来绑定命名空间模块时,写起来可能比较繁琐
computed:{
...mapState({
a:state=>state.some.nested.module.a,
b:state=>state.some.nested.module.b
})
},
methods:{
...mapActions([
'some/nested/module/foo',
'some/nested/module/bar'
])
}
对于上述情况,可以将模块的空间名称字符串作为第一个参数传递给上述函数,
computed:{
...mapStated('some/nested/module',{
a:state=>state.a,
b:state=>state.b
})
},
methods:{
...mapActions('some/nested/module',[
'foo',
'bar'
])
}