How to use provide to implement state management in Vue3

How to use provide to implement state management in Vue3

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 brings some costs. For smaller projects, it is not worth introducing Vuex just to store a small amount of data such as user information.

Vue2.2.x later provided the provide/inject API to help us achieve communication between cross-level components.

Vue3.x also puts provide on the application API, which makes it easier for us to implement basic state management on this basis.

How to implement Vuex functions through provide/inject

First, let's think about the general logic, make it into a plug-in, and register it to the application instance through the use method.

In the install method, the data is mounted on the root component through the app.provide method. The data should be responsive data, and for data security, the change of the data should be restricted. It follows the design of unidirectional data flow and cannot be modified directly by users. Therefore, when exposing data, the data should be processed as readonly.

Implement a useStore function similar to Vuex, allowing users to access data through this method.

Implement mapState, mapMutations and mapActions methods similar to Vuex to simplify operations.

The usage is exactly the same as Vuex.

Register this plugin 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')

Plugin entry file

In the entry file, export all methods directly.

// sky-vuex/index.ts
export * from './main/index'

Create a store and mount the corresponding data on the root component

The store itself is an object that contains state properties and methods such as commit and dispatch. The main functions of the store are to allow all components to get the store object to obtain the data in the state and call related methods to modify the 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 obtain the store object return inject(mainStoreSky)
}

Implementing 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)
}

Used 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
    }
  },
})

// Home.vue
<template>
  <div class="home">
    <button @click="handleAge(23)">Modify 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>

Summarize

So far, the basic Vuex functions have been implemented. You can practice it yourself and optimize it. If you have any questions, you are welcome to ask

This is the end of this article about how to use provide in vue3 to implement state management. For more information about vue3 provide to implement state management, please search for previous articles on 123WORDPRESS.COM or continue to browse the following related articles. I hope you will support 123WORDPRESS.COM in the future!

You may also be interested in:
  • Vue front-end development auxiliary function state management detailed example
  • The practical process of login status management in the vuex project
  • Quickly master how to get started with Vuex state management in Vue3.0
  • How to understand Vue's simple state management store mode
  • Application of mapState idea in vuex
  • How to use vuex in Vue project
  • Use of vuex namespace
  • Vue state management: using Pinia instead of Vuex

<<:  HTML table markup tutorial (9): cell spacing attribute CELLSPACING

>>:  Detailed explanation of the implementation process of Nginx enabling Brotli compression algorithm

Recommend

Solve the problem of garbled data in MySQL database migration

Under the instructions of my leader, I took over ...

React implements infinite loop scrolling information

This article shares the specific code of react to...

JavaScript canvas implements graphics and text with shadows

Use canvas to create graphics and text with shado...

Ten popular rules for interface design

<br />This is an article I collected a long ...

The shortest JS to determine whether it is IE6 (IE writing method)

Commonly used JavaScript code to detect which ver...

Example code for implementing the wavy water ball effect using CSS

Today I learned a new CSS special effect, the wav...

Detailed Introduction to MySQL Innodb Index Mechanism

1. What is an index? An index is a data structure...

Sharing of two website page translation plug-ins

TranslateThis URL: http://translateth.is Google T...

Summary of how to use bootstrap Table

This article shares with you how to use bootstrap...

Installation of mysql-community-server. 5.7.18-1.el6 under centos 6.5

Use the following command to check whether MySQL ...

Steps for packaging and configuring SVG components in Vue projects

I just joined a new company recently. After getti...

How to change the terminal to a beautiful command line prompt in Ubuntu 18

I reinstalled VMware and Ubuntu, but the command ...