Compare commits
26 Commits
fb5a59d81b
...
003c43140b
Author | SHA1 | Date | |
---|---|---|---|
003c43140b | |||
6850347b89 | |||
4e625111ce | |||
8524ae7666 | |||
d6ae85d218 | |||
89719abe34 | |||
8d7f91dccf | |||
33ade53904 | |||
ab9c84d831 | |||
7651816495 | |||
4539fe01fb | |||
4e9839bd48 | |||
a15b683b1d | |||
9df8d2f55f | |||
710374398a | |||
52318c106d | |||
9027632bef | |||
b96c46baa9 | |||
2d31d7dc62 | |||
e538355f2b | |||
f89835578c | |||
4eb77eac78 | |||
adca2e26be | |||
ff576f3f42 | |||
8fcc70d73d | |||
923eb98a5c |
13
.vscode/settings.json
vendored
@ -4,7 +4,18 @@
|
|||||||
"source.organizeImports": "never"
|
"source.organizeImports": "never"
|
||||||
},
|
},
|
||||||
"editor.formatOnSave": false,
|
"editor.formatOnSave": false,
|
||||||
"eslint.validate": ["html", "css", "scss", "json", "jsonc"],
|
"eslint.validate": [
|
||||||
|
"html",
|
||||||
|
"css",
|
||||||
|
"scss",
|
||||||
|
"json",
|
||||||
|
"jsonc",
|
||||||
|
"javascript",
|
||||||
|
"javascriptreact",
|
||||||
|
"typescript",
|
||||||
|
"typescriptreact",
|
||||||
|
"vue"
|
||||||
|
],
|
||||||
"i18n-ally.displayLanguage": "zh-cn",
|
"i18n-ally.displayLanguage": "zh-cn",
|
||||||
"i18n-ally.enabledParsers": ["ts"],
|
"i18n-ally.enabledParsers": ["ts"],
|
||||||
"i18n-ally.enabledFrameworks": ["vue"],
|
"i18n-ally.enabledFrameworks": ["vue"],
|
||||||
|
39
CHANGELOG.md
@ -1,5 +1,44 @@
|
|||||||
# 更新日志
|
# 更新日志
|
||||||
|
|
||||||
|
## [v1.1.3](https://gitee.com/xlsea/ruoyi-plus-soybean/compare/v1.1.2...v1.1.3) (2025-08-16)
|
||||||
|
|
||||||
|
### 🐞 Bug 修复
|
||||||
|
|
||||||
|
- **hooks**:
|
||||||
|
- 非安全环境下不使用流式下载 - by @m-xlsea [<samp>(f8983)</samp>](https://gitee.com/xlsea/ruoyi-plus-soybean/commit/f8983557)
|
||||||
|
- 修复oss下载时未转码问题 - by **AN** [<samp>(2d31d)</samp>](https://gitee.com/xlsea/ruoyi-plus-soybean/commit/2d31d7dc)
|
||||||
|
- **project**:
|
||||||
|
- 关闭多租户功能后仍然遍历租户列表导致控制台报错的问题 - by **wang_rui** [<samp>(b96c4)</samp>](https://gitee.com/xlsea/ruoyi-plus-soybean/commit/b96c46ba)
|
||||||
|
- 关闭多租户功能后仍然遍历租户列表导致控制台报错的问题 Merge pull request !25 from littleghost2016/dev - by **不寻俗** [<samp>(90276)</samp>](https://gitee.com/xlsea/ruoyi-plus-soybean/commit/9027632b)
|
||||||
|
- **projects**:
|
||||||
|
- 修复一级菜单隐藏失效问题 - by **AN** [<samp>(8fcc7)</samp>](https://gitee.com/xlsea/ruoyi-plus-soybean/commit/8fcc70d7)
|
||||||
|
- 修复日期搜索条件清除问题 - by **AN** [<samp>(52318)</samp>](https://gitee.com/xlsea/ruoyi-plus-soybean/commit/52318c10)
|
||||||
|
- 修复登录过期事件监听未被重置 - by @m-xlsea [<samp>(71037)</samp>](https://gitee.com/xlsea/ruoyi-plus-soybean/commit/71037439)
|
||||||
|
- 修复用户新增时角色下拉包含超级管理员问题 - by **AN** [<samp>(a15b6)</samp>](https://gitee.com/xlsea/ruoyi-plus-soybean/commit/a15b683b)
|
||||||
|
- 修复用户导入功能无法更新问题 - by **AN** [<samp>(4e983)</samp>](https://gitee.com/xlsea/ruoyi-plus-soybean/commit/4e9839bd)
|
||||||
|
- Fix the icon size in the image preview toolbar - by @m-xlsea [<samp>(4539f)</samp>](https://gitee.com/xlsea/ruoyi-plus-soybean/commit/4539fe01)
|
||||||
|
- 修复新增用户未查询角色列表问题 - by **AN** [<samp>(d6ae8)</samp>](https://gitee.com/xlsea/ruoyi-plus-soybean/commit/d6ae85d2)
|
||||||
|
- **readme**:
|
||||||
|
- update GitHub stars and forks links for gitee - by @soybeanjs [<samp>(923eb)</samp>](https://gitee.com/xlsea/ruoyi-plus-soybean/commit/923eb98a)
|
||||||
|
|
||||||
|
### 💅 重构
|
||||||
|
|
||||||
|
- **menu**:
|
||||||
|
- 菜单管理中隐藏的菜单显示灰色 - by **NicholasLD** [<samp>(adca2)</samp>](https://gitee.com/xlsea/ruoyi-plus-soybean/commit/adca2e26)
|
||||||
|
- 菜单管理中隐藏的菜单显示灰色 Merge pull request !24 from NicholasLD/N/A - by **不寻俗** [<samp>(4eb77)</samp>](https://gitee.com/xlsea/ruoyi-plus-soybean/commit/4eb77eac)
|
||||||
|
- **projects**:
|
||||||
|
- 菜单列表新增禁用菜单样式 - by @m-xlsea [<samp>(e5383)</samp>](https://gitee.com/xlsea/ruoyi-plus-soybean/commit/e538355f)
|
||||||
|
|
||||||
|
### 🏡 杂项
|
||||||
|
|
||||||
|
- **other**: update the ESLint validation configuration to support more file types. - by **Azir-11** [<samp>(8d7f9)</samp>](https://gitee.com/xlsea/ruoyi-plus-soybean/commit/8d7f91dc)
|
||||||
|
- **readme**: remove DartNode sponsorship badge from README files - by @soybeanjs [<samp>(33ade)</samp>](https://gitee.com/xlsea/ruoyi-plus-soybean/commit/33ade539)
|
||||||
|
|
||||||
|
### ❤️ 贡献者
|
||||||
|
|
||||||
|
[](https://github.com/soybeanjs) [](https://github.com/m-xlsea) [](https://gitee.com/elio-an) [](https://github.com/Azir-11) [](https://github.com/NicholasLD)
|
||||||
|
[wang_rui](mailto:wrr1996@163.com)
|
||||||
|
|
||||||
## [v1.1.2](https://gitee.com/xlsea/ruoyi-plus-soybean/compare/v1.1.1...v1.1.2) (2025-07-24)
|
## [v1.1.2](https://gitee.com/xlsea/ruoyi-plus-soybean/compare/v1.1.1...v1.1.2) (2025-07-24)
|
||||||
|
|
||||||
### 🐞 Bug 修复
|
### 🐞 Bug 修复
|
||||||
|
@ -22,7 +22,7 @@
|
|||||||
|
|
||||||
# 📢 重要通知
|
# 📢 重要通知
|
||||||
|
|
||||||
1.1.2 版本已经正式发布,工作流版本迎来首个版本(请切换 [flow](https://gitee.com/xlsea/ruoyi-plus-soybean/tree/flow/) 分支查看),但仍然建议:
|
1.1.3 版本已经正式发布(工作流版本请切换 [flow](https://gitee.com/xlsea/ruoyi-plus-soybean/tree/flow/) 分支查看),但仍然建议:
|
||||||
- 在生产环境使用前进行充分测试
|
- 在生产环境使用前进行充分测试
|
||||||
- 关注项目更新,及时获取最新版本
|
- 关注项目更新,及时获取最新版本
|
||||||
- 积极反馈问题,帮助我们快速迭代
|
- 积极反馈问题,帮助我们快速迭代
|
||||||
@ -40,6 +40,11 @@
|
|||||||
|
|
||||||
<p style="font-weight: bold; font-size: 24px;">后端需要替换代码生成模板与菜单 SQL,详细请看 <a href="#代码生成与菜单更新">代码生成与菜单更新</a></p>
|
<p style="font-weight: bold; font-size: 24px;">后端需要替换代码生成模板与菜单 SQL,详细请看 <a href="#代码生成与菜单更新">代码生成与菜单更新</a></p>
|
||||||
|
|
||||||
|
# 💎 友情链接
|
||||||
|
|
||||||
|
- [Snail Job Pro](https://pro.snailjob.opensnail.com/home) - 灵活,可靠和快速的分布式任务重试和分布式任务调度平台
|
||||||
|
- [AiZuDa - 爱组搭(飞龙工作流企业版)](https://naiveui.aizuda.com) - 像搭积木一样进行低代码甚至零代码快速构建应用
|
||||||
|
|
||||||
## 📋 项目概述
|
## 📋 项目概述
|
||||||
|
|
||||||
RuoYi-Plus-Soybean 是一个现代化的企业级多租户管理系统,它结合了 RuoYi-Vue-Plus 的强大后端功能和 Soybean Admin 的现代化前端特性,为开发者提供了完整的企业管理解决方案。
|
RuoYi-Plus-Soybean 是一个现代化的企业级多租户管理系统,它结合了 RuoYi-Vue-Plus 的强大后端功能和 Soybean Admin 的现代化前端特性,为开发者提供了完整的企业管理解决方案。
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "ruoyi-vue-plus",
|
"name": "ruoyi-vue-plus",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"version": "1.1.2",
|
"version": "1.1.3",
|
||||||
"description": "结合了 RuoYi-Vue-Plus 的强大后端功能和 Soybean Admin 的现代化前端特性,为开发者提供了完整的企业管理解决方案。",
|
"description": "结合了 RuoYi-Vue-Plus 的强大后端功能和 Soybean Admin 的现代化前端特性,为开发者提供了完整的企业管理解决方案。",
|
||||||
"author": {
|
"author": {
|
||||||
"name": "xlsea",
|
"name": "xlsea",
|
||||||
|
BIN
public/logo.png
Before Width: | Height: | Size: 124 KiB After Width: | Height: | Size: 58 KiB |
Before Width: | Height: | Size: 18 KiB After Width: | Height: | Size: 14 KiB |
Before Width: | Height: | Size: 42 KiB After Width: | Height: | Size: 29 KiB |
Before Width: | Height: | Size: 3.0 KiB After Width: | Height: | Size: 2.5 KiB |
Before Width: | Height: | Size: 7.8 KiB After Width: | Height: | Size: 6.3 KiB |
Before Width: | Height: | Size: 14 KiB After Width: | Height: | Size: 11 KiB |
Before Width: | Height: | Size: 20 KiB After Width: | Height: | Size: 15 KiB |
Before Width: | Height: | Size: 22 KiB After Width: | Height: | Size: 16 KiB |
Before Width: | Height: | Size: 48 KiB After Width: | Height: | Size: 33 KiB |
Before Width: | Height: | Size: 2.7 KiB After Width: | Height: | Size: 2.3 KiB |
Before Width: | Height: | Size: 54 KiB After Width: | Height: | Size: 36 KiB |
Before Width: | Height: | Size: 4.8 KiB After Width: | Height: | Size: 3.8 KiB |
Before Width: | Height: | Size: 8.7 KiB After Width: | Height: | Size: 7.2 KiB |
Before Width: | Height: | Size: 12 KiB After Width: | Height: | Size: 9.3 KiB |
Before Width: | Height: | Size: 5.5 KiB After Width: | Height: | Size: 4.6 KiB |
Before Width: | Height: | Size: 5.4 KiB After Width: | Height: | Size: 4.4 KiB |
Before Width: | Height: | Size: 24 KiB After Width: | Height: | Size: 18 KiB |
Before Width: | Height: | Size: 5.4 KiB After Width: | Height: | Size: 4.4 KiB |
Before Width: | Height: | Size: 5.3 KiB After Width: | Height: | Size: 4.4 KiB |
Before Width: | Height: | Size: 15 KiB After Width: | Height: | Size: 11 KiB |
Before Width: | Height: | Size: 5.3 KiB After Width: | Height: | Size: 4.4 KiB |
Before Width: | Height: | Size: 13 KiB After Width: | Height: | Size: 9.8 KiB |
Before Width: | Height: | Size: 34 KiB After Width: | Height: | Size: 24 KiB |
Before Width: | Height: | Size: 13 KiB After Width: | Height: | Size: 9.8 KiB |
Before Width: | Height: | Size: 21 KiB After Width: | Height: | Size: 15 KiB |
Before Width: | Height: | Size: 57 KiB After Width: | Height: | Size: 38 KiB |
Before Width: | Height: | Size: 21 KiB After Width: | Height: | Size: 15 KiB |
Before Width: | Height: | Size: 29 KiB After Width: | Height: | Size: 21 KiB |
Before Width: | Height: | Size: 83 KiB After Width: | Height: | Size: 52 KiB |
Before Width: | Height: | Size: 29 KiB After Width: | Height: | Size: 21 KiB |
Before Width: | Height: | Size: 62 KiB After Width: | Height: | Size: 47 KiB |
Before Width: | Height: | Size: 103 KiB After Width: | Height: | Size: 62 KiB |
Before Width: | Height: | Size: 1.2 KiB After Width: | Height: | Size: 1.1 KiB |
Before Width: | Height: | Size: 3.6 KiB After Width: | Height: | Size: 2.8 KiB |
Before Width: | Height: | Size: 3.6 KiB After Width: | Height: | Size: 2.8 KiB |
Before Width: | Height: | Size: 6.4 KiB After Width: | Height: | Size: 4.7 KiB |
Before Width: | Height: | Size: 2.2 KiB After Width: | Height: | Size: 1.8 KiB |
Before Width: | Height: | Size: 6.1 KiB After Width: | Height: | Size: 4.6 KiB |
Before Width: | Height: | Size: 6.1 KiB After Width: | Height: | Size: 4.6 KiB |
Before Width: | Height: | Size: 10 KiB After Width: | Height: | Size: 7.0 KiB |
Before Width: | Height: | Size: 3.6 KiB After Width: | Height: | Size: 2.8 KiB |
Before Width: | Height: | Size: 9.3 KiB After Width: | Height: | Size: 6.5 KiB |
Before Width: | Height: | Size: 9.3 KiB After Width: | Height: | Size: 6.5 KiB |
Before Width: | Height: | Size: 16 KiB After Width: | Height: | Size: 10 KiB |
Before Width: | Height: | Size: 278 KiB After Width: | Height: | Size: 59 KiB |
Before Width: | Height: | Size: 16 KiB After Width: | Height: | Size: 10 KiB |
Before Width: | Height: | Size: 26 KiB After Width: | Height: | Size: 15 KiB |
Before Width: | Height: | Size: 8.7 KiB After Width: | Height: | Size: 6.1 KiB |
Before Width: | Height: | Size: 21 KiB After Width: | Height: | Size: 13 KiB |
Before Width: | Height: | Size: 24 KiB After Width: | Height: | Size: 14 KiB |
@ -21,27 +21,27 @@ const attrs: SelectProps = useAttrs();
|
|||||||
|
|
||||||
const { loading: postLoading, startLoading: startPostLoading, endLoading: endPostLoading } = useLoading();
|
const { loading: postLoading, startLoading: startPostLoading, endLoading: endPostLoading } = useLoading();
|
||||||
|
|
||||||
/** the enabled role options */
|
/** the enabled post options */
|
||||||
const roleOptions = ref<CommonType.Option<CommonType.IdType>[]>([]);
|
const postOptions = ref<CommonType.Option<CommonType.IdType>[]>([]);
|
||||||
|
|
||||||
watch(
|
watch(
|
||||||
() => props.deptId,
|
() => props.deptId,
|
||||||
() => {
|
() => {
|
||||||
if (!props.deptId) {
|
if (!props.deptId) {
|
||||||
roleOptions.value = [];
|
postOptions.value = [];
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
getRoleOptions();
|
getPostOptions();
|
||||||
},
|
},
|
||||||
{ immediate: true }
|
{ immediate: true }
|
||||||
);
|
);
|
||||||
|
|
||||||
async function getRoleOptions() {
|
async function getPostOptions() {
|
||||||
startPostLoading();
|
startPostLoading();
|
||||||
const { error, data } = await fetchGetPostSelect(props.deptId!);
|
const { error, data } = await fetchGetPostSelect(props.deptId!);
|
||||||
|
|
||||||
if (!error) {
|
if (!error) {
|
||||||
roleOptions.value = data.map(item => ({
|
postOptions.value = data.map(item => ({
|
||||||
label: item.postName,
|
label: item.postName,
|
||||||
value: item.postId
|
value: item.postId
|
||||||
}));
|
}));
|
||||||
@ -54,7 +54,7 @@ async function getRoleOptions() {
|
|||||||
<NSelect
|
<NSelect
|
||||||
v-model:value="value"
|
v-model:value="value"
|
||||||
:loading="postLoading"
|
:loading="postLoading"
|
||||||
:options="roleOptions"
|
:options="postOptions"
|
||||||
v-bind="attrs"
|
v-bind="attrs"
|
||||||
placeholder="请选择岗位"
|
placeholder="请选择岗位"
|
||||||
/>
|
/>
|
||||||
|
@ -16,6 +16,12 @@ export function useDownload() {
|
|||||||
const isHttpProxy = import.meta.env.DEV && import.meta.env.VITE_HTTP_PROXY === 'Y';
|
const isHttpProxy = import.meta.env.DEV && import.meta.env.VITE_HTTP_PROXY === 'Y';
|
||||||
const { baseURL } = getServiceBaseURL(import.meta.env, isHttpProxy);
|
const { baseURL } = getServiceBaseURL(import.meta.env, isHttpProxy);
|
||||||
|
|
||||||
|
const isHttps = () => {
|
||||||
|
const protocol = document.location.protocol;
|
||||||
|
const hostname = document.location.hostname;
|
||||||
|
return protocol === 'https' || hostname === 'localhost' || hostname === '127.0.0.1';
|
||||||
|
};
|
||||||
|
|
||||||
/** 获取通用请求头 */
|
/** 获取通用请求头 */
|
||||||
const getCommonHeaders = (contentType = 'application/octet-stream') => ({
|
const getCommonHeaders = (contentType = 'application/octet-stream') => ({
|
||||||
Authorization: `Bearer ${localStg.get('token')}`,
|
Authorization: `Bearer ${localStg.get('token')}`,
|
||||||
@ -109,9 +115,10 @@ export function useDownload() {
|
|||||||
|
|
||||||
await handleResponse(response);
|
await handleResponse(response);
|
||||||
|
|
||||||
const finalFilename = filename || response.headers.get('Download-Filename') || `download-${timestamp}`;
|
const rawHeader = response.headers.get('Download-Filename');
|
||||||
|
const finalFilename = filename || (rawHeader ? decodeURIComponent(rawHeader) : null) || `download-${timestamp}`;
|
||||||
|
|
||||||
if (response.body) {
|
if (response.body && isHttps()) {
|
||||||
const contentLength = Number(response.headers.get('Content-Length'));
|
const contentLength = Number(response.headers.get('Content-Length'));
|
||||||
await downloadByStream(response.body, finalFilename, contentLength);
|
await downloadByStream(response.body, finalFilename, contentLength);
|
||||||
return;
|
return;
|
||||||
|
@ -77,7 +77,7 @@ export const request = createFlatRequest<App.Service.Response, RequestInstanceSt
|
|||||||
// when the backend response code is in `modalLogoutCodes`, it means the user will be logged out by displaying a modal
|
// when the backend response code is in `modalLogoutCodes`, it means the user will be logged out by displaying a modal
|
||||||
const modalLogoutCodes = import.meta.env.VITE_SERVICE_MODAL_LOGOUT_CODES?.split(',') || [];
|
const modalLogoutCodes = import.meta.env.VITE_SERVICE_MODAL_LOGOUT_CODES?.split(',') || [];
|
||||||
if (modalLogoutCodes.includes(responseCode) && isLogin) {
|
if (modalLogoutCodes.includes(responseCode) && isLogin) {
|
||||||
const isExist = request.state.errMsgStack && request.state.errMsgStack.includes(response.data.msg);
|
const isExist = request.state.errMsgStack?.includes(response.data.msg);
|
||||||
if (isExist) {
|
if (isExist) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
@ -85,23 +85,24 @@ export const request = createFlatRequest<App.Service.Response, RequestInstanceSt
|
|||||||
logoutAndCleanup();
|
logoutAndCleanup();
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
request.state.errMsgStack = [...(request.state.errMsgStack || []), response.data.msg];
|
request.state.errMsgStack = [...(request.state.errMsgStack || []), response.data.msg];
|
||||||
// prevent the user from refreshing the page
|
|
||||||
window.addEventListener('beforeunload', handleLogout);
|
|
||||||
|
|
||||||
window.$dialog?.warning({
|
window.$dialog?.warning({
|
||||||
title: '系统提示',
|
title: '系统提示',
|
||||||
content: '登录状态已过期,您可以继续留在该页面,或者重新登录',
|
content: '登录状态已过期,请重新登录',
|
||||||
positiveText: '重新登录',
|
positiveText: '重新登录',
|
||||||
negativeText: '取消',
|
|
||||||
maskClosable: false,
|
maskClosable: false,
|
||||||
closeOnEsc: false,
|
closeOnEsc: false,
|
||||||
|
onAfterEnter() {
|
||||||
|
// prevent the user from refreshing the page
|
||||||
|
window.addEventListener('beforeunload', handleLogout);
|
||||||
|
},
|
||||||
onPositiveClick() {
|
onPositiveClick() {
|
||||||
logoutAndCleanup();
|
logoutAndCleanup();
|
||||||
},
|
},
|
||||||
onClose() {
|
onClose() {
|
||||||
window.removeEventListener('beforeunload', handleLogout);
|
logoutAndCleanup();
|
||||||
request.state.errMsgStack = request.state.errMsgStack.filter(msg => msg !== response.data.msg);
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
request.cancelAllRequest();
|
request.cancelAllRequest();
|
||||||
|
@ -67,6 +67,8 @@ export const useRouteStore = defineStore(SetupStoreId.Route, () => {
|
|||||||
if (authRouteMode.value === 'dynamic') {
|
if (authRouteMode.value === 'dynamic') {
|
||||||
if (route.path === '/' && route.children?.length) {
|
if (route.path === '/' && route.children?.length) {
|
||||||
const child = route.children[0];
|
const child = route.children[0];
|
||||||
|
// @ts-expect-error no hidden field
|
||||||
|
child.hidden = route.hidden;
|
||||||
parseRouter(child);
|
parseRouter(child);
|
||||||
child.name = Math.random().toString(36).slice(2, 12);
|
child.name = Math.random().toString(36).slice(2, 12);
|
||||||
Object.assign(route, child);
|
Object.assign(route, child);
|
||||||
@ -123,7 +125,6 @@ export const useRouteStore = defineStore(SetupStoreId.Route, () => {
|
|||||||
} else if (!isNotNull(route.meta.icon)) {
|
} else if (!isNotNull(route.meta.icon)) {
|
||||||
route.meta.icon = defaultIcon;
|
route.meta.icon = defaultIcon;
|
||||||
}
|
}
|
||||||
|
|
||||||
// @ts-expect-error no hidden field
|
// @ts-expect-error no hidden field
|
||||||
route.meta.hideInMenu = route.hidden;
|
route.meta.hideInMenu = route.hidden;
|
||||||
if (route.meta.hideInMenu && parent) {
|
if (route.meta.hideInMenu && parent) {
|
||||||
|
@ -13,6 +13,13 @@
|
|||||||
border-color: var(--un-default-border-color, #e5e7eb); /* 2 */
|
border-color: var(--un-default-border-color, #e5e7eb); /* 2 */
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* [Naive UI] Fix the icon size in the image preview toolbar
|
||||||
|
*/
|
||||||
|
.n-image-preview-toolbar .n-base-icon {
|
||||||
|
box-sizing: unset !important;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
1. Use a consistent sensible line-height in all browsers.
|
1. Use a consistent sensible line-height in all browsers.
|
||||||
2. Prevent adjustments of font size after orientation changes in iOS.
|
2. Prevent adjustments of font size after orientation changes in iOS.
|
||||||
|
2
src/typings/api/system.api.d.ts
vendored
@ -163,6 +163,8 @@ declare namespace Api {
|
|||||||
postIds: string[];
|
postIds: string[];
|
||||||
/** user role ids */
|
/** user role ids */
|
||||||
roleIds: string[];
|
roleIds: string[];
|
||||||
|
/** roles */
|
||||||
|
roles: Role[];
|
||||||
};
|
};
|
||||||
|
|
||||||
/** user list */
|
/** user list */
|
||||||
|
@ -53,12 +53,14 @@ async function handleFetchTenantList() {
|
|||||||
const { data, error } = await fetchTenantList();
|
const { data, error } = await fetchTenantList();
|
||||||
if (error) return;
|
if (error) return;
|
||||||
tenantEnabled.value = data.tenantEnabled;
|
tenantEnabled.value = data.tenantEnabled;
|
||||||
tenantOption.value = data.voList.map(tenant => {
|
if (data.tenantEnabled) {
|
||||||
return {
|
tenantOption.value = data.voList.map(tenant => {
|
||||||
label: tenant.companyName,
|
return {
|
||||||
value: tenant.tenantId
|
label: tenant.companyName,
|
||||||
};
|
value: tenant.tenantId
|
||||||
});
|
};
|
||||||
|
});
|
||||||
|
}
|
||||||
endTenantLoading();
|
endTenantLoading();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -21,9 +21,12 @@ const dateRangeCreateTime = ref<[string, string] | null>(null);
|
|||||||
const model = defineModel<Api.System.ConfigSearchParams>('model', { required: true });
|
const model = defineModel<Api.System.ConfigSearchParams>('model', { required: true });
|
||||||
|
|
||||||
function onDateRangeCreateTimeUpdate(value: [string, string] | null) {
|
function onDateRangeCreateTimeUpdate(value: [string, string] | null) {
|
||||||
if (value?.length) {
|
const params = model.value.params!;
|
||||||
model.value.params!.beginTime = value[0];
|
if (value && value.length === 2) {
|
||||||
model.value.params!.endTime = value[1];
|
[params.beginTime, params.endTime] = value;
|
||||||
|
} else {
|
||||||
|
params.beginTime = undefined;
|
||||||
|
params.endTime = undefined;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -113,6 +113,24 @@ function renderLabel({ option }: { option: TreeOption }) {
|
|||||||
if (label?.startsWith('route.') || label?.startsWith('menu.')) {
|
if (label?.startsWith('route.') || label?.startsWith('menu.')) {
|
||||||
label = $t(label as App.I18n.I18nKey);
|
label = $t(label as App.I18n.I18nKey);
|
||||||
}
|
}
|
||||||
|
// 禁用的菜单显示红色
|
||||||
|
if (option.status === '1') {
|
||||||
|
return (
|
||||||
|
<div class="flex items-center gap-4px text-error-200">
|
||||||
|
{label}
|
||||||
|
<SvgIcon icon="ri:prohibited-line" class="text-16px" />
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
// 隐藏的菜单显示灰色
|
||||||
|
if (option.visible === '1') {
|
||||||
|
return (
|
||||||
|
<div class="flex items-center gap-4px text-gray-400">
|
||||||
|
{label}
|
||||||
|
<SvgIcon icon="codex:hidden" class="text-21px" />
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
return <div>{label}</div>;
|
return <div>{label}</div>;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -21,9 +21,12 @@ const dateRangeCreateTime = ref<[string, string] | null>(null);
|
|||||||
const model = defineModel<Api.System.OssSearchParams>('model', { required: true });
|
const model = defineModel<Api.System.OssSearchParams>('model', { required: true });
|
||||||
|
|
||||||
function onDateRangeCreateTimeUpdate(value: [string, string] | null) {
|
function onDateRangeCreateTimeUpdate(value: [string, string] | null) {
|
||||||
if (value?.length) {
|
const params = model.value.params!;
|
||||||
model.value.params!.beginCreateTime = value[0];
|
if (value && value.length === 2) {
|
||||||
model.value.params!.endCreateTime = value[1];
|
[params.beginTime, params.endTime] = value;
|
||||||
|
} else {
|
||||||
|
params.beginTime = undefined;
|
||||||
|
params.endTime = undefined;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -168,9 +168,12 @@ const dateRangeCreateTime = ref<[string, string] | null>(null);
|
|||||||
const datePickerRef = ref<InstanceType<typeof NDatePicker>>();
|
const datePickerRef = ref<InstanceType<typeof NDatePicker>>();
|
||||||
|
|
||||||
function onDateRangeCreateTimeUpdate(value: [string, string] | null) {
|
function onDateRangeCreateTimeUpdate(value: [string, string] | null) {
|
||||||
if (value?.length) {
|
const params = searchParams.params!;
|
||||||
searchParams.params!.beginTime = value[0];
|
if (value && value.length === 2) {
|
||||||
searchParams.params!.endTime = value[1];
|
[params.beginTime, params.endTime] = value;
|
||||||
|
} else {
|
||||||
|
params.beginTime = undefined;
|
||||||
|
params.endTime = undefined;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -24,9 +24,12 @@ const model = defineModel<Api.System.RoleSearchParams>('model', { required: true
|
|||||||
const { options: sysNormalDisableOptions } = useDict('sys_normal_disable', false);
|
const { options: sysNormalDisableOptions } = useDict('sys_normal_disable', false);
|
||||||
|
|
||||||
function onDateRangeCreateTimeUpdate(value: [string, string] | null) {
|
function onDateRangeCreateTimeUpdate(value: [string, string] | null) {
|
||||||
if (value?.length) {
|
const params = model.value.params!;
|
||||||
model.value.params!.beginTime = `${value[0]} 00:00:00`;
|
if (value && value.length === 2) {
|
||||||
model.value.params!.endTime = `${value[1]} 23:59:59`;
|
[params.beginTime, params.endTime] = value;
|
||||||
|
} else {
|
||||||
|
params.beginTime = undefined;
|
||||||
|
params.endTime = undefined;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -91,6 +91,7 @@ function handleDownloadTemplate() {
|
|||||||
|
|
||||||
watch(visible, () => {
|
watch(visible, () => {
|
||||||
if (visible.value) {
|
if (visible.value) {
|
||||||
|
data.value.updateSupport = false;
|
||||||
fileList.value = [];
|
fileList.value = [];
|
||||||
success.value = false;
|
success.value = false;
|
||||||
message.value = '';
|
message.value = '';
|
||||||
@ -140,7 +141,7 @@ watch(visible, () => {
|
|||||||
</NUploadDragger>
|
</NUploadDragger>
|
||||||
</NUpload>
|
</NUpload>
|
||||||
<div class="flex-center">
|
<div class="flex-center">
|
||||||
<NCheckbox v-model="data.updateSupport">{{ $t('common.updateExisting') }}</NCheckbox>
|
<NCheckbox v-model:checked="data.updateSupport">{{ $t('common.updateExisting') }}</NCheckbox>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<NAlert v-if="message" :title="$t('common.importResult')" :type="success ? 'success' : 'error'" :bordered="false">
|
<NAlert v-if="message" :title="$t('common.importResult')" :type="success ? 'success' : 'error'" :bordered="false">
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { computed, reactive, watch } from 'vue';
|
import { computed, reactive, ref, watch } from 'vue';
|
||||||
import { useLoading } from '@sa/hooks';
|
import { useLoading } from '@sa/hooks';
|
||||||
import { fetchCreateUser, fetchGetUserInfo, fetchUpdateUser } from '@/service/api/system';
|
import { fetchCreateUser, fetchGetUserInfo, fetchUpdateUser } from '@/service/api/system';
|
||||||
import { useFormRules, useNaiveForm } from '@/hooks/common/form';
|
import { useFormRules, useNaiveForm } from '@/hooks/common/form';
|
||||||
@ -49,6 +49,8 @@ type Model = Api.System.UserOperateParams;
|
|||||||
|
|
||||||
const model: Model = reactive(createDefaultModel());
|
const model: Model = reactive(createDefaultModel());
|
||||||
|
|
||||||
|
const roleOptions = ref<CommonType.Option<CommonType.IdType>[]>([]);
|
||||||
|
|
||||||
function createDefaultModel(): Model {
|
function createDefaultModel(): Model {
|
||||||
return {
|
return {
|
||||||
deptId: null,
|
deptId: null,
|
||||||
@ -76,18 +78,23 @@ const rules: Record<RuleKey, App.Global.FormRule[]> = {
|
|||||||
roleIds: [{ ...createRequiredRule('请选择角色'), type: 'array' }]
|
roleIds: [{ ...createRequiredRule('请选择角色'), type: 'array' }]
|
||||||
};
|
};
|
||||||
|
|
||||||
async function getUserInfo() {
|
async function getUserInfo(id: CommonType.IdType = '') {
|
||||||
startLoading();
|
startLoading();
|
||||||
const { error, data } = await fetchGetUserInfo(props.rowData?.userId);
|
const { error, data } = await fetchGetUserInfo(id);
|
||||||
if (!error) {
|
if (!error) {
|
||||||
model.roleIds = data.roleIds;
|
model.roleIds = data.roleIds;
|
||||||
model.postIds = data.postIds;
|
model.postIds = data.postIds;
|
||||||
|
roleOptions.value = data.roles.map(role => ({
|
||||||
|
label: role.roleName,
|
||||||
|
value: role.roleId
|
||||||
|
}));
|
||||||
}
|
}
|
||||||
endLoading();
|
endLoading();
|
||||||
}
|
}
|
||||||
|
|
||||||
function handleUpdateModelWhenEdit() {
|
function handleUpdateModelWhenEdit() {
|
||||||
if (props.operateType === 'add') {
|
if (props.operateType === 'add') {
|
||||||
|
getUserInfo();
|
||||||
Object.assign(model, createDefaultModel());
|
Object.assign(model, createDefaultModel());
|
||||||
model.deptId = props.deptId;
|
model.deptId = props.deptId;
|
||||||
return;
|
return;
|
||||||
@ -97,7 +104,7 @@ function handleUpdateModelWhenEdit() {
|
|||||||
startDeptLoading();
|
startDeptLoading();
|
||||||
Object.assign(model, props.rowData);
|
Object.assign(model, props.rowData);
|
||||||
model.password = '';
|
model.password = '';
|
||||||
getUserInfo();
|
getUserInfo(props.rowData.userId);
|
||||||
endDeptLoading();
|
endDeptLoading();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -209,7 +216,14 @@ watch(visible, () => {
|
|||||||
<PostSelect v-model:value="model.postIds" :dept-id="model.deptId" multiple clearable />
|
<PostSelect v-model:value="model.postIds" :dept-id="model.deptId" multiple clearable />
|
||||||
</NFormItem>
|
</NFormItem>
|
||||||
<NFormItem :label="$t('page.system.user.roleIds')" path="roleIds">
|
<NFormItem :label="$t('page.system.user.roleIds')" path="roleIds">
|
||||||
<RoleSelect v-model:value="model.roleIds" multiple clearable />
|
<NSelect
|
||||||
|
v-model:value="model.roleIds"
|
||||||
|
:loading="loading"
|
||||||
|
:options="roleOptions"
|
||||||
|
multiple
|
||||||
|
clearable
|
||||||
|
placeholder="请选择角色"
|
||||||
|
/>
|
||||||
</NFormItem>
|
</NFormItem>
|
||||||
<NFormItem :label="$t('page.system.user.status')" path="status">
|
<NFormItem :label="$t('page.system.user.status')" path="status">
|
||||||
<DictRadio v-model:value="model.status" dict-code="sys_normal_disable" />
|
<DictRadio v-model:value="model.status" dict-code="sys_normal_disable" />
|
||||||
|
@ -24,9 +24,12 @@ const datePickerRef = ref<InstanceType<typeof NDatePicker>>();
|
|||||||
const model = defineModel<Api.System.UserSearchParams>('model', { required: true });
|
const model = defineModel<Api.System.UserSearchParams>('model', { required: true });
|
||||||
|
|
||||||
function onDateRangeCreateTimeUpdate(value: [string, string] | null) {
|
function onDateRangeCreateTimeUpdate(value: [string, string] | null) {
|
||||||
if (value?.length) {
|
const params = model.value.params!;
|
||||||
model.value.params!.beginTime = value[0];
|
if (value && value.length === 2) {
|
||||||
model.value.params!.endTime = value[1];
|
[params.beginTime, params.endTime] = value;
|
||||||
|
} else {
|
||||||
|
params.beginTime = undefined;
|
||||||
|
params.endTime = undefined;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|