ctrl_price_controller.js 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204
  1. 'use strict';
  2. /**
  3. *
  4. *
  5. * @author Mai
  6. * @date 2021/10/27
  7. * @version
  8. */
  9. const stdDataAddType = {
  10. withParent: 1,
  11. child: 2,
  12. next: 3,
  13. };
  14. const LzString = require('lz-string');
  15. module.exports = app => {
  16. class CtrlPriceController extends app.BaseController {
  17. _getSpreadSetting() {
  18. const spreadSetting = {
  19. cols: [
  20. {title: '项目节编号', colSpan: '1', rowSpan: '2', field: 'code', hAlign: 0, width: 180, formatter: '@', cellType: 'tree'},
  21. {title: '清单编号', colSpan: '1', rowSpan: '2', field: 'b_code', hAlign: 0, width: 120, formatter: '@'},
  22. {title: '名称', colSpan: '1', rowSpan: '2', field: 'name', hAlign: 0, width: 230, formatter: '@'},
  23. {title: '单位', colSpan: '1', rowSpan: '2', field: 'unit', hAlign: 1, width: 50, formatter: '@', cellType: 'unit', comboEdit: true},
  24. {title: '设计数量|数量1', colSpan: '2|1', rowSpan: '1|1', field: 'dgn_qty1', hAlign: 2, width: 80, type: 'Number'},
  25. {title: '|数量2', colSpan: '|1', rowSpan: '|1', field: 'dgn_qty2', hAlign: 2, width: 80, type: 'Number'},
  26. {title: '经济指标', colSpan: '1', rowSpan: '2', field: 'dgn_price', hAlign: 2, width: 80, type: 'Number', readOnly: true},
  27. {title: '清单数量', colSpan: '1', rowSpan: '2', field: 'quantity', hAlign: 2, width: 80, type: 'Number'},
  28. {title: '单价', colSpan: '1', rowSpan: '2', field: 'unit_price', hAlign: 2, width: 80, type: 'Number'},
  29. {title: '金额', colSpan: '1', rowSpan: '2', field: 'total_price', hAlign: 2, width: 80, type: 'Number'},
  30. {title: '图册号', colSpan: '1', rowSpan: '2', field: 'drawing_code', hAlign: 0, width: 100, formatter: '@'},
  31. {title: '备注', colSpan: '1', rowSpan: '2', field: 'memo', hAlign: 0, width: 100, formatter: '@'},
  32. ],
  33. emptyRows: 3,
  34. headRows: 2,
  35. headRowHeight: [25, 25],
  36. defaultRowHeight: 21,
  37. headerFont: '12px 微软雅黑',
  38. font: '12px 微软雅黑',
  39. localCache: { key: 'budget', colWidth: true },
  40. };
  41. return spreadSetting;
  42. }
  43. async index(ctx) {
  44. try {
  45. await this.ctx.service.controlPrice.checkInit(ctx.tender);
  46. const [stdBills, stdChapters] = await this.ctx.service.valuation.getValuationStdList(
  47. ctx.tender.data.valuation, ctx.tender.data.measure_type);
  48. const renderData = {
  49. spreadSetting: this._getSpreadSetting(),
  50. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.tender.ctrlPrice),
  51. stdBills,
  52. stdChapters,
  53. };
  54. await this.layout('tender/ctrl_price.ejs', renderData, 'tender/ctrl_price_modal.ejs');
  55. } catch (err) {
  56. ctx.log(err);
  57. }
  58. }
  59. async load(ctx) {
  60. try {
  61. ctx.body = {
  62. err: 0, msg: '',
  63. data: await this.ctx.service.controlPrice.getData(ctx.tender.id),
  64. }
  65. } catch (err) {
  66. ctx.log(err);
  67. ctx.ajaxErrorBody(err, '获取数据错误');
  68. }
  69. }
  70. _checkCanUpdate(ctx) {
  71. if (ctx.tender.isTourist) throw '您无权修改数据';
  72. }
  73. async _billsBase(relaService, type, data) {
  74. if (isNaN(data.id) || data.id <= 0) throw '数据错误';
  75. if (type !== 'add') {
  76. if (isNaN(data.count) || data.count <= 0) data.count = 1;
  77. }
  78. switch (type) {
  79. case 'add':
  80. return await relaService.addNodeBatch(this.ctx.tender.id, data.id, {}, data.count);
  81. case 'delete':
  82. return await relaService.delete(this.ctx.tender.id, data.id, data.count);
  83. case 'up-move':
  84. return await relaService.upMoveNode(this.ctx.tender.id, data.id, data.count);
  85. case 'down-move':
  86. return await relaService.downMoveNode(this.ctx.tender.id, data.id, data.count);
  87. case 'up-level':
  88. return await relaService.upLevelNode(this.ctx.tender.id, data.id, data.count);
  89. case 'down-level':
  90. return await relaService.downLevelNode(this.ctx.tender.id, data.id, data.count);
  91. }
  92. }
  93. async _addStd(relaService, data) {
  94. if ((isNaN(data.id) || data.id <= 0) || !data.stdType || !data.stdNode) throw '参数错误';
  95. let stdLib, addType;
  96. switch (data.stdType) {
  97. case 'xmj':
  98. stdLib = this.ctx.service.stdXmj;
  99. addType = stdDataAddType.withParent;
  100. break;
  101. case 'gcl':
  102. stdLib = this.ctx.service.stdGcl;
  103. const selectNode = await relaService.getDataByKid(this.ctx.tender.id, data.id);
  104. addType = selectNode.b_code ? stdDataAddType.next : stdDataAddType.child;
  105. break;
  106. default:
  107. throw '未知标准库';
  108. }
  109. const stdData = await stdLib.getDataByDataId(data.stdLibId, data.stdNode);
  110. switch (addType) {
  111. case stdDataAddType.child:
  112. return await relaService.addStdNodeAsChild(this.ctx.tender.id, data.id, stdData);
  113. case stdDataAddType.next:
  114. return await relaService.addStdNode(this.ctx.tender.id, data.id, stdData);
  115. case stdDataAddType.withParent:
  116. return await relaService.addStdNodeWithParent(this.ctx.tender.id, stdData, stdLib);
  117. default:
  118. throw '未知添加方式';
  119. }
  120. }
  121. async _pasteBlock(relaService, data) {
  122. if ((isNaN(data.id) || data.id <= 0) ||
  123. (!data.tid && data.tid <= 0) ||
  124. (!data.block || data.block.length <= 0)) throw '参数错误';
  125. return await relaService.pasteBlockData(this.ctx.tender.id, data.id, data.block);
  126. }
  127. async update(ctx) {
  128. try {
  129. if (!ctx.tender) throw '项目数据错误';
  130. this._checkCanUpdate(ctx);
  131. const relaService = this.ctx.service.controlPrice;
  132. const data = JSON.parse(ctx.request.body.data);
  133. if (!data.postType || !data.postData) throw '数据错误';
  134. const responseData = { err: 0, msg: '', data: {} };
  135. switch (data.postType) {
  136. case 'add':
  137. case 'delete':
  138. case 'up-move':
  139. case 'down-move':
  140. case 'up-level':
  141. case 'down-level':
  142. responseData.data = await this._billsBase(relaService, data.postType, data.postData);
  143. break;
  144. case 'update':
  145. ctx.helper.checkDgnQtyPrecision(data.postData);
  146. responseData.data = await relaService.updateCalc(ctx.tender.id, data.postData);
  147. break;
  148. case 'add-std':
  149. responseData.data = await this._addStd(relaService, data.postData);
  150. break;
  151. case 'paste-block':
  152. responseData.data = await this._pasteBlock(relaService, data.postData);
  153. break;
  154. default:
  155. throw '未知操作';
  156. }
  157. ctx.body = responseData;
  158. } catch (err) {
  159. this.log(err);
  160. ctx.body = this.ajaxErrorBody(err, '数据错误');
  161. }
  162. }
  163. async uploadExcel(ctx) {
  164. try {
  165. if (!ctx.tender) throw '项目数据错误';
  166. this._checkCanUpdate(ctx);
  167. const ueType = ctx.params.ueType;
  168. const compressData = ctx.request.body.data;
  169. const data = JSON.parse(LzString.decompressFromUTF16(compressData));
  170. const responseData = { err: 0, msg: '', data: {} };
  171. switch (ueType) {
  172. case 'tz':
  173. const templateId = await this.ctx.service.valuation.getValuationTemplate(
  174. this.ctx.tender.data.valuation, this.ctx.tender.data.measure_type);
  175. responseData.data = await this.ctx.service.controlPrice.importExcel(templateId, data.sheet, data.filter);
  176. break;
  177. case 'gcl2xmj':
  178. responseData.data = await this.ctx.service.controlPrice.importGclExcel(data.id, data.sheet);
  179. break;
  180. default:
  181. throw '数据错误';
  182. }
  183. ctx.body = responseData;
  184. } catch (err) {
  185. this.log(err);
  186. ctx.body = { err: 1, msg: err.toString(), data: null };
  187. }
  188. }
  189. }
  190. return CtrlPriceController;
  191. };