提交 be545bf7 authored 作者: lidongxu's avatar lidongxu

feat(report): 新增_管理员分配报表权限给个人功能_个人查看被分配报表列表和预览功能

上级 0bb1469d
import request from '@/utils/request'
// 获取被分配的报表列表
export function getReportListAPI(params) {
return request({
url: '/report/jimuReport/query/list',
params
})
}
...@@ -31,6 +31,7 @@ export * from './system/menu' ...@@ -31,6 +31,7 @@ export * from './system/menu'
export * from './system/notice' export * from './system/notice'
export * from './system/operlog' export * from './system/operlog'
export * from './system/post' export * from './system/post'
export * from './system/report'
export * from './system/role' export * from './system/role'
export * from './system/user' export * from './system/user'
export * from './tool/gen' export * from './tool/gen'
...@@ -28,4 +28,5 @@ export function getReportFolderListAPI() { ...@@ -28,4 +28,5 @@ export function getReportFolderListAPI() {
}, },
method: 'GET' method: 'GET'
}) })
} }
\ No newline at end of file
import request from '@/utils/request'
// 管理员查询所有报表接口
export const selReportListAPI = (params) => {
return request({
url: `/report/jimuReport/query/all/list`,
params
})
}
// 授权单个报表给单个/多个用户
export const batchAuthReportAPI = (data) => {
return request({
url: `/report/jimuReportAuth/core/authUsers`,
method: "PUT",
data
})
}
// 报表分类下拉列表接口
export const selReportCategoryAPI = () => {
return request({
url: `/report/jimuReportCategory/query/list`
})
}
// 查询-报表已分配用户列表
export const selReportAuthUserListAPI = (params) => {
return request({
url: `/report/jimuReportAuth/query/list/user`,
params
})
}
// cover some element-ui styles // cover some element-ui styles
body .el-loading-mask{
z-index: 10000 !important;
}
.el-breadcrumb__inner, .el-breadcrumb__inner,
.el-breadcrumb__inner a { .el-breadcrumb__inner a {
font-weight: 400 !important; font-weight: 400 !important;
...@@ -142,11 +138,7 @@ body .el-loading-mask{ ...@@ -142,11 +138,7 @@ body .el-loading-mask{
flex: 1; flex: 1;
} }
/* 全局覆盖Element Plus加载遮罩的背景色 */
.el-loading-mask {
z-index: 2 !important;
// background-color: red !important;
}
.el-pagination { .el-pagination {
font-size: var(--xl-fontsize) !important; font-size: var(--xl-fontsize) !important;
......
<template>
<div class="app-container">
<el-card class="container">
<template #header>
<div class="card-header">
<span>常用报表</span>
<el-button class="custom"
type="primary"
link
@click="visible = true">
自定义
</el-button>
</div>
</template>
<div v-for="item in groups"
class="group_item">
<div>
<p class="group_name">
<el-icon color="blue"
size="18">
<Folder />
</el-icon>
<span>
{{ item.groupName }}
</span>
</p>
<div class="link_wrap">
<div class="link_item"
v-for="obj in item.items">
<el-icon color="green"
size="16">
<Document></Document>
</el-icon>
<el-link :href="baseURL + obj.previewUrl"
target="_blank">{{ obj.reportName }}</el-link>
</div>
</div>
</div>
</div>
</el-card>
<el-dialog title="编辑常用报表"
v-model="visible"
draggable overflow>
<div class="wrap">
<div class="left">
<draggable v-model="reportList"
group="reports"
ghost-class="dragging-item-left"
item-key="id">
<template #item="{ element }">
<div class="link_item">
<el-icon color="green"
size="16">
<Document></Document>
</el-icon>
<el-link :href="baseURL + element.previewUrl"
target="_blank">{{ element.reportName }}</el-link>
</div>
</template>
</draggable>
</div>
<div class="right">
<div class="group-container">
<el-card v-for="group in groups"
:key="group.id"
class="group_card"
:style="{ width: '100%' }">
<template #header>
<div class="card-header">
<span>{{ group.groupName }}</span>
</div>
</template>
<draggable v-model="group.items"
group="reports"
ghost-class="dragging-item"
item-key="id"
@change="groupChange">
<template #item="{ element }">
<div class="link_item">
<el-icon color="green"
size="16">
<Document></Document>
</el-icon>
<el-link :href="baseURL + element.previewUrl"
target="_blank">{{ element.reportName }}</el-link>
</div>
</template>
</draggable>
</el-card>
</div>
</div>
</div>
</el-dialog>
</div>
</template>
<script setup>
import { getReportListAPI, getReportListBySelfAPI, getReportListByGroupAPI, addReportGroupAPI } from '@/api'
import draggable from 'vuedraggable'
const visible = ref(false)
const groups = ref([])
const baseURL = ref(import.meta.env.VITE_APP_REPORT_URL + '/report')
const reportList = ref([])
const getReportList = async () => {
// 积木报表
const { data } = await getReportListAPI({
categoryName: '财务报表'
})
// 其他报表
const { data: otherData } = await getReportListBySelfAPI()
reportList.value = data.concat(otherData)
// 右侧分组列表
const res = await getReportListByGroupAPI({
categoryName: '财务报表'
})
groups.value = res.data
// 左侧报表里要删除右侧里面有的了,用 reportId 来判断
reportList.value = reportList.value.filter(item => {
return !groups.value.some(group => group.items.some(groupItem => groupItem.reportId === item.reportId))
})
}
// 拖入拖出+顺序改变
const groupChange = async (e) => {
groups.value.forEach(o => {
o.id = o.groupId
})
await addReportGroupAPI(groups.value)
}
getReportList()
</script>
<style scoped
lang="scss">
.el-card__header {
.custom {
float: right;
padding: 3px 0;
}
}
.group_item {
.group_name {
display: flex;
align-items: center;
gap: 10px;
font-size: 16px;
font-weight: 800;
}
.link_wrap {
display: flex;
flex-wrap: wrap;
padding: 0 0 0 20px;
.link_item {
padding: 10px 0;
width: 25%;
/* 超出宽度文字省略号 */
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
}
}
}
.link_item {
padding: 5px 0;
display: flex;
align-items: center;
gap: 10px;
}
.group_card {
margin-top: 20px;
}
.wrap {
display: flex;
gap: 20px;
width: 100%;
height: 100%;
.left {
width: 100%;
height: 100%;
overflow-y: scroll;
overflow-x: hidden;
}
.right {
width: 100%;
height: 100%;
overflow-y: scroll;
overflow-x: hidden;
}
}
/* 添加拖拽时的样式 */
.dragging-item-left {
background-color: transparent;
}
.dragging-item {
background-color: #a1ccf2;
}
</style>
\ No newline at end of file
<template>
<div class="app-container">
<div class="container">
<el-form :model="queryParams"
ref="queryRef"
inline
label-width="68px"
v-show="showSearch">
<el-form-item label="报表名称"
prop="name">
<el-input v-model="queryParams.name"
placeholder="搜索报表名称"
clearable
style="width: 240px"
@input="getReportList" />
</el-form-item>
<el-form-item label="报表类目"
prop="categoryName">
<el-select v-model="queryParams.categoryName"
placeholder="选择报表类目"
clearable
filterable
@change="getReportList"
style="width: 240px">
<el-option v-for="item in categoryList"
:label="item.name"
:value="item.name" />
</el-select>
</el-form-item>
<el-form-item>
<el-button icon="Refresh"
@click="resetQuery">重置</el-button>
</el-form-item>
</el-form>
<!-- 表格数据 -->
<el-table v-loading="loading"
:data="reportList">
<el-table-column label="报表名称"
key="reportName"
prop="reportName"
align="left"
sortable>
<template #default="scope">
<div style="display: flex; align-items: center">
<el-icon>
<List color="green" />
</el-icon>
<span style="margin-left: 10px">{{ scope.row.reportName }}</span>
</div>
</template>
</el-table-column>
<el-table-column label="创建人"
key="createBy"
prop="createBy"
align="left"
width="180"
sortable />
<el-table-column label="创建时间"
prop="createTime"
align="left"
width="180"
sortable>
<template #default="scope">
<span>{{ parseTime(scope.row.createTime) }}</span>
</template>
</el-table-column>
<!-- 操作预览 -->
<el-table-column label="操作"
key="operation"
align="center"
width="180">
<template #default="scope">
<el-button link
type="primary"
icon="Reading"
@click="previewReport(scope.row)">预览</el-button>
</template>
</el-table-column>
</el-table>
<!-- <pagination v-show="total > 0"
:total="total"
v-model:page="queryParams.pageNum"
v-model:limit="queryParams.pageSize"
@pagination="getReportList" /> -->
</div>
</div>
</template>
<script setup
name="Role">
import { getReportListAPI, selReportCategoryAPI } from "@/api"
import { getToken } from '@/utils/auth'
import useUserStore from '@/store/modules/user'
const userStore = useUserStore()
const reportBaseURL = import.meta.env.VITE_APP_REPORT_URL
const { proxy } = getCurrentInstance()
/*************** 报表列表部分 ****************/
// 报表搜索
const showSearch = ref(true)
const queryParams = reactive({
name: undefined, // 报表名称
categoryName: undefined, // 报表类目
pageNum: 1,
pageSize: 10
})
// 获取设置报表类目下拉菜单
const categoryList = ref([])
const getCategoryList = () => {
selReportCategoryAPI().then(res => {
categoryList.value = res.data;
})
}
getCategoryList()
// 重置
function resetQuery() {
proxy.resetForm("queryRef")
getReportList()
}
// 获取报表列表
const loading = ref(true)
const reportList = ref([])
// 查询报表列表
function getReportList() {
loading.value = true
getReportListAPI(queryParams).then(response => {
reportList.value = response.data
loading.value = false
})
}
getReportList()
// 点击某行报表
function previewReport(row) {
window.open(`${reportBaseURL}/report/jmreport/view/${row.id}?token=${getToken()}&tenantId=${userStore.$state.userInfo.deptId}`)
}
</script>
<style scoped
lang="scss">
</style>
\ No newline at end of file
...@@ -9,9 +9,8 @@ import iFrame from '@/components/iFrame/index' ...@@ -9,9 +9,8 @@ import iFrame from '@/components/iFrame/index'
// 获取 pinia 里保存的用户部门 id // 获取 pinia 里保存的用户部门 id
import useUserStore from '@/store/modules/user' import useUserStore from '@/store/modules/user'
// 这里写暴露的统一的网关地址 const userStore = useUserStore()
const store = useUserStore() const openUrl = ref(`${import.meta.env.VITE_APP_REPORT_URL}/report/jmreport/list?token=` + getToken() + `&tenantId=${userStore.$state.userInfo.deptId}`)
const openUrl = ref(`${import.meta.env.VITE_APP_REPORT_URL}/report/jmreport/list?token=` + getToken() + `&tenantId=${store.$state.userInfo.deptId}`)
</script> </script>
<style lang="scss" <style lang="scss"
......
...@@ -3,29 +3,29 @@ ...@@ -3,29 +3,29 @@
<div class="container"> <div class="container">
<el-form :model="queryParams" <el-form :model="queryParams"
ref="queryRef" ref="queryRef"
:inline="true" inline
label-width="68px"> label-width="68px"
v-show="showSearch">
<el-form-item label="用户名称" <el-form-item label="报表名称"
prop="nickName"> prop="name">
<el-input v-model="queryParams.nickName" <el-input v-model="queryParams.name"
placeholder="请输入用户名称" placeholder="搜索报表名称"
clearable
style="width: 240px" />
</el-form-item>
<el-form-item label="用户工号"
prop="userName">
<el-input v-model="queryParams.userName"
placeholder="请输入用户工号"
clearable clearable
style="width: 240px" /> style="width: 240px"
@input="getReportList" />
</el-form-item> </el-form-item>
<el-form-item label="手机号码" <el-form-item label="报表类目"
prop="phonenumber"> prop="category">
<el-input v-model="queryParams.phonenumber" <el-select v-model="queryParams.category"
placeholder="请输入手机号码" placeholder="选择报表类目"
clearable clearable
style="width: 240px" /> filterable
@change="getReportList"
style="width: 240px">
<el-option v-for="item in categoryList"
:label="item.name"
:value="item.id" />
</el-select>
</el-form-item> </el-form-item>
<el-form-item> <el-form-item>
<el-button icon="Refresh" <el-button icon="Refresh"
...@@ -33,41 +33,71 @@ ...@@ -33,41 +33,71 @@
</el-form-item> </el-form-item>
</el-form> </el-form>
<el-row :gutter="10"
class="mb8">
<!-- <el-col :span="1.5">
<el-button type="primary"
plain
icon="Plus"
:disabled="reportMultipleBtnDis"
@click="handleAssign">分配用户</el-button>
</el-col> -->
<right-toolbar v-model:showSearch="showSearch"
@queryTable="getReportList"></right-toolbar>
</el-row>
<!-- 表格数据 --> <!-- 表格数据 -->
<el-table v-loading="loading" <el-table v-loading="loading"
:data="userList"> :data="reportList"
<el-table-column type="selection" @selection-change="handleSelectionReport">
width="50" /> <!-- <el-table-column type="selection"
<el-table-column label="用户工号" width="50" /> -->
key="userName" <el-table-column label="报表名称"
prop="userName" key="name"
align="left" /> prop="name"
<el-table-column label="用户昵称" align="left"
key="nickName" sortable>
prop="nickName" <template #default="scope">
align="left" /> <div style="display: flex; align-items: center">
<el-table-column label="手机号码" <el-icon>
key="phonenumber" <List color="green" />
prop="phonenumber" </el-icon>
align="left" /> <span style="margin-left: 10px">{{ scope.row.name }}</span>
</div>
</template>
</el-table-column>
<el-table-column label="报表类目"
key="categoryName"
prop="categoryName"
align="left"
width="180"
sortable />
<el-table-column label="创建人"
key="createBy"
prop="createBy"
align="left"
width="180"
sortable />
<el-table-column label="创建时间" <el-table-column label="创建时间"
prop="createTime" prop="createTime"
align="left"> align="left"
width="180"
sortable>
<template #default="scope"> <template #default="scope">
<span>{{ parseTime(scope.row.createTime) }}</span> <span>{{ parseTime(scope.row.createTime) }}</span>
</template> </template>
</el-table-column> </el-table-column>
<el-table-column label="分配报表" <el-table-column label="操作"
class-name="small-padding fixed-width" class-name="small-padding fixed-width"
align="left"> align="left"
width="100">
<template #default="scope"> <template #default="scope">
<xl-tool-tip content="分配报表" <xl-tool-tip content="分配用户"
placement="top" placement="top">
v-if="scope.row.userId !== 1">
<el-button link <el-button link
type="primary" type="primary"
icon="Reading" icon="Reading"
@click="handleAuthRole(scope.row)"></el-button> @click="handleAssign(scope.row)"></el-button>
</xl-tool-tip> </xl-tool-tip>
</template> </template>
</el-table-column> </el-table-column>
...@@ -77,73 +107,70 @@ ...@@ -77,73 +107,70 @@
:total="total" :total="total"
v-model:page="queryParams.pageNum" v-model:page="queryParams.pageNum"
v-model:limit="queryParams.pageSize" v-model:limit="queryParams.pageSize"
@pagination="getList" /> @pagination="getReportList" />
<!-- 分配用户 --> <!-- 分配用户 -->
<el-dialog title="选择用户" <el-dialog v-model="showSelUser"
v-model="showSelUser"
top="5vh"
append-to-body
draggable draggable
overflow> overflow
<el-form :model="queryParams" modal-class="assign-user-dialog"
ref="queryRef" @closed="closeSelUserDialog">
:inline="true"> <template #header>
<el-form-item label="部门选择" <el-form :model="userParams"
prop="roleName"> inline>
<el-tree-select v-model="queryParams.deptId" <el-form-item label="部门选择"
:data="deptOptions" prop="roleName">
filterable <el-tree-select v-model="userParams.deptId"
highlight-current :data="deptOptions"
node-key="id" filterable
check-strictly clearable
:default-expanded-keys="defaultExpandedKeys" highlight-current
style="width: 240px" /> node-key="id"
</el-form-item> check-strictly
<el-form-item label="用户工号" :default-expanded-keys="defaultExpandedKeys"
prop="userName"> style="width: 200px"
<el-input v-model="queryParams.userName" @change="getUserList" />
placeholder="请输入用户工号" </el-form-item>
clearable <el-form-item label="用户昵称"
style="width: 180px" prop="nickName">
@input="handleQuery" /> <el-input v-model="userParams.nickName"
</el-form-item> placeholder="请输入用户昵称"
<el-form-item label="用户昵称" clearable
prop="nickName"> style="width: 150px"
<el-input v-model="queryParams.nickName" @input="getUserList" />
placeholder="请输入用户昵称" </el-form-item>
clearable <el-form-item label="用户工号"
style="width: 180px" prop="userName">
@input="handleQuery" /> <el-input v-model="userParams.userName"
</el-form-item> placeholder="请输入用户工号"
<el-form-item label="手机号码" clearable
prop="phonenumber"> style="width: 150px"
<el-input v-model="queryParams.phonenumber" @input="getUserList" />
placeholder="请输入手机号码" </el-form-item>
clearable <el-form-item label="手机号码"
style="width: 180px" prop="phonenumber">
@input="handleQuery" /> <el-input v-model="userParams.phonenumber"
</el-form-item> placeholder="请输入手机号码"
<el-form-item> clearable
<el-button icon="Refresh" style="width: 150px"
@click="resetQuery">重置</el-button> @input="getUserList" />
</el-form-item> </el-form-item>
</el-form> </el-form>
</template>
<el-row> <el-row>
<el-table @row-click="clickRow" <el-table ref="refTable"
ref="refTable"
:data="userList" :data="userList"
@selection-change="handleSelectionChange"> :loading="userLoading"
@select="handleSelectChange"
@selectAll="handleSelectAll">
<el-table-column type="selection" <el-table-column type="selection"
width="55"></el-table-column> width="55"></el-table-column>
<el-table-column label="用户工号"
prop="userName"
:show-overflow-tooltip="true" />
<el-table-column label="用户昵称" <el-table-column label="用户昵称"
prop="nickName" prop="nickName"
:show-overflow-tooltip="true" /> :show-overflow-tooltip="true" />
<el-table-column label="邮箱" <el-table-column label="用户工号"
prop="email" prop="userName"
:show-overflow-tooltip="true" /> :show-overflow-tooltip="true" />
<el-table-column label="手机" <el-table-column label="手机"
prop="phonenumber" prop="phonenumber"
...@@ -165,19 +192,20 @@ ...@@ -165,19 +192,20 @@
</template> </template>
</el-table-column> </el-table-column>
</el-table> </el-table>
<div class="pagination-wrap">
<pagination v-show="total > 0"
:total="total"
v-model:page="queryParams.pageNum"
v-model:limit="queryParams.pageSize"
@pagination="getList" />
</div>
</el-row> </el-row>
<template #footer> <template #footer>
<div class="dialog-footer"> <div class="dialog-footer">
<div class="pagination-wrap">
<pagination v-show="userTotal > 0"
:total="userTotal"
v-model:page="userParams.pageNum"
v-model:limit="userParams.pageSize"
@pagination="getUserList" />
</div>
<el-button type="primary" <el-button type="primary"
@click="handleSelectUser">确 定</el-button> @click="handleSelectUser">确 定</el-button>
<el-button @click="visible = false">取 消</el-button> <el-button @click="showSelUser = false">取 消</el-button>
</div> </div>
</template> </template>
</el-dialog> </el-dialog>
...@@ -187,38 +215,90 @@ ...@@ -187,38 +215,90 @@
<script setup <script setup
name="Role"> name="Role">
import { addRole, changeRoleStatus, dataScope, delRole, getRole, listRole, updateRole, deptTreeSelect } from "@/api"; import { selReportListAPI, selReportCategoryAPI, deptTreeSelectList, listUser, batchAuthReportAPI, selReportAuthUserListAPI } from "@/api"
import { deptTreeSelectList, listUser, getReportFolderListAPI, roleMenuTreeselect, treeselect as menuTreeselect } from "@/api";
const { proxy } = getCurrentInstance()
const router = useRouter();
const { proxy } = getCurrentInstance(); /*************** 报表列表部分 ****************/
const { sys_normal_disable } = proxy.useDict("sys_normal_disable"); // 报表搜索
const showSearch = ref(true)
const roleList = ref([]); const queryParams = reactive({
const open = ref(false); name: undefined, // 报表名称
const loading = ref(true); category: undefined, // 报表类目
const showSearch = ref(true); pageNum: 1,
const ids = ref([]); pageSize: 10
const single = ref(true); })
const multiple = ref(true); // 获取设置报表类目下拉菜单
const categoryList = ref([])
const getCategoryList = () => {
selReportCategoryAPI().then(res => {
categoryList.value = res.data;
})
}
getCategoryList()
// 重置
function resetQuery() {
proxy.resetForm("queryRef")
getReportList()
}
const title = ref(""); // 批量分配报表
const dateRange = ref([]); const reportMultipleBtnDis = ref(true) // 是否多选报表
const menuOptions = ref([]); const checkedReportRowList = ref([]) // 选择的报表对象
const menuExpand = ref(false); const handleSelectionReport = (val) => {
const menuNodeAll = ref(false); checkedReportRowList.value = val
const deptExpand = ref(true); reportMultipleBtnDis.value = !(checkedReportRowList.value.length > 0)
const deptNodeAll = ref(false); }
const menuRef = ref(null); // 获取报表列表
const loading = ref(true)
const reportList = ref([])
const total = ref(0)
// 查询报表列表
function getReportList() {
loading.value = true
selReportListAPI(queryParams).then(response => {
reportList.value = response.data.rows
total.value = response.data.total
loading.value = false
})
}
getReportList()
/**************** 分配用户 ****************/
const { sys_normal_disable } = proxy.useDict("sys_normal_disable");
const showSelUser = ref(false)
/** 查询用户列表 */
const userList = ref([])
const userLoading = ref(false)
const userTotal = ref(0)
const userParams = ref({
deptId: undefined, // 部门 ID
pageNum: 1,
pageSize: 10
})
const selUserList = ref([]) // 当前表已经分配给的用户列表
// 开始分配
const handleAssign = async (obj) => {
const res = await selReportAuthUserListAPI({
reportId: obj.id,
pageSize: 100
})
console.log('查询到', res)
selUserList.value = res.data || []
showSelUser.value = true
if (obj.id) {
// 如果 obj 有值则是单个分配
checkedReportRowList.value = [obj]
}
handleRowClick()
}
// 部门查询 // 部门查询
const deptRef = ref(null)
const deptOptions = ref([]); // 部门树结构 const deptOptions = ref([]); // 部门树结构
const defaultExpandedKeys = ref([]) // 默认展开 id const defaultExpandedKeys = ref([]) // 默认展开 id
function getDeptList() { function getDeptList() {
loading.value = true;
deptTreeSelectList().then(response => { deptTreeSelectList().then(response => {
deptOptions.value = response.data; deptOptions.value = response.data;
defaultExpandedKeys.value = [response.data[0].id]; defaultExpandedKeys.value = [response.data[0].id];
...@@ -226,352 +306,94 @@ ...@@ -226,352 +306,94 @@
} }
getDeptList() getDeptList()
/** 查询用户列表 */
const showSelUser = ref(false)
const userList = ref([])
const total = ref(0);
function getUserList() { function getUserList() {
loading.value = true; userLoading.value = true;
listUser().then(res => { listUser(userParams.value).then(res => {
console.log('用户', res) userLoading.value = false;
loading.value = false;
userList.value = res.data.rows; userList.value = res.data.rows;
total.value = res.data.total; userTotal.value = res.data.total;
handleRowClick()
}) })
} }
getUserList() getUserList()
// 分配报表 // 单选用户
function handleAuthRole(row) { const handleSelectChange = (selection, row) => {
console.log('分配报表', row) if (selection.includes(row)) {
showSelUser.value = true; // 如果当前用户不在已分配用户列表中
} if (!selUserList.value.some(item => item.userId === row.userId)) {
selUserList.value.push(row)
// 获取积木报表列表
const reportFolderList = ref([])
function getReportFolderList() {
getReportFolderListAPI().then(res => {
console.log('报表', res)
reportFolderList.value = res.data;
})
}
getReportFolderList()
/** 数据范围选项*/
const dataScopeOptions = ref([
{ value: "1", label: "全部数据权限" },
{ value: "2", label: "自定数据权限" },
{ value: "3", label: "本部门数据权限" },
{ value: "4", label: "本部门及以下数据权限" },
{ value: "5", label: "仅本人数据权限" }
]);
const data = reactive({
form: {},
queryParams: {
pageNum: 1,
pageSize: 10,
roleName: undefined,
roleKey: undefined,
status: undefined,
deptId: undefined,
nickName: undefined
},
rules: {
roleName: [{ required: true, message: "角色名称不能为空", trigger: "blur" }],
roleKey: [{ required: true, message: "权限字符不能为空", trigger: "blur" }],
roleSort: [{ required: true, message: "角色顺序不能为空", trigger: "blur" }]
},
});
const { queryParams, form, rules } = toRefs(data);
/** 查询角色列表 */
function getList() {
loading.value = true;
listRole(proxy.addDateRange(queryParams.value, dateRange.value)).then(response => {
roleList.value = response.data.rows;
total.value = response.data.total;
loading.value = false;
});
}
/** 搜索按钮操作 */
function handleQuery() {
queryParams.value.pageNum = 1;
getList();
}
/** 重置按钮操作 */
function resetQuery() {
dateRange.value = [];
proxy.resetForm("queryRef");
handleQuery();
}
/** 删除按钮操作 */
function handleDelete(row) {
const roleIds = row.roleId || ids.value;
proxy.$modal.confirm('是否确认删除角色编号为"' + roleIds + '"的数据项?').then(function () {
return delRole(roleIds);
}).then(() => {
getList();
proxy.$modal.msgSuccess("删除成功");
}).catch(() => { });
}
/** 导出按钮操作 */
function handleExport() {
proxy.download("system/role/export", {
...queryParams.value,
}, `role_${new Date().getTime()}.xlsx`);
}
/** 多选框选中数据 */
function handleSelectionChange(selection) {
ids.value = selection.map(item => item.roleId);
single.value = selection.length != 1;
multiple.value = !selection.length;
}
/** 角色状态修改 */
function handleStatusChange(row) {
let text = row.status === "0" ? "启用" : "停用";
proxy.$modal.confirm('确认要"' + text + '""' + row.roleName + '"角色吗?').then(function () {
return changeRoleStatus(row.roleId, row.status);
}).then(() => {
proxy.$modal.msgSuccess(text + "成功");
}).catch(function () {
row.status = row.status === "0" ? "1" : "0";
});
}
/** 更多操作 */
function handleCommand(command, row) {
switch (command) {
case "handleDataScope":
handleDataScope(row);
break;
case "handleAuthUser":
handleAuthUser(row);
break;
default:
break;
}
}
/** 分配用户 */
function handleAuthUser(row) {
router.push("/system/role-auth/user/" + row.roleId);
}
/** 查询菜单树结构 */
function getMenuTreeselect() {
menuTreeselect().then(response => {
menuOptions.value = response.data;
});
}
/** 所有部门节点数据 */
function getDeptAllCheckedKeys() {
// 目前被选中的部门节点
let checkedKeys = deptRef.value.getCheckedKeys();
// 半选中的部门节点
let halfCheckedKeys = deptRef.value.getHalfCheckedKeys();
checkedKeys.unshift.apply(checkedKeys, halfCheckedKeys);
return checkedKeys;
}
/** 重置新增的表单以及其他数据 */
function reset() {
if (menuRef.value != undefined) {
menuRef.value.setCheckedKeys([]);
}
menuExpand.value = false;
menuNodeAll.value = false;
deptExpand.value = true;
deptNodeAll.value = false;
form.value = {
roleId: undefined,
roleName: undefined,
roleKey: undefined,
roleSort: 0,
status: "0",
menuIds: [],
deptIds: [],
menuCheckStrictly: true,
deptCheckStrictly: true,
remark: undefined
};
proxy.resetForm("roleRef");
}
/** 添加角色 */
function handleAdd() {
reset();
getMenuTreeselect();
open.value = true;
title.value = "添加角色";
}
/** 修改角色 */
function handleUpdate(row) {
reset();
const roleId = row.roleId || ids.value;
const roleMenu = getRoleMenuTreeselect(roleId);
getRole(roleId).then(response => {
form.value = response.data;
form.value.roleSort = Number(form.value.roleSort);
open.value = true;
nextTick(() => {
roleMenu.then((res) => {
let checkedKeys = res.checkedKeys;
checkedKeys.forEach((v) => {
nextTick(() => {
menuRef.value.setChecked(v, true, false);
});
});
});
});
});
title.value = "修改角色";
}
/** 根据角色ID查询菜单树结构 */
function getRoleMenuTreeselect(roleId) {
return roleMenuTreeselect(roleId).then(response => {
menuOptions.value = response.menus;
return response;
});
}
/** 根据角色ID查询部门树结构 */
function getDeptTree(roleId) {
return deptTreeSelect(roleId).then(response => {
deptOptions.value = response.depts;
return response;
});
}
/** 树权限(展开/折叠)*/
function handleCheckedTreeExpand(value, type) {
if (type == "menu") {
let treeList = menuOptions.value;
for (let i = 0; i < treeList.length; i++) {
menuRef.value.store.nodesMap[treeList[i].id].expanded = value;
} }
} else if (type == "dept") { } else {
let treeList = deptOptions.value; // 取消选择
for (let i = 0; i < treeList.length; i++) { const index = selUserList.value.findIndex(o => o.userId === row.userId)
deptRef.value.store.nodesMap[treeList[i].id].expanded = value; if (index !== -1) {
selUserList.value.splice(index, 1)
} }
} }
}
/** 树权限(全选/全不选) */
function handleCheckedTreeNodeAll(value, type) {
if (type == "menu") {
menuRef.value.setCheckedNodes(value ? menuOptions.value : []);
} else if (type == "dept") {
deptRef.value.setCheckedNodes(value ? deptOptions.value : []);
}
}
/** 树权限(父子联动) */
function handleCheckedTreeConnect(value, type) {
if (type == "menu") {
form.value.menuCheckStrictly = value ? true : false;
} else if (type == "dept") {
form.value.deptCheckStrictly = value ? true : false;
}
}
/** 所有菜单节点数据 */ console.log(selUserList.value.map(o => o.userId), '当前')
function getMenuAllCheckedKeys() {
// 目前被选中的菜单节点
let checkedKeys = menuRef.value.getCheckedKeys();
// 半选中的菜单节点
let halfCheckedKeys = menuRef.value.getHalfCheckedKeys();
checkedKeys.unshift.apply(checkedKeys, halfCheckedKeys);
return checkedKeys;
} }
// 全选反选用户
/** 提交按钮 */ const handleSelectAll = (selection) => {
function submitForm() { if (selection.length === 0) {
proxy.$refs["roleRef"].validate(valid => { // 取消当前页
if (valid) { userList.value.forEach(o => {
if (form.value.roleId != undefined) { const index = selUserList.value.findIndex(item => item.userId === o.userId)
form.value.menuIds = getMenuAllCheckedKeys(); if (index !== -1) {
updateRole(form.value).then(response => { selUserList.value.splice(index, 1)
proxy.$modal.msgSuccess("修改成功");
open.value = false;
getList();
});
} else {
form.value.menuIds = getMenuAllCheckedKeys();
addRole(form.value).then(response => {
proxy.$modal.msgSuccess("新增成功");
open.value = false;
getList();
});
} }
} })
}); } else {
} // 全选当前页
userList.value.forEach(o => {
// 如果当前用户不在已分配用户列表中
if (!selUserList.value.some(item => item.userId === o.userId)) {
selUserList.value.push(o)
}
})
}
/** 取消按钮 */ console.log(selUserList.value, '当前')
function cancel() {
open.value = false;
reset();
} }
/** 选择角色权限范围触发 */ // 确定分配给用户
function dataScopeSelectChange(value) { const handleSelectUser = async () => {
if (value !== "2") { const reportId = checkedReportRowList.value[0].id
deptRef.value.setCheckedKeys([]); const userIds = selUserList.value.map(o => o.userId)
} await batchAuthReportAPI({
} reportId,
userIds
})
// 通知成功并关闭弹窗
proxy.$modal.msgSuccess('分配成功')
showSelUser.value = false
/** 分配数据权限操作 */
function handleDataScope(row) {
reset();
const deptTreeSelect = getDeptTree(row.roleId);
getRole(row.roleId).then(response => {
form.value = response.data;
openDataScope.value = true;
nextTick(() => {
deptTreeSelect.then(res => {
nextTick(() => {
if (deptRef.value) {
deptRef.value.setCheckedKeys(res.checkedKeys);
}
});
});
});
});
title.value = "分配数据权限";
} }
/** 提交按钮(数据权限) */ // 表格动态设置复选框选中效果
function submitDataScope() { const handleRowClick = () => {
if (form.value.roleId != undefined) { // 这里添加设置选中状态的逻辑
form.value.deptIds = getDeptAllCheckedKeys(); nextTick(() => {
dataScope(form.value).then(response => { if (proxy.$refs.refTable) {
proxy.$modal.msgSuccess("修改成功"); proxy.$refs.refTable.clearSelection()
openDataScope.value = false; const selectedIds = selUserList.value ? selUserList.value.map(o => o.userId) : []
getList(); userList.value.forEach(row => {
}); if (selectedIds.includes(row.userId)) {
} proxy.$refs.refTable.toggleRowSelection(row, true)
}
})
}
})
} }
/** 取消按钮(数据权限)*/ // 关闭分配用户弹窗
function cancelDataScope() { const closeSelUserDialog = () => {
openDataScope.value = false; userParams.value.pageNum = 1
reset(); userParams.value.pageSize = 10
getUserList()
} }
getList();
</script> </script>
<style scoped <style scoped
lang="scss"> lang="scss">
...@@ -584,4 +406,15 @@ ...@@ -584,4 +406,15 @@
background-color: transparent; background-color: transparent;
} }
} }
/* 分配用户 */
::v-deep(.assign-user-dialog) {
color: white;
.el-overlay-dialog .el-dialog .el-dialog__header {
height: auto !important;
padding-bottom: 0px;
}
}
</style> </style>
\ No newline at end of file
...@@ -39,7 +39,7 @@ export default defineConfig(({ mode, command }) => { ...@@ -39,7 +39,7 @@ export default defineConfig(({ mode, command }) => {
rewrite: (p) => p.replace(/^\/ql_local/, '') rewrite: (p) => p.replace(/^\/ql_local/, '')
}, },
'/bc_local': { '/bc_local': {
target: 'http://192.168.101.103:8180', target: 'http://192.168.100.103:8180',
changeOrigin: true, changeOrigin: true,
rewrite: (p) => p.replace(/^\/bc_local/, '') rewrite: (p) => p.replace(/^\/bc_local/, '')
} }
......
Markdown 格式
0%
您添加了 0 到此讨论。请谨慎行事。
请先完成此评论的编辑!
注册 或者 后发表评论