index.vue 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349
  1. <script lang="ts">
  2. export default {
  3. name: "dict"
  4. };
  5. </script>
  6. <script setup lang="ts">
  7. import XEUtils from "xe-utils";
  8. import Config from "./config.vue";
  9. import { cloneDeep } from "lodash-unified";
  10. import { templateRef } from "@vueuse/core";
  11. import { reactive, ref, unref, nextTick } from "vue";
  12. import { useCopyToClipboard } from "/@/utils/useCopyToClipboard";
  13. import {
  14. VXETable,
  15. VxeTableInstance,
  16. VxeTableEvents,
  17. VxeFormPropTypes
  18. } from "vxe-table";
  19. type onEditNRow = {
  20. name: string;
  21. model: string;
  22. };
  23. const dictData = reactive({
  24. submitLoading: false,
  25. showEdit: false,
  26. selectRow: null,
  27. filterName: "",
  28. tableData: [
  29. {
  30. id: 1,
  31. name: "状态",
  32. model: "",
  33. children: [
  34. {
  35. id: "1-1",
  36. name: "服务状态",
  37. model: "serviceStatus"
  38. },
  39. {
  40. id: "1-2",
  41. name: "在线状态",
  42. model: "onlienStatus"
  43. }
  44. ]
  45. },
  46. { id: 2, name: "操作系统", model: "operatingSystem" }
  47. ],
  48. formData: {
  49. name: "",
  50. model: ""
  51. },
  52. formItems: [
  53. {
  54. field: "name",
  55. title: "字典名称",
  56. span: 24,
  57. itemRender: {
  58. name: "$input",
  59. props: { placeholder: "请输入字典名称" }
  60. }
  61. },
  62. {
  63. field: "model",
  64. title: "字典类型",
  65. span: 24,
  66. itemRender: {
  67. name: "$input",
  68. props: {
  69. placeholder: "请输入字典类型",
  70. //这里vxe-table文档并没有提到,可以配置所选组件的所有属性,比如这里可以配置关于vxe-input的所有属性
  71. disabled: true
  72. }
  73. }
  74. },
  75. {
  76. align: "right",
  77. span: 24,
  78. itemRender: {
  79. name: "$buttons",
  80. children: [
  81. { props: { type: "submit", content: "提交", status: "primary" } },
  82. { props: { type: "reset", content: "重置" } }
  83. ]
  84. }
  85. }
  86. ] as VxeFormPropTypes.Items
  87. });
  88. let originData = cloneDeep(dictData.tableData);
  89. const xTree = templateRef<HTMLElement | any>("xTree", null);
  90. const handleSearch = () => {
  91. const filterName = XEUtils.toValueString(dictData.filterName).trim();
  92. if (filterName) {
  93. const options = { children: "children" };
  94. const searchProps = ["name"];
  95. dictData.tableData = XEUtils.searchTree(
  96. originData,
  97. item =>
  98. searchProps.some(
  99. key => XEUtils.toValueString(item[key]).indexOf(filterName) > -1
  100. ),
  101. options
  102. );
  103. // 搜索之后默认展开所有子节点
  104. nextTick(() => {
  105. const $table = xTree.value;
  106. $table.setAllTreeExpand(true);
  107. });
  108. } else {
  109. dictData.tableData = originData;
  110. }
  111. };
  112. // 创建一个防防抖函数,调用频率间隔 100 毫秒
  113. const searchEvent = XEUtils.debounce(
  114. function () {
  115. handleSearch();
  116. },
  117. 100,
  118. { leading: false, trailing: true }
  119. );
  120. const confirmEvent = async () => {
  121. const type = await VXETable.modal.confirm("您确定要删除吗?");
  122. (await type) === "confirm" &&
  123. VXETable.modal.message({
  124. content: "测试数据,不可删除",
  125. status: "error"
  126. });
  127. };
  128. function commonFn(value, disabled) {
  129. dictData.selectRow = value;
  130. dictData.showEdit = true;
  131. dictData.formItems[1].itemRender.props.disabled = disabled;
  132. }
  133. // 新增
  134. function onAdd() {
  135. commonFn(null, false);
  136. }
  137. // 新增子类型
  138. function onAddChild(row?: object) {
  139. console.log("onAddChild", row);
  140. commonFn(null, false);
  141. }
  142. // 编辑
  143. function onEdit(row?: onEditNRow) {
  144. dictData.formData = {
  145. name: row.name,
  146. model: row.model ? row.model : "暂无字典类型"
  147. };
  148. commonFn(row, true);
  149. // VXETable.modal.message({
  150. // content: "测试数据,不可编辑",
  151. // status: "error"
  152. // });
  153. }
  154. // 拷贝当前列表项的数据(字典类型)
  155. const { clipboardRef } = useCopyToClipboard();
  156. const cellDBLClickEvent: VxeTableEvents.CellDblclick = ({ row }) => {
  157. clipboardRef.value = unref(row).model;
  158. };
  159. const xTable = ref({} as VxeTableInstance);
  160. const submitEvent = () => {
  161. dictData.submitLoading = true;
  162. setTimeout(() => {
  163. const $table = xTable.value;
  164. dictData.submitLoading = false;
  165. dictData.showEdit = false;
  166. if (dictData.selectRow) {
  167. VXETable.modal.message({ content: "保存成功", status: "success" });
  168. Object.assign(dictData.selectRow, dictData.formData);
  169. } else {
  170. VXETable.modal.message({ content: "新增成功", status: "success" });
  171. $table.insert(dictData.formData);
  172. }
  173. }, 500);
  174. };
  175. let drawer = ref(false);
  176. function onDeploy(value?: object) {
  177. console.log("onDeploy", value);
  178. drawer.value = true;
  179. }
  180. function handleClose() {
  181. drawer.value = false;
  182. }
  183. </script>
  184. <template>
  185. <div>
  186. <!-- 工具栏 -->
  187. <vxe-toolbar>
  188. <template #buttons>
  189. <vxe-input
  190. v-model="dictData.filterName"
  191. :placeholder="$t('buttons.hssearch')"
  192. @keyup="searchEvent"
  193. ></vxe-input>
  194. </template>
  195. <template #tools>
  196. <vxe-button
  197. icon="fa fa-plus-square-o"
  198. status="primary"
  199. @click="onAdd"
  200. >{{ $t("buttons.hsadd") }}</vxe-button
  201. >
  202. <vxe-button
  203. icon="fa fa-folder-open-o"
  204. status="primary"
  205. @click="$refs.xTree.setAllTreeExpand(true)"
  206. >{{ $t("buttons.hsexpendAll") }}</vxe-button
  207. >
  208. <vxe-button
  209. icon="fa fa-folder-o"
  210. status="primary"
  211. @click="$refs.xTree.clearTreeExpand()"
  212. >{{ $t("buttons.hscollapseAll") }}</vxe-button
  213. >
  214. </template>
  215. </vxe-toolbar>
  216. <!-- 列表 -->
  217. <vxe-table
  218. ref="xTree"
  219. border
  220. resizable
  221. :tree-config="{
  222. children: 'children',
  223. iconOpen: 'fa fa-minus-square-o',
  224. iconClose: 'fa fa-plus-square-o'
  225. }"
  226. :data="dictData.tableData"
  227. @cell-dblclick="cellDBLClickEvent"
  228. >
  229. <vxe-table-column
  230. tree-node
  231. field="name"
  232. title="字典名称"
  233. ></vxe-table-column>
  234. <vxe-table-column title="字典类型">
  235. <template #default="{ row }">
  236. <el-tooltip
  237. effect="dark"
  238. :content="'双击复制:' + row.model"
  239. placement="right"
  240. >
  241. <span class="text-model">{{ row.model }}</span>
  242. </el-tooltip>
  243. </template>
  244. </vxe-table-column>
  245. <vxe-table-column title="操作" width="330" fixed="right">
  246. <template #default="{ row }">
  247. <vxe-button
  248. type="text"
  249. icon="fa fa-pencil-square-o"
  250. @click="onEdit(row)"
  251. >编辑</vxe-button
  252. >
  253. <vxe-button
  254. type="text"
  255. icon="fa fa-plus-square-o"
  256. @click="onAddChild(row)"
  257. >新增子类型</vxe-button
  258. >
  259. <vxe-button
  260. v-show="row.model"
  261. type="text"
  262. icon="fa fa-cog"
  263. @click="onDeploy(row)"
  264. >字典配置</vxe-button
  265. >
  266. <vxe-button type="text" icon="fa fa-trash-o" @click="confirmEvent"
  267. >删除</vxe-button
  268. >
  269. </template>
  270. </vxe-table-column>
  271. </vxe-table>
  272. <!-- 修改、添加弹框 -->
  273. <vxe-modal
  274. resize
  275. width="450"
  276. v-model="dictData.showEdit"
  277. :title="dictData.selectRow ? '编辑' : '新增'"
  278. :loading="dictData.submitLoading"
  279. @hide="$refs.xForm.reset()"
  280. >
  281. <template #default>
  282. <vxe-form
  283. ref="xForm"
  284. :data="dictData.formData"
  285. :items="dictData.formItems"
  286. title-align="right"
  287. title-width="100"
  288. @submit="submitEvent"
  289. ></vxe-form>
  290. </template>
  291. </vxe-modal>
  292. <Config :drawer="drawer" drawTitle="字典列表" @handleClose="handleClose" />
  293. </div>
  294. </template>
  295. <style lang="scss" scoped>
  296. .vxe-input + .vxe-button,
  297. .vxe-input + .vxe-button--dropdown,
  298. .vxe-button + .vxe-button,
  299. .vxe-button + .vxe-button--dropdown {
  300. margin-left: 0;
  301. }
  302. .vxe-button.type--button:not(.is--round) {
  303. border-radius: 0;
  304. }
  305. .vxe-toolbar.size--medium {
  306. padding: 10px;
  307. }
  308. .vxe-table--render-default.size--medium {
  309. margin-top: 12px;
  310. }
  311. .vxe-button.size--medium.type--button {
  312. margin-right: 0.07em;
  313. }
  314. .text-model {
  315. &:hover {
  316. cursor: pointer;
  317. }
  318. }
  319. </style>