index.vue 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493
  1. <template>
  2. <div class="box pagePadding MenuOperator">
  3. <div class="MenuOperator-main" v-if="powers.some((i) => i == '001')">
  4. <el-row class="MenuOperator-search clear" style="padding: 10px 0">
  5. <!-- <el-col :span="24" style="width: 90px; float: right">
  6. <el-button
  7. type="success"
  8. :size="searchSize"
  9. style="float: right"
  10. @click="btnVisible = true"
  11. >
  12. 添加按钮
  13. </el-button>
  14. </el-col> -->
  15. <el-col :span="24" style="width: 66px; float: right">
  16. <el-button
  17. v-if="powers.some((i) => i == '003')"
  18. type="success"
  19. :size="searchSize"
  20. style="float: right"
  21. @click="openModal(0, '0', 1, false, true, {})"
  22. >
  23. 添加
  24. </el-button>
  25. </el-col>
  26. <el-col :span="3" style="width: 66px; float: right">
  27. <el-button
  28. type="primary"
  29. :size="searchSize"
  30. style="float: right; margin-left: 5px"
  31. @click="searchList"
  32. >
  33. 刷新
  34. </el-button>
  35. </el-col>
  36. </el-row>
  37. <div style="padding: 50px 0 0 0">
  38. <el-table
  39. :data="tableData"
  40. style="width: 100%; margin-bottom: 20px"
  41. row-key="id"
  42. v-loading="loading"
  43. border
  44. :default-expand-all="false"
  45. size="mini"
  46. :tree-props="{ children: 'child', hasChildren: 'hasChildren' }"
  47. >
  48. <el-table-column prop="menu_name" label="菜单名称" width="180" />
  49. <el-table-column prop="name" label="图标" width="50">
  50. <template slot-scope="scope"> <i :class="scope.row.menu_img" /></template
  51. ></el-table-column>
  52. <el-table-column prop="menu_route" label="path" width="180" />
  53. <el-table-column prop="weight" label="权重" width="80" />
  54. <el-table-column prop="address" label="类型" width="100">
  55. <template slot-scope="scope">
  56. <el-tag
  57. :size="tablebtnSize"
  58. :type="Number(scope.row.pid) == 0 ? 'success' : ''"
  59. v-text="Number(scope.row.pid) == 0 ? '菜单' : '页面'"
  60. ></el-tag> </template
  61. ></el-table-column>
  62. <el-table-column prop="status" label="状态" width="100">
  63. <template slot-scope="scope">
  64. <el-tag
  65. :size="tablebtnSize"
  66. :type="Number(scope.row.status) == 0 ? 'warning' : ''"
  67. v-text="
  68. (
  69. statusOptions.find((item) => item.id == Number(scope.row.status)) ||
  70. {}
  71. ).label || '--'
  72. "
  73. ></el-tag>
  74. </template>
  75. </el-table-column>
  76. <el-table-column prop="menu_url" label="文件路径" />
  77. <el-table-column prop="address" label="操作" width="180">
  78. <template slot-scope="scope">
  79. <el-tooltip
  80. v-if="powers.some((i) => i == '007')"
  81. effect="dark"
  82. content="详情"
  83. placement="top"
  84. >
  85. <i
  86. class="el-icon-view tb-icon"
  87. size="mini"
  88. @click="
  89. openModal(
  90. scope.row.id,
  91. scope.row.pid,
  92. Number(scope.row.pid) === 0 ? 1 : 2,
  93. true,
  94. false,
  95. scope.row
  96. )
  97. "
  98. ></i>
  99. </el-tooltip>
  100. <el-tooltip
  101. v-if="powers.some((i) => i == '005')"
  102. effect="dark"
  103. content="修改"
  104. placement="top"
  105. >
  106. <i
  107. class="el-icon-edit tb-icon"
  108. size="mini"
  109. @click="
  110. openModal(
  111. scope.row.id,
  112. scope.row.pid,
  113. Number(scope.row.pid) === 0 ? 1 : 2,
  114. false,
  115. false,
  116. scope.row
  117. )
  118. "
  119. ></i>
  120. </el-tooltip>
  121. <el-tooltip
  122. v-if="powers.some((i) => i == '004') && scope.row.status + '' === '1'"
  123. effect="dark"
  124. content="禁用"
  125. placement="top"
  126. >
  127. <i
  128. class="el-icon-video-pause tb-icon"
  129. size="mini"
  130. @click="
  131. changeStatus(
  132. scope.row.id,
  133. scope.row.pid,
  134. scope.row.level,
  135. scope.row.status
  136. )
  137. "
  138. ></i>
  139. </el-tooltip>
  140. <el-tooltip
  141. v-if="powers.some((i) => i == '004') && scope.row.status + '' === '0'"
  142. effect="dark"
  143. content="启用"
  144. placement="top"
  145. >
  146. <i
  147. class="el-icon-video-play tb-icon"
  148. size="mini"
  149. @click="
  150. changeStatus(
  151. scope.row.id,
  152. scope.row.pid,
  153. scope.row.level,
  154. scope.row.status
  155. )
  156. "
  157. ></i>
  158. </el-tooltip>
  159. <el-tooltip
  160. v-if="powers.some((i) => i == '006')"
  161. effect="dark"
  162. content="删除"
  163. placement="top"
  164. >
  165. <i
  166. class="el-icon-delete tb-icon"
  167. size="mini"
  168. @click="
  169. deleteItem(
  170. scope.row.id,
  171. scope.row.pid,
  172. scope.row.level,
  173. scope.row.status
  174. )
  175. "
  176. ></i>
  177. </el-tooltip>
  178. <el-tooltip
  179. v-if="powers.some((i) => i == '003') && Number(scope.row.pid) === 0"
  180. effect="dark"
  181. content="添加"
  182. placement="top"
  183. >
  184. <i
  185. class="el-icon-circle-plus-outline tb-icon"
  186. size="mini"
  187. @click="openModal(0, scope.row.id, 2, false, true, {})"
  188. ></i>
  189. </el-tooltip>
  190. </template>
  191. </el-table-column>
  192. </el-table>
  193. </div>
  194. <add-edit
  195. :id="modelId"
  196. :show-model="showModel"
  197. :level="level"
  198. :parent-ids="parentIds"
  199. :is-add="isAdd"
  200. :form-data="formData"
  201. :type="parmValue.type"
  202. :is-detail="isDetail"
  203. @refresh="submitRefresh"
  204. @cancel="showModel = false"
  205. />
  206. </div>
  207. <div v-else>
  208. <no-auth></no-auth>
  209. </div>
  210. <ButtonModal :visible.sync="btnVisible" />
  211. </div>
  212. </template>
  213. <script>
  214. import mixinPage from "@/mixins/elPaginationHandle";
  215. import asyncRequest from "@/apis/service/system/menu";
  216. import addEdit from "@/views/system/menuOperator/addEdit";
  217. import ButtonModal from "./button-modal.vue"
  218. import resToken from "@/mixins/resToken";
  219. import { mapGetters } from "vuex";
  220. export default {
  221. name: "MenuOperator",
  222. components: {
  223. addEdit,
  224. ButtonModal
  225. },
  226. mixins: [mixinPage, resToken],
  227. computed: {
  228. ...mapGetters(["tablebtnSize", "searchSize", "size"]),
  229. powers() {
  230. const { btnList } = this.$store.getters;
  231. const tran = btnList.find((item) => item.menu_route == "menuOperator") || {};
  232. const { action } = tran ?? {};
  233. return action ?? [];
  234. },
  235. },
  236. data() {
  237. return {
  238. btnVisible:false,
  239. // 状态
  240. statusOptions: [
  241. { id: 0, label: "禁用" },
  242. { id: 1, label: "启用" },
  243. ],
  244. loading: true,
  245. showModel: false,
  246. showInterface: false,
  247. isDetail: false,
  248. isAdd: true,
  249. level: 1,
  250. modelId: 0,
  251. formData: {},
  252. parentIds: "",
  253. parmValue: {
  254. parentId: 0, // 父级Id
  255. code: "", // 编码
  256. status: "", // 状态(1启用 2禁用)
  257. type: 1, // 类型(1.运营菜单 2.物业菜单)
  258. menuCategory: "", // 菜单类别(1.导航菜单 2.按钮)
  259. current: 1, // 页码
  260. size: 15, // 每页显示条数
  261. },
  262. tableData: [],
  263. };
  264. },
  265. mounted() {
  266. this.searchList();
  267. },
  268. methods: {
  269. openModal(id, parentIds, level, isDetail, isAdd, formData) {
  270. this.showModel = true;
  271. this.modelId = id;
  272. this.level = level;
  273. this.parentIds = parentIds;
  274. this.isDetail = isDetail;
  275. this.isAdd = isAdd;
  276. this.formData = formData;
  277. },
  278. async submitRefresh(id, parentIds, level) {
  279. if (level < 3) {
  280. await this.searchList();
  281. } else {
  282. this.getchildList(parentIds);
  283. }
  284. },
  285. async changeStatus(id, parentIds, level, status) {
  286. await this.$confirm(`确定要${status + "" === "0" ? "启用" : "禁用"}?`, {
  287. confirmButtonText: "确定",
  288. cancelButtonText: "取消",
  289. type: "warning",
  290. })
  291. .then(async () => {
  292. const model = {
  293. id: id,
  294. status: status + "" === "1" ? "0" : "1",
  295. };
  296. let res = {};
  297. if (level === 3) {
  298. res = await asyncRequest.actionstatus(model);
  299. } else {
  300. res = await asyncRequest.menustatus(model);
  301. }
  302. if (res && res.code === 0) {
  303. this.$notify.success({
  304. title: "修改成功!",
  305. message: "",
  306. });
  307. if (level === 3) {
  308. this.getchildList(parentIds);
  309. } else {
  310. await this.searchList();
  311. }
  312. } else if (res && res.code >= 100 && res.code <= 104) {
  313. await this.logout();
  314. } else {
  315. this.$message.warning(res.message);
  316. }
  317. })
  318. .catch(() => {
  319. console.log("取消");
  320. });
  321. },
  322. async deleteItem(id, parentIds, level, status) {
  323. await this.$confirm(`确定要删除?`, {
  324. confirmButtonText: "确定",
  325. cancelButtonText: "取消",
  326. type: "warning",
  327. })
  328. .then(async () => {
  329. const model = {
  330. id: id,
  331. };
  332. let res = {};
  333. if (level === 3) {
  334. res = await asyncRequest.actiondel(model);
  335. } else {
  336. res = await asyncRequest.menudel(model);
  337. }
  338. if (res && res.code === 0) {
  339. this.$notify.success({
  340. title: "删除成功!",
  341. message: "",
  342. });
  343. if (level === 3) {
  344. this.getchildList(parentIds);
  345. } else {
  346. await this.searchList();
  347. }
  348. } else if (res && res.code >= 100 && res.code <= 104) {
  349. await this.logout();
  350. }
  351. })
  352. .catch(() => {
  353. console.log("取消");
  354. });
  355. },
  356. openChildren(id, parentIds, level, isOpen) {
  357. if (isOpen) {
  358. if (level === 1) {
  359. this.tableData = this.tableData.map((v) => {
  360. if (v.id + "" === id + "") {
  361. v.child = [];
  362. }
  363. return v;
  364. });
  365. } else {
  366. this.tableData = this.tableData.map((v) => {
  367. if (v.id + "" === parentIds + "") {
  368. v.child = v.child.map((val) => {
  369. if (val.id === id) val.child = [];
  370. return val;
  371. });
  372. }
  373. return v;
  374. });
  375. }
  376. } else {
  377. if (level === 1) {
  378. this.concatData(id, "0", level);
  379. } else {
  380. this.getchildList(id);
  381. }
  382. }
  383. },
  384. async getchildList(id) {
  385. this.loading = true;
  386. const res = await asyncRequest.actionList({ id: id });
  387. let tableData = [];
  388. if (res && res.code === 0 && res.data) {
  389. tableData = res.data;
  390. // tableData = tableData.map((val) => {
  391. // let index = btnList.findIndex((e) => val.action_code === e.code);
  392. // if (index !== -1) {
  393. // val.menu_name = btnList[index].name;
  394. // }
  395. // return val;
  396. // });
  397. } else if (res && res.code >= 100 && res.code <= 104) {
  398. await this.logout();
  399. } else {
  400. this.$message.warning(res.message);
  401. }
  402. let arr = JSON.parse(JSON.stringify(this.tableData));
  403. arr = arr.map((val, index) => {
  404. if (val && val.child && val.child.length > 0) {
  405. val.child.forEach((el, eli) => {
  406. if (id === el.id) {
  407. el.child = tableData;
  408. }
  409. return el;
  410. });
  411. }
  412. return val;
  413. });
  414. this.tableData = arr;
  415. this.loading = false;
  416. },
  417. async concatData(id, parentIds, level) {
  418. this.loading = true;
  419. const res = await asyncRequest.list({});
  420. let tableData = [];
  421. if (res && res.code === 0 && res.data) {
  422. tableData = res.data;
  423. } else if (res && res.code >= 100 && res.code <= 104) {
  424. await this.logout();
  425. } else {
  426. this.$message.warning(res.message);
  427. }
  428. let arr = JSON.parse(JSON.stringify(this.tableData));
  429. arr = arr.map((val, index) => {
  430. if (level === 1 ? id : parentIds + "" === val.id + "") {
  431. val.child = tableData[index].child;
  432. }
  433. return val;
  434. });
  435. this.tableData = arr;
  436. this.loading = false;
  437. },
  438. async searchList() {
  439. this.loading = true;
  440. const res = await asyncRequest.list({});
  441. this.tableData = [];
  442. if (res && res.code === 0 && res.data) {
  443. this.tableData = res.data;
  444. } else if (res && res.code >= 100 && res.code <= 104) {
  445. await this.logout();
  446. } else {
  447. this.tableData = [];
  448. }
  449. this.loading = false;
  450. },
  451. },
  452. };
  453. </script>
  454. <style lang="scss" scoped>
  455. .MenuOperator {
  456. position: relative;
  457. height: 100%;
  458. width: 100%;
  459. box-sizing: border-box;
  460. .MenuOperator-main {
  461. position: relative;
  462. height: 100%;
  463. width: 100%;
  464. box-sizing: border-box;
  465. .MenuOperator-search {
  466. position: absolute;
  467. top: 0;
  468. left: 0;
  469. width: 100%;
  470. z-index: 2;
  471. background: #fff;
  472. }
  473. .MenuOperator-show {
  474. position: relative;
  475. height: 100%;
  476. padding: 40px 0 0 0;
  477. overflow-y: scroll;
  478. }
  479. }
  480. }
  481. </style>