preface
In the Vue ecosystem, Vuex, the official state management library, brings us very convenient functions in Vue application development. However, the size of Vuex 20K+ also comes with some costs, and for smaller applications, it’s not worth introducing Vuex to store a small amount of data, such as user information. Vue2.2.x provides the Provide/Inject API in the future to help us communicate between components across the hierarchy.
Vue3. X also puts provide in the application API, which makes it easier to implement a basic state management on top of it.
How to provide/ Inject Vuex
First let’s think about the general logic and make it a plug-in that registers with the application instance using the use method.
- In the install method, the data should be mounted to the root component through the app.provide method. The data should be responsive data, and for the sake of data security, the change of data should be limited, follow the design of one-way data flow, and users can not directly modify the data. Therefore, when exposing data, The data should be readonly.
- Vuex – like useStore function is implemented to allow users to access data through this method.
- Realize mapState, mapMutations and mapActions methods similar to Vuex to simplify operation.
- The usage is directly similar to Vuex.
Register the plug-in in your application
// main.ts
import { createApp } from 'vue'
import App from './App.vue'
import router from './router'
import store from './store'
const app = createApp(App)
app.use(router).use(store).mount('#app')
Copy the code
The entry file for the plug-in
In the entry file, export all methods directly.
// sky-vuex/index.ts
export * from './main/index'
Copy the code
Create a store and mount the corresponding data to the root component
The store itself is an object that contains the state property and methods like Commit and Dispatch. Some of the main functions of a Store are to allow all components to access store objects to retrieve data from state and call methods to modify state.
// sky-vuex/main/index.ts
import {inject, reactive, readonly} from 'vue'
const mainStoreSky = Symbol('main store key') interface storeOptions { state? : any actions? : any mutations? : any }export const createStore = (options: storeOptions = {}) = > { // Create a store object
const initOptions = {
state: {},
actions: {},
mutations: {},}const mergeOptions: storeOptions = Object.assign(initOptions, options)
const state = reactive(mergeOptions.state)
const store = {
state: readonly(state),
dispatch(eventName: string, ... args: any[]){ mergeOptions.actions[eventName](store, ... args) },commit(eventName: string, ... args: any[]){... }},return {
install(app: any) {
app.provide(mainStoreSky, store)
},
}
}
export const useStore = (): any= > { // Other components use this method to get the store object
return inject(mainStoreSky)
}
Copy the code
Implement mapState, mapMutations and mapActions methods
export const mapState = () = > {
const store = useStore()
return store.state
}
export const mapActions = (eventName: string) = > {
const store = useStore()
return (. args: any[]) = >store.dispatch(eventName, ... args) }export const mapMutations = (eventName: string) = > {
const store = useStore()
return (. args: any[]) = >store.commit(eventName, ... args) }Copy the code
Use in components
// store/index.ts
import { createStore } from '.. /sky-vuex/index'
export default createStore({
state: {
age: 18
},
mutations: {
setAge(state: any, data: number) {
state.age = data
}
},
})
Copy the code
// Home.vue
<template>
<div class="home">
<button @click="handleAge(23)">Modify the data</button>
<h1>{{ state.age }}</h1>
</div>
</template>
<script lang="ts">
import { defineComponent } from 'vue'
import { useStore, mapActions, mapMutations } from '@/sky-vuex/index'
export default defineComponent({
name: 'Home'.setup() {
const store = useStore()
const handleAge = mapMutations('setAge')
// const handleAge = mapActions('setAge')
// const handleAge = () => {
// store.dispatch('setAge', 5)
// }
return {
state: store.state,
handleAge,
}
},
})
</script>
Copy the code
conclusion
So far, the basic Vuex function has been realized. You can practice it yourself and optimize it. If you have any questions, welcome to ask.