index.vue 6.2 KB

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