插件

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

const myPlugin = store => {
  // 在 store 初始化之后调用
  store.subscribe((mutation, state) => {
    // 在每个 mutation 完成后调用
    // mutation 按照这种格式 { type, payload }。
  })
}

然后像这样使用:

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

在插件中提交 Mutation

在插件中不允许直接修改状态 —— 类似于组件,只能通过提交 mutation 来触发改变。

通过提交 mutation,插件可以用来同步一个数据源到 store。例如,同步一个 websocket 数据源到 store (下面是个大概例子,实际上 createPlugin 方法可以获得更多选项来完成复杂任务):

export default function createWebSocketPlugin (socket) {
  return store => {
    socket.on('data', data => {
      store.commit('receiveData', data)
    })
    store.subscribe(mutation => {
      if (mutation.type === 'UPDATE_DATA') {
        socket.emit('update', mutation.payload)
      }
    })
  }
}
const plugin = createWebSocketPlugin(socket)

const store = new Vuex.Store({
  state,
  mutations,
  plugins: [plugin]
})

生成 State 快照

有时候插件需要获得状态『快照』,还有比较改变的前后状态。想要实现这些,你需要对状态对象进行深拷贝:

const myPluginWithSnapshot = store => {
  let prevState = _.cloneDeep(store.state)
  store.subscribe((mutation, state) => {
    let nextState = _.cloneDeep(state)

    // 对比 prevState 和 nextState...

    // 保存状态,用于下一次 mutation
    prevState = nextState
  })
}

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

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

The plugin will be used by default. For production, you will need DefinePlugin for Webpack or envify for Browserify to convert the value of process.env.NODE_ENV !== 'production' to false for the final build. 上面插件默认会被使用,在生成发布阶段,你需要使用 Webpack 的 DefinePlugin 或者是 Browserify 的 envify 来转换 process.env.NODE_ENV !== 'production' 的值为 false

内置 Logger 插件

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

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

import createLogger from 'vuex/dist/logger'

const store = new Vuex.Store({
  plugins: [createLogger()]
})

createLogger 方法有几个配置项:

const logger = createLogger({
  collapsed: false, // 自动展开记录的 mutation
  transformer (state) {
    // 在记录之前前进行转换
    // 例如,只返回指定的子树
    return state.subTree
  },
  mutationTransformer (mutation) {
    // mutation 按照这种 { type, payload } 记录
    // 我们可以按照想要的方式对个格式化
    return mutation.type
  }
})

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

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