Vuex 插件


Vuex 的 store 接收 plugins 选项,这个选项暴露出每个 mutation 的钩子。一个 Vuex 的插件就是一个简单的方法,接收 store 作为唯一参数:

  1. const myPlugin = store => {
  2. // 当 store 在被初始化完成时被调用
  3. store.subscribe((mutation, state) => {
  4. // mutation 之后被调用
  5. // mutation 的格式为 {type, payload}。
  6. })
  7. }

然后像这样使用:

  1. const store = new Vuex.Store({
  2. // ...
  3. plugins: [myPlugin]
  4. })

在插件内提交 Mutations

插件不能直接修改状态 - 这就像你的组件,它们只能被 mutations 来触发改变。

通过提交 mutations,插件可以用来同步数据源到 store。例如, 为了同步 websocket 数据源到 store (这只是为说明用法的例子,在实际中,createPlugin 方法会附加更多的可选项,来完成复杂的任务)。

  1. export default function createWebSocketPlugin (socket) {
  2. return store => {
  3. socket.on('data', data => {
  4. store.commit('receiveData', data)
  5. })
  6. store.subscribe(mutation => {
  7. if (mutation.type === 'UPDATE_DATA') {
  8. socket.emit('update', mutation.payload)
  9. }
  10. })
  11. }
  12. }
  1. const plugin = createWebSocketPlugin(socket)
  2. const store = new Vuex.Store({
  3. state,
  4. mutations,
  5. plugins: [plugin]
  6. })

生成状态快照

有时候插件想获取状态 “快照” 和状态的改变前后的变化。为了实现这些功能,需要对状态对象进行深拷贝:

  1. const myPluginWithSnapshot = store => {
  2. let prevState = _.cloneDeep(store.state)
  3. store.subscribe((mutation, state) => {
  4. let nextState = _.cloneDeep(state)
  5. // 对比 prevState 和 nextState...
  6. // 保存状态,用于下一次 mutation
  7. prevState = nextState
  8. })
  9. }

** 生成状态快照的插件只能在开发阶段使用,使用 Webpack 或 Browserify,让构建工具帮我们处理:

  1. const store = new Vuex.Store({
  2. // ...
  3. plugins: process.env.NODE_ENV !== 'production'
  4. ? [myPluginWithSnapshot]
  5. : []
  6. })

插件默认会被起用。为了发布产品,你需要用 Webpack 的 DefinePlugin 或者 Browserify 的 envify 来转换 process.env.NODE_ENV !== 'production' 的值为 false

内置 Logger 插件

如果你正在使用 vue-devtools,你可能不需要。

Vuex 带来一个日志插件用于一般的调试:

  1. import createLogger from 'vuex/dist/logger'
  2. const store = new Vuex.Store({
  3. plugins: [createLogger()]
  4. })

createLogger 方法有几个配置项:

  1. const logger = createLogger({
  2. collapsed: false, // 自动展开记录 mutation
  3. transformer (state) {
  4. // 在记录之前前进行转换
  5. // 例如,只返回指定的子树
  6. return state.subTree
  7. },
  8. mutationTransformer (mutation) {
  9. // mutation 格式 { type, payload }
  10. // 我们可以按照想要的方式进行格式化
  11. return mutation.type
  12. }
  13. })

日志插件还可以直接通过 <script> 标签,然后它会提供全局方法 createVuexLogger

要注意,logger 插件会生成状态快照,所以仅在开发环境使用。