index.vue 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465
  1. <template>
  2. <div>
  3. <div class="app-container">
  4. <el-form :model="queryParams" ref="queryRef" :inline="true" v-show="showSearch" label-width="68px">
  5. <el-form-item label="模板名称" prop="tmplName">
  6. <el-input
  7. v-model="queryParams.tmplName"
  8. placeholder="请输入模板名称"
  9. clearable
  10. @keyup.enter="handleQuery"
  11. />
  12. </el-form-item>
  13. <!-- <el-form-item label="创建人" prop="createBy">
  14. <el-input
  15. v-model="queryParams.createBy"
  16. placeholder="请输入创建人"
  17. clearable
  18. @keyup.enter="handleQuery"
  19. />
  20. </el-form-item> -->
  21. <!-- <el-form-item label="创建时间" prop="createTime">
  22. <el-date-picker clearable
  23. v-model="queryParams.createTime"
  24. type="date"
  25. value-format="YYYY-MM-DD"
  26. placeholder="请选择创建时间">
  27. </el-date-picker>
  28. </el-form-item> -->
  29. <!-- <el-form-item label="更新人" prop="updateBy">
  30. <el-input
  31. v-model="queryParams.updateBy"
  32. placeholder="请输入更新人"
  33. clearable
  34. @keyup.enter="handleQuery"
  35. />
  36. </el-form-item> -->
  37. <!-- <el-form-item label="更新时间" prop="updateTime">
  38. <el-date-picker clearable
  39. v-model="queryParams.updateTime"
  40. type="date"
  41. value-format="YYYY-MM-DD"
  42. placeholder="请选择更新时间">
  43. </el-date-picker>
  44. </el-form-item> -->
  45. <el-form-item>
  46. <el-button type="primary" icon="Search" @click="handleQuery">搜索</el-button>
  47. <el-button icon="Refresh" @click="resetQuery">重置</el-button>
  48. </el-form-item>
  49. </el-form>
  50. <el-row :gutter="10" class="mb8">
  51. <el-col :span="1.5">
  52. <el-button
  53. type="primary"
  54. plain
  55. icon="Plus"
  56. @click="handleAdd"
  57. v-hasPermi="['biz:dirTemplate:add']"
  58. >新增
  59. </el-button>
  60. </el-col>
  61. <el-col :span="1.5">
  62. <el-button
  63. type="success"
  64. plain
  65. icon="Edit"
  66. :disabled="single"
  67. @click="handleUpdate"
  68. v-hasPermi="['biz:dirTemplate:edit']"
  69. >修改
  70. </el-button>
  71. </el-col>
  72. <el-col :span="1.5">
  73. <el-button
  74. type="danger"
  75. plain
  76. icon="Delete"
  77. :disabled="multiple"
  78. @click="handleDelete"
  79. v-hasPermi="['biz:dirTemplate:remove']"
  80. >删除
  81. </el-button>
  82. </el-col>
  83. <el-col :span="1.5">
  84. <el-button
  85. type="warning"
  86. plain
  87. icon="Download"
  88. @click="handleExport"
  89. v-hasPermi="['biz:dirTemplate:export']"
  90. >导出
  91. </el-button>
  92. </el-col>
  93. <right-toolbar v-model:showSearch="showSearch" @queryTable="getList"></right-toolbar>
  94. </el-row>
  95. <el-table v-loading="loading" :data="dirTemplateList" @selection-change="handleSelectionChange">
  96. <el-table-column type="selection" width="55" align="center"/>
  97. <el-table-column label="模板ID" align="center" prop="tmplId"/>
  98. <el-table-column label="模板名称" align="center" prop="tmplName"/>
  99. <el-table-column label="创建人" align="center" prop="createBy"/>
  100. <el-table-column label="创建时间" align="center" prop="createTime" width="180">
  101. <template #default="scope">
  102. <span>{{ parseTime(scope.row.createTime, '{y}-{m}-{d} {h}:{i}:{s}') }}</span>
  103. </template>
  104. </el-table-column>
  105. <el-table-column label="更新人" align="center" prop="updateBy"/>
  106. <el-table-column label="更新时间" align="center" prop="updateTime" width="180">
  107. <template #default="scope">
  108. <span>{{ parseTime(scope.row.updateTime, '{y}-{m}-{d} {h}:{i}:{s}') }}</span>
  109. </template>
  110. </el-table-column>
  111. <el-table-column label="备注" align="center" prop="remark"/>
  112. <el-table-column label="操作" align="center" class-name="small-padding fixed-width">
  113. <template #default="scope">
  114. <el-button link type="primary" icon="Edit" @click="handleUpdate(scope.row)"
  115. v-hasPermi="['biz:dirTemplate:edit']">修改
  116. </el-button>
  117. <el-button link type="primary" icon="Delete" @click="handleDelete(scope.row)"
  118. v-hasPermi="['biz:dirTemplate:remove']">删除
  119. </el-button>
  120. </template>
  121. </el-table-column>
  122. </el-table>
  123. <pagination
  124. v-show="total>0"
  125. :total="total"
  126. v-model:page="queryParams.pageNum"
  127. v-model:limit="queryParams.pageSize"
  128. @pagination="getList"
  129. />
  130. <!-- 添加或修改目录模板对话框 -->
  131. <el-dialog :title="title" :close-on-click-modal="false" v-model="open" width="500px" append-to-body @close="cancelAddTemp">
  132. <el-form ref="dirTemplateRef" :model="form" :rules="rules" label-width="80px">
  133. <el-form-item label="模板名称" prop="tmplName">
  134. <el-input v-model="tempTitle" placeholder="请输入模板名称"/>
  135. </el-form-item>
  136. <el-form-item label="模板内容" prop="tmplContent">
  137. <el-tree
  138. :data="tempData"
  139. default-expand-all
  140. node-key="id"
  141. style="width:100%;height:300px;overflow-y: auto;"
  142. :expand-on-click-node="false"
  143. >
  144. <template #default="{ node, data }">
  145. <span class="custom-tree-node">
  146. <span>{{ node.label }}</span>
  147. <span>
  148. <a class="btm" @click="append(data)"> 增加 </a>
  149. <a class="up_btn" v-if="data.id != 1" @click="updataNode(data)"> 修改 </a>
  150. <a class="del_btn" v-if="data.id != 1" style="margin-left: 8px" @click="remove(node, data)"> 删除 </a>
  151. </span>
  152. </span>
  153. </template>
  154. </el-tree>
  155. </el-form-item>
  156. <el-form-item label="备注" prop="remark">
  157. <el-input v-model="form.remark" placeholder="请输入备注"/>
  158. </el-form-item>
  159. </el-form>
  160. <template #footer>
  161. <div class="dialog-footer">
  162. <el-button type="primary" @click="submitForm" :disabled="addDisabled">确 定</el-button>
  163. <el-button @click="cancelAddTemp">取 消</el-button>
  164. </div>
  165. </template>
  166. </el-dialog>
  167. <el-dialog :close-on-click-modal="false" title="目录名称" v-model="showMenuName" width="500px">
  168. <el-input v-model="newName" maxlength="32" placeholder="请输入目录名称"/>
  169. <template #footer>
  170. <div class="dialog-footer">
  171. <el-button type="primary" @click="submitMenuName">确 定</el-button>
  172. <el-button @click="cancelMenuName">取 消</el-button>
  173. </div>
  174. </template>
  175. </el-dialog>
  176. </div>
  177. </div>
  178. </template>
  179. <script setup name="DirTemplate">
  180. import {
  181. addDirTemplate,
  182. delDirTemplate,
  183. getDirTemplate,
  184. listDirTemplate,
  185. updateDirTemplate
  186. } from "@/api/biz/dirTemplate";
  187. import { ElMessage} from 'element-plus'
  188. const {proxy} = getCurrentInstance();
  189. const dirTemplateList = ref([]);
  190. const open = ref(false);
  191. const loading = ref(true);
  192. const showSearch = ref(true);
  193. const ids = ref([]);
  194. const single = ref(true);
  195. const multiple = ref(true);
  196. const total = ref(0);
  197. const title = ref("");
  198. const showMenuName = ref(false)
  199. const newName = ref()
  200. const tempId = ref(1000)
  201. const upNodeData = ref()
  202. const tempTitle = ref('')
  203. const addDisabled = ref(false);
  204. const tempData = ref([
  205. {id:1,label:tempTitle || '请输入模板名称',children:[
  206. ]}
  207. ])
  208. const data = reactive({
  209. form: {},
  210. queryParams: {
  211. pageNum: 1,
  212. pageSize: 10,
  213. tmplName: null,
  214. createBy: null,
  215. createTime: null,
  216. updateBy: null,
  217. updateTime: null,
  218. },
  219. rules: {}
  220. });
  221. const {queryParams, form, rules} = toRefs(data);
  222. /** 查询目录模板列表 */
  223. function getList() {
  224. loading.value = true;
  225. listDirTemplate(queryParams.value).then(response => {
  226. dirTemplateList.value = response.rows;
  227. total.value = response.total;
  228. loading.value = false;
  229. });
  230. }
  231. // 取消按钮
  232. function cancel() {
  233. open.value = false;
  234. reset();
  235. }
  236. // 表单重置
  237. function reset() {
  238. form.value = {
  239. tmplId: null,
  240. tmplName: null,
  241. tmplContent: null,
  242. createBy: null,
  243. createTime: null,
  244. updateBy: null,
  245. updateTime: null,
  246. remark: null
  247. };
  248. proxy.resetForm("dirTemplateRef");
  249. }
  250. /** 搜索按钮操作 */
  251. function handleQuery() {
  252. queryParams.value.pageNum = 1;
  253. getList();
  254. }
  255. /** 重置按钮操作 */
  256. function resetQuery() {
  257. proxy.resetForm("queryRef");
  258. handleQuery();
  259. }
  260. // 多选框选中数据
  261. function handleSelectionChange(selection) {
  262. ids.value = selection.map(item => item.tmplId);
  263. single.value = selection.length != 1;
  264. multiple.value = !selection.length;
  265. }
  266. /** 新增按钮操作 */
  267. function handleAdd() {
  268. reset();
  269. open.value = true;
  270. title.value = "添加目录模板";
  271. }
  272. /** 修改按钮操作 */
  273. function handleUpdate(row) {
  274. reset();
  275. const _tmplId = row.tmplId || ids.value
  276. getDirTemplate(_tmplId).then(response => {
  277. console.log('response',response);
  278. form.value = response.data;
  279. const arr = JSON.parse(response.data.tmplContent)
  280. const getNewtempId = (arr)=>{
  281. arr.map(item=>{
  282. if(item.id >= tempId.value){
  283. console.log('1',item.id);
  284. tempId.value = item.id +1
  285. }
  286. if(item.children?.length>0){
  287. getNewtempId(item.children)
  288. }
  289. })
  290. }
  291. getNewtempId(arr)
  292. tempTitle.value = response.data.tmplName
  293. tempData.value = [{id:1,label:tempTitle,children:[]}]
  294. tempData.value[0].children = arr
  295. open.value = true;
  296. title.value = "修改目录模板";
  297. });
  298. }
  299. /** 提交按钮 */
  300. function submitForm() {
  301. proxy.$refs["dirTemplateRef"].validate(valid => {
  302. if(!tempTitle.value){
  303. return ElMessage({
  304. message: "模板名不能为空",
  305. type: "error"
  306. })
  307. }
  308. const str = JSON.stringify(tempData.value[0].children)
  309. form.value.tmplContent = str
  310. form.value.tmplName = tempTitle.value
  311. if (valid) {
  312. addDisabled.value = true
  313. if (form.value.tmplId != null) {
  314. updateDirTemplate(form.value).then(response => {
  315. proxy.$modal.msgSuccess("修改成功");
  316. open.value = false;
  317. addDisabled.value = false
  318. getList();
  319. });
  320. } else {
  321. addDirTemplate(form.value).then(response => {
  322. proxy.$modal.msgSuccess("新增成功");
  323. open.value = false;
  324. addDisabled.value = false
  325. getList();
  326. });
  327. }
  328. }
  329. });
  330. }
  331. /** 删除按钮操作 */
  332. function handleDelete(row) {
  333. const _tmplIds = row.tmplId || ids.value;
  334. proxy.$modal.confirm('是否确认删除目录模板编号为"' + _tmplIds + '"的数据项?').then(function () {
  335. return delDirTemplate(_tmplIds);
  336. }).then(() => {
  337. getList();
  338. proxy.$modal.msgSuccess("删除成功");
  339. }).catch(() => {
  340. });
  341. }
  342. /** 导出按钮操作 */
  343. function handleExport() {
  344. proxy.download('biz/dirTemplate/export', {
  345. ...queryParams.value
  346. }, `dirTemplate_${new Date().getTime()}.xlsx`)
  347. }
  348. // 新增目录节点
  349. const append = (data)=>{
  350. // console.log('data',data);
  351. const newChild = {id: tempId.value, label: 'newMenu', children: [] }
  352. tempId.value = tempId.value+1
  353. if (!data.children) {
  354. data.children = []
  355. }
  356. data.children.push(newChild)
  357. tempData.value = [...tempData.value]
  358. }
  359. // 移除目录节点
  360. const remove = (node, data)=>{
  361. const parent = node.parent
  362. const children = parent.data.children || parent.data
  363. const index = children.findIndex((d) => d.id === data.id)
  364. children.splice(index, 1)
  365. tempData.value = [...tempData.value]
  366. }
  367. // 修改目录节点
  368. const updataNode = (data)=>{
  369. newName.value = ''
  370. newName.value = data.label
  371. showMenuName.value = true
  372. upNodeData.value = data
  373. }
  374. const submitMenuName = ()=>{
  375. if(!newName.value){
  376. return ElMessage({
  377. message: "目录名不能为空",
  378. type: "error"
  379. })
  380. }
  381. const oldArr = tempData.value
  382. const getNewData = (data)=>{
  383. data.map(item=>{
  384. if(item.id == upNodeData.value.id){
  385. // console.log(1);
  386. item.label = newName.value
  387. return
  388. }
  389. if(item.children?.length>0){
  390. getNewData(item.children)
  391. }
  392. })
  393. }
  394. getNewData(oldArr)
  395. tempData.value= JSON.parse(JSON.stringify(oldArr))
  396. setTimeout(() => {
  397. showMenuName.value = false
  398. // console.log('tempData.value',tempData.value);
  399. // console.log('upNodeData.value',upNodeData.value);
  400. }, 200);
  401. }
  402. const submitAddTemp = ()=>{
  403. }
  404. const cancelMenuName = ()=>{
  405. newName.value = ''
  406. showMenuName.value = false
  407. }
  408. const cancelAddTemp = ()=>{
  409. open.value = false
  410. newName.value = ''
  411. tempTitle.value = ''
  412. tempData.value = [
  413. {id:1,label:tempTitle || '请输入模板名称',children:[
  414. ]}
  415. ]
  416. }
  417. getList();
  418. </script>
  419. <style lang="scss" scoped>
  420. .custom-tree-node {
  421. width: 100%;
  422. flex: 1;
  423. display: flex;
  424. align-items: center;
  425. justify-content: space-between;
  426. font-size: 14px;
  427. padding-right: 8px;
  428. }
  429. .btm{
  430. color: #409eff;
  431. }
  432. .up_btn{
  433. color: #E6a23c;
  434. }
  435. .del_btn{
  436. color: #F56c6c;
  437. }
  438. </style>