datascope.vue 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732
  1. <template>
  2. <basic-container>
  3. <avue-crud :option="option"
  4. :table-loading="loading"
  5. :data="data"
  6. ref="crud"
  7. v-model="form"
  8. :permission="permissionList"
  9. :before-open="beforeOpen"
  10. :before-close="beforeClose"
  11. @row-del="rowDel"
  12. @row-update="rowUpdate"
  13. @row-save="rowSave"
  14. @search-change="searchChange"
  15. @search-reset="searchReset"
  16. @selection-change="selectionChange"
  17. @current-change="currentChange"
  18. @size-change="sizeChange"
  19. @refresh-change="refreshChange"
  20. @on-load="onLoad"
  21. @tree-load="treeLoad">
  22. <template slot-scope="{row}" slot="menu">
  23. <el-button type="text"
  24. icon="el-icon-setting"
  25. size="small"
  26. v-if="permission.data_scope_setting"
  27. plain
  28. style="border: 0;background-color: transparent !important;"
  29. @click.stop="handleDataScope(row)">权限配置
  30. </el-button>
  31. </template>
  32. <template slot-scope="{row}" slot="source">
  33. <div style="text-align:center">
  34. <i :class="row.source"/>
  35. </div>
  36. </template>
  37. </avue-crud>
  38. <el-drawer :title="`[${scopeMenuName}] 数据权限配置`" :visible.sync="drawerVisible" :direction="direction"
  39. append-to-body
  40. :before-close="handleDrawerClose" size="1000px">
  41. <basic-container>
  42. <avue-crud :option="optionScope"
  43. :data="dataScope"
  44. :page="pageScope"
  45. v-model="formScope"
  46. :table-loading="scopeLoading"
  47. ref="crudScope"
  48. @row-del="rowDelScope"
  49. @row-update="rowUpdateScope"
  50. @row-save="rowSaveScope"
  51. :before-open="beforeOpenScope"
  52. @search-change="searchChangeScope"
  53. @search-reset="searchResetScope"
  54. @selection-change="selectionChangeScope"
  55. @current-change="currentChangeScope"
  56. @size-change="sizeChangeScope"
  57. @on-load="onLoadScope">
  58. <template slot="menuLeft">
  59. <el-button type="danger"
  60. size="small"
  61. icon="el-icon-delete"
  62. plain
  63. @click="handleDeleteScope">删 除
  64. </el-button>
  65. </template>
  66. <template slot-scope="{row}"
  67. slot="scopeType">
  68. <el-tag>{{row.scopeTypeName}}</el-tag>
  69. </template>
  70. </avue-crud>
  71. </basic-container>
  72. </el-drawer>
  73. </basic-container>
  74. </template>
  75. <script>
  76. import {
  77. add,
  78. remove,
  79. update,
  80. getLazyMenuList,
  81. getMenu
  82. } from "@/api/system/menu";
  83. import {
  84. addDataScope,
  85. removeDataScope,
  86. updateDataScope,
  87. getListDataScope,
  88. getMenuDataScope
  89. } from "@/api/system/scope";
  90. import {mapGetters} from "vuex";
  91. import iconList from "@/config/iconList";
  92. import func from "@/util/func";
  93. export default {
  94. data() {
  95. return {
  96. form: {},
  97. selectionList: [],
  98. query: {},
  99. loading: true,
  100. parentId: 0,
  101. page: {
  102. pageSize: 10,
  103. currentPage: 1,
  104. total: 0
  105. },
  106. drawerVisible: false,
  107. direction: 'rtl',
  108. scopeMenuId: 0,
  109. scopeMenuCode: '',
  110. scopeMenuName: "菜单",
  111. scopeLoading: false,
  112. menu: true,
  113. watchMode: true,
  114. option: {
  115. lazy: true,
  116. tip: false,
  117. simplePage: true,
  118. searchShow: true,
  119. searchMenuSpan: 6,
  120. dialogWidth: "60%",
  121. tree: true,
  122. border: true,
  123. index: true,
  124. selection: true,
  125. viewBtn: false,
  126. editBtn: false,
  127. addBtn: false,
  128. delBtn: false,
  129. menuWidth: 150,
  130. dialogClickModal: false,
  131. column: [
  132. {
  133. label: "菜单名称",
  134. prop: "name",
  135. search: true,
  136. rules: [
  137. {
  138. required: true,
  139. message: "请输入菜单名称",
  140. trigger: "blur"
  141. }
  142. ]
  143. },
  144. {
  145. label: "路由地址",
  146. prop: "path",
  147. rules: [
  148. {
  149. required: true,
  150. message: "请输入路由地址",
  151. trigger: "blur"
  152. }
  153. ]
  154. },
  155. {
  156. label: "上级菜单",
  157. prop: "parentId",
  158. type: "tree",
  159. dicUrl: "/api/blade-system/menu/tree",
  160. hide: true,
  161. props: {
  162. label: "title"
  163. },
  164. rules: [
  165. {
  166. required: false,
  167. message: "请选择上级菜单",
  168. trigger: "click"
  169. }
  170. ]
  171. },
  172. {
  173. label: "菜单图标",
  174. prop: "source",
  175. type: "icon",
  176. slot: true,
  177. width: 80,
  178. iconList: iconList,
  179. rules: [
  180. {
  181. required: true,
  182. message: "请输入菜单图标",
  183. trigger: "click"
  184. }
  185. ]
  186. },
  187. {
  188. label: "菜单编号",
  189. prop: "code",
  190. search: true,
  191. rules: [
  192. {
  193. required: true,
  194. message: "请输入菜单编号",
  195. trigger: "blur"
  196. }
  197. ]
  198. },
  199. {
  200. label: "菜单类型",
  201. prop: "category",
  202. type: "radio",
  203. dicData: [
  204. {
  205. label: "菜单",
  206. value: 1
  207. },
  208. {
  209. label: "按钮",
  210. value: 2
  211. }
  212. ],
  213. hide: true,
  214. rules: [
  215. {
  216. required: true,
  217. message: "请选择菜单类型",
  218. trigger: "blur"
  219. }
  220. ]
  221. },
  222. {
  223. label: "菜单别名",
  224. prop: "alias",
  225. rules: [
  226. {
  227. required: true,
  228. message: "请输入菜单别名",
  229. trigger: "blur"
  230. }
  231. ]
  232. },
  233. {
  234. label: "按钮功能",
  235. prop: "action",
  236. type: "radio",
  237. dicData: [
  238. {
  239. label: "工具栏",
  240. value: 0
  241. },
  242. {
  243. label: "操作栏",
  244. value: 1
  245. },
  246. {
  247. label: "工具操作栏",
  248. value: 2
  249. }
  250. ],
  251. hide: true,
  252. rules: [
  253. {
  254. required: true,
  255. message: "请选择按钮功能",
  256. trigger: "blur"
  257. }
  258. ]
  259. },
  260. {
  261. label: "菜单排序",
  262. prop: "sort",
  263. type: "number",
  264. width: 80,
  265. rules: [
  266. {
  267. required: true,
  268. message: "请输入菜单排序",
  269. trigger: "blur"
  270. }
  271. ]
  272. },
  273. {
  274. label: "新窗口",
  275. prop: "isOpen",
  276. type: "radio",
  277. dicData: [
  278. {
  279. label: "否",
  280. value: 0
  281. },
  282. {
  283. label: "是",
  284. value: 1
  285. },
  286. ],
  287. hide: true
  288. },
  289. {
  290. label: "菜单备注",
  291. prop: "remark",
  292. type: "textarea",
  293. span: 24,
  294. minRows: 6,
  295. hide: true
  296. }
  297. ]
  298. },
  299. data: [],
  300. formScope: {},
  301. selectionListScope: [],
  302. pageScope: {
  303. pageSize: 10,
  304. currentPage: 1,
  305. total: 0
  306. },
  307. optionScope: {
  308. tip: false,
  309. searchShow: true,
  310. searchMenuSpan: 6,
  311. border: true,
  312. index: true,
  313. viewBtn: true,
  314. selection: true,
  315. menuWidth: 200,
  316. dialogWidth: 900,
  317. dialogClickModal: false,
  318. column: [
  319. {
  320. label: "权限名称",
  321. prop: "scopeName",
  322. search: true,
  323. value: "",
  324. rules: [{
  325. required: true,
  326. message: "请输入数据权限名称",
  327. trigger: "blur"
  328. }]
  329. },
  330. {
  331. label: "权限编号",
  332. prop: "resourceCode",
  333. search: true,
  334. width: 100,
  335. rules: [{
  336. required: true,
  337. message: "请输入数据权限编号",
  338. trigger: "blur"
  339. }]
  340. },
  341. {
  342. label: "权限字段",
  343. prop: "scopeColumn",
  344. width: 130,
  345. rules: [{
  346. required: true,
  347. message: "请输入数据权限编号",
  348. trigger: "blur"
  349. }]
  350. },
  351. {
  352. label: "规则类型",
  353. type: "select",
  354. dicUrl: "/api/blade-system/dict/dictionary?code=data_scope_type",
  355. props: {
  356. label: "dictValue",
  357. value: "dictKey"
  358. },
  359. dataType: "number",
  360. slot: true,
  361. width: 140,
  362. prop: "scopeType",
  363. rules: [{
  364. required: true,
  365. message: "请输入通知类型",
  366. trigger: "blur"
  367. }]
  368. },
  369. {
  370. label: "可见字段",
  371. prop: "scopeField",
  372. span: 24,
  373. hide: true,
  374. value: "*",
  375. rules: [{
  376. required: true,
  377. message: "请输入数据权限可见的字段",
  378. trigger: "blur"
  379. }],
  380. },
  381. {
  382. label: "权限类名",
  383. prop: "scopeClass",
  384. span: 24,
  385. hide: true,
  386. rules: [{
  387. required: true,
  388. message: "请输入MybatisMapper对应方法的完整类名路径",
  389. trigger: "blur"
  390. }],
  391. },
  392. {
  393. label: "规则值",
  394. prop: "scopeValue",
  395. span: 24,
  396. minRows: 5,
  397. type: "textarea",
  398. display: true,
  399. hide: true,
  400. },
  401. {
  402. label: "备注",
  403. prop: "remark",
  404. span: 24,
  405. hide: true,
  406. },
  407. ]
  408. },
  409. dataScope: []
  410. };
  411. },
  412. watch: {
  413. 'formScope.scopeType'() {
  414. this.initScope();
  415. }
  416. },
  417. computed: {
  418. ...mapGetters(["permission"]),
  419. permissionList() {
  420. return {
  421. addBtn: this.vaildData(this.permission.menu_add, false),
  422. viewBtn: this.vaildData(this.permission.menu_view, false),
  423. delBtn: this.vaildData(this.permission.menu_delete, false),
  424. editBtn: this.vaildData(this.permission.menu_edit, false)
  425. };
  426. },
  427. ids() {
  428. let ids = [];
  429. this.selectionList.forEach(ele => {
  430. ids.push(ele.id);
  431. });
  432. return ids.join(",");
  433. },
  434. scopeIds() {
  435. let ids = [];
  436. this.selectionListScope.forEach(ele => {
  437. ids.push(ele.id);
  438. });
  439. return ids.join(",");
  440. }
  441. },
  442. methods: {
  443. initScope() {
  444. const scopeType = func.toInt(this.formScope.scopeType);
  445. const watchMode = this.watchMode;
  446. let column = "-", name = "暂无";
  447. if (scopeType === 1) {
  448. column = "-";
  449. name = "全部可见";
  450. } else if (scopeType === 2) {
  451. column = "create_user";
  452. name = "本人可见";
  453. } else if (scopeType === 3) {
  454. column = "create_dept";
  455. name = "所在机构可见";
  456. } else if (scopeType === 4) {
  457. column = "create_dept";
  458. name = "所在机构可见及子级可见";
  459. } else if (scopeType === 5) {
  460. column = "";
  461. name = "自定义";
  462. }
  463. this.$refs.crudScope.option.column.filter(item => {
  464. if (watchMode) {
  465. if (item.prop === "scopeName") {
  466. this.formScope.scopeName = `${this.scopeMenuName} [${name}]`;
  467. }
  468. if (item.prop === "resourceCode") {
  469. this.formScope.resourceCode = this.scopeMenuCode;
  470. }
  471. if (item.prop === "scopeColumn") {
  472. this.formScope.scopeColumn = column;
  473. }
  474. }
  475. if (item.prop === "scopeValue") {
  476. item.display = scopeType === 5;
  477. }
  478. });
  479. },
  480. // 菜单管理模块
  481. rowSave(row, done, loading) {
  482. add(row).then(() => {
  483. this.onLoad(this.page);
  484. this.$message({
  485. type: "success",
  486. message: "操作成功!"
  487. });
  488. done();
  489. }, error => {
  490. window.console.log(error);
  491. loading();
  492. });
  493. },
  494. rowUpdate(row, index, done, loading) {
  495. update(row).then(() => {
  496. this.onLoad(this.page);
  497. this.$message({
  498. type: "success",
  499. message: "操作成功!"
  500. });
  501. done();
  502. }, error => {
  503. window.console.log(error);
  504. loading();
  505. });
  506. },
  507. rowDel(row) {
  508. this.$confirm("确定将选择数据删除?", {
  509. confirmButtonText: "确定",
  510. cancelButtonText: "取消",
  511. type: "warning"
  512. })
  513. .then(() => {
  514. return remove(row.id);
  515. })
  516. .then(() => {
  517. this.onLoad(this.page);
  518. this.$message({
  519. type: "success",
  520. message: "操作成功!"
  521. });
  522. });
  523. },
  524. searchReset() {
  525. this.query = {};
  526. this.parentId = 0;
  527. this.onLoad(this.page);
  528. },
  529. searchChange(params, done) {
  530. this.query = params;
  531. this.parentId = '';
  532. this.page.currentPage = 1;
  533. this.onLoad(this.page, params);
  534. done();
  535. },
  536. selectionChange(list) {
  537. this.selectionList = list;
  538. },
  539. selectionClear() {
  540. this.selectionList = [];
  541. this.$refs.crud.toggleSelection();
  542. },
  543. handleDelete() {
  544. if (this.selectionList.length === 0) {
  545. this.$message.warning("请选择至少一条数据");
  546. return;
  547. }
  548. this.$confirm("确定将选择数据删除?", {
  549. confirmButtonText: "确定",
  550. cancelButtonText: "取消",
  551. type: "warning"
  552. })
  553. .then(() => {
  554. return remove(this.ids);
  555. })
  556. .then(() => {
  557. this.onLoad(this.page);
  558. this.$message({
  559. type: "success",
  560. message: "操作成功!"
  561. });
  562. this.$refs.crud.toggleSelection();
  563. });
  564. },
  565. beforeOpen(done, type) {
  566. if (["edit", "view"].includes(type)) {
  567. getMenu(this.form.id).then(res => {
  568. this.form = res.data.data;
  569. });
  570. }
  571. done();
  572. },
  573. beforeClose(done) {
  574. this.formScope = {};
  575. done();
  576. },
  577. currentChange(currentPage) {
  578. this.page.currentPage = currentPage;
  579. },
  580. sizeChange(pageSize) {
  581. this.page.pageSize = pageSize;
  582. },
  583. refreshChange() {
  584. this.onLoad(this.page, this.query);
  585. },
  586. onLoad(page, params = {}) {
  587. this.loading = true;
  588. getLazyMenuList(this.parentId, Object.assign(params, this.query)).then(res => {
  589. this.data = res.data.data;
  590. this.loading = false;
  591. this.selectionClear();
  592. });
  593. },
  594. treeLoad(tree, treeNode, resolve) {
  595. const parentId = tree.id;
  596. getLazyMenuList(parentId).then(res => {
  597. resolve(res.data.data);
  598. });
  599. },
  600. // 数据权限模块
  601. handleDataScope(row) {
  602. this.drawerVisible = true;
  603. this.scopeMenuId = row.id;
  604. this.scopeMenuCode = row.code;
  605. this.scopeMenuName = row.name;
  606. this.onLoadScope(this.pageScope)
  607. },
  608. handleDrawerClose(hide) {
  609. hide();
  610. },
  611. rowSaveScope(row, done, loading) {
  612. row = {
  613. ...row,
  614. menuId: this.scopeMenuId,
  615. };
  616. addDataScope(row).then(() => {
  617. this.onLoadScope(this.pageScope);
  618. this.$message({
  619. type: "success",
  620. message: "操作成功!"
  621. });
  622. done();
  623. }, error => {
  624. window.console.log(error);
  625. loading();
  626. });
  627. },
  628. rowUpdateScope(row, index, done, loading) {
  629. row = {
  630. ...row,
  631. menuId: this.scopeMenuId,
  632. };
  633. updateDataScope(row).then(() => {
  634. this.onLoadScope(this.pageScope);
  635. this.$message({
  636. type: "success",
  637. message: "操作成功!"
  638. });
  639. done();
  640. }, error => {
  641. window.console.log(error);
  642. loading();
  643. });
  644. },
  645. rowDelScope(row) {
  646. this.$confirm("确定将选择数据删除?", {
  647. confirmButtonText: "确定",
  648. cancelButtonText: "取消",
  649. type: "warning"
  650. })
  651. .then(() => {
  652. return removeDataScope(row.id);
  653. })
  654. .then(() => {
  655. this.onLoadScope(this.pageScope);
  656. this.$message({
  657. type: "success",
  658. message: "操作成功!"
  659. });
  660. });
  661. },
  662. handleDeleteScope() {
  663. if (this.selectionListScope.length === 0) {
  664. this.$message.warning("请选择至少一条数据");
  665. return;
  666. }
  667. this.$confirm("确定将选择数据删除?", {
  668. confirmButtonText: "确定",
  669. cancelButtonText: "取消",
  670. type: "warning"
  671. })
  672. .then(() => {
  673. return removeDataScope(this.scopeIds);
  674. })
  675. .then(() => {
  676. this.onLoadScope(this.pageScope);
  677. this.$message({
  678. type: "success",
  679. message: "操作成功!"
  680. });
  681. this.$refs.crudScope.toggleSelection();
  682. });
  683. },
  684. beforeOpenScope(done, type) {
  685. if (["add"].includes(type)) {
  686. this.watchMode = true;
  687. this.initScope();
  688. }
  689. if (["edit", "view"].includes(type)) {
  690. this.watchMode = false;
  691. getMenuDataScope(this.formScope.id).then(res => {
  692. this.formScope = res.data.data;
  693. });
  694. }
  695. done();
  696. },
  697. searchResetScope() {
  698. this.onLoadScope(this.pageScope);
  699. },
  700. searchChangeScope(params, done) {
  701. this.onLoadScope(this.pageScope, params);
  702. done();
  703. },
  704. selectionChangeScope(list) {
  705. this.selectionListScope = list;
  706. },
  707. currentChangeScope(currentPage) {
  708. this.pageScope.currentPage = currentPage;
  709. },
  710. sizeChangeScope(pageSize) {
  711. this.pageScope.pageSize = pageSize;
  712. },
  713. onLoadScope(page, params = {}) {
  714. this.scopeLoading = true;
  715. const values = {
  716. ...params,
  717. menuId: this.scopeMenuId,
  718. }
  719. getListDataScope(page.currentPage, page.pageSize, Object.assign(values, this.query)).then(res => {
  720. const data = res.data.data;
  721. this.pageScope.total = data.total;
  722. this.dataScope = data.records;
  723. this.selectionListScope = [];
  724. this.scopeLoading = false;
  725. });
  726. },
  727. }
  728. };
  729. </script>