IntegrityCheckingDrawer.vue 9.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318
  1. <template>
  2. <a-drawer title="物料检测标准" :width="1200" v-model:open="visible" :body-style="{ paddingBottom: '24px' }" @close="closeDraw">
  3. <a-row :gutter="16" style="margin-bottom: 20px">
  4. <a-col :span="3">
  5. <a-button type="primary" @click="submitAll" :loading="loading" :disabled="submitDisable"> 保存 </a-button>
  6. </a-col>
  7. </a-row>
  8. <a-table :columns="columns" :data-source="data" :pagination="false">
  9. <template v-for="col in ['weight']" #[col]="text, record, index" :key="col">
  10. <div>
  11. <a-input-number
  12. v-if="record.editable"
  13. style="margin: -5px 0"
  14. :value="text"
  15. :autoFocus="true"
  16. @change="(e) => handleChange(e, record.key, col)"
  17. :min="0"
  18. :step="0.1"
  19. :formatter="
  20. (value) => {
  21. let reg = /^(-)*(\d+)\.(\d).*$/;
  22. return `${value}`.replace(/\B(?=(\d{3})+(?!\d))/g, ',').replace(reg, '$1$2.$3');
  23. }
  24. "
  25. :parser="
  26. (value) => {
  27. let reg = /^(-)*(\d+)\.(\d).*$/;
  28. return value.replace(/\s?|(,*)/g, '').replace(reg, '$1$2.$3');
  29. }
  30. "
  31. />
  32. <template v-else>
  33. {{ text }}
  34. </template>
  35. </div>
  36. </template>
  37. <template v-for="col in ['limitMin']" #[col]="text, record, index" :key="col">
  38. <div>
  39. <a-input-number
  40. v-if="record.editable"
  41. style="margin: -5px 0"
  42. :value="text"
  43. @change="(e) => handleChange(e, record.key, col)"
  44. :min="0"
  45. :step="1"
  46. :autoFocus="true"
  47. :formatter="
  48. (value) => {
  49. if (typeof value === 'string') {
  50. return !isNaN(Number(value)) ? value.replace(/^(0+)|[^\d]/g, '') : '';
  51. } else if (typeof value === 'number') {
  52. return !isNaN(value) ? String(value).replace(/^(0+)|[^\d]/g, '') : '';
  53. } else {
  54. return '';
  55. }
  56. }
  57. "
  58. />
  59. <template v-else>
  60. {{ text }}
  61. </template>
  62. </div>
  63. </template>
  64. <template #required="text, record">
  65. <a-switch
  66. checked-children="是"
  67. un-checked-children="否"
  68. :autoFocus="true"
  69. :checked="record.required === 0"
  70. @change="changeRequired(record)"
  71. :loading="changeLoading"
  72. />
  73. </template>
  74. <template #operation="text, record, index">
  75. <div class="editable-row-operations">
  76. <span v-if="record.editable">
  77. <a @click="() => save(record.key)">保存</a>
  78. <a-divider type="vertical" />
  79. <a-popconfirm title="确定取消吗?" @confirm="() => cancel(record.key)">
  80. <a>取消</a>
  81. </a-popconfirm>
  82. </span>
  83. <span v-else>
  84. <a :disabled="editingKey !== ''" @click="() => edit(record.key)">编辑</a>
  85. </span>
  86. </div>
  87. </template>
  88. <template #sort="text, record, index">
  89. <a-tag color="pink" @click="up(index)" style="cursor: pointer" :class="index == 0 ? 'disabled' : ''">
  90. <a-icon type="arrow-up" />
  91. </a-tag>
  92. <a-tag color="blue" @click="down(index)" style="cursor: pointer" :class="index === data.length - 1 ? 'disabled' : ''">
  93. <a-icon type="arrow-down" />
  94. </a-tag>
  95. </template>
  96. </a-table>
  97. </a-drawer>
  98. </template>
  99. <script>
  100. import { getAction, postAction } from '/src/api/manage/manage';
  101. export default {
  102. name: 'IntegrityCheckingDrawer',
  103. data() {
  104. return {
  105. visible: false,
  106. columns: [
  107. {
  108. title: '物料名称',
  109. dataIndex: 'materialLabel',
  110. align: 'center',
  111. },
  112. {
  113. title: '最低限制',
  114. dataIndex: 'limitMin',
  115. align: 'center',
  116. scopedSlots: { customRender: 'limitMin' },
  117. },
  118. {
  119. title: '评分占比',
  120. dataIndex: 'weight',
  121. align: 'center',
  122. scopedSlots: { customRender: 'weight' },
  123. },
  124. {
  125. title: '是否必填',
  126. dataIndex: 'required',
  127. align: 'center',
  128. scopedSlots: { customRender: 'required' },
  129. },
  130. {
  131. title: '排序',
  132. dataIndex: 'sort',
  133. align: 'center',
  134. scopedSlots: { customRender: 'sort' },
  135. },
  136. {
  137. title: '操作',
  138. dataIndex: 'operation',
  139. align: 'center',
  140. scopedSlots: { customRender: 'operation' },
  141. },
  142. ],
  143. data: [],
  144. cacheData: [],
  145. editingKey: '',
  146. sortChanged: false, // 判断排序是否改变,排序改变则在关闭的时候进行统一保存
  147. loading: false,
  148. loading2: false,
  149. id: '',
  150. changeLoading: false,
  151. submitDisable: false,
  152. };
  153. },
  154. methods: {
  155. submitAll() {
  156. let d = this.data;
  157. let weight = 0;
  158. for (let i in d) {
  159. weight += parseFloat(d[i].weight);
  160. }
  161. if (parseFloat(weight) !== 100) {
  162. this.$message.error('权重之和应为100');
  163. return;
  164. }
  165. this.loading = true;
  166. postAction('/serp/seoMarketPlan/saveTemplate', {
  167. checkList: JSON.stringify(d),
  168. planId: this.id,
  169. })
  170. .then((res) => {
  171. this.loading = false;
  172. if (res.code == 200) {
  173. this.$message.success('保存成功');
  174. this.showDrawer(this.id);
  175. } else {
  176. this.$message.error(res.message);
  177. }
  178. })
  179. .catch((e) => {
  180. this.$message.error('保存数据失败!');
  181. });
  182. },
  183. showDrawer(id) {
  184. let that = this;
  185. that.id = id;
  186. getAction('/serp/seoMarketPlan/integrityChecking?planId=' + id)
  187. .then((res) => {
  188. if (res.success) {
  189. that.data = res.result;
  190. for (let i = 0; i < that.data.length; i++) {
  191. let item = that.data[i];
  192. item.key = item.id;
  193. }
  194. that.cacheData = that.data.map((item) => ({ ...item }));
  195. }
  196. })
  197. .catch((e) => {
  198. this.$message.warning('获取数据失败!');
  199. });
  200. this.visible = true;
  201. },
  202. closeDraw() {
  203. let that = this;
  204. that.visible = false;
  205. that.data = [];
  206. that.cacheData = [];
  207. that.sortChanged = false;
  208. that.record = {};
  209. },
  210. handleChange(value, key, column) {
  211. const newData = [...this.data];
  212. const target = newData.filter((item) => key === item.key)[0];
  213. if (target) {
  214. target[column] = value == null ? '' : value;
  215. this.data = newData;
  216. }
  217. },
  218. edit(key) {
  219. const newData = [...this.data];
  220. const target = newData.filter((item) => key === item.key)[0];
  221. this.editingKey = key;
  222. if (target) {
  223. target.editable = true;
  224. this.data = newData;
  225. }
  226. this.submitDisable = true;
  227. },
  228. save(key) {
  229. const newData = [...this.data];
  230. const newCacheData = [...this.cacheData];
  231. const target = newData.filter((item) => key === item.key)[0];
  232. const targetCache = newCacheData.filter((item) => key === item.key)[0];
  233. if (target && targetCache) {
  234. delete target.editable;
  235. this.data = newData;
  236. Object.assign(targetCache, target);
  237. this.cacheData = newCacheData;
  238. }
  239. this.editingKey = '';
  240. this.submitDisable = false;
  241. },
  242. cancel(key) {
  243. const newData = [...this.data];
  244. const target = newData.filter((item) => key === item.key)[0];
  245. this.editingKey = '';
  246. if (target) {
  247. Object.assign(target, this.cacheData.filter((item) => key === item.key)[0]);
  248. delete target.editable;
  249. this.data = newData;
  250. }
  251. this.submitDisable = false;
  252. },
  253. up(index) {
  254. let that = this;
  255. if (that.editingKey !== '') {
  256. that.$message.warn('请先完成编辑!');
  257. return;
  258. }
  259. if (index === 0) {
  260. return;
  261. }
  262. that.sortChanged = true;
  263. // 修改展示顺序
  264. let data2 = that.data;
  265. let temp1 = data2[index];
  266. data2[index] = data2[index - 1];
  267. data2[index - 1] = temp1;
  268. that.cacheData = data2.map((item) => ({ ...item }));
  269. that.data = that.cacheData;
  270. },
  271. down(index) {
  272. let that = this;
  273. if (that.editingKey !== '') {
  274. that.$message.warn('请先完成编辑!');
  275. return;
  276. }
  277. if (index === that.data.length - 1) {
  278. return;
  279. }
  280. that.sortChanged = true;
  281. // 修改展示顺序
  282. let data2 = that.data;
  283. let temp1 = data2[index];
  284. data2[index] = data2[index + 1];
  285. data2[index + 1] = temp1;
  286. that.cacheData = data2.map((item) => ({ ...item }));
  287. that.data = that.cacheData;
  288. },
  289. changeRequired(record) {
  290. let that = this;
  291. that.changeLoading = true;
  292. postAction('/materialcollect/changeRequired?id=' + record.id + '&planId=' + record.planId).then((res) => {
  293. if (res.success) {
  294. that.$message.success(res.result);
  295. that.showDrawer(that.id);
  296. } else {
  297. that.$message.error(res.message);
  298. }
  299. that.changeLoading = false;
  300. });
  301. },
  302. },
  303. };
  304. </script>
  305. <style lang="less" scoped>
  306. .ant-tag.disabled {
  307. cursor: no-drop !important;
  308. opacity: 0.3;
  309. }
  310. </style>