Vuex 模块化实现待办事项的状态管理
sogr5916
8年前
<h2>前言</h2> <p>在vue里,组件之间的作用域是独立的,父组件跟子组件之间的通讯可以通过prop属性来传参,但是在兄弟组件之间通讯就比较麻烦了。比如A组件要告诉一件事给B组件,那么A就要先告诉他们的爸组件,然后爸组件再告诉B。当组件比较多,要互相通讯的事情很多的话,爸组件要管他们那么多事,很累的。vuex正是为了解决这个问题,让多个子组件之间可以方便的通讯。</p> <h2>项目介绍</h2> <p style="text-align:center"><img src="https://simg.open-open.com/show/36a3a843bab55acbdec74300883886b3.gif"></p> <p>待办事项中的一个事件,它可能拥有几个状态,未完成、已完成、已取消或被删除等。这个事件需要在这多种状态之间切换,那么使用vuex来管理也是非常方便的。</p> <p>来看一下vuex怎么完成状态管理的:</p> <p style="text-align:center"><img src="https://simg.open-open.com/show/226d9c29c5cdf9cf4adee069709e7701.png"></p> <p>所有组件都是调用actions,分发mutation去修改state,然后state经过getter又更新到各个组件里。state又通过localStorage存储数据到本地,下次重新打开时再读取保存的数据。</p> <h2>模块化</h2> <p>为什么要用模块化?当我们的项目比较大,组件很多,功能也多,会导致state里要存放很多内容,整个 store 都会很庞大,很难管理。</p> <p>我模块化的store目录如下:</p> <pre> <code class="language-javascript">|-store/ // 存放vuex代码 | |-eventModule // 事件模块 | | |-actions.js | | |-getters.js | | |-index.js | | |-mutations.js | | |-state.js | |-themeModule // 主题颜色模块 | | |-actions.js | | |-getters.js | | |-index.js | | |-mutations.js | | |-state.js | |-index.js // vuex的核心,创建一个store</code></pre> <p>可以看到,每个模块拥有自己的state、mutation、action、getter,这样子我们就可以把我们的项目根据功能划分为多个模块去使用vuex了,而且后期维护也不会一脸懵逼。</p> <h2>状态管理</h2> <p>接下来,我们来看看vuex完成状态管理的一个流程。 举个栗子:一个待办事项,勾选之后,会在未完成列表里移除,并在已完成的列表里出现。这个过程,是这个待办事项的状态发生了改变。勾选的时候,是执行了一个方法,那我们就先写这个方法。在 event_list.vue 文件里新建一个moveToDone方法。</p> <pre> <code class="language-javascript">methods: { moveToDone(id){ //移至已完成 this.$store.dispatch('eventdone', id); } }</code></pre> <p>在 moveToDone 方法中通过 store.dispatch 方法触发 action, 接下来我们在 eventModule/actions.js 中来注册这个 action, 接受一个 id 的参数。</p> <pre> <code class="language-javascript">module.exports = { eventdone = ({ commit }, param) =>{ commit('EVENTDONE',{id: param}); } }</code></pre> <p>action 通过调用 store.commit 提交载荷(也就是{id: param}这个对象)到名为'EVENTDONE'的 mutation,那我们再来注册这个 mutation</p> <pre> <code class="language-javascript">module.exports = { EVENTDONE(states,obj){ for (let i = 0; i < states.event.length; i++) { if (states.event[i].id === obj.id) { states.event[i].type = 2; states.event[i].time = getDate(); var item = states.event[i]; states.event.splice(i, 1); // 把该事件在数组中删除 break; } } states.event.unshift(item); // 把该事件存到数组的第一个元素 local.set(states); // 将整个状态存到本地 } }</code></pre> <p>通过 mutation 去修改 state, state里我们存放了一个 event 属性</p> <pre> <code class="language-javascript">const state = { event: [] }; export default state;</code></pre> <p>在组件中要获得这个 state 里的 event, 那就需要写个getters</p> <pre> <code class="language-javascript">const getters = { getDone(states){ return states.event.filter(function (d) { if (d.type === 2) { // type == 2表示已完成 return d; // 返回已完成的事件 } }); } }; export default getters;</code></pre> <p>然后每个module里都有一个index.js文件,把自己的state、mutation、action、getters都集合起来,就是一个module</p> <pre> <code class="language-javascript">import * as func from '../function'; import * as actions from './actions.js'; import * as mutations from './mutations.js'; import state from './state.js'; import getters from './getters.js'; module.exports = { state, getters, actions, mutations }</code></pre> <p>在 store/index.js 里创建一个 store 对象来存放这个module</p> <pre> <code class="language-javascript">import Vue from 'vue'; import Vuex from 'vuex'; import event from './eventModule'; Vue.use(Vuex); module.exports = new Vuex.Store({ modules: { event } });</code></pre> <p>最后在 event_list.vue 组件上,我们通过计算属性 computed 来获取到这个从未完成的状态改变到已完成的状态,我们要用到 store 这个对象里的getters</p> <pre> <code class="language-javascript">computed: { getDone(){ return this.$store.getters.getDone; } }</code></pre> <p>这样子,完成了 '未完成' => '已完成' 从提交修改到更新视图读取的整个流程,也是 vuex 工作的整个流程。通过 module 的封装,更加方便多模块项目的开发和维护。</p> <p> </p> <p> </p> <p>来自:http://www.w3ctech.com/topic/1974</p> <p> </p>