Compare commits
	
		
			31 Commits
		
	
	
		
			v1.3.2
			...
			tauri-v1.3
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					b00ae45108 | ||
| 
						 | 
					75997c1655 | ||
| 
						 | 
					c4e16102cb | ||
| 
						 | 
					be8556cd2b | ||
| 
						 | 
					27b5222cfb | ||
| 
						 | 
					79b2a28b5a | ||
| 
						 | 
					531bfaf1b9 | ||
| 
						 | 
					c3f1f69833 | ||
| 
						 | 
					a44ea624f1 | ||
| 
						 | 
					7cb43fc3c2 | ||
| 
						 | 
					4c9f4e09b1 | ||
| 
						 | 
					9c012c7d13 | ||
| 
						 | 
					878d9c395c | ||
| 
						 | 
					3830ec7a69 | ||
| 
						 | 
					ef6cf93d96 | ||
| 
						 | 
					7fa55905fc | ||
| 
						 | 
					2fa400b113 | ||
| 
						 | 
					73e917ad59 | ||
| 
						 | 
					e9a2ee4a23 | ||
| 
						 | 
					a1c14a1570 | ||
| 
						 | 
					b16a96323a | ||
| 
						 | 
					26c93dff9a | ||
| 
						 | 
					5094f0eea8 | ||
| 
						 | 
					a8f923eb38 | ||
| 
						 | 
					e6aa25e9f8 | ||
| 
						 | 
					968370b1aa | ||
| 
						 | 
					cb1d44525d | ||
| 
						 | 
					939c512f02 | ||
| 
						 | 
					3f23993a83 | ||
| 
						 | 
					ebc838c3e0 | ||
| 
						 | 
					5646a50ddf | 
							
								
								
									
										5
									
								
								.env
									
									
									
									
									
								
							
							
						
						@@ -36,7 +36,7 @@ VITE_SERVICE_LOGOUT_CODES=8888,8889
 | 
			
		||||
VITE_SERVICE_MODAL_LOGOUT_CODES=7777,7778
 | 
			
		||||
 | 
			
		||||
# token expired codes of backend service, when the code is received, it will refresh the token and resend the request
 | 
			
		||||
VITE_SERVICE_EXPIRED_TOKEN_CODES=9999,9998
 | 
			
		||||
VITE_SERVICE_EXPIRED_TOKEN_CODES=9999,9998,3333
 | 
			
		||||
 | 
			
		||||
# when the route mode is static, the defined super role
 | 
			
		||||
VITE_STATIC_SUPER_ROLE=R_SUPER
 | 
			
		||||
@@ -46,3 +46,6 @@ VITE_SOURCE_MAP=N
 | 
			
		||||
 | 
			
		||||
# Used to differentiate storage across different domains
 | 
			
		||||
VITE_STORAGE_PREFIX=SOY_
 | 
			
		||||
 | 
			
		||||
# used to control whether the program automatically detects updates
 | 
			
		||||
VITE_AUTOMATICALLY_DETECT_UPDATE=Y
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										87
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						@@ -1,6 +1,93 @@
 | 
			
		||||
# Changelog
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
## [v1.3.5](https://github.com/soybeanjs/soybean-admin/compare/v1.3.4...v1.3.5) (2024-09-07)
 | 
			
		||||
 | 
			
		||||
###    🚀 Features
 | 
			
		||||
 | 
			
		||||
- **packages**:
 | 
			
		||||
  - @sa/axios: add response to flatRequest when success  -  by @soybeanjs [<samp>(c4e16)</samp>](https://github.com/soybeanjs/soybean-admin/commit/c4e1610)
 | 
			
		||||
- **projects**:
 | 
			
		||||
  - README.zh_CN.md 添加合作推广  -  by @PZ-18664918826 in https://github.com/soybeanjs/soybean-admin/issues/601 [<samp>(2fa40)</samp>](https://github.com/soybeanjs/soybean-admin/commit/2fa400b)
 | 
			
		||||
  - Add more commit types according to Apache specifications  -  by @Azir-11 in https://github.com/soybeanjs/soybean-admin/issues/610 [<samp>(878d9)</samp>](https://github.com/soybeanjs/soybean-admin/commit/878d9c3)
 | 
			
		||||
  - does the configuration support automatic updates. close#612  -  by @Azir-11 in https://github.com/soybeanjs/soybean-admin/issues/617 and https://github.com/soybeanjs/soybean-admin/issues/612 [<samp>(4c9f4)</samp>](https://github.com/soybeanjs/soybean-admin/commit/4c9f4e0)
 | 
			
		||||
  - add app error handler. close #587  -  by @soybeanjs in https://github.com/soybeanjs/soybean-admin/issues/587 [<samp>(be855)</samp>](https://github.com/soybeanjs/soybean-admin/commit/be8556c)
 | 
			
		||||
 | 
			
		||||
###    🐞 Bug Fixes
 | 
			
		||||
 | 
			
		||||
- **deps**:
 | 
			
		||||
  - move json5 from devDependencies to dependencies to support production usage  -  by @mufeng889 in https://github.com/soybeanjs/soybean-admin/issues/618 [<samp>(7cb43)</samp>](https://github.com/soybeanjs/soybean-admin/commit/7cb43fc)
 | 
			
		||||
- **projects**:
 | 
			
		||||
  - avoid retrieving cached HTML  -  by @Azir-11 in https://github.com/soybeanjs/soybean-admin/issues/605 [<samp>(ef6cf)</samp>](https://github.com/soybeanjs/soybean-admin/commit/ef6cf93)
 | 
			
		||||
  - fix login redirect  -  by @soybeanjs [<samp>(3830e)</samp>](https://github.com/soybeanjs/soybean-admin/commit/3830ec7)
 | 
			
		||||
  - fix vertical-mix-menu when sider collapse. fixed #608  -  by @soybeanjs in https://github.com/soybeanjs/soybean-admin/issues/608 [<samp>(c3f1f)</samp>](https://github.com/soybeanjs/soybean-admin/commit/c3f1f69)
 | 
			
		||||
  - fix breadcrumb when activeMenu is parent menu. fixed #589  -  by @soybeanjs in https://github.com/soybeanjs/soybean-admin/issues/589 [<samp>(79b2a)</samp>](https://github.com/soybeanjs/soybean-admin/commit/79b2a28)
 | 
			
		||||
  - fix refresh token when meet multi requests. fixed #581  -  by @soybeanjs in https://github.com/soybeanjs/soybean-admin/issues/581 [<samp>(27b52)</samp>](https://github.com/soybeanjs/soybean-admin/commit/27b5222)
 | 
			
		||||
- **types**:
 | 
			
		||||
  - fix the type of TableApiFn  -  by @Azir-11 in https://github.com/soybeanjs/soybean-admin/issues/599 [<samp>(26c93)</samp>](https://github.com/soybeanjs/soybean-admin/commit/26c93df)
 | 
			
		||||
 | 
			
		||||
###    🛠 Optimizations
 | 
			
		||||
 | 
			
		||||
- **projects**: optimize menu selectedKey  -  by @soybeanjs [<samp>(531bf)</samp>](https://github.com/soybeanjs/soybean-admin/commit/531bfaf)
 | 
			
		||||
 | 
			
		||||
###    📖 Documentation
 | 
			
		||||
 | 
			
		||||
- **projects**:
 | 
			
		||||
  - update README  -  by @mufeng889 in https://github.com/soybeanjs/soybean-admin/issues/594 [<samp>(a8f92)</samp>](https://github.com/soybeanjs/soybean-admin/commit/a8f923e)
 | 
			
		||||
  - update README  -  by @soybeanjs [<samp>(e9a2e)</samp>](https://github.com/soybeanjs/soybean-admin/commit/e9a2ee4)
 | 
			
		||||
  - update README  -  by @soybeanjs [<samp>(73e91)</samp>](https://github.com/soybeanjs/soybean-admin/commit/73e917a)
 | 
			
		||||
  - update the location of important information in the document  -  by **Azir** [<samp>(9c012)</samp>](https://github.com/soybeanjs/soybean-admin/commit/9c012c7)
 | 
			
		||||
 | 
			
		||||
###    🏡 Chore
 | 
			
		||||
 | 
			
		||||
- **deps**:
 | 
			
		||||
  - update deps  -  by @soybeanjs [<samp>(a1c14)</samp>](https://github.com/soybeanjs/soybean-admin/commit/a1c14a1)
 | 
			
		||||
  - update deps  -  by @soybeanjs [<samp>(7fa55)</samp>](https://github.com/soybeanjs/soybean-admin/commit/7fa5590)
 | 
			
		||||
  - update deps  -  by @soybeanjs [<samp>(a44ea)</samp>](https://github.com/soybeanjs/soybean-admin/commit/a44ea62)
 | 
			
		||||
- **projects**:
 | 
			
		||||
  - use json5 resolve env `VITE_OTHER_SERVICE_BASE_URL` & fix proxy enable  -  by @soybeanjs [<samp>(b16a9)</samp>](https://github.com/soybeanjs/soybean-admin/commit/b16a963)
 | 
			
		||||
 | 
			
		||||
###    🎨 Styles
 | 
			
		||||
 | 
			
		||||
- **projects**: rename script czh to commit:zh  -  by @Azir-11 in https://github.com/soybeanjs/soybean-admin/issues/597 [<samp>(5094f)</samp>](https://github.com/soybeanjs/soybean-admin/commit/5094f0e)
 | 
			
		||||
 | 
			
		||||
###    ❤️ Contributors
 | 
			
		||||
 | 
			
		||||
[](https://github.com/soybeanjs)  [](https://github.com/mufeng889)  [](https://github.com/Azir-11)  [](https://github.com/PZ-18664918826)  
 | 
			
		||||
[Azir](mailto:2075125282@qq.com), 
 | 
			
		||||
 | 
			
		||||
## [v1.3.4](https://github.com/honghuangdc/soybean-admin/compare/v1.3.3...v1.3.4) (2024-08-01)
 | 
			
		||||
 | 
			
		||||
###    🚨 Breaking Changes
 | 
			
		||||
 | 
			
		||||
- **projects**: don't reset active menu of vertical-mix layout when it is mixSiderFixed  -  by @honghuangdc [<samp>(939c5)</samp>](https://github.com/honghuangdc/soybean-admin/commit/939c512)
 | 
			
		||||
 | 
			
		||||
###    🛠 Optimizations
 | 
			
		||||
 | 
			
		||||
- **projects**: optimize code  -  by @honghuangdc [<samp>(cb1d4)</samp>](https://github.com/honghuangdc/soybean-admin/commit/cb1d445)
 | 
			
		||||
 | 
			
		||||
###    🏡 Chore
 | 
			
		||||
 | 
			
		||||
- **projects**: update deps & fix vue-router type  -  by @honghuangdc [<samp>(96837)</samp>](https://github.com/honghuangdc/soybean-admin/commit/968370b)
 | 
			
		||||
 | 
			
		||||
###    ❤️ Contributors
 | 
			
		||||
 | 
			
		||||
[](https://github.com/honghuangdc)  
 | 
			
		||||
 | 
			
		||||
## [v1.3.3](https://github.com/honghuangdc/soybean-admin/compare/v1.3.2...v1.3.3) (2024-07-30)
 | 
			
		||||
 | 
			
		||||
###    🐞 Bug Fixes
 | 
			
		||||
 | 
			
		||||
- **projects**: fix watermark settings  -  by @honghuangdc [<samp>(5646a)</samp>](https://github.com/honghuangdc/soybean-admin/commit/5646a50)
 | 
			
		||||
 | 
			
		||||
###    📖 Documentation
 | 
			
		||||
 | 
			
		||||
- **projects**: update CHANGELOG  -  by @honghuangdc [<samp>(ebc83)</samp>](https://github.com/honghuangdc/soybean-admin/commit/ebc838c)
 | 
			
		||||
 | 
			
		||||
###    ❤️ Contributors
 | 
			
		||||
 | 
			
		||||
[](https://github.com/honghuangdc)  
 | 
			
		||||
 | 
			
		||||
## [v1.3.2](https://github.com/honghuangdc/soybean-admin/compare/v1.3.1...v1.3.2) (2024-07-30)
 | 
			
		||||
 | 
			
		||||
###    🚀 Features
 | 
			
		||||
 
 | 
			
		||||
@@ -1,5 +1,39 @@
 | 
			
		||||
# 更新日志
 | 
			
		||||
 | 
			
		||||
## [v1.3.2](https://github.com/honghuangdc/soybean-admin/compare/v1.3.1...v1.3.2) (2024-07-30)
 | 
			
		||||
 | 
			
		||||
###    🚀 新功能
 | 
			
		||||
 | 
			
		||||
- **项目**:
 | 
			
		||||
  - 添加颜色渐变模式。关闭 #567  -  由 @Azir-11 在 https://github.com/honghuangdc/soybean-admin/issues/569 和 https://github.com/honghuangdc/soybean-admin/issues/567 [<samp>(4dde4)</samp>](https://github.com/honghuangdc/soybean-admin/commit/4dde4c2)
 | 
			
		||||
  - 添加全屏水印。关闭 #571  -  由 @paynezhuang 在 https://github.com/honghuangdc/soybean-admin/issues/573 和 https://github.com/honghuangdc/soybean-admin/issues/571 [<samp>(ea8aa)</samp>](https://github.com/honghuangdc/soybean-admin/commit/ea8aa6c)
 | 
			
		||||
 | 
			
		||||
###    🐞 Bug 修复
 | 
			
		||||
 | 
			
		||||
- **项目**: 修复 vertical-mix 菜单选中问题  -  由 @honghuangdc [<samp>(59f07)</samp>](https://github.com/honghuangdc/soybean-admin/commit/59f07d8)
 | 
			
		||||
 | 
			
		||||
###    🛠 优化
 | 
			
		||||
 | 
			
		||||
- **项目**:
 | 
			
		||||
  - 添加类型 WatermarkProps  -  由 @honghuangdc [<samp>(f26d0)</samp>](https://github.com/honghuangdc/soybean-admin/commit/f26d0a6)
 | 
			
		||||
  - 移除 home NAlert 可关闭  -  由 @honghuangdc [<samp>(98b75)</samp>](https://github.com/honghuangdc/soybean-admin/commit/98b75c2)
 | 
			
		||||
  - 优化响应代码比较  -  由 @honghuangdc [<samp>(cf67d)</samp>](https://github.com/honghuangdc/soybean-admin/commit/cf67d55)
 | 
			
		||||
- **类型**:
 | 
			
		||||
  - 移除无用类型。  -  由 **Azir** [<samp>(eed61)</samp>](https://github.com/honghuangdc/soybean-admin/commit/eed617f)
 | 
			
		||||
 | 
			
		||||
###    📖 文档
 | 
			
		||||
 | 
			
		||||
- **项目**: 更新更新日志  -  由 @honghuangdc [<samp>(d3759)</samp>](https://github.com/honghuangdc/soybean-admin/commit/d37591d)
 | 
			
		||||
 | 
			
		||||
###    🏡 杂项
 | 
			
		||||
 | 
			
		||||
- **依赖**: 更新依赖  -  由 @honghuangdc [<samp>(993e9)</samp>](https://github.com/honghuangdc/soybean-admin/commit/993e9ca)
 | 
			
		||||
 | 
			
		||||
###    ❤️ 贡献者
 | 
			
		||||
 | 
			
		||||
[](https://github.com/honghuangdc)  [](https://github.com/paynezhuang)  [](https://github.com/Azir-11)  
 | 
			
		||||
[Azir](mailto:2075125282@qq.com), 
 | 
			
		||||
 | 
			
		||||
## [v1.3.1](https://github.com/honghuangdc/soybean-admin/compare/v1.3.0...v1.3.1) (2024-07-22)
 | 
			
		||||
 | 
			
		||||
###    🐞 Bug 修复
 | 
			
		||||
 
 | 
			
		||||
@@ -111,6 +111,7 @@ pnpm build
 | 
			
		||||
 | 
			
		||||
## Ecosystem
 | 
			
		||||
 | 
			
		||||
- [react-soybean-admin](https://github.com/mufeng889/react-soybean-admin): SoybeanAdmin based version of React.
 | 
			
		||||
- [electron-mock-admin](https://github.com/lixin59/electron-mock-api): A Mock Api management system that helps front-end developers quickly implement interface mocks.
 | 
			
		||||
- [T-Shell](https://github.com/TheBlindM/T-Shell): A terminal emulator and SSH client with configurable command prompts.
 | 
			
		||||
- [pea](https://github.com/haitang1894/pea) : Adopting SpringBoot3.2 + JDK21, MyBatis-Plus, SpringSecurity security framework, etc., suitable for the simple permission system developed by [soybean-admin](https://gitee.com/honghuangdc/soybean-admin).
 | 
			
		||||
@@ -155,7 +156,7 @@ Thanks the following people for their contributions. If you want to contribute t
 | 
			
		||||
 | 
			
		||||
  <div>
 | 
			
		||||
  	<p>QQ Group</p>
 | 
			
		||||
    <img src="https://soybeanjs-1300612522.cos.ap-guangzhou.myqcloud.com/uPic/qq-soybean-admin-2.jpg" style="width:200px" />
 | 
			
		||||
    <img src="https://soybeanjs-1300612522.cos.ap-guangzhou.myqcloud.com/uPic/qq-soybean-admin-3.jpg" style="width:200px" />
 | 
			
		||||
  </div>
 | 
			
		||||
	<!-- <div>
 | 
			
		||||
		<p>WeChat Group</p>
 | 
			
		||||
@@ -165,10 +166,6 @@ Thanks the following people for their contributions. If you want to contribute t
 | 
			
		||||
		<p>Add the following WeChat to invite to the WeChat group</p>
 | 
			
		||||
		<img src="https://soybeanjs-1300612522.cos.ap-guangzhou.myqcloud.com/uPic/wechat-soybeanjs.jpg" style="width:200px" />
 | 
			
		||||
	</div>
 | 
			
		||||
  <div>
 | 
			
		||||
    <p>Add Soybean's WeChat for business consultation, cooperation, project architecture, one-on-one guidance, etc.</p>
 | 
			
		||||
    <img src="https://soybeanjs-1300612522.cos.ap-guangzhou.myqcloud.com/uPic/wechat-soybean.jpg" style="width:200px" />
 | 
			
		||||
  </div>
 | 
			
		||||
 | 
			
		||||
## Star Trend
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -57,6 +57,36 @@
 | 
			
		||||
- [地址](https://docs.soybeanjs.cn)
 | 
			
		||||
- [旧版文档](https://legacy-docs.soybeanjs.cn)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
## 合作事项
 | 
			
		||||
 | 
			
		||||
我们非常感谢大家对 [`SoybeanAdmin`](https://github.com/soybeanjs/soybean-admin) 的支持!为了进一步回馈社区,并助力企业和开发者实现个性化需求,我们现提供多种合作服务,期待与您携手共赢。
 | 
			
		||||
 | 
			
		||||
##### 1、定制化管理后台开发
 | 
			
		||||
 | 
			
		||||
针对企业和开发者的特定业务需求,我们提供基于 [`SoybeanAdmin`](https://github.com/soybeanjs/soybean-admin) 的定制化管理后台开发服务。我们的团队具备丰富的行业经验,能够迅速理解并实现您的需求,打造高效、灵活且安全的定制化解决方案。
 | 
			
		||||
 | 
			
		||||
- **定制开发**:我们将根据您的具体需求,提供从需求分析、UI设计到功能实现的全方位服务,确保项目高效交付。
 | 
			
		||||
- **功能扩展**:在 [`SoybeanAdmin`](https://github.com/soybeanjs/soybean-admin) 基础上,扩展您所需的特定功能模块,提升管理后台的功能和用户体验。
 | 
			
		||||
 | 
			
		||||
##### 2、企业外包服务
 | 
			
		||||
 | 
			
		||||
我们承接各类企业级外包项目,特别是在管理后台系统的开发、集成与运维方面。我们以精益求精的态度,确保项目的质量和进度,为您的业务提供强有力的技术支持。
 | 
			
		||||
 | 
			
		||||
- **项目开发**:无论是全新的项目,还是现有系统的优化与集成,我们都将为您量身打造高效可靠的解决方案。
 | 
			
		||||
- **系统集成与维护**:我们也提供基于 [`SoybeanAdmin`](https://github.com/soybeanjs/soybean-admin) 的系统集成与长期维护服务,确保您的系统稳定、安全地运行。
 | 
			
		||||
 | 
			
		||||
##### 3、联系方式
 | 
			
		||||
 | 
			
		||||
如有合作意向或项目咨询,请通过以下方式与我们联系:
 | 
			
		||||
 | 
			
		||||
- **Email**: [soybeanjs@outlook.com](mailto:soybeanjs@outlook.com)
 | 
			
		||||
- **GitHub Issues**: 欢迎通过 [GitHub Issues](https://github.com/soybeanjs/soybean-admin/issues/new) 联系我们,进行初步的合作洽谈。
 | 
			
		||||
- **商务合作微信**:  honghuangdc
 | 
			
		||||
 | 
			
		||||
期待与您开展深入合作,共同推动 SoybeanAdmin 项目及其在更多领域的成功应用!
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
## 示例图片
 | 
			
		||||
 | 
			
		||||

 | 
			
		||||
@@ -109,6 +139,7 @@ pnpm build
 | 
			
		||||
 | 
			
		||||
## 周边生态
 | 
			
		||||
 | 
			
		||||
- [react-soybean-admin](https://github.com/mufeng889/react-soybean-admin): 基于SoybeanAdmin的React版本.
 | 
			
		||||
- [electron-mock-admin](https://github.com/lixin59/electron-mock-api): 一个 Mock Api 管理系统,帮助前端开发伙伴快速实现接口的 mock。
 | 
			
		||||
- [T-Shell](https://github.com/TheBlindM/T-Shell): 是一个可配置命令提示的终端模拟器和 SSH 客户端。
 | 
			
		||||
- [pea](https://github.com/haitang1894/pea) : 采用SpringBoot3.2 + JDK21、MyBatis-Plus、SpringSecurity安全框架等,适配 [soybean-admin](https://gitee.com/honghuangdc/soybean-admin) 开发的简单权限系统。
 | 
			
		||||
@@ -155,7 +186,7 @@ pnpm build
 | 
			
		||||
 | 
			
		||||
  <div>
 | 
			
		||||
  	<p>QQ交流群</p>
 | 
			
		||||
    <img src="https://soybeanjs-1300612522.cos.ap-guangzhou.myqcloud.com/uPic/qq-soybean-admin-2.jpg" style="width:200px" />
 | 
			
		||||
    <img src="https://soybeanjs-1300612522.cos.ap-guangzhou.myqcloud.com/uPic/qq-soybean-admin-3.jpg" style="width:200px" />
 | 
			
		||||
  </div>
 | 
			
		||||
	<!-- <div>
 | 
			
		||||
		<p>微信群</p>
 | 
			
		||||
@@ -165,10 +196,6 @@ pnpm build
 | 
			
		||||
		<p>添加下面微信邀请进微信群</p>
 | 
			
		||||
		<img src="https://soybeanjs-1300612522.cos.ap-guangzhou.myqcloud.com/uPic/wechat-soybeanjs.jpg" style="width:200px" />
 | 
			
		||||
	</div>
 | 
			
		||||
  <div>
 | 
			
		||||
    <p>添加 Soybean 的微信,业务咨询、合作、项目架构、一对一指导等</p>
 | 
			
		||||
    <img src="https://soybeanjs-1300612522.cos.ap-guangzhou.myqcloud.com/uPic/wechat-soybean.jpg" style="width:200px" />
 | 
			
		||||
  </div>
 | 
			
		||||
 | 
			
		||||
## Star 趋势
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -5,10 +5,10 @@ import { createServiceConfig } from '../../src/utils/service';
 | 
			
		||||
 * Set http proxy
 | 
			
		||||
 *
 | 
			
		||||
 * @param env - The current env
 | 
			
		||||
 * @param isDev - Is development environment
 | 
			
		||||
 * @param enable - If enable http proxy
 | 
			
		||||
 */
 | 
			
		||||
export function createViteProxy(env: Env.ImportMeta, isDev: boolean) {
 | 
			
		||||
  const isEnableHttpProxy = isDev && env.VITE_HTTP_PROXY === 'Y';
 | 
			
		||||
export function createViteProxy(env: Env.ImportMeta, enable: boolean) {
 | 
			
		||||
  const isEnableHttpProxy = enable && env.VITE_HTTP_PROXY === 'Y';
 | 
			
		||||
 | 
			
		||||
  if (!isEnableHttpProxy) return undefined;
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -1,7 +1,7 @@
 | 
			
		||||
import { defineConfig } from '@soybeanjs/eslint-config';
 | 
			
		||||
 | 
			
		||||
export default defineConfig(
 | 
			
		||||
  { vue: true, unocss: true },
 | 
			
		||||
  { vue: true, unocss: true, ignores: ['src-tauri/target'] },
 | 
			
		||||
  {
 | 
			
		||||
    rules: {
 | 
			
		||||
      'vue/multi-word-component-names': [
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										69
									
								
								package.json
									
									
									
									
									
								
							
							
						
						@@ -1,7 +1,7 @@
 | 
			
		||||
{
 | 
			
		||||
  "name": "soybean-admin",
 | 
			
		||||
  "type": "module",
 | 
			
		||||
  "version": "1.3.2",
 | 
			
		||||
  "version": "1.3.5",
 | 
			
		||||
  "description": "A fresh and elegant admin template, based on Vue3、Vite3、TypeScript、NaiveUI and UnoCSS. 一个基于Vue3、Vite3、TypeScript、NaiveUI and UnoCSS的清新优雅的中后台模版。",
 | 
			
		||||
  "author": {
 | 
			
		||||
    "name": "Soybean",
 | 
			
		||||
@@ -32,17 +32,20 @@
 | 
			
		||||
  },
 | 
			
		||||
  "scripts": {
 | 
			
		||||
    "build": "vite build --mode prod",
 | 
			
		||||
    "build:tauri": "pnpm tauri build",
 | 
			
		||||
    "build:test": "vite build --mode test",
 | 
			
		||||
    "cleanup": "sa cleanup",
 | 
			
		||||
    "commit": "sa git-commit",
 | 
			
		||||
    "czh": "sa git-commit -l=zh-cn",
 | 
			
		||||
    "commit:zh": "sa git-commit -l=zh-cn",
 | 
			
		||||
    "dev": "vite --mode test",
 | 
			
		||||
    "dev:prod": "vite --mode prod",
 | 
			
		||||
    "dev:tauri": "pnpm tauri dev",
 | 
			
		||||
    "gen-route": "sa gen-route",
 | 
			
		||||
    "lint": "eslint . --fix",
 | 
			
		||||
    "prepare": "simple-git-hooks",
 | 
			
		||||
    "preview": "vite preview",
 | 
			
		||||
    "release": "sa release",
 | 
			
		||||
    "tauri-icon": "pnpm tauri icon ./public/logo.png",
 | 
			
		||||
    "typecheck": "vue-tsc --noEmit --skipLibCheck",
 | 
			
		||||
    "update-pkg": "sa update-pkg"
 | 
			
		||||
  },
 | 
			
		||||
@@ -54,50 +57,52 @@
 | 
			
		||||
    "@sa/hooks": "workspace:*",
 | 
			
		||||
    "@sa/materials": "workspace:*",
 | 
			
		||||
    "@sa/utils": "workspace:*",
 | 
			
		||||
    "@vueuse/core": "10.11.0",
 | 
			
		||||
    "@vueuse/core": "11.0.3",
 | 
			
		||||
    "clipboard": "2.0.11",
 | 
			
		||||
    "dayjs": "1.11.12",
 | 
			
		||||
    "dayjs": "1.11.13",
 | 
			
		||||
    "echarts": "5.5.1",
 | 
			
		||||
    "json5": "2.2.3",
 | 
			
		||||
    "naive-ui": "2.39.0",
 | 
			
		||||
    "nprogress": "0.2.0",
 | 
			
		||||
    "pinia": "2.2.0",
 | 
			
		||||
    "tailwind-merge": "2.4.0",
 | 
			
		||||
    "vue": "3.4.34",
 | 
			
		||||
    "vue-draggable-plus": "0.5.2",
 | 
			
		||||
    "vue-i18n": "9.13.1",
 | 
			
		||||
    "vue-router": "4.4.0"
 | 
			
		||||
    "pinia": "2.2.2",
 | 
			
		||||
    "tailwind-merge": "2.5.2",
 | 
			
		||||
    "vue": "3.5.3",
 | 
			
		||||
    "vue-draggable-plus": "0.5.3",
 | 
			
		||||
    "vue-i18n": "9.14.0",
 | 
			
		||||
    "vue-router": "4.4.3"
 | 
			
		||||
  },
 | 
			
		||||
  "devDependencies": {
 | 
			
		||||
    "@elegant-router/vue": "0.3.7",
 | 
			
		||||
    "@iconify/json": "2.2.232",
 | 
			
		||||
    "@elegant-router/vue": "0.3.8",
 | 
			
		||||
    "@iconify/json": "2.2.245",
 | 
			
		||||
    "@sa/scripts": "workspace:*",
 | 
			
		||||
    "@sa/uno-preset": "workspace:*",
 | 
			
		||||
    "@soybeanjs/eslint-config": "1.3.7",
 | 
			
		||||
    "@types/node": "22.0.0",
 | 
			
		||||
    "@soybeanjs/eslint-config": "1.4.0",
 | 
			
		||||
    "@tauri-apps/cli": "1.6.1",
 | 
			
		||||
    "@types/node": "22.5.4",
 | 
			
		||||
    "@types/nprogress": "0.2.3",
 | 
			
		||||
    "@unocss/eslint-config": "0.61.7",
 | 
			
		||||
    "@unocss/preset-icons": "0.61.7",
 | 
			
		||||
    "@unocss/preset-uno": "0.61.7",
 | 
			
		||||
    "@unocss/transformer-directives": "0.61.7",
 | 
			
		||||
    "@unocss/transformer-variant-group": "0.61.7",
 | 
			
		||||
    "@unocss/vite": "0.61.7",
 | 
			
		||||
    "@vitejs/plugin-vue": "5.1.1",
 | 
			
		||||
    "@vitejs/plugin-vue-jsx": "4.0.0",
 | 
			
		||||
    "eslint": "9.8.0",
 | 
			
		||||
    "eslint-plugin-vue": "9.27.0",
 | 
			
		||||
    "lint-staged": "15.2.7",
 | 
			
		||||
    "sass": "1.77.8",
 | 
			
		||||
    "@unocss/eslint-config": "0.62.3",
 | 
			
		||||
    "@unocss/preset-icons": "0.62.3",
 | 
			
		||||
    "@unocss/preset-uno": "0.62.3",
 | 
			
		||||
    "@unocss/transformer-directives": "0.62.3",
 | 
			
		||||
    "@unocss/transformer-variant-group": "0.62.3",
 | 
			
		||||
    "@unocss/vite": "0.62.3",
 | 
			
		||||
    "@vitejs/plugin-vue": "5.1.3",
 | 
			
		||||
    "@vitejs/plugin-vue-jsx": "4.0.1",
 | 
			
		||||
    "eslint": "9.10.0",
 | 
			
		||||
    "eslint-plugin-vue": "9.28.0",
 | 
			
		||||
    "lint-staged": "15.2.10",
 | 
			
		||||
    "sass": "1.78.0",
 | 
			
		||||
    "simple-git-hooks": "2.11.1",
 | 
			
		||||
    "tsx": "4.16.2",
 | 
			
		||||
    "tsx": "4.19.0",
 | 
			
		||||
    "typescript": "5.5.4",
 | 
			
		||||
    "unplugin-icons": "0.19.1",
 | 
			
		||||
    "unplugin-vue-components": "0.27.3",
 | 
			
		||||
    "vite": "5.3.5",
 | 
			
		||||
    "unplugin-icons": "0.19.3",
 | 
			
		||||
    "unplugin-vue-components": "0.27.4",
 | 
			
		||||
    "vite": "5.4.3",
 | 
			
		||||
    "vite-plugin-progress": "0.0.7",
 | 
			
		||||
    "vite-plugin-svg-icons": "2.0.1",
 | 
			
		||||
    "vite-plugin-vue-devtools": "7.3.7",
 | 
			
		||||
    "vite-plugin-vue-devtools": "7.4.4",
 | 
			
		||||
    "vue-eslint-parser": "9.4.3",
 | 
			
		||||
    "vue-tsc": "2.0.29"
 | 
			
		||||
    "vue-tsc": "2.1.6"
 | 
			
		||||
  },
 | 
			
		||||
  "simple-git-hooks": {
 | 
			
		||||
    "commit-msg": "pnpm sa git-commit-verify",
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,6 @@
 | 
			
		||||
{
 | 
			
		||||
  "name": "@sa/axios",
 | 
			
		||||
  "version": "1.3.2",
 | 
			
		||||
  "version": "1.3.5",
 | 
			
		||||
  "exports": {
 | 
			
		||||
    ".": "./src/index.ts"
 | 
			
		||||
  },
 | 
			
		||||
@@ -11,9 +11,9 @@
 | 
			
		||||
  },
 | 
			
		||||
  "dependencies": {
 | 
			
		||||
    "@sa/utils": "workspace:*",
 | 
			
		||||
    "axios": "1.7.2",
 | 
			
		||||
    "axios-retry": "4.4.2",
 | 
			
		||||
    "qs": "6.12.3"
 | 
			
		||||
    "axios": "1.7.7",
 | 
			
		||||
    "axios-retry": "4.5.0",
 | 
			
		||||
    "qs": "6.13.0"
 | 
			
		||||
  },
 | 
			
		||||
  "devDependencies": {
 | 
			
		||||
    "@types/qs": "6.9.15"
 | 
			
		||||
 
 | 
			
		||||
@@ -162,12 +162,12 @@ export function createFlatRequest<ResponseData = any, State = Record<string, unk
 | 
			
		||||
      if (responseType === 'json') {
 | 
			
		||||
        const data = opts.transformBackendResponse(response);
 | 
			
		||||
 | 
			
		||||
        return { data, error: null };
 | 
			
		||||
        return { data, error: null, response };
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      return { data: response.data as MappedType<R, T>, error: null };
 | 
			
		||||
    } catch (error) {
 | 
			
		||||
      return { data: null, error };
 | 
			
		||||
      return { data: null, error, response: (error as AxiosError<ResponseData>).response };
 | 
			
		||||
    }
 | 
			
		||||
  } as FlatRequestInstance<State, ResponseData>;
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -92,18 +92,20 @@ export interface RequestInstance<S = Record<string, unknown>> extends RequestIns
 | 
			
		||||
  <T = any, R extends ResponseType = 'json'>(config: CustomAxiosRequestConfig<R>): Promise<MappedType<R, T>>;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export type FlatResponseSuccessData<T = any> = {
 | 
			
		||||
export type FlatResponseSuccessData<T = any, ResponseData = any> = {
 | 
			
		||||
  data: T;
 | 
			
		||||
  error: null;
 | 
			
		||||
  response: AxiosResponse<ResponseData>;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
export type FlatResponseFailData<ResponseData = any> = {
 | 
			
		||||
  data: null;
 | 
			
		||||
  error: AxiosError<ResponseData>;
 | 
			
		||||
  response: AxiosResponse<ResponseData>;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
export type FlatResponseData<T = any, ResponseData = any> =
 | 
			
		||||
  | FlatResponseSuccessData<T>
 | 
			
		||||
  | FlatResponseSuccessData<T, ResponseData>
 | 
			
		||||
  | FlatResponseFailData<ResponseData>;
 | 
			
		||||
 | 
			
		||||
export interface FlatRequestInstance<S = Record<string, unknown>, ResponseData = any> extends RequestInstanceCommon<S> {
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,6 @@
 | 
			
		||||
{
 | 
			
		||||
  "name": "@sa/color",
 | 
			
		||||
  "version": "1.3.2",
 | 
			
		||||
  "version": "1.3.5",
 | 
			
		||||
  "exports": {
 | 
			
		||||
    ".": "./src/index.ts"
 | 
			
		||||
  },
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,6 @@
 | 
			
		||||
{
 | 
			
		||||
  "name": "@sa/hooks",
 | 
			
		||||
  "version": "1.3.2",
 | 
			
		||||
  "version": "1.3.5",
 | 
			
		||||
  "exports": {
 | 
			
		||||
    ".": "./src/index.ts"
 | 
			
		||||
  },
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,6 @@
 | 
			
		||||
{
 | 
			
		||||
  "name": "@sa/materials",
 | 
			
		||||
  "version": "1.3.2",
 | 
			
		||||
  "version": "1.3.5",
 | 
			
		||||
  "exports": {
 | 
			
		||||
    ".": "./src/index.ts"
 | 
			
		||||
  },
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,6 @@
 | 
			
		||||
{
 | 
			
		||||
  "name": "@sa/fetch",
 | 
			
		||||
  "version": "1.3.2",
 | 
			
		||||
  "version": "1.3.5",
 | 
			
		||||
  "exports": {
 | 
			
		||||
    ".": "./src/index.ts"
 | 
			
		||||
  },
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,6 @@
 | 
			
		||||
{
 | 
			
		||||
  "name": "@sa/scripts",
 | 
			
		||||
  "version": "1.3.2",
 | 
			
		||||
  "version": "1.3.5",
 | 
			
		||||
  "bin": {
 | 
			
		||||
    "sa": "./bin.ts"
 | 
			
		||||
  },
 | 
			
		||||
@@ -14,14 +14,14 @@
 | 
			
		||||
  },
 | 
			
		||||
  "devDependencies": {
 | 
			
		||||
    "@soybeanjs/changelog": "0.3.24",
 | 
			
		||||
    "bumpp": "9.4.1",
 | 
			
		||||
    "c12": "1.11.1",
 | 
			
		||||
    "bumpp": "9.5.2",
 | 
			
		||||
    "c12": "1.11.2",
 | 
			
		||||
    "cac": "6.7.14",
 | 
			
		||||
    "consola": "3.2.3",
 | 
			
		||||
    "enquirer": "2.4.1",
 | 
			
		||||
    "execa": "9.3.0",
 | 
			
		||||
    "execa": "9.3.1",
 | 
			
		||||
    "kolorist": "1.8.0",
 | 
			
		||||
    "npm-check-updates": "16.14.20",
 | 
			
		||||
    "npm-check-updates": "17.1.1",
 | 
			
		||||
    "rimraf": "6.0.1"
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -11,13 +11,15 @@ export const locales = {
 | 
			
		||||
    },
 | 
			
		||||
    gitCommitTypes: [
 | 
			
		||||
      ['feat', '新功能'],
 | 
			
		||||
      ['feat-wip', '开发中的功能,比如某功能的部分代码'],
 | 
			
		||||
      ['fix', '修复Bug'],
 | 
			
		||||
      ['docs', '只更新文档'],
 | 
			
		||||
      ['docs', '只涉及文档更新'],
 | 
			
		||||
      ['typo', '代码或文档勘误,比如错误拼写'],
 | 
			
		||||
      ['style', '修改代码风格,不影响代码含义的变更'],
 | 
			
		||||
      ['refactor', '代码重构,既不修复 bug 也不添加功能的代码变更'],
 | 
			
		||||
      ['perf', '可提高性能的代码更改'],
 | 
			
		||||
      ['optimize', '优化代码质量的代码更改'],
 | 
			
		||||
      ['test', '添加缺失的测试或更正现有测'],
 | 
			
		||||
      ['test', '添加缺失的测试或更正现有测试'],
 | 
			
		||||
      ['build', '影响构建系统或外部依赖项的更改'],
 | 
			
		||||
      ['ci', '对 CI 配置文件和脚本的更改'],
 | 
			
		||||
      ['chore', '没有修改src或测试文件的其他变更'],
 | 
			
		||||
@@ -47,8 +49,10 @@ export const locales = {
 | 
			
		||||
    },
 | 
			
		||||
    gitCommitTypes: [
 | 
			
		||||
      ['feat', 'A new feature'],
 | 
			
		||||
      ['feat-wip', 'Features in development, such as partial code for a certain feature'],
 | 
			
		||||
      ['fix', 'A bug fix'],
 | 
			
		||||
      ['docs', 'Documentation only changes'],
 | 
			
		||||
      ['typo', 'Code or document corrections, such as spelling errors'],
 | 
			
		||||
      ['style', 'Changes that do not affect the meaning of the code'],
 | 
			
		||||
      ['refactor', 'A code change that neither fixes a bug nor adds a feature'],
 | 
			
		||||
      ['perf', 'A code change that improves performance'],
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,6 @@
 | 
			
		||||
{
 | 
			
		||||
  "name": "@sa/uno-preset",
 | 
			
		||||
  "version": "1.3.2",
 | 
			
		||||
  "version": "1.3.5",
 | 
			
		||||
  "exports": {
 | 
			
		||||
    ".": "./src/index.ts"
 | 
			
		||||
  },
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,6 @@
 | 
			
		||||
{
 | 
			
		||||
  "name": "@sa/utils",
 | 
			
		||||
  "version": "1.3.2",
 | 
			
		||||
  "version": "1.3.5",
 | 
			
		||||
  "exports": {
 | 
			
		||||
    ".": "./src/index.ts"
 | 
			
		||||
  },
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										4061
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							
							
						
						
							
								
								
									
										
											BIN
										
									
								
								public/logo.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 20 KiB  | 
							
								
								
									
										3
									
								
								src-tauri/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,3 @@
 | 
			
		||||
# Generated by Cargo
 | 
			
		||||
# will have compiled files and executables
 | 
			
		||||
/target/
 | 
			
		||||
							
								
								
									
										3664
									
								
								src-tauri/Cargo.lock
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										26
									
								
								src-tauri/Cargo.toml
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,26 @@
 | 
			
		||||
[package]
 | 
			
		||||
name = "app"
 | 
			
		||||
version = "0.1.0"
 | 
			
		||||
description = "A Tauri App"
 | 
			
		||||
authors = ["you"]
 | 
			
		||||
license = ""
 | 
			
		||||
repository = ""
 | 
			
		||||
default-run = "app"
 | 
			
		||||
edition = "2021"
 | 
			
		||||
rust-version = "1.60"
 | 
			
		||||
 | 
			
		||||
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
 | 
			
		||||
 | 
			
		||||
[build-dependencies]
 | 
			
		||||
tauri-build = { version = "1.5.1", features = [] }
 | 
			
		||||
 | 
			
		||||
[dependencies]
 | 
			
		||||
serde_json = "1.0"
 | 
			
		||||
serde = { version = "1.0", features = ["derive"] }
 | 
			
		||||
tauri = { version = "1.6.1", features = [] }
 | 
			
		||||
 | 
			
		||||
[features]
 | 
			
		||||
# this feature is used for production builds or when `devPath` points to the filesystem and the built-in dev server is disabled.
 | 
			
		||||
# If you use cargo directly instead of tauri's cli you can use this feature flag to switch between tauri's `dev` and `build` modes.
 | 
			
		||||
# DO NOT REMOVE!!
 | 
			
		||||
custom-protocol = [ "tauri/custom-protocol" ]
 | 
			
		||||
							
								
								
									
										3
									
								
								src-tauri/build.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,3 @@
 | 
			
		||||
fn main() {
 | 
			
		||||
  tauri_build::build()
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										
											BIN
										
									
								
								src-tauri/icons/128x128.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 8.8 KiB  | 
							
								
								
									
										
											BIN
										
									
								
								src-tauri/icons/128x128@2x.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 19 KiB  | 
							
								
								
									
										
											BIN
										
									
								
								src-tauri/icons/32x32.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 1.7 KiB  | 
							
								
								
									
										
											BIN
										
									
								
								src-tauri/icons/Square107x107Logo.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 7.4 KiB  | 
							
								
								
									
										
											BIN
										
									
								
								src-tauri/icons/Square142x142Logo.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 9.9 KiB  | 
							
								
								
									
										
											BIN
										
									
								
								src-tauri/icons/Square150x150Logo.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 10 KiB  | 
							
								
								
									
										
											BIN
										
									
								
								src-tauri/icons/Square284x284Logo.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 21 KiB  | 
							
								
								
									
										
											BIN
										
									
								
								src-tauri/icons/Square30x30Logo.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 1.6 KiB  | 
							
								
								
									
										
											BIN
										
									
								
								src-tauri/icons/Square310x310Logo.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 23 KiB  | 
							
								
								
									
										
											BIN
										
									
								
								src-tauri/icons/Square44x44Logo.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 2.6 KiB  | 
							
								
								
									
										
											BIN
										
									
								
								src-tauri/icons/Square71x71Logo.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 4.7 KiB  | 
							
								
								
									
										
											BIN
										
									
								
								src-tauri/icons/Square89x89Logo.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 6.1 KiB  | 
							
								
								
									
										
											BIN
										
									
								
								src-tauri/icons/StoreLogo.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 3.0 KiB  | 
							
								
								
									
										
											BIN
										
									
								
								src-tauri/icons/icon.icns
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										
											BIN
										
									
								
								src-tauri/icons/icon.ico
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 31 KiB  | 
							
								
								
									
										
											BIN
										
									
								
								src-tauri/icons/icon.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| 
		 After Width: | Height: | Size: 39 KiB  | 
							
								
								
									
										8
									
								
								src-tauri/src/main.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,8 @@
 | 
			
		||||
// Prevents additional console window on Windows in release, DO NOT REMOVE!!
 | 
			
		||||
#![cfg_attr(not(debug_assertions), windows_subsystem = "windows")]
 | 
			
		||||
 | 
			
		||||
fn main() {
 | 
			
		||||
  tauri::Builder::default()
 | 
			
		||||
    .run(tauri::generate_context!())
 | 
			
		||||
    .expect("error while running tauri application");
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										60
									
								
								src-tauri/tauri.conf.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,60 @@
 | 
			
		||||
{
 | 
			
		||||
  "$schema": "../node_modules/@tauri-apps/cli/schema.json",
 | 
			
		||||
  "build": {
 | 
			
		||||
    "beforeBuildCommand": "npm run build",
 | 
			
		||||
    "beforeDevCommand": "npm run dev",
 | 
			
		||||
    "devPath": "http://localhost:9527",
 | 
			
		||||
    "distDir": "../dist"
 | 
			
		||||
  },
 | 
			
		||||
  "package": {
 | 
			
		||||
    "productName": "soybean-admin",
 | 
			
		||||
    "version": "1.0.0"
 | 
			
		||||
  },
 | 
			
		||||
  "tauri": {
 | 
			
		||||
    "allowlist": {
 | 
			
		||||
      "all": false
 | 
			
		||||
    },
 | 
			
		||||
    "bundle": {
 | 
			
		||||
      "active": true,
 | 
			
		||||
      "category": "DeveloperTool",
 | 
			
		||||
      "copyright": "",
 | 
			
		||||
      "deb": {
 | 
			
		||||
        "depends": []
 | 
			
		||||
      },
 | 
			
		||||
      "externalBin": [],
 | 
			
		||||
      "icon": ["icons/32x32.png", "icons/128x128.png", "icons/128x128@2x.png", "icons/icon.icns", "icons/icon.ico"],
 | 
			
		||||
      "identifier": "cn.soybeanjs.admin",
 | 
			
		||||
      "longDescription": "",
 | 
			
		||||
      "macOS": {
 | 
			
		||||
        "entitlements": null,
 | 
			
		||||
        "exceptionDomain": "",
 | 
			
		||||
        "frameworks": [],
 | 
			
		||||
        "providerShortName": null,
 | 
			
		||||
        "signingIdentity": null
 | 
			
		||||
      },
 | 
			
		||||
      "resources": [],
 | 
			
		||||
      "shortDescription": "",
 | 
			
		||||
      "targets": "all",
 | 
			
		||||
      "windows": {
 | 
			
		||||
        "certificateThumbprint": null,
 | 
			
		||||
        "digestAlgorithm": "sha256",
 | 
			
		||||
        "timestampUrl": ""
 | 
			
		||||
      }
 | 
			
		||||
    },
 | 
			
		||||
    "security": {
 | 
			
		||||
      "csp": null
 | 
			
		||||
    },
 | 
			
		||||
    "updater": {
 | 
			
		||||
      "active": false
 | 
			
		||||
    },
 | 
			
		||||
    "windows": [
 | 
			
		||||
      {
 | 
			
		||||
        "fullscreen": false,
 | 
			
		||||
        "height": 768,
 | 
			
		||||
        "resizable": true,
 | 
			
		||||
        "title": "SoybeanAdmin",
 | 
			
		||||
        "width": 1366
 | 
			
		||||
      }
 | 
			
		||||
    ]
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
@@ -25,7 +25,7 @@ const naiveDateLocale = computed(() => {
 | 
			
		||||
 | 
			
		||||
const watermarkProps = computed<WatermarkProps>(() => {
 | 
			
		||||
  return {
 | 
			
		||||
    content: themeStore.watermark.text,
 | 
			
		||||
    content: themeStore.watermark?.text || 'SoybeanAdmin',
 | 
			
		||||
    cross: true,
 | 
			
		||||
    fullscreen: true,
 | 
			
		||||
    fontSize: 16,
 | 
			
		||||
@@ -50,7 +50,7 @@ const watermarkProps = computed<WatermarkProps>(() => {
 | 
			
		||||
  >
 | 
			
		||||
    <AppProvider>
 | 
			
		||||
      <RouterView class="bg-layout" />
 | 
			
		||||
      <NWatermark v-if="themeStore.watermark.visible" v-bind="watermarkProps" />
 | 
			
		||||
      <NWatermark v-if="themeStore.watermark?.visible" v-bind="watermarkProps" />
 | 
			
		||||
    </AppProvider>
 | 
			
		||||
  </NConfigProvider>
 | 
			
		||||
</template>
 | 
			
		||||
 
 | 
			
		||||
@@ -93,11 +93,15 @@ export function useRouterPush(inSetup = true) {
 | 
			
		||||
    return routerPushByKey('login', { query, params: { module } });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /** Redirect from login */
 | 
			
		||||
  async function redirectFromLogin() {
 | 
			
		||||
  /**
 | 
			
		||||
   * Redirect from login
 | 
			
		||||
   *
 | 
			
		||||
   * @param [needRedirect=true] Whether to redirect after login. Default is `true`
 | 
			
		||||
   */
 | 
			
		||||
  async function redirectFromLogin(needRedirect = true) {
 | 
			
		||||
    const redirect = route.value.query?.redirect as string;
 | 
			
		||||
 | 
			
		||||
    if (redirect) {
 | 
			
		||||
    if (needRedirect && redirect) {
 | 
			
		||||
      routerPush(redirect);
 | 
			
		||||
    } else {
 | 
			
		||||
      toHome();
 | 
			
		||||
 
 | 
			
		||||
@@ -8,6 +8,7 @@ export const { setupStore: setupMixMenuContext, useStore: useMixMenuContext } =
 | 
			
		||||
function useMixMenu() {
 | 
			
		||||
  const route = useRoute();
 | 
			
		||||
  const routeStore = useRouteStore();
 | 
			
		||||
  const { selectedKey } = useMenu();
 | 
			
		||||
 | 
			
		||||
  const activeFirstLevelMenuKey = ref('');
 | 
			
		||||
 | 
			
		||||
@@ -16,12 +17,7 @@ function useMixMenu() {
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  function getActiveFirstLevelMenuKey() {
 | 
			
		||||
    const { hideInMenu, activeMenu } = route.meta;
 | 
			
		||||
    const name = route.name as string;
 | 
			
		||||
 | 
			
		||||
    const routeName = (hideInMenu ? activeMenu : name) || name;
 | 
			
		||||
 | 
			
		||||
    const [firstLevelRouteName] = routeName.split('_');
 | 
			
		||||
    const [firstLevelRouteName] = selectedKey.value.split('_');
 | 
			
		||||
 | 
			
		||||
    setActiveFirstLevelMenuKey(firstLevelRouteName);
 | 
			
		||||
  }
 | 
			
		||||
@@ -68,3 +64,20 @@ function useMixMenu() {
 | 
			
		||||
    getActiveFirstLevelMenuKey
 | 
			
		||||
  };
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export function useMenu() {
 | 
			
		||||
  const route = useRoute();
 | 
			
		||||
 | 
			
		||||
  const selectedKey = computed(() => {
 | 
			
		||||
    const { hideInMenu, activeMenu } = route.meta;
 | 
			
		||||
    const name = route.name as string;
 | 
			
		||||
 | 
			
		||||
    const routeName = (hideInMenu ? activeMenu : name) || name;
 | 
			
		||||
 | 
			
		||||
    return routeName;
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  return {
 | 
			
		||||
    selectedKey
 | 
			
		||||
  };
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,26 +1,16 @@
 | 
			
		||||
<script setup lang="ts">
 | 
			
		||||
import { computed } from 'vue';
 | 
			
		||||
import { useRoute } from 'vue-router';
 | 
			
		||||
import { GLOBAL_HEADER_MENU_ID } from '@/constants/app';
 | 
			
		||||
import { useRouteStore } from '@/store/modules/route';
 | 
			
		||||
import { useRouterPush } from '@/hooks/common/router';
 | 
			
		||||
import { useMenu } from '../../../context';
 | 
			
		||||
 | 
			
		||||
defineOptions({
 | 
			
		||||
  name: 'HorizontalMenu'
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
const route = useRoute();
 | 
			
		||||
const routeStore = useRouteStore();
 | 
			
		||||
const { routerPushByKeyWithMetaQuery } = useRouterPush();
 | 
			
		||||
 | 
			
		||||
const selectedKey = computed(() => {
 | 
			
		||||
  const { hideInMenu, activeMenu } = route.meta;
 | 
			
		||||
  const name = route.name as string;
 | 
			
		||||
 | 
			
		||||
  const routeName = (hideInMenu ? activeMenu : name) || name;
 | 
			
		||||
 | 
			
		||||
  return routeName;
 | 
			
		||||
});
 | 
			
		||||
const { selectedKey } = useMenu();
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
<template>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,31 +1,20 @@
 | 
			
		||||
<script setup lang="ts">
 | 
			
		||||
import { computed } from 'vue';
 | 
			
		||||
import { useRoute } from 'vue-router';
 | 
			
		||||
import { GLOBAL_HEADER_MENU_ID, GLOBAL_SIDER_MENU_ID } from '@/constants/app';
 | 
			
		||||
import { useAppStore } from '@/store/modules/app';
 | 
			
		||||
import { useThemeStore } from '@/store/modules/theme';
 | 
			
		||||
import { useRouterPush } from '@/hooks/common/router';
 | 
			
		||||
import FirstLevelMenu from '../components/first-level-menu.vue';
 | 
			
		||||
import { useMixMenuContext } from '../../../context';
 | 
			
		||||
import { useMenu, useMixMenuContext } from '../../../context';
 | 
			
		||||
 | 
			
		||||
defineOptions({
 | 
			
		||||
  name: 'HorizontalMixMenu'
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
const route = useRoute();
 | 
			
		||||
const appStore = useAppStore();
 | 
			
		||||
const themeStore = useThemeStore();
 | 
			
		||||
const { allMenus, childLevelMenus, activeFirstLevelMenuKey, setActiveFirstLevelMenuKey } = useMixMenuContext();
 | 
			
		||||
const { routerPushByKeyWithMetaQuery } = useRouterPush();
 | 
			
		||||
 | 
			
		||||
const selectedKey = computed(() => {
 | 
			
		||||
  const { hideInMenu, activeMenu } = route.meta;
 | 
			
		||||
  const name = route.name as string;
 | 
			
		||||
 | 
			
		||||
  const routeName = (hideInMenu ? activeMenu : name) || name;
 | 
			
		||||
 | 
			
		||||
  return routeName;
 | 
			
		||||
});
 | 
			
		||||
const { allMenus, childLevelMenus, activeFirstLevelMenuKey, setActiveFirstLevelMenuKey } = useMixMenuContext();
 | 
			
		||||
const { selectedKey } = useMenu();
 | 
			
		||||
 | 
			
		||||
function handleSelectMixMenu(menu: App.Global.Menu) {
 | 
			
		||||
  setActiveFirstLevelMenuKey(menu.key);
 | 
			
		||||
@@ -56,9 +45,7 @@ function handleSelectMixMenu(menu: App.Global.Menu) {
 | 
			
		||||
      :theme-color="themeStore.themeColor"
 | 
			
		||||
      @select="handleSelectMixMenu"
 | 
			
		||||
      @toggle-sider-collapse="appStore.toggleSiderCollapse"
 | 
			
		||||
    >
 | 
			
		||||
      <slot></slot>
 | 
			
		||||
    </FirstLevelMenu>
 | 
			
		||||
    />
 | 
			
		||||
  </Teleport>
 | 
			
		||||
</template>
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -1,5 +1,5 @@
 | 
			
		||||
<script setup lang="ts">
 | 
			
		||||
import { computed, ref, watch } from 'vue';
 | 
			
		||||
import { ref, watch } from 'vue';
 | 
			
		||||
import { useRoute } from 'vue-router';
 | 
			
		||||
import type { RouteKey } from '@elegant-router/types';
 | 
			
		||||
import { SimpleScrollbar } from '@sa/materials';
 | 
			
		||||
@@ -8,7 +8,7 @@ import { useAppStore } from '@/store/modules/app';
 | 
			
		||||
import { useThemeStore } from '@/store/modules/theme';
 | 
			
		||||
import { useRouteStore } from '@/store/modules/route';
 | 
			
		||||
import { useRouterPush } from '@/hooks/common/router';
 | 
			
		||||
import { useMixMenuContext } from '../../../context';
 | 
			
		||||
import { useMenu, useMixMenuContext } from '../../../context';
 | 
			
		||||
 | 
			
		||||
defineOptions({
 | 
			
		||||
  name: 'ReversedHorizontalMixMenu'
 | 
			
		||||
@@ -18,6 +18,7 @@ const route = useRoute();
 | 
			
		||||
const appStore = useAppStore();
 | 
			
		||||
const themeStore = useThemeStore();
 | 
			
		||||
const routeStore = useRouteStore();
 | 
			
		||||
const { routerPushByKeyWithMetaQuery } = useRouterPush();
 | 
			
		||||
const {
 | 
			
		||||
  firstLevelMenus,
 | 
			
		||||
  childLevelMenus,
 | 
			
		||||
@@ -25,16 +26,7 @@ const {
 | 
			
		||||
  setActiveFirstLevelMenuKey,
 | 
			
		||||
  isActiveFirstLevelMenuHasChildren
 | 
			
		||||
} = useMixMenuContext();
 | 
			
		||||
const { routerPushByKeyWithMetaQuery } = useRouterPush();
 | 
			
		||||
 | 
			
		||||
const selectedKey = computed(() => {
 | 
			
		||||
  const { hideInMenu, activeMenu } = route.meta;
 | 
			
		||||
  const name = route.name as string;
 | 
			
		||||
 | 
			
		||||
  const routeName = (hideInMenu ? activeMenu : name) || name;
 | 
			
		||||
 | 
			
		||||
  return routeName;
 | 
			
		||||
});
 | 
			
		||||
const { selectedKey } = useMenu();
 | 
			
		||||
 | 
			
		||||
function handleSelectMixMenu(key: RouteKey) {
 | 
			
		||||
  setActiveFirstLevelMenuKey(key);
 | 
			
		||||
 
 | 
			
		||||
@@ -7,6 +7,7 @@ import { useThemeStore } from '@/store/modules/theme';
 | 
			
		||||
import { useRouteStore } from '@/store/modules/route';
 | 
			
		||||
import { useRouterPush } from '@/hooks/common/router';
 | 
			
		||||
import { GLOBAL_SIDER_MENU_ID } from '@/constants/app';
 | 
			
		||||
import { useMenu } from '../../../context';
 | 
			
		||||
 | 
			
		||||
defineOptions({
 | 
			
		||||
  name: 'VerticalMenu'
 | 
			
		||||
@@ -17,18 +18,10 @@ const appStore = useAppStore();
 | 
			
		||||
const themeStore = useThemeStore();
 | 
			
		||||
const routeStore = useRouteStore();
 | 
			
		||||
const { routerPushByKeyWithMetaQuery } = useRouterPush();
 | 
			
		||||
const { selectedKey } = useMenu();
 | 
			
		||||
 | 
			
		||||
const inverted = computed(() => !themeStore.darkMode && themeStore.sider.inverted);
 | 
			
		||||
 | 
			
		||||
const selectedKey = computed(() => {
 | 
			
		||||
  const { hideInMenu, activeMenu } = route.meta;
 | 
			
		||||
  const name = route.name as string;
 | 
			
		||||
 | 
			
		||||
  const routeName = (hideInMenu ? activeMenu : name) || name;
 | 
			
		||||
 | 
			
		||||
  return routeName;
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
const expandedKeys = ref<string[]>([]);
 | 
			
		||||
 | 
			
		||||
function updateExpandedKeys() {
 | 
			
		||||
 
 | 
			
		||||
@@ -9,7 +9,7 @@ import { useRouteStore } from '@/store/modules/route';
 | 
			
		||||
import { useRouterPush } from '@/hooks/common/router';
 | 
			
		||||
import { $t } from '@/locales';
 | 
			
		||||
import { GLOBAL_SIDER_MENU_ID } from '@/constants/app';
 | 
			
		||||
import { useMixMenuContext } from '../../../context';
 | 
			
		||||
import { useMenu, useMixMenuContext } from '../../../context';
 | 
			
		||||
import FirstLevelMenu from '../components/first-level-menu.vue';
 | 
			
		||||
import GlobalLogo from '../../global-logo/index.vue';
 | 
			
		||||
 | 
			
		||||
@@ -31,6 +31,7 @@ const {
 | 
			
		||||
  getActiveFirstLevelMenuKey
 | 
			
		||||
  //
 | 
			
		||||
} = useMixMenuContext();
 | 
			
		||||
const { selectedKey } = useMenu();
 | 
			
		||||
 | 
			
		||||
const inverted = computed(() => !themeStore.darkMode && themeStore.sider.inverted);
 | 
			
		||||
 | 
			
		||||
@@ -49,19 +50,13 @@ function handleSelectMixMenu(menu: App.Global.Menu) {
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function handleResetActiveMenu() {
 | 
			
		||||
  getActiveFirstLevelMenuKey();
 | 
			
		||||
  setDrawerVisible(false);
 | 
			
		||||
 | 
			
		||||
  if (!appStore.mixSiderFixed) {
 | 
			
		||||
    getActiveFirstLevelMenuKey();
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const selectedKey = computed(() => {
 | 
			
		||||
  const { hideInMenu, activeMenu } = route.meta;
 | 
			
		||||
  const name = route.name as string;
 | 
			
		||||
 | 
			
		||||
  const routeName = (hideInMenu ? activeMenu : name) || name;
 | 
			
		||||
 | 
			
		||||
  return routeName;
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
const expandedKeys = ref<string[]>([]);
 | 
			
		||||
 | 
			
		||||
function updateExpandedKeys() {
 | 
			
		||||
@@ -119,9 +114,6 @@ watch(
 | 
			
		||||
              mode="vertical"
 | 
			
		||||
              :value="selectedKey"
 | 
			
		||||
              :options="childLevelMenus"
 | 
			
		||||
              :collapsed="appStore.siderCollapse"
 | 
			
		||||
              :collapsed-width="themeStore.sider.collapsedWidth"
 | 
			
		||||
              :collapsed-icon-size="22"
 | 
			
		||||
              :inverted="inverted"
 | 
			
		||||
              :indent="18"
 | 
			
		||||
              @update:value="routerPushByKeyWithMetaQuery"
 | 
			
		||||
 
 | 
			
		||||
@@ -101,10 +101,10 @@ const isWrapperScrollMode = computed(() => themeStore.layout.scrollMode === 'wra
 | 
			
		||||
    >
 | 
			
		||||
      <NSwitch v-model:value="themeStore.footer.right" />
 | 
			
		||||
    </SettingItem>
 | 
			
		||||
    <SettingItem key="8" :label="$t('theme.watermark.visible')">
 | 
			
		||||
    <SettingItem v-if="themeStore.watermark" key="8" :label="$t('theme.watermark.visible')">
 | 
			
		||||
      <NSwitch v-model:value="themeStore.watermark.visible" />
 | 
			
		||||
    </SettingItem>
 | 
			
		||||
    <SettingItem v-if="themeStore.watermark.visible" key="8-1" :label="$t('theme.watermark.text')">
 | 
			
		||||
    <SettingItem v-if="themeStore.watermark?.visible" key="8-1" :label="$t('theme.watermark.text')">
 | 
			
		||||
      <NInput
 | 
			
		||||
        v-model:value="themeStore.watermark.text"
 | 
			
		||||
        autosize
 | 
			
		||||
 
 | 
			
		||||
@@ -1,8 +1,20 @@
 | 
			
		||||
import { h } from 'vue';
 | 
			
		||||
import type { App } from 'vue';
 | 
			
		||||
import { NButton } from 'naive-ui';
 | 
			
		||||
import { $t } from '../locales';
 | 
			
		||||
import { $t } from '@/locales';
 | 
			
		||||
 | 
			
		||||
export function setupAppErrorHandle(app: App) {
 | 
			
		||||
  app.config.errorHandler = (err, vm, info) => {
 | 
			
		||||
    // eslint-disable-next-line no-console
 | 
			
		||||
    console.error(err, vm, info);
 | 
			
		||||
  };
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export function setupAppVersionNotification() {
 | 
			
		||||
  const canAutoUpdateApp = import.meta.env.VITE_AUTOMATICALLY_DETECT_UPDATE === 'Y';
 | 
			
		||||
 | 
			
		||||
  if (!canAutoUpdateApp) return;
 | 
			
		||||
 | 
			
		||||
  let isShow = false;
 | 
			
		||||
 | 
			
		||||
  document.addEventListener('visibilitychange', async () => {
 | 
			
		||||
@@ -52,9 +64,7 @@ export function setupAppVersionNotification() {
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
async function getHtmlBuildTime() {
 | 
			
		||||
  const baseURL = import.meta.env.VITE_BASE_URL;
 | 
			
		||||
 | 
			
		||||
  const res = await fetch(`${baseURL}index.html`);
 | 
			
		||||
  const res = await fetch(`/index.html?time=${Date.now()}`);
 | 
			
		||||
 | 
			
		||||
  const html = await res.text();
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -97,10 +97,13 @@ function transformElegantRouteToVueRoute(
 | 
			
		||||
    if (component) {
 | 
			
		||||
      if (isSingleLevelRoute(route)) {
 | 
			
		||||
        const { layout, view } = getSingleLevelRouteComponent(component);
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
        const singleLevelRoute: RouteRecordRaw = {
 | 
			
		||||
          path,
 | 
			
		||||
          component: layouts[layout],
 | 
			
		||||
          meta: {
 | 
			
		||||
            title: route.meta?.title || ''
 | 
			
		||||
          },
 | 
			
		||||
          children: [
 | 
			
		||||
            {
 | 
			
		||||
              name,
 | 
			
		||||
@@ -110,36 +113,35 @@ function transformElegantRouteToVueRoute(
 | 
			
		||||
            } as RouteRecordRaw
 | 
			
		||||
          ]
 | 
			
		||||
        };
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
        return [singleLevelRoute];
 | 
			
		||||
      }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
      if (isLayout(component)) {
 | 
			
		||||
        const layoutName = getLayoutName(component);
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
        vueRoute.component = layouts[layoutName];
 | 
			
		||||
      }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
      if (isView(component)) {
 | 
			
		||||
        const viewName = getViewName(component);
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
        vueRoute.component = views[viewName];
 | 
			
		||||
      }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
  } catch (error: any) {
 | 
			
		||||
    console.error(`Error transforming route "${route.name}": ${error.toString()}`);
 | 
			
		||||
    return [];
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  
 | 
			
		||||
  // add redirect to child
 | 
			
		||||
  if (children?.length && !vueRoute.redirect) {
 | 
			
		||||
    vueRoute.redirect = {
 | 
			
		||||
      name: children[0].name
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
  if (children?.length) {
 | 
			
		||||
    const childRoutes = children.flatMap(child => transformElegantRouteToVueRoute(child, layouts, views));
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -4,7 +4,7 @@ import { useAuthStore } from '@/store/modules/auth';
 | 
			
		||||
import { $t } from '@/locales';
 | 
			
		||||
import { localStg } from '@/utils/storage';
 | 
			
		||||
import { getServiceBaseURL } from '@/utils/service';
 | 
			
		||||
import { handleRefreshToken, showErrorMsg } from './shared';
 | 
			
		||||
import { getAuthorization, handleExpiredRequest, showErrorMsg } from './shared';
 | 
			
		||||
import type { RequestInstanceState } from './type';
 | 
			
		||||
 | 
			
		||||
const isHttpProxy = import.meta.env.DEV && import.meta.env.VITE_HTTP_PROXY === 'Y';
 | 
			
		||||
@@ -19,12 +19,8 @@ export const request = createFlatRequest<App.Service.Response, RequestInstanceSt
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    async onRequest(config) {
 | 
			
		||||
      const { headers } = config;
 | 
			
		||||
 | 
			
		||||
      // set token
 | 
			
		||||
      const token = localStg.get('token');
 | 
			
		||||
      const Authorization = token ? `Bearer ${token}` : null;
 | 
			
		||||
      Object.assign(headers, { Authorization });
 | 
			
		||||
      const Authorization = getAuthorization();
 | 
			
		||||
      Object.assign(config.headers, { Authorization });
 | 
			
		||||
 | 
			
		||||
      return config;
 | 
			
		||||
    },
 | 
			
		||||
@@ -83,15 +79,13 @@ export const request = createFlatRequest<App.Service.Response, RequestInstanceSt
 | 
			
		||||
      // when the backend response code is in `expiredTokenCodes`, it means the token is expired, and refresh token
 | 
			
		||||
      // the api `refreshToken` can not return error code in `expiredTokenCodes`, otherwise it will be a dead loop, should return `logoutCodes` or `modalLogoutCodes`
 | 
			
		||||
      const expiredTokenCodes = import.meta.env.VITE_SERVICE_EXPIRED_TOKEN_CODES?.split(',') || [];
 | 
			
		||||
      if (expiredTokenCodes.includes(responseCode) && !request.state.isRefreshingToken) {
 | 
			
		||||
        request.state.isRefreshingToken = true;
 | 
			
		||||
      if (expiredTokenCodes.includes(responseCode)) {
 | 
			
		||||
        const success = await handleExpiredRequest(request.state);
 | 
			
		||||
        if (success) {
 | 
			
		||||
          const Authorization = getAuthorization();
 | 
			
		||||
          Object.assign(response.config.headers, { Authorization });
 | 
			
		||||
 | 
			
		||||
        const refreshConfig = await handleRefreshToken(response.config);
 | 
			
		||||
 | 
			
		||||
        request.state.isRefreshingToken = false;
 | 
			
		||||
 | 
			
		||||
        if (refreshConfig) {
 | 
			
		||||
          return instance.request(refreshConfig) as Promise<AxiosResponse>;
 | 
			
		||||
          return instance.request(response.config) as Promise<AxiosResponse>;
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -1,34 +1,44 @@
 | 
			
		||||
import type { AxiosRequestConfig } from 'axios';
 | 
			
		||||
import { useAuthStore } from '@/store/modules/auth';
 | 
			
		||||
import { localStg } from '@/utils/storage';
 | 
			
		||||
import { fetchRefreshToken } from '../api';
 | 
			
		||||
import type { RequestInstanceState } from './type';
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * refresh token
 | 
			
		||||
 *
 | 
			
		||||
 * @param axiosConfig - request config when the token is expired
 | 
			
		||||
 */
 | 
			
		||||
export async function handleRefreshToken(axiosConfig: AxiosRequestConfig) {
 | 
			
		||||
export function getAuthorization() {
 | 
			
		||||
  const token = localStg.get('token');
 | 
			
		||||
  const Authorization = token ? `Bearer ${token}` : null;
 | 
			
		||||
 | 
			
		||||
  return Authorization;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/** refresh token */
 | 
			
		||||
async function handleRefreshToken() {
 | 
			
		||||
  const { resetStore } = useAuthStore();
 | 
			
		||||
 | 
			
		||||
  const refreshToken = localStg.get('refreshToken') || '';
 | 
			
		||||
  const { error, data } = await fetchRefreshToken(refreshToken);
 | 
			
		||||
  const rToken = localStg.get('refreshToken') || '';
 | 
			
		||||
  const { error, data } = await fetchRefreshToken(rToken);
 | 
			
		||||
  if (!error) {
 | 
			
		||||
    localStg.set('token', data.token);
 | 
			
		||||
    localStg.set('refreshToken', data.refreshToken);
 | 
			
		||||
 | 
			
		||||
    const config = { ...axiosConfig };
 | 
			
		||||
    if (config.headers) {
 | 
			
		||||
      config.headers.Authorization = data.token;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    return config;
 | 
			
		||||
    return true;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  resetStore();
 | 
			
		||||
 | 
			
		||||
  return null;
 | 
			
		||||
  return false;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export async function handleExpiredRequest(state: RequestInstanceState) {
 | 
			
		||||
  if (!state.refreshTokenFn) {
 | 
			
		||||
    state.refreshTokenFn = handleRefreshToken();
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  const success = await state.refreshTokenFn;
 | 
			
		||||
 | 
			
		||||
  setTimeout(() => {
 | 
			
		||||
    state.refreshTokenFn = null;
 | 
			
		||||
  }, 1000);
 | 
			
		||||
 | 
			
		||||
  return success;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export function showErrorMsg(state: RequestInstanceState, message: string) {
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,6 @@
 | 
			
		||||
export interface RequestInstanceState {
 | 
			
		||||
  /** whether the request is refreshing token */
 | 
			
		||||
  isRefreshingToken: boolean;
 | 
			
		||||
  refreshTokenFn: Promise<boolean> | null;
 | 
			
		||||
  /** the request error message stack */
 | 
			
		||||
  errMsgStack: string[];
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -71,9 +71,7 @@ export const useAuthStore = defineStore(SetupStoreId.Auth, () => {
 | 
			
		||||
      if (pass) {
 | 
			
		||||
        await routeStore.initAuthRoute();
 | 
			
		||||
 | 
			
		||||
        if (redirect) {
 | 
			
		||||
          await redirectFromLogin();
 | 
			
		||||
        }
 | 
			
		||||
        await redirectFromLogin(redirect);
 | 
			
		||||
 | 
			
		||||
        if (routeStore.isInitAuthRoute) {
 | 
			
		||||
          window.$notification?.success({
 | 
			
		||||
 
 | 
			
		||||
@@ -281,15 +281,25 @@ export function getBreadcrumbsByRoute(
 | 
			
		||||
  const key = route.name as string;
 | 
			
		||||
  const activeKey = route.meta?.activeMenu;
 | 
			
		||||
 | 
			
		||||
  const menuKey = activeKey || key;
 | 
			
		||||
 | 
			
		||||
  for (const menu of menus) {
 | 
			
		||||
    if (menu.key === menuKey) {
 | 
			
		||||
      const breadcrumbMenu = menuKey !== activeKey ? menu : getGlobalMenuByBaseRoute(route);
 | 
			
		||||
 | 
			
		||||
    if (menu.key === key) {
 | 
			
		||||
      const breadcrumbMenu = menu;
 | 
			
		||||
      return [transformMenuToBreadcrumb(breadcrumbMenu)];
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (menu.key === activeKey) {
 | 
			
		||||
      const ROUTE_DEGREE_SPLITTER = '_';
 | 
			
		||||
 | 
			
		||||
      const parentKey = key.split(ROUTE_DEGREE_SPLITTER).slice(0, -1).join(ROUTE_DEGREE_SPLITTER);
 | 
			
		||||
 | 
			
		||||
      const breadcrumbMenu = getGlobalMenuByBaseRoute(route);
 | 
			
		||||
      if (parentKey !== activeKey) {
 | 
			
		||||
        return [transformMenuToBreadcrumb(breadcrumbMenu)];
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      return [transformMenuToBreadcrumb(menu), transformMenuToBreadcrumb(breadcrumbMenu)];
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (menu.children?.length) {
 | 
			
		||||
      const result = getBreadcrumbsByRoute(route, menu.children);
 | 
			
		||||
      if (result.length > 0) {
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										3
									
								
								src/typings/api.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						@@ -20,6 +20,9 @@ declare namespace Api {
 | 
			
		||||
      records: T[];
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /** common search params of table */
 | 
			
		||||
    type CommonSearchParams = Pick<Common.PaginatingCommonParams, 'current' | 'size'>;
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * enable status
 | 
			
		||||
     *
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										2
									
								
								src/typings/app.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						@@ -96,7 +96,7 @@ declare namespace App {
 | 
			
		||||
        right: boolean;
 | 
			
		||||
      };
 | 
			
		||||
      /** Watermark */
 | 
			
		||||
      watermark: {
 | 
			
		||||
      watermark?: {
 | 
			
		||||
        /** Whether to show the watermark */
 | 
			
		||||
        visible: boolean;
 | 
			
		||||
        /** Watermark text */
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										4
									
								
								src/typings/elegant-router.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						@@ -37,7 +37,7 @@ declare module "@elegant-router/types" {
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * custom route key
 | 
			
		||||
   */ 
 | 
			
		||||
   */
 | 
			
		||||
  export type CustomRouteKey = Extract<
 | 
			
		||||
    RouteKey,
 | 
			
		||||
    | "root"
 | 
			
		||||
@@ -46,7 +46,7 @@ declare module "@elegant-router/types" {
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * the generated route key
 | 
			
		||||
   */ 
 | 
			
		||||
   */
 | 
			
		||||
  export type GeneratedRouteKey = Exclude<RouteKey, CustomRouteKey>;
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										2
									
								
								src/typings/env.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						@@ -103,6 +103,8 @@ declare namespace Env {
 | 
			
		||||
    readonly VITE_ICONIFY_URL?: string;
 | 
			
		||||
    /** Used to differentiate storage across different domains */
 | 
			
		||||
    readonly VITE_STORAGE_PREFIX?: string;
 | 
			
		||||
    /** Whether to automatically detect updates after configuring application packaging */
 | 
			
		||||
    readonly VITE_AUTOMATICALLY_DETECT_UPDATE?: CommonType.YesOrNo;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										2
									
								
								src/typings/naive-ui.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						@@ -26,7 +26,7 @@ declare namespace NaiveUI {
 | 
			
		||||
 | 
			
		||||
  type TableColumn<T> = TableColumnWithKey<T> | DataTableSelectionColumn<T> | DataTableExpandColumn<T>;
 | 
			
		||||
 | 
			
		||||
  type TableApiFn<T = any, R = Api.SystemManage.CommonSearchParams> = (
 | 
			
		||||
  type TableApiFn<T = any, R = Api.Common.CommonSearchParams> = (
 | 
			
		||||
    params: R
 | 
			
		||||
  ) => Promise<FlatResponseData<Api.Common.PaginatingQueryRecord<T>>>;
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -1,3 +1,5 @@
 | 
			
		||||
import json5 from 'json5';
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Create service config by current env
 | 
			
		||||
 *
 | 
			
		||||
@@ -8,10 +10,10 @@ export function createServiceConfig(env: Env.ImportMeta) {
 | 
			
		||||
 | 
			
		||||
  let other = {} as Record<App.Service.OtherBaseURLKey, string>;
 | 
			
		||||
  try {
 | 
			
		||||
    other = JSON.parse(VITE_OTHER_SERVICE_BASE_URL);
 | 
			
		||||
    other = json5.parse(VITE_OTHER_SERVICE_BASE_URL);
 | 
			
		||||
  } catch (error) {
 | 
			
		||||
    // eslint-disable-next-line no-console
 | 
			
		||||
    console.error('VITE_OTHER_SERVICE_BASE_URL is not a valid JSON string');
 | 
			
		||||
    console.error('VITE_OTHER_SERVICE_BASE_URL is not a valid json5 string');
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  const httpConfig: App.Service.SimpleServiceConfig = {
 | 
			
		||||
 
 | 
			
		||||
@@ -9,6 +9,8 @@ export default defineConfig(configEnv => {
 | 
			
		||||
 | 
			
		||||
  const buildTime = getBuildTime();
 | 
			
		||||
 | 
			
		||||
  const enableProxy = configEnv.command === 'serve' && !configEnv.isPreview;
 | 
			
		||||
 | 
			
		||||
  return {
 | 
			
		||||
    base: viteEnv.VITE_BASE_URL,
 | 
			
		||||
    resolve: {
 | 
			
		||||
@@ -32,7 +34,7 @@ export default defineConfig(configEnv => {
 | 
			
		||||
      host: '0.0.0.0',
 | 
			
		||||
      port: 9527,
 | 
			
		||||
      open: true,
 | 
			
		||||
      proxy: createViteProxy(viteEnv, configEnv.command === 'serve'),
 | 
			
		||||
      proxy: createViteProxy(viteEnv, enableProxy),
 | 
			
		||||
      fs: {
 | 
			
		||||
        cachedChecks: false
 | 
			
		||||
      }
 | 
			
		||||
 
 | 
			
		||||