menu.data.ts 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428
  1. import { BasicColumn } from '/@/components/Table';
  2. import { FormSchema } from '/@/components/Table';
  3. import { h } from 'vue';
  4. import { Icon } from '/@/components/Icon';
  5. import { duplicateCheck } from '../user/user.api';
  6. import { ajaxGetDictItems, checkPermDuplication } from './menu.api';
  7. import { render } from '/@/utils/common/renderUtils';
  8. const isDir = (type) => type === 0;
  9. const isMenu = (type) => type === 1;
  10. const isButton = (type) => type === 2;
  11. // 定义可选择的组件类型
  12. export enum ComponentTypes {
  13. Default = 'layouts/default/index',
  14. IFrame = 'sys/iframe/FrameBlank',
  15. // IFrame = 'sys/iframe/index',
  16. }
  17. export const columns: BasicColumn[] = [
  18. {
  19. title: '菜单名称',
  20. dataIndex: 'name',
  21. width: 200,
  22. align: 'left',
  23. },
  24. {
  25. title: '菜单类型',
  26. dataIndex: 'menuType',
  27. width: 150,
  28. customRender: ({ text }) => {
  29. return render.renderDict(text, 'menu_type');
  30. },
  31. },
  32. {
  33. title: '图标',
  34. dataIndex: 'icon',
  35. width: 50,
  36. customRender: ({ record }) => {
  37. return h(Icon, { icon: record.icon });
  38. },
  39. },
  40. {
  41. title: '组件',
  42. dataIndex: 'component',
  43. align: 'left',
  44. width: 150,
  45. },
  46. {
  47. title: '路径',
  48. dataIndex: 'url',
  49. align: 'left',
  50. width: 150,
  51. },
  52. {
  53. title: '排序',
  54. dataIndex: 'sortNo',
  55. width: 50,
  56. },
  57. ];
  58. export const searchFormSchema: FormSchema[] = [
  59. {
  60. field: 'name',
  61. label: '菜单名称',
  62. component: 'Input',
  63. colProps: { span: 8 },
  64. },
  65. ];
  66. export const formSchema: FormSchema[] = [
  67. {
  68. label: 'id',
  69. field: 'id',
  70. component: 'Input',
  71. show: false,
  72. },
  73. {
  74. field: 'menuType',
  75. label: '菜单类型',
  76. component: 'RadioButtonGroup',
  77. defaultValue: 0,
  78. componentProps: ({ formActionType, formModel }) => {
  79. return {
  80. options: [
  81. { label: '一级菜单', value: 0 },
  82. { label: '子菜单', value: 1 },
  83. { label: '按钮/权限', value: 2 },
  84. ],
  85. onChange: (e) => {
  86. const { updateSchema, clearValidate } = formActionType;
  87. const label = isButton(e) ? '按钮/权限' : '菜单名称';
  88. //清除校验
  89. clearValidate();
  90. updateSchema([
  91. {
  92. field: 'name',
  93. label: label,
  94. },
  95. {
  96. field: 'url',
  97. required: !isButton(e),
  98. },
  99. ]);
  100. //update-begin---author:wangshuai ---date:20220729 for:[VUEN-1834]只有一级菜单,才默认值,子菜单的时候,清空------------
  101. if (isMenu(e) && !formModel.id && formModel.component == 'layouts/RouteView') {
  102. formModel.component = '';
  103. }
  104. //update-end---author:wangshuai ---date:20220729 for:[VUEN-1834]只有一级菜单,才默认值,子菜单的时候,清空------------
  105. },
  106. };
  107. },
  108. },
  109. {
  110. field: 'name',
  111. label: '菜单名称',
  112. component: 'Input',
  113. required: true,
  114. },
  115. {
  116. field: 'parentId',
  117. label: '上级菜单',
  118. component: 'TreeSelect',
  119. required: true,
  120. componentProps: {
  121. replaceFields: {
  122. title: 'name',
  123. key: 'id',
  124. value: 'id',
  125. },
  126. dropdownStyle: {
  127. maxHeight: '50vh',
  128. },
  129. getPopupContainer: (node) => node.parentNode,
  130. },
  131. ifShow: ({ values }) => !isDir(values.menuType),
  132. },
  133. {
  134. field: 'url',
  135. label: '访问路径',
  136. component: 'Input',
  137. required: true,
  138. ifShow: ({ values }) => !(values.component === ComponentTypes.IFrame && values.internalOrExternal) && values.menuType !== 2,
  139. //update-begin-author:zyf date:2022-11-02 for: 聚合路由允许路径重复
  140. dynamicRules: ({ model, schema }) => {
  141. return checkPermDuplication(model, schema, true);
  142. },
  143. //update-end-author:zyf date:2022-11-02 for: 聚合路由允许路径重复
  144. },
  145. {
  146. field: 'component',
  147. label: '前端组件',
  148. component: 'Input',
  149. componentProps: {
  150. placeholder: '请输入前端组件',
  151. },
  152. defaultValue: 'layouts/RouteView',
  153. required: true,
  154. ifShow: ({ values }) => !isButton(values.menuType),
  155. },
  156. {
  157. field: 'componentName',
  158. label: '组件名称',
  159. component: 'Input',
  160. componentProps: {
  161. placeholder: '请输入组件名称',
  162. },
  163. helpMessage: [
  164. '此处名称应和vue组件的name属性保持一致。',
  165. '组件名称不能重复,主要用于路由缓存功能。',
  166. '如果组件名称和vue组件的name属性不一致,则会导致路由缓存失效。',
  167. '非必填,留空则会根据访问路径自动生成。',
  168. ],
  169. defaultValue: '',
  170. ifShow: ({ values }) => !isButton(values.menuType),
  171. },
  172. {
  173. field: 'frameSrc',
  174. label: 'Iframe地址',
  175. component: 'Input',
  176. rules: [
  177. { required: true, message: '请输入Iframe地址' },
  178. { type: 'url', message: '请输入正确的url地址' },
  179. ],
  180. ifShow: ({ values }) => !isButton(values.menuType) && values.component === ComponentTypes.IFrame,
  181. },
  182. {
  183. field: 'redirect',
  184. label: '默认跳转地址',
  185. component: 'Input',
  186. ifShow: ({ values }) => isDir(values.menuType),
  187. },
  188. {
  189. field: 'perms',
  190. label: '授权标识',
  191. component: 'Input',
  192. ifShow: ({ values }) => isButton(values.menuType),
  193. dynamicRules: ({ model }) => {
  194. return [
  195. {
  196. required: false,
  197. validator: (_, value) => {
  198. return new Promise((resolve, reject) => {
  199. const params = {
  200. tableName: 'sys_permission',
  201. fieldName: 'perms',
  202. fieldVal: value,
  203. dataId: model.id,
  204. };
  205. duplicateCheck(params)
  206. .then((res) => {
  207. res.success ? resolve() : reject(res.message || '校验失败');
  208. })
  209. .catch((err) => {
  210. reject(err.message || '校验失败');
  211. });
  212. });
  213. },
  214. },
  215. ];
  216. },
  217. },
  218. {
  219. field: 'permsType',
  220. label: '授权策略',
  221. component: 'RadioGroup',
  222. defaultValue: '1',
  223. helpMessage: ['可见/可访问(授权后可见/可访问)', '可编辑(未授权时禁用)'],
  224. componentProps: {
  225. options: [
  226. { label: '可见/可访问', value: '1' },
  227. { label: '可编辑', value: '2' },
  228. ],
  229. },
  230. ifShow: ({ values }) => isButton(values.menuType),
  231. },
  232. {
  233. field: 'status',
  234. label: '状态',
  235. component: 'RadioGroup',
  236. defaultValue: '1',
  237. componentProps: {
  238. options: [
  239. { label: '有效', value: '1' },
  240. { label: '无效', value: '0' },
  241. ],
  242. },
  243. ifShow: ({ values }) => isButton(values.menuType),
  244. },
  245. {
  246. field: 'icon',
  247. label: '菜单图标',
  248. component: 'IconPicker',
  249. ifShow: ({ values }) => !isButton(values.menuType),
  250. },
  251. {
  252. field: 'sortNo',
  253. label: '排序',
  254. component: 'InputNumber',
  255. defaultValue: 1,
  256. ifShow: ({ values }) => !isButton(values.menuType),
  257. },
  258. {
  259. field: 'route',
  260. label: '是否路由菜单',
  261. component: 'Switch',
  262. defaultValue: true,
  263. componentProps: {
  264. checkedChildren: '是',
  265. unCheckedChildren: '否',
  266. },
  267. ifShow: ({ values }) => !isButton(values.menuType),
  268. },
  269. {
  270. field: 'hidden',
  271. label: '隐藏路由',
  272. component: 'Switch',
  273. defaultValue: 0,
  274. componentProps: {
  275. checkedChildren: '是',
  276. unCheckedChildren: '否',
  277. },
  278. ifShow: ({ values }) => !isButton(values.menuType),
  279. },
  280. {
  281. field: 'hideTab',
  282. label: '隐藏Tab',
  283. component: 'Switch',
  284. defaultValue: 0,
  285. componentProps: {
  286. checkedChildren: '是',
  287. unCheckedChildren: '否',
  288. },
  289. ifShow: ({ values }) => !isButton(values.menuType),
  290. },
  291. {
  292. field: 'keepAlive',
  293. label: '是否缓存路由',
  294. component: 'Switch',
  295. defaultValue: false,
  296. componentProps: {
  297. checkedChildren: '是',
  298. unCheckedChildren: '否',
  299. },
  300. ifShow: ({ values }) => !isButton(values.menuType),
  301. },
  302. {
  303. field: 'alwaysShow',
  304. label: '聚合路由',
  305. component: 'Switch',
  306. defaultValue: false,
  307. componentProps: {
  308. checkedChildren: '是',
  309. unCheckedChildren: '否',
  310. },
  311. ifShow: ({ values }) => !isButton(values.menuType),
  312. },
  313. {
  314. field: 'internalOrExternal',
  315. label: '打开方式',
  316. component: 'Switch',
  317. defaultValue: false,
  318. componentProps: {
  319. checkedChildren: '外部',
  320. unCheckedChildren: '内部',
  321. },
  322. ifShow: ({ values }) => !isButton(values.menuType),
  323. },
  324. {
  325. field: 'ver',
  326. label: '菜单是否可点',
  327. component: 'RadioButtonGroup',
  328. defaultValue: '1',
  329. componentProps: {
  330. options: [
  331. { label: '是', value: '1' },
  332. { label: '否', value: '0' },
  333. ],
  334. },
  335. },
  336. ];
  337. export const dataRuleColumns: BasicColumn[] = [
  338. {
  339. title: '规则名称',
  340. dataIndex: 'ruleName',
  341. width: 150,
  342. },
  343. {
  344. title: '规则字段',
  345. dataIndex: 'ruleColumn',
  346. width: 100,
  347. },
  348. {
  349. title: '规则值',
  350. dataIndex: 'ruleValue',
  351. width: 100,
  352. },
  353. ];
  354. export const dataRuleSearchFormSchema: FormSchema[] = [
  355. {
  356. field: 'ruleName',
  357. label: '规则名称',
  358. component: 'Input',
  359. colProps: { span: 6 },
  360. },
  361. {
  362. field: 'ruleValue',
  363. label: '规则值',
  364. component: 'Input',
  365. colProps: { span: 6 },
  366. },
  367. ];
  368. export const dataRuleFormSchema: FormSchema[] = [
  369. {
  370. label: 'id',
  371. field: 'id',
  372. component: 'Input',
  373. show: false,
  374. },
  375. {
  376. field: 'ruleName',
  377. label: '规则名称',
  378. component: 'Input',
  379. required: true,
  380. },
  381. {
  382. field: 'ruleColumn',
  383. label: '规则字段',
  384. component: 'Input',
  385. ifShow: ({ values }) => {
  386. return values.ruleConditions !== 'USE_SQL_RULES';
  387. },
  388. },
  389. {
  390. field: 'ruleConditions',
  391. label: '条件规则',
  392. required: true,
  393. component: 'ApiSelect',
  394. componentProps: {
  395. api: ajaxGetDictItems,
  396. params: { code: 'rule_conditions' },
  397. labelField: 'text',
  398. valueField: 'value',
  399. getPopupContainer: (node) => document.body,
  400. },
  401. },
  402. {
  403. field: 'ruleValue',
  404. label: '规则值',
  405. component: 'Input',
  406. required: true,
  407. },
  408. {
  409. field: 'status',
  410. label: '状态',
  411. component: 'RadioButtonGroup',
  412. defaultValue: '1',
  413. componentProps: {
  414. options: [
  415. { label: '无效', value: '0' },
  416. { label: '有效', value: '1' },
  417. ],
  418. },
  419. },
  420. ];