Преглед изворни кода

BUG修复/更改高德IOS_KEY

fanghuisheng пре 1 година
родитељ
комит
a768a49b93

+ 6 - 4
src/manifest.json

@@ -101,8 +101,9 @@
                 "ad" : {},
                 "maps" : {
                     "amap" : {
-                        "appkey_ios" : "ffc71dfd4e576596027f8f45a1b8fb2f",
-                        "appkey_android" : "ffc71dfd4e576596027f8f45a1b8fb2f"
+                        "appkey_ios" : "fb35d03fbb17cbf7a8743a522da3c7fc",
+                        "appkey_android" : "ffc71dfd4e576596027f8f45a1b8fb2f",
+                        "name" : "amapBOujshtbA"
                     }
                 },
                 "geolocation" : {
@@ -111,8 +112,9 @@
                     },
                     "amap" : {
                         "__platform__" : [ "ios", "android" ],
-                        "appkey_ios" : "ffc71dfd4e576596027f8f45a1b8fb2f",
-                        "appkey_android" : "ffc71dfd4e576596027f8f45a1b8fb2f"
+                        "appkey_ios" : "fb35d03fbb17cbf7a8743a522da3c7fc",
+                        "appkey_android" : "ffc71dfd4e576596027f8f45a1b8fb2f",
+                        "name" : "amapBOujshtbA"
                     }
                 },
                 "push" : {

+ 3 - 1
src/pages/business/zhaf/xunJian/collect/index.vue

@@ -119,7 +119,9 @@ const messageList = reactive({
 });
 function handleInsert(e) {
   uni.getLocation({
-    type: "wgs84",
+    type: "gcj02",
+    geocode: true,
+    highAccuracyExpireTime: 5000,
     success: function (res) {
       console.log("当前位置的经度:" + res.longitude);
       console.log("当前位置的纬度:" + res.latitude);

+ 3 - 3
src/pages/index.vue

@@ -267,10 +267,10 @@ function getLocation() {
 
   //#ifdef APP-PLUS
   uni.getLocation({
-    type: "wgs84",
+    type: "gcj02",
+    geocode: true,
+    highAccuracyExpireTime: 5000,
     success: function (res) {
-      // console.log("当前位置的经度:" + res.longitude);
-      // console.log("当前位置的纬度:" + res.latitude);
       getWeather(res.latitude + ":" + res.longitude);
     },
     fail: function (res) {

+ 1 - 1
src/pages/mine/info/index.vue

@@ -21,7 +21,7 @@
         <view class="list-cell list-cell-arrow" @click="handleUserModal('修改手机号')">
           <view class="menu-item">
             <view class="title">手机号</view>
-            <view style="margin: 0 15px 0 auto; font-size: 14px; color: #909399">{{ user.phonenumber.substr(0, 3) + "******" + user.phonenumber.substr(9) }}</view>
+            <view style="margin: 0 15px 0 auto; font-size: 14px; color: #909399">{{ user.phonenumber }}</view>
           </view>
         </view>
         <view class="list-cell list-cell-arrow" @click="handleUserModal('修改邮箱')">

+ 4 - 8
src/pages/mine/secure/index.vue

@@ -10,7 +10,7 @@
         <view class="list-cell list-cell-arrow" @click="handleUserModal('修改手机号')">
           <view class="menu-item">
             <view class="title">手机号</view>
-            <view style="margin: 0 15px 0 auto; font-size: 14px; color: #909399">{{ phonenumber }}</view>
+            <view style="margin: 0 15px 0 auto; font-size: 14px; color: #909399">{{ user.phonenumber }}</view>
           </view>
         </view>
         <view class="list-cell list-cell-arrow" @click="handleUserModal('修改密码')">
@@ -100,7 +100,7 @@
 
       <view v-if="modalTitle === '手机号验证'">
         <view class="mb5" style="text-align: center; color: #000; font-weight: 600">{{ modalTitle }}</view>
-        <view class="mb15" style="text-align: center; color: #000; font-size: 0.75rem; width: 90%; margin: 0 auto"> 请填写完整的手机号 {{ phonenumber }} 以验证身份 </view>
+        <view class="mb15" style="text-align: center; color: #000; font-size: 0.75rem; width: 90%; margin: 0 auto"> 请填写完整的手机号 {{ user.phonenumber }} 以验证身份 </view>
         <u-input v-model="phone" placeholder="请输入手机号" :maxlength="11" border="bottom" />
         <u-input v-model="verify" placeholder="请输入验证码" :maxlength="6" border="bottom">
           <template #suffix>
@@ -142,10 +142,6 @@ const data = reactive({
 const { modalShow, modalTitle, oldPassword, oldPasswordBool, newPassword, newPasswordBool, confirmPassword, confirmPasswordBool, phone, verify } = toRefs(data);
 const { avatar, user, userArr } = toRefs(useStore.$state);
 
-const phonenumber = computed(() => {
-  return user.value.phonenumber.substr(0, 3) + "******" + user.value.phonenumber.substr(9);
-});
-
 function handleUserModal(title) {
   modalShow.value = true;
   modalTitle.value = title;
@@ -208,7 +204,7 @@ function handleConfirm() {
   }
 
   if (modalTitle.value == "手机号验证") {
-    if (phone.value != user.value.phonenumber) {
+    if (phone.value != userArr.value.phonenumber) {
       proxy.$modal.msg("请输入正确的手机号");
       return;
     }
@@ -257,7 +253,7 @@ function handleCancel() {
 
 /** 点击发送验证码 */
 function getVerifyCode() {
-  if (modalTitle.value == "手机号验证" && phone.value != user.value.phonenumber) {
+  if (modalTitle.value == "手机号验证" && phone.value != userArr.value.phonenumber) {
     proxy.$modal.msg("请输入正确的手机号");
     return;
   }

+ 2 - 7
src/store/index.js

@@ -1,13 +1,8 @@
 import { createPinia } from "pinia";
-import { createPersistedState } from 'pinia-plugin-persistedstate';
+import { createUnistorage } from '@/uni_modules/pinia-plugin-unistorage'
 
 const store = createPinia()
 
-store.use(createPersistedState({
-    serializer: { // 指定参数序列化器
-        serialize: JSON.stringify,
-        deserialize: JSON.parse,
-    },
-}))
+store.use(createUnistorage())
 
 export default store

+ 0 - 9
src/store/modules/common.js

@@ -151,15 +151,6 @@ const commonStore = defineStore("common", {
 
     nfcWaiting: '请将手机靠近NFC标签',//nfc读取页面-提示文字-数据存储
   }),
-  persist: {
-    storage: window ? window.localStorage : uni.setStorageSync(), // 指定换成地址
-    beforeRestore: (context) => {
-      // console.log("Before" + context);
-    },
-    afterRestore: (context) => {
-      // console.log("After" + context);
-    },
-  },
   actions: {
     getDate() {
       var date = new Date();

+ 1 - 10
src/store/modules/setting.js

@@ -8,7 +8,7 @@ import { useStores } from "@/store/modules/index";
 
 import { uploadAvatar, updateUserProfile } from "@/api/system/user";
 
-const settingStore = defineStore("setting", {
+const settingStore = defineStore("storage-setting", {
     state: () => ({
         currentSize: "",//APP缓存
         barHeight: 0,//微信小程序顶部安全距离
@@ -20,15 +20,6 @@ const settingStore = defineStore("setting", {
         themeColor: storage.get("themeColor"),//主题
         fingerprintUserList: storage.get("fingerprintUserList"),//指纹登录用户数据
     }),
-    persist: {
-        storage: window ? window.localStorage : uni.setStorageSync(), // 指定换成地址
-        beforeRestore: (context) => {
-            // console.log("Before" + context);
-        },
-        afterRestore: (context) => {
-            // console.log("After" + context);
-        },
-    },
     actions: {
         /**
          * @系统主题颜色

+ 2 - 13
src/store/modules/user.js

@@ -32,18 +32,6 @@ const useStores = defineStore("useStores", {
     codeTime: 0,//验证码倒计时
     codeTimeInterval: null,//验证码倒计时定时器
   }),
-  persist: {
-    // 自定义数据持久化方式
-    // key: 'store-key', 指定key进行存储,此时非key的值不会持久化,刷新就会丢失
-    storage: window ? window.localStorage : uni.setStorageSync(), // 指定换成地址
-    // paths: ['nested.data'],// 指定需要持久化的state的路径名称
-    beforeRestore: (context) => {
-      // console.log("Before" + context);
-    },
-    afterRestore: (context) => {
-      // console.log("After" + context);
-    },
-  },
   actions: {
     // 登录
     Login(data) {
@@ -84,6 +72,7 @@ const useStores = defineStore("useStores", {
     getUser() {
       getUserProfile().then((response) => {
         this.user = response.data.user;
+        this.user.phonenumber = response.data.user.phonenumber.substr(0, 3) + "******" + response.data.user.phonenumber.substr(9);
         this.userArr = JSON.parse(JSON.stringify(response.data.user));
         this.postGroup = response.postGroup;
         this.roleGroup = response.roleGroup;
@@ -106,7 +95,7 @@ const useStores = defineStore("useStores", {
 
             this.SET_NAME(data.user.userName);
             this.SET_NICKNAME(data.user.nickName);
-            this.SET_PHONE(data.user.phonenumber);
+            this.SET_PHONE(data.user.phonenumber.substr(0, 3) + "******" + data.user.phonenumber.substr(9));
             this.SET_AVATAR(data.user.avatar);
 
             resolve(res);

+ 3 - 14
src/store/modules/xunJian.js

@@ -1,6 +1,6 @@
 import { defineStore } from "pinia";
 
-const xunJianStore = defineStore("xunJian", {
+const xunJianStore = defineStore("storage-xunJian", {
   state: () => ({
     planTabs: 0, //巡检计划页 tabs切换值
     siteId: null, //巡检计划页 站点id
@@ -11,20 +11,9 @@ const xunJianStore = defineStore("xunJian", {
     inspectionStatus: null,//巡检页 是否已巡检状态 1.未巡检 2.已巡检
     contentArray: {},
 
-    collectDataList: [], //巡检采集
+    collectDataList: [], //巡检点位采集数据存储
   }),
-  persist: {
-    // 自定义数据持久化方式
-    // key: 'store-key', 指定key进行存储,此时非key的值不会持久化,刷新就会丢失
-    storage: window ? window.localStorage : uni.setStorageSync(), // 指定换成地址
-    // paths: ['nested.data'],// 指定需要持久化的state的路径名称
-    beforeRestore: (context) => {
-      // console.log("Before" + context);
-    },
-    afterRestore: (context) => {
-      // console.log("After" + context);
-    },
-  },
+  unistorage: true,
   actions: {},
 });
 

+ 21 - 0
src/uni_modules/pinia-plugin-unistorage/changelog.md

@@ -0,0 +1,21 @@
+## 0.0.19(2024-01-18)
+fix: 重新构建,不需要默认参数
+## 0.0.16(2023-05-06)
+
+fix: 修复全局 key 移除
+
+## 0.0.14(2023-04-29)
+
+fix: 修复全局 global key 选项
+
+## 0.0.12(2023-04-07)
+
+- fix: 修复类型错误
+
+## 0.0.11(2023-03-22)
+
+- chore: ts 支持
+
+## 0.0.7(2022-04-29)
+
+- 更新 README

+ 86 - 0
src/uni_modules/pinia-plugin-unistorage/index.js

@@ -0,0 +1,86 @@
+function get(state, path) {
+  return path.reduce((obj, p) => {
+    return obj?.[p];
+  }, state);
+}
+function set(state, path, val) {
+  return path.slice(0, -1).reduce((obj, p) => {
+    if (!/^(__proto__)$/.test(p)) {
+      return obj[p] = obj[p] || {};
+    } else
+      return {};
+  }, state)[path[path.length - 1]] = val, state;
+}
+function pick(baseState, paths) {
+  return paths.reduce((substate, path) => {
+    const pathArray = path.split(".");
+    return set(
+      substate,
+      pathArray,
+      get(baseState, pathArray)
+    );
+  }, {});
+}
+
+const isObject = (v) => typeof v === "object" && v !== null;
+const normalizeOptions = (options, globalOptions) => {
+  options = isObject(options) ? options : /* @__PURE__ */ Object.create(null);
+  return new Proxy(options, {
+    get(t, p, r) {
+      return Reflect.get(t, p, r) || Reflect.get(globalOptions, p, r);
+    }
+  });
+};
+
+function passage(key) {
+  return key;
+}
+function createUnistorage(globalOptions = {}) {
+  const { key: normalizeKey = passage } = globalOptions || {};
+  if (globalOptions?.key) {
+    delete globalOptions.key;
+  }
+  return function(ctx) {
+    {
+      const { store, options } = ctx;
+      let { unistorage } = options || {};
+      if (!unistorage)
+        return;
+      const {
+        paths = null,
+        afterRestore,
+        beforeRestore,
+        serializer = {
+          serialize: JSON.stringify,
+          deserialize: JSON.parse
+        },
+        key = store.$id
+      } = normalizeOptions(unistorage, globalOptions);
+      beforeRestore?.(ctx);
+      const normalizedKey = normalizeKey(key);
+      try {
+        const fromStorage = uni.getStorageSync(normalizedKey);
+        if (fromStorage) {
+          store.$patch(serializer.deserialize(fromStorage));
+        }
+      } catch (_error) {
+      }
+      afterRestore?.(ctx);
+      store.$subscribe(
+        (_, state) => {
+          try {
+            const toStore = Array.isArray(paths) ? pick(state, paths) : state;
+            uni.setStorageSync(
+              normalizedKey,
+              serializer.serialize(toStore)
+            );
+          } catch (_error) {
+          }
+        },
+        { detached: true }
+      );
+    }
+  };
+}
+
+export { createUnistorage };

+ 84 - 0
src/uni_modules/pinia-plugin-unistorage/package.json

@@ -0,0 +1,84 @@
+{
+  "id": "pinia-plugin-unistorage",
+  "displayName": "pinia-plugin-unistorage",
+  "version": "0.0.19",
+  "description": "uniapp 下 pinia 的本地数据缓存插件",
+  "keywords": [
+    "pinia",
+    "uniapp",
+    "storage",
+    "pinia-plugin",
+    "persistence"
+],
+  "engines": {
+    "HBuilderX": "^3.4.7"
+  },
+  "dcloudext": {
+    "sale": {
+      "regular": {
+        "price": "0.00"
+      },
+      "sourcecode": {
+        "price": "0.00"
+      }
+    },
+    "contact": {
+      "qq": ""
+    },
+    "declaration": {
+      "ads": "无",
+      "data": "无",
+      "permissions": "无"
+    },
+    "npmurl": "https://www.npmjs.com/package/pinia-plugin-unistorage",
+    "type": "sdk-js"
+  },
+  "uni_modules": {
+    "dependencies": [],
+    "encrypt": [],
+    "platforms": {
+      "cloud": {
+        "tcb": "y",
+        "aliyun": "y"
+      },
+      "client": {
+        "Vue": {
+          "vue2": "y",
+          "vue3": "y"
+        },
+        "App": {
+          "app-vue": "y",
+          "app-nvue": "y"
+        },
+        "H5-mobile": {
+          "Safari": "y",
+          "Android Browser": "y",
+          "微信浏览器(Android)": "y",
+          "QQ浏览器(Android)": "y"
+        },
+        "H5-pc": {
+          "Chrome": "y",
+          "IE": "y",
+          "Edge": "y",
+          "Firefox": "y",
+          "Safari": "y"
+        },
+        "小程序": {
+          "微信": "y",
+          "阿里": "y",
+          "百度": "y",
+          "字节跳动": "y",
+          "QQ": "y",
+          "钉钉": "y",
+          "快手": "y",
+          "飞书": "y",
+          "京东": "y"
+        },
+        "快应用": {
+          "华为": "y",
+          "联盟": "y"
+        }
+      }
+    }
+  }
+}

+ 226 - 0
src/uni_modules/pinia-plugin-unistorage/readme.md

@@ -0,0 +1,226 @@
+<div align="center">
+    <img width="200px" height="200px" src="https://gitee.com/dishait/pinia-plugin-unistorage/raw/main/static/favicon.png" />
+    <h1>pinia-plugin-unistorage</h1>
+    <p>uniapp 下 pinia 的本地数据缓存插件</p>
+</div>
+
+<br />
+<br />
+
+<div align="center">
+    <img width="100%" height="100%" src="https://gitee.com/dishait/pinia-plugin-unistorage/raw/main/static/pinia-plugin-unistorage.gif" />
+</div>
+
+<br />
+<br />
+
+## 引用
+
+该插件是
+[pinia-plugin-persistedstate](https://github.com/prazdevs/pinia-plugin-persistedstate)
+的 `uniapp` 版本,如果你需要在纯 `vue` 或者 `nuxt` 项目中使用 `pinia`
+的本地数据缓存,请使用
+[pinia-plugin-persistedstate](https://github.com/prazdevs/pinia-plugin-persistedstate)。
+
+<br />
+<br />
+
+## 动机
+
+为了实现多端的更简单的全局本地数据缓存
+
+<br />
+<br />
+
+## 组织 🦔
+
+欢迎关注 **帝莎编程**
+
+- [官网](http://dishaxy.dishait.cn/)
+- [Gitee](https://gitee.com/dishait)
+- [Github](https://github.com/dishait)
+- [网易云课堂](https://study.163.com/provider/480000001892585/index.htm?share=2&shareId=480000001892585)
+
+<br />
+<br />
+
+## 使用
+
+### 安装
+
+#### 1. `cli` 创建的 `uniapp` 项目
+
+```shell
+npm i pinia-plugin-unistorage -D
+```
+
+```js
+// main.js
+import { createSSRApp } from "vue";
+import * as Pinia from "pinia";
+import { createUnistorage } from "pinia-plugin-unistorage";
+
+export function createApp() {
+  const app = createSSRApp(App);
+
+  const store = Pinia.createPinia();
+
+  // 关键代码 👇
+  store.use(createUnistorage());
+
+  app.use(store);
+
+  return {
+    app,
+    Pinia, // 此处必须将 Pinia 返回
+  };
+}
+```
+
+<br />
+
+#### 2. `hbuilderx` 创建的 `uniapp` 项目
+
+直接插件市场安装后引入注册
+
+```js
+// main.js
+import { createSSRApp } from "vue";
+import * as Pinia from "pinia";
+import { createUnistorage } from "./uni_modules/pinia-plugin-unistorage";
+
+export function createApp() {
+  const app = createSSRApp(App);
+
+  const store = Pinia.createPinia();
+
+  // 关键代码 👇
+  store.use(createUnistorage());
+
+  app.use(store);
+
+  return {
+    app,
+    Pinia, // 此处必须将 Pinia 返回
+  };
+}
+```
+
+### 基础
+
+```js
+import { defineStore } from "pinia";
+
+export const useStore = defineStore("main", {
+  state() {
+    return {
+      someState: "hello pinia",
+    };
+  },
+  unistorage: true, // 开启后对 state 的数据读写都将持久化
+});
+```
+
+或者 `setup` 语法也是支持的
+
+```js
+import { defineStore } from "pinia";
+
+export const useStore = defineStore(
+  "main",
+  () => {
+    const someState = ref("hello pinia");
+    return { someState };
+  },
+  {
+    unistorage: true, // 开启后对 state 的数据读写都将持久化
+  },
+);
+```
+
+<br />
+
+### 选项
+
+#### 钩子
+
+```js
+import { defineStore } from "pinia";
+
+export const useStore = defineStore("main", {
+  state() {
+    return {
+      someState: "hello pinia",
+    };
+  },
+  unistorage: {
+    // 初始化恢复前触发
+    beforeRestore(ctx) {},
+    // 初始化恢复后触发
+    afterRestore(ctx) {},
+  },
+});
+```
+
+<br />
+
+#### 序列化
+
+大多数情况下你并不需要了解该选项
+
+```js
+import { defineStore } from "pinia";
+
+export const useStore = defineStore("main", {
+  state() {
+    return {
+      someState: "hello pinia",
+    };
+  },
+  unistorage: {
+    serializer: {
+      // 序列化,默认为 JSON.stringify
+      serialize(v) {
+        return JSON.stringify(v);
+      },
+      // 反序列化,默认为 JSON.parse
+      deserialize(v) {
+        return JSON.parse(v);
+      },
+    },
+  },
+});
+```
+
+<br />
+
+#### 其他
+
+```js
+import { defineStore } from "pinia";
+
+export const useStore = defineStore("main", {
+  state() {
+    return {
+      foo: "foo",
+      nested: {
+        data: "nested pinia",
+      },
+      someState: "hello pinia",
+    };
+  },
+  unistorage: {
+    key: "foo", // 缓存的键,默认为该 store 的 id,这里是 main,
+    paths: ["foo", "nested.data"], // 需要缓存的路径,这里设置 foo 和 nested 下的 data 会被缓存
+  },
+});
+```
+
+<br />
+<br />
+
+## License
+
+Made with [markthree](https://github.com/markthree)
+
+Published under [MIT License](./LICENSE).

+ 28 - 0
src/uni_modules/pinia-plugin-unistorage/src/filter.ts

@@ -0,0 +1,28 @@
+import type { StateTree } from "pinia";
+
+function get(state: StateTree, path: Array<string>) {
+  return path.reduce((obj, p) => {
+    return obj?.[p];
+  }, state);
+}
+
+function set(state: StateTree, path: Array<string>, val: unknown): StateTree {
+  return (
+    (path.slice(0, -1).reduce((obj, p) => {
+      if (!/^(__proto__)$/.test(p)) {
+        return (obj[p] = obj[p] || {});
+      } else return {};
+    }, state)[path[path.length - 1]] = val), state
+  );
+}
+
+export function pick(baseState: StateTree, paths: string[]): StateTree {
+  return paths.reduce((substate, path) => {
+    const pathArray = path.split(".");
+    return set(
+      substate,
+      pathArray,
+      get(baseState, pathArray),
+    );
+  }, {});
+}

+ 65 - 0
src/uni_modules/pinia-plugin-unistorage/src/index.ts

@@ -0,0 +1,65 @@
+import { pick } from "./filter";
+import type { PiniaPlugin } from "pinia";
+import { normalizeOptions } from "./normalize";
+import { PersistedStateFactoryOptions } from "./types";
+
+function passage(key: string) {
+  return key;
+}
+
+export function createUnistorage(
+  globalOptions: PersistedStateFactoryOptions = {},
+): PiniaPlugin {
+  const { key: normalizeKey = passage } = globalOptions || {};
+  if (globalOptions?.key) {
+    delete globalOptions.key;
+  }
+  return function (ctx) {
+    {
+      const { store, options } = ctx;
+      // @ts-ignore
+      let { unistorage } = options || {};
+
+      if (!unistorage) return;
+
+      const {
+        paths = null,
+        afterRestore,
+        beforeRestore,
+        serializer = {
+          serialize: JSON.stringify,
+          deserialize: JSON.parse,
+        },
+        key = store.$id,
+      } = normalizeOptions(unistorage, globalOptions);
+
+      beforeRestore?.(ctx);
+
+      const normalizedKey = normalizeKey(key);
+
+      try {
+        // @ts-ignore
+        const fromStorage = uni.getStorageSync(normalizedKey);
+        if (fromStorage) {
+          store.$patch(serializer.deserialize(fromStorage));
+        }
+      } catch (_error) {}
+
+      afterRestore?.(ctx);
+
+      store.$subscribe(
+        (_, state) => {
+          try {
+            const toStore = Array.isArray(paths) ? pick(state, paths) : state;
+            // @ts-ignore
+            uni.setStorageSync(
+              normalizedKey,
+              serializer.serialize(toStore),
+            );
+          } catch (_error) {}
+        },
+        { detached: true },
+      );
+    }
+  };
+}

+ 23 - 0
src/uni_modules/pinia-plugin-unistorage/src/normalize.ts

@@ -0,0 +1,23 @@
+import type {
+  PersistedStateFactoryOptions,
+  PersistedStateOptions,
+} from "./types";
+
+const isObject = (v: unknown): v is Object =>
+  typeof v === "object" && v !== null;
+
+export const normalizeOptions = (
+  options: boolean | PersistedStateOptions | undefined,
+  globalOptions: PersistedStateFactoryOptions,
+): PersistedStateOptions => {
+  options = isObject(options) ? options : Object.create(null);
+
+  return new Proxy(options as object, {
+    get(t, p, r) {
+      return (
+        Reflect.get(t, p, r) ||
+        Reflect.get(globalOptions, p, r)
+      );
+    },
+  });
+};

+ 69 - 0
src/uni_modules/pinia-plugin-unistorage/src/types.ts

@@ -0,0 +1,69 @@
+import type { PiniaPluginContext, StateTree } from "pinia";
+
+type Prettify<T> = { [K in keyof T]: T[K] };
+
+export type StorageLike = Pick<Storage, "getItem" | "setItem">;
+
+export interface Serializer {
+  /**
+   * Serializes state into string before storing
+   * @default JSON.stringify
+   */
+  serialize: (value: StateTree) => string;
+
+  /**
+   * Deserializes string into state before hydrating
+   * @default JSON.parse
+   */
+  deserialize: (value: string) => StateTree;
+}
+
+export interface PersistedStateOptions {
+  /**
+   * Storage key to use.
+   * @default $store.id
+   */
+  key?: string;
+
+  /**
+   * Dot-notation paths to partially save state. Saves everything if undefined.
+   * @default undefined
+   */
+  paths?: Array<string>;
+
+  /**
+   * Customer serializer to serialize/deserialize state.
+   */
+  serializer?: Serializer;
+
+  /**
+   * Hook called before state is hydrated from storage.
+   * @default null
+   */
+  beforeRestore?: (context: PiniaPluginContext) => void;
+
+  /**
+   * Hook called after state is hydrated from storage.
+   * @default undefined
+   */
+  afterRestore?: (context: PiniaPluginContext) => void;
+}
+
+export type PersistedStateFactoryOptions = Prettify<
+  Pick<
+    PersistedStateOptions,
+    "serializer" | "afterRestore" | "beforeRestore"
+  > & {
+    /**
+     * Global key generator, allows pre/postfixing store keys.
+     * @default storeKey => storeKey
+     */
+    key?: (storeKey: string) => string;
+  }
+>;
+
+declare module "pinia" {
+  export interface DefineStoreOptionsBase<S extends StateTree, Store> {
+    unistorage?: boolean | PersistedStateOptions;
+  }
+}