Files
Yi.Admin/Yi.Vben5.Vue3/apps/web-antd/src/views/system/user/user-drawer.vue
wcg f5da80de9e refactor(system): 移除岗位关联部门功能并优化岗位加载逻辑
- 删除岗位表单中的部门选择组件及相关联动逻辑
- 修改岗位接口调用为无参数版本,加载全部岗位数据
- 优化用户手机号字段验证规则,支持字符串和数字
- 用户界面中岗位选择不再依赖部门选择,初始化时加载全部岗位
- 移除岗位抽屉中部门树获取及展示逻辑,简化表单结构
- 调整岗位数据提交时移除部门字段,确保数据一致性
- 修正岗位选择组件占位符及禁用状态逻辑,提升用户体验
2026-02-11 16:43:08 +08:00

312 lines
8.2 KiB
Vue
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
<script setup lang="ts">
import type { Role } from '#/api/system/user/model';
import { computed, h, onMounted, ref } from 'vue';
import { useVbenDrawer } from '@vben/common-ui';
import { $t } from '@vben/locales';
import { addFullName, cloneDeep, getPopupContainer } from '@vben/utils';
import { Tag } from 'ant-design-vue';
import { useVbenForm } from '#/adapter/form';
import { configInfoByKey } from '#/api/system/config';
import { postOptionSelect } from '#/api/system/post';
import { roleOptionSelect } from '#/api/system/role';
import {
findUserInfo,
getDeptTree,
userAdd,
userUpdate,
} from '#/api/system/user';
import { defaultFormValueGetter, useBeforeCloseDiff } from '#/utils/popup';
import { authScopeOptions } from '#/views/system/role/data';
import { drawerSchema } from './data';
const emit = defineEmits<{ reload: [] }>();
const isUpdate = ref(false);
const title = computed(() => {
return isUpdate.value ? $t('pages.common.edit') : $t('pages.common.add');
});
const [BasicForm, formApi] = useVbenForm({
commonConfig: {
formItemClass: 'col-span-2',
componentProps: {
class: 'w-full',
},
labelWidth: 80,
},
schema: drawerSchema(),
showDefaultActions: false,
wrapperClass: 'grid-cols-2',
});
/**
* 生成角色的自定义label
* 也可以用option插槽来做
* renderComponentContent: () => ({
option: ({value, label, [disabled, key, title]}) => '',
}),
*/
function genRoleOptionlabel(role: Role) {
const found = authScopeOptions.find((item) => item.value === role.dataScope);
if (!found) {
return role.roleName;
}
return h('div', { class: 'flex items-center gap-[6px]' }, [
h('span', null, role.roleName),
h(Tag, { color: found.color }, () => found.label),
]);
}
/**
* 加载岗位列表(不再强依赖部门选择)
* @param deptId 部门ID可选不传时尝试加载全部
*/
async function setupPostOptions() {
try {
const postListResp = await postOptionSelect();
// 确保返回的是数组
const postList = Array.isArray(postListResp) ? postListResp : [];
const options = postList.map((item) => ({
label: item.postName,
value: item.id,
}));
const placeholder = options.length > 0 ? '请选择岗位' : '暂无岗位数据';
formApi.updateSchema([
{
componentProps: {
disabled: options.length === 0,
options,
placeholder,
},
fieldName: 'postIds',
},
]);
} catch (error) {
console.error('加载岗位信息失败:', error);
formApi.updateSchema([
{
componentProps: {
disabled: true,
options: [],
placeholder: '加载岗位失败',
},
fieldName: 'postIds',
},
]);
}
}
/**
* 初始化部门选择
*/
async function setupDeptSelect() {
try {
// updateSchema
const deptTree = await getDeptTree();
// 确保返回的是数组
const deptList = Array.isArray(deptTree) ? deptTree : [];
// 选中后显示在输入框的值 即父节点 / 子节点
addFullName(deptList, 'deptName', ' / ');
formApi.updateSchema([
{
componentProps: {
class: 'w-full',
fieldNames: {
label: 'deptName',
key: 'id',
value: 'id',
children: 'children',
},
getPopupContainer,
placeholder: '请选择',
showSearch: true,
treeData: deptList,
treeDefaultExpandAll: true,
treeLine: { showLeafIcon: false },
// 筛选的字段
treeNodeFilterProp: 'deptName',
// 选中后显示在输入框的值
treeNodeLabelProp: 'fullName',
},
fieldName: 'deptId',
},
]);
} catch (error) {
console.error('加载部门树失败:', error);
// 加载失败时设置空树
formApi.updateSchema([
{
componentProps: {
placeholder: '加载部门失败',
treeData: [],
},
fieldName: 'deptId',
},
]);
}
}
const defaultPassword = ref('');
onMounted(async () => {
const password = await configInfoByKey('sys.user.initPassword');
if (password) {
defaultPassword.value = password;
}
});
/**
* 新增时候 从参数设置获取默认密码
*/
async function loadDefaultPassword(update: boolean) {
if (!update && defaultPassword.value) {
formApi.setFieldValue('password', defaultPassword.value);
}
}
const { onBeforeClose, markInitialized, resetInitialized } = useBeforeCloseDiff(
{
initializedGetter: defaultFormValueGetter(formApi),
currentGetter: defaultFormValueGetter(formApi),
},
);
const [BasicDrawer, drawerApi] = useVbenDrawer({
onBeforeClose,
onClosed: handleClosed,
onConfirm: handleConfirm,
async onOpenChange(isOpen) {
if (!isOpen) {
// 重置岗位选择(不再提示必须先选部门)
formApi.updateSchema([
{
componentProps: {
disabled: false,
options: [],
placeholder: '请选择岗位',
},
fieldName: 'postIds',
},
]);
return null;
}
drawerApi.drawerLoading(true);
try {
const { id } = drawerApi.getData() as { id?: number | string };
isUpdate.value = !!id;
/** update时 禁用用户名修改 不显示密码框 */
formApi.updateSchema([
{ componentProps: { disabled: isUpdate.value }, fieldName: 'userName' },
{
dependencies: { show: () => !isUpdate.value, triggerFields: ['id'] },
fieldName: 'password',
},
]);
let user: any | null = null;
if (isUpdate.value && id) {
// 编辑模式从用户详情中获取用户信息含岗位、角色ID
user = await findUserInfo(id);
}
// 角色下拉统一使用 roleOptionSelect
const roleListResp = await roleOptionSelect();
const allRoles = Array.isArray(roleListResp) ? (roleListResp as Role[]) : [];
const userRoles = user?.roles ?? [];
const posts = user?.posts ?? [];
const postIds = posts.map((item: any) => item.id);
const roleIds = userRoles.map((item: any) => item.roleId ?? item.id);
const postOptions = posts.map((item: any) => ({
label: item.postName,
value: item.id,
}));
formApi.updateSchema([
{
componentProps: {
// title用于选中后回填到输入框 默认为label
optionLabelProp: 'title',
options: allRoles.map((item: any) => ({
label: genRoleOptionlabel(item),
// title用于选中后回填到输入框 默认为label
title: item.roleName,
value: item.roleId ?? item.id,
})),
},
fieldName: 'roleIds',
},
]);
// 部门选择、初始密码
const promises = [
setupDeptSelect(),
// 岗位列表初始化加载(部门为空也可加载)
setupPostOptions(),
loadDefaultPassword(isUpdate.value),
];
if (user) {
const userData = {
...user,
};
promises.push(
formApi.setValues(userData),
formApi.setFieldValue('postIds', postIds),
formApi.setFieldValue('roleIds', roleIds),
);
} else {
// 新增模式:直接加载岗位(不依赖部门)
await setupPostOptions();
}
// 并行处理
await Promise.all(promises);
await markInitialized();
} catch (error) {
console.error('加载用户信息失败:', error);
} finally {
drawerApi.drawerLoading(false);
}
},
});
async function handleConfirm() {
try {
drawerApi.lock(true);
const { valid } = await formApi.validate();
if (!valid) {
return;
}
const data = cloneDeep(await formApi.getValues());
await (isUpdate.value ? userUpdate(data) : userAdd(data));
resetInitialized();
emit('reload');
drawerApi.close();
} catch (error) {
console.error(error);
} finally {
drawerApi.lock(false);
}
}
async function handleClosed() {
formApi.resetForm();
resetInitialized();
}
</script>
<template>
<BasicDrawer :title="title" class="w-[600px]">
<BasicForm />
</BasicDrawer>
</template>