优势
dev-tools 支持
- 跟踪动作、突变的时间线
- Store 出现在使用它们的组件中
- time travel 和 更容易的调试
热模块更换
- 在不重新加载页面的情况下修改您的 Store
- 在开发时保持任何现有状态
插件:使用插件扩展 Pinia 功能
为 JS 用户提供适当的 TypeScript 支持或 autocompletion
服务器端渲染支持
与 vuex 3.x/4.x 对比
- mutations 不再存在
- 自动注入、导入函数、调用函数
- 支持ts,且内容都是类型化
- 无需动态添加 Store,默认情况下它们都是动态的
- 没有命名空间模块。
- 没有module的嵌套结构
Store
- Store是使用 defineStore() 定义的,并且它需要一个唯一名称,作为第一个参数传递
定义一个Store
1 | import { defineStore } from 'pinia' |
在 Options API 中使用 setup()
1 | import { defineStore } from 'pinia', |
在 Options API 中不使用 setup()
- 使用 mapState() 映射为只读计算属性
- 使用 mapWritableState() 可以写入
- 示例:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27import { mapState } from 'pinia'
import { useCounterStore } from '../stores/counterStore'
export default {
computed: {
💠 使用 mapState() 映射为只读计算属性
// 允许访问组件内部的 this.counter
// 与从 store.counter 读取相同
...mapState(useCounterStore, {
myOwnName: 'counter',
double: store => store.counter * 2,// 还可以编写一个访问 store 的函数
magicValue(store) { // 可以读取“this”,但无法写入...
return store.someGetter + this.counter + this.double
},
}),
💠 使用 mapWritableState() 可以写入
...mapWritableState(useCounterStore, ['counter']),// 与从 store.counter 读取相同
...mapWritableState(useCounterStore, {// 与上面相同,但将其注册为 this.myOwnName
myOwnName: 'counter',
}),
//对于像数组这样的集合,不需要mapWritableState(),除非用cartItems = []替换整个数组
},
methods:{
...mapActions(useCounterStore, { myOwnName: 'doubleCounter' }),
}
}
修改 store – 更改都出现在 devtool
- 1、直接用 store.counter++
- 2、调用 $patch 使用部分“state”对象同时应用多个更改
1
2
3
4cartStore.$patch((state) => {
state.items.push({name: 'shoes', quantity: 1})
state.hasChanged = true
})
替换state — $state 替换整个state
- store.$state = { counter: 666, name: ‘Paimon’ }
订阅状态 — $subscribe(): 只会在 patches 之后触发一次
将参数传递给getter — 此操作 getter 不再缓存
1 | //store |
访问其他store的getter/action
1 | import { useOtherStore } from './other-store' |
订阅Actions — store.$onAction()
1 | const unsubscribe = someStore.$onAction( |
插件
Pinia 插件是一个
函数
,可以选择返回
要添加
到store
的属性。
创建Pinia
1 | import { createPinia } from 'pinia' |
页面使用
1 | export function myPiniaPlugin(context) { |
扩充store
- 1、简单的在插件中返回它们的对象来为每个store添加属性
1
pinia.use(() => ({hello:'world'}))
- 2、直接在store上设置属性
1
2
3pinia.use(({ store }) => {
store.hello = 'world'
}) - 3、每个store都是用reactive包装,自动展开任何Ref(ref(),computed(,…))
1
2
3
4
5
6
7
8
9const shareRef = ref('shared')
pinia.use(({store}) => {
store.hello = ref('secret')
store.hello // secret
// 所有store都共享 shared属性
store.shared = sharedRef
store.shared // shared
})
添加新状态
- 添加位置:
- 在store上,可以使用store.myState访问
- 在store.$state上,在SSR期间被序列化
- 共享ref或computed属性
1
2
3
4
5
6
7
8
9
10
11const globalSecret = ref('secret')
pinia.use(({store}) => {
// secret在所有state之间共享
store.$state.secret = globalSecret
store.secret = globalSecret
store.router = markRaw(router) // 添加外部插件
store.$subscribe(()=>{}) // 在存储变化的时候执行
store.$onAction(()=>{}) // 在 action 的时候执行
store.secret
}) - 注意:插件中发生的状态更改或添加(包括调用store.$patch())发生在存储处于活动状态之前,因此不会触发任何订阅。
在组件外使用存储
单页应用程序
- 没有进行任何 SSR(服务器端渲染),则在使用 app.use(pinia) 安装 pinia 插件后,任何useStore() 调用都将起作用:
服务端渲染应用
- 必须将 pinia 实例传递给 useStore()
服务端渲染
vue and vite
- 1.只要在 setup 函数、getters 和 actions 的顶部调用 useStore() 函数,使用 Pinia 创建store可以立即用于 SSR
1
2
3
4
5
6export default defineComponent({
setup() {
const main = useMainStore()
return { main }
},
}) - 2.在setup()之外使用存储:则需要将 pinia 实例传递给 useStore() 函数调用
1
2
3
4
5
6
7
8
9
10const pinia = createPinia()
const app = createApp(App)
app.use(router)
app.use(pinia)
router.beforeEach((to) => {
const main = useMainStore(pinia)
if (to.meta.requiresAuth && !main.isLoggedIn) return '/login'
}) - 3.Pinia 可以将自身作为 $pinia 添加到应用程序中,以便可以在serverPrefetch() 之类的函数中使用它:
1
2
3
4
5export default {
serverPrefetch() {
const store = useStore(this.$pinia)
},
} - 4.State hydration
- 为了 hydration 初始状态,需要确保 rootState 包含在 HTML 中的某个位置,以便 Pinia 稍后获取它。出于安全原因,应该转义状态。可以使用 JSON.stringify()/JSON.parse() 序列化和解析你的状态