Vuex を安全に使うための型定義
Meguro.es #13 @ ラクスル
@ktsn
Vuex?
状態とミューテーション
const store = new Vuex.Store({
// 状態
state: {
count: 0
},
// ミューテーション
mutations: {
increment (state, payload) {
state.count += payload.amount
}
}
})
// 使う側
store.commit('increment', { amount: 1 })
console.log(store.state.count) // -> 1
モジュール
// カウンターモジュール
const counter = {
namespaced: true,
state: { /* ... */ },
mutations: { /* ... */ }
}
const store = new Vuex.Store({
modules: { counter }
})
store.commit('counter/increment', { amount: 1 })
コンポーネントで使う
import Vue from 'vue'
import { mapState, mapMutations } from 'vuex'
export default Vue.extend({
// increment を this に生やす
methods: mapMutations(['increment']),
// count を this に生やす
computed: mapState(['count']),
created () {
// ミューテーション
this.increment({ amount: 1 })
// 状態
console.log(this.count)
}
})
型付けがつらい
const store = new Vuex.Store({
/* ... */
mutations: {
increment (state, payload: { amount: number }) {
state.count += payload.amount
}
}
})
// 有効な mutation 名がわからない
// payload の型がわからない
store.commit('increment', 123)
namespace の型付けが困難
定義するとき
increment (state, payload) { ... }
呼び出すとき
commit('counter/increment', payload)
無理では?
ゼロから型付けできる API で作ったほうが良い
試しに書いてみたら意外にできた
// 状態: 普通に状態の型を定義
interface CounterState {
count: number
}
// ミューテーション
// キー: ミューテーション名
// 値: ペイロードの型
interface CounterMutations {
increment: { amount: number }
}
まず状態とミューテーションの
インターフェースを定義
DefineModule
import { DefineModule } from 'vuex'
// モジュールの型注釈に DefineModule を指定
const counter: DefineModule<CounterState, {}, CounterMutations, {}> = {
// namespace 付けても良い
namespaced: true,
state: {
count: 0
},
mutations: {
// 型注釈付けなくても全部型がついてる
increment (state, payload) {
state.count += payload.amount
}
}
}
createNamespacedHelpers
import { createNamespacedHelpers } from 'vuex'
// 型パラメーターに状態、ミューテーションのインターフェースを渡す
// 引数には実際の namespace の値を渡す
export const counterHelpers
= createNamespacedHelpers<CounterState, {}, CounterMutations, {}>('counter')
// コンポーネント内でヘルパーを使う
export default Vue.extend({
methods: counterHelpers.mapMutations([
'increment' // 名前が正しいかチェックされる!
]),
created () {
// ちゃんと型付けされる!
this.increment({ amount: 1 })
}
})
型安全バンザイ 🎉
Vuex を安全に使うための型定義 #meguroes
By katashin
Vuex を安全に使うための型定義 #meguroes
- 3,195