权限列表

创建分支

  • 由于要开发一个新的功能模块,需要创建一个新的分支rights
1
2
git checkout -b rights
git branch
  • 然后推送到云端,由于是第一次推送,需要-u参数
1
git push -u origin rights

权限路由

  • 创建power/Rights.vue组件,并在home子路由下创建路由规则
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
const router = new VueRouter({
routes: [
{ path: '/', redirect: '/login' },
{ path: '/login', component: Login },
{
path: '/home',
component: Home,
redirect: '/welcome',
children: [
{ path: '/welcome', component: Welcome },
{ path: '/users', component: Users },
{ path: '/rights', component: Rights}
]
}
]
})
  • 绘制基本的视图后,请求权限列表的数据
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
export default {
name: 'Rights',
data() {
return {
// 权限列表
rightsList: [],
}
},
created() {
// 获取所有权限
this.getRightsList()
},
methods: {
// 获取权限列表
async getRightsList() {
const { data: res } = await this.$http.get('rights/list')
if(res.meta.status !== 200){
return this.$message.error('获取权限列表失败')
}
this.rightsList = res.data
},
},
}

渲染视图

  • 使用到了表格,面包屑,卡片和Tag标签UI组件
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
<template>
<div>
<!--面包屑导航区-->
<el-breadcrumb separator-class="el-icon-arrow-right">
<el-breadcrumb-item :to="{ path: '/home' }">首页</el-breadcrumb-item>
<el-breadcrumb-item>权限管理</el-breadcrumb-item>
<el-breadcrumb-item>权限列表</el-breadcrumb-item>
</el-breadcrumb>

<!--卡片视图-->
<el-card>
<el-table :data="rightsList" border stripe>
<el-table-column label="#" type="index"></el-table-column>
<el-table-column label="权限名称" prop="authName"></el-table-column>
<el-table-column label="路径" prop="path"></el-table-column>
<el-table-column label="权限等级">
<template slot-scope="scope">
<el-tag v-if="scope.row.level === '0'">一级</el-tag>
<el-tag type="success" v-if="scope.row.level === '1'">二级</el-tag>
<el-tag type="warning" v-if="scope.row.level === '2'">三级</el-tag>
</template>
</el-table-column>
</el-table>
</el-card>
</div>
</template>

  • 通过权限管理模块控制不同的用户可以进行哪些操作,具体可以通过角色的方式进行控制,即每个用户分配一个特定的角色,角色包括不同的功能权限

角色列表

角色路由

  • 创建power/Roles.vue组件,并在home下配置子路由
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
const router = new VueRouter({
routes: [
{ path: '/', redirect: '/login' },
{ path: '/login', component: Login },
{
path: '/home',
component: Home,
redirect: '/welcome',
children: [
{ path: '/welcome', component: Welcome },
{ path: '/users', component: Users },
{ path: '/rights', component: Rights},
{ path: '/roles', component: Roles}
]
}
]
})
  • 并发送请求获取数据
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
export default {
name: 'Roles',
data() {
return {
// 所有角色列表数据
roleList: [],
}
},
created() {
this.getRolesList()
},
methods: {
// 获取所有角色的列表
async getRolesList() {
const { data: res } = await this.$http.get('roles')
if (res.meta.status !== 200) {
return this.$message.error('获取角色列表失败')
}
this.roleList = res.data
console.log(this.roleList)
},
},
}

渲染视图

  • 通过表格渲染,并且具有展开列,里面存放一二三级权限,通过一二三层for循环实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
<template>
<div>
<!--面包屑导航区-->
<el-breadcrumb separator-class="el-icon-arrow-right">
<el-breadcrumb-item :to="{ path: '/home' }">首页</el-breadcrumb-item>
<el-breadcrumb-item>权限管理</el-breadcrumb-item>
<el-breadcrumb-item>角色列表</el-breadcrumb-item>
</el-breadcrumb>
<!--卡片视图-->
<el-card>
<el-row>
<el-col>
<el-button type="primary">添加角色</el-button>
</el-col>
</el-row>

<!--角色列表区域-->
<el-table :data="roleList" border stripe>
<!--展开列-->
<el-table-column type="expand">
<template slot-scope="scope">
<!--使用如下方式格式化查看-->
<!-- <pre>{{scope.row}}</pre> -->
<el-row
:class="['vcenter','bdbottom', i1 === 0 ? 'bdtop' : 'bdbottom']"
v-for="(item1, i1) in scope.row.children"
:key="item1.id"
>
<!--渲染一级权限-->
<el-col :span="5">
<el-tag>{{ item1.authName }}</el-tag>
<i class="el-icon-caret-right e"></i>
</el-col>
<!--渲染二级和三级权限-->
<el-col :span="19">
<!--通过for循环嵌套渲染二级权限-->
<el-row
:class="['vcenter','bdbottom', i1 === 0 ? 'bdtop' : 'bdbottom']"
v-for="(item2, i2) in item1.children"
:key="item2.id"
>
<el-col :span="6">
<el-tag type="success">{{ item2.authName }}</el-tag>
<i class="el-icon-caret-right e"></i>
</el-col>
<!-- for循环渲染三级权限-->
<el-col :span="18">
<el-tag type="warning" v-for="(item3, i3) in item2.children" :key="item3.id">{{item3.authName}}</el-tag>
</el-col>
</el-row>
</el-col>
</el-row>
</template>
</el-table-column>
<!--索引列-->
<el-table-column type="index" label="#"></el-table-column>
<el-table-column label="角色名称" prop="roleName"></el-table-column>
<el-table-column label="角色描述" prop="roleDesc"></el-table-column>
<el-table-column label="操作">
<template slot-scope="scope">
<el-button type="primary" icon="el-icon-edit" size="mini"
>编辑</el-button
>
<el-button type="danger" icon="el-icon-delete" size="mini"
>删除</el-button
>
<el-button type="warning" icon="el-icon-setting" size="mini"
>分配权限</el-button
>
</template>
</el-table-column>
</el-table>
</el-card>
</div>
</template>
1
2
3
4
5
6
7
8
9
10
11
12
13
.el-tag {
margin: 7px;
}
.bdtop {
border-top: 1px solid #eee;
}
.bdbottom {
border-bottom: 1px solid #eee;
}
.vcenter {
display: flex;
align-items: center;
}

删除功能

  • 每个权限标签tag都可以删除,只需要添加属性closeable即可。并绑定回调事件 @close="removeRightByID"实现删除权限功能
1
<el-tag type="warning" v-for="(item3, i3) in item2.children" :key="item3.id" closable @close="removeRightByID(scope.row,item3.id)">{{item3.authName}}</el-tag>
  • 发送请求
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27

// 根据id删除对应权限
async removeRightByID(role, rightId) {
// 弹框提示是否删除
const confirmResult = await this.$confirm(
'此操作将永久删除该文件, 是否继续?',
'提示',
{
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'warning',
}
).catch((err) => err)
if (confirmResult !== 'confirm') {
return this.$message.info('已取消删除')
}
// 发送请求删除权限
const { data: res } = await this.$http.delete(
`roles/${role.id}/rights/${rightId}`
)
if (res.meta.status !== 200) {
return this.$message.error('删除权限失败')
}
// 重新更新数据,不能重新发送请求,这样会导致展开列关闭
// 使用重新赋值比较好
role.children = res.data
}

分配权限

权限对话框

  • 点击分配权限按钮需要弹出对话框
1
2
3
4
5
6
7
8
9
10
11
<el-dialog
title="分配权限"
:visible.sync="setRightDialogVisible"
width="50%"
>
<span>这是一段信息</span>
<span slot="footer" class="dialog-footer">
<el-button @click="setRightDialogVisible = false">取 消</el-button>
<el-button type="primary" @click="setRightDialogVisible = false">确 定</el-button>
</span>
</el-dialog>
  • 打开对话框后需要发送请求获取所有的权限
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
export default {
name: 'Roles',
data() {
return {
// 所有角色列表数据
roleList: [],
setRightDialogVisible: false, // 控制分配权限对话框的显示与隐藏
tightsList:[], // 所有权限的数据
}
},
created() {
this.getRolesList()
},
methods: {
//..
// 展示分配权限的对话框
async showSetRightDialog() {
// 获取所有权限的数据
const { data: res } = await this.$http.get('rights/tree')
if(res.meta.status !== 200) {
return this.$message.error('获取权限失败')
}
// 获取到的权限数据保存到data中
this.rightsList = res.data
console.log(this.rightsList)
this.setRightDialogVisible = true
},
},
}

树形视图

  • 使用到了Tree树形组件
1
2
3
4
5
6
7
8
9
<!--树形控件-->
<el-tree
:data="rightsList"
:props="treeProps"
show-checkbox
node-key="id"
:default-expand-all="true"
:default-checked-keys="defKeys"
></el-tree>
  • 需要把三级权限的ID保存起来实现默认选中,在对话框关闭后需要清空数组
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
export default {
name: 'Roles',
data() {
return {
// 所有角色列表数据
roleList: [],
setRightDialogVisible: false, // 控制分配权限对话框的显示与隐藏
rightsList: [], // 所有权限的数据
treeProps: {
// 树形组件的绑定对象
label: 'authName', // 内容
children: 'children', // 父子节点
},
// 默认选中的节点ID值数组((只需三级权限ID即可)
defKeys: [],
}
},
created() {
this.getRolesList()
},
methods: {
//...
// 展示分配权限的对话框
async showSetRightDialog(role) {
// 获取所有权限的数据
const { data: res } = await this.$http.get('rights/tree')
if (res.meta.status !== 200) {
return this.$message.error('获取权限失败')
}
// 获取到的权限数据保存到data中
this.rightsList = res.data
// 递归获取三级节点的ID
this.getLeafKeys(role, this.defKeys)
this.setRightDialogVisible = true
},
// 通过递归的形式获取角色下所有三级权限的id并保存
getLeafKeys(node, arr) {
// node节点不包含childran属性即为第三级别
if (!node.children) {
return arr.push(node.id)
}
// 递归获取
node.children.forEach((item) => {
this.getLeafKeys(item, arr)
})
},
// 监听分配权限关闭对话框的事件
setRightDialogClosed(){
this.defKeys = []
}
},
}

发送请求

  • 发送请求前需要获取所有选中的权限的ID中,只需要使用到Tree组件引用的对象的一个方法getCheckedKeysgetHalfCheckedKeys
1
2
3
4
<!--树形控件,使用ref指定引用对象-->
<el-tree ref="treeRef"></el-tree>
<!--点击按钮触发事件分配权限-->
<el-button type="primary" @click="allotRights">确 定</el-button>
  • 发送请求
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
// 为角色分配权限
async allotRights() {
const keys = [
...this.$refs.treeRef.getCheckedKeys(),
...this.$refs.treeRef.getHalfCheckedKeys(),
]
// 所有ID以,逗号进行拼接
const idStr = keys.join(',')
const { data: res } = await this.$http.post(
`roles/${this.roleId}/rights`,
{ rids: idStr }
)
if (res.meta.status !== 200) {
return this.$message.error('分配权限失败')
}
this.$message.success('分配角色成功')
// 重新获取角色列表
this.getRolesList()
// 关闭对话框
this.setRightDialogVisible = false
}

分配角色

分配对话框

  • 使用到了Select 选择器组件
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
<!--分配角色的对话框-->
<el-dialog title="分配角色" :visible.sync="setRoledialogVisible" width="50%">
<div>
<p>当前的用户:{{ userInfo.username }}</p>
<p>当前的角色:{{ userInfo.role_name }}</p>
<p>
分配新角色:
<el-select v-model="selectRoleId" placeholder="请选择">
<el-option v-for="item in rolesList" :key="item.id" :label="item.roleName" :value="item.id">
</el-option>
</el-select>
</p>
</div>
<span slot="footer" class="dialog-footer">
<el-button @click="setRoledialogVisible = false">取 消</el-button>
<el-button type="primary" @click="setRoledialogVisible = false">确 定</el-button>
</span>
</el-dialog>
  • 绑定相关数据和事件
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
export default {
name: 'Users',
data() {
return {
// 控制分配角色对话框的显示与隐藏
setRoledialogVisible: false,
userInfo: {}, // 需要被分配角色的用户信息
rolesList: [], // 所有角色的数据列表
selectRoleId: '' // 已选中的角色ID值
}
},
created() {
this.getUserList()
},
methods: {
// 分配角色的事件
async setRole(userInfo) {
this.userInfo = userInfo
// 在展示对话框之前,获取所有角色的列表
const { data: res } = await this.$http.get('roles')
if (res.meta.status !== 200) {
return this.$message.error('获取角色列表失败')
}
this.rolesList = res.data
this.setRoledialogVisible = true
}
}
}

发送请求

  • 给对话框确定按钮绑定事件,关闭对话框也绑定事件清除下拉框内容
1
2
3
<el-dialog @close="setRoledialogClosed">
<el-button type="primary" @click="saveRoleInfo">确 定</el-button>
</el-dialog>
  • 发送请求
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
 // 点击按钮分配角色
async saveRoleInfo() {
if (!this.selectRoleId) {
return this.$message.error('请选择分配的角色')
}
// 发送请求
const { data: res } = await this.$http.put(
`users/${this.userInfo.id}/role`,
{ rid: this.selectRoleId }
)
if (res.meta.status !== 200) {
return this.$message.error('更新角色失败')
}
this.$message.success('更新角色成功')
this.getUserList()
this.setRoledialogVisible = false
},
// 监听分配角色对话框的事件
setRoledialogClosed(){
this.selectRoleId = ''
this.userInfo = {}
}

提交码云

  • 关于权限管理模块已经完成了,需要上传到码云
1
2
3
4
5
6
# 查看是否是right分支
git branch
git add .
git commit -m "完成了权限管理功能"
# 云端已经有了right分支就直接push
git push
  • right分支合并到master分支
1
2
3
4
git checkout master
git merge rights
# 更新云端matser分支
git push