editMenu.vue 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240
  1. <template>
  2. <div class="sys-menu-container">
  3. <el-dialog v-model="isShowDialog" width="769px">
  4. <template #header>
  5. <div style="font-size: large" v-drag="['.el-dialog','.el-dialog__header']">
  6. {{ title }}
  7. </div>
  8. </template>
  9. <el-form :model="ruleForm" :rules="ruleRules" ref="ruleFormRef" size="default" label-width="80px">
  10. <el-row :gutter="35">
  11. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24" class="mb20">
  12. <el-form-item label="上级菜单">
  13. <el-cascader :options="menuData"
  14. :props="{ checkStrictly: true, value: 'id', label: 'title' }" placeholder="请选择上级菜单"
  15. clearable class="w100" v-model="ruleForm.pid">
  16. <template #default="{ node, data }">
  17. <span>{{ data.title }}</span>
  18. <span v-if="!node.isLeaf"> ({{ data.children.length }}) </span>
  19. </template>
  20. </el-cascader>
  21. </el-form-item>
  22. </el-col>
  23. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24" class="mb20">
  24. <el-form-item label="菜单类型" prop="type">
  25. <el-radio-group v-model="ruleForm.type">
  26. <el-radio v-for="dict in menuType" :key="dict.value" :label="dict.value">
  27. {{ dict.label }}
  28. </el-radio>
  29. </el-radio-group>
  30. </el-form-item>
  31. </el-col>
  32. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  33. <el-form-item label="菜单名称" prop="title">
  34. <el-input v-model="ruleForm.title" placeholder="菜单名称" clearable></el-input>
  35. </el-form-item>
  36. </el-col>
  37. <template v-if="ruleForm.type === 1 || ruleForm.type === 2">
  38. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  39. <el-form-item label="路由名称">
  40. <el-input v-model="ruleForm.name" placeholder="路由名称" clearable></el-input>
  41. </el-form-item>
  42. </el-col>
  43. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  44. <el-form-item label="路由路径">
  45. <el-input v-model="ruleForm.path" placeholder="路由路径" clearable></el-input>
  46. </el-form-item>
  47. </el-col>
  48. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  49. <el-form-item label="组件路径">
  50. <el-input v-model="ruleForm.component" placeholder="组件路径" clearable></el-input>
  51. </el-form-item>
  52. </el-col>
  53. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  54. <el-form-item label="菜单图标">
  55. <IconSelector v-model="ruleForm.icon" placeholder="菜单图标" type="all" />
  56. </el-form-item>
  57. </el-col>
  58. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  59. <el-form-item label="重定向">
  60. <el-input v-model="ruleForm.redirect" placeholder="重定向地址" clearable></el-input>
  61. </el-form-item>
  62. </el-col>
  63. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  64. <el-form-item label="链接地址">
  65. <el-input v-model="ruleForm.outLink" placeholder="外链/内嵌时链接地址" clearable>
  66. </el-input>
  67. </el-form-item>
  68. </el-col>
  69. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  70. <el-form-item label="菜单排序">
  71. <el-input-number v-model="ruleForm.order" placeholder="排序" class="w100" />
  72. </el-form-item>
  73. </el-col>
  74. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  75. <el-form-item label="是否隐藏">
  76. <el-radio-group v-model="ruleForm.isHide">
  77. <el-radio :label="true">隐藏</el-radio>
  78. <el-radio :label="false">不隐藏</el-radio>
  79. </el-radio-group>
  80. </el-form-item>
  81. </el-col>
  82. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  83. <el-form-item label="是否缓存">
  84. <el-radio-group v-model="ruleForm.isKeepAlive">
  85. <el-radio :label="true">缓存</el-radio>
  86. <el-radio :label="false">不缓存</el-radio>
  87. </el-radio-group>
  88. </el-form-item>
  89. </el-col>
  90. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  91. <el-form-item label="是否固定">
  92. <el-radio-group v-model="ruleForm.isAffix">
  93. <el-radio :label="true">固定</el-radio>
  94. <el-radio :label="false">不固定</el-radio>
  95. </el-radio-group>
  96. </el-form-item>
  97. </el-col>
  98. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  99. <el-form-item label="是否内嵌">
  100. <el-radio-group v-model="ruleForm.isIframe">
  101. <el-radio :label="true">内嵌</el-radio>
  102. <el-radio :label="false">不内嵌</el-radio>
  103. </el-radio-group>
  104. </el-form-item>
  105. </el-col>
  106. </template>
  107. <template v-if="ruleForm.type === 3">
  108. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  109. <el-form-item label="权限标识">
  110. <el-input v-model="ruleForm.permission" placeholder="权限标识" clearable></el-input>
  111. </el-form-item>
  112. </el-col>
  113. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  114. <el-form-item label="菜单排序">
  115. <el-input-number v-model="ruleForm.order" placeholder="排序" class="w100" />
  116. </el-form-item>
  117. </el-col>
  118. </template>
  119. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" class="mb20">
  120. <el-form-item label="是否启用">
  121. <el-radio-group v-model="ruleForm.status">
  122. <el-radio :label="1">启用</el-radio>
  123. <el-radio :label="2">不启用</el-radio>
  124. </el-radio-group>
  125. </el-form-item>
  126. </el-col>
  127. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24" class="mb20">
  128. <el-form-item label="备注">
  129. <el-input v-model="ruleForm.remark" placeholder="请输入备注内容" clearable type="textarea">
  130. </el-input>
  131. </el-form-item>
  132. </el-col>
  133. </el-row>
  134. </el-form>
  135. <template #footer>
  136. <span class="dialog-footer">
  137. <el-button @click="cancel" size="default">取 消</el-button>
  138. <el-button type="primary" @click="submit" size="default">确 定</el-button>
  139. </span>
  140. </template>
  141. </el-dialog>
  142. </div>
  143. </template>
  144. <script lang="ts">
  145. import { reactive, toRefs, defineComponent, getCurrentInstance, ref, unref } from 'vue';
  146. import IconSelector from '/@/components/iconSelector/index.vue';
  147. import { getAPI } from '/@/utils/axios-utils';
  148. import { SysMenuApi } from '/@/api-services/api';
  149. export default defineComponent({
  150. name: 'sysEditMenu',
  151. components: { IconSelector },
  152. props: {
  153. // 弹窗标题
  154. title: {
  155. type: String,
  156. default: "",
  157. },
  158. // 菜单数据
  159. menuData: {
  160. type: Array,
  161. default: () => [],
  162. }
  163. },
  164. setup() {
  165. const { proxy } = getCurrentInstance() as any;
  166. const ruleFormRef = ref<HTMLElement | null>(null);
  167. const state = reactive({
  168. isShowDialog: false,
  169. ruleForm: {
  170. id: 0, // Id
  171. pid: 0, // 父节点Id
  172. type: 1, // 菜单类型
  173. name: '', // 路由名称(全局唯一)
  174. component: '', // 组件路径
  175. redirect: '', // 路由重定向(有子集 children 时)
  176. permission: '', // 权限标识
  177. path: '', // 路由路径
  178. title: '', // 菜单名称
  179. icon: '', // 菜单图标
  180. isHide: false, // 是否隐藏
  181. isKeepAlive: true, // 是否缓存
  182. isAffix: false, // 是否固定
  183. outLink: '', // 外链/内嵌时链接地址
  184. isIframe: false, // 是否内嵌
  185. order: 10, // 排序
  186. status: 1, // 是否启用
  187. remark: '', // 备注
  188. },
  189. menuType: [{ value: 1, label: "目录" }, { value: 2, label: "菜单" }, { value: 3, label: "按钮" }],
  190. ruleRules: {
  191. type: [{ required: true, message: "菜单类型不能为空", trigger: "blur" }],
  192. title: [{ required: true, message: "菜单名称不能为空", trigger: "blur" }],
  193. },
  194. });
  195. // 打开弹窗
  196. const openDialog = (row: any) => {
  197. state.ruleForm = row;
  198. state.isShowDialog = true;
  199. };
  200. // 关闭弹窗
  201. const closeDialog = () => {
  202. proxy.mittBus.emit("submitRefresh");
  203. state.isShowDialog = false;
  204. };
  205. // 取消
  206. const cancel = () => {
  207. state.isShowDialog = false;
  208. };
  209. // 提交
  210. const submit = () => {
  211. const formWrap = unref(ruleFormRef) as any;
  212. if (!formWrap) return;
  213. // 取父节点Id
  214. if (Array.isArray(state.ruleForm.pid))
  215. state.ruleForm.pid = state.ruleForm.pid[state.ruleForm.pid.length - 1];
  216. formWrap.validate(async () => {
  217. if (state.ruleForm.id != undefined && state.ruleForm.id > 0) {
  218. await getAPI(SysMenuApi).sysMenuUpdatePost(state.ruleForm);
  219. }
  220. else {
  221. await getAPI(SysMenuApi).sysMenuAddPost(state.ruleForm);
  222. }
  223. closeDialog();
  224. })
  225. };
  226. return {
  227. ruleFormRef,
  228. openDialog,
  229. closeDialog,
  230. cancel,
  231. submit,
  232. ...toRefs(state),
  233. };
  234. },
  235. });
  236. </script>