index.vue 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244
  1. <template>
  2. <div class="ele-body">
  3. <el-card shadow="never">
  4. <!-- 搜索表单 -->
  5. <menu-search @search="reload" />
  6. <!-- 数据表格 -->
  7. <ele-pro-table
  8. ref="table"
  9. row-key="id"
  10. :columns="columns"
  11. :datasource="datasource"
  12. :default-expand-all="false"
  13. :need-page="false"
  14. :parse-data="parseData"
  15. cache-key="systemMenuTable"
  16. @done="onDone"
  17. >
  18. <!-- 表头工具栏 -->
  19. <template v-slot:toolbar>
  20. <el-button
  21. size="small"
  22. type="primary"
  23. icon="el-icon-plus"
  24. class="ele-btn-icon"
  25. @click="openEdit()"
  26. >
  27. 添加
  28. </el-button>
  29. <el-button class="ele-btn-icon" size="small" @click="expandAll">
  30. 展开全部
  31. </el-button>
  32. <el-button class="ele-btn-icon" size="small" @click="foldAll">
  33. 折叠全部
  34. </el-button>
  35. <!-- <div class="el-button--primary">122222</div> -->
  36. </template>
  37. <!-- 标题列 -->
  38. <template v-slot:name="{ row }">
  39. <i :class="row.icon"></i> {{ row.name }}
  40. </template>
  41. <!-- 类型列 -->
  42. <template v-slot:type="{ row }">
  43. <el-tag
  44. type="primary"
  45. size="mini"
  46. :disable-transitions="true"
  47. >
  48. {{row.type==1?'菜单':(row.type==2?'按钮':'')}}
  49. </el-tag>
  50. </template>
  51. <!-- 操作列 -->
  52. <template v-slot:action="{ row }">
  53. <el-link
  54. type="primary"
  55. :underline="false"
  56. icon="el-icon-plus"
  57. @click="openEdit(null, row.id)"
  58. >
  59. 添加
  60. </el-link>
  61. <el-link
  62. type="primary"
  63. :underline="false"
  64. icon="el-icon-edit"
  65. @click="openEdit(row)"
  66. >
  67. 修改
  68. </el-link>
  69. <el-popconfirm
  70. class="ele-action"
  71. title="确定要删除吗?"
  72. @confirm="remove(row)"
  73. >
  74. <template v-slot:reference>
  75. <el-link type="danger" :underline="false" icon="el-icon-delete">
  76. 删除
  77. </el-link>
  78. </template>
  79. </el-popconfirm>
  80. </template>
  81. </ele-pro-table>
  82. </el-card>
  83. <!-- 编辑弹窗 -->
  84. <menu-edit
  85. :data="current"
  86. :parent-id="parentId"
  87. :menu-list="menuList"
  88. :visible.sync="showEdit"
  89. @done="reload"
  90. />
  91. </div>
  92. </template>
  93. <script>
  94. import MenuSearch from './components/menu-search.vue';
  95. import MenuEdit from './components/menu-edit.vue';
  96. import { listMenus, deleteMenu } from '@/api/system/menu';
  97. export default {
  98. name: 'SystemMenu',
  99. components: { MenuSearch, MenuEdit },
  100. data() {
  101. return {
  102. // 表格列配置
  103. columns: [
  104. {
  105. columnKey: 'index',
  106. type: 'index',
  107. width: 45,
  108. align: 'center',
  109. showOverflowTooltip: true,
  110. fixed: 'left'
  111. },
  112. {
  113. prop: 'name',
  114. label: '菜单名称',
  115. showOverflowTooltip: true,
  116. minWidth: 110,
  117. slot: 'name'
  118. },
  119. {
  120. prop: 'url',
  121. label: '路由地址',
  122. showOverflowTooltip: true,
  123. minWidth: 110
  124. },
  125. {
  126. prop: 'sort',
  127. label: '排序',
  128. align: 'center',
  129. showOverflowTooltip: true,
  130. width: 60
  131. },
  132. {
  133. prop: 'type',
  134. label: '类型',
  135. align: 'center',
  136. showOverflowTooltip: true,
  137. width: 60,
  138. slot: 'type'
  139. },
  140. {
  141. prop: 'createTime',
  142. label: '创建时间',
  143. showOverflowTooltip: true,
  144. minWidth: 110,
  145. formatter: (_row, _column, cellValue) => {
  146. return this.$util.toDateString(cellValue);
  147. }
  148. },
  149. {
  150. columnKey: 'action',
  151. label: '操作',
  152. width: 190,
  153. align: 'center',
  154. resizable: false,
  155. slot: 'action',
  156. showOverflowTooltip: true
  157. }
  158. ],
  159. // 当前编辑数据
  160. current: null,
  161. // 是否显示编辑弹窗
  162. showEdit: false,
  163. // 全部菜单数据
  164. menuList: [],
  165. // 上级菜单id
  166. parentId: null
  167. };
  168. },
  169. methods: {
  170. /* 表格数据源 */
  171. datasource({ where }) {
  172. return listMenus({ ...where });
  173. },
  174. /* 数据转为树形结构 */
  175. parseData(data) {
  176. // data.map(item=>{
  177. // item.id = parseInt(item.id)
  178. // })
  179. return this.$util.toTreeData({
  180. data: data,
  181. idField: 'id',
  182. parentIdField: 'parentId'
  183. });
  184. },
  185. /* 表格渲染完成回调 */
  186. onDone({ data }) {
  187. //if (!this.menuList.length) {
  188. this.menuList = data;
  189. //}
  190. },
  191. /* 刷新表格 */
  192. reload(where) {
  193. this.$refs.table.reload({ where: where });
  194. },
  195. /* 显示编辑 */
  196. openEdit(row, parentId) {
  197. this.current = row;
  198. this.parentId = parentId;
  199. this.showEdit = true;
  200. },
  201. /* 删除 */
  202. remove(row) {
  203. if (row.children?.length) {
  204. this.$message.error('请先删除子节点');
  205. return;
  206. }
  207. const loading = this.$loading({ lock: true });
  208. deleteMenu([row.id])
  209. .then(res => {
  210. loading.close();
  211. this.$message.success('菜单删除成功');
  212. this.reload();
  213. })
  214. .catch((e) => {
  215. loading.close();
  216. this.$message.error(e.message);
  217. });
  218. },
  219. /* 展开全部 */
  220. expandAll() {
  221. this.$refs.table.toggleRowExpansionAll(true);
  222. },
  223. /* 折叠全部 */
  224. foldAll() {
  225. this.$refs.table.toggleRowExpansionAll(false);
  226. },
  227. /* 判断是否是外链 */
  228. isUrl(url) {
  229. return !!(
  230. url &&
  231. (url.startsWith('http://') ||
  232. url.startsWith('https://') ||
  233. url.startsWith('//'))
  234. );
  235. },
  236. /* 判断是否是目录 */
  237. isDirectory(d) {
  238. return !!d.children?.length && !d.component;
  239. }
  240. }
  241. };
  242. </script>