user.ts 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422
  1. import type { UserInfo, LoginInfo } from '/#/store';
  2. import type { ErrorMessageMode } from '/#/axios';
  3. import { defineStore } from 'pinia';
  4. import { store } from '/@/store';
  5. import { PageEnum } from '/@/enums/pageEnum';
  6. import { ROLES_KEY, TOKEN_KEY, USER_INFO_KEY, LOGIN_INFO_KEY, DB_DICT_DATA_KEY, TENANT_ID, PWD_KEY } from '/@/enums/cacheEnum';
  7. import { getAuthCache, setAuthCache } from '/@/utils/auth';
  8. import { AutoLoginParams, GetUserInfoModel, LoginParams, ThirdLoginParams } from '/@/api/sys/model/userModel';
  9. import { autoLoginApi, doLogout, getUserInfo, loginApi, phoneLoginApi, thirdLogin } from '/@/api/sys/user';
  10. import { useI18n } from '/@/hooks/web/useI18n';
  11. import { useMessage } from '/@/hooks/web/useMessage';
  12. import { router } from '/@/router';
  13. import { usePermissionStore } from '/@/store/modules/permission';
  14. import { RouteRecordRaw } from 'vue-router';
  15. import { PAGE_NOT_FOUND_ROUTE, QIANKUN_ROUTE } from '/@/router/routes/basic';
  16. import { isArray } from '/@/utils/is';
  17. import { useGlobSetting } from '/@/hooks/setting';
  18. import { JDragConfigEnum } from '/@/enums/jeecgEnum';
  19. import { RoleEnum } from '/@/enums/roleEnum';
  20. import { useSso } from '/@/hooks/web/useSso';
  21. import { getActions } from '/@/qiankun/state';
  22. import { MOCK_LOGIN_PASSWORD, MOCK_LOGIN_UESRNAME } from '../constant';
  23. interface UserState {
  24. userInfo: Nullable<UserInfo>;
  25. token?: string;
  26. roleList: RoleEnum[];
  27. dictItems?: [];
  28. sessionTimeout?: boolean;
  29. lastUpdateTime: number;
  30. tenantid?: string | number;
  31. shareTenantId?: Nullable<string | number>;
  32. loginInfo?: Nullable<LoginInfo>;
  33. }
  34. export const useUserStore = defineStore({
  35. id: 'app-user',
  36. state: (): UserState => ({
  37. // 用户信息
  38. userInfo: null,
  39. // token
  40. token: undefined,
  41. // 角色列表
  42. roleList: [],
  43. // 字典
  44. dictItems: [],
  45. // session过期时间
  46. sessionTimeout: false,
  47. // Last fetch time
  48. lastUpdateTime: 0,
  49. //租户id
  50. tenantid: '',
  51. // 分享租户ID
  52. // 用于分享页面所属租户与当前用户登录租户不一致的情况
  53. shareTenantId: null,
  54. //登录返回信息
  55. loginInfo: null,
  56. }),
  57. getters: {
  58. getUserInfo(): UserInfo {
  59. return this.userInfo || getAuthCache<UserInfo>(USER_INFO_KEY) || {};
  60. },
  61. getLoginInfo(): LoginInfo {
  62. return this.loginInfo || getAuthCache<LoginInfo>(LOGIN_INFO_KEY) || {};
  63. },
  64. getToken(): string {
  65. return this.token || getAuthCache<string>(TOKEN_KEY);
  66. },
  67. getAllDictItems(): [] {
  68. return this.dictItems || getAuthCache(DB_DICT_DATA_KEY);
  69. },
  70. getRoleList(): RoleEnum[] {
  71. return this.roleList.length > 0 ? this.roleList : getAuthCache<RoleEnum[]>(ROLES_KEY);
  72. },
  73. getSessionTimeout(): boolean {
  74. return !!this.sessionTimeout;
  75. },
  76. getLastUpdateTime(): number {
  77. return this.lastUpdateTime;
  78. },
  79. getTenant(): string | number {
  80. return this.tenantid || getAuthCache<string | number>(TENANT_ID);
  81. },
  82. // 是否有分享租户id
  83. hasShareTenantId(): boolean {
  84. return this.shareTenantId != null && this.shareTenantId !== '';
  85. },
  86. // 目前用户角色列表为空数组,所以使用既定的用户名判断
  87. getIsMockLogin(): boolean {
  88. return this.getUserInfo.username === MOCK_LOGIN_UESRNAME;
  89. },
  90. },
  91. actions: {
  92. /** 设置用户密码并加密,理论上加密、解密的工作应仅在此模块进行 */
  93. setPassword(password: string) {
  94. // setAuthCache(PWD_KEY, AES.encrypt(password, PWD_KEY));
  95. setAuthCache(PWD_KEY, btoa(password));
  96. },
  97. setToken(info: string | undefined) {
  98. this.token = info ? info : ''; // for null or undefined value
  99. setAuthCache(TOKEN_KEY, info);
  100. },
  101. setRoleList(roleList: RoleEnum[]) {
  102. this.roleList = roleList;
  103. setAuthCache(ROLES_KEY, roleList);
  104. },
  105. setUserInfo(info: UserInfo | null) {
  106. this.userInfo = info;
  107. this.lastUpdateTime = new Date().getTime();
  108. setAuthCache(USER_INFO_KEY, info);
  109. },
  110. setLoginInfo(info: LoginInfo | null) {
  111. this.loginInfo = info;
  112. setAuthCache(LOGIN_INFO_KEY, info);
  113. },
  114. setAllDictItems(dictItems) {
  115. this.dictItems = dictItems;
  116. setAuthCache(DB_DICT_DATA_KEY, dictItems);
  117. },
  118. setTenant(id) {
  119. this.tenantid = id;
  120. setAuthCache(TENANT_ID, id);
  121. },
  122. setShareTenantId(id: NonNullable<typeof this.shareTenantId>) {
  123. this.shareTenantId = id;
  124. },
  125. setSessionTimeout(flag: boolean) {
  126. this.sessionTimeout = flag;
  127. },
  128. resetState() {
  129. this.userInfo = null;
  130. this.dictItems = [];
  131. this.token = '';
  132. this.roleList = [];
  133. this.sessionTimeout = false;
  134. },
  135. /**
  136. * 登录事件
  137. */
  138. async login(
  139. params: LoginParams & {
  140. goHome?: boolean;
  141. mode?: ErrorMessageMode;
  142. }
  143. ): Promise<GetUserInfoModel | null> {
  144. try {
  145. const { goHome = true, mode, ...loginParams } = params;
  146. const data = await loginApi(loginParams, mode);
  147. const { token, userInfo } = data;
  148. // save token
  149. this.setToken(token);
  150. this.setTenant(userInfo.loginTenantId);
  151. return this.afterLoginAction(goHome, data);
  152. } catch (error) {
  153. return Promise.reject(error);
  154. }
  155. },
  156. /**
  157. * 扫码登录事件
  158. */
  159. async qrCodeLogin(token): Promise<GetUserInfoModel | null> {
  160. try {
  161. // save token
  162. this.setToken(token);
  163. return this.afterLoginAction(true, {});
  164. } catch (error) {
  165. return Promise.reject(error);
  166. }
  167. },
  168. /**
  169. * 登录完成处理
  170. * @param goHome
  171. */
  172. async afterLoginAction(goHome?: boolean, data?: any): Promise<any | null> {
  173. const glob = useGlobSetting();
  174. if (!this.getToken) return null;
  175. //获取用户信息
  176. const userInfo = await this.getUserInfoAction();
  177. const sessionTimeout = this.sessionTimeout;
  178. if (sessionTimeout) {
  179. this.setSessionTimeout(false);
  180. } else {
  181. const permissionStore = usePermissionStore();
  182. if (!permissionStore.isDynamicAddedRoute) {
  183. const routes = await permissionStore.buildRoutesAction();
  184. routes.forEach((route) => {
  185. router.addRoute(route as unknown as RouteRecordRaw);
  186. });
  187. router.addRoute(PAGE_NOT_FOUND_ROUTE as unknown as RouteRecordRaw);
  188. router.addRoute(QIANKUN_ROUTE as unknown as RouteRecordRaw);
  189. permissionStore.setDynamicAddedRoute(true);
  190. }
  191. this.setLoginInfo({ ...data, isLogin: true });
  192. //update-begin-author:liusq date:2022-5-5 for:登录成功后缓存拖拽模块的接口前缀
  193. localStorage.setItem(JDragConfigEnum.DRAG_BASE_URL, useGlobSetting().domainUrl);
  194. //update-end-author:liusq date:2022-5-5 for: 登录成功后缓存拖拽模块的接口前缀
  195. goHome && (await router.replace((userInfo && userInfo.homePath) || glob.homePath || PageEnum.BASE_HOME));
  196. // update-begin-author:sunjianlei date:20230306 for: 修复登录成功后,没有正确重定向的问题
  197. const redirect = router.currentRoute.value?.query?.redirect as string;
  198. // 判断是否有 redirect 重定向地址
  199. //update-begin---author:wangshuai ---date:20230424 for:【QQYUN-5195】登录之后直接刷新页面导致没有进入创建组织页面------------
  200. if (redirect && goHome) {
  201. //update-end---author:wangshuai ---date:20230424 for:【QQYUN-5195】登录之后直接刷新页面导致没有进入创建组织页面------------
  202. // 当前页面打开
  203. window.open(redirect, '_self');
  204. return data;
  205. }
  206. // update-end-author:sunjianlei date:20230306 for: 修复登录成功后,没有正确重定向的问题
  207. goHome && (await router.replace((userInfo && userInfo.homePath) || glob.homePath || PageEnum.BASE_HOME));
  208. }
  209. if (useGlobSetting().openQianKun) {
  210. const actions = getActions();
  211. actions.setGlobalState({ token: this.getToken, userInfo: userInfo, isMounted: false });
  212. }
  213. return data;
  214. },
  215. /**
  216. * 手机号登录
  217. * @param params
  218. */
  219. async phoneLogin(
  220. params: LoginParams & {
  221. goHome?: boolean;
  222. mode?: ErrorMessageMode;
  223. }
  224. ): Promise<GetUserInfoModel | null> {
  225. try {
  226. const { goHome = true, mode, ...loginParams } = params;
  227. const data = await phoneLoginApi(loginParams, mode);
  228. const { token } = data;
  229. // save token
  230. this.setToken(token);
  231. return this.afterLoginAction(goHome, data);
  232. } catch (error) {
  233. return Promise.reject(error);
  234. }
  235. },
  236. /**
  237. * 获取用户信息
  238. */
  239. async getUserInfoAction(): Promise<UserInfo | null> {
  240. if (!this.getToken) {
  241. return null;
  242. }
  243. const { userInfo, sysAllDictItems } = await getUserInfo();
  244. if (userInfo) {
  245. const { roles = [] } = userInfo;
  246. if (isArray(roles)) {
  247. const roleList = roles.map((item) => item.value) as RoleEnum[];
  248. this.setRoleList(roleList);
  249. } else {
  250. userInfo.roles = [];
  251. this.setRoleList([]);
  252. }
  253. this.setUserInfo(userInfo);
  254. }
  255. /**
  256. * 添加字典信息到缓存
  257. * @updateBy:lsq
  258. * @updateDate:2021-09-08
  259. */
  260. if (sysAllDictItems) {
  261. this.setAllDictItems(sysAllDictItems);
  262. }
  263. return userInfo;
  264. },
  265. /**
  266. * 退出登录
  267. */
  268. async logout(goLogin = false) {
  269. if (this.getToken) {
  270. try {
  271. await doLogout();
  272. } catch {
  273. console.log('注销Token失败');
  274. }
  275. }
  276. // //update-begin-author:taoyan date:2022-5-5 for: src/layouts/default/header/index.vue showLoginSelect方法 获取tenantId 退出登录后再次登录依然能获取到值,没有清空
  277. // let username:any = this.userInfo && this.userInfo.username;
  278. // if(username){
  279. // removeAuthCache(username)
  280. // }
  281. // //update-end-author:taoyan date:2022-5-5 for: src/layouts/default/header/index.vue showLoginSelect方法 获取tenantId 退出登录后再次登录依然能获取到值,没有清空
  282. this.setToken('');
  283. setAuthCache(TOKEN_KEY, null);
  284. this.setSessionTimeout(false);
  285. this.setUserInfo(null);
  286. this.setLoginInfo(null);
  287. this.setTenant(null);
  288. //update-begin-author:liusq date:2022-5-5 for:退出登录后清除拖拽模块的接口前缀
  289. localStorage.removeItem(JDragConfigEnum.DRAG_BASE_URL);
  290. //update-end-author:liusq date:2022-5-5 for: 退出登录后清除拖拽模块的接口前缀
  291. //如果开启单点登录,则跳转到单点统一登录中心
  292. const openSso = useGlobSetting().openSso;
  293. if (openSso == 'true') {
  294. await useSso().ssoLoginOut();
  295. }
  296. //update-begin---author:wangshuai ---date:20230224 for:[QQYUN-3440]新建企业微信和钉钉配置表,通过租户模式隔离------------
  297. //退出登录的时候需要用的应用id
  298. // if(isOAuth2AppEnv()){
  299. // let tenantId = getAuthCache(OAUTH2_THIRD_LOGIN_TENANT_ID);
  300. // removeAuthCache(OAUTH2_THIRD_LOGIN_TENANT_ID);
  301. // goLogin && await router.push({ name:"Login",query:{ tenantId:tenantId }})
  302. // }else{
  303. // // update-begin-author:sunjianlei date:20230306 for: 修复登录成功后,没有正确重定向的问题
  304. // goLogin && (await router.push({
  305. // path: PageEnum.BASE_LOGIN,
  306. // query: {
  307. // // 传入当前的路由,登录成功后跳转到当前路由
  308. // redirect: router.currentRoute.value.fullPath,
  309. // }
  310. // }));
  311. // // update-end-author:sunjianlei date:20230306 for: 修复登录成功后,没有正确重定向的问题
  312. // }
  313. goLogin &&
  314. (await router.push({
  315. path: PageEnum.BASE_LOGIN,
  316. query: {
  317. // 传入当前的路由,登录成功后跳转到当前路由
  318. redirect: router.currentRoute.value.fullPath,
  319. },
  320. }));
  321. //update-end---author:wangshuai ---date:20230224 for:[QQYUN-3440]新建企业微信和钉钉配置表,通过租户模式隔离------------
  322. },
  323. /**
  324. * 登录事件
  325. */
  326. async ThirdLogin(
  327. params: ThirdLoginParams & {
  328. goHome?: boolean;
  329. mode?: ErrorMessageMode;
  330. }
  331. ): Promise<any | null> {
  332. try {
  333. const { goHome = true, mode, ...ThirdLoginParams } = params;
  334. const data = await thirdLogin(ThirdLoginParams, mode);
  335. const { token } = data;
  336. // save token
  337. this.setToken(token);
  338. return this.afterLoginAction(goHome, data);
  339. } catch (error) {
  340. return Promise.reject(error);
  341. }
  342. },
  343. /**
  344. * 退出询问
  345. */
  346. confirmLoginOut() {
  347. // debugger;
  348. const { createConfirm } = useMessage();
  349. const { t } = useI18n();
  350. createConfirm({
  351. iconType: 'warning',
  352. title: t('sys.app.logoutTip'),
  353. content: t('sys.app.logoutMessage'),
  354. onOk: async () => {
  355. await this.logout(true);
  356. },
  357. });
  358. },
  359. /** 模拟用户登录行为,使用既定的账号,账号权限将受限 */
  360. async mockLogin(
  361. params: Partial<LoginParams> & {
  362. goHome?: boolean;
  363. mode?: ErrorMessageMode;
  364. } = {}
  365. ) {
  366. try {
  367. const loginParams = {
  368. username: MOCK_LOGIN_UESRNAME,
  369. password: MOCK_LOGIN_PASSWORD,
  370. checkKey: new Date().getTime(),
  371. ...params,
  372. };
  373. return this.login(loginParams);
  374. } catch (error) {
  375. return Promise.reject(error);
  376. }
  377. },
  378. /** 续登录,即登出后再次模拟登录并刷新当前页面,不需要用户重复登录"自动"登录账户 */
  379. async redoMockLogin(
  380. params: Partial<LoginParams> & {
  381. goHome?: boolean;
  382. mode?: ErrorMessageMode;
  383. } = {}
  384. ) {
  385. await this.logout();
  386. await this.mockLogin({
  387. goHome: false,
  388. ...params,
  389. });
  390. router.go(0);
  391. },
  392. /** 用户自动登录,即不需要用户密码即可登录 */
  393. async autoLogin(
  394. params: AutoLoginParams & {
  395. goHome?: boolean;
  396. mode?: ErrorMessageMode;
  397. }
  398. ) {
  399. const { goHome = true, mode, ...loginParams } = params;
  400. const data = await autoLoginApi(loginParams, mode);
  401. const { token, userInfo } = data;
  402. // save token
  403. this.setToken(token);
  404. this.setTenant(userInfo.loginTenantId);
  405. await this.afterLoginAction(goHome, data);
  406. return;
  407. },
  408. },
  409. });
  410. // Need to be used outside the setup
  411. export function useUserStoreWithOut() {
  412. return useUserStore(store);
  413. }