feat: 🚀 角色权限-首页默认选择-父级半选中传给后台

This commit is contained in:
2025-10-27 15:35:16 +08:00
parent 5f423ba282
commit 3aadf27102
9 changed files with 458 additions and 96 deletions

BIN
dist.zip

Binary file not shown.

View File

@@ -77,7 +77,6 @@
:check-strictly="false" :check-strictly="false"
show-checkbox show-checkbox
style="width: 224px" style="width: 224px"
:collapse-tags="true"
ref="treeSelectRef" ref="treeSelectRef"
@change="handleTreeSelectChange(item)" @change="handleTreeSelectChange(item)"
@remove-tag="handleRemoveTag(item, $event)" @remove-tag="handleRemoveTag(item, $event)"

View File

@@ -46,7 +46,6 @@ const authStore = useAuthStore();
const globalStore = useGlobalStore(); const globalStore = useGlobalStore();
const isCollapse = computed(() => globalStore.isCollapse); const isCollapse = computed(() => globalStore.isCollapse);
const menuList = computed(() => authStore.showMenuListGet); const menuList = computed(() => authStore.showMenuListGet);
console.log(menuList, "=menuList=");
const activeMenu = computed(() => (route.meta.activeMenu ? route.meta.activeMenu : route.path) as string); const activeMenu = computed(() => (route.meta.activeMenu ? route.meta.activeMenu : route.path) as string);
</script> </script>

View File

@@ -40,6 +40,7 @@ watch(
); );
const handleOpenPage = (item: any) => { const handleOpenPage = (item: any) => {
console.log(item, "===========item==========");
$router.push({ $router.push({
path: item.path path: item.path
}); });

View File

@@ -3,11 +3,23 @@
<!-- 封面图 --> <!-- 封面图 -->
<div> <div>
<h5 style="margin: 0; margin-bottom: 16px; font-size: 14px">封面图</h5> <h5 style="margin: 0; margin-bottom: 16px; font-size: 14px">封面图</h5>
<UploadImg v-model:image-url="imgInfoDataStore.cover_image">
<template #tip> <div style="display: flex">
<div style="width: 150px; text-align: center">图片尺寸800x800</div> <div>
</template> <UploadImg v-model:image-url="imgInfoDataStore.cover_image">
</UploadImg> <template #tip>
<div style="width: 150px; text-align: center">图片尺寸800x800</div>
</template>
</UploadImg>
</div>
<!-- <div style="margin-left: 20px">
<UploadImg v-model:image-url="imgInfoDataStore.cover_image">
<template #tip>
<div style="width: 150px; text-align: center">图片尺寸800x800</div>
</template>
</UploadImg>
</div> -->
</div>
</div> </div>
<el-divider /> <el-divider />
<!-- 属性 --> <!-- 属性 -->
@@ -116,7 +128,6 @@ const findAttrById = (id: any) => {
// 在 row 的 attrs 数组中查找对应 attrId 的对象 // 在 row 的 attrs 数组中查找对应 attrId 的对象
const findAttrObjInRow = (row: any, attrId: any) => { const findAttrObjInRow = (row: any, attrId: any) => {
console.log(row.attrs, "=======row===========");
let obj = row.attrs.find((item: any) => item.attr_id === attrId.toString()); let obj = row.attrs.find((item: any) => item.attr_id === attrId.toString());
if (!obj) { if (!obj) {
obj = { attr_id: attrId.toString(), attr_value: "" }; obj = { attr_id: attrId.toString(), attr_value: "" };

View File

@@ -35,27 +35,10 @@ export const FORM_DATA: FormItem[] = [
prop: "treeIds", prop: "treeIds",
prop1: "category_id", prop1: "category_id",
placeholder: "请选择", placeholder: "请选择",
type: "treeSelect", type: "treeSelect", //treeSelect
isArray: true, isArray: true,
label: "产品分类: ", label: "产品分类: ",
options: [ options: []
{
value: "1",
label: "Level one 1",
children: [
{
value: "1-1",
label: "Level two 1-1",
children: [
{
value: "1-1-1",
label: "Level three 1-1-1"
}
]
}
]
}
]
}, },
{ {
prop: "Time", prop: "Time",

View File

@@ -76,7 +76,6 @@ export const initDetailParams = (dataStore: any, data: any, editorRef: any) => {
editorRef?.value?.clearEditor(); // 调用子组件的清空方法 editorRef?.value?.clearEditor(); // 调用子组件的清空方法
} else { } else {
dataStore.detail = htmlDecode(data.detail); //htmlDecode(data.detail); dataStore.detail = htmlDecode(data.detail); //htmlDecode(data.detail);
console.log(data.detail, "=======detail========");
} }
//图片 //图片

View File

@@ -50,6 +50,7 @@
highlight-current highlight-current
:props="defaultProps" :props="defaultProps"
@check-change="handleTreeCheckChange" @check-change="handleTreeCheckChange"
:check-strictly="false"
/> />
</div> </div>
</div> </div>
@@ -77,49 +78,54 @@ import {
getRoleListEditUpApi, getRoleListEditUpApi,
getRoleListSaveApi getRoleListSaveApi
} from "@/api/modules/roleList"; } from "@/api/modules/roleList";
//getMenusListApi //权限列表接口
import { getRoleMenusListApi } from "@/api/modules/webMenusList"; import { getRoleMenusListApi } from "@/api/modules/webMenusList";
//深拷贝方法 //深拷贝方法
import { cloneDeep } from "lodash-es"; import { cloneDeep } from "lodash-es";
//表格和搜索 //表格和搜索
import { RULE_FORM, FORM_DATA, COLUMNS, EDIT_FORM_DATA, EDIT_RULE_FORM, RULES } from "./constant/index"; import { RULE_FORM, FORM_DATA, COLUMNS, EDIT_FORM_DATA, EDIT_RULE_FORM, RULES } from "./constant/index";
// 获取 ProTable 元素,调用其获取刷新数据方法(还能获取到当前查询参数,方便导出携带参数)
// 组件引用
const proTableRef = ref<any>(null); const proTableRef = ref<any>(null);
const formRef: any = ref(null); const formRef: any = ref(null);
const treeRef: any = ref(null); const treeRef: any = ref(null);
const defaultProps = { const defaultProps = {
children: "children", children: "children",
label: "title" label: "title",
disabled: (data: any) => data.id === 7 // 首页节点ID为7禁止选择
}; };
// 数据源 // 数据源
const dataStore = reactive<any>({ const dataStore = reactive<any>({
treeData: [], //权限 treeData: [], //权限树数据
allCheck: false, allCheck: false, //全选框状态
title: "添加角色", //抽屉标题 title: "添加角色", //抽屉标题
columns: COLUMNS, //列表配置项 columns: COLUMNS, //列表配置项
rules: cloneDeep(RULES), //抽屉表单验证 rules: cloneDeep(RULES), //表单验证规则
editRuleForm: cloneDeep(EDIT_RULE_FORM), editRuleForm: cloneDeep(EDIT_RULE_FORM), //编辑表单数据
editFormData: cloneDeep(EDIT_FORM_DATA), //抽屉表单配置项 editFormData: cloneDeep(EDIT_FORM_DATA), //表单配置项
initParam: cloneDeep(RULE_FORM), // 初始化搜索条件|重置搜索条件 initParam: cloneDeep(RULE_FORM), //初始化搜索条件
ruleForm: cloneDeep(RULE_FORM), // 搜索參數 ruleForm: cloneDeep(RULE_FORM), //搜索参数
formData: FORM_DATA, //搜索配置项 formData: FORM_DATA, //搜索配置项
selectedMenuIds: [], // 新增属性,用于保存选中的菜单 id selectedMenuIds: [], //选中的菜单ID含全选和半选
isIndeterminate: false, //全选框样式控制 isIndeterminate: false, //全选框半选样式
visible: false, //抽屉控制 visible: false, //抽屉显示状态
selectRow: {} //当前选择的row selectRow: {} //当前选中行数据
}); });
// 处理全选框状态变化 // 处理全选框状态变化(排除首页节点)
const handleAllCheckChange = (checked: any) => { const handleAllCheckChange = (checked: any) => {
const allNodeKeys = getAllNodeKeys(dataStore.treeData); const allNodeKeys = getAllNodeKeys(dataStore.treeData).filter(id => id !== 7); // 排除首页ID
if (checked) { if (checked) {
treeRef.value.setCheckedKeys(allNodeKeys); // 全选时选中所有节点(包括父节点,但首页已禁用)
treeRef.value.setCheckedKeys([7, ...allNodeKeys]); // 强制包含首页
} else { } else {
treeRef.value.setCheckedKeys([]); // 取消全选时保留首页选中状态
treeRef.value.setCheckedKeys([7]);
} }
}; };
// 获取所有节点的 key // 获取所有节点的ID用于全选计算
const getAllNodeKeys = (data: any[]) => { const getAllNodeKeys = (data: any[]) => {
let keys: number[] = []; let keys: number[] = [];
data.forEach(item => { data.forEach(item => {
@@ -131,139 +137,187 @@ const getAllNodeKeys = (data: any[]) => {
return keys; return keys;
}; };
// 处理树节点选中状态变化 // 处理树节点选中状态变化(确保首页始终选中)
const handleTreeCheckChange = () => { const handleTreeCheckChange = () => {
const allNodeKeys = getAllNodeKeys(dataStore.treeData); // 移除 (data: any) 参数
// 防止手动取消首页选中(虽然已禁用,但做双重保险)
if (!dataStore.selectedMenuIds.some((item: any) => item.menu_id === 7)) {
dataStore.selectedMenuIds.unshift({ menu_id: 7 });
}
const allNodeKeys = getAllNodeKeys(dataStore.treeData).filter(id => id !== 7); // 排除首页
// 获取全选节点ID包括父节点全选和叶子节点选中
const checkedKeys = treeRef.value.getCheckedKeys(); const checkedKeys = treeRef.value.getCheckedKeys();
if (checkedKeys.length === allNodeKeys.length) { // 获取半选节点ID父节点部分子节点选中
const halfCheckedKeys = treeRef.value.getHalfCheckedKeys();
// 合并全选和半选ID并去重强制包含首页
const allSelectedKeys = [...new Set([7, ...checkedKeys, ...halfCheckedKeys])];
// 更新全选框状态(排除首页计算)
const selectedWithoutHome = allSelectedKeys.filter(id => id !== 7);
if (selectedWithoutHome.length === allNodeKeys.length) {
dataStore.allCheck = true; dataStore.allCheck = true;
dataStore.isIndeterminate = false; dataStore.isIndeterminate = false;
} else if (checkedKeys.length === 0) { } else if (selectedWithoutHome.length === 0) {
dataStore.allCheck = false; dataStore.allCheck = false;
dataStore.isIndeterminate = false; dataStore.isIndeterminate = false;
} else { } else {
dataStore.allCheck = false; dataStore.allCheck = false;
dataStore.isIndeterminate = true; dataStore.isIndeterminate = true;
} }
// 更新选中的菜单 id
dataStore.selectedMenuIds = checkedKeys.map((id: any) => ({ menu_id: id })); // 保存所有选中的菜单ID强制包含首页
dataStore.selectedMenuIds = allSelectedKeys.map((id: number) => ({ menu_id: id }));
}; };
//抽屉确认 // 抽屉确认按钮
const handleConfirmClick = () => { const handleConfirmClick = () => {
if (!formRef.value!.ruleFormRef) return; if (!formRef.value?.ruleFormRef) return;
formRef!.value!.ruleFormRef!.validate((valid: any) => { formRef.value.ruleFormRef.validate((valid: boolean) => {
if (valid) { if (valid) {
console.log("submit!");
dataStore.title === "添加角色" ? getRoleListSave() : getRoleListEditUp(); dataStore.title === "添加角色" ? getRoleListSave() : getRoleListEditUp();
} else { } else {
console.log("error submit!"); console.log("表单验证失败");
return false; return false;
} }
}); });
}; };
//重置验证状态
// 重置表单验证状态
const resetFields = () => { const resetFields = () => {
if (!formRef.value!.ruleFormRef) return; if (formRef.value?.ruleFormRef) {
formRef!.value!.ruleFormRef.resetFields(); formRef.value.ruleFormRef.resetFields();
}
}; };
//抽屉重置
// 抽屉重置按钮
const handleResetClick = () => { const handleResetClick = () => {
if (dataStore.title === "添加角色") { if (dataStore.title === "添加角色") {
resetFields(); resetFields();
// 重置树选择状态(保留首页选中)
treeRef.value?.setCheckedKeys([7]);
dataStore.selectedMenuIds = [{ menu_id: 7 }];
} else { } else {
// 编辑时重新获取详情数据
getRoleListDetails(dataStore.editRuleForm.id); getRoleListDetails(dataStore.editRuleForm.id);
} }
}; };
//添加
// 打开添加抽屉
const handleAdd = () => { const handleAdd = () => {
dataStore.title = "添加角色"; dataStore.title = "添加角色";
dataStore.visible = true; dataStore.visible = true;
// 重置表单和树状态(默认选中首页)
dataStore.editRuleForm = cloneDeep(EDIT_RULE_FORM);
resetFields();
nextTick(() => {
treeRef.value?.setCheckedKeys([7]);
});
}; };
//抽屉关闭前的钩子
// 抽屉关闭前处理
const handleBeforeClone = () => { const handleBeforeClone = () => {
dataStore.selectedMenuIds = []; dataStore.selectedMenuIds = [];
treeRef.value.setCheckedKeys([]); treeRef.value?.setCheckedKeys([]);
dataStore.editRuleForm = cloneDeep(EDIT_RULE_FORM); dataStore.editRuleForm = cloneDeep(EDIT_RULE_FORM);
resetFields(); resetFields();
dataStore.visible = false; dataStore.visible = false;
}; };
//按钮点击事件
const handleBtnClick = (type: any, row: any) => { // 编辑/删除按钮点击
const handleBtnClick = (type: string, row: any) => {
dataStore.selectRow = row; dataStore.selectRow = row;
//编辑
if (type === "编辑") { if (type === "编辑") {
dataStore.visible = true; dataStore.visible = true;
dataStore.title = "编辑角色"; dataStore.title = "编辑角色";
getRoleListDetails(row.id); getRoleListDetails(row.id);
return; } else if (type === "删除") {
}
//删除
if (type === "删除") {
getRoleListDel(row.id); getRoleListDel(row.id);
} }
}; };
//删除 // 删除角色
const getRoleListDel = (id: any) => { const getRoleListDel = (id: number) => {
messageBox("你确定要删除?", async () => { messageBox("你确定要删除?", async () => {
const result = await getRoleListDelApi(id); const result = await getRoleListDelApi(id);
if (result?.code === 0) { if (result?.code === 0) {
const { msg } = result; useMsg("success", result.msg);
useMsg("success", msg); proTableRef.value?.getTableList();
proTableRef?.value?.getTableList();
} }
}); });
}; };
//详情
const getRoleListDetails = async (id: any) => { // 获取角色详情并回显(确保首页始终选中)
const getRoleListDetails = async (id: number) => {
const result = await getRoleListDetailsApi(id); const result = await getRoleListDetailsApi(id);
if (result?.code === 0) { if (result?.code === 0) {
dataStore.editRuleForm = result?.data; dataStore.editRuleForm = result.data;
const menuIds = dataStore.editRuleForm.authorities.map((item: any) => item.menu_id); // 提取后台返回的权限ID列表强制包含首页
// 设置树组件的默认选中节点 let savedMenuIds = dataStore.editRuleForm.authorities.map((item: any) => item.menu_id);
savedMenuIds = [...new Set([7, ...savedMenuIds])]; // 确保首页ID存在
nextTick(() => { nextTick(() => {
if (treeRef.value) { if (treeRef.value) {
treeRef.value.setCheckedKeys(menuIds); // 筛选出所有叶子节点ID用于正确触发父节点半选状态
const leafIds: number[] = [];
const findLeafNodes = (nodes: any[]) => {
nodes.forEach(node => {
if (!node.children || node.children.length === 0) {
// 叶子节点且在保存的ID中才选中
if (savedMenuIds.includes(node.id)) {
leafIds.push(node.id);
}
} else {
findLeafNodes(node.children);
}
});
};
findLeafNodes(dataStore.treeData);
// 设置叶子节点选中状态,强制包含首页
treeRef.value.setCheckedKeys([7, ...leafIds]);
// 触发状态更新
handleTreeCheckChange();
} }
}); });
} }
}; };
//保存
// 保存新角色
const getRoleListSave = async () => { const getRoleListSave = async () => {
const result = await getRoleListSaveApi({ const result = await getRoleListSaveApi({
...dataStore.editRuleForm, ...dataStore.editRuleForm,
menu_permission: JSON.stringify(dataStore.selectedMenuIds) menu_permission: JSON.stringify(dataStore.selectedMenuIds)
}); });
if (result?.code === 0) { if (result?.code === 0) {
const { msg } = result; useMsg("success", result.msg);
useMsg("success", msg);
dataStore.visible = false; dataStore.visible = false;
dataStore.editRuleForm = cloneDeep(EDIT_RULE_FORM); proTableRef.value?.getTableList();
proTableRef?.value?.getTableList();
} }
}; };
//更新
// 更新角色信息
const getRoleListEditUp = async () => { const getRoleListEditUp = async () => {
const result = await getRoleListEditUpApi({ const result = await getRoleListEditUpApi({
...dataStore.editRuleForm, ...dataStore.editRuleForm,
menu_permission: JSON.stringify(dataStore.selectedMenuIds) menu_permission: JSON.stringify(dataStore.selectedMenuIds)
}); });
if (result?.code === 0) { if (result?.code === 0) {
const { msg } = result; useMsg("success", result.msg);
useMsg("success", msg);
dataStore.visible = false; dataStore.visible = false;
dataStore.editRuleForm = cloneDeep(EDIT_RULE_FORM); proTableRef.value?.getTableList();
proTableRef?.value?.getTableList();
} }
}; };
//权限管理
// 获取权限树数据
const getMenusList = async () => { const getMenusList = async () => {
const result = await getRoleMenusListApi(); const result = await getRoleMenusListApi();
if (result?.code === 0) { if (result?.code === 0) {
dataStore.treeData = result?.data; dataStore.treeData = result.data;
} }
}; };
// 初始化加载权限树
getMenusList(); getMenusList();
</script> </script>

View File

@@ -0,0 +1,316 @@
<template>
<div class="table-box">
<div style="padding-bottom: 16px">
<el-button type="primary" @click="handleAdd"> 添加 </el-button>
</div>
<ProTable
ref="proTableRef"
:formData="dataStore.formData"
:columns="dataStore.columns"
:request-api="getRoleListApi"
:init-param="dataStore.initParam"
>
<template #operation="scope">
<el-button size="small" type="primary" @click="handleBtnClick('编辑', scope.row)">编辑</el-button>
<el-button size="small" type="danger" @click="handleBtnClick('删除', scope.row)">删除</el-button>
</template>
</ProTable>
<el-drawer
v-model="dataStore.visible"
:show-close="true"
:size="600"
:close-on-click-modal="false"
:close-on-press-escape="false"
:before-close="handleBeforeClone"
destroy-on-close
>
<template #header="{ titleId, titleClass }">
<h4 :id="titleId" :class="titleClass">{{ dataStore.title }}</h4>
</template>
<div>
<rulesForm
:ruleForm="dataStore.editRuleForm"
:formData="dataStore.editFormData"
:rules="dataStore.rules"
:indeterminate="dataStore.isIndeterminate"
ref="formRef"
>
</rulesForm>
<div style="margin-left: 65px; font-size: 14px; color: #606266">
<div style="display: flex; align-items: center; margin-bottom: 10px">
<span style="margin-right: 10px">权限分配:</span>
<el-checkbox v-model="dataStore.allCheck" @change="handleAllCheckChange" label="全选" size="large" />
<!-- <el-checkbox v-model="dataStore.allCheck" @change="handleAllCheckChange" label="全选" size="large" /> -->
</div>
<el-tree
ref="treeRef"
style="max-width: 600px"
:data="dataStore.treeData"
show-checkbox
node-key="id"
highlight-current
:props="defaultProps"
@check-change="handleTreeCheckChange"
/>
</div>
</div>
<template #footer>
<div style="flex: auto">
<el-button @click="handleResetClick">重置</el-button>
<el-button type="primary" @click="handleConfirmClick">确认</el-button>
</div>
</template>
</el-drawer>
</div>
</template>
<script setup lang="ts" name="roleListIndex">
import ProTable from "@/components/ProTable/index.vue";
import rulesForm from "@/components/rulesForm/index.vue";
import { messageBox } from "@/utils/messageBox";
import { nextTick } from "vue";
import { useMsg } from "@/hooks/useMsg";
//列表接口
import {
getRoleListApi,
getRoleListDetailsApi,
getRoleListDelApi,
getRoleListEditUpApi,
getRoleListSaveApi
} from "@/api/modules/roleList";
//权限列表接口
import { getRoleMenusListApi } from "@/api/modules/webMenusList";
//深拷贝方法
import { cloneDeep } from "lodash-es";
//表格和搜索条件
import { RULE_FORM, FORM_DATA, COLUMNS, EDIT_FORM_DATA, EDIT_RULE_FORM, RULES } from "./constant/index";
// 组件引用
const proTableRef = ref<any>(null);
const formRef: any = ref(null);
const treeRef: any = ref(null);
const defaultProps = {
children: "children",
label: "title"
};
// 数据源
const dataStore = reactive<any>({
treeData: [], //权限树数据
allCheck: false, //全选框状态
title: "添加角色", //抽屉标题
columns: COLUMNS, //列表配置项
rules: cloneDeep(RULES), //表单验证规则
editRuleForm: cloneDeep(EDIT_RULE_FORM), //编辑表单数据
editFormData: cloneDeep(EDIT_FORM_DATA), //表单配置项
initParam: cloneDeep(RULE_FORM), //初始化搜索条件
ruleForm: cloneDeep(RULE_FORM), //搜索参数
formData: FORM_DATA, //搜索配置项
selectedMenuIds: [], //选中的菜单ID含全选和半选
isIndeterminate: false, //全选框半选样式
visible: false, //抽屉显示状态
selectRow: {} //当前选中行数据
});
// 处理全选框状态变化
const handleAllCheckChange = (checked: any) => {
const allNodeKeys = getAllNodeKeys(dataStore.treeData);
if (checked) {
// 全选时选中所有节点(包括父节点)
treeRef.value.setCheckedKeys(allNodeKeys);
} else {
// 取消全选时清空所有选中
treeRef.value.setCheckedKeys([]);
}
};
// 获取所有节点的ID
const getAllNodeKeys = (data: any[]) => {
let keys: number[] = [];
data.forEach(item => {
keys.push(item.id);
if (item.children && item.children.length > 0) {
keys = keys.concat(getAllNodeKeys(item.children));
}
});
return keys;
};
// 处理树节点选中状态变化
const handleTreeCheckChange = () => {
const allNodeKeys = getAllNodeKeys(dataStore.treeData);
// 获取全选节点ID包括父节点全选和叶子节点选中
const checkedKeys = treeRef.value.getCheckedKeys();
// 获取半选节点ID父节点部分子节点选中
const halfCheckedKeys = treeRef.value.getHalfCheckedKeys();
// 合并全选和半选ID并去重
const allSelectedKeys = [...new Set([...checkedKeys, ...halfCheckedKeys])];
// 更新全选框状态
if (allSelectedKeys.length === allNodeKeys.length) {
dataStore.allCheck = true;
dataStore.isIndeterminate = false;
} else if (allSelectedKeys.length === 0) {
dataStore.allCheck = false;
dataStore.isIndeterminate = false;
} else {
dataStore.allCheck = false;
dataStore.isIndeterminate = true;
}
// 保存所有选中的菜单ID含半选父节点
dataStore.selectedMenuIds = allSelectedKeys.map((id: number) => ({ menu_id: id }));
};
// 抽屉确认按钮
const handleConfirmClick = () => {
if (!formRef.value?.ruleFormRef) return;
formRef.value.ruleFormRef.validate((valid: boolean) => {
if (valid) {
dataStore.title === "添加角色" ? getRoleListSave() : getRoleListEditUp();
} else {
console.log("表单验证失败");
return false;
}
});
};
// 重置表单验证状态
const resetFields = () => {
if (formRef.value?.ruleFormRef) {
formRef.value.ruleFormRef.resetFields();
}
};
// 抽屉重置按钮
const handleResetClick = () => {
if (dataStore.title === "添加角色") {
resetFields();
// 重置树选择状态
treeRef.value?.setCheckedKeys([]);
dataStore.selectedMenuIds = [];
} else {
// 编辑时重新获取详情数据
getRoleListDetails(dataStore.editRuleForm.id);
}
};
// 打开添加抽屉
const handleAdd = () => {
dataStore.title = "添加角色";
dataStore.visible = true;
// 重置表单和树状态
dataStore.editRuleForm = cloneDeep(EDIT_RULE_FORM);
resetFields();
nextTick(() => {
treeRef.value?.setCheckedKeys([]);
});
};
// 抽屉关闭前处理
const handleBeforeClone = () => {
dataStore.selectedMenuIds = [];
treeRef.value?.setCheckedKeys([]);
dataStore.editRuleForm = cloneDeep(EDIT_RULE_FORM);
resetFields();
dataStore.visible = false;
};
// 编辑/删除按钮点击
const handleBtnClick = (type: string, row: any) => {
dataStore.selectRow = row;
if (type === "编辑") {
dataStore.visible = true;
dataStore.title = "编辑角色";
getRoleListDetails(row.id);
} else if (type === "删除") {
getRoleListDel(row.id);
}
};
// 删除角色
const getRoleListDel = (id: number) => {
messageBox("你确定要删除?", async () => {
const result = await getRoleListDelApi(id);
if (result?.code === 0) {
useMsg("success", result.msg);
proTableRef.value?.getTableList();
}
});
};
// 获取角色详情并回显
const getRoleListDetails = async (id: number) => {
const result = await getRoleListDetailsApi(id);
if (result?.code === 0) {
dataStore.editRuleForm = result.data;
// 提取后台返回的权限ID列表
const savedMenuIds = dataStore.editRuleForm.authorities.map((item: any) => item.menu_id);
nextTick(() => {
if (treeRef.value) {
// 筛选出所有叶子节点ID用于正确触发父节点半选状态
const leafIds: number[] = [];
const findLeafNodes = (nodes: any[]) => {
nodes.forEach(node => {
if (!node.children || node.children.length === 0) {
// 叶子节点且在保存的ID中才选中
if (savedMenuIds.includes(node.id)) {
leafIds.push(node.id);
}
} else {
findLeafNodes(node.children);
}
});
};
findLeafNodes(dataStore.treeData);
// 设置叶子节点选中状态,父节点会自动计算半选/全选
treeRef.value.setCheckedKeys(leafIds);
// 触发状态更新
handleTreeCheckChange();
}
});
}
};
// 保存新角色
const getRoleListSave = async () => {
const result = await getRoleListSaveApi({
...dataStore.editRuleForm,
menu_permission: JSON.stringify(dataStore.selectedMenuIds)
});
if (result?.code === 0) {
useMsg("success", result.msg);
dataStore.visible = false;
proTableRef.value?.getTableList();
}
};
// 更新角色信息
const getRoleListEditUp = async () => {
const result = await getRoleListEditUpApi({
...dataStore.editRuleForm,
menu_permission: JSON.stringify(dataStore.selectedMenuIds)
});
if (result?.code === 0) {
useMsg("success", result.msg);
dataStore.visible = false;
proTableRef.value?.getTableList();
}
};
// 获取权限树数据
const getMenusList = async () => {
const result = await getRoleMenusListApi();
if (result?.code === 0) {
dataStore.treeData = result.data;
console.log(result.data, "==============data===============");
}
};
// 初始化加载权限树
getMenusList();
</script>
<style scoped></style>