mirror of
				https://github.com/soybeanjs/soybean-admin.git
				synced 2025-10-25 19:13:42 +08:00 
			
		
		
		
	Compare commits
	
		
			23 Commits
		
	
	
		
			tauri-v1.1
			...
			tauri-v1.2
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 1bd769f1c0 | ||
|  | 22004ff4dc | ||
|  | 08827a42a3 | ||
|  | 6a6eb9afd6 | ||
|  | ff51b72dac | ||
|  | fe06b8c499 | ||
|  | 4b63bbcf67 | ||
|  | 5531a68641 | ||
|  | 584cd54d6d | ||
|  | 2bec899031 | ||
|  | 83862ba76e | ||
|  | 060c0a91c7 | ||
|  | 76649e2a08 | ||
|  | ddf3823a55 | ||
|  | e75fd73f34 | ||
|  | 1f4647b995 | ||
|  | 910dfca8bb | ||
|  | ffb48b15bc | ||
|  | e6086f0f35 | ||
|  | 9b05d73e6d | ||
|  | bd69c00e74 | ||
|  | d08a3817d1 | ||
|  | d0380ce5cb | 
| @@ -1,5 +1,5 @@ | |||||||
| # backend service base url, prod environment | # backend service base url, prod environment | ||||||
| VITE_SERVICE_BASE_URL=https://mock.apifox.com/m1/3109515-0-default | VITE_SERVICE_BASE_URL=https://mock.apifox.cn/m1/3109515-0-default | ||||||
|  |  | ||||||
| # other backend service base url, prod environment | # other backend service base url, prod environment | ||||||
| VITE_OTHER_SERVICE_BASE_URL= `{ | VITE_OTHER_SERVICE_BASE_URL= `{ | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| # backend service base url, test environment | # backend service base url, test environment | ||||||
| VITE_SERVICE_BASE_URL=https://mock.apifox.com/m1/3109515-0-default | VITE_SERVICE_BASE_URL=https://mock.apifox.cn/m1/3109515-0-default | ||||||
|  |  | ||||||
| # other backend service base url, test environment | # other backend service base url, test environment | ||||||
| VITE_OTHER_SERVICE_BASE_URL= `{ | VITE_OTHER_SERVICE_BASE_URL= `{ | ||||||
|   | |||||||
							
								
								
									
										3
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							| @@ -14,5 +14,6 @@ | |||||||
|   "i18n-ally.localesPaths": ["src/locales/langs"], |   "i18n-ally.localesPaths": ["src/locales/langs"], | ||||||
|   "prettier.enable": false, |   "prettier.enable": false, | ||||||
|   "typescript.tsdk": "node_modules/typescript/lib", |   "typescript.tsdk": "node_modules/typescript/lib", | ||||||
|   "unocss.root": ["./"] |   "unocss.root": ["./"], | ||||||
|  |   "vue.server.hybridMode": true | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										81
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										81
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -1,6 +1,87 @@ | |||||||
| # Changelog | # Changelog | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [v1.2.1](https://github.com/honghuangdc/soybean-admin/compare/v1.2.0...v1.2.1) (2024-06-07) | ||||||
|  |  | ||||||
|  | ###    🐞 Bug Fixes | ||||||
|  |  | ||||||
|  | - **projects**: | ||||||
|  |   - fix get user info when page reload  -  by @soybeanjs [<samp>(ff51b)</samp>](https://github.com/honghuangdc/soybean-admin/commit/ff51b72) | ||||||
|  |   - fix setupAppVersionNotification render  -  by @soybeanjs [<samp>(6a6eb)</samp>](https://github.com/honghuangdc/soybean-admin/commit/6a6eb9a) | ||||||
|  |  | ||||||
|  | ###    📖 Documentation | ||||||
|  |  | ||||||
|  | - **projects**: update CHANGELOG  -  by @soybeanjs [<samp>(fe06b)</samp>](https://github.com/honghuangdc/soybean-admin/commit/fe06b8c) | ||||||
|  |  | ||||||
|  | ###    🏡 Chore | ||||||
|  |  | ||||||
|  | - **deps**: update deps  -  by @soybeanjs [<samp>(08827)</samp>](https://github.com/honghuangdc/soybean-admin/commit/08827a4) | ||||||
|  |  | ||||||
|  | ###    ❤️ Contributors | ||||||
|  |  | ||||||
|  | [](https://github.com/soybeanjs)   | ||||||
|  |  | ||||||
|  | ## [v1.2.0](https://github.com/soybeanjs/soybean-admin/compare/v1.1.5...v1.2.0) (2024-06-06) | ||||||
|  |  | ||||||
|  | ###    🚀 Features | ||||||
|  |  | ||||||
|  | - **projects**: | ||||||
|  |   - support system new version update notification. close #420  -  by @soybeanjs in https://github.com/soybeanjs/soybean-admin/issues/420 [<samp>(584cd)</samp>](https://github.com/soybeanjs/soybean-admin/commit/584cd54) | ||||||
|  |   - get user info in router guard and remove in localStorage. close #459  -  by @soybeanjs in https://github.com/soybeanjs/soybean-admin/issues/459 [<samp>(5531a)</samp>](https://github.com/soybeanjs/soybean-admin/commit/5531a68) | ||||||
|  |  | ||||||
|  | ###    📖 Documentation | ||||||
|  |  | ||||||
|  | - **projects**: update CHANGELOG  -  by @soybeanjs [<samp>(2bec8)</samp>](https://github.com/soybeanjs/soybean-admin/commit/2bec899) | ||||||
|  |  | ||||||
|  | ###    ❤️ Contributors | ||||||
|  |  | ||||||
|  | [](https://github.com/soybeanjs)   | ||||||
|  |  | ||||||
|  | ## [v1.1.5](https://github.com/soybeanjs/soybean-admin/compare/v1.1.4...v1.1.5) (2024-06-06) | ||||||
|  |  | ||||||
|  | ###    🐞 Bug Fixes | ||||||
|  |  | ||||||
|  | - **projects**: fix register name, CodeLogin => Register  -  by @m-xlsea in https://github.com/soybeanjs/soybean-admin/issues/478 [<samp>(ddf38)</samp>](https://github.com/soybeanjs/soybean-admin/commit/ddf3823) | ||||||
|  |  | ||||||
|  | ###    🏡 Chore | ||||||
|  |  | ||||||
|  | - **deps**: update deps  -  by @soybeanjs [<samp>(060c0)</samp>](https://github.com/soybeanjs/soybean-admin/commit/060c0a9) | ||||||
|  | - **projects**: update vscode settings: vue official  -  by @soybeanjs [<samp>(76649)</samp>](https://github.com/soybeanjs/soybean-admin/commit/76649e2) | ||||||
|  |  | ||||||
|  | ###    ❤️ Contributors | ||||||
|  |  | ||||||
|  | [](https://github.com/soybeanjs)  [](https://github.com/m-xlsea)   | ||||||
|  |  | ||||||
|  | ## [v1.1.4](https://github.com/honghuangdc/soybean-admin/compare/v1.1.3...v1.1.4) (2024-06-06) | ||||||
|  |  | ||||||
|  | ###    🐞 Bug Fixes | ||||||
|  |  | ||||||
|  | - **utils**: modalLogout bug when esc is pressed  -  by @sigma-plus in https://github.com/honghuangdc/soybean-admin/issues/470 [<samp>(bd69c)</samp>](https://github.com/honghuangdc/soybean-admin/commit/bd69c00) | ||||||
|  |  | ||||||
|  | ###    🛠 Optimizations | ||||||
|  |  | ||||||
|  | - **projects**: optimize RouteMeta remarks  -  by @soybeanjs [<samp>(ffb48)</samp>](https://github.com/honghuangdc/soybean-admin/commit/ffb48b1) | ||||||
|  |  | ||||||
|  | ###    📖 Documentation | ||||||
|  |  | ||||||
|  | - **projects**: | ||||||
|  |   - update CHANGELOG  -  by @soybeanjs [<samp>(756f8)</samp>](https://github.com/honghuangdc/soybean-admin/commit/756f84a) | ||||||
|  |   - update Node&pnpm version  -  by @Azir-11 in https://github.com/honghuangdc/soybean-admin/issues/472 [<samp>(9b05d)</samp>](https://github.com/honghuangdc/soybean-admin/commit/9b05d73) | ||||||
|  |  | ||||||
|  | ###    🏡 Chore | ||||||
|  |  | ||||||
|  | - **deps**: | ||||||
|  |   - update deps  -  by @soybeanjs [<samp>(d0380)</samp>](https://github.com/honghuangdc/soybean-admin/commit/d0380ce) | ||||||
|  |   - update deps  -  by @soybeanjs [<samp>(1f464)</samp>](https://github.com/honghuangdc/soybean-admin/commit/1f4647b) | ||||||
|  | - **projects**: | ||||||
|  |   - close http proxy  -  by @soybeanjs [<samp>(d08a3)</samp>](https://github.com/honghuangdc/soybean-admin/commit/d08a381) | ||||||
|  |   - update mock url  -  by @soybeanjs [<samp>(e6086)</samp>](https://github.com/honghuangdc/soybean-admin/commit/e6086f0) | ||||||
|  |   - update vscode settings  -  by @soybeanjs [<samp>(910df)</samp>](https://github.com/honghuangdc/soybean-admin/commit/910dfca) | ||||||
|  |  | ||||||
|  | ###    ❤️ Contributors | ||||||
|  |  | ||||||
|  | [](https://github.com/soybeanjs)  [](https://github.com/Azir-11)  [](https://github.com/sigma-plus)   | ||||||
|  |  | ||||||
| ## [v1.1.3](https://github.com/soybeanjs/soybean-admin/compare/v1.1.2...v1.1.3) (2024-06-02) | ## [v1.1.3](https://github.com/soybeanjs/soybean-admin/compare/v1.1.2...v1.1.3) (2024-06-02) | ||||||
|  |  | ||||||
| ###    🐞 Bug Fixes | ###    🐞 Bug Fixes | ||||||
|   | |||||||
| @@ -1,5 +1,67 @@ | |||||||
| # 更新日志 | # 更新日志 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## [v1.2.0](https://github.com/soybeanjs/soybean-admin/compare/v1.1.5...v1.2.0) (2024-06-06) | ||||||
|  |  | ||||||
|  | ###    🚀 功能 | ||||||
|  |  | ||||||
|  | - **项目**: | ||||||
|  |   - 支持系统新版本更新通知。关闭 #420  -  由 @soybeanjs 在 https://github.com/soybeanjs/soybean-admin/issues/420 [<samp>(584cd)</samp>](https://github.com/soybeanjs/soybean-admin/commit/584cd54) | ||||||
|  |   - 在路由守卫中获取用户信息并从localStorage中移除。关闭 #459  -  由 @soybeanjs 在 https://github.com/soybeanjs/soybean-admin/issues/459 [<samp>(5531a)</samp>](https://github.com/soybeanjs/soybean-admin/commit/5531a68) | ||||||
|  |  | ||||||
|  | ###    📖 文档 | ||||||
|  |  | ||||||
|  | - **项目**: 更新CHANGELOG  -  由 @soybeanjs [<samp>(2bec8)</samp>](https://github.com/soybeanjs/soybean-admin/commit/2bec899) | ||||||
|  |  | ||||||
|  | ###    ❤️ 贡献者 | ||||||
|  |  | ||||||
|  | [](https://github.com/soybeanjs)   | ||||||
|  |  | ||||||
|  | ## [v1.1.5](https://github.com/soybeanjs/soybean-admin/compare/v1.1.4...v1.1.5) (2024-06-06) | ||||||
|  |  | ||||||
|  | ###    🐞 错误修复 | ||||||
|  |  | ||||||
|  | - **项目**: 修复注册组件名,CodeLogin => Register  -  由 @m-xlsea 在 https://github.com/soybeanjs/soybean-admin/issues/478 [<samp>(ddf38)</samp>](https://github.com/soybeanjs/soybean-admin/commit/ddf3823) | ||||||
|  |  | ||||||
|  | ###    🏡 杂务 | ||||||
|  |  | ||||||
|  | - **依赖**: 更新依赖  -  由 @soybeanjs [<samp>(060c0)</samp>](https://github.com/soybeanjs/soybean-admin/commit/060c0a9) | ||||||
|  | - **项目**: 更新 vscode 设置: vue 官方  -  由 @soybeanjs [<samp>(76649)</samp>](https://github.com/soybeanjs/soybean-admin/commit/76649e2) | ||||||
|  |  | ||||||
|  | ###    ❤️ 贡献者 | ||||||
|  |  | ||||||
|  | [](https://github.com/soybeanjs)  [](https://github.com/m-xlsea)   | ||||||
|  |  | ||||||
|  | ## [v1.1.4](https://github.com/honghuangdc/soybean-admin/compare/v1.1.3...v1.1.4) (2024-06-06) | ||||||
|  |  | ||||||
|  | ###    🐞 错误修复 | ||||||
|  |  | ||||||
|  | - **utils**: 修复了按esc键时modalLogout的错误  -  由 @sigma-plus 在 https://github.com/honghuangdc/soybean-admin/issues/470 中提出 [<samp>(bd69c)</samp>](https://github.com/honghuangdc/soybean-admin/commit/bd69c00) | ||||||
|  |  | ||||||
|  | ###    🛠 优化 | ||||||
|  |  | ||||||
|  | - **projects**: 优化了RouteMeta的备注  -  由 @soybeanjs 提出 [<samp>(ffb48)</samp>](https://github.com/honghuangdc/soybean-admin/commit/ffb48b1) | ||||||
|  |  | ||||||
|  | ###    📖 文档 | ||||||
|  |  | ||||||
|  | - **projects**: | ||||||
|  |   - 更新了CHANGELOG  -  由 @soybeanjs 提出 [<samp>(756f8)</samp>](https://github.com/honghuangdc/soybean-admin/commit/756f84a) | ||||||
|  |   - 更新了Node&pnpm版本  -  由 @Azir-11 在 https://github.com/honghuangdc/soybean-admin/issues/472 中提出 [<samp>(9b05d)</samp>](https://github.com/honghuangdc/soybean-admin/commit/9b05d73) | ||||||
|  |  | ||||||
|  | ###    🏡 杂项 | ||||||
|  |  | ||||||
|  | - **deps**: | ||||||
|  |   - 更新了依赖  -  由 @soybeanjs 提出 [<samp>(d0380)</samp>](https://github.com/honghuangdc/soybean-admin/commit/d0380ce) | ||||||
|  |   - 更新了依赖  -  由 @soybeanjs 提出 [<samp>(1f464)</samp>](https://github.com/honghuangdc/soybean-admin/commit/1f4647b) | ||||||
|  | - **projects**: | ||||||
|  |   - 关闭了http代理  -  由 @soybeanjs 提出 [<samp>(d08a3)</samp>](https://github.com/honghuangdc/soybean-admin/commit/d08a381) | ||||||
|  |   - 更新了mock url  -  由 @soybeanjs 提出 [<samp>(e6086)</samp>](https://github.com/honghuangdc/soybean-admin/commit/e6086f0) | ||||||
|  |   - 更新了vscode设置  -  由 @soybeanjs 提出 [<samp>(910df)</samp>](https://github.com/honghuangdc/soybean-admin/commit/910dfca) | ||||||
|  |  | ||||||
|  | ###    ❤️ 贡献者 | ||||||
|  |  | ||||||
|  | [](https://github.com/soybeanjs)  [](https://github.com/Azir-11)  [](https://github.com/sigma-plus)   | ||||||
|  |  | ||||||
| ## [v1.1.3](https://github.com/soybeanjs/soybean-admin/compare/v1.1.2...v1.1.3) (2024-06-02) | ## [v1.1.3](https://github.com/soybeanjs/soybean-admin/compare/v1.1.2...v1.1.3) (2024-06-02) | ||||||
|  |  | ||||||
| ###    🐞 错误修复 | ###    🐞 错误修复 | ||||||
|   | |||||||
| @@ -78,8 +78,8 @@ | |||||||
| Make sure your environment meets the following requirements: | Make sure your environment meets the following requirements: | ||||||
|  |  | ||||||
| - **git**: you need git to clone and manage project versions. | - **git**: you need git to clone and manage project versions. | ||||||
| - **NodeJS**: >=18.0.0, recommended 18.19.0 or higher. | - **NodeJS**: >=18.12.0, recommended 18.19.0 or higher. | ||||||
| - **pnpm**: >= 8.0.0, recommended 8.14.0 or higher. | - **pnpm**: >= 8.7.0, recommended 8.14.0 or higher. | ||||||
|  |  | ||||||
| **Clone Project** | **Clone Project** | ||||||
|  |  | ||||||
|   | |||||||
| @@ -77,8 +77,8 @@ | |||||||
| 确保你的环境满足以下要求: | 确保你的环境满足以下要求: | ||||||
|  |  | ||||||
| - **git**: 你需要git来克隆和管理项目版本。 | - **git**: 你需要git来克隆和管理项目版本。 | ||||||
| - **NodeJS**: >=18.0.0,推荐 18.19.0 或更高。 | - **NodeJS**: >=18.12.0,推荐 18.19.0 或更高。 | ||||||
| - **pnpm**: >= 8.0.0,推荐 8.14.0 或更高。 | - **pnpm**: >= 8.7.0,推荐 8.14.0 或更高。 | ||||||
|  |  | ||||||
| **克隆项目** | **克隆项目** | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										13
									
								
								build/plugins/html.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								build/plugins/html.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | |||||||
|  | import type { Plugin } from 'vite'; | ||||||
|  |  | ||||||
|  | export function setupHtmlPlugin(buildTime: string) { | ||||||
|  |   const plugin: Plugin = { | ||||||
|  |     name: 'html-plugin', | ||||||
|  |     apply: 'build', | ||||||
|  |     transformIndexHtml(html) { | ||||||
|  |       return html.replace('<head>', `<head>\n    <meta name="buildTime" content="${buildTime}">`); | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   return plugin; | ||||||
|  | } | ||||||
| @@ -6,8 +6,9 @@ import progress from 'vite-plugin-progress'; | |||||||
| import { setupElegantRouter } from './router'; | import { setupElegantRouter } from './router'; | ||||||
| import { setupUnocss } from './unocss'; | import { setupUnocss } from './unocss'; | ||||||
| import { setupUnplugin } from './unplugin'; | import { setupUnplugin } from './unplugin'; | ||||||
|  | import { setupHtmlPlugin } from './html'; | ||||||
|  |  | ||||||
| export function setupVitePlugins(viteEnv: Env.ImportMeta) { | export function setupVitePlugins(viteEnv: Env.ImportMeta, buildTime: string) { | ||||||
|   const plugins: PluginOption = [ |   const plugins: PluginOption = [ | ||||||
|     vue({ |     vue({ | ||||||
|       script: { |       script: { | ||||||
| @@ -19,7 +20,8 @@ export function setupVitePlugins(viteEnv: Env.ImportMeta) { | |||||||
|     setupElegantRouter(), |     setupElegantRouter(), | ||||||
|     setupUnocss(viteEnv), |     setupUnocss(viteEnv), | ||||||
|     ...setupUnplugin(viteEnv), |     ...setupUnplugin(viteEnv), | ||||||
|     progress() |     progress(), | ||||||
|  |     setupHtmlPlugin(buildTime) | ||||||
|   ]; |   ]; | ||||||
|  |  | ||||||
|   return plugins; |   return plugins; | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ export default defineConfig( | |||||||
|       'vue/multi-word-component-names': [ |       'vue/multi-word-component-names': [ | ||||||
|         'warn', |         'warn', | ||||||
|         { |         { | ||||||
|           ignores: ['index', 'App', '[id]', '[url]'] |           ignores: ['index', 'App', 'Register', '[id]', '[url]'] | ||||||
|         } |         } | ||||||
|       ], |       ], | ||||||
|       'vue/component-name-in-template-casing': [ |       'vue/component-name-in-template-casing': [ | ||||||
|   | |||||||
							
								
								
									
										44
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										44
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,7 +1,7 @@ | |||||||
| { | { | ||||||
|   "name": "soybean-admin", |   "name": "soybean-admin", | ||||||
|   "type": "module", |   "type": "module", | ||||||
|   "version": "1.1.3", |   "version": "1.2.1", | ||||||
|   "description": "A fresh and elegant admin template, based on Vue3、Vite3、TypeScript、NaiveUI and UnoCSS. 一个基于Vue3、Vite3、TypeScript、NaiveUI and UnoCSS的清新优雅的中后台模版。", |   "description": "A fresh and elegant admin template, based on Vue3、Vite3、TypeScript、NaiveUI and UnoCSS. 一个基于Vue3、Vite3、TypeScript、NaiveUI and UnoCSS的清新优雅的中后台模版。", | ||||||
|   "author": { |   "author": { | ||||||
|     "name": "Soybean", |     "name": "Soybean", | ||||||
| @@ -56,7 +56,7 @@ | |||||||
|     "@sa/hooks": "workspace:*", |     "@sa/hooks": "workspace:*", | ||||||
|     "@sa/materials": "workspace:*", |     "@sa/materials": "workspace:*", | ||||||
|     "@sa/utils": "workspace:*", |     "@sa/utils": "workspace:*", | ||||||
|     "@vueuse/core": "10.9.0", |     "@vueuse/core": "10.10.0", | ||||||
|     "clipboard": "2.0.11", |     "clipboard": "2.0.11", | ||||||
|     "dayjs": "1.11.11", |     "dayjs": "1.11.11", | ||||||
|     "echarts": "5.5.0", |     "echarts": "5.5.0", | ||||||
| @@ -66,42 +66,42 @@ | |||||||
|     "pinia": "2.1.7", |     "pinia": "2.1.7", | ||||||
|     "tailwind-merge": "^2.3.0", |     "tailwind-merge": "^2.3.0", | ||||||
|     "vue": "3.4.27", |     "vue": "3.4.27", | ||||||
|     "vue-draggable-plus": "0.4.1", |     "vue-draggable-plus": "0.5.0", | ||||||
|     "vue-i18n": "9.13.1", |     "vue-i18n": "9.13.1", | ||||||
|     "vue-router": "4.3.2" |     "vue-router": "4.3.2" | ||||||
|   }, |   }, | ||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
|     "@elegant-router/vue": "0.3.7", |     "@elegant-router/vue": "0.3.7", | ||||||
|     "@iconify/json": "2.2.211", |     "@iconify/json": "2.2.217", | ||||||
|     "@sa/scripts": "workspace:*", |     "@sa/scripts": "workspace:*", | ||||||
|     "@sa/uno-preset": "workspace:*", |     "@sa/uno-preset": "workspace:*", | ||||||
|     "@soybeanjs/eslint-config": "1.3.5", |     "@soybeanjs/eslint-config": "1.3.6", | ||||||
|     "@tauri-apps/cli": "1.5.11", |     "@tauri-apps/cli": "1.5.14", | ||||||
|     "@types/lodash-es": "4.17.12", |     "@types/lodash-es": "4.17.12", | ||||||
|     "@types/node": "20.12.12", |     "@types/node": "20.14.2", | ||||||
|     "@types/nprogress": "0.2.3", |     "@types/nprogress": "0.2.3", | ||||||
|     "@unocss/eslint-config": "0.60.2", |     "@unocss/eslint-config": "0.60.4", | ||||||
|     "@unocss/preset-icons": "0.60.2", |     "@unocss/preset-icons": "0.60.4", | ||||||
|     "@unocss/preset-uno": "0.60.2", |     "@unocss/preset-uno": "0.60.4", | ||||||
|     "@unocss/transformer-directives": "0.60.2", |     "@unocss/transformer-directives": "0.60.4", | ||||||
|     "@unocss/transformer-variant-group": "0.60.2", |     "@unocss/transformer-variant-group": "0.60.4", | ||||||
|     "@unocss/vite": "0.60.2", |     "@unocss/vite": "0.60.4", | ||||||
|     "@vitejs/plugin-vue": "5.0.4", |     "@vitejs/plugin-vue": "5.0.5", | ||||||
|     "@vitejs/plugin-vue-jsx": "3.1.0", |     "@vitejs/plugin-vue-jsx": "4.0.0", | ||||||
|     "eslint": "9.3.0", |     "eslint": "9.4.0", | ||||||
|     "eslint-plugin-vue": "9.26.0", |     "eslint-plugin-vue": "9.26.0", | ||||||
|     "lint-staged": "15.2.2", |     "lint-staged": "15.2.5", | ||||||
|     "sass": "1.77.2", |     "sass": "1.77.4", | ||||||
|     "simple-git-hooks": "2.11.1", |     "simple-git-hooks": "2.11.1", | ||||||
|     "tsx": "4.10.5", |     "tsx": "4.12.0", | ||||||
|     "typescript": "5.4.5", |     "typescript": "5.4.5", | ||||||
|     "unplugin-icons": "0.19.0", |     "unplugin-icons": "0.19.0", | ||||||
|     "unplugin-vue-components": "0.27.0", |     "unplugin-vue-components": "0.27.0", | ||||||
|     "vite": "5.2.11", |     "vite": "5.2.12", | ||||||
|     "vite-plugin-progress": "0.0.7", |     "vite-plugin-progress": "0.0.7", | ||||||
|     "vite-plugin-svg-icons": "2.0.1", |     "vite-plugin-svg-icons": "2.0.1", | ||||||
|     "vite-plugin-vue-devtools": "7.2.0", |     "vite-plugin-vue-devtools": "7.2.1", | ||||||
|     "vue-eslint-parser": "9.4.2", |     "vue-eslint-parser": "9.4.3", | ||||||
|     "vue-tsc": "2.0.19" |     "vue-tsc": "2.0.19" | ||||||
|   }, |   }, | ||||||
|   "simple-git-hooks": { |   "simple-git-hooks": { | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| { | { | ||||||
|   "name": "@sa/axios", |   "name": "@sa/axios", | ||||||
|   "version": "1.1.3", |   "version": "1.2.1", | ||||||
|   "exports": { |   "exports": { | ||||||
|     ".": "./src/index.ts" |     ".": "./src/index.ts" | ||||||
|   }, |   }, | ||||||
| @@ -11,8 +11,8 @@ | |||||||
|   }, |   }, | ||||||
|   "dependencies": { |   "dependencies": { | ||||||
|     "@sa/utils": "workspace:*", |     "@sa/utils": "workspace:*", | ||||||
|     "axios": "1.6.8", |     "axios": "1.7.2", | ||||||
|     "axios-retry": "4.2.0", |     "axios-retry": "4.4.0", | ||||||
|     "qs": "6.12.1" |     "qs": "6.12.1" | ||||||
|   }, |   }, | ||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| { | { | ||||||
|   "name": "@sa/color", |   "name": "@sa/color", | ||||||
|   "version": "1.1.3", |   "version": "1.2.1", | ||||||
|   "exports": { |   "exports": { | ||||||
|     ".": "./src/index.ts" |     ".": "./src/index.ts" | ||||||
|   }, |   }, | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| { | { | ||||||
|   "name": "@sa/hooks", |   "name": "@sa/hooks", | ||||||
|   "version": "1.1.3", |   "version": "1.2.1", | ||||||
|   "exports": { |   "exports": { | ||||||
|     ".": "./src/index.ts" |     ".": "./src/index.ts" | ||||||
|   }, |   }, | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| { | { | ||||||
|   "name": "@sa/materials", |   "name": "@sa/materials", | ||||||
|   "version": "1.1.3", |   "version": "1.2.1", | ||||||
|   "exports": { |   "exports": { | ||||||
|     ".": "./src/index.ts" |     ".": "./src/index.ts" | ||||||
|   }, |   }, | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| { | { | ||||||
|   "name": "@sa/fetch", |   "name": "@sa/fetch", | ||||||
|   "version": "1.1.3", |   "version": "1.2.1", | ||||||
|   "exports": { |   "exports": { | ||||||
|     ".": "./src/index.ts" |     ".": "./src/index.ts" | ||||||
|   }, |   }, | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| { | { | ||||||
|   "name": "@sa/scripts", |   "name": "@sa/scripts", | ||||||
|   "version": "1.1.3", |   "version": "1.2.1", | ||||||
|   "bin": { |   "bin": { | ||||||
|     "sa": "./bin.ts" |     "sa": "./bin.ts" | ||||||
|   }, |   }, | ||||||
| @@ -13,13 +13,13 @@ | |||||||
|     } |     } | ||||||
|   }, |   }, | ||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
|     "@soybeanjs/changelog": "0.3.23", |     "@soybeanjs/changelog": "0.3.24", | ||||||
|     "bumpp": "9.4.1", |     "bumpp": "9.4.1", | ||||||
|     "c12": "1.10.0", |     "c12": "1.10.0", | ||||||
|     "cac": "6.7.14", |     "cac": "6.7.14", | ||||||
|     "consola": "3.2.3", |     "consola": "3.2.3", | ||||||
|     "enquirer": "2.4.1", |     "enquirer": "2.4.1", | ||||||
|     "execa": "9.1.0", |     "execa": "9.2.0", | ||||||
|     "kolorist": "1.8.0", |     "kolorist": "1.8.0", | ||||||
|     "npm-check-updates": "16.14.20", |     "npm-check-updates": "16.14.20", | ||||||
|     "rimraf": "5.0.7" |     "rimraf": "5.0.7" | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| { | { | ||||||
|   "name": "@sa/uno-preset", |   "name": "@sa/uno-preset", | ||||||
|   "version": "1.1.3", |   "version": "1.2.1", | ||||||
|   "exports": { |   "exports": { | ||||||
|     ".": "./src/index.ts" |     ".": "./src/index.ts" | ||||||
|   }, |   }, | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| { | { | ||||||
|   "name": "@sa/utils", |   "name": "@sa/utils", | ||||||
|   "version": "1.1.3", |   "version": "1.2.1", | ||||||
|   "exports": { |   "exports": { | ||||||
|     ".": "./src/index.ts" |     ".": "./src/index.ts" | ||||||
|   }, |   }, | ||||||
|   | |||||||
							
								
								
									
										1708
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										1708
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,6 +1,10 @@ | |||||||
| const local: App.I18n.Schema = { | const local: App.I18n.Schema = { | ||||||
|   system: { |   system: { | ||||||
|     title: 'SoybeanAdmin' |     title: 'SoybeanAdmin', | ||||||
|  |     updateTitle: 'System Version Update Notification', | ||||||
|  |     updateContent: 'A new version of the system has been detected. Do you want to refresh the page immediately?', | ||||||
|  |     updateConfirm: 'Refresh immediately', | ||||||
|  |     updateCancel: 'Later' | ||||||
|   }, |   }, | ||||||
|   common: { |   common: { | ||||||
|     action: 'Action', |     action: 'Action', | ||||||
|   | |||||||
| @@ -1,6 +1,10 @@ | |||||||
| const local: App.I18n.Schema = { | const local: App.I18n.Schema = { | ||||||
|   system: { |   system: { | ||||||
|     title: 'Soybean 管理系统' |     title: 'Soybean 管理系统', | ||||||
|  |     updateTitle: '系统版本更新通知', | ||||||
|  |     updateContent: '检测到系统有新版本发布,是否立即刷新页面?', | ||||||
|  |     updateConfirm: '立即刷新', | ||||||
|  |     updateCancel: '稍后再说' | ||||||
|   }, |   }, | ||||||
|   common: { |   common: { | ||||||
|     action: '操作', |     action: '操作', | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| import { createApp } from 'vue'; | import { createApp } from 'vue'; | ||||||
| import './plugins/assets'; | import './plugins/assets'; | ||||||
| import { setupDayjs, setupIconifyOffline, setupLoading, setupNProgress } from './plugins'; | import { setupAppVersionNotification, setupDayjs, setupIconifyOffline, setupLoading, setupNProgress } from './plugins'; | ||||||
| import { setupStore } from './store'; | import { setupStore } from './store'; | ||||||
| import { setupRouter } from './router'; | import { setupRouter } from './router'; | ||||||
| import { setupI18n } from './locales'; | import { setupI18n } from './locales'; | ||||||
| @@ -23,6 +23,8 @@ async function setupApp() { | |||||||
|  |  | ||||||
|   setupI18n(app); |   setupI18n(app); | ||||||
|  |  | ||||||
|  |   setupAppVersionNotification(); | ||||||
|  |  | ||||||
|   app.mount('#app'); |   app.mount('#app'); | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										53
									
								
								src/plugins/app.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								src/plugins/app.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,53 @@ | |||||||
|  | import { h } from 'vue'; | ||||||
|  | import { NButton } from 'naive-ui'; | ||||||
|  | import { $t } from '../locales'; | ||||||
|  |  | ||||||
|  | export function setupAppVersionNotification() { | ||||||
|  |   document.addEventListener('visibilitychange', async () => { | ||||||
|  |     const buildTime = await getHtmlBuildTime(); | ||||||
|  |  | ||||||
|  |     if (buildTime !== BUILD_TIME && document.visibilityState === 'visible') { | ||||||
|  |       const n = window.$notification?.create({ | ||||||
|  |         title: $t('system.updateTitle'), | ||||||
|  |         content: $t('system.updateContent'), | ||||||
|  |         action() { | ||||||
|  |           return h('div', { style: { display: 'flex', justifyContent: 'end', gap: '12px', width: '325px' } }, [ | ||||||
|  |             h( | ||||||
|  |               NButton, | ||||||
|  |               { | ||||||
|  |                 onClick() { | ||||||
|  |                   n?.destroy(); | ||||||
|  |                 } | ||||||
|  |               }, | ||||||
|  |               () => $t('system.updateCancel') | ||||||
|  |             ), | ||||||
|  |             h( | ||||||
|  |               NButton, | ||||||
|  |               { | ||||||
|  |                 type: 'primary', | ||||||
|  |                 onClick() { | ||||||
|  |                   location.reload(); | ||||||
|  |                 } | ||||||
|  |               }, | ||||||
|  |               () => $t('system.updateConfirm') | ||||||
|  |             ) | ||||||
|  |           ]); | ||||||
|  |         } | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function getHtmlBuildTime() { | ||||||
|  |   const baseURL = import.meta.env.VITE_BASE_URL; | ||||||
|  |  | ||||||
|  |   const res = await fetch(`${baseURL}index.html`); | ||||||
|  |  | ||||||
|  |   const html = await res.text(); | ||||||
|  |  | ||||||
|  |   const match = html.match(/<meta name="buildTime" content="(.*)">/); | ||||||
|  |  | ||||||
|  |   const buildTime = match?.[1] || ''; | ||||||
|  |  | ||||||
|  |   return buildTime; | ||||||
|  | } | ||||||
| @@ -2,3 +2,4 @@ export * from './loading'; | |||||||
| export * from './nprogress'; | export * from './nprogress'; | ||||||
| export * from './iconify'; | export * from './iconify'; | ||||||
| export * from './dayjs'; | export * from './dayjs'; | ||||||
|  | export * from './app'; | ||||||
|   | |||||||
| @@ -92,6 +92,7 @@ export function createRouteGuard(router: Router) { | |||||||
|  * @param to to route |  * @param to to route | ||||||
|  */ |  */ | ||||||
| async function initRoute(to: RouteLocationNormalized): Promise<RouteLocationRaw | null> { | async function initRoute(to: RouteLocationNormalized): Promise<RouteLocationRaw | null> { | ||||||
|  |   const authStore = useAuthStore(); | ||||||
|   const routeStore = useRouteStore(); |   const routeStore = useRouteStore(); | ||||||
|  |  | ||||||
|   const notFoundRoute: RouteKey = 'not-found'; |   const notFoundRoute: RouteKey = 'not-found'; | ||||||
| @@ -160,6 +161,8 @@ async function initRoute(to: RouteLocationNormalized): Promise<RouteLocationRaw | |||||||
|     return location; |     return location; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   await authStore.initUserInfo(); | ||||||
|  |  | ||||||
|   // initialize the auth route |   // initialize the auth route | ||||||
|   await routeStore.initAuthRoute(); |   await routeStore.initAuthRoute(); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -64,9 +64,10 @@ export const request = createFlatRequest<App.Service.Response, RequestInstanceSt | |||||||
|  |  | ||||||
|         window.$dialog?.error({ |         window.$dialog?.error({ | ||||||
|           title: 'Error', |           title: 'Error', | ||||||
|           content: response.data.code, |           content: response.data.msg, | ||||||
|           positiveText: $t('common.confirm'), |           positiveText: $t('common.confirm'), | ||||||
|           maskClosable: false, |           maskClosable: false, | ||||||
|  |           closeOnEsc: false, | ||||||
|           onPositiveClick() { |           onPositiveClick() { | ||||||
|             logoutAndCleanup(); |             logoutAndCleanup(); | ||||||
|           }, |           }, | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ import { fetchGetUserInfo, fetchLogin } from '@/service/api'; | |||||||
| import { localStg } from '@/utils/storage'; | import { localStg } from '@/utils/storage'; | ||||||
| import { $t } from '@/locales'; | import { $t } from '@/locales'; | ||||||
| import { useRouteStore } from '../route'; | import { useRouteStore } from '../route'; | ||||||
| import { clearAuthStorage, getToken, getUserInfo } from './shared'; | import { clearAuthStorage, getToken } from './shared'; | ||||||
|  |  | ||||||
| export const useAuthStore = defineStore(SetupStoreId.Auth, () => { | export const useAuthStore = defineStore(SetupStoreId.Auth, () => { | ||||||
|   const route = useRoute(); |   const route = useRoute(); | ||||||
| @@ -18,7 +18,12 @@ export const useAuthStore = defineStore(SetupStoreId.Auth, () => { | |||||||
|  |  | ||||||
|   const token = ref(getToken()); |   const token = ref(getToken()); | ||||||
|  |  | ||||||
|   const userInfo: Api.Auth.UserInfo = reactive(getUserInfo()); |   const userInfo: Api.Auth.UserInfo = reactive({ | ||||||
|  |     userId: '', | ||||||
|  |     userName: '', | ||||||
|  |     roles: [], | ||||||
|  |     buttons: [] | ||||||
|  |   }); | ||||||
|  |  | ||||||
|   /** is super role in static route */ |   /** is super role in static route */ | ||||||
|   const isStaticSuper = computed(() => { |   const isStaticSuper = computed(() => { | ||||||
| @@ -87,14 +92,23 @@ export const useAuthStore = defineStore(SetupStoreId.Auth, () => { | |||||||
|     localStg.set('token', loginToken.token); |     localStg.set('token', loginToken.token); | ||||||
|     localStg.set('refreshToken', loginToken.refreshToken); |     localStg.set('refreshToken', loginToken.refreshToken); | ||||||
|  |  | ||||||
|  |     // 2. get user info | ||||||
|  |     const pass = await getUserInfo(); | ||||||
|  |  | ||||||
|  |     if (pass) { | ||||||
|  |       token.value = loginToken.token; | ||||||
|  |  | ||||||
|  |       return true; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return false; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   async function getUserInfo() { | ||||||
|     const { data: info, error } = await fetchGetUserInfo(); |     const { data: info, error } = await fetchGetUserInfo(); | ||||||
|  |  | ||||||
|     if (!error) { |     if (!error) { | ||||||
|       // 2. store user info |       // update store | ||||||
|       localStg.set('userInfo', info); |  | ||||||
|  |  | ||||||
|       // 3. update store |  | ||||||
|       token.value = loginToken.token; |  | ||||||
|       Object.assign(userInfo, info); |       Object.assign(userInfo, info); | ||||||
|  |  | ||||||
|       return true; |       return true; | ||||||
| @@ -103,6 +117,18 @@ export const useAuthStore = defineStore(SetupStoreId.Auth, () => { | |||||||
|     return false; |     return false; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   async function initUserInfo() { | ||||||
|  |     const hasToken = getToken(); | ||||||
|  |  | ||||||
|  |     if (hasToken) { | ||||||
|  |       const pass = await getUserInfo(); | ||||||
|  |  | ||||||
|  |       if (!pass) { | ||||||
|  |         resetStore(); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|   return { |   return { | ||||||
|     token, |     token, | ||||||
|     userInfo, |     userInfo, | ||||||
| @@ -110,6 +136,7 @@ export const useAuthStore = defineStore(SetupStoreId.Auth, () => { | |||||||
|     isLogin, |     isLogin, | ||||||
|     loginLoading, |     loginLoading, | ||||||
|     resetStore, |     resetStore, | ||||||
|     login |     login, | ||||||
|  |     initUserInfo | ||||||
|   }; |   }; | ||||||
| }); | }); | ||||||
|   | |||||||
| @@ -5,27 +5,8 @@ export function getToken() { | |||||||
|   return localStg.get('token') || ''; |   return localStg.get('token') || ''; | ||||||
| } | } | ||||||
|  |  | ||||||
| /** Get user info */ |  | ||||||
| export function getUserInfo() { |  | ||||||
|   const emptyInfo: Api.Auth.UserInfo = { |  | ||||||
|     userId: '', |  | ||||||
|     userName: '', |  | ||||||
|     roles: [], |  | ||||||
|     buttons: [] |  | ||||||
|   }; |  | ||||||
|   const userInfo = localStg.get('userInfo') || emptyInfo; |  | ||||||
|  |  | ||||||
|   // fix new property: buttons, this will be removed in the next version `1.1.0` |  | ||||||
|   if (!userInfo.buttons) { |  | ||||||
|     userInfo.buttons = []; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   return userInfo; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** Clear auth storage */ | /** Clear auth storage */ | ||||||
| export function clearAuthStorage() { | export function clearAuthStorage() { | ||||||
|   localStg.remove('token'); |   localStg.remove('token'); | ||||||
|   localStg.remove('refreshToken'); |   localStg.remove('refreshToken'); | ||||||
|   localStg.remove('userInfo'); |  | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										4
									
								
								src/typings/app.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								src/typings/app.d.ts
									
									
									
									
										vendored
									
									
								
							| @@ -251,6 +251,10 @@ declare namespace App { | |||||||
|     type Schema = { |     type Schema = { | ||||||
|       system: { |       system: { | ||||||
|         title: string; |         title: string; | ||||||
|  |         updateTitle: string; | ||||||
|  |         updateContent: string; | ||||||
|  |         updateConfirm: string; | ||||||
|  |         updateCancel: string; | ||||||
|       }; |       }; | ||||||
|       common: { |       common: { | ||||||
|         action: string; |         action: string; | ||||||
|   | |||||||
							
								
								
									
										7
									
								
								src/typings/router.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										7
									
								
								src/typings/router.d.ts
									
									
									
									
										vendored
									
									
								
							| @@ -27,7 +27,7 @@ declare module 'vue-router' { | |||||||
|     /** |     /** | ||||||
|      * Is constant route |      * Is constant route | ||||||
|      * |      * | ||||||
|      * Does not need to login, and the route is defined in the front-end |      * when it is set to true, there will be no login verification and no permission verification to access the route | ||||||
|      */ |      */ | ||||||
|     constant?: boolean | null; |     constant?: boolean | null; | ||||||
|     /** |     /** | ||||||
| @@ -57,7 +57,10 @@ declare module 'vue-router' { | |||||||
|      *   the route is "user_detail", if it is set to "user_list", the menu "user_list" will be activated |      *   the route is "user_detail", if it is set to "user_list", the menu "user_list" will be activated | ||||||
|      */ |      */ | ||||||
|     activeMenu?: import('@elegant-router/types').RouteKey | null; |     activeMenu?: import('@elegant-router/types').RouteKey | null; | ||||||
|     /** By default, the same route path will use one tab, if set to true, it will use multiple tabs */ |     /** | ||||||
|  |      * By default, the same route path will use one tab, even with different query, if set true, the route with | ||||||
|  |      * different query will use different tabs | ||||||
|  |      */ | ||||||
|     multiTab?: boolean | null; |     multiTab?: boolean | null; | ||||||
|     /** If set, the route will be fixed in tabs, and the value is the order of fixed tabs */ |     /** If set, the route will be fixed in tabs, and the value is the order of fixed tabs */ | ||||||
|     fixedIndexInTab?: number | null; |     fixedIndexInTab?: number | null; | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								src/typings/storage.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								src/typings/storage.d.ts
									
									
									
									
										vendored
									
									
								
							| @@ -18,8 +18,6 @@ declare namespace StorageType { | |||||||
|     mixSiderFixed: CommonType.YesOrNo; |     mixSiderFixed: CommonType.YesOrNo; | ||||||
|     /** The refresh token */ |     /** The refresh token */ | ||||||
|     refreshToken: string; |     refreshToken: string; | ||||||
|     /** The user info */ |  | ||||||
|     userInfo: Api.Auth.UserInfo; |  | ||||||
|     /** The theme color */ |     /** The theme color */ | ||||||
|     themeColor: string; |     themeColor: string; | ||||||
|     /** The theme settings */ |     /** The theme settings */ | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ import { useFormRules, useNaiveForm } from '@/hooks/common/form'; | |||||||
| import { useCaptcha } from '@/hooks/business/captcha'; | import { useCaptcha } from '@/hooks/business/captcha'; | ||||||
|  |  | ||||||
| defineOptions({ | defineOptions({ | ||||||
|   name: 'CodeLogin' |   name: 'Register' | ||||||
| }); | }); | ||||||
|  |  | ||||||
| const { toggleLoginModule } = useRouterPush(); | const { toggleLoginModule } = useRouterPush(); | ||||||
|   | |||||||
| @@ -25,7 +25,7 @@ export default defineConfig(configEnv => { | |||||||
|         } |         } | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     plugins: setupVitePlugins(viteEnv), |     plugins: setupVitePlugins(viteEnv, buildTime), | ||||||
|     define: { |     define: { | ||||||
|       BUILD_TIME: JSON.stringify(buildTime) |       BUILD_TIME: JSON.stringify(buildTime) | ||||||
|     }, |     }, | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user