menu.vue 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294
  1. <template>
  2. <div>
  3. <div class="button-box clearflex">
  4. <el-button @click="addMenu('0')" type="primary">新增根菜单</el-button>
  5. </div>
  6. <!-- 由于此处菜单跟左侧列表一一对应所以不需要分页 pageSize默认999 -->
  7. <el-table :data="tableData" border row-key="ID" stripe>
  8. <el-table-column label="ID" min-width="100" prop="ID"></el-table-column>
  9. <el-table-column label="路由Name" min-width="160" prop="name"></el-table-column>
  10. <el-table-column label="是否隐藏" min-width="100" prop="hidden">
  11. <template slot-scope="scope">
  12. <span>{{scope.row.hidden?"隐藏":"显示"}}</span>
  13. </template>
  14. </el-table-column>
  15. <el-table-column label="父节点" min-width="90" prop="parentId"></el-table-column>
  16. <el-table-column label="排序" min-width="70" prop="sort"></el-table-column>
  17. <el-table-column label="文件路径" min-width="360" prop="component"></el-table-column>
  18. <el-table-column label="展示名称" min-width="120" prop="authorityName">
  19. <template slot-scope="scope">
  20. <span>{{scope.row.meta.title}}</span>
  21. </template>
  22. </el-table-column>
  23. <el-table-column label="图标" min-width="140" prop="authorityName">
  24. <template slot-scope="scope">
  25. <span>{{scope.row.meta.icon}}</span>
  26. </template>
  27. </el-table-column>
  28. <el-table-column fixed="right" label="操作" width="300">
  29. <template slot-scope="scope">
  30. <el-button @click="deleteMenu(scope.row.ID)" size="small" type="text">删除菜单</el-button>
  31. <el-button @click="editMenu(scope.row.ID)" size="small" type="text">编辑菜单</el-button>
  32. <el-button @click="addMenu(scope.row.ID)" size="small" type="text">添加子菜单</el-button>
  33. </template>
  34. </el-table-column>
  35. </el-table>
  36. <el-dialog :before-close="handleClose" :visible.sync="dialogFormVisible" :title="dialogTitle">
  37. <el-form :inline="true" :model="form" :rules="rules" label-width="85px" ref="menuForm">
  38. <el-form-item label="路由name" prop="path">
  39. <el-input autocomplete="off" placeholder="唯一英文字符串" v-model="form.path"></el-input>
  40. </el-form-item>
  41. <el-form-item label="是否隐藏">
  42. <el-select placeholder="是否在列表隐藏" v-model="form.hidden">
  43. <el-option :value="false" label="否"></el-option>
  44. <el-option :value="true" label="是"></el-option>
  45. </el-select>
  46. </el-form-item>
  47. <el-form-item label="父节点Id">
  48. <el-select
  49. placeholder="请选择"
  50. v-model="form.parentId"
  51. :disabled="!this.isEdit"
  52. filterable
  53. >
  54. <el-option
  55. :disabled="canSelect(item)"
  56. :key="item.ID"
  57. :label="item.title"
  58. :value="item.ID"
  59. v-for="item in menuOption"
  60. >
  61. </el-option>
  62. </el-select>
  63. </el-form-item>
  64. <el-form-item label="文件路径" prop="component">
  65. <el-input autocomplete="off" v-model="form.component"></el-input>
  66. </el-form-item>
  67. <el-form-item label="展示名称" prop="meta.title">
  68. <el-input autocomplete="off" v-model="form.meta.title"></el-input>
  69. </el-form-item>
  70. <el-form-item label="图标" prop="meta.icon">
  71. <el-input autocomplete="off" v-model="form.meta.icon"></el-input>
  72. </el-form-item>
  73. <el-form-item label="排序标记" prop="sort">
  74. <el-input autocomplete="off" v-model.number="form.sort"></el-input>
  75. </el-form-item>
  76. <el-form-item label="keepAlive" prop="meta.keepAlive">
  77. <el-select placeholder="是否keepAlive缓存页面" v-model="form.meta.keepAlive">
  78. <el-option :value="false" label="否"></el-option>
  79. <el-option :value="true" label="是"></el-option>
  80. </el-select>
  81. </el-form-item>
  82. </el-form>
  83. <div class="warning">新增菜单需要在角色管理内配置权限才可使用</div>
  84. <div class="dialog-footer" slot="footer">
  85. <el-button @click="closeDialog">取 消</el-button>
  86. <el-button @click="enterDialog" type="primary">确 定</el-button>
  87. </div>
  88. </el-dialog>
  89. </div>
  90. </template>
  91. <script>
  92. // 获取列表内容封装在mixins内部 getTableData方法 初始化已封装完成
  93. import {
  94. updateBaseMenu,
  95. getMenuList,
  96. addBaseMenu,
  97. deleteBaseMenu,
  98. getBaseMenuById
  99. } from '@/api/menu'
  100. import infoList from '@/components/mixins/infoList'
  101. export default {
  102. name: 'Menus',
  103. mixins: [infoList],
  104. data() {
  105. return {
  106. listApi: getMenuList,
  107. dialogFormVisible: false,
  108. dialogTitle:"新增菜单",
  109. menuOption:[
  110. {
  111. ID:"0",
  112. title:"根菜单"
  113. }
  114. ],
  115. form: {
  116. ID: 0,
  117. path: '',
  118. name: '',
  119. hidden: '',
  120. parentId: '',
  121. component: '',
  122. meta: {
  123. title: '',
  124. icon: '',
  125. defaultMenu:false,
  126. keepAlive:false
  127. }
  128. },
  129. rules: {
  130. path: [{ required: true, message: '请输入菜单name', trigger: 'blur' }],
  131. component: [
  132. { required: true, message: '请输入文件路径', trigger: 'blur' }
  133. ],
  134. 'meta.title': [
  135. { required: true, message: '请输入菜单展示名称', trigger: 'blur' }
  136. ]
  137. },
  138. isEdit: false
  139. }
  140. },
  141. methods: {
  142. handleClose(done) {
  143. this.initForm()
  144. done()
  145. },
  146. // 懒加载子菜单
  147. load(tree, treeNode, resolve) {
  148. resolve([
  149. {
  150. id: 31,
  151. date: '2016-05-01',
  152. name: '王小虎',
  153. address: '上海市普陀区金沙江路 1519 弄'
  154. },
  155. {
  156. id: 32,
  157. date: '2016-05-01',
  158. name: '王小虎',
  159. address: '上海市普陀区金沙江路 1519 弄'
  160. }
  161. ])
  162. },
  163. // 删除菜单
  164. deleteMenu(ID) {
  165. this.$confirm('此操作将永久删除所有角色下该菜单, 是否继续?', '提示', {
  166. confirmButtonText: '确定',
  167. cancelButtonText: '取消',
  168. type: 'warning'
  169. })
  170. .then(async () => {
  171. const res = await deleteBaseMenu({ ID })
  172. if (res.code == 0) {
  173. this.$message({
  174. type: 'success',
  175. message: '删除成功!'
  176. })
  177. this.getTableData()
  178. }
  179. })
  180. .catch(() => {
  181. this.$message({
  182. type: 'info',
  183. message: '已取消删除'
  184. })
  185. })
  186. },
  187. // 初始化弹窗内表格方法
  188. initForm() {
  189. this.$refs.menuForm.resetFields()
  190. },
  191. // 关闭弹窗
  192. closeDialog() {
  193. this.initForm()
  194. this.dialogFormVisible = false
  195. },
  196. // 添加menu
  197. async enterDialog() {
  198. this.$refs.menuForm.validate(async valid => {
  199. if (valid) {
  200. let res
  201. this.form.name = this.form.path
  202. if (this.isEdit) {
  203. res = await updateBaseMenu(this.form)
  204. } else {
  205. res = await addBaseMenu(this.form)
  206. }
  207. if (res.code == 0) {
  208. this.$message({
  209. type: 'success',
  210. message: '添加成功!'
  211. })
  212. this.getTableData()
  213. } else {
  214. this.$message({
  215. type: 'error',
  216. message: '添加失败!'
  217. })
  218. }
  219. this.initForm()
  220. this.dialogFormVisible = false
  221. }
  222. })
  223. },
  224. // 添加菜单方法,id为 0则为添加根菜单
  225. addMenu(id) {
  226. this.dialogTitle = "新增菜单"
  227. this.form.parentId = String(id)
  228. this.isEdit = false
  229. this.dialogFormVisible = true
  230. },
  231. // 修改菜单方法
  232. async editMenu(id) {
  233. this.dialogTitle = "编辑菜单"
  234. const res = await getBaseMenuById({ id })
  235. this.form = res.data.menu
  236. this.dialogFormVisible = true
  237. this.isEdit = true
  238. },
  239. getMenuList(MenuData){
  240. MenuData.map(item=>{
  241. this.menuOption.push({
  242. ID:String(item.ID),
  243. title:item.meta.title
  244. })
  245. if(item.children){
  246. this.getMenuList(item.children)
  247. }
  248. })
  249. },
  250. findAuthoritySelf(mune,muneData,outData){
  251. muneData&&muneData.some(item=>{
  252. if(item.ID == mune.ID){
  253. outData.push(item)
  254. return true
  255. }
  256. this.findAuthoritySelf(mune,item.children,outData)
  257. })
  258. },
  259. findAllChild(menu,array){
  260. menu&&menu.map(item=>{
  261. array.push(String(item.ID))
  262. this.findAllChild(item.children,array)
  263. })
  264. },
  265. canSelect(authority){
  266. const array = []
  267. const arrayIds = []
  268. this.findAuthoritySelf({ID:this.form.ID},this.tableData,array)
  269. this.findAllChild(array,arrayIds)
  270. return arrayIds.indexOf(authority.ID)>-1
  271. },
  272. },
  273. async created() {
  274. this.pageSize = 999
  275. await this.getTableData()
  276. await this.getMenuList(this.tableData)
  277. }
  278. }
  279. </script>
  280. <style scoped lang="scss">
  281. .button-box {
  282. padding: 10px 20px;
  283. .el-button {
  284. float: right;
  285. }
  286. }
  287. .warning {
  288. color: #dc143c;
  289. }
  290. </style>