flowProcessEdit.vue 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271
  1. <template>
  2. <div>
  3. <a-row :gutter="16" style="margin-bottom: 20px">
  4. <a-col :span="24">
  5. <a-button type="primary" @click="submitAll" :loading="loading" style="margin-right: 6px"> 保存流程 </a-button>
  6. <a-button @click="addNew" :loading="loading"> 新增节点 </a-button>
  7. </a-col>
  8. </a-row>
  9. <a-table
  10. :columns="columns"
  11. :rowKey="(record, index) => index"
  12. :data-source="data"
  13. :pagination="false"
  14. :scroll="{ x: true }"
  15. :loading="tableLoading"
  16. class="j-table-force-nowrap"
  17. size="middle"
  18. >
  19. <template #bodyCell="{ column, record, index, text }">
  20. <template v-if="['name', 'description', 'timeDesc', 'proportion'].includes(column.dataIndex)">
  21. <div>
  22. <a-input
  23. v-if="editableData[record.key]"
  24. style="margin: -5px 0"
  25. v-model:value="editableData[record.key][column.dataIndex]"
  26. @change="(e) => handleChange(e.target.value, record.key, column)"
  27. />
  28. <template v-else>
  29. {{ text }}
  30. </template>
  31. </div>
  32. </template>
  33. <template v-if="column.dataIndex === 'id'">
  34. <a-tag color="pink" @click="up(record, index)" style="cursor: pointer" :class="index == 0 ? 'disabled' : ''">
  35. <a-icon type="arrow-up" />
  36. </a-tag>
  37. <a-tag color="blue" @click="down(record, index)" style="cursor: pointer" :class="index == data.length - 1 ? 'disabled' : ''">
  38. <a-icon type="arrow-down" />
  39. </a-tag>
  40. </template>
  41. <template v-if="column.dataIndex === 'action'">
  42. <div class="editable-row-operations">
  43. <span v-if="editableData[record.key]">
  44. <a @click="() => save(record.key)" style="margin-right: 10px">保存</a>
  45. <a-popconfirm title="确定取消吗?" @confirm="cancel(record.key)">
  46. <a>取消</a>
  47. </a-popconfirm>
  48. </span>
  49. <span v-else>
  50. <a v-if="editableData[record.key]" disabled @click="() => edit(record.key)">编辑</a>
  51. <a v-else @click="() => edit(record.key)">编辑</a>
  52. <a-divider type="vertical" />
  53. <a v-if="editableData[record.key]" disabled @click="() => deleteRow(index)">删除</a>
  54. <a v-else @click="() => deleteRow(index)">删除</a>
  55. </span>
  56. </div>
  57. </template>
  58. </template>
  59. </a-table>
  60. </div>
  61. </template>
  62. <script lang="ts" setup>
  63. import { getAction, postAction } from '/@/api/manage/manage';
  64. import { reactive, ref, UnwrapRef } from 'vue';
  65. import { useMessage } from '@/hooks/web/useMessage';
  66. import { cloneDeep } from 'lodash-es';
  67. const columns = [
  68. {
  69. title: '节点名称',
  70. dataIndex: 'name',
  71. scopedSlots: { customRender: 'name' },
  72. },
  73. {
  74. title: '节点描述',
  75. dataIndex: 'description',
  76. scopedSlots: { customRender: 'description' },
  77. },
  78. {
  79. title: '预计时间',
  80. dataIndex: 'timeDesc',
  81. scopedSlots: { customRender: 'timeDesc' },
  82. },
  83. {
  84. title: '权重',
  85. dataIndex: 'proportion',
  86. scopedSlots: { customRender: 'proportion', title: 'proportionTitle' },
  87. },
  88. {
  89. title: '排序',
  90. dataIndex: 'id',
  91. scopedSlots: { customRender: 'id' },
  92. },
  93. {
  94. title: '操作',
  95. dataIndex: 'action',
  96. scopedSlots: { customRender: 'action' },
  97. },
  98. ];
  99. let data = ref([{}]);
  100. const editableData: UnwrapRef<Record<string, {}>> = reactive({});
  101. const hostId = ref('');
  102. const tableLoading = ref(false);
  103. const loading = ref(false);
  104. const url = ref('');
  105. let record = reactive({});
  106. let cacheData = reactive([{}]);
  107. const { createMessage } = useMessage();
  108. function submitAll() {
  109. let d = data.value;
  110. let proportion = 0;
  111. for (let i in d) {
  112. proportion += parseInt(d[i].proportion);
  113. }
  114. if (parseInt(proportion) !== 100) {
  115. createMessage.warn('权重之和应为100');
  116. return;
  117. }
  118. loading.value = true;
  119. // for(let i in d){
  120. // delete d[i].key
  121. // }
  122. postAction('/adweb/executeNode/saveFlowTemplate', d, 120000)
  123. .then((res) => {
  124. loading.value = false;
  125. if (res.code == 200) {
  126. createMessage.success('保存成功');
  127. } else {
  128. createMessage.warn(res.message);
  129. }
  130. })
  131. .catch((e) => {
  132. createMessage.warn('保存数据失败!');
  133. });
  134. }
  135. function addNew() {
  136. data.value.push({
  137. createTime: null,
  138. delFlag: 0,
  139. description: '',
  140. finTime: null,
  141. finUid: null,
  142. hostId: hostId,
  143. id: '',
  144. name: '',
  145. nodeType: 3,
  146. proportion: null,
  147. sort: 1,
  148. status: null,
  149. timeDesc: null,
  150. type: 1,
  151. key: data.value.length,
  152. });
  153. cacheData = data.value;
  154. }
  155. //排序
  156. function up(r, i) {
  157. if (i == 0) {
  158. return;
  159. }
  160. data.value.splice(i - 1, 0, data[i]);
  161. data.value.splice(i + 1, 1);
  162. }
  163. function down(r, i) {
  164. if (i == data.value.length - 1) {
  165. return;
  166. }
  167. data.value.splice(i + 2, 0, data[i]);
  168. data.value.splice(i, 1);
  169. }
  170. function deleteRow(i) {
  171. data.value.splice(i, 1);
  172. }
  173. function onClose() {
  174. data.value = [];
  175. }
  176. function init(r, t) {
  177. record = r;
  178. if (t == 'seo') {
  179. url.value = '/adweb/executeNode/querySEOTemplateFlow?planId=' + r.id;
  180. } else if (t == 'enquiry') {
  181. url.value = '/adweb/executeNode/queryEnquiryVerifyTemplateFlow';
  182. } else {
  183. url.value = '/adweb/executeNode/querySiteBuildTemplateFlow';
  184. }
  185. showStep();
  186. }
  187. function showStep() {
  188. tableLoading.value = true;
  189. getAction(url.value, {})
  190. .then((res) => {
  191. tableLoading.value = false;
  192. if (res.success) {
  193. res.result.map((item, index) => {
  194. data.value.push(Object.assign({}, item, { key: index }));
  195. cacheData = data.value;
  196. hostId.value = res.result[0].hostId;
  197. });
  198. }
  199. })
  200. .catch((e) => {
  201. createMessage.warn('获取数据失败!', e.message());
  202. });
  203. }
  204. function handleChange(value, key, column) {
  205. const newData = [...data.value];
  206. const target = newData.filter((item) => key === item.key)[0];
  207. if (target) {
  208. target[column] = value;
  209. data.value = newData;
  210. }
  211. }
  212. // function edit(key) {
  213. // const newData = [...data.value];
  214. // const target = newData.filter((item) => key === item.key)[0];
  215. // editingKey.value = key;
  216. // if (target) {
  217. // target.editable = true;
  218. // data.value = newData;
  219. // }
  220. // }
  221. // function save(key) {
  222. // const newData = [...data.value];
  223. // const newCacheData = [...cacheData];
  224. // const target = newData.filter((item) => key === item.key)[0];
  225. // const targetCache = newCacheData.filter((item) => key === item.key)[0];
  226. // if (target && targetCache) {
  227. // delete target.editable;
  228. // data.value = newData;
  229. //
  230. // console.log(data.value, '123123');
  231. // Object.assign(targetCache, target);
  232. // cacheData = newCacheData;
  233. // }
  234. // editingKey.value = '';
  235. // }
  236. const edit = (key: string) => {
  237. editableData[key] = cloneDeep(data.value.filter((item) => key === item.key)[0]);
  238. };
  239. const save = (key: string) => {
  240. Object.assign(data.value.filter((item) => key === item.key)[0], editableData[key]);
  241. delete editableData[key];
  242. };
  243. const cancel = (key: string) => {
  244. delete editableData[key];
  245. };
  246. // function cancel(key: string) {
  247. // delete data.value[key];
  248. // }
  249. defineExpose({ onClose, init });
  250. </script>
  251. <style scoped lang="less">
  252. .ant-tag.disabled {
  253. cursor: no-drop !important;
  254. opacity: 0.3;
  255. }
  256. </style>