经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » JS/JS库/框架 » TypeScript » 查看文章
Vite4+Typescript+Vue3+Pinia 从零搭建(6) - 状态管理pina
来源:cnblogs  作者:唯之为之  时间:2023/12/5 12:07:39  对本文有异议

项目代码同步至码云 weiz-vue3-template
pina 是 vue3 官方推荐的状态管理库,由 Vue 核心团队维护,旨在替代 vuex。pina 的更多介绍,可从 pina官网 查看

特点

  • 更简洁直接的 API,提供组合式风格的 API
  • 支持模块热更新和服务端渲染
  • 对TS支持更为友好

安装

  1. npm i pinia

使用

1. 创建实例

src目录下新建store文件夹,并新建index.ts文件

  1. import { createPinia } from 'pinia'
  2. const store = createPinia()
  3. export default store

2. 使用实例

在main.ts里引入并使用

  1. import { createApp } from 'vue'
  2. import pinia from '@/store'
  3. import './style.css'
  4. import App from './App.vue'
  5. import router from '@/router/index'
  6. createApp(App).use(router).use(pinia).mount('#app')

3. 创建store

types文件夹下创建store.ts类型声明

  1. export interface UserState {
  2. token: string
  3. userInfo: { name?: string; phone?: string }
  4. }

store目录下新建modules文件夹,并新建user.ts文件

  1. import { defineStore } from 'pinia'
  2. import { UserState } from 'types/store'
  3. // 第一个参数是id,唯一
  4. export const useUserStore = defineStore('user', {
  5. state: () => {
  6. return {
  7. token: 'EFA68205747CB561BB7C0F85D5689856',
  8. userInfo: { name: 'weizwz', phone: '18392016879' }
  9. }
  10. },
  11. getters: {
  12. namePic: (state) => state.userInfo.name.substring(0, 1)
  13. },
  14. actions: {
  15. setToken(token: string) {
  16. this.token = token
  17. },
  18. setUserInfo(userInfo: UserState['userInfo']) {
  19. this.userInfo = { ...this.userInfo, ...userInfo }
  20. }
  21. }
  22. })

4. 使用store

修改view文件夹下的home.vue来获取store

  1. <script setup lang="ts">
  2. import { computed } from 'vue'
  3. import { storeToRefs } from 'pinia'
  4. import { useUserStore } from '@store/user'
  5. defineOptions({
  6. name: 'V-home'
  7. })
  8. const userStore = useUserStore()
  9. // 获取state使用computed或者使用storeToRefs,直接使用不具备响应式(拿到的永远是初次的值)
  10. const username = computed(() => userStore.userInfo.name)
  11. // 获取getter使用storeToRefs,或者直接使用,在模板里 userStore.namePic
  12. const { namePic, token } = storeToRefs(userStore)
  13. </script>
  14. <template>
  15. <div>Hello: {{ namePic }}, your name is {{ username }}, your token is {{ token }}</div>
  16. </template>
  17. <style scoped></style>

修改view文件夹下的login.vue来设置store

  1. <script setup lang="ts">
  2. import { ref } from 'vue'
  3. import { storeToRefs } from 'pinia'
  4. import { useUserStore } from '@store/user'
  5. defineOptions({
  6. name: 'V-login'
  7. })
  8. const userStore = useUserStore()
  9. const { userInfo, token } = storeToRefs(userStore)
  10. const userName = ref(userInfo.value.name)
  11. const userToken = ref(token)
  12. const updateUserName = () => {
  13. userStore.setUserInfo({
  14. name: userName.value
  15. })
  16. }
  17. const updateUserToken = () => {
  18. userStore.setToken(userToken.value)
  19. }
  20. </script>
  21. <template>
  22. <div>login page</div>
  23. name:
  24. <input type="text" v-model="userName" @input="updateUserName" />
  25. <br />
  26. token:
  27. <input type="text" v-model="userToken" @input="updateUserToken" />
  28. </template>
  29. <style scoped></style>

如果@store/user不能识别,请在tsconfig.json中的paths里加入此路径

  1. "paths": {
  2. "@store/*": ["src/store/modules/*"],
  3. },

持久化

由于刷新界面会导致store重置,所以一般通过将store存储到cookie或者storage里使其持久化。cookie方面的插件,我这里推荐使用js-cookie

1. 安装

  1. npm i js-cookie -S
  2. npm i @types/js-cookie -S

2.封装

在src下新建utils文件夹,并新建auth.ts

  1. import Cookies from 'js-cookie'
  2. export const TokenKey = 'weiz-token'
  3. type ExpiresData = Date | number
  4. export interface TokenInfo {
  5. token: string
  6. expires: ExpiresData
  7. }
  8. export function getToken() {
  9. return Cookies.get(TokenKey)
  10. }
  11. export function setToken(data: TokenInfo) {
  12. const { token, expires } = data
  13. return expires ? Cookies.set(TokenKey, token, { expires: expires }) : Cookies.set(TokenKey, token)
  14. }
  15. export function removeToken() {
  16. return Cookies.remove(TokenKey)
  17. }

3. 使用

修改store下的user.ts

  1. import { getToken, setToken } from '@/utils/auth'
  2. // 省略
  3. state: () => {
  4. return {
  5. token: getToken() || 'EFA68205747',
  6. userInfo: { name: 'weizwz', phone: '18392016879' }
  7. }
  8. }
  9. // 省略
  10. setToken(token: string) {
  11. this.token = token
  12. setToken({
  13. token,
  14. expires: 30
  15. })
  16. }

效果展示

image

原文链接:https://www.cnblogs.com/weizwz/p/17872189.html

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号