mirror of
				https://github.com/pure-admin/vue-pure-admin.git
				synced 2025-11-03 13:44:47 +08:00 
			
		
		
		
	feat: 支持多标签页打开已经登录的系统后无需再登录并添加7天内免登录功能 (#747)
				
					
				
			* feat: 支持多标签页打开已经登录的系统后无需再登录 * feat: 添加`7`天内免登录功能
This commit is contained in:
		
							parent
							
								
									be2de405ab
								
							
						
					
					
						commit
						7e7b6fee7a
					
				@ -116,7 +116,8 @@ login:
 | 
				
			|||||||
  username: Username
 | 
					  username: Username
 | 
				
			||||||
  password: Password
 | 
					  password: Password
 | 
				
			||||||
  verifyCode: VerifyCode
 | 
					  verifyCode: VerifyCode
 | 
				
			||||||
  remember: Remember Password
 | 
					  remember: No need to login for 7 days
 | 
				
			||||||
 | 
					  rememberInfo: After checking and logging in, you will automatically log in to the system without entering your username and password within 7 days
 | 
				
			||||||
  sure: Sure Password
 | 
					  sure: Sure Password
 | 
				
			||||||
  forget: Forget Password?
 | 
					  forget: Forget Password?
 | 
				
			||||||
  login: Login
 | 
					  login: Login
 | 
				
			||||||
 | 
				
			|||||||
@ -116,7 +116,8 @@ login:
 | 
				
			|||||||
  username: 账号
 | 
					  username: 账号
 | 
				
			||||||
  password: 密码
 | 
					  password: 密码
 | 
				
			||||||
  verifyCode: 验证码
 | 
					  verifyCode: 验证码
 | 
				
			||||||
  remember: 记住密码
 | 
					  remember: 7天内免登录
 | 
				
			||||||
 | 
					  rememberInfo: 勾选并登录后,7天内无需输入用户名和密码会自动登入系统
 | 
				
			||||||
  sure: 确认密码
 | 
					  sure: 确认密码
 | 
				
			||||||
  forget: 忘记密码?
 | 
					  forget: 忘记密码?
 | 
				
			||||||
  login: 登录
 | 
					  login: 登录
 | 
				
			||||||
 | 
				
			|||||||
@ -140,7 +140,7 @@
 | 
				
			|||||||
    "prettier": "^3.0.3",
 | 
					    "prettier": "^3.0.3",
 | 
				
			||||||
    "rimraf": "^5.0.5",
 | 
					    "rimraf": "^5.0.5",
 | 
				
			||||||
    "rollup-plugin-visualizer": "^5.9.2",
 | 
					    "rollup-plugin-visualizer": "^5.9.2",
 | 
				
			||||||
    "sass": "^1.68.0",
 | 
					    "sass": "^1.69.0",
 | 
				
			||||||
    "sass-loader": "^13.3.2",
 | 
					    "sass-loader": "^13.3.2",
 | 
				
			||||||
    "stylelint": "^15.10.3",
 | 
					    "stylelint": "^15.10.3",
 | 
				
			||||||
    "stylelint-config-html": "^1.1.0",
 | 
					    "stylelint-config-html": "^1.1.0",
 | 
				
			||||||
@ -157,7 +157,7 @@
 | 
				
			|||||||
    "tailwindcss": "^3.3.3",
 | 
					    "tailwindcss": "^3.3.3",
 | 
				
			||||||
    "terser": "^5.21.0",
 | 
					    "terser": "^5.21.0",
 | 
				
			||||||
    "typescript": "^5.2.2",
 | 
					    "typescript": "^5.2.2",
 | 
				
			||||||
    "vite": "^4.4.10",
 | 
					    "vite": "^4.4.11",
 | 
				
			||||||
    "vite-plugin-cdn-import": "^0.3.5",
 | 
					    "vite-plugin-cdn-import": "^0.3.5",
 | 
				
			||||||
    "vite-plugin-compression": "^0.5.1",
 | 
					    "vite-plugin-compression": "^0.5.1",
 | 
				
			||||||
    "vite-plugin-mock": "2.9.6",
 | 
					    "vite-plugin-mock": "2.9.6",
 | 
				
			||||||
 | 
				
			|||||||
							
								
								
									
										80
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										80
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							@ -73,7 +73,7 @@ specifiers:
 | 
				
			|||||||
  responsive-storage: ^2.2.0
 | 
					  responsive-storage: ^2.2.0
 | 
				
			||||||
  rimraf: ^5.0.5
 | 
					  rimraf: ^5.0.5
 | 
				
			||||||
  rollup-plugin-visualizer: ^5.9.2
 | 
					  rollup-plugin-visualizer: ^5.9.2
 | 
				
			||||||
  sass: ^1.68.0
 | 
					  sass: ^1.69.0
 | 
				
			||||||
  sass-loader: ^13.3.2
 | 
					  sass-loader: ^13.3.2
 | 
				
			||||||
  sortablejs: ^1.15.0
 | 
					  sortablejs: ^1.15.0
 | 
				
			||||||
  stylelint: ^15.10.3
 | 
					  stylelint: ^15.10.3
 | 
				
			||||||
@ -96,7 +96,7 @@ specifiers:
 | 
				
			|||||||
  v-contextmenu: 3.0.0
 | 
					  v-contextmenu: 3.0.0
 | 
				
			||||||
  v3-infinite-loading: ^1.3.1
 | 
					  v3-infinite-loading: ^1.3.1
 | 
				
			||||||
  version-rocket: ^1.7.0
 | 
					  version-rocket: ^1.7.0
 | 
				
			||||||
  vite: ^4.4.10
 | 
					  vite: ^4.4.11
 | 
				
			||||||
  vite-plugin-cdn-import: ^0.3.5
 | 
					  vite-plugin-cdn-import: ^0.3.5
 | 
				
			||||||
  vite-plugin-compression: ^0.5.1
 | 
					  vite-plugin-compression: ^0.5.1
 | 
				
			||||||
  vite-plugin-mock: 2.9.6
 | 
					  vite-plugin-mock: 2.9.6
 | 
				
			||||||
@ -194,8 +194,8 @@ devDependencies:
 | 
				
			|||||||
  '@types/sortablejs': 1.15.3
 | 
					  '@types/sortablejs': 1.15.3
 | 
				
			||||||
  '@typescript-eslint/eslint-plugin': 6.7.4_sjhwt3bl5psuxqi3hx6z7r6ola
 | 
					  '@typescript-eslint/eslint-plugin': 6.7.4_sjhwt3bl5psuxqi3hx6z7r6ola
 | 
				
			||||||
  '@typescript-eslint/parser': 6.7.4_jk7qbkaijtltyu4ajmze3dfiwa
 | 
					  '@typescript-eslint/parser': 6.7.4_jk7qbkaijtltyu4ajmze3dfiwa
 | 
				
			||||||
  '@vitejs/plugin-vue': 4.4.0_vite@4.4.10+vue@3.3.4
 | 
					  '@vitejs/plugin-vue': 4.4.0_vite@4.4.11+vue@3.3.4
 | 
				
			||||||
  '@vitejs/plugin-vue-jsx': 3.0.2_vite@4.4.10+vue@3.3.4
 | 
					  '@vitejs/plugin-vue-jsx': 3.0.2_vite@4.4.11+vue@3.3.4
 | 
				
			||||||
  '@vue/eslint-config-prettier': 8.0.0_rj7fo27gtcc4oitmthuutitbrm
 | 
					  '@vue/eslint-config-prettier': 8.0.0_rj7fo27gtcc4oitmthuutitbrm
 | 
				
			||||||
  '@vue/eslint-config-typescript': 12.0.0_ljkbukdqy6rudcxzcb5p2o2hbq
 | 
					  '@vue/eslint-config-typescript': 12.0.0_ljkbukdqy6rudcxzcb5p2o2hbq
 | 
				
			||||||
  autoprefixer: 10.4.16_postcss@8.4.31
 | 
					  autoprefixer: 10.4.16_postcss@8.4.31
 | 
				
			||||||
@ -214,8 +214,8 @@ devDependencies:
 | 
				
			|||||||
  prettier: 3.0.3
 | 
					  prettier: 3.0.3
 | 
				
			||||||
  rimraf: 5.0.5
 | 
					  rimraf: 5.0.5
 | 
				
			||||||
  rollup-plugin-visualizer: 5.9.2
 | 
					  rollup-plugin-visualizer: 5.9.2
 | 
				
			||||||
  sass: 1.68.0
 | 
					  sass: 1.69.0
 | 
				
			||||||
  sass-loader: 13.3.2_sass@1.68.0
 | 
					  sass-loader: 13.3.2_sass@1.69.0
 | 
				
			||||||
  stylelint: 15.10.3_typescript@5.2.2
 | 
					  stylelint: 15.10.3_typescript@5.2.2
 | 
				
			||||||
  stylelint-config-html: 1.1.0_a6l2rvr7enkswjarqif24xxgi4
 | 
					  stylelint-config-html: 1.1.0_a6l2rvr7enkswjarqif24xxgi4
 | 
				
			||||||
  stylelint-config-recess-order: 4.3.0_stylelint@15.10.3
 | 
					  stylelint-config-recess-order: 4.3.0_stylelint@15.10.3
 | 
				
			||||||
@ -231,10 +231,10 @@ devDependencies:
 | 
				
			|||||||
  tailwindcss: 3.3.3
 | 
					  tailwindcss: 3.3.3
 | 
				
			||||||
  terser: 5.21.0
 | 
					  terser: 5.21.0
 | 
				
			||||||
  typescript: 5.2.2
 | 
					  typescript: 5.2.2
 | 
				
			||||||
  vite: 4.4.10_aoxrcfqgusexnpex5mio6763sm
 | 
					  vite: 4.4.11_e5w4bvq32mzkrz2cg5gbeogbay
 | 
				
			||||||
  vite-plugin-cdn-import: 0.3.5
 | 
					  vite-plugin-cdn-import: 0.3.5
 | 
				
			||||||
  vite-plugin-compression: 0.5.1_vite@4.4.10
 | 
					  vite-plugin-compression: 0.5.1_vite@4.4.11
 | 
				
			||||||
  vite-plugin-mock: 2.9.6_mockjs@1.1.0+vite@4.4.10
 | 
					  vite-plugin-mock: 2.9.6_mockjs@1.1.0+vite@4.4.11
 | 
				
			||||||
  vite-plugin-remove-console: 2.1.1
 | 
					  vite-plugin-remove-console: 2.1.1
 | 
				
			||||||
  vite-svg-loader: 4.0.0
 | 
					  vite-svg-loader: 4.0.0
 | 
				
			||||||
  vue-eslint-parser: 9.3.1_eslint@8.50.0
 | 
					  vue-eslint-parser: 9.3.1_eslint@8.50.0
 | 
				
			||||||
@ -1095,7 +1095,7 @@ packages:
 | 
				
			|||||||
      ajv: 6.12.6
 | 
					      ajv: 6.12.6
 | 
				
			||||||
      debug: 4.3.4
 | 
					      debug: 4.3.4
 | 
				
			||||||
      espree: 9.6.1
 | 
					      espree: 9.6.1
 | 
				
			||||||
      globals: 13.22.0
 | 
					      globals: 13.23.0
 | 
				
			||||||
      ignore: 5.2.4
 | 
					      ignore: 5.2.4
 | 
				
			||||||
      import-fresh: 3.3.0
 | 
					      import-fresh: 3.3.0
 | 
				
			||||||
      js-yaml: 4.1.0
 | 
					      js-yaml: 4.1.0
 | 
				
			||||||
@ -1241,7 +1241,7 @@ packages:
 | 
				
			|||||||
    dependencies:
 | 
					    dependencies:
 | 
				
			||||||
      '@intlify/bundle-utils': 7.4.0_vue-i18n@9.5.0
 | 
					      '@intlify/bundle-utils': 7.4.0_vue-i18n@9.5.0
 | 
				
			||||||
      '@intlify/shared': 9.5.0
 | 
					      '@intlify/shared': 9.5.0
 | 
				
			||||||
      '@rollup/pluginutils': 5.0.4
 | 
					      '@rollup/pluginutils': 5.0.5
 | 
				
			||||||
      '@vue/compiler-sfc': 3.3.4
 | 
					      '@vue/compiler-sfc': 3.3.4
 | 
				
			||||||
      debug: 4.3.4
 | 
					      debug: 4.3.4
 | 
				
			||||||
      fast-glob: 3.3.1
 | 
					      fast-glob: 3.3.1
 | 
				
			||||||
@ -1716,11 +1716,11 @@ packages:
 | 
				
			|||||||
      picomatch: 2.3.1
 | 
					      picomatch: 2.3.1
 | 
				
			||||||
    dev: true
 | 
					    dev: true
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  /@rollup/pluginutils/5.0.4:
 | 
					  /@rollup/pluginutils/5.0.5:
 | 
				
			||||||
    resolution: {integrity: sha512-0KJnIoRI8A+a1dqOYLxH8vBf8bphDmty5QvIm2hqm7oFCFYKCAZWWd2hXgMibaPsNDhI0AtpYfQZJG47pt/k4g==}
 | 
					    resolution: {integrity: sha512-6aEYR910NyP73oHiJglti74iRyOwgFU4x3meH/H8OJx6Ry0j6cOVZ5X/wTvub7G7Ao6qaHBEaNsV3GLJkSsF+Q==}
 | 
				
			||||||
    engines: {node: '>=14.0.0'}
 | 
					    engines: {node: '>=14.0.0'}
 | 
				
			||||||
    peerDependencies:
 | 
					    peerDependencies:
 | 
				
			||||||
      rollup: ^1.20.0||^2.0.0||^3.0.0
 | 
					      rollup: ^1.20.0||^2.0.0||^3.0.0||^4.0.0
 | 
				
			||||||
    peerDependenciesMeta:
 | 
					    peerDependenciesMeta:
 | 
				
			||||||
      rollup:
 | 
					      rollup:
 | 
				
			||||||
        optional: true
 | 
					        optional: true
 | 
				
			||||||
@ -2108,7 +2108,7 @@ packages:
 | 
				
			|||||||
      nanoid: 3.3.6
 | 
					      nanoid: 3.3.6
 | 
				
			||||||
    dev: false
 | 
					    dev: false
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  /@vitejs/plugin-vue-jsx/3.0.2_vite@4.4.10+vue@3.3.4:
 | 
					  /@vitejs/plugin-vue-jsx/3.0.2_vite@4.4.11+vue@3.3.4:
 | 
				
			||||||
    resolution: {integrity: sha512-obF26P2Z4Ogy3cPp07B4VaW6rpiu0ue4OT2Y15UxT5BZZ76haUY9guOsZV3uWh/I6xc+VeiW+ZVabRE82FyzWw==}
 | 
					    resolution: {integrity: sha512-obF26P2Z4Ogy3cPp07B4VaW6rpiu0ue4OT2Y15UxT5BZZ76haUY9guOsZV3uWh/I6xc+VeiW+ZVabRE82FyzWw==}
 | 
				
			||||||
    engines: {node: ^14.18.0 || >=16.0.0}
 | 
					    engines: {node: ^14.18.0 || >=16.0.0}
 | 
				
			||||||
    peerDependencies:
 | 
					    peerDependencies:
 | 
				
			||||||
@ -2118,20 +2118,20 @@ packages:
 | 
				
			|||||||
      '@babel/core': 7.23.0
 | 
					      '@babel/core': 7.23.0
 | 
				
			||||||
      '@babel/plugin-transform-typescript': 7.22.15_@babel+core@7.23.0
 | 
					      '@babel/plugin-transform-typescript': 7.22.15_@babel+core@7.23.0
 | 
				
			||||||
      '@vue/babel-plugin-jsx': 1.1.5_@babel+core@7.23.0
 | 
					      '@vue/babel-plugin-jsx': 1.1.5_@babel+core@7.23.0
 | 
				
			||||||
      vite: 4.4.10_aoxrcfqgusexnpex5mio6763sm
 | 
					      vite: 4.4.11_e5w4bvq32mzkrz2cg5gbeogbay
 | 
				
			||||||
      vue: 3.3.4
 | 
					      vue: 3.3.4
 | 
				
			||||||
    transitivePeerDependencies:
 | 
					    transitivePeerDependencies:
 | 
				
			||||||
      - supports-color
 | 
					      - supports-color
 | 
				
			||||||
    dev: true
 | 
					    dev: true
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  /@vitejs/plugin-vue/4.4.0_vite@4.4.10+vue@3.3.4:
 | 
					  /@vitejs/plugin-vue/4.4.0_vite@4.4.11+vue@3.3.4:
 | 
				
			||||||
    resolution: {integrity: sha512-xdguqb+VUwiRpSg+nsc2HtbAUSGak25DXYvpQQi4RVU1Xq1uworyoH/md9Rfd8zMmPR/pSghr309QNcftUVseg==}
 | 
					    resolution: {integrity: sha512-xdguqb+VUwiRpSg+nsc2HtbAUSGak25DXYvpQQi4RVU1Xq1uworyoH/md9Rfd8zMmPR/pSghr309QNcftUVseg==}
 | 
				
			||||||
    engines: {node: ^14.18.0 || >=16.0.0}
 | 
					    engines: {node: ^14.18.0 || >=16.0.0}
 | 
				
			||||||
    peerDependencies:
 | 
					    peerDependencies:
 | 
				
			||||||
      vite: ^4.0.0
 | 
					      vite: ^4.0.0
 | 
				
			||||||
      vue: ^3.2.25
 | 
					      vue: ^3.2.25
 | 
				
			||||||
    dependencies:
 | 
					    dependencies:
 | 
				
			||||||
      vite: 4.4.10_aoxrcfqgusexnpex5mio6763sm
 | 
					      vite: 4.4.11_e5w4bvq32mzkrz2cg5gbeogbay
 | 
				
			||||||
      vue: 3.3.4
 | 
					      vue: 3.3.4
 | 
				
			||||||
    dev: true
 | 
					    dev: true
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -2950,7 +2950,7 @@ packages:
 | 
				
			|||||||
    hasBin: true
 | 
					    hasBin: true
 | 
				
			||||||
    dependencies:
 | 
					    dependencies:
 | 
				
			||||||
      caniuse-lite: 1.0.30001546
 | 
					      caniuse-lite: 1.0.30001546
 | 
				
			||||||
      electron-to-chromium: 1.4.542
 | 
					      electron-to-chromium: 1.4.543
 | 
				
			||||||
      node-releases: 2.0.13
 | 
					      node-releases: 2.0.13
 | 
				
			||||||
      update-browserslist-db: 1.0.13_browserslist@4.22.1
 | 
					      update-browserslist-db: 1.0.13_browserslist@4.22.1
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -3883,8 +3883,8 @@ packages:
 | 
				
			|||||||
      - '@vue/composition-api'
 | 
					      - '@vue/composition-api'
 | 
				
			||||||
    dev: false
 | 
					    dev: false
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  /electron-to-chromium/1.4.542:
 | 
					  /electron-to-chromium/1.4.543:
 | 
				
			||||||
    resolution: {integrity: sha512-6+cpa00G09N3sfh2joln4VUXHquWrOFx3FLZqiVQvl45+zS9DskDBTPvob+BhvFRmTBkyDSk0vvLMMRo/qc6mQ==}
 | 
					    resolution: {integrity: sha512-t2ZP4AcGE0iKCCQCBx/K2426crYdxD3YU6l0uK2EO3FZH0pbC4pFz/sZm2ruZsND6hQBTcDWWlo/MLpiOdif5g==}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  /element-plus/2.3.14_vue@3.3.4:
 | 
					  /element-plus/2.3.14_vue@3.3.4:
 | 
				
			||||||
    resolution: {integrity: sha512-9yvxUaU4jXf2ZNPdmIxoj/f8BG8CDcGM6oHa9JIqxLjQlfY4bpzR1E5CjNimnOX3rxO93w1TQ0jTVt0RSxh9kA==}
 | 
					    resolution: {integrity: sha512-9yvxUaU4jXf2ZNPdmIxoj/f8BG8CDcGM6oHa9JIqxLjQlfY4bpzR1E5CjNimnOX3rxO93w1TQ0jTVt0RSxh9kA==}
 | 
				
			||||||
@ -4171,7 +4171,7 @@ packages:
 | 
				
			|||||||
      file-entry-cache: 6.0.1
 | 
					      file-entry-cache: 6.0.1
 | 
				
			||||||
      find-up: 5.0.0
 | 
					      find-up: 5.0.0
 | 
				
			||||||
      glob-parent: 6.0.2
 | 
					      glob-parent: 6.0.2
 | 
				
			||||||
      globals: 13.22.0
 | 
					      globals: 13.23.0
 | 
				
			||||||
      graphemer: 1.4.0
 | 
					      graphemer: 1.4.0
 | 
				
			||||||
      ignore: 5.2.4
 | 
					      ignore: 5.2.4
 | 
				
			||||||
      imurmurhash: 0.1.4
 | 
					      imurmurhash: 0.1.4
 | 
				
			||||||
@ -4625,8 +4625,8 @@ packages:
 | 
				
			|||||||
    resolution: {integrity: sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==}
 | 
					    resolution: {integrity: sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==}
 | 
				
			||||||
    engines: {node: '>=4'}
 | 
					    engines: {node: '>=4'}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  /globals/13.22.0:
 | 
					  /globals/13.23.0:
 | 
				
			||||||
    resolution: {integrity: sha512-H1Ddc/PbZHTDVJSnj8kWptIRSD6AM3pK+mKytuIVF4uoBV7rshFlhhvA58ceJ5wp3Er58w6zj7bykMpYXt3ETw==}
 | 
					    resolution: {integrity: sha512-XAmF0RjlrjY23MA51q3HltdlGxUpXPvg0GioKiD9X6HD28iMjo2dKC8Vqwm7lne4GNr78+RHTfliktR6ZH09wA==}
 | 
				
			||||||
    engines: {node: '>=8'}
 | 
					    engines: {node: '>=8'}
 | 
				
			||||||
    dependencies:
 | 
					    dependencies:
 | 
				
			||||||
      type-fest: 0.20.2
 | 
					      type-fest: 0.20.2
 | 
				
			||||||
@ -7589,7 +7589,7 @@ packages:
 | 
				
			|||||||
    resolution: {integrity: sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==}
 | 
					    resolution: {integrity: sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==}
 | 
				
			||||||
    dev: false
 | 
					    dev: false
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  /sass-loader/13.3.2_sass@1.68.0:
 | 
					  /sass-loader/13.3.2_sass@1.69.0:
 | 
				
			||||||
    resolution: {integrity: sha512-CQbKl57kdEv+KDLquhC+gE3pXt74LEAzm+tzywcA0/aHZuub8wTErbjAoNI57rPUWRYRNC5WUnNl8eGJNbDdwg==}
 | 
					    resolution: {integrity: sha512-CQbKl57kdEv+KDLquhC+gE3pXt74LEAzm+tzywcA0/aHZuub8wTErbjAoNI57rPUWRYRNC5WUnNl8eGJNbDdwg==}
 | 
				
			||||||
    engines: {node: '>= 14.15.0'}
 | 
					    engines: {node: '>= 14.15.0'}
 | 
				
			||||||
    peerDependencies:
 | 
					    peerDependencies:
 | 
				
			||||||
@ -7611,11 +7611,11 @@ packages:
 | 
				
			|||||||
        optional: true
 | 
					        optional: true
 | 
				
			||||||
    dependencies:
 | 
					    dependencies:
 | 
				
			||||||
      neo-async: 2.6.2
 | 
					      neo-async: 2.6.2
 | 
				
			||||||
      sass: 1.68.0
 | 
					      sass: 1.69.0
 | 
				
			||||||
    dev: true
 | 
					    dev: true
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  /sass/1.68.0:
 | 
					  /sass/1.69.0:
 | 
				
			||||||
    resolution: {integrity: sha512-Lmj9lM/fef0nQswm1J2HJcEsBUba4wgNx2fea6yJHODREoMFnwRpZydBnX/RjyXw2REIwdkbqE4hrTo4qfDBUA==}
 | 
					    resolution: {integrity: sha512-l3bbFpfTOGgQZCLU/gvm1lbsQ5mC/WnLz3djL2v4WCJBDrWm58PO+jgngcGRNnKUh6wSsdm50YaovTqskZ0xDQ==}
 | 
				
			||||||
    engines: {node: '>=14.0.0'}
 | 
					    engines: {node: '>=14.0.0'}
 | 
				
			||||||
    hasBin: true
 | 
					    hasBin: true
 | 
				
			||||||
    dependencies:
 | 
					    dependencies:
 | 
				
			||||||
@ -7782,7 +7782,7 @@ packages:
 | 
				
			|||||||
    resolution: {integrity: sha512-kN9dJbvnySHULIluDHy32WHRUu3Og7B9sbY7tsFLctQkIqnMh3hErYgdMjTYuqmcXX+lK5T1lnUt3G7zNswmZA==}
 | 
					    resolution: {integrity: sha512-kN9dJbvnySHULIluDHy32WHRUu3Og7B9sbY7tsFLctQkIqnMh3hErYgdMjTYuqmcXX+lK5T1lnUt3G7zNswmZA==}
 | 
				
			||||||
    dependencies:
 | 
					    dependencies:
 | 
				
			||||||
      spdx-expression-parse: 3.0.1
 | 
					      spdx-expression-parse: 3.0.1
 | 
				
			||||||
      spdx-license-ids: 3.0.15
 | 
					      spdx-license-ids: 3.0.16
 | 
				
			||||||
    dev: true
 | 
					    dev: true
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  /spdx-exceptions/2.3.0:
 | 
					  /spdx-exceptions/2.3.0:
 | 
				
			||||||
@ -7793,11 +7793,11 @@ packages:
 | 
				
			|||||||
    resolution: {integrity: sha512-cbqHunsQWnJNE6KhVSMsMeH5H/L9EpymbzqTQ3uLwNCLZ1Q481oWaofqH7nO6V07xlXwY6PhQdQ2IedWx/ZK4Q==}
 | 
					    resolution: {integrity: sha512-cbqHunsQWnJNE6KhVSMsMeH5H/L9EpymbzqTQ3uLwNCLZ1Q481oWaofqH7nO6V07xlXwY6PhQdQ2IedWx/ZK4Q==}
 | 
				
			||||||
    dependencies:
 | 
					    dependencies:
 | 
				
			||||||
      spdx-exceptions: 2.3.0
 | 
					      spdx-exceptions: 2.3.0
 | 
				
			||||||
      spdx-license-ids: 3.0.15
 | 
					      spdx-license-ids: 3.0.16
 | 
				
			||||||
    dev: true
 | 
					    dev: true
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  /spdx-license-ids/3.0.15:
 | 
					  /spdx-license-ids/3.0.16:
 | 
				
			||||||
    resolution: {integrity: sha512-lpT8hSQp9jAKp9mhtBU4Xjon8LPGBvLIuBiSVhMEtmLecTh2mO0tlqrAMp47tBXzMr13NJMQ2lf7RpQGLJ3HsQ==}
 | 
					    resolution: {integrity: sha512-eWN+LnM3GR6gPu35WxNgbGl8rmY1AEmoMDvL/QD6zYmPWgywxWqJWNdLGT+ke8dKNWrcYgYjPpG5gbTfghP8rw==}
 | 
				
			||||||
    dev: true
 | 
					    dev: true
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  /split2/3.2.2:
 | 
					  /split2/3.2.2:
 | 
				
			||||||
@ -8568,7 +8568,7 @@ packages:
 | 
				
			|||||||
  /unimport/3.4.0:
 | 
					  /unimport/3.4.0:
 | 
				
			||||||
    resolution: {integrity: sha512-M/lfFEgufIT156QAr/jWHLUn55kEmxBBiQsMxvRSIbquwmeJEyQYgshHDEvQDWlSJrVOOTAgnJ3FvlsrpGkanA==}
 | 
					    resolution: {integrity: sha512-M/lfFEgufIT156QAr/jWHLUn55kEmxBBiQsMxvRSIbquwmeJEyQYgshHDEvQDWlSJrVOOTAgnJ3FvlsrpGkanA==}
 | 
				
			||||||
    dependencies:
 | 
					    dependencies:
 | 
				
			||||||
      '@rollup/pluginutils': 5.0.4
 | 
					      '@rollup/pluginutils': 5.0.5
 | 
				
			||||||
      escape-string-regexp: 5.0.0
 | 
					      escape-string-regexp: 5.0.0
 | 
				
			||||||
      fast-glob: 3.3.1
 | 
					      fast-glob: 3.3.1
 | 
				
			||||||
      local-pkg: 0.4.3
 | 
					      local-pkg: 0.4.3
 | 
				
			||||||
@ -8721,7 +8721,7 @@ packages:
 | 
				
			|||||||
      - rollup
 | 
					      - rollup
 | 
				
			||||||
    dev: true
 | 
					    dev: true
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  /vite-plugin-compression/0.5.1_vite@4.4.10:
 | 
					  /vite-plugin-compression/0.5.1_vite@4.4.11:
 | 
				
			||||||
    resolution: {integrity: sha512-5QJKBDc+gNYVqL/skgFAP81Yuzo9R+EAf19d+EtsMF/i8kFUpNi3J/H01QD3Oo8zBQn+NzoCIFkpPLynoOzaJg==}
 | 
					    resolution: {integrity: sha512-5QJKBDc+gNYVqL/skgFAP81Yuzo9R+EAf19d+EtsMF/i8kFUpNi3J/H01QD3Oo8zBQn+NzoCIFkpPLynoOzaJg==}
 | 
				
			||||||
    peerDependencies:
 | 
					    peerDependencies:
 | 
				
			||||||
      vite: '>=2.0.0'
 | 
					      vite: '>=2.0.0'
 | 
				
			||||||
@ -8729,12 +8729,12 @@ packages:
 | 
				
			|||||||
      chalk: 4.1.2
 | 
					      chalk: 4.1.2
 | 
				
			||||||
      debug: 4.3.4
 | 
					      debug: 4.3.4
 | 
				
			||||||
      fs-extra: 10.1.0
 | 
					      fs-extra: 10.1.0
 | 
				
			||||||
      vite: 4.4.10_aoxrcfqgusexnpex5mio6763sm
 | 
					      vite: 4.4.11_e5w4bvq32mzkrz2cg5gbeogbay
 | 
				
			||||||
    transitivePeerDependencies:
 | 
					    transitivePeerDependencies:
 | 
				
			||||||
      - supports-color
 | 
					      - supports-color
 | 
				
			||||||
    dev: true
 | 
					    dev: true
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  /vite-plugin-mock/2.9.6_mockjs@1.1.0+vite@4.4.10:
 | 
					  /vite-plugin-mock/2.9.6_mockjs@1.1.0+vite@4.4.11:
 | 
				
			||||||
    resolution: {integrity: sha512-/Rm59oPppe/ncbkSrUuAxIQihlI2YcBmnbR4ST1RA2VzM1C0tEQc1KlbQvnUGhXECAGTaQN2JyasiwXP6EtKgg==}
 | 
					    resolution: {integrity: sha512-/Rm59oPppe/ncbkSrUuAxIQihlI2YcBmnbR4ST1RA2VzM1C0tEQc1KlbQvnUGhXECAGTaQN2JyasiwXP6EtKgg==}
 | 
				
			||||||
    engines: {node: '>=12.0.0'}
 | 
					    engines: {node: '>=12.0.0'}
 | 
				
			||||||
    peerDependencies:
 | 
					    peerDependencies:
 | 
				
			||||||
@ -8751,7 +8751,7 @@ packages:
 | 
				
			|||||||
      fast-glob: 3.3.1
 | 
					      fast-glob: 3.3.1
 | 
				
			||||||
      mockjs: 1.1.0
 | 
					      mockjs: 1.1.0
 | 
				
			||||||
      path-to-regexp: 6.2.1
 | 
					      path-to-regexp: 6.2.1
 | 
				
			||||||
      vite: 4.4.10_aoxrcfqgusexnpex5mio6763sm
 | 
					      vite: 4.4.11_e5w4bvq32mzkrz2cg5gbeogbay
 | 
				
			||||||
    transitivePeerDependencies:
 | 
					    transitivePeerDependencies:
 | 
				
			||||||
      - rollup
 | 
					      - rollup
 | 
				
			||||||
      - supports-color
 | 
					      - supports-color
 | 
				
			||||||
@ -8768,8 +8768,8 @@ packages:
 | 
				
			|||||||
      svgo: 3.0.2
 | 
					      svgo: 3.0.2
 | 
				
			||||||
    dev: true
 | 
					    dev: true
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  /vite/4.4.10_aoxrcfqgusexnpex5mio6763sm:
 | 
					  /vite/4.4.11_e5w4bvq32mzkrz2cg5gbeogbay:
 | 
				
			||||||
    resolution: {integrity: sha512-TzIjiqx9BEXF8yzYdF2NTf1kFFbjMjUSV0LFZ3HyHoI3SGSPLnnFUKiIQtL3gl2AjHvMrprOvQ3amzaHgQlAxw==}
 | 
					    resolution: {integrity: sha512-ksNZJlkcU9b0lBwAGZGGaZHCMqHsc8OpgtoYhsQ4/I2v5cnpmmmqe5pM4nv/4Hn6G/2GhTdj0DhZh2e+Er1q5A==}
 | 
				
			||||||
    engines: {node: ^14.18.0 || >=16.0.0}
 | 
					    engines: {node: ^14.18.0 || >=16.0.0}
 | 
				
			||||||
    hasBin: true
 | 
					    hasBin: true
 | 
				
			||||||
    peerDependencies:
 | 
					    peerDependencies:
 | 
				
			||||||
@ -8800,7 +8800,7 @@ packages:
 | 
				
			|||||||
      esbuild: 0.18.20
 | 
					      esbuild: 0.18.20
 | 
				
			||||||
      postcss: 8.4.31
 | 
					      postcss: 8.4.31
 | 
				
			||||||
      rollup: 3.29.4
 | 
					      rollup: 3.29.4
 | 
				
			||||||
      sass: 1.68.0
 | 
					      sass: 1.69.0
 | 
				
			||||||
      terser: 5.21.0
 | 
					      terser: 5.21.0
 | 
				
			||||||
    optionalDependencies:
 | 
					    optionalDependencies:
 | 
				
			||||||
      fsevents: 2.3.3
 | 
					      fsevents: 2.3.3
 | 
				
			||||||
 | 
				
			|||||||
@ -8,13 +8,6 @@ import {
 | 
				
			|||||||
  nextTick,
 | 
					  nextTick,
 | 
				
			||||||
  onBeforeMount
 | 
					  onBeforeMount
 | 
				
			||||||
} from "vue";
 | 
					} from "vue";
 | 
				
			||||||
import {
 | 
					 | 
				
			||||||
  useDark,
 | 
					 | 
				
			||||||
  debounce,
 | 
					 | 
				
			||||||
  useGlobal,
 | 
					 | 
				
			||||||
  storageLocal,
 | 
					 | 
				
			||||||
  storageSession
 | 
					 | 
				
			||||||
} from "@pureadmin/utils";
 | 
					 | 
				
			||||||
import { getConfig } from "@/config";
 | 
					import { getConfig } from "@/config";
 | 
				
			||||||
import { useRouter } from "vue-router";
 | 
					import { useRouter } from "vue-router";
 | 
				
			||||||
import panel from "../panel/index.vue";
 | 
					import panel from "../panel/index.vue";
 | 
				
			||||||
@ -27,6 +20,7 @@ import { useAppStoreHook } from "@/store/modules/app";
 | 
				
			|||||||
import { toggleTheme } from "@pureadmin/theme/dist/browser-utils";
 | 
					import { toggleTheme } from "@pureadmin/theme/dist/browser-utils";
 | 
				
			||||||
import { useMultiTagsStoreHook } from "@/store/modules/multiTags";
 | 
					import { useMultiTagsStoreHook } from "@/store/modules/multiTags";
 | 
				
			||||||
import { useDataThemeChange } from "@/layout/hooks/useDataThemeChange";
 | 
					import { useDataThemeChange } from "@/layout/hooks/useDataThemeChange";
 | 
				
			||||||
 | 
					import { useDark, debounce, useGlobal, storageLocal } from "@pureadmin/utils";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import dayIcon from "@/assets/svg/day.svg?component";
 | 
					import dayIcon from "@/assets/svg/day.svg?component";
 | 
				
			||||||
import darkIcon from "@/assets/svg/dark.svg?component";
 | 
					import darkIcon from "@/assets/svg/dark.svg?component";
 | 
				
			||||||
@ -133,7 +127,6 @@ const multiTagsCacheChange = () => {
 | 
				
			|||||||
function onReset() {
 | 
					function onReset() {
 | 
				
			||||||
  removeToken();
 | 
					  removeToken();
 | 
				
			||||||
  storageLocal().clear();
 | 
					  storageLocal().clear();
 | 
				
			||||||
  storageSession().clear();
 | 
					 | 
				
			||||||
  const { Grey, Weak, MultiTagsCache, EpThemeColor, Layout } = getConfig();
 | 
					  const { Grey, Weak, MultiTagsCache, EpThemeColor, Layout } = getConfig();
 | 
				
			||||||
  useAppStoreHook().setLayout(Layout);
 | 
					  useAppStoreHook().setLayout(Layout);
 | 
				
			||||||
  setEpThemeColor(EpThemeColor);
 | 
					  setEpThemeColor(EpThemeColor);
 | 
				
			||||||
 | 
				
			|||||||
@ -1,16 +1,13 @@
 | 
				
			|||||||
import "@/utils/sso";
 | 
					import "@/utils/sso";
 | 
				
			||||||
 | 
					import Cookies from "js-cookie";
 | 
				
			||||||
import { getConfig } from "@/config";
 | 
					import { getConfig } from "@/config";
 | 
				
			||||||
import NProgress from "@/utils/progress";
 | 
					import NProgress from "@/utils/progress";
 | 
				
			||||||
import { transformI18n } from "@/plugins/i18n";
 | 
					import { transformI18n } from "@/plugins/i18n";
 | 
				
			||||||
import { sessionKey, type DataInfo } from "@/utils/auth";
 | 
					import { buildHierarchyTree } from "@/utils/tree";
 | 
				
			||||||
 | 
					import remainingRouter from "./modules/remaining";
 | 
				
			||||||
import { useMultiTagsStoreHook } from "@/store/modules/multiTags";
 | 
					import { useMultiTagsStoreHook } from "@/store/modules/multiTags";
 | 
				
			||||||
import { usePermissionStoreHook } from "@/store/modules/permission";
 | 
					import { usePermissionStoreHook } from "@/store/modules/permission";
 | 
				
			||||||
import {
 | 
					import { isUrl, openLink, storageLocal, isAllEmpty } from "@pureadmin/utils";
 | 
				
			||||||
  Router,
 | 
					 | 
				
			||||||
  createRouter,
 | 
					 | 
				
			||||||
  RouteRecordRaw,
 | 
					 | 
				
			||||||
  RouteComponent
 | 
					 | 
				
			||||||
} from "vue-router";
 | 
					 | 
				
			||||||
import {
 | 
					import {
 | 
				
			||||||
  ascending,
 | 
					  ascending,
 | 
				
			||||||
  getTopMenu,
 | 
					  getTopMenu,
 | 
				
			||||||
@ -22,10 +19,18 @@ import {
 | 
				
			|||||||
  formatTwoStageRoutes,
 | 
					  formatTwoStageRoutes,
 | 
				
			||||||
  formatFlatteningRoutes
 | 
					  formatFlatteningRoutes
 | 
				
			||||||
} from "./utils";
 | 
					} from "./utils";
 | 
				
			||||||
import { buildHierarchyTree } from "@/utils/tree";
 | 
					import {
 | 
				
			||||||
import { isUrl, openLink, storageSession, isAllEmpty } from "@pureadmin/utils";
 | 
					  Router,
 | 
				
			||||||
 | 
					  createRouter,
 | 
				
			||||||
import remainingRouter from "./modules/remaining";
 | 
					  RouteRecordRaw,
 | 
				
			||||||
 | 
					  RouteComponent
 | 
				
			||||||
 | 
					} from "vue-router";
 | 
				
			||||||
 | 
					import {
 | 
				
			||||||
 | 
					  type DataInfo,
 | 
				
			||||||
 | 
					  userKey,
 | 
				
			||||||
 | 
					  removeToken,
 | 
				
			||||||
 | 
					  multipleTabsKey
 | 
				
			||||||
 | 
					} from "@/utils/auth";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/** 自动导入全部静态路由,无需再手动引入!匹配 src/router/modules 目录(任何嵌套级别)中具有 .ts 扩展名的所有文件,除了 remaining.ts 文件
 | 
					/** 自动导入全部静态路由,无需再手动引入!匹配 src/router/modules 目录(任何嵌套级别)中具有 .ts 扩展名的所有文件,除了 remaining.ts 文件
 | 
				
			||||||
 * 如何匹配所有文件请看:https://github.com/mrmlnc/fast-glob#basic-syntax
 | 
					 * 如何匹配所有文件请看:https://github.com/mrmlnc/fast-glob#basic-syntax
 | 
				
			||||||
@ -109,7 +114,7 @@ router.beforeEach((to: ToRouteType, _from, next) => {
 | 
				
			|||||||
      handleAliveRoute(to);
 | 
					      handleAliveRoute(to);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
  const userInfo = storageSession().getItem<DataInfo<number>>(sessionKey);
 | 
					  const userInfo = storageLocal().getItem<DataInfo<number>>(userKey);
 | 
				
			||||||
  NProgress.start();
 | 
					  NProgress.start();
 | 
				
			||||||
  const externalLink = isUrl(to?.name as string);
 | 
					  const externalLink = isUrl(to?.name as string);
 | 
				
			||||||
  if (!externalLink) {
 | 
					  if (!externalLink) {
 | 
				
			||||||
@ -125,7 +130,7 @@ router.beforeEach((to: ToRouteType, _from, next) => {
 | 
				
			|||||||
  function toCorrectRoute() {
 | 
					  function toCorrectRoute() {
 | 
				
			||||||
    whiteList.includes(to.fullPath) ? next(_from.fullPath) : next();
 | 
					    whiteList.includes(to.fullPath) ? next(_from.fullPath) : next();
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
  if (userInfo) {
 | 
					  if (Cookies.get(multipleTabsKey) && userInfo) {
 | 
				
			||||||
    // 无权限跳转403页面
 | 
					    // 无权限跳转403页面
 | 
				
			||||||
    if (to.meta?.roles && !isOneOfArray(to.meta?.roles, userInfo?.roles)) {
 | 
					    if (to.meta?.roles && !isOneOfArray(to.meta?.roles, userInfo?.roles)) {
 | 
				
			||||||
      next({ path: "/error/403" });
 | 
					      next({ path: "/error/403" });
 | 
				
			||||||
@ -187,6 +192,7 @@ router.beforeEach((to: ToRouteType, _from, next) => {
 | 
				
			|||||||
      if (whiteList.indexOf(to.path) !== -1) {
 | 
					      if (whiteList.indexOf(to.path) !== -1) {
 | 
				
			||||||
        next();
 | 
					        next();
 | 
				
			||||||
      } else {
 | 
					      } else {
 | 
				
			||||||
 | 
					        removeToken();
 | 
				
			||||||
        next({ path: "/login" });
 | 
					        next({ path: "/login" });
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
    } else {
 | 
					    } else {
 | 
				
			||||||
 | 
				
			|||||||
@ -13,13 +13,13 @@ import {
 | 
				
			|||||||
  cloneDeep,
 | 
					  cloneDeep,
 | 
				
			||||||
  isAllEmpty,
 | 
					  isAllEmpty,
 | 
				
			||||||
  intersection,
 | 
					  intersection,
 | 
				
			||||||
  storageSession,
 | 
					  storageLocal,
 | 
				
			||||||
  isIncludeAllChildren
 | 
					  isIncludeAllChildren
 | 
				
			||||||
} from "@pureadmin/utils";
 | 
					} from "@pureadmin/utils";
 | 
				
			||||||
import { getConfig } from "@/config";
 | 
					import { getConfig } from "@/config";
 | 
				
			||||||
import { menuType } from "@/layout/types";
 | 
					import { menuType } from "@/layout/types";
 | 
				
			||||||
import { buildHierarchyTree } from "@/utils/tree";
 | 
					import { buildHierarchyTree } from "@/utils/tree";
 | 
				
			||||||
import { sessionKey, type DataInfo } from "@/utils/auth";
 | 
					import { userKey, type DataInfo } from "@/utils/auth";
 | 
				
			||||||
import { useMultiTagsStoreHook } from "@/store/modules/multiTags";
 | 
					import { useMultiTagsStoreHook } from "@/store/modules/multiTags";
 | 
				
			||||||
import { usePermissionStoreHook } from "@/store/modules/permission";
 | 
					import { usePermissionStoreHook } from "@/store/modules/permission";
 | 
				
			||||||
const IFrame = () => import("@/layout/frameView.vue");
 | 
					const IFrame = () => import("@/layout/frameView.vue");
 | 
				
			||||||
@ -81,10 +81,10 @@ function isOneOfArray(a: Array<string>, b: Array<string>) {
 | 
				
			|||||||
    : true;
 | 
					    : true;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/** 从sessionStorage里取出当前登陆用户的角色roles,过滤无权限的菜单 */
 | 
					/** 从localStorage里取出当前登陆用户的角色roles,过滤无权限的菜单 */
 | 
				
			||||||
function filterNoPermissionTree(data: RouteComponent[]) {
 | 
					function filterNoPermissionTree(data: RouteComponent[]) {
 | 
				
			||||||
  const currentRoles =
 | 
					  const currentRoles =
 | 
				
			||||||
    storageSession().getItem<DataInfo<number>>(sessionKey)?.roles ?? [];
 | 
					    storageLocal().getItem<DataInfo<number>>(userKey)?.roles ?? [];
 | 
				
			||||||
  const newTree = cloneDeep(data).filter((v: any) =>
 | 
					  const newTree = cloneDeep(data).filter((v: any) =>
 | 
				
			||||||
    isOneOfArray(v.meta?.roles, currentRoles)
 | 
					    isOneOfArray(v.meta?.roles, currentRoles)
 | 
				
			||||||
  );
 | 
					  );
 | 
				
			||||||
@ -184,9 +184,9 @@ function handleAsyncRoutes(routeList) {
 | 
				
			|||||||
/** 初始化路由(`new Promise` 写法防止在异步请求中造成无限循环)*/
 | 
					/** 初始化路由(`new Promise` 写法防止在异步请求中造成无限循环)*/
 | 
				
			||||||
function initRouter() {
 | 
					function initRouter() {
 | 
				
			||||||
  if (getConfig()?.CachingAsyncRoutes) {
 | 
					  if (getConfig()?.CachingAsyncRoutes) {
 | 
				
			||||||
    // 开启动态路由缓存本地sessionStorage
 | 
					    // 开启动态路由缓存本地localStorage
 | 
				
			||||||
    const key = "async-routes";
 | 
					    const key = "async-routes";
 | 
				
			||||||
    const asyncRouteList = storageSession().getItem(key) as any;
 | 
					    const asyncRouteList = storageLocal().getItem(key) as any;
 | 
				
			||||||
    if (asyncRouteList && asyncRouteList?.length > 0) {
 | 
					    if (asyncRouteList && asyncRouteList?.length > 0) {
 | 
				
			||||||
      return new Promise(resolve => {
 | 
					      return new Promise(resolve => {
 | 
				
			||||||
        handleAsyncRoutes(asyncRouteList);
 | 
					        handleAsyncRoutes(asyncRouteList);
 | 
				
			||||||
@ -196,7 +196,7 @@ function initRouter() {
 | 
				
			|||||||
      return new Promise(resolve => {
 | 
					      return new Promise(resolve => {
 | 
				
			||||||
        getAsyncRoutes().then(({ data }) => {
 | 
					        getAsyncRoutes().then(({ data }) => {
 | 
				
			||||||
          handleAsyncRoutes(cloneDeep(data));
 | 
					          handleAsyncRoutes(cloneDeep(data));
 | 
				
			||||||
          storageSession().setItem(key, data);
 | 
					          storageLocal().setItem(key, data);
 | 
				
			||||||
          resolve(router);
 | 
					          resolve(router);
 | 
				
			||||||
        });
 | 
					        });
 | 
				
			||||||
      });
 | 
					      });
 | 
				
			||||||
 | 
				
			|||||||
@ -41,4 +41,5 @@ export type userType = {
 | 
				
			|||||||
  roles?: Array<string>;
 | 
					  roles?: Array<string>;
 | 
				
			||||||
  verifyCode?: string;
 | 
					  verifyCode?: string;
 | 
				
			||||||
  currentPage?: number;
 | 
					  currentPage?: number;
 | 
				
			||||||
 | 
					  isRemembered?: boolean;
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
				
			|||||||
@ -3,24 +3,25 @@ import { store } from "@/store";
 | 
				
			|||||||
import { userType } from "./types";
 | 
					import { userType } from "./types";
 | 
				
			||||||
import { routerArrays } from "@/layout/types";
 | 
					import { routerArrays } from "@/layout/types";
 | 
				
			||||||
import { router, resetRouter } from "@/router";
 | 
					import { router, resetRouter } from "@/router";
 | 
				
			||||||
import { storageSession } from "@pureadmin/utils";
 | 
					import { storageLocal } from "@pureadmin/utils";
 | 
				
			||||||
import { getLogin, refreshTokenApi } from "@/api/user";
 | 
					import { getLogin, refreshTokenApi } from "@/api/user";
 | 
				
			||||||
import { UserResult, RefreshTokenResult } from "@/api/user";
 | 
					import { UserResult, RefreshTokenResult } from "@/api/user";
 | 
				
			||||||
import { useMultiTagsStoreHook } from "@/store/modules/multiTags";
 | 
					import { useMultiTagsStoreHook } from "@/store/modules/multiTags";
 | 
				
			||||||
import { type DataInfo, setToken, removeToken, sessionKey } from "@/utils/auth";
 | 
					import { type DataInfo, setToken, removeToken, userKey } from "@/utils/auth";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export const useUserStore = defineStore({
 | 
					export const useUserStore = defineStore({
 | 
				
			||||||
  id: "pure-user",
 | 
					  id: "pure-user",
 | 
				
			||||||
  state: (): userType => ({
 | 
					  state: (): userType => ({
 | 
				
			||||||
    // 用户名
 | 
					    // 用户名
 | 
				
			||||||
    username:
 | 
					    username: storageLocal().getItem<DataInfo<number>>(userKey)?.username ?? "",
 | 
				
			||||||
      storageSession().getItem<DataInfo<number>>(sessionKey)?.username ?? "",
 | 
					 | 
				
			||||||
    // 页面级别权限
 | 
					    // 页面级别权限
 | 
				
			||||||
    roles: storageSession().getItem<DataInfo<number>>(sessionKey)?.roles ?? [],
 | 
					    roles: storageLocal().getItem<DataInfo<number>>(userKey)?.roles ?? [],
 | 
				
			||||||
    // 前端生成的验证码(按实际需求替换)
 | 
					    // 前端生成的验证码(按实际需求替换)
 | 
				
			||||||
    verifyCode: "",
 | 
					    verifyCode: "",
 | 
				
			||||||
    // 判断登录页面显示哪个组件(0:登录(默认)、1:手机登录、2:二维码登录、3:注册、4:忘记密码)
 | 
					    // 判断登录页面显示哪个组件(0:登录(默认)、1:手机登录、2:二维码登录、3:注册、4:忘记密码)
 | 
				
			||||||
    currentPage: 0
 | 
					    currentPage: 0,
 | 
				
			||||||
 | 
					    // 是否勾选了7天内免登录
 | 
				
			||||||
 | 
					    isRemembered: false
 | 
				
			||||||
  }),
 | 
					  }),
 | 
				
			||||||
  actions: {
 | 
					  actions: {
 | 
				
			||||||
    /** 存储用户名 */
 | 
					    /** 存储用户名 */
 | 
				
			||||||
@ -39,6 +40,10 @@ export const useUserStore = defineStore({
 | 
				
			|||||||
    SET_CURRENTPAGE(value: number) {
 | 
					    SET_CURRENTPAGE(value: number) {
 | 
				
			||||||
      this.currentPage = value;
 | 
					      this.currentPage = value;
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
 | 
					    /** 存储是否勾选了7天内免登录 */
 | 
				
			||||||
 | 
					    SET_ISREMEMBERED(bool: boolean) {
 | 
				
			||||||
 | 
					      this.isRemembered = bool;
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
    /** 登入 */
 | 
					    /** 登入 */
 | 
				
			||||||
    async loginByUsername(data) {
 | 
					    async loginByUsername(data) {
 | 
				
			||||||
      return new Promise<UserResult>((resolve, reject) => {
 | 
					      return new Promise<UserResult>((resolve, reject) => {
 | 
				
			||||||
 | 
				
			|||||||
@ -1,5 +1,5 @@
 | 
				
			|||||||
import Cookies from "js-cookie";
 | 
					import Cookies from "js-cookie";
 | 
				
			||||||
import { storageSession } from "@pureadmin/utils";
 | 
					import { storageLocal } from "@pureadmin/utils";
 | 
				
			||||||
import { useUserStoreHook } from "@/store/modules/user";
 | 
					import { useUserStoreHook } from "@/store/modules/user";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export interface DataInfo<T> {
 | 
					export interface DataInfo<T> {
 | 
				
			||||||
@ -15,22 +15,29 @@ export interface DataInfo<T> {
 | 
				
			|||||||
  roles?: Array<string>;
 | 
					  roles?: Array<string>;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export const sessionKey = "user-info";
 | 
					export const userKey = "user-info";
 | 
				
			||||||
export const TokenKey = "authorized-token";
 | 
					export const TokenKey = "authorized-token";
 | 
				
			||||||
 | 
					/**
 | 
				
			||||||
 | 
					 * 通过`multiple-tabs`是否在`cookie`中,判断用户是否已经登录系统,
 | 
				
			||||||
 | 
					 * 从而支持多标签页打开已经登录的系统后无需再登录。
 | 
				
			||||||
 | 
					 * 浏览器完全关闭后`multiple-tabs`将自动从`cookie`中销毁,
 | 
				
			||||||
 | 
					 * 再次打开浏览器需要重新登录系统
 | 
				
			||||||
 | 
					 * */
 | 
				
			||||||
 | 
					export const multipleTabsKey = "multiple-tabs";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/** 获取`token` */
 | 
					/** 获取`token` */
 | 
				
			||||||
export function getToken(): DataInfo<number> {
 | 
					export function getToken(): DataInfo<number> {
 | 
				
			||||||
  // 此处与`TokenKey`相同,此写法解决初始化时`Cookies`中不存在`TokenKey`报错
 | 
					  // 此处与`TokenKey`相同,此写法解决初始化时`Cookies`中不存在`TokenKey`报错
 | 
				
			||||||
  return Cookies.get(TokenKey)
 | 
					  return Cookies.get(TokenKey)
 | 
				
			||||||
    ? JSON.parse(Cookies.get(TokenKey))
 | 
					    ? JSON.parse(Cookies.get(TokenKey))
 | 
				
			||||||
    : storageSession().getItem(sessionKey);
 | 
					    : storageLocal().getItem(userKey);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/**
 | 
					/**
 | 
				
			||||||
 * @description 设置`token`以及一些必要信息并采用无感刷新`token`方案
 | 
					 * @description 设置`token`以及一些必要信息并采用无感刷新`token`方案
 | 
				
			||||||
 * 无感刷新:后端返回`accessToken`(访问接口使用的`token`)、`refreshToken`(用于调用刷新`accessToken`的接口时所需的`token`,`refreshToken`的过期时间(比如30天)应大于`accessToken`的过期时间(比如2小时))、`expires`(`accessToken`的过期时间)
 | 
					 * 无感刷新:后端返回`accessToken`(访问接口使用的`token`)、`refreshToken`(用于调用刷新`accessToken`的接口时所需的`token`,`refreshToken`的过期时间(比如30天)应大于`accessToken`的过期时间(比如2小时))、`expires`(`accessToken`的过期时间)
 | 
				
			||||||
 * 将`accessToken`、`expires`这两条信息放在key值为authorized-token的cookie里(过期自动销毁)
 | 
					 * 将`accessToken`、`expires`这两条信息放在key值为authorized-token的cookie里(过期自动销毁)
 | 
				
			||||||
 * 将`username`、`roles`、`refreshToken`、`expires`这四条信息放在key值为`user-info`的sessionStorage里(浏览器关闭自动销毁)
 | 
					 * 将`username`、`roles`、`refreshToken`、`expires`这四条信息放在key值为`user-info`的localStorage里(利用`multipleTabsKey`当浏览器完全关闭后自动销毁)
 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
export function setToken(data: DataInfo<Date>) {
 | 
					export function setToken(data: DataInfo<Date>) {
 | 
				
			||||||
  let expires = 0;
 | 
					  let expires = 0;
 | 
				
			||||||
@ -44,10 +51,20 @@ export function setToken(data: DataInfo<Date>) {
 | 
				
			|||||||
      })
 | 
					      })
 | 
				
			||||||
    : Cookies.set(TokenKey, cookieString);
 | 
					    : Cookies.set(TokenKey, cookieString);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  function setSessionKey(username: string, roles: Array<string>) {
 | 
					  Cookies.set(
 | 
				
			||||||
 | 
					    multipleTabsKey,
 | 
				
			||||||
 | 
					    "true",
 | 
				
			||||||
 | 
					    useUserStoreHook().isRemembered
 | 
				
			||||||
 | 
					      ? {
 | 
				
			||||||
 | 
					          expires: 7
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      : {}
 | 
				
			||||||
 | 
					  );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  function setUserKey(username: string, roles: Array<string>) {
 | 
				
			||||||
    useUserStoreHook().SET_USERNAME(username);
 | 
					    useUserStoreHook().SET_USERNAME(username);
 | 
				
			||||||
    useUserStoreHook().SET_ROLES(roles);
 | 
					    useUserStoreHook().SET_ROLES(roles);
 | 
				
			||||||
    storageSession().setItem(sessionKey, {
 | 
					    storageLocal().setItem(userKey, {
 | 
				
			||||||
      refreshToken,
 | 
					      refreshToken,
 | 
				
			||||||
      expires,
 | 
					      expires,
 | 
				
			||||||
      username,
 | 
					      username,
 | 
				
			||||||
@ -57,20 +74,21 @@ export function setToken(data: DataInfo<Date>) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
  if (data.username && data.roles) {
 | 
					  if (data.username && data.roles) {
 | 
				
			||||||
    const { username, roles } = data;
 | 
					    const { username, roles } = data;
 | 
				
			||||||
    setSessionKey(username, roles);
 | 
					    setUserKey(username, roles);
 | 
				
			||||||
  } else {
 | 
					  } else {
 | 
				
			||||||
    const username =
 | 
					    const username =
 | 
				
			||||||
      storageSession().getItem<DataInfo<number>>(sessionKey)?.username ?? "";
 | 
					      storageLocal().getItem<DataInfo<number>>(userKey)?.username ?? "";
 | 
				
			||||||
    const roles =
 | 
					    const roles =
 | 
				
			||||||
      storageSession().getItem<DataInfo<number>>(sessionKey)?.roles ?? [];
 | 
					      storageLocal().getItem<DataInfo<number>>(userKey)?.roles ?? [];
 | 
				
			||||||
    setSessionKey(username, roles);
 | 
					    setUserKey(username, roles);
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/** 删除`token`以及key值为`user-info`的session信息 */
 | 
					/** 删除`token`以及key值为`user-info`的localStorage信息 */
 | 
				
			||||||
export function removeToken() {
 | 
					export function removeToken() {
 | 
				
			||||||
  Cookies.remove(TokenKey);
 | 
					  Cookies.remove(TokenKey);
 | 
				
			||||||
  sessionStorage.clear();
 | 
					  Cookies.remove(multipleTabsKey);
 | 
				
			||||||
 | 
					  storageLocal().removeItem(userKey);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/** 格式化token(jwt格式) */
 | 
					/** 格式化token(jwt格式) */
 | 
				
			||||||
 | 
				
			|||||||
@ -37,6 +37,7 @@ import globalization from "@/assets/svg/globalization.svg?component";
 | 
				
			|||||||
import Lock from "@iconify-icons/ri/lock-fill";
 | 
					import Lock from "@iconify-icons/ri/lock-fill";
 | 
				
			||||||
import Check from "@iconify-icons/ep/check";
 | 
					import Check from "@iconify-icons/ep/check";
 | 
				
			||||||
import User from "@iconify-icons/ri/user-3-fill";
 | 
					import User from "@iconify-icons/ri/user-3-fill";
 | 
				
			||||||
 | 
					import Info from "@iconify-icons/ri/information-line";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
defineOptions({
 | 
					defineOptions({
 | 
				
			||||||
  name: "Login"
 | 
					  name: "Login"
 | 
				
			||||||
@ -107,6 +108,9 @@ onBeforeUnmount(() => {
 | 
				
			|||||||
watch(imgCode, value => {
 | 
					watch(imgCode, value => {
 | 
				
			||||||
  useUserStoreHook().SET_VERIFYCODE(value);
 | 
					  useUserStoreHook().SET_VERIFYCODE(value);
 | 
				
			||||||
});
 | 
					});
 | 
				
			||||||
 | 
					watch(checked, bool => {
 | 
				
			||||||
 | 
					  useUserStoreHook().SET_ISREMEMBERED(bool);
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
</script>
 | 
					</script>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
<template>
 | 
					<template>
 | 
				
			||||||
@ -225,7 +229,16 @@ watch(imgCode, value => {
 | 
				
			|||||||
              <el-form-item>
 | 
					              <el-form-item>
 | 
				
			||||||
                <div class="w-full h-[20px] flex justify-between items-center">
 | 
					                <div class="w-full h-[20px] flex justify-between items-center">
 | 
				
			||||||
                  <el-checkbox v-model="checked">
 | 
					                  <el-checkbox v-model="checked">
 | 
				
			||||||
 | 
					                    <span class="flex">
 | 
				
			||||||
                      {{ t("login.remember") }}
 | 
					                      {{ t("login.remember") }}
 | 
				
			||||||
 | 
					                      <el-tooltip
 | 
				
			||||||
 | 
					                        effect="dark"
 | 
				
			||||||
 | 
					                        placement="top"
 | 
				
			||||||
 | 
					                        :content="t('login.rememberInfo')"
 | 
				
			||||||
 | 
					                      >
 | 
				
			||||||
 | 
					                        <IconifyIconOffline :icon="Info" class="ml-1" />
 | 
				
			||||||
 | 
					                      </el-tooltip>
 | 
				
			||||||
 | 
					                    </span>
 | 
				
			||||||
                  </el-checkbox>
 | 
					                  </el-checkbox>
 | 
				
			||||||
                  <el-button
 | 
					                  <el-button
 | 
				
			||||||
                    link
 | 
					                    link
 | 
				
			||||||
 | 
				
			|||||||
@ -1,6 +1,6 @@
 | 
				
			|||||||
<script setup lang="ts">
 | 
					<script setup lang="ts">
 | 
				
			||||||
import { initRouter } from "@/router/utils";
 | 
					import { initRouter } from "@/router/utils";
 | 
				
			||||||
import { storageSession } from "@pureadmin/utils";
 | 
					import { storageLocal } from "@pureadmin/utils";
 | 
				
			||||||
import { type CSSProperties, ref, computed } from "vue";
 | 
					import { type CSSProperties, ref, computed } from "vue";
 | 
				
			||||||
import { useUserStoreHook } from "@/store/modules/user";
 | 
					import { useUserStoreHook } from "@/store/modules/user";
 | 
				
			||||||
import { usePermissionStoreHook } from "@/store/modules/permission";
 | 
					import { usePermissionStoreHook } from "@/store/modules/permission";
 | 
				
			||||||
@ -34,7 +34,7 @@ function onChange() {
 | 
				
			|||||||
    .loginByUsername({ username: username.value, password: "admin123" })
 | 
					    .loginByUsername({ username: username.value, password: "admin123" })
 | 
				
			||||||
    .then(res => {
 | 
					    .then(res => {
 | 
				
			||||||
      if (res.success) {
 | 
					      if (res.success) {
 | 
				
			||||||
        storageSession().removeItem("async-routes");
 | 
					        storageLocal().removeItem("async-routes");
 | 
				
			||||||
        usePermissionStoreHook().clearAllCachePage();
 | 
					        usePermissionStoreHook().clearAllCachePage();
 | 
				
			||||||
        initRouter();
 | 
					        initRouter();
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
 | 
				
			|||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user