user.ts 7.04 KB
import type { UserInfo, UserUpdateInfo } from '/#/external/store';
import type { ErrorMessageMode } from '/#/external/axios';
import { defineStore } from 'pinia';
import { pinia as store } from '@/store';
import { RoleEnum } from '@/enums/external/roleEnum';
import { JWT_TOKEN_KEY, REFRESH_TOKEN_KEY, ROLES_KEY, SHARE_JWT_TOKEN_KEY, SHARE_REFRESH_TOKEN_KEY, USER_INFO_KEY } from '@/enums/external/cacheEnum';
import { getAuthCache, getJwtToken, getRefreshToken, getUserInfo, setAuthCache } from '@/utils/external/auth';
import {
  LoginParams,
  LoginResultModel,
  RefreshTokenParams,
  SmsLoginParams,
} from '@/api/external/sys/model/userModel';
import { doRefreshToken, getMyInfo, loginApi } from '@/api/external/sys/user';
import router from '@/router';
import { createLocalStorage } from '@/utils/external/cache';
import { useI18n } from 'vue-i18n';
import { useDialog } from 'naive-ui';
import { PageEnum } from '@/enums/external/pageEnum';

interface UserState {
  platInfo: any;
  enterPriseInfo: any;
  userInfo: Nullable<UserInfo>;
  userUpdateInfo?: Nullable<UserUpdateInfo>;
  token?: string;
  roleList: RoleEnum[];
  sessionTimeout?: boolean;
  lastUpdateTime: number;
  jwtToken?: string;
  refreshToken?: string;
  shareJwtToken?: string;
  shareRefreshToken?: string;
  outTarget?: string;
  thirdTokenIsExp?: boolean;
}

const storage = createLocalStorage();
export const useUserStore = defineStore({
  id: 'app-user',
  state: (): UserState => ({
    //平台信息
    platInfo: storage.get('platformInfo') || null,
    enterPriseInfo: storage.get('enterPriseInfo') || null,

    // user info
    userInfo: getUserInfo() || null,
    userUpdateInfo: null,
    // token
    jwtToken: getJwtToken() as string || undefined,
    //refresh Token
    refreshToken: getRefreshToken() as string || undefined,
    shareJwtToken: undefined,
    shareRefreshToken: undefined,
    // roleList
    roleList: [],
    // Whether the login expired
    sessionTimeout: false,
    // Last fetch time
    lastUpdateTime: 0,
    // 用于三方接口Token是否过期
    thirdTokenIsExp: false 
  }),

  getters: {
    getThirdTokenIsExp(): boolean {
      return this.thirdTokenIsExp!;
    },
    getPlatInfo(): any {
      return this.platInfo;
    },
    getUserInfo(): UserInfo {
      return this.userInfo || getAuthCache<UserInfo>(USER_INFO_KEY) || {};
    },

    getUserUpdateInfo(): UserUpdateInfo {
      return this.userUpdateInfo || {};
    },
    getJwtToken(): string {
      return this.jwtToken || getAuthCache<string>(JWT_TOKEN_KEY);
    },
    getRefreshToken(): string {
      return this.refreshToken || getAuthCache<string>(REFRESH_TOKEN_KEY);
    },
    getRoleList(): RoleEnum[] {
      return this.roleList.length > 0 ? this.roleList : getAuthCache<RoleEnum[]>(ROLES_KEY);
    },
    getSessionTimeout(): boolean {
      return !!this.sessionTimeout;
    },
    getLastUpdateTime(): number {
      return this.lastUpdateTime;
    },
  },
  actions: {
    setThirdTokenIsExp(expStatus:boolean) {
      this.thirdTokenIsExp = expStatus
    },
    setPlatInfo(platInfo: any) {
      this.platInfo = platInfo;
    },
    setEnterPriseInfo(enterPriseInfo: any) {
      this.enterPriseInfo = enterPriseInfo;
    },

    storeToken(jwtToken: string, refreshToken: string) {
      this.jwtToken = jwtToken;
      this.refreshToken = refreshToken;
      setAuthCache(JWT_TOKEN_KEY, jwtToken);
      setAuthCache(REFRESH_TOKEN_KEY, refreshToken);
    },
    storeShareToken(jwtToken: string, refreshToken: string) {
      this.shareJwtToken = jwtToken;
      this.shareRefreshToken = refreshToken;
      setAuthCache(SHARE_JWT_TOKEN_KEY, jwtToken);
      setAuthCache(SHARE_REFRESH_TOKEN_KEY, refreshToken);
    },
    setToken(info: string | undefined) {
      this.jwtToken = info;
      setAuthCache(JWT_TOKEN_KEY, info);
    },
    setRoleList(roleList: RoleEnum[]) {
      this.roleList = roleList;
      setAuthCache(ROLES_KEY, roleList);
    },
    setUserInfo(info: UserInfo | null) {
      this.userInfo = info;
      this.lastUpdateTime = new Date().getTime();
      setAuthCache(USER_INFO_KEY, info);
    },
    setUserUpdateInfo(info: UserUpdateInfo) {
      this.userUpdateInfo = info;
    },
    setSessionTimeout(flag: boolean) {
      this.sessionTimeout = flag;
    },
    resetState() {
      this.userInfo = null;
      this.token = '';
      this.roleList = [];
      this.sessionTimeout = false;
    },
    /**
     * @description: login
     */
    async login(
      params: LoginParams & {
        goHome?: boolean;
        mode?: ErrorMessageMode;
      }
    ) {
      try {
        const { goHome = true, mode, ...loginParams } = params;
        const data = await loginApi(loginParams, mode);
        return this.process(data, goHome);
      } catch (error) {
        return Promise.reject(error);
      }
    },
    async process(data: LoginResultModel, goHome?: boolean) {
      const { token, refreshToken } = data;
      this.storeToken(token, refreshToken);
      // get user info
      const userInfo = await this.getMyUserInfoAction();

      const sessionTimeout = this.sessionTimeout;
      if (sessionTimeout) {
        this.setSessionTimeout(false);
      } else if (goHome) {
        await router.replace(userInfo.homePath || PageEnum.BASE_HOME);
      }
      return userInfo;
    },
    async getMyUserInfoAction() {
      const userInfo = await getMyInfo();
      this.setUserInfo(userInfo);
      const { roles } = userInfo;
      const roleList = roles.map((item) => item) as RoleEnum[];
      this.setRoleList(roleList);
      return userInfo;
    },
    /**
     * @description: logout
     */
    async logout(goLogin = false) {
      this.setToken(undefined);
      this.setSessionTimeout(false);
      setAuthCache(REFRESH_TOKEN_KEY, undefined);
      this.setUserInfo(null);
      window.localStorage.clear();
      window.localStorage.removeItem('updateUserInfo');
      goLogin && router.push(PageEnum.BASE_LOGIN);
    },

    async doRefresh() {
      try {
        const req = { refreshToken: this.refreshToken } as RefreshTokenParams;
        const data = await doRefreshToken(req);
        const { token, refreshToken } = data;
        this.storeToken(token, refreshToken);
      } catch (error) {
        this.logout();
      }
    },

    /**
     * @description 刷新分享页面的token
     */
    async doShareRefresh() {
      try {
        const req = { refreshToken: this.shareRefreshToken } as RefreshTokenParams
        const { token, refreshToken } = await doRefreshToken(req)
        this.storeShareToken(token, refreshToken)
      } catch (error) {
        window.location.reload()
        throw error
      }
    },

    /**
     * @description: Confirm before logging out
     */
    confirmLoginOut() {
      const { warning } = useDialog();
      const { t } = useI18n();
      warning({
        type: 'warning',
        title: '温馨提醒',
        content: '是否确认退出系统?',
        onPositiveClick: async () => {
          await this.logout(true);
        },
      });
    },
  },
});

// Need to be used outside the setup
export function useUserStoreWithOut() {
  return useUserStore(store);
}