pinia
coderhyh 2/10/2022 笔记pinia
# 前言
Pinia.js 有如下特点:
- 完整的 ts 的支持;
- 足够轻量,压缩后的体积只有1kb左右;
- 去除 mutations, modules,只有 state,getters,actions;
- actions 支持同步和异步;
- 代码扁平化没有模块嵌套,只有 store 的概念,store 之间可以自由使用,每一个store都是独立的
- 无需手动添加 store,store 一旦创建便会自动添加;
- 支持Vue3 和 Vue2
# 安装
npm install pinia
# 引入注册Vue
Vue3 使用
import { createApp } from 'vue'
import App from './App.vue'
import {createPinia} from 'pinia'
const store = createPinia()
let app = createApp(App)
app.use(store)
app.mount('#app')
1
2
3
4
5
6
7
8
9
2
3
4
5
6
7
8
9
Vue2 使用
import { createPinia, PiniaVuePlugin } from 'pinia'
Vue.use(PiniaVuePlugin)
const pinia = createPinia()
new Vue({
el: '#app',
// other options...
// ...
// note the same `pinia` instance can be used across multiple Vue apps on
// the same page
pinia,
})
1
2
3
4
5
6
7
8
9
10
11
12
13
2
3
4
5
6
7
8
9
10
11
12
13
# 初始化仓库Store
新建一个文件夹Store
新建文件[name].ts
定义仓库Store
import { defineStore } from 'pinia'
储存是使用定义的
defineStore()
,并且它需要一个唯一的名称,作为第一个参数传递新建文件 store-namespace/index.ts
export const enum Names { Test = 'TEST' }
1
2
3store 引入
import { defineStore } from 'pinia' import { Names } from './store-namespace' //这个名称,也称为id,是必要的,Pania 使用它来连接到 devtools。将返回的函数命名为use...是可组合项之间的约定,以使其使用习惯。 export const useTestStore = defineStore(Names.Test, { })
1
2
3
4
5
6定义值
State 箭头函数 返回一个对象 在对象里面定义值
import { defineStore } from 'pinia' import { Names } from './store-namespce' export const useTestStore = defineStore(Names.Test, { state:()=>{ return { current:1 } }, //类似于computed 可以帮我们去修饰我们的值 getters:{ }, //可以操作异步 和 同步提交state actions:{ } })
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
# 操作State
# 1.State 是允许直接修改值的
<template>
<div>
<button @click="Add">+</button>
<div>
{{Test.current}}
</div>
</div>
</template>
<script setup lang='ts'>
import {useTestStore} from './store'
const Test = useTestStore()
const Add = () => {
Test.current++
}
</script>
<style>
</style>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
# 2.批量修改State的值
在他的实例上有$patch方法可以批量修改多个值
<template>
<div>
<button @click="Add">+</button>
<div> {{Test.current}} </div>
<div> {{Test.age}} </div>
</div>
</template>
<script setup lang='ts'>
import {useTestStore} from './store'
const Test = useTestStore()
const Add = () => {
Test.$patch({
current:200,
age:300
})
}
</script>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
# 3.批量修改函数形式
使用函数形式 可以自定义修改逻辑
<template>
<div>
<button @click="Add">+</button>
<div> {{Test.current}} </div>
<div> {{Test.age}} </div>
</div>
</template>
<script setup lang='ts'>
import {useTestStore} from './store'
const Test = useTestStore()
const Add = () => {
Test.$patch((state)=>{ // 传入一个回调函数 state参数就是数据
state.current++;
state.age = 40
})
}
</script>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
# 4.通过原始对象修改整个实例
$state
可以通过将store的属性设置为新对象来替换store的整个状态
缺点就是必须修改整个对象的所有属性
<template>
<div>
<button @click="Add">+</button>
<div> {{Test.current}} </div>
<div> {{Test.age}} </div>
</div>
</template>
<script setup lang='ts'>
import {useTestStore} from './store'
const Test = useTestStore()
const Add = () => {
Test.$state = {
current:10,
age:30
}
}
</script>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
# 5.通过actions修改
定义Actions
在actions 中直接使用this就可以指到state里面的值
import { defineStore } from 'pinia'
import { Names } from './store-naspace'
export const useTestStore = defineStore(Names.TEST, {
state:()=>{
return {
current:1,
age:30
}
},
actions:{
setCurrent () {
this.current++
}
}
})
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
使用方法直接在实例调用
<template>
<div>
<button @click="Add">+</button>
<div> {{Test.current}} </div>
<div> {{Test.age}} </div>
</div>
</template>
<script setup lang='ts'>
import {useTestStore} from './store'
const Test = useTestStore()
const Add = () => {
Test.setCurrent()
}
</script>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
2
3
4
5
6
7
8
9
10
11
12
13
14
15
# 解构store
在Pinia是不允许直接解构是会失去响应性的
const Test = useTestStore()
const { current, name } = Test
console.log(current, name);
1
2
3
2
3
解决方案可以使用 storeToRefs
import { storeToRefs } from 'pinia'
const Test = useTestStore()
const { current, name } = storeToRefs(Test)
1
2
3
2
3
其原理跟toRefs 一样的给里面的数据包裹一层toref
# API
# 1.$reset
重置store
到他的初始状态
state: () => ({
user: <Result>{},
name: "default",
current:1
}),
1
2
3
4
5
2
3
4
5
例如我把值改变到了10 => 然后调用$reset() => 将会把state所有值 重置回 原始定义态
# 2.订阅state的改变
只要有state 的变化就会走这个函数
Test.$subscribe((args,state)=>{
console.log(args,state);
})
1
2
3
2
3
第一个参数是一个回调函数
如果你的组件卸载之后还想继续调用可以设置第二个参数
Test.$subscribe((args,state)=>{
console.log(args,state);
},{
detached:true
})
1
2
3
4
5
2
3
4
5
# 3.订阅Actions的调用
只要有actions被调用就会走这个函数
Test.$onAction((args)=>{
console.log(args);
})
1
2
3
2
3
# pinia插件
pinia 和 vuex 都有一个通病 页面刷新状态会丢失
可以写一个pinia 插件缓存他的值
import {createPinia} from 'pinia'
const __piniaKey = '__PINIAKEY__'
type OptPinia = Partial<Options>
const setStorage = (key: string, value: any): void => {
localStorage.setItem(key, JSON.stringify(value))
}
const getStorage = (key: string) => {
return (localStorage.getItem(key) ? JSON.parse(localStorage.getItem(key) as string) : {})
}
const piniaPlugin = (options: OptPinia) => {
return (context: PiniaPluginContext) => {
const { store } = context;
const data = getStorage(`${options?.key ?? __piniaKey}-${store.$id}`)
store.$subscribe(() => {
setStorage(`${options?.key ?? __piniaKey}-${store.$id}`, toRaw(store.$state));
})
return {
...store.$state,
...data
}
}
}
const pinia = createPinia()
pinia.use(piniaPlugin({
key: "pinia"
}))
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
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
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
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45