TypeScript实战之在Vuex4中使用TS
简介
虽然TypeScript
知识点学了很多,但是在实际项目中很多小伙伴还并不知道怎么用,今天笔者结合Vuex4
来使用TypeScript
实战一下。
本文分vuex4
类型 Api
分析和vuex4
实战两部分讲述。
首先我们来分析下 vuex4
的类型 Api
createStore
我们可以先看看createStore
方法,发现它需要传递一个泛型,并且这个泛型会应用到state
上。
export function createStore<S>(options: StoreOptions<S>): Store<S>;
export interface StoreOptions<S> {
state?: S | (() => S);
getters?: GetterTree<S, S>;
actions?: ActionTree<S, S>;
mutations?: MutationTree<S>;
modules?: ModuleTree<S>;
plugins?: Plugin<S>[];
strict?: boolean;
devtools?: boolean;
}
GetterTree
我们可以看到getters
的类型GetterTree
接收了两个泛型。
export interface GetterTree<S, R> {
[key: string]: Getter<S, R>;
}
这两个泛型分别应用在本模块state
上和根state
上。
export type Getter<S, R> = (state: S, getters: any, rootState: R, rootGetters: any) => any;
MutationTree
我们可以看到mutations
的类型MutationTree
只接收了一个泛型。
export interface MutationTree<S> {
[key: string]: Mutation<S>;
}
并且这个泛型只应用到本模块state
上。
export type Mutation<S> = (state: S, payload?: any) => any;
ActionTree
我们可以看到actions
的类型ActionTree
接收了两个泛型。
export interface ActionTree<S, R> {
[key: string]: Action<S, R>;
}
并且也将这两个泛型分别应用在本模块state
上和根state
上。我们还可以看到,由于action
支持对象和方法形式的写法,所以Action
的类型是ActionHandler
和ActionObject
的联合类型。
export type Action<S, R> = ActionHandler<S, R> | ActionObject<S, R>;
export type ActionHandler<S, R> = (this: Store<R>, injectee: ActionContext<S, R>, payload?: any) => any;
export interface ActionContext<S, R> {
dispatch: Dispatch;
commit: Commit;
state: S;
getters: any;
rootState: R;
rootGetters: any;
}
ModuleTree
我们可以看到modules
的类型ModuleTree
只接收了一个泛型。并且将这个泛型传递到了Module
里面。
export interface ModuleTree<R> {
[key: string]: Module<any, R>;
}
我们可以发现,module
的类型和createStore
的StoreOptions
是非常像的。
因为模块本身也有自己的state、getters、actions、mutations、modules
export interface Module<S, R> {
namespaced?: boolean;
state?: S | (() => S);
getters?: GetterTree<S, R>;
actions?: ActionTree<S, R>;
mutations?: MutationTree<S>;
modules?: ModuleTree<R>;
}
了解了vuex4
类型 Api
接下来我们就进入到实战环节了。
实战
首先我们需要使用vuecli
创建一个TypeScript
的项目。
整体目录结构
笔者store
整体目录结构如下
-
index.ts
和以前的还是一样,用来创建根store
并导出根store
。 -
interfaces.ts
用来存放store
的类型。 -
modules.ts
和以前的还是一样,用来存放模块。
首先定义根state
的类型
// interfaces.ts
type Info = { address: string };
export interface IRootState {
name: string;
age: number;
info: Info;
}
在创建store
的时候将根state
的类型传递进去。
// index.ts
import { createStore, Store } from "vuex";
import { InjectionKey } from "vue";
import { IRootState } from "./interfaces";
export default createStore<IRootState>({
state: {
name: "root",
age: 0,
info: { address: "" },
},
getters: {
getRootName(state) {
return state.name;
},
getRootInfo(state) {
return state.info;
},
},
mutations: {},
actions: {},
});
并且需要导出key
// index.ts
export const key: InjectionKey<Store<IRootState>> = Symbol();
在Vue
实例使用store
的时候将key
一并传入。
// main.ts
import store, { key } from "@/store";
createApp(App).use(store, key).mount("#app");
在vue组件使用
这样我们在vue
组件就能享受到TypeScript
的优势啦。
注意这里的useStore()
,也需要我们把key
传递进去。
import { useStore } from "vuex";
import { key } from "@/store";
setup() {
const store = useStore(key);
return {
rootName: store.state.name,
};
},
可以看到,我们使用state
的时候就会被自动提示啦。
并且当你使用不存在的属性时会在我们编写代码的时候就会直接报错提示。
相较js
,大大提高了开发效率不说,还减少了bug
。
自定义useStore()方法
如果觉得每次useStore()
,还需要我们把key
传递进去麻烦的话,我们可以创建自己的useStore()
方法。
// index.ts
import { useStore as baseUseStore } from "vuex";
export function useStore() {
return baseUseStore(key);
}
这样我们在vue
组件使用store
的时候引入自己的useStore
方法就可以啦。
import { useStore } from "@/store";
setup() {
const store = useStore();
return {
rootName: store.state.name,
};
},
我们知道,在实际项目中只创建一个根store
是远远不够的。一般我们都会使用modules
。下面笔者介绍下怎么使用modules
。
modules的使用
模块的类型是Module
,并且需要传递本模块state
类型和根state
类型。
本模块state
类型需要传递进去我们可以理解,但是为什么要传递根state
类型呢?
因为我们的getters
和actions
参数里面是有rootState
的,所以需要引入根state
类型。
// modeuls/test1.ts
import { Module } from "vuex";
import { IRootState, ITest1State } from "../interfaces";
const Test1: Module<ITest1State, IRootState> = {
state: {
name: "test1",
count: 0,
},
getters: {
getTest1Name(state) {
return state.name;
},
getAllName(state, rootState) {
return state.name + rootState.age;
},
},
};
export default Test1;
创建好模块后我们需要在根store
里面引入进去,引入方式和以前还是一样。
并且我们需要把模块的state
类型一并传递到InjectionKey
中和根state
类型形成交叉类型,并重新生成key
。
// index.ts
import { createStore, Store, useStore as baseUseStore } from "vuex";
import { InjectionKey } from "vue";
import { IRootState, ITest1State } from "./interfaces";
import test1 from "./modeuls/test1";
export default createStore<IRootState>({
// ...
modules: {
test1: test1,
// ...多个模块,类似
},
});
// 定义模块类型
type Modules = {
test1: ITest1State;
// ...多个模块,类似
};
// 使用交叉类型形成新的key
export const key: InjectionKey<Store<IRootState & Modules>> = Symbol();
我们来看看在vue
组件中的使用效果。
我们可以发现,当我们使用state
的时候,test1
模块也会被提示出来
并且它里面的属性也会被直接提示出来
好了,实战环节就讲述的差不多了,小伙伴们时候都懂了呢?
总结
虽然vuex4
对TypeScript
有了很好的支持,但是笔者觉得还是不够的。
比如 使用麻烦,每次需要定义一个key
,还需要把可以传递到vue
的use
方法里面,并且在使用useStore
的时候也还需要将key
传递进去,无疑增加了开发成本。
其次,这样的配置并不会对getters、mutations、actions
生效,只会对state
有提示。
扩展
对于vuex
的模块和命名空间的使用,可能有些小伙伴还会有疑问,这里笔者再总结一下。
模块
当vuex
使用模块后,state
会按模块读取,而getter、mutation、action
不受影响。
state
和rootState
不一样,state
只是本模块的状态,rootState
是所有模块状态。
getters
和rootGetters
是一样的,都是全局的。
只开启模块的话,在当前模块是可以提交其他模块的mutation
和action
。
// user模块
{
state: {
name: 'randy'
},
getters: {
getName(state, getters, rootState, rootGetters) {
return state.name
}
},
mutations: {
setName(state, payload) {
state.name = payload
}
},
actions: {
asyncSetName({commit, dispatch, state, rootState, getters, rootGetters}, payload) {
commit("setName", payload)
}
}
}
比如上面的例子,除了state
获取方式会变化,其他的都不变
store.state.user.name
store.getters.getName
commit("setName", 'demi')
dispatch("asyncSetName")
命名空间
如果想要getters、mutations、actions
也区分模块,可以开启命名空间。
{
namespaced: true
}
当开启命名空间后,getters、mutations、actions
都需要通过模块名/key
的方式获取。
比如上面的例子,使用的话需要添加模块名。
store.state.user.name
store.getters['user/getName']
commit("user/setName", 'demi')
dispatch("user/asyncSetName")
并且
getters
和rootGetters
不再一样的。getters
只包含本模块的getters
,rootGetters
是所有模块的getters。
本模块不能定义全局action
,除非添加root: true
,并将这个 action
的定义放在函数 handler
中。
globalAction: {
root: true,
handler(context, payload) {
context.commit("mutation", payload);
},
},
本模块不能提交其他模块的mutation
和action
。如果一定要提交,需要传递第三个参数root: true}
commit("otherModuleMutation", payload, { root: true });
dispatch("otherModuleAction", payload, { root: true });
系列文章
后记
感谢小伙伴们的耐心观看,本文为笔者个人学习笔记,如有谬误,还请告知,万分感谢!如果本文对你有所帮助,还请点个关注点个赞~,您的支持是笔者不断更新的动力!
转载自:https://juejin.cn/post/7112641239488413704