mirror of
https://gitee.com/ssssssss-team/magic-boot.git
synced 2025-04-05 04:41:51 +08:00
370 lines
11 KiB
Vue
370 lines
11 KiB
Vue
<template>
|
|
<div class="app-container">
|
|
|
|
<div class="filter-container">
|
|
<el-form :inline="true">
|
|
<el-form-item label="搜索菜单名称">
|
|
<el-input v-model="searchValue" @input="searchMenu"></el-input>
|
|
</el-form-item>
|
|
<el-button class="filter-item" style="margin-bottom:10px;" type="primary" icon="el-icon-edit" @click="addSubMenu('0')">
|
|
添加菜单
|
|
</el-button>
|
|
</el-form>
|
|
</div>
|
|
|
|
<pd-table ref="table" v-bind="tableOptions" v-if="menuData && menuData.length > 0" />
|
|
|
|
<pd-dialog ref="menuFormDialog" width="800px" :title="textMap[dialogStatus]" @confirm-click="save()">
|
|
<template #content>
|
|
<el-form ref="dataForm" :rules="rules" :model="temp" label-position="left" label-width="100px" style="width: 600px; margin-left:50px;">
|
|
<el-form-item label="上级菜单" prop="pid">
|
|
<treeselect v-model="temp.pid" :options="menuTree" />
|
|
</el-form-item>
|
|
<el-form-item label="菜单名称" prop="name">
|
|
<el-input v-model="temp.name" />
|
|
</el-form-item>
|
|
<el-form-item label="菜单链接" prop="url">
|
|
<el-input v-model="temp.url" />
|
|
</el-form-item>
|
|
<el-form-item label="权限标识" prop="permission">
|
|
<el-input v-model="temp.permission" />
|
|
</el-form-item>
|
|
<el-form-item label="图标" prop="icon">
|
|
<a @click="openIcons">
|
|
<el-input placeholder="请选择图标" v-model="temp.icon" class="input-with-select">
|
|
<el-button class="icon-btn" slot="append">
|
|
<i v-html="generateIconCode(temp.icon)"></i>
|
|
</el-button>
|
|
</el-input>
|
|
</a>
|
|
</el-form-item>
|
|
<el-form-item label="排序" prop="sort">
|
|
<el-input v-model="temp.sort" />
|
|
</el-form-item>
|
|
<el-form-item label="描述">
|
|
<el-input v-model="temp.descRibe" :autosize="{ minRows: 4, maxRows: 6}" type="textarea" />
|
|
</el-form-item>
|
|
<el-form-item label="是否显示">
|
|
<el-checkbox v-model="temp.isShow">显示</el-checkbox>
|
|
</el-form-item>
|
|
</el-form>
|
|
</template>
|
|
</pd-dialog>
|
|
|
|
<pd-dialog ref="iconDialog">
|
|
<template #content>
|
|
<menu-icons :select-icon="selectIcon" />
|
|
</template>
|
|
</pd-dialog>
|
|
|
|
</div>
|
|
</template>
|
|
|
|
<script>
|
|
import MenuIcons from './menu-icons'
|
|
import Treeselect from '@riophae/vue-treeselect'
|
|
import '@riophae/vue-treeselect/dist/vue-treeselect.css'
|
|
|
|
export default {
|
|
components: { MenuIcons, Treeselect },
|
|
data() {
|
|
return {
|
|
menuData: [],
|
|
menuTree: [],
|
|
searchValue: '',
|
|
tableOptions: {
|
|
el: {
|
|
'default-expand-all': true,
|
|
'tree-props': { children: 'children', hasChildren: 'hasChildren' },
|
|
'row-key': 'id'
|
|
},
|
|
showNo: false,
|
|
page: false,
|
|
cols: [
|
|
{
|
|
field: 'name',
|
|
title: '菜单名称',
|
|
align: 'left',
|
|
type: 'html'
|
|
},
|
|
{
|
|
field: 'url',
|
|
title: '路径',
|
|
align: 'left'
|
|
},
|
|
{
|
|
field: 'permission',
|
|
title: '权限标识',
|
|
width: 150,
|
|
align: 'left'
|
|
},
|
|
{
|
|
field: 'icon',
|
|
title: '图标',
|
|
width: 45,
|
|
align: 'left',
|
|
templet: (row) => {
|
|
return this.generateIconCode(row.icon)
|
|
}
|
|
},
|
|
{
|
|
field: 'sort',
|
|
title: '排序',
|
|
width: 60
|
|
},
|
|
{
|
|
field: 'isShow',
|
|
title: '是否显示',
|
|
type: 'switch',
|
|
width: 100,
|
|
change: (row) => {
|
|
this.$get('menu/change/show', {
|
|
id: row.id,
|
|
isShow: row.isShow
|
|
})
|
|
}
|
|
},
|
|
{
|
|
title: '操作',
|
|
type: 'btns',
|
|
width: 262,
|
|
fixed: 'right',
|
|
align: 'left',
|
|
btns: [
|
|
{
|
|
title: '添加下级菜单',
|
|
type: 'text',
|
|
click: (row) => {
|
|
this.addSubMenu(row.id)
|
|
}
|
|
},
|
|
{
|
|
title: '编辑',
|
|
type: 'text',
|
|
click: (row) => {
|
|
this.handleUpdate(row)
|
|
}
|
|
},
|
|
{
|
|
title: '删除',
|
|
type: 'text',
|
|
click: (row) => {
|
|
this.$common.handleDelete({
|
|
url: 'menu/delete',
|
|
id: row.id,
|
|
done: () => this.reloadTable()
|
|
})
|
|
}
|
|
}
|
|
]
|
|
}
|
|
]
|
|
},
|
|
dialogFormVisible: false,
|
|
dialogStatus: '',
|
|
textMap: {
|
|
update: '修改',
|
|
create: '添加'
|
|
},
|
|
temp: this.getTemp(),
|
|
listConfigDialogVisible: false,
|
|
formConfigDialogVisible: false,
|
|
rules: {
|
|
pid: [{ required: true, message: '请选择上级菜单', trigger: 'change' }],
|
|
name: [{ required: true, message: '请输入菜单名称', trigger: 'change' }]
|
|
}
|
|
}
|
|
},
|
|
mounted() {
|
|
this.reloadTable()
|
|
},
|
|
watch: {
|
|
menuData() {
|
|
this.menuTree = [{
|
|
label: '根节点',
|
|
id: '0',
|
|
children: this.genMenuTree(this.menuData)
|
|
}]
|
|
}
|
|
},
|
|
methods: {
|
|
isChildren(children, id) {
|
|
var result = false
|
|
for(var i in children) {
|
|
var chi = children[i]
|
|
if(chi.id == id){
|
|
result = true
|
|
}
|
|
if(chi.children && children.length > 0){
|
|
if(this.isChildren(chi.children, id)){
|
|
result = true
|
|
}
|
|
}
|
|
}
|
|
return result
|
|
},
|
|
queryChildren(children, id) {
|
|
var result = []
|
|
for(var i in children){
|
|
var chi = children[i]
|
|
if(chi.id == id){
|
|
if(chi.children && chi.children.length > 0){
|
|
result = chi.children
|
|
}
|
|
}else{
|
|
var qc = this.queryChildren(chi.children, id)
|
|
if(qc.length > 0){
|
|
result = qc
|
|
}
|
|
}
|
|
}
|
|
return result
|
|
},
|
|
genMenuTree(children) {
|
|
var menuTree = []
|
|
for(var i in children){
|
|
var chi = {}
|
|
chi.label = children[i].name
|
|
chi.id = children[i].id
|
|
if(children[i].children && children[i].children.length > 0){
|
|
chi.children = this.genMenuTree(children[i].children)
|
|
}
|
|
menuTree.push(chi)
|
|
}
|
|
return menuTree
|
|
},
|
|
searchMenu() {
|
|
if(this.searchValue){
|
|
this.$set(this.tableOptions, 'data', this.recursionSearch(this.$common.copyNew(this.menuData), this.searchValue))
|
|
}else{
|
|
this.$set(this.tableOptions, 'data', this.menuData)
|
|
}
|
|
},
|
|
recursionSearch(data, text){
|
|
var searchData = []
|
|
for(var i in data){
|
|
var treeNode = data[i]
|
|
var children = treeNode.children
|
|
if(children && children.length > 0){
|
|
var childrenSearch = this.recursionSearch(children, text)
|
|
treeNode.children = childrenSearch && childrenSearch.length > 0 ? childrenSearch : treeNode.children
|
|
if(treeNode.name.indexOf(text) != -1 || childrenSearch.length > 0){
|
|
treeNode.name = treeNode.name.replace(text, `<font color="#FAA353">${text}</font>`)
|
|
searchData.push(treeNode)
|
|
}
|
|
}else{
|
|
if(treeNode.name.indexOf(text) != -1){
|
|
treeNode.name = treeNode.name.replace(text, `<font color="#FAA353">${text}</font>`)
|
|
searchData.push(treeNode)
|
|
}
|
|
}
|
|
}
|
|
return searchData
|
|
},
|
|
generateIconCode(symbol) {
|
|
return `<svg style="width: 20px;height: 20px" aria-hidden="true" class="svg-icon disabled"><use href="#icon-${symbol}"></use></svg>`
|
|
},
|
|
selectIcon(symbol) {
|
|
this.$set(this.temp, 'icon', symbol)
|
|
this.$refs.iconDialog.hide()
|
|
},
|
|
openIcons() {
|
|
this.$refs.iconDialog.show()
|
|
},
|
|
getTemp() {
|
|
return {
|
|
id: '',
|
|
name: '',
|
|
url: '',
|
|
permission: '',
|
|
sort: 0,
|
|
descRibe: '',
|
|
isShow: true,
|
|
pid: '',
|
|
icon: ''
|
|
}
|
|
},
|
|
resetTemp() {
|
|
this.temp = this.getTemp()
|
|
},
|
|
getSort() {
|
|
this.$get('menu/sort', { pid: this.temp.pid }).then(res => {
|
|
this.temp.sort = res.data
|
|
})
|
|
},
|
|
addSubMenu(id) {
|
|
this.resetTemp()
|
|
this.temp.pid = id
|
|
this.temp.id = this.$common.uuid()
|
|
this.getSort()
|
|
this.dialogStatus = 'create'
|
|
this.$refs.menuFormDialog.show()
|
|
this.$nextTick(() => {
|
|
this.$refs['dataForm'].clearValidate()
|
|
})
|
|
},
|
|
save() {
|
|
this.$refs['dataForm'].validate((valid) => {
|
|
if (valid) {
|
|
if(this.temp.pid == this.temp.id){
|
|
this.$notify({
|
|
title: '失败',
|
|
message: '上级菜单不能选当前菜单',
|
|
type: 'error',
|
|
duration: 2000
|
|
})
|
|
return
|
|
}
|
|
if(this.isChildren(this.queryChildren(this.menuData, this.temp.id), this.temp.pid)){
|
|
this.$notify({
|
|
title: '失败',
|
|
message: '上级菜单不能选当前菜单子级',
|
|
type: 'error',
|
|
duration: 2000
|
|
})
|
|
return
|
|
}
|
|
this.temp.isShow = this.temp.isShow === true ? 1 : 0
|
|
this.$post('menu/save', this.temp).then(() => {
|
|
this.reloadTable()
|
|
this.$refs.menuFormDialog.hide()
|
|
this.$notify({
|
|
title: '成功',
|
|
message: (this.dialogStatus === 'create' ? '创建' : '修改') + '成功',
|
|
type: 'success',
|
|
duration: 2000
|
|
})
|
|
})
|
|
}
|
|
})
|
|
},
|
|
reloadTable() {
|
|
this.$get('menu/tree').then(res => {
|
|
this.menuData = res.data.list
|
|
this.$set(this.tableOptions, 'data', this.menuData)
|
|
})
|
|
},
|
|
handleUpdate(row) {
|
|
for (var t in this.temp) {
|
|
if (t === 'isShow') {
|
|
if (row[t] === 1) {
|
|
this.temp[t] = true
|
|
} else {
|
|
this.temp[t] = false
|
|
}
|
|
} else {
|
|
this.temp[t] = row[t]
|
|
}
|
|
}
|
|
this.temp.name = this.temp.name.replaceAll(/<font.*?>(.*?)<\/font>/g,'$1')
|
|
this.dialogStatus = 'update'
|
|
this.$refs.menuFormDialog.show()
|
|
this.$nextTick(() => {
|
|
this.$refs['dataForm'].clearValidate()
|
|
})
|
|
}
|
|
}
|
|
}
|
|
|
|
</script>
|