index.vue 6.5 KB

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