华西海圻ELN前端工程
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

499 lines
18 KiB

  1. <template>
  2. <div class="app-container">
  3. <el-form :model="queryParams" ref="queryForm" :inline="true" v-show="showSearch">
  4. <el-form-item :label="$t('page.system.menu.name')+':'" prop="menuName">
  5. <el-input
  6. v-model="queryParams.menuName"
  7. :placeholder="$t('form.placeholderInput')"
  8. clearable
  9. @keyup.enter.native="handleQuery"
  10. />
  11. </el-form-item>
  12. <el-form-item :label="$t('page.system.menu.status')+':'" prop="visible">
  13. <el-select v-model="queryParams.visible" :placeholder="$t('form.placeholderSelect')">
  14. <el-option :label="$t('page.system.menu.statusVisible')" :value="0" />
  15. <el-option :label="$t('page.system.menu.statusHide')" :value="1" />
  16. </el-select>
  17. </el-form-item>
  18. <el-form-item>
  19. <el-button type="primary" icon="el-icon-search" @click="handleQuery">{{$t('form.search')}}</el-button>
  20. <el-button icon="el-icon-refresh" @click="resetQuery">{{$t('form.reset')}}</el-button>
  21. </el-form-item>
  22. </el-form>
  23. <el-row :gutter="10" class="mb8">
  24. <el-col :span="1.5">
  25. <el-button
  26. type="primary"
  27. plain
  28. icon="el-icon-plus"
  29. @click="handleAdd"
  30. v-hasPermi="['system:menu:add']"
  31. >{{$t('form.add')}}</el-button>
  32. </el-col>
  33. <!-- <el-col :span="1.5">
  34. <el-button
  35. type="info"
  36. plain
  37. icon="el-icon-sort"
  38. @click="toggleExpandAll"
  39. >展开/折叠</el-button>
  40. </el-col> -->
  41. <!-- <right-toolbar :showSearch.sync="showSearch" @queryTable="getList"></right-toolbar> -->
  42. </el-row>
  43. <el-table
  44. v-if="refreshTable"
  45. v-loading="loading"
  46. :data="menuList"
  47. row-key="menuId"
  48. :default-expand-all="isExpandAll"
  49. :tree-props="{children: 'children', hasChildren: 'hasChildren'}"
  50. >
  51. <el-table-column prop="menuName" :label="$t('page.system.menu.name')" :show-overflow-tooltip="true"></el-table-column>
  52. <el-table-column prop="orderNum" :label="$t('page.system.menu.sort')" width="100"></el-table-column>
  53. <el-table-column prop="path" :label="$t('page.system.menu.router')" :show-overflow-tooltip="true"></el-table-column>
  54. <el-table-column prop="menuType" :label="$t('page.system.menu.type')" width="100">
  55. <template slot-scope="scope">
  56. <span v-if="scope.row.menuType==='M'">{{$t('page.system.menu.typeDir')}}</span>
  57. <span v-if="scope.row.menuType==='C'">{{$t('page.system.menu.typeMenu')}}</span>
  58. <span v-if="scope.row.menuType==='F'">{{$t('page.system.menu.typeBtn')}}</span>
  59. </template>
  60. </el-table-column>
  61. <el-table-column prop="visible" :label="$t('page.system.menu.status')" width="100">
  62. <template slot-scope="scope">
  63. <span v-if="scope.row.visible==='0'">{{$t('page.system.menu.statusVisible')}}</span>
  64. <span v-if="scope.row.visible==='1'">{{$t('page.system.menu.statusHide')}}</span>
  65. </template>
  66. </el-table-column>
  67. <!-- <el-table-column prop="perms" label="权限标识" :show-overflow-tooltip="true"></el-table-column>
  68. <el-table-column prop="component" label="组件路径" :show-overflow-tooltip="true"></el-table-column>
  69. <el-table-column prop="status" label="状态" width="80">
  70. <template slot-scope="scope">
  71. <dict-tag :options="dict.type.sys_normal_disable" :value="scope.row.status"/>
  72. </template>
  73. </el-table-column>
  74. <el-table-column label="创建时间" align="center" prop="createTime">
  75. <template slot-scope="scope">
  76. <span>{{ parseTime(scope.row.createTime) }}</span>
  77. </template>
  78. </el-table-column> -->
  79. <el-table-column :label="$t('form.operate')" align="center" class-name="small-padding fixed-width" width="200">
  80. <template slot-scope="scope">
  81. <el-button
  82. type="text"
  83. @click="handleAdd(scope.row)"
  84. v-hasPermi="['system:menu:add']"
  85. >{{$t('page.system.menu.add')}}</el-button>
  86. <el-button
  87. type="text"
  88. @click="handleUpdate(scope.row)"
  89. v-hasPermi="['system:menu:edit']"
  90. >{{$t('page.system.menu.edit')}}</el-button>
  91. <el-button
  92. type="text"
  93. @click="handleDelete(scope.row)"
  94. v-hasPermi="['system:menu:remove']"
  95. >{{$t('page.system.menu.delete')}}</el-button>
  96. </template>
  97. </el-table-column>
  98. </el-table>
  99. <!-- 添加或修改菜单对话框 -->
  100. <el-dialog :close-on-click-modal="false" :close-on-press-escape="false" :title="title" :visible.sync="open" width="680px" append-to-body>
  101. <el-form ref="form" :model="form" :rules="rules" label-width="100px">
  102. <el-row>
  103. <el-col :span="24">
  104. <el-form-item :label="$t('page.system.menu.parent')+':'" prop="parentId">
  105. <treeselect
  106. v-model="form.parentId"
  107. :options="menuOptions"
  108. :normalizer="normalizer"
  109. :show-count="true"
  110. />
  111. </el-form-item>
  112. </el-col>
  113. </el-row>
  114. <el-row>
  115. <el-col :span="24">
  116. <el-form-item :label="$t('page.system.menu.type')+':'" prop="menuType">
  117. <el-radio-group v-model="form.menuType">
  118. <el-radio label="M">{{$t('page.system.menu.typeDir')}}</el-radio>
  119. <el-radio label="C">{{$t('page.system.menu.typeMenu')}}</el-radio>
  120. <el-radio label="F">{{$t('page.system.menu.typeBtn')}}</el-radio>
  121. </el-radio-group>
  122. </el-form-item>
  123. </el-col>
  124. </el-row>
  125. <el-row>
  126. <!-- <el-col :span="12" v-if="form.menuType != 'F'">
  127. <el-form-item prop="isFrame">
  128. <span slot="label">
  129. <el-tooltip content="选择是外链则路由地址需要以`http(s)://`开头" placement="top">
  130. <i class="el-icon-question"></i>
  131. </el-tooltip>
  132. 是否外链
  133. </span>
  134. <el-radio-group v-model="form.isFrame">
  135. <el-radio label="0"></el-radio>
  136. <el-radio label="1"></el-radio>
  137. </el-radio-group>
  138. </el-form-item>
  139. </el-col> -->
  140. <el-col :span="24" v-if="form.menuType != 'F'">
  141. <el-form-item prop="path">
  142. <span slot="label">
  143. <!-- <el-tooltip content="访问的路由地址,如:`user`,如外网地址需内链访问则以`http(s)://`开头" placement="top">
  144. <i class="el-icon-question"></i>
  145. </el-tooltip> -->
  146. {{$t('page.system.menu.router')+':'}}
  147. </span>
  148. <el-input v-model="form.path" :placeholder="$t('form.placeholderInput')" />
  149. </el-form-item>
  150. </el-col>
  151. </el-row>
  152. <el-row>
  153. <el-col :span="24">
  154. <el-form-item :label="$t('page.system.menu.name')+':'" prop="menuName">
  155. <el-input v-model="form.menuName" :placeholder="$t('form.placeholderInput')" />
  156. </el-form-item>
  157. </el-col>
  158. <!-- <el-col :span="12" v-if="form.menuType == 'C'">
  159. <el-form-item prop="routeName">
  160. <el-input v-model="form.routeName" placeholder="请输入路由名称" />
  161. <span slot="label">
  162. <el-tooltip content="默认不填则和路由地址相同:如地址为:`user`,则名称为`User`(注意:为避免名字的冲突,特殊情况下请自定义,保证唯一性)" placement="top">
  163. <i class="el-icon-question"></i>
  164. </el-tooltip>
  165. 路由名称
  166. </span>
  167. </el-form-item>
  168. </el-col> -->
  169. <el-col :span="24">
  170. <el-form-item :label="$t('page.system.menu.sort')+':'" prop="orderNum">
  171. <el-input-number style="width:100%" v-model="form.orderNum" controls-position="right" :min="0" :placeholder="$t('form.placeholderInput')" />
  172. </el-form-item>
  173. </el-col>
  174. </el-row>
  175. <el-row>
  176. <el-col :span="24" v-if="form.menuType != 'F'">
  177. <el-form-item prop="visible">
  178. <span slot="label">
  179. <!-- <el-tooltip content="选择隐藏则路由将不会出现在侧边栏,但仍然可以访问" placement="top">
  180. <i class="el-icon-question"></i>
  181. </el-tooltip> -->
  182. {{$t('page.system.menu.status')+':'}}
  183. </span>
  184. <el-radio-group v-model="form.visible">
  185. <el-radio key="0" label="0" >{{$t('page.system.menu.statusVisible')}}</el-radio>
  186. <el-radio key="1" label="1" >{{$t('page.system.menu.statusHide')}}</el-radio>
  187. </el-radio-group>
  188. </el-form-item>
  189. </el-col>
  190. <!-- <el-col :span="12">
  191. <el-form-item prop="status">
  192. <span slot="label">
  193. <el-tooltip content="选择停用则路由将不会出现在侧边栏,也不能被访问" placement="top">
  194. <i class="el-icon-question"></i>
  195. </el-tooltip>
  196. 菜单状态
  197. </span>
  198. <el-radio-group v-model="form.status">
  199. <el-radio
  200. v-for="dict in dict.type.sys_normal_disable"
  201. :key="dict.value"
  202. :label="dict.value"
  203. >{{dict.label}}</el-radio>
  204. </el-radio-group>
  205. </el-form-item>
  206. </el-col> -->
  207. </el-row>
  208. <el-row>
  209. <el-col :span="24" v-if="form.menuType == 'C'">
  210. <el-form-item prop="component">
  211. <span slot="label">
  212. <!-- <el-tooltip content="访问的组件路径,如:`system/user/index`,默认在`views`目录下" placement="top">
  213. <i class="el-icon-question"></i>
  214. </el-tooltip> -->
  215. {{$t('page.system.menu.comp')+':'}}
  216. </span>
  217. <el-input v-model="form.component" :placeholder="$t('form.placeholderInput')" />
  218. </el-form-item>
  219. </el-col>
  220. <el-col :span="24" v-if="form.menuType != 'M'">
  221. <el-form-item prop="perms">
  222. <el-input v-model="form.perms" :placeholder="$t('form.placeholderInput')" maxlength="100" />
  223. <span slot="label">
  224. <!-- <el-tooltip content="控制器中定义的权限字符,如:@PreAuthorize(`@ss.hasPermi('system:user:list')`)" placement="top">
  225. <i class="el-icon-question"></i>
  226. </el-tooltip> -->
  227. {{$t('page.system.menu.permit')+':'}}
  228. </span>
  229. </el-form-item>
  230. </el-col>
  231. </el-row>
  232. <el-row>
  233. <el-col :span="24" v-if="form.menuType != 'F'">
  234. <el-form-item :label="$t('page.system.menu.icon')+':'" prop="icon">
  235. <el-popover
  236. placement="bottom-start"
  237. width="460"
  238. trigger="click"
  239. @show="$refs['iconSelect'].reset()"
  240. >
  241. <IconSelect ref="iconSelect" @selected="selected" :active-icon="form.icon" />
  242. <el-input slot="reference" v-model="form.icon" :placeholder="$t('form.placeholderSelect')" readonly>
  243. <svg-icon
  244. v-if="form.icon"
  245. slot="prefix"
  246. :icon-class="form.icon"
  247. style="width: 25px;"
  248. />
  249. <i v-else slot="prefix" class="el-icon-search el-input__icon" />
  250. </el-input>
  251. </el-popover>
  252. </el-form-item>
  253. </el-col>
  254. </el-row>
  255. <el-row>
  256. <!-- <el-col :span="12" v-if="form.menuType == 'C'">
  257. <el-form-item prop="query">
  258. <el-input v-model="form.query" placeholder="请输入路由参数" maxlength="255" />
  259. <span slot="label">
  260. <el-tooltip content='访问路由的默认传递参数,如:`{"id": 1, "name": "ry"}`' placement="top">
  261. <i class="el-icon-question"></i>
  262. </el-tooltip>
  263. 路由参数
  264. </span>
  265. </el-form-item>
  266. </el-col> -->
  267. <el-col :span="24" v-if="form.menuType == 'C'">
  268. <el-form-item prop="isCache">
  269. <span slot="label">
  270. <el-tooltip content="选择是则会被`keep-alive`缓存,需要匹配组件的`name`和地址保持一致" placement="top">
  271. <i class="el-icon-question"></i>
  272. </el-tooltip>
  273. 是否缓存
  274. </span>
  275. <el-radio-group v-model="form.isCache">
  276. <el-radio label="0">缓存</el-radio>
  277. <el-radio label="1">不缓存</el-radio>
  278. </el-radio-group>
  279. </el-form-item>
  280. </el-col>
  281. </el-row>
  282. </el-form>
  283. <div slot="footer" class="dialog-footer">
  284. <el-button type="primary" @click="submitForm">{{$t('form.saveConfirm')}}</el-button>
  285. <el-button @click="cancel">{{$t('form.cancel')}}</el-button>
  286. </div>
  287. </el-dialog>
  288. </div>
  289. </template>
  290. <script>
  291. import { listMenu, getMenu, delMenu, addMenu, updateMenu } from "@/api/system/menu"
  292. import Treeselect from "@riophae/vue-treeselect"
  293. import "@riophae/vue-treeselect/dist/vue-treeselect.css"
  294. import IconSelect from "@/components/IconSelect"
  295. export default {
  296. name: "Menu",
  297. dicts: ['sys_show_hide', 'sys_normal_disable'],
  298. components: { Treeselect, IconSelect },
  299. data() {
  300. return {
  301. // 遮罩层
  302. loading: true,
  303. // 显示搜索条件
  304. showSearch: true,
  305. // 菜单表格树数据
  306. menuList: [],
  307. // 菜单树选项
  308. menuOptions: [],
  309. // 弹出层标题
  310. title: "",
  311. // 是否显示弹出层
  312. open: false,
  313. // 是否展开,默认全部折叠
  314. isExpandAll: false,
  315. // 重新渲染表格状态
  316. refreshTable: true,
  317. // 查询参数
  318. queryParams: {
  319. menuName: undefined,
  320. visible: ''
  321. },
  322. // 表单参数
  323. form: {},
  324. // 表单校验
  325. rules: {
  326. menuType: [
  327. { required: true, message: this.$t('form.placeholderSelect'), trigger: "change" }
  328. ],
  329. menuName: [
  330. { required: true, message: this.$t('form.placeholderInput'), trigger: "blur" }
  331. ],
  332. orderNum: [
  333. { required: true, message: this.$t('form.placeholderInput'), trigger: "blur" }
  334. ],
  335. path: [
  336. { required: true, message: this.$t('form.placeholderInput'), trigger: "blur" }
  337. ],
  338. component: [
  339. { required: true, message: this.$t('form.placeholderInput'), trigger: "blur" }
  340. ],
  341. visible: [
  342. { required: true, message: this.$t('form.placeholderSelect'), trigger: "change" }
  343. ],
  344. }
  345. }
  346. },
  347. created() {
  348. this.getList()
  349. },
  350. methods: {
  351. // 选择图标
  352. selected(name) {
  353. this.form.icon = name
  354. },
  355. /** 查询菜单列表 */
  356. getList() {
  357. this.loading = true
  358. listMenu(this.queryParams).then(response => {
  359. this.menuList = this.handleTree(response.data, "menuId")
  360. this.loading = false
  361. })
  362. },
  363. /** 转换菜单数据结构 */
  364. normalizer(node) {
  365. if (node.children && !node.children.length) {
  366. delete node.children
  367. }
  368. return {
  369. id: node.menuId,
  370. label: node.menuName,
  371. children: node.children
  372. }
  373. },
  374. /** 查询菜单下拉树结构 */
  375. getTreeselect() {
  376. listMenu().then(response => {
  377. this.menuOptions = []
  378. const menu = { menuId: 0, menuName: this.$t('page.system.menu.root'), children: [] }
  379. menu.children = this.handleTree(response.data, "menuId")
  380. this.menuOptions.push(menu)
  381. })
  382. },
  383. // 取消按钮
  384. cancel() {
  385. this.open = false
  386. this.reset()
  387. },
  388. // 表单重置
  389. reset() {
  390. this.form = {
  391. menuId: undefined,
  392. parentId: 0,
  393. menuName: undefined,
  394. icon: undefined,
  395. menuType: "M",
  396. orderNum: undefined,
  397. isFrame: "1",
  398. isCache: "0",
  399. visible: "0",
  400. status: "0"
  401. }
  402. this.resetForm("form")
  403. },
  404. /** 搜索按钮操作 */
  405. handleQuery() {
  406. this.getList()
  407. },
  408. /** 重置按钮操作 */
  409. resetQuery() {
  410. this.resetForm("queryForm")
  411. this.handleQuery()
  412. },
  413. /** 新增按钮操作 */
  414. handleAdd(row) {
  415. debugger
  416. // this.saveSimpleLog({name:'',nameEn:'',jcmc:'新增菜单',jcmcEn:'Add Menu'})
  417. this.reset()
  418. this.getTreeselect()
  419. if (row != null && row.menuId) {
  420. this.form.parentId = row.menuId
  421. } else {
  422. this.form.parentId = 0
  423. }
  424. this.open = true
  425. this.title = this.$t('page.system.menu.addMenu')
  426. },
  427. /** 展开/折叠操作 */
  428. toggleExpandAll() {
  429. this.refreshTable = false
  430. this.isExpandAll = !this.isExpandAll
  431. this.$nextTick(() => {
  432. this.refreshTable = true
  433. })
  434. },
  435. /** 修改按钮操作 */
  436. handleUpdate(row) {
  437. // this.saveSimpleLog({name:row.menuName,nameEn:row.menuName,jcmc:'编辑菜单',jcmcEn:'Edit Menu'})
  438. this.reset()
  439. this.getTreeselect()
  440. getMenu(row.menuId).then(response => {
  441. this.form = response.data
  442. this.open = true
  443. this.title = this.$t('page.system.menu.modifyMenu')
  444. })
  445. },
  446. /** 提交按钮 */
  447. submitForm: function() {
  448. this.$refs["form"].validate(valid => {
  449. if (valid) {
  450. if (this.form.menuId != undefined) {
  451. updateMenu(this.form).then(response => {
  452. // this.$modal.msgSuccess("修改成功")
  453. this.open = false
  454. this.getList()
  455. })
  456. } else {
  457. addMenu(this.form).then(response => {
  458. // this.$modal.msgSuccess("新增成功")
  459. this.open = false
  460. this.getList()
  461. })
  462. }
  463. }
  464. })
  465. },
  466. /** 删除按钮操作 */
  467. handleDelete(row) {
  468. // this.saveSimpleLog({name:row.menuName,nameEn:row.menuName,jcmc:'删除菜单',jcmcEn:'Remove Menu'})
  469. this.$modal.confirm(this.$t('form.confirmDelete')).then(function() {
  470. return delMenu(row.menuId)
  471. }).then(() => {
  472. this.getList()
  473. }).catch(() => {})
  474. }
  475. }
  476. }
  477. </script>