vue3中Pinia的使用之plugins

简介: vue3中Pinia的使用之plugins

Plugins



插件意思,用来扩展pinia的能力



由于是底层 API,Pania Store可以完全扩展。 以下是您可以执行的操作列表:



  • 向 Store 添加新属性
  • 定义 Store 时添加新选项
  • 为 Store 添加新方法
  • 包装现有方法
  • 更改甚至取消操作
  • 实现本地存储等副作用
  • 仅适用于特定 Store

一、向 Store 添加新属性


通常,为store添加属性时候,都是每个store添加各自的属性。如果想要为所有store添加相同的对象属性,那就不太方便,所以pinia提供了plugins,它能够做到。这对于添加全局对象(如路由器、模式或 toast 管理器)很有用。


// /src/main.js
import { createApp } from 'vue'
import { createPinia } from 'pinia'
import App from './App.vue'

import { userPlugins } from './store/plugins'
const pinia = createPinia();
pinia.use(userPlugins)

createApp(App)
.use(pinia)
.mount('#app')




注意⚠️:插件仅适用于在将pinia传递给应用程序后创建的 store ,否则将不会被应用。所以上面代码piniaPlugins顺序为什么在上面,是因为我的store在App.vue一开始就用了。如果App没有用直接放到最下面也可以的


只有 user 和 register 两个 store 被插件使用了, 每个 store 上都会有 plugin 返回的参数.


// /src/store/plugins.ts
export const userPlugins = () => {
    return {
        userName'pinia'
    }
}

// src/App.vue
<script setup>
import { userStore, useRegtores } from "@/store";
const user = userStore();
const reg = useRegtores();
</script>

<template>
   <div>userStore中用户名称为:{ { user.userName }}</
div>
   <div>useRegtores中用户名称为:{ { reg.userName }}</div>
</template>

<style scoped>
</
style>

alt
alt


当然我们也可以下面这样添加属性。但是在devTools中无法跟踪它。所以如果可以,请使用返回版本,以便 devtools 可以自动跟踪它们:


export const userPlugins = ({ store }) => {
  
  
    store.userName = "pinia";
}

这里默认是无法在devTools里面跟踪的。但是为了让userName在devtools中可见,如果你想调试它,请确保将它添加到store._customProperties仅在开发模式 开发工具:


// 从上面的例子
pinia.use(({ store }) => {
  store.userName = 'pinia'
  // 确保您的打包器可以处理这个问题。 webpack 和 vite 应该默认这样做
  if (process.env.NODE_ENV === 'development') {
    // 添加您在 store 中设置的任何 keys
    store._customProperties.add('userName')
  }
})

二、给store添加新的state


如果你想给store添加state属性,有两种方式:


1、store.[属性名](不可以在 devtools 中使用)


2、store.$state(可以在 devtools 中使用,并且在 SSR 期间被序列化)


下面看第二种


import { ref, toRef } from 'vue';
export const userPlugins = ({ store }) => {
    if(!store.$state.hasOwnProperty('hasError')){
        const hasError = ref(false);
        store.$state.hasError = hasError;
    }
    store.hasError = toRef(store.$state,'hasError')
}

1️⃣ 首先, 为了正确处理 SSR, 需要确保不覆盖任何已存在的值. 因此先判断是否存在 hasError


2️⃣ 如果不存在, 那么使用 ref 定义. 这样每个 store 都会有自己独立的 hasError


3️⃣ 其次, 如果已经存在 hasError, 我们需要将 hasError 从 state 转移到 store, 这样既可以通过 store.hasError 访问, 也可以通过 store.$state.hasError 访问.


📕这种情况下, 最后不要在 return 时返回 hasError 了. 因为返回值会被展示在开发者工具中的 state 部分, 又定义又返回就会展示两次了.


📕在 plugin 中的增加 state 或修改 state, 都不触发任何的订阅, 因为这时 store 并不活跃


三、添加新的外部属性


当添加外部属性、来自其他库的类实例或仅仅是非响应式的东西时,您应该在将对象传递给 pinia 之前使用 markRaw() 包装对象。 这是一个将路由添加到每个 store 的示例:


import { markRaw } from 'vue'
// 根据您的路由所在的位置进行调整
import { router } from './router'

pinia.use(({ store }) => {
  store.router = markRaw(router)
})

四、调用 subscribe


pinia.use(({ store }) => {
  
  
  store.$subscribe(() => {
    // 在存储变化的时候执行
  })
  store.$onAction(() => {
    // 在 action 的时候执行
  })
})

五、TypeScript


上面显示的所有内容都可以通过键入支持来完成,因此您无需使用 any 或 @ts-ignore。


1、Typing 插件


Pinia 插件可以按如下方式引入:


import { ref, toRef } from "vue";
import { PiniaPluginContext } from "pinia";

export const userPlugins = ({ store }: PiniaPluginContext) => {
  if (!store.$state.hasOwnProperty("hasError")) {
    const hasError = ref(false);
    store.$state.hasError = hasError;
  }
  store.hasError = toRef(store.$state, "hasError");
};

2、引入新的 store 属性


向 store 添加新属性时,您还应该扩展 PiniaCustomProperties 接口。


import 'pinia'
declare module 'pinia' {
  export interface PiniaCustomProperties {
    // 通过使用 setter,我们可以同时允许字符串和引用
    set hello(value: string | Ref<string>)
    get hello(): string

    // 你也可以定义更简单的值
    simpleNumber: number
  }
}

然后可以安全地写入和读取它:


pinia.use(({ store }) => {
  
  
  store.hello = 'Hola'
  store.hello = ref('Hola')

  store.number = Math.random()
  // @ts-expect-error: we haven't typed this correctly
  store.number = ref(Math.random())
})

我们可以使用 4 种通用类型的 PiniaCustomProperties 来正确输入:


import 'pinia'

declare module 'pinia' {
  export interface PiniaCustomProperties<Id, S, G, A> {
    $options: {
      id: Id
      state?: () => S
      getters?: G
      actions?: A
    }
  }
}

提示
在泛型中扩展类型时,它们的命名必须与源代码中的完全相同。 Id不能命名为id或I,S不能命名为State。 以下是每个字母所代表的含义:



  • S: State
  • G: Getters
  • A: Actions
  • SS: Setup Store / Store


关于pinia的基础知识就这里完结了,下一节将在项目中实际使用、总结pinia。



相关文章
|
4天前
|
JavaScript 前端开发
在Vue2或Vue3中项目中使用 Isotope(同位素) 过滤和排序神奇的布局神器,全网独家实现!
本文介绍了在Vue2或Vue3项目中如何使用Isotope(同位素)布局库来创建动态的网格布局,并提供了详细的代码实现和效果展示,包括过滤和排序功能。
12 0
在Vue2或Vue3中项目中使用 Isotope(同位素) 过滤和排序神奇的布局神器,全网独家实现!
|
4天前
|
前端开发 JavaScript API
基于Vue3+Hooks实现4位随机数和60秒倒计时
本文介绍了如何在Vue3中使用Hooks API来实现生成4位随机数和执行60秒倒计时的功能,并提供了详细的代码示例和运行效果展示。
22 1
基于Vue3+Hooks实现4位随机数和60秒倒计时
|
4天前
|
数据可视化 JavaScript
Vue3项目使用G6可视化组件实现一个树形机构图
在Vue 3项目中使用G6可视化组件库实现树形机构图的构建和展示。
35 1
Vue3项目使用G6可视化组件实现一个树形机构图
|
3天前
|
JavaScript 算法 API
Vue 3有哪些新特性
【8月更文挑战第16天】Vue 3有哪些新特性
20 1
|
4天前
|
JavaScript UED
如何在Vue3项目中使用防抖节流技巧
在Vue 3项目中使用防抖和节流技巧以优化组件性能,包括使用`lodash`库和自定义实现这两种方法。
9 0
如何在Vue3项目中使用防抖节流技巧
|
4天前
|
前端开发 JavaScript
基于Vue3实现鼠标按下某个元素进行拖动,实时改变左侧或右侧元素的宽度,以及点击收起或展开的功能
本文介绍了如何在Vue3项目中实现一个鼠标拖动调整元素宽度的功能,并展示了点击按钮收起或展开侧边栏的效果,提供了完整的实现代码和操作演示。
50 0
基于Vue3实现鼠标按下某个元素进行拖动,实时改变左侧或右侧元素的宽度,以及点击收起或展开的功能
|
4天前
|
JavaScript
创建 Vue3 项目
创建 Vue3 项目
10 0
|
4天前
|
JavaScript
在Vue3+ElementPlus项目中使用具有懒加载的el-tree树形控件
在Vue 3和Element Plus项目中实现具有懒加载功能的el-tree树形控件,以优化大数据量时的页面性能。
10 0
|
4天前
|
JavaScript 前端开发
在Vue3+ElementPlus项目中实现一个简单的新增/移除行记录的小组件
在Vue 3和Element Plus项目中创建一个支持新增和移除行记录的简单表格组件。
29 0
|
4天前
在Vue3项目中使用 vue3-seamless-scroll 无缝滚动插件
本文介绍了如何在Vue3项目中使用`vue3-seamless-scroll`插件实现无缝滚动效果,并提供了详细的示例代码和运行效果。
83 0