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。



相关文章
|
2月前
|
缓存 JavaScript UED
Vue3中v-model在处理自定义组件双向数据绑定时有哪些注意事项?
在使用`v-model`处理自定义组件双向数据绑定时,要仔细考虑各种因素,确保数据的准确传递和更新,同时提供良好的用户体验和代码可维护性。通过合理的设计和注意事项的遵循,能够更好地发挥`v-model`的优势,实现高效的双向数据绑定效果。
147 64
|
2月前
|
JavaScript 前端开发 API
Vue 3 中 v-model 与 Vue 2 中 v-model 的区别是什么?
总的来说,Vue 3 中的 `v-model` 在灵活性、与组合式 API 的结合、对自定义组件的支持等方面都有了明显的提升和改进,使其更适应现代前端开发的需求和趋势。但需要注意的是,在迁移过程中可能需要对一些代码进行调整和适配。
117 60
|
12天前
|
JavaScript API 数据处理
vue3使用pinia中的actions,需要调用接口的话
通过上述步骤,您可以在Vue 3中使用Pinia和actions来管理状态并调用API接口。Pinia的简洁设计使得状态管理和异步操作更加直观和易于维护。无论是安装配置、创建Store还是在组件中使用Store,都能轻松实现高效的状态管理和数据处理。
44 3
|
2月前
|
前端开发 JavaScript 测试技术
Vue3中v-model在处理自定义组件双向数据绑定时,如何避免循环引用?
Web 组件化是一种有效的开发方法,可以提高项目的质量、效率和可维护性。在实际项目中,要结合项目的具体情况,合理应用 Web 组件化的理念和技术,实现项目的成功实施和交付。通过不断地探索和实践,将 Web 组件化的优势充分发挥出来,为前端开发领域的发展做出贡献。
41 8
|
2月前
|
存储 JavaScript 数据管理
除了provide/inject,Vue3中还有哪些方式可以避免v-model的循环引用?
需要注意的是,在实际开发中,应根据具体的项目需求和组件结构来选择合适的方式来避免`v-model`的循环引用。同时,要综合考虑代码的可读性、可维护性和性能等因素,以确保系统的稳定和高效运行。
34 1
|
2月前
|
JavaScript
Vue3中使用provide/inject来避免v-model的循环引用
`provide`和`inject`是 Vue 3 中非常有用的特性,在处理一些复杂的组件间通信问题时,可以提供一种灵活的解决方案。通过合理使用它们,可以帮助我们更好地避免`v-model`的循环引用问题,提高代码的质量和可维护性。
44 1
|
2月前
|
JavaScript
在 Vue 3 中,如何使用 v-model 来处理自定义组件的双向数据绑定?
需要注意的是,在实际开发中,根据具体的业务需求和组件设计,可能需要对上述步骤进行适当的调整和优化,以确保双向数据绑定的正确性和稳定性。同时,深入理解 Vue 3 的响应式机制和组件通信原理,将有助于更好地运用 `v-model` 实现自定义组件的双向数据绑定。
|
2月前
|
JavaScript 索引
Vue 3.x 版本中双向数据绑定的底层实现有哪些变化
从Vue 2.x的`Object.defineProperty`到Vue 3.x的`Proxy`,实现了更高效的数据劫持与响应式处理。`Proxy`不仅能够代理整个对象,动态响应属性的增删,还优化了嵌套对象的处理和依赖追踪,减少了不必要的视图更新,提升了性能。同时,Vue 3.x对数组的响应式处理也更加灵活,简化了开发流程。
|
2月前
|
JavaScript 前端开发 API
从Vue 2到Vue 3的演进
从Vue 2到Vue 3的演进
48 0
|
2月前
|
JavaScript 前端开发 API
Vue.js响应式原理深度解析:从Vue 2到Vue 3的演进
Vue.js响应式原理深度解析:从Vue 2到Vue 3的演进
67 0

热门文章

最新文章