tender_controller.js 74 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511
  1. 'use strict';
  2. /**
  3. * 标段管理控制器
  4. *
  5. * @author CaiAoLin
  6. * @date 2018/2/5
  7. * @version
  8. */
  9. const tenderConst = require('../const/tender');
  10. const codeRuleConst = require('../const/code_rule');
  11. const settingConst = require('../const/setting.js');
  12. const tenderMenu = require('../../config/menu').tenderMenu;
  13. const auditConst = require('../const/audit');
  14. const shenpiConst = require('../const/shenpi');
  15. const accountGroup = require('../const/account_group').group;
  16. const accountPermission = require('../const/account_permission');
  17. const measureType = require('../const/tender').measureType;
  18. const billsPosConvert = require('../lib/bills_pos_convert');
  19. const path = require('path');
  20. const sendToWormhole = require('stream-wormhole');
  21. const scheduleConst = require('../const/schedule');
  22. const changeConst = require('../const/change');
  23. const tenderInfoModel = require('../lib/tender_info');
  24. const mapConst = require('../const/map');
  25. const advanceConst = require('../const/advance');
  26. module.exports = app => {
  27. class TenderController extends app.BaseController {
  28. /**
  29. * 构造函数
  30. *
  31. * @param {Object} ctx - egg全局变量
  32. * @return {void}
  33. */
  34. constructor(ctx) {
  35. super(ctx);
  36. ctx.showProject = true;
  37. ctx.showTitle = true;
  38. }
  39. async _getLedgerAuditInfo(tender) {
  40. tender.cur_flow = {
  41. title: '台账',
  42. status: auditConst.ledger.tiStatusString[tender.ledger_status],
  43. status_class: auditConst.ledger.tiStatusStringClass[tender.ledger_status],
  44. };
  45. if (tender.ledger_status === auditConst.ledger.status.uncheck) {
  46. tender.cur_flow.name = tender.user_name;
  47. } else {
  48. const cur = tender.ledger_status === auditConst.ledger.status.checkNo
  49. ? await this.ctx.service.ledgerAudit.getLastestAuditor(tender.id, tender.ledger_times - 1, auditConst.ledger.status.checkNo)
  50. : await this.ctx.service.ledgerAudit.getLastestAuditor(tender.id, tender.ledger_times, tender.ledger_status);
  51. if (cur) {
  52. tender.cur_flow.name = cur.name;
  53. if (cur.audit_order === 1) {
  54. tender.pre_flow = { name: tender.user_name, time: cur.begin_time };
  55. } else {
  56. const pre = await this.ctx.service.ledgerAudit.getAuditorByOrder(tender.id, cur.audit_order - 1, cur.times);
  57. if (pre) tender.pre_flow = { name: pre.name, time: pre.end_time };
  58. }
  59. } else {
  60. tender.cur_flow.name = '';
  61. }
  62. }
  63. }
  64. async _getStageAuditInfo(tender, stage) {
  65. tender.cur_flow = {
  66. title: '第' + stage.order + '期',
  67. status: auditConst.stage.tiStatusString[stage.status],
  68. status_class: auditConst.stage.tiStatusStringClass[stage.status],
  69. };
  70. if (stage.status === auditConst.stage.status.uncheck) {
  71. if (tender.user_id === stage.user_id) {
  72. tender.cur_flow.name = tender.user_name;
  73. } else {
  74. const user = await this.ctx.service.projectAccount.getDataById(stage.user_id);
  75. tender.cur_flow.name = user.name;
  76. }
  77. if (stage.order > 1) {
  78. const preStage = await this.ctx.service.stage.getDataByCondition({ tid: tender.id, order: stage.order - 1 });
  79. if (!preStage) return;
  80. const pre = await this.ctx.service.stageAudit.getLastestAuditor(preStage.id, preStage.times, auditConst.stage.status.checked);
  81. if (pre) tender.pre_flow = { name: pre.name, time: pre.end_time };
  82. }
  83. } else {
  84. let cur;
  85. if (stage.status === auditConst.stage.status.checkNo) {
  86. cur = await this.ctx.service.stageAudit.getLastestAuditor(stage.id, stage.times - 1, auditConst.stage.status.checkNo);
  87. } else if (stage.status === auditConst.stage.status.checked) {
  88. cur = await this.ctx.service.stageAudit.getLastestAuditor(stage.id, stage.times, auditConst.stage.status.checked);
  89. } else {
  90. cur = await this.ctx.service.stageAudit.getCurAuditor(stage.id, stage.times);
  91. }
  92. if (cur) {
  93. tender.cur_flow.name = cur.name;
  94. if (cur.order === 1) {
  95. tender.pre_flow = {};
  96. if (tender.user_id === stage.user_id) {
  97. tender.pre_flow.name = tender.user_name;
  98. } else {
  99. const user = await this.ctx.service.projectAccount.getDataById(stage.user_id);
  100. tender.pre_flow.name = user.name;
  101. }
  102. tender.pre_flow.time = cur.begin_time;
  103. } else {
  104. const pre = await this.ctx.service.stageAudit.getAuditorByOrder(stage.id, cur.order - 1, cur.times);
  105. if (pre) tender.pre_flow = { name: pre.name, time: pre.end_time };
  106. }
  107. } else {
  108. tender.cur_flow.name = '';
  109. }
  110. }
  111. }
  112. async _listDetail(view, modal = '') {
  113. try {
  114. // 获取用户新建标段权利
  115. const accountInfo = await this.ctx.service.projectAccount.getDataById(this.ctx.session.sessionUser.accountId);
  116. const userPermission = accountInfo !== undefined && accountInfo.permission !== ''
  117. ? JSON.parse(accountInfo.permission) : null;
  118. const tenderList = await this.ctx.service.tender.getList('', userPermission, this.ctx.session.sessionUser.is_admin);
  119. for (const t of tenderList) {
  120. const tenderInfo = await this.ctx.service.tenderInfo.getTenderInfo(t.id);
  121. t.contract_price = tenderInfo.deal_param.contractPrice;
  122. let bCalcTp = t.user_id === this.ctx.session.sessionUser.accountId && (
  123. t.ledger_status === auditConst.ledger.status.checkNo || t.ledger_status === auditConst.ledger.status.uncheck);
  124. t.advance_tp = await this.ctx.service.advance.getSumAdvance(t.id);
  125. if (t.ledger_status === auditConst.ledger.status.checked) {
  126. t.lastStage = await this.ctx.service.stage.getLastestStage(t.id, true);
  127. if (t.lastStage && t.lastStage.status === auditConst.stage.status.uncheck &&
  128. t.lastStage.user_id !== this.ctx.session.sessionUser.accountId) {
  129. t.lastStage = await this.ctx.service.stage.getLastestStage(t.id);
  130. }
  131. if (t.lastStage) await this.ctx.service.stage.checkStageGatherData(t.lastStage, this.ctx.session.sessionUser.is_admin);
  132. t.completeStage = await this.ctx.service.stage.getLastestCompleteStage(t.id);
  133. if ((!bCalcTp) && t.measure_type === measureType.gcl.value) {
  134. bCalcTp = t.lastStage && t.lastStage.status !== auditConst.stage.status.checked && !t.lastStage.readOnly;
  135. }
  136. }
  137. if (bCalcTp) {
  138. const sum = await this.ctx.service.ledger.addUp({ tender_id: t.id/* , is_leaf: true*/ });
  139. t.total_price = sum.total_price;
  140. t.deal_tp = sum.deal_tp;
  141. }
  142. if (t.lastStage) {
  143. await this._getStageAuditInfo(t, t.lastStage);
  144. } else {
  145. await this._getLedgerAuditInfo(t);
  146. }
  147. }
  148. const categoryData = await this.ctx.service.category.getAllCategory(this.ctx.session.sessionProject.id);
  149. const valuations = await this.ctx.service.valuation.getProjectValidValuation(this.ctx.session.sessionProject.id);
  150. const renderData = {
  151. tenderList,
  152. tenderConst,
  153. settingConst,
  154. categoryData,
  155. measureType: tenderConst.measureType,
  156. jsFiles: this.app.jsFiles.common.concat(this.jsFiles),
  157. auditConst,
  158. userPermission,
  159. valuations,
  160. uid: this.ctx.session.sessionUser.accountId,
  161. pid: this.ctx.session.sessionProject.id,
  162. };
  163. await this.layout(view, renderData, modal);
  164. } catch (err) {
  165. this.log(err);
  166. this.ctx.redirect('/dashboard');
  167. }
  168. }
  169. async _list(view, renderData, modal = '', list_status = '') {
  170. try {
  171. renderData.tenderList = await this.ctx.service.tender.getList(list_status, renderData.userPermission, this.ctx.session.sessionUser.is_admin);
  172. for (const t of renderData.tenderList) {
  173. if (t.ledger_status === auditConst.ledger.status.checked) {
  174. t.lastStage = await this.ctx.service.stage.getLastestStage(t.id, true);
  175. t.completeStage = await this.ctx.service.stage.getLastestCompleteStage(t.id);
  176. }
  177. if (t.lastStage) {
  178. if (t.lastStage.status === auditConst.stage.status.uncheck) {
  179. const status_name = await this.ctx.service.projectAccount.getAccountInfoById(t.lastStage.user_id);
  180. t.status_users = status_name ? status_name.name : '';
  181. // const status_name = await this.ctx.service.stageAudit.getStatusName(t.lastStage.id, t.lastStage.times - 1);
  182. // t.status_users = status_name ? status_name.name : '';
  183. } else {
  184. const status_name = await this.ctx.service.stageAudit.getAuditorByStatus(t.lastStage.id, t.lastStage.status, t.lastStage.times);
  185. t.status_users = status_name ? status_name.name : '';
  186. }
  187. } else {
  188. if (t.ledger_status !== auditConst.ledger.status.uncheck) {
  189. const status_name = await this.ctx.service.ledgerAudit.getStatusName(t.id, t.ledger_times);
  190. t.status_users = status_name ? status_name.name : '';
  191. }
  192. }
  193. }
  194. renderData.categoryData = await this.ctx.service.category.getAllCategory(this.ctx.session.sessionProject.id);
  195. renderData.valuations = await this.ctx.service.valuation.getProjectValidValuation(this.ctx.session.sessionProject.id);
  196. renderData.tenderConst = tenderConst;
  197. renderData.settingConst = settingConst;
  198. renderData.measureType = tenderConst.measureType;
  199. renderData.jsFiles = this.app.jsFiles.common.concat(this.jsFiles);
  200. renderData.auditConst = auditConst;
  201. renderData.uid = this.ctx.session.sessionUser.accountId;
  202. renderData.pid = this.ctx.session.sessionProject.id;
  203. await this.layout(view, renderData, modal);
  204. } catch (err) {
  205. this.log(err);
  206. this.ctx.redirect('/dashboard');
  207. }
  208. }
  209. async listDefault(ctx) {
  210. this.jsFiles = this.app.jsFiles.tender.list;
  211. const accountInfo = await this.ctx.service.projectAccount.getDataById(this.ctx.session.sessionUser.accountId);
  212. const userPermission = accountInfo !== undefined && accountInfo.permission !== '' ? JSON.parse(accountInfo.permission) : null;
  213. const renderData = {
  214. accountInfo,
  215. userPermission,
  216. };
  217. await this._list('tender/index.ejs', renderData, 'tender/modal.ejs');
  218. }
  219. /**
  220. * 标段概况(Get)
  221. *
  222. * @param {object} ctx - egg全局变量
  223. * @return {void}
  224. */
  225. async listInfo(ctx) {
  226. this.jsFiles = this.app.jsFiles.tender.info;
  227. await this._listDetail('tender/info.ejs', 'tender/modal.ejs');
  228. }
  229. /**
  230. * 计量进度(Get)
  231. *
  232. * @param ctx
  233. * @return {Promise<void>}
  234. */
  235. async listProgress(ctx) {
  236. this.jsFiles = this.app.jsFiles.tender.progress;
  237. await this._listDetail('tender/progress.ejs', 'tender/modal.ejs');
  238. }
  239. /**
  240. * 标段管理(Get)
  241. *
  242. * @param ctx
  243. * @return {Promise<void>}
  244. */
  245. async listManage(ctx) {
  246. this.jsFiles = this.app.jsFiles.tender.manage;
  247. // 先判断权限
  248. // 获取用户新建标段权利
  249. const accountInfo = await this.ctx.service.projectAccount.getDataById(this.ctx.session.sessionUser.accountId);
  250. const userPermission = accountInfo !== undefined && accountInfo.permission !== '' ? JSON.parse(accountInfo.permission) : null;
  251. if (userPermission !== null && userPermission.tender !== undefined && userPermission.tender.indexOf('1') !== -1) {
  252. const renderData = {
  253. accountInfo,
  254. userPermission,
  255. };
  256. await this._list('tender/manage.ejs', renderData, 'tender/manage_modal.ejs', 'manage');
  257. // 获取用户新建标段权利
  258. // const accountInfo = await this.ctx.service.projectAccount.getDataById(this.ctx.session.sessionUser.accountId);
  259. // const userPermission = accountInfo !== undefined && accountInfo.permission !== '' ? JSON.parse(accountInfo.permission) : null;
  260. //
  261. // const tenderList = await this.ctx.service.tender.getList('manage', userPermission);
  262. // for (const t of tenderList) {
  263. // t.lastStage = await this.ctx.service.stage.getLastestStage(t.id, true);
  264. // t.completeStage = await this.ctx.service.stage.getLastestCompleteStage(t.id);
  265. // }
  266. // const categoryData = await this.ctx.service.category.getAllCategory(this.ctx.session.sessionProject.id);
  267. // const valuations = await this.ctx.service.valuation.getProjectValidValuation(this.ctx.session.sessionProject.id);
  268. // const renderData = {
  269. // tenderList,
  270. // tenderConst,
  271. // settingConst,
  272. // categoryData,
  273. // measureType: tenderConst.measureType,
  274. // jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.tender.manage),
  275. // auditConst,
  276. // userPermission,
  277. // valuations,
  278. // };
  279. // await this.layout('tender/manage.ejs', renderData, 'tender/manage_modal.ejs');
  280. } else {
  281. this.ctx.redirect(ctx.request.header.referer);
  282. }
  283. }
  284. /**
  285. * 新增标段(Ajax)
  286. *
  287. * @param ctx
  288. * @return {Promise<void>}
  289. */
  290. async addTender(ctx) {
  291. try {
  292. const responseData = {
  293. err: 0, msg: '', data: null,
  294. };
  295. // 获取用户新建标段权利
  296. const accountInfo = await this.ctx.service.projectAccount.getDataById(this.ctx.session.sessionUser.accountId);
  297. const userPermission = accountInfo !== undefined && accountInfo.permission !== '' ? JSON.parse(accountInfo.permission) : null;
  298. if (userPermission === null || userPermission.tender === undefined || userPermission.tender.indexOf('1') === -1) {
  299. throw '当前用户没有创建标段的权限';
  300. }
  301. const data = JSON.parse(ctx.request.body.data);
  302. if (!data.name || data.name === '' || !data.valuation) {
  303. throw '标段信息不完整';
  304. }
  305. responseData.data = await ctx.service.tender.add(data);
  306. ctx.body = responseData;
  307. } catch (error) {
  308. this.log(error);
  309. ctx.body = { err: 1, msg: error.toString(), data: null };
  310. }
  311. }
  312. /**
  313. * 编辑标段(Ajax)
  314. *
  315. * @param ctx
  316. * @return {Promise<void>}
  317. */
  318. async updateTender(ctx) {
  319. try {
  320. const responseData = {
  321. err: 0, msg: '', data: null,
  322. };
  323. const data = JSON.parse(ctx.request.body.data);
  324. if (!data.id) {
  325. throw '提交信息错误';
  326. }
  327. if (!data.name || data.name === '') {
  328. throw '标段信息不完整';
  329. }
  330. if (await ctx.service.tender.save(data, data.id)) {
  331. responseData.data = await ctx.service.tender.getTender(data.id);
  332. }
  333. ctx.body = responseData;
  334. } catch (error) {
  335. this.log(error);
  336. ctx.body = { err: 1, msg: error.toString(), data: null };
  337. }
  338. }
  339. /**
  340. * 删除标段(Ajax)
  341. *
  342. * @param ctx
  343. * @return {Promise<void>}
  344. */
  345. async deleteTender(ctx) {
  346. try {
  347. const data = JSON.parse(ctx.request.body.data),
  348. result = [];
  349. if (!(data instanceof Array) && (data.length === 0)) {
  350. throw '提交数据有误';
  351. }
  352. for (const id of data) {
  353. if (await ctx.service.tender.deleteTenderNoBackup(id)) {
  354. result.push(id);
  355. }
  356. }
  357. ctx.body = { err: 0, msg: '', data: result };
  358. } catch (err) {
  359. ctx.body = { err: 1, msg: err.toString(), data: [] };
  360. }
  361. }
  362. /**
  363. * 标段概况(Get)
  364. *
  365. * @param ctx
  366. * @return {Promise<void>}
  367. */
  368. async tenderInfo(ctx) {
  369. try {
  370. const tender = ctx.tender.data;
  371. let bCalcTp = tender.user_id === this.ctx.session.sessionUser.accountId && (
  372. tender.ledger_status === auditConst.ledger.status.checkNo || tender.ledger_status === auditConst.ledger.status.uncheck);
  373. const stages = await ctx.service.stage.getValidStages(ctx.tender.id);
  374. const lastStage = stages.length > 0 ? stages[0] : null; // await ctx.service.stage.getLastestStage(ctx.tender.id);
  375. if (lastStage) {
  376. await this.ctx.service.stage.checkStageGatherData(lastStage, this.ctx.session.sessionUser.is_admin);
  377. if ((!bCalcTp) && tender.measure_type === measureType.gcl.value) {
  378. bCalcTp = lastStage.status !== auditConst.stage.status.checked && !lastStage.readOnly;
  379. }
  380. if (bCalcTp) {
  381. const sum = await this.ctx.service.ledger.addUp({ tender_id: tender.id/* , is_leaf: true*/ });
  382. tender.total_price = sum.total_price;
  383. tender.deal_tp = sum.deal_tp;
  384. }
  385. tender.gather_tp = ctx.helper.sum([lastStage.contract_tp, lastStage.qc_tp, lastStage.pc_tp]);
  386. tender.end_contract_tp = ctx.helper.sum([lastStage.contract_tp, lastStage.pre_contract_tp, lastStage.contract_pc_tp]);
  387. tender.end_qc_tp = ctx.helper.sum([lastStage.qc_tp, lastStage.pre_qc_tp, lastStage.qc_pc_tp]);
  388. tender.end_positive_qc_tp = ctx.helper.sum([lastStage.positive_qc_tp, lastStage.pre_positive_qc_tp, lastStage.positive_qc_pc_tp]);
  389. tender.end_negative_qc_tp = ctx.helper.sum([lastStage.negative_qc_tp, lastStage.pre_negative_qc_tp, lastStage.negative_qc_pc_tp]);
  390. tender.end_gather_tp = ctx.helper.add(tender.end_contract_tp, tender.end_qc_tp);
  391. tender.pre_gather_tp = ctx.helper.add(lastStage.pre_contract_tp, lastStage.pre_qc_tp);
  392. tender.yf_tp = lastStage.yf_tp;
  393. const [change_tp, change_p_tp, change_n_tp] = await ctx.service.change.getChangeTp(tender.id);
  394. tender.change_tp = change_tp;
  395. tender.change_p_tp = change_p_tp;
  396. tender.change_n_tp = change_n_tp;
  397. tender.qc_ratio = ctx.helper.mul(ctx.helper.div(tender.end_qc_tp, ctx.tender.info.deal_param.contractPrice, 2), 100);
  398. tender.sum = ctx.helper.add(tender.total_price, tender.change_tp);
  399. tender.pre_ratio = ctx.helper.mul(ctx.helper.div(tender.pre_gather_tp, tender.sum, 2), 100);
  400. tender.cur_ratio = ctx.helper.mul(ctx.helper.div(tender.gather_tp, tender.sum, 2), 100);
  401. tender.other_tp = ctx.helper.sub(ctx.helper.sub(tender.sum, tender.pre_gather_tp), tender.gather_tp);
  402. tender.other_ratio = Math.max(0, 100 - tender.pre_ratio - tender.cur_ratio);
  403. tender.end_yf_tp = ctx.helper.add(lastStage.yf_tp, lastStage.pre_yf_tp);
  404. tender.end_sf_tp = ctx.helper.add(lastStage.sf_tp, lastStage.pre_sf_tp);
  405. tender.undone_tp = ctx.helper.sub(ctx.helper.sub(ctx.helper.add(tender.total_price, change_tp), tender.end_contract_tp), tender.end_qc_tp);
  406. if (lastStage.status === auditConst.stage.status.uncheck) {
  407. const status_name = await this.ctx.service.projectAccount.getAccountInfoById(lastStage.user_id);
  408. lastStage.status_users = status_name ? status_name.name : '';
  409. lastStage.auditors = [];
  410. } else {
  411. lastStage.status = lastStage.status === auditConst.stage.status.checkNoPre ? auditConst.stage.status.checking : lastStage.status;
  412. let cur;
  413. if (lastStage.status === auditConst.stage.status.checked) {
  414. cur = await this.ctx.service.stageAudit.getLastestAuditor(lastStage.id, lastStage.times, auditConst.stage.status.checked);
  415. } else if (lastStage.status === auditConst.stage.status.checking) {
  416. cur = await this.ctx.service.stageAudit.getCurAuditor(lastStage.id, lastStage.times);
  417. } else {
  418. cur = await this.ctx.service.stageAudit.getAuditorByStatus(lastStage.id, lastStage.status, lastStage.times);
  419. }
  420. // const status_name = await this.ctx.service.stageAudit.getAuditorByStatus(lastStage.id, lastStage.status, lastStage.times);
  421. lastStage.status_users = cur ? cur.name : '';
  422. const times = lastStage.status === auditConst.stage.status.checkNo ? lastStage.times - 1 : lastStage.times;
  423. lastStage.auditors = await ctx.service.stageAudit.getFinalAuditGroup(lastStage.id, times);
  424. }
  425. } else {
  426. if (bCalcTp) {
  427. const sum = await this.ctx.service.ledger.addUp({ tender_id: tender.id/* , is_leaf: true*/ });
  428. tender.total_price = sum.total_price;
  429. tender.deal_tp = sum.deal_tp;
  430. }
  431. if (tender.ledger_status !== auditConst.ledger.status.uncheck) {
  432. const status_name = await this.ctx.service.ledgerAudit.getStatusName(tender.id, tender.ledger_times);
  433. tender.status_users = status_name ? status_name.name : '';
  434. const times = tender.ledger_status === auditConst.ledger.status.checkNo ? tender.ledger_times - 1 : tender.ledger_times;
  435. tender.auditors = await ctx.service.ledgerAudit.getFinalAuditGroup(tender.id, times);
  436. } else {
  437. const status_name = await this.ctx.service.projectAccount.getAccountInfoById(tender.user_id);
  438. tender.status_users = status_name ? status_name.name : '';
  439. }
  440. }
  441. const tiModel = new tenderInfoModel(ctx);
  442. const gclChapter = await tiModel.gatherChapter();
  443. const monthProgress = [];
  444. for (const s of stages) {
  445. if (s.s_time) {
  446. let progress = monthProgress.find(function(x) {
  447. return x.month === s.s_time;
  448. });
  449. if (!progress) {
  450. progress = { month: s.s_time };
  451. monthProgress.push(progress);
  452. }
  453. progress.tp = ctx.helper.add(ctx.helper.add(progress.tp, s.contract_tp), s.qc_tp);
  454. }
  455. }
  456. monthProgress.sort(function(x, y) {
  457. return Date.parse(x.month) - Date.parse(y.month);
  458. });
  459. let sum = 0;
  460. for (const p of monthProgress) {
  461. p.ratio = ctx.helper.mul(ctx.helper.div(p.tp, tender.sum, 4), 100);
  462. sum = ctx.helper.add(sum, p.tp);
  463. p.end_tp = sum;
  464. p.end_ratio = ctx.helper.mul(ctx.helper.div(p.end_tp, tender.sum, 4), 100);
  465. }
  466. const revise = await ctx.service.ledgerRevise.getLastestRevise(tender.id);
  467. const tenders = await ctx.service.tender.getList('', null, 1);
  468. const categoryData = await ctx.service.category.getAllCategory(ctx.session.sessionProject.id);
  469. // 变更图表数据
  470. const change_done_total = await ctx.service.change.getCountByStatus2(tender.id, auditConst.filter.status.checked);
  471. const change_doing_total = await ctx.service.change.getCountByStatus2(tender.id, auditConst.filter.status.checking);
  472. const change_uncheck_total = await ctx.service.change.getCountByStatus2(tender.id, auditConst.filter.status.uncheck);
  473. const change_status_total = [
  474. { num: change_uncheck_total, name: '待上报' },
  475. { num: change_doing_total, name: '审批中' },
  476. { num: change_done_total, name: '已完成' },
  477. ];
  478. const change_common_total = await ctx.service.change.getCountByQuality(tender.id, changeConst.quality.common.value);
  479. const change_more_total = await ctx.service.change.getCountByQuality(tender.id, changeConst.quality.more.value);
  480. const change_great_total = await ctx.service.change.getCountByQuality(tender.id, changeConst.quality.great.value);
  481. const change_quality_total = [
  482. { num: change_common_total, name: '一般变更' },
  483. { num: change_more_total, name: '较大变更' },
  484. { num: change_great_total, name: '重大变更' },
  485. ];
  486. // 调差最新期数据
  487. const materials = await ctx.service.material.getValidMaterials(ctx.tender.id);
  488. let materialData = null;
  489. if (materials && materials.length > 0) {
  490. materialData = materials[0];
  491. materialData.curAuditor = await ctx.service.materialAudit.getAuditorByStatus(materialData.id, materialData.status, materialData.times);
  492. const times = materialData.status === auditConst.material.status.checkNo ? materialData.times - 1 : materialData.times;
  493. materialData.auditors = materialData.status === auditConst.material.status.uncheck ? [] : await ctx.service.materialAudit.getFinalAuditGroup(materialData.id, times);
  494. }
  495. // 修订完成数目
  496. // const reviseNum = await ctx.service.ledgerRevise.count({ tid: tender.id, status: auditConst.revise.status.checked });
  497. // 计量完成概况
  498. // tender.total_price
  499. const stage_total = [
  500. { num: tender.end_contract_tp ? tender.end_contract_tp : 0, name: '合同完成' },
  501. { num: tender.end_qc_tp ? tender.end_qc_tp : 0, name: '变更完成' },
  502. { num: tender.undone_tp ? tender.undone_tp : 0, name: '未完成' },
  503. ];
  504. // 地图
  505. const tenderALLInfo = await ctx.service.tender.getDataById(tender.id);
  506. const hadMap = tenderALLInfo.had_map === 0 ? 1 : tenderALLInfo.had_map;// 0为初始值,因为默认可能会变化,所以暂时把0都默认为1。共三种模式坐标模式1,图片模式2,无图模式3。
  507. const tenderMapList = await ctx.service.tenderMap.getAllDataByCondition({ where: { tid: tender.id } });
  508. // 默认坐标,否则则取办事处坐标
  509. const projectData = await ctx.service.project.getDataById(ctx.session.sessionProject.id);
  510. let map_json = {
  511. province: mapConst.map[0].province,
  512. lng: mapConst.map[0].lng,
  513. lat: mapConst.map[0].lat,
  514. level: 15,
  515. };
  516. if (projectData.map_json) {
  517. map_json = JSON.parse(projectData.map_json);
  518. } else {
  519. const mapInfo = ctx.helper._.find(mapConst.map, { office: projectData.manager_office });
  520. if (mapInfo) {
  521. map_json.province = mapInfo.province;
  522. map_json.lng = mapInfo.lng;
  523. map_json.lat = mapInfo.lat;
  524. }
  525. }
  526. const renderData = {
  527. tenders,
  528. categoryData,
  529. tender,
  530. revise,
  531. tenderInfo: ctx.tender.info,
  532. tenderMenu: this.menu.tenderMenu,
  533. preUrl: '/tender/' + ctx.tender.id,
  534. cooperation: ctx.session.sessionUser.cooperation,
  535. lastStage,
  536. stages: stages.reverse(),
  537. monthProgress,
  538. audit: auditConst,
  539. change_status_total,
  540. change_quality_total,
  541. materialData,
  542. // reviseNum,
  543. stage_total,
  544. hadMap,
  545. map_pic: tenderALLInfo.map_pic,
  546. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.tender.tenderInfo),
  547. gclChapter,
  548. tenderMapList,
  549. map_json,
  550. fujianOssPath: ctx.app.config.fujianOssPath,
  551. };
  552. if (ctx.session.sessionUser.is_admin) {
  553. renderData.tourists = await ctx.service.tenderTourist.getTourists(tender.id);
  554. for (const t of renderData.tourists) {
  555. t.permission = await ctx.service.tenderTourist.getTouristPermission(t);
  556. }
  557. // 获取所有项目参与者
  558. const accountList = await ctx.service.projectAccount.getAllDataByCondition({
  559. where: { project_id: ctx.session.sessionProject.id, enable: 1 },
  560. columns: ['id', 'name', 'company', 'role', 'enable', 'is_admin', 'account_group', 'mobile'],
  561. });
  562. const accountGroupList = accountGroup.map((item, idx) => {
  563. const groupList = accountList.filter(item => item.account_group === idx);
  564. return { groupName: item, groupList };
  565. });
  566. renderData.accountList = accountList;
  567. renderData.accountGroup = accountGroupList;
  568. }
  569. if (ctx.session.sessionProject.page_show.xxjd && ctx.session.sessionUser.is_admin) {
  570. // 投资进度内容
  571. renderData.scheduleAuditList = await ctx.service.scheduleAudit.getAllDataByCondition({ where: { tid: tender.id } });
  572. renderData.scPermission = scheduleConst.permission;
  573. }
  574. await this.layout('tender/detail.ejs', renderData, 'tender/detail_modal.ejs');
  575. } catch (error) {
  576. this.log(error);
  577. this.ctx.redirect('/list');
  578. }
  579. }
  580. /**
  581. * 保存标段属性等(Ajax)
  582. *
  583. * @param ctx
  584. * @return {Promise<void>}
  585. */
  586. async saveTenderInfo(ctx) {
  587. try {
  588. const data = JSON.parse(ctx.request.body.data);
  589. if (!data) {
  590. throw '提交数据错误';
  591. }
  592. // 针对查阅所有标段者但非原报和审批人提示
  593. const times = ctx.tender.data.ledger_status === auditConst.ledger.status.checkNo ? ctx.tender.data.ledger_times - 1 : ctx.tender.data.ledger_times;
  594. const auditors = await this.service.ledgerAudit.getAuditors(ctx.tender.id, times);
  595. const auditorsId = ctx.helper._.map(auditors, 'audit_id');
  596. const stageAuditors = await this.service.stageAudit.getAllAuditors(ctx.tender.id);
  597. const stageAUditorsId = ctx.helper._.map(stageAuditors, 'aid');
  598. const accountId = ctx.session.sessionUser.accountId;
  599. if (auditorsId.indexOf(accountId) === -1 && ctx.tender.data.user_id !== accountId &&
  600. stageAUditorsId.indexOf(accountId) === -1) {
  601. throw '您无权修改标段设置内容';
  602. }
  603. if (ctx.tender.data.ledger_status === auditConst.ledger.status.checked) {
  604. if (data.deal_param) {
  605. const lastStage = await this.ctx.service.stage.getLastestStage(ctx.tender.id, true);
  606. if (lastStage) {
  607. if (lastStage.order > 1 || (lastStage.status === auditConst.stage.status.checked || lastStage.status === auditConst.stage.status.checking)) throw '第一期上报后不可修改合同参数';
  608. if (lastStage.user_id !== ctx.session.sessionUser.accountId) throw '仅原报可修改合同参数';
  609. }
  610. }
  611. }
  612. if (data.decimal) {
  613. if (ctx.tender.data.user_id !== ctx.session.sessionUser.accountId) throw '仅原报可修改小数位数';
  614. await ctx.service.tenderInfo.saveDecimal(ctx.tender.id, data.decimal, ctx.tender.info.decimal);
  615. } else if (data.precision) {
  616. if (ctx.tender.data.user_id !== ctx.session.sessionUser.accountId) throw '仅原报可修改清单精度';
  617. await ctx.service.tenderInfo.savePrecision(ctx.tender.id,
  618. data.precision, ctx.tender.info.precision, ctx.tender.info.decimal);
  619. } else {
  620. await ctx.service.tenderInfo.saveTenderInfo(ctx.tender.id, data);
  621. }
  622. ctx.body = { err: 0, msg: '', data: JSON.parse(ctx.request.body.data) };
  623. } catch (err) {
  624. this.log(err);
  625. ctx.body = this.ajaxErrorBody(err, '保存标段设置失败');
  626. }
  627. }
  628. async saveTenderInfo2(ctx) {
  629. try {
  630. const data = JSON.parse(ctx.request.body.data);
  631. if (!data || (!data.ledger_check && !data.fun_rela)) throw '提交数据错误';
  632. if (!ctx.session.sessionUser.is_admin) throw '您无权修改该内容';
  633. const updateData = {};
  634. if (data.ledger_check) updateData.ledger_check = data.ledger_check;
  635. if (data.fun_rela) updateData.fun_rela = data.fun_rela;
  636. await ctx.service.tenderInfo.saveTenderInfo(ctx.tender.id, data);
  637. ctx.body = { err: 0, msg: '', data: JSON.parse(ctx.request.body.data) };
  638. } catch (err) {
  639. this.log(err);
  640. ctx.body = this.ajaxErrorBody(err, '保存失败');
  641. }
  642. }
  643. /**
  644. * 设置标段计量类型并调整到标段概况(Get)
  645. *
  646. * @param ctx
  647. * @return {Promise<void>}
  648. */
  649. async tenderType(ctx) {
  650. try {
  651. const tenderId = ctx.params.id,
  652. type = ctx.query.type;
  653. if (!tenderId) {
  654. throw '当前未打开标段';
  655. }
  656. await ctx.service.tender.checkTender(tenderId);
  657. if (!ctx.tender) {
  658. throw '标段数据错误';
  659. }
  660. if (!ctx.tender.measure_type) {
  661. await ctx.service.tender.setTenderType(ctx.tender, parseInt(type));
  662. }
  663. ctx.redirect('/tender/' + tenderId);
  664. } catch (error) {
  665. ctx.helper.log(error);
  666. this.postError(error, '设置标段类型错误');
  667. ctx.redirect('/list');
  668. }
  669. }
  670. /**
  671. * 标段协作办公
  672. *
  673. * @param {Object} ctx - egg全局变量
  674. * @return {void}
  675. */
  676. async tenderCooperation(ctx) {
  677. const tenderId = ctx.params.id;
  678. try {
  679. if (!ctx.session.sessionUser.cooperation) {
  680. throw '权限不足';
  681. }
  682. const tender = await ctx.service.tender.getDataById(tenderId);
  683. const user = await ctx.service.projectAccount.getDataById(ctx.session.sessionUser.accountId);
  684. // 获取已参与协作用户列表
  685. const cooperationArray = [];
  686. if (tender.cooperation !== null && tender.cooperation !== '') {
  687. const cooperationList = JSON.parse(tender.cooperation);
  688. for (const cl in cooperationList) {
  689. const clArray = [];
  690. for (const audit of cooperationList[cl]) {
  691. const userInfo = await ctx.service.projectAccount.getDataById(audit);
  692. clArray.push(userInfo);
  693. }
  694. cooperationArray[cl] = clArray;
  695. }
  696. }
  697. const renderData = {
  698. user,
  699. tender,
  700. tenderMenu: this.menu.tenderMenu,
  701. preUrl: '/tender/' + tenderId,
  702. tenderPermissionList: accountPermission.tenderPermissionList,
  703. cooperationArray,
  704. };
  705. await this.layout('tender/cooperation.ejs', renderData, 'tender/cooperationModal.ejs');
  706. } catch (error) {
  707. ctx.helper.log(error);
  708. this.ctx.redirect('/tender/' + tenderId);
  709. }
  710. }
  711. /**
  712. * 添加标段操作
  713. *
  714. * @param {Object} ctx - egg全局变量
  715. * @return {void}
  716. */
  717. async add(ctx) {
  718. try {
  719. const rule = ctx.service.tender.rule('add');
  720. ctx.helper.validate(rule);
  721. const result = ctx.service.tender.add(ctx.request.body);
  722. if (!result) {
  723. throw '新增标段失败';
  724. }
  725. } catch (error) {
  726. ctx.helper.log(error);
  727. this.setMessage(error.toString(), this.messageType.ERROR);
  728. }
  729. ctx.redirect(ctx.request.header.referer);
  730. }
  731. /**
  732. * 保存标段操作
  733. *
  734. * @param {Object} ctx - egg全局变量
  735. * @return {void}
  736. */
  737. async save(ctx) {
  738. let id = ctx.request.body.tenderId;
  739. id = parseInt(id);
  740. try {
  741. if (isNaN(id) || id < 0) {
  742. throw '参数错误';
  743. }
  744. // 获取数据规则
  745. const rule = ctx.service.tender.rule('save');
  746. ctx.validate(rule);
  747. const result = await ctx.service.tender.save(ctx.request.body, id);
  748. if (!result) {
  749. throw '保存标段数据失败';
  750. }
  751. this.setMessage('保存标段数据成功', this.messageType.SUCCESS);
  752. } catch (error) {
  753. this.postError(error, '保存标段数据失败');
  754. }
  755. ctx.redirect(ctx.request.header.referer);
  756. }
  757. /**
  758. * 删除标段
  759. *
  760. * @param {Object} ctx -egg全局变量
  761. * @return {void}
  762. */
  763. async delete(ctx) {
  764. let id = ctx.request.body.tenderId;
  765. id = parseInt(id);
  766. try {
  767. if (isNaN(id) || id <= 0) {
  768. throw '参数错误';
  769. }
  770. const result = ctx.service.tender.deleteTenderById(id);
  771. if (!result) {
  772. throw '删除标段失败';
  773. }
  774. this.setMessage('删除标段成功', this.messageType.SUCCESS);
  775. } catch (error) {
  776. this.postError(error, '删除标段失败');
  777. }
  778. ctx.redirect(ctx.request.header.referer);
  779. }
  780. async rule(ctx) {
  781. const responseData = {
  782. err: 0,
  783. msg: '',
  784. data: {},
  785. };
  786. try {
  787. const tenderId = ctx.session.sessionUser.tenderId;
  788. if (!tenderId) {
  789. throw '当前未打开标段';
  790. }
  791. const data = JSON.parse(ctx.request.body.data);
  792. if (!data.rule || !JSON.parse(data.data).length || !codeRuleConst.ruleField[data.rule]) {
  793. throw '请选择组件再添加';
  794. }
  795. if (!data.connector) {
  796. throw '请选择连接符';
  797. }
  798. const updateData = {
  799. id: tenderId,
  800. };
  801. if (data.type) {
  802. const tenderData = await ctx.service.tender.getDataById(tenderId);
  803. const c_code_rules = tenderData.c_code_rules ? JSON.parse(tenderData.c_code_rules) : {};
  804. c_code_rules[data.type + '_rule'] = JSON.parse(data.data);
  805. c_code_rules[data.type + '_rule_first'] = 0;
  806. c_code_rules[data.type + '_connector'] = data.connector;
  807. updateData.c_code_rules = JSON.stringify(c_code_rules);
  808. } else {
  809. updateData[codeRuleConst.ruleField[data.rule]] = data.data;
  810. updateData.c_connector = data.connector;
  811. updateData.c_rule_first = 0;
  812. }
  813. const result = await ctx.service.tender.db.update(ctx.service.tender.tableName, updateData);
  814. if (result.affectedRows !== 1) {
  815. throw '更新规则失败';
  816. }
  817. } catch (err) {
  818. ctx.helper.log(err);
  819. responseData.err = 1;
  820. responseData.msg = err.toString();
  821. }
  822. ctx.body = responseData;
  823. }
  824. async ruleFirst(ctx) {
  825. const responseData = {
  826. err: 0,
  827. msg: '',
  828. data: {},
  829. };
  830. try {
  831. const tenderId = ctx.session.sessionUser.tenderId;
  832. if (!tenderId) {
  833. throw '当前未打开标段';
  834. }
  835. const updateData = {
  836. id: tenderId,
  837. };
  838. const data = JSON.parse(ctx.request.body.data);
  839. if (data && data.type) {
  840. const tenderData = await ctx.service.tender.getDataById(tenderId);
  841. const c_code_rules = tenderData.c_code_rules ? JSON.parse(tenderData.c_code_rules) : {};
  842. c_code_rules[data.type + '_rule_first'] = 0;
  843. updateData.c_code_rules = JSON.stringify(c_code_rules);
  844. } else {
  845. updateData.c_rule_first = 0;
  846. }
  847. const result = await ctx.service.tender.db.update(ctx.service.tender.tableName, updateData);
  848. if (result.affectedRows !== 1) {
  849. throw '更新规则失败';
  850. }
  851. } catch (err) {
  852. ctx.helper.log(err);
  853. responseData.err = 1;
  854. responseData.msg = err.toString();
  855. }
  856. ctx.body = responseData;
  857. }
  858. async shenpiSet(ctx) {
  859. if (ctx.session.sessionUser.is_admin === 0) {
  860. ctx.request.headers.referer ? ctx.redirect(ctx.request.headers.referer) : ctx.redirect('/list');
  861. }
  862. // 获取所有项目参与者
  863. const accountList = await ctx.service.projectAccount.getAllDataByCondition({
  864. where: { project_id: ctx.session.sessionProject.id, enable: 1 },
  865. columns: ['id', 'name', 'company', 'role', 'enable', 'is_admin', 'account_group', 'mobile'],
  866. });
  867. const accountGroupList = accountGroup.map((item, idx) => {
  868. const groupList = accountList.filter(item => item.account_group === idx);
  869. return { groupName: item, groupList };
  870. });
  871. // 获取固定审批流 or 固定终审
  872. for (const sp of shenpiConst.sp_lc) {
  873. sp.status = ctx.tender.info.shenpi ? ctx.tender.info.shenpi[sp.code] : shenpiConst.sp_status.sqspr;
  874. if (sp.status === shenpiConst.sp_status.gdspl) {
  875. sp.auditList = await ctx.service.shenpiAudit.getAuditList(ctx.tender.id, sp.type, sp.status);
  876. } else if (sp.status === shenpiConst.sp_status.gdzs) {
  877. sp.audit = await ctx.service.shenpiAudit.getAudit(ctx.tender.id, sp.type, sp.status);
  878. }
  879. }
  880. const tenders = await ctx.service.tender.getList('', null, 1);
  881. const removeTenders = [];
  882. for (const tender of tenders) {
  883. const shenpiInfo = await ctx.service.tenderInfo.getTenderShenpiInfo(tender.id);
  884. if (!shenpiInfo) {
  885. removeTenders.push(tender.id);
  886. } else {
  887. tender.shenpiInfo = shenpiInfo;
  888. // 获取所有的固定审批流或固定终审
  889. const shenpiauditList = {};
  890. for (const shenpi in tender.shenpiInfo) {
  891. if (tender.shenpiInfo[shenpi] === shenpiConst.sp_status.gdspl) {
  892. const shenpiList = await ctx.service.shenpiAudit.getAllDataByCondition({ where: { tid: tender.id, sp_type: shenpiConst.sp_type[shenpi], sp_status: tender.shenpiInfo[shenpi] } });
  893. const shenpiIdList = ctx.helper._.map(shenpiList, 'audit_id');
  894. shenpiauditList[shenpi] = shenpiIdList.length ? shenpiIdList : null;
  895. } else if (tender.shenpiInfo[shenpi] === shenpiConst.sp_status.gdzs) {
  896. const shenpiInfo = await ctx.service.shenpiAudit.getDataByCondition({ tid: tender.id, sp_type: shenpiConst.sp_type[shenpi], sp_status: tender.shenpiInfo[shenpi] });
  897. shenpiauditList[shenpi] = shenpiInfo && shenpiInfo.audit_id ? [shenpiInfo.audit_id] : null;
  898. }
  899. }
  900. tender.shenpiauditList = shenpiauditList;
  901. }
  902. }
  903. if (removeTenders.length > 0) {
  904. ctx.helper._.remove(tenders, function(n) {
  905. return removeTenders.indexOf(n.id) !== -1;
  906. });
  907. }
  908. const categoryData = await ctx.service.category.getAllCategory(ctx.session.sessionProject.id);
  909. // 是否修订中
  910. const lastRevise = await ctx.service.ledgerRevise.getLastestRevise(ctx.tender.id);
  911. const revising = (lastRevise && lastRevise.status !== auditConst.revise.status.checked) || false;
  912. const cooperationNum = await ctx.service.ledgerCooperation.count({ tid: ctx.tender.id, status: 1 });
  913. const renderData = {
  914. shenpi: shenpiConst,
  915. accountList,
  916. accountGroup: accountGroupList,
  917. tenders,
  918. categoryData,
  919. auditConst,
  920. revising,
  921. measureType,
  922. cooperationNum,
  923. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.tender.shenpi),
  924. };
  925. await this.layout('tender/shenpi.ejs', renderData, 'tender/shenpi_modal.ejs');
  926. }
  927. async saveTenderInfoShenpi(ctx) {
  928. try {
  929. const data = JSON.parse(ctx.request.body.data);
  930. if (!data) {
  931. throw '提交数据错误';
  932. }
  933. // 判断修改权限
  934. if (ctx.session.sessionUser.is_admin === 0) {
  935. throw '你没有权限修改审批流程';
  936. }
  937. // let postData = {};
  938. // if (!ctx.tender.info.shenpi) {
  939. // for (const sp of shenpiConst.sp_lc) {
  940. // if (sp.code === data.code) {
  941. // postData[sp.code] = data.status;
  942. // } else {
  943. // postData[sp.code] = shenpiConst.sp_status.sqspr;
  944. // }
  945. // }
  946. // } else {
  947. const postData = ctx.tender.info.shenpi;
  948. postData[data.code] = data.status;
  949. if (data.code === shenpiConst.sp_lc[shenpiConst.sp_type.stage - 1].code) {
  950. const status = parseInt(data.status) === shenpiConst.sp_status.gdspl ? 1 : 0;
  951. await ctx.service.ledgerCooperation.changeAllStatus(status);
  952. }
  953. // }
  954. await ctx.service.tenderInfo.saveTenderInfo(ctx.tender.id, { shenpi: postData });
  955. let auditList = [];
  956. if (data.status === shenpiConst.sp_status.gdspl) {
  957. auditList = await ctx.service.shenpiAudit.getAuditList(ctx.tender.id, shenpiConst.sp_type[data.code], data.status);
  958. } else if (data.status === shenpiConst.sp_status.gdzs) {
  959. auditList = await ctx.service.shenpiAudit.getAudit(ctx.tender.id, shenpiConst.sp_type[data.code], data.status);
  960. }
  961. ctx.body = { err: 0, msg: '', data: auditList };
  962. } catch (err) {
  963. this.log(err);
  964. ctx.body = this.ajaxErrorBody(err, '保存审批流程设置失败');
  965. }
  966. }
  967. async saveShenpiAudit(ctx) {
  968. try {
  969. const data = JSON.parse(ctx.request.body.data);
  970. if (!data) {
  971. throw '提交数据错误';
  972. }
  973. // 判断修改权限
  974. if (ctx.session.sessionUser.is_admin === 0) {
  975. throw '你没有权限修改审批流程';
  976. }
  977. let info = '';
  978. switch (data.type) {
  979. case 'add':
  980. const result = await ctx.service.shenpiAudit.addAudit(data);
  981. if (result) {
  982. throw '添加审批人失败';
  983. }
  984. break;
  985. case 'del':
  986. await ctx.service.shenpiAudit.removeAudit(data);
  987. break;
  988. case 'copy2ot':
  989. await ctx.service.shenpiAudit.copyAudit2otherTender(data);
  990. break;
  991. case 'copy2os':
  992. await ctx.service.shenpiAudit.copyAudit2otherShenpi(data);
  993. break;
  994. case 'pwd':
  995. info = await ctx.service.ledgerCooperation.save(data);
  996. break;
  997. case 'company':
  998. info = await ctx.service.ledgerCooperation.saveCompany(data);
  999. break;
  1000. default:break;
  1001. }
  1002. ctx.body = { err: 0, msg: '', data: info };
  1003. } catch (err) {
  1004. this.log(err);
  1005. ctx.body = this.ajaxErrorBody(err, '保存审批流程设置失败');
  1006. }
  1007. }
  1008. /**
  1009. * 签名,上传图片 (Ajax)
  1010. * @param ctx
  1011. * @return {Promise<void>}
  1012. */
  1013. async saveCooperateSign(ctx) {
  1014. try {
  1015. const stream = await ctx.getFileStream();
  1016. const create_time = Date.parse(new Date()) / 1000;
  1017. const id = stream.fields.id;
  1018. const fileInfo = path.parse(stream.filename);
  1019. const fileName = path.join('public/upload', ctx.tender.id.toString(), 'sign', 'signImg_' + create_time + fileInfo.ext);
  1020. await ctx.helper.saveStreamFile(stream, path.join(this.app.baseDir, 'app', fileName));
  1021. if (stream) {
  1022. await sendToWormhole(stream);
  1023. }
  1024. await ctx.service.ledgerCooperation.saveSign(id, fileName);
  1025. ctx.body = { err: 0, msg: '', data: fileName };
  1026. } catch (err) {
  1027. this.log(err);
  1028. ctx.body = { err: 1, msg: err.toString(), data: null };
  1029. }
  1030. }
  1031. /**
  1032. * 拷贝设置
  1033. * @param {object} ctx - 上下文
  1034. */
  1035. async copyTender(ctx) {
  1036. try {
  1037. const id = ctx.tender.data.id;
  1038. const { id: copy_id = '', type: typeArr = [] } = JSON.parse(ctx.request.body.data);
  1039. await ctx.service.tenderInfo.copyTenderHandler(id, copy_id, typeArr);
  1040. ctx.body = { err: 0, msg: '' };
  1041. } catch (error) {
  1042. this.log(error);
  1043. ctx.body = this.ajaxErrorBody(error, '保存审批流程设置失败');
  1044. }
  1045. }
  1046. /**
  1047. * 游客账号设置
  1048. * @param {object} ctx - 上下文
  1049. */
  1050. async saveTourist(ctx) {
  1051. try {
  1052. const data = JSON.parse(ctx.request.body.data);
  1053. if (!data) {
  1054. throw '提交数据错误';
  1055. }
  1056. // 判断修改权限
  1057. if (ctx.session.sessionUser.is_admin === 0) {
  1058. throw '你没有权限修改游客账号';
  1059. }
  1060. let info = '';
  1061. switch (data.type) {
  1062. case 'add':
  1063. const result = await ctx.service.tenderTourist.addAudit(data);
  1064. if (!result) {
  1065. throw '添加审批人失败';
  1066. }
  1067. info = result;
  1068. break;
  1069. case 'del':
  1070. await ctx.service.tenderTourist.removeAudit(data);
  1071. break;
  1072. case 'permission':
  1073. await ctx.service.tenderTourist.setPermission(data);
  1074. break;
  1075. default:break;
  1076. }
  1077. ctx.body = { err: 0, msg: '', data: info };
  1078. } catch (err) {
  1079. this.log(err);
  1080. ctx.body = this.ajaxErrorBody(err, '保存游客账号设置失败');
  1081. }
  1082. }
  1083. /**
  1084. * 获取部位明细数据(Ajax)
  1085. *
  1086. * @param ctx
  1087. * @return {Promise<void>}
  1088. */
  1089. async loadLedgerData(ctx) {
  1090. try {
  1091. const ledgerData = await ctx.service.ledger.getData(ctx.tender.id);
  1092. const posData = ctx.tender.data.measure_type === measureType.tz.value
  1093. ? await ctx.service.pos.getPosData({ tid: ctx.tender.id }) : [];
  1094. const convert = new billsPosConvert(ctx);
  1095. convert.loadData(ledgerData, posData, []);
  1096. const result = await convert.convert();
  1097. const ledgerCooperationList = await ctx.service.ledgerCooperation.getAllDataByCondition({ where: { tid: ctx.tender.id, status: 1 } });
  1098. ctx.body = { err: 0, msg: '', data: { ledgerList: result, ledgerCooperationList } };
  1099. } catch (err) {
  1100. this.log(err);
  1101. ctx.body = { err: 1, msg: err.toString(), data: [] };
  1102. }
  1103. }
  1104. async billsTag(ctx) {
  1105. try {
  1106. const isValidTourist = ctx.tender.isTourist && ctx.tender.touristPermission.tag;
  1107. if (ctx.stage) {
  1108. const isAuditor = ctx.stage.users.indexOf(this.ctx.session.sessionUser.accountId) >= 0;
  1109. if (!isAuditor && !isValidTourist) throw '您无权进行该操作';
  1110. } else if (ctx.revise) {
  1111. const isAuditor = ctx.revise.reviseUsers.indexOf(this.ctx.session.sessionUser.accountId) >= 0;
  1112. if (!isAuditor && !isValidTourist) throw '您无权进行该操作';
  1113. } else {
  1114. const isAuditor = ctx.tender.ledgerUsers.indexOf(this.ctx.session.sessionUser.accountId) >= 0;
  1115. if (!isAuditor && !isValidTourist) throw '您无权进行该操作';
  1116. }
  1117. const data = JSON.parse(ctx.request.body.data);
  1118. const result = await ctx.service.ledgerTag.update(data);
  1119. ctx.body = { err: 0, msg: '', data: result };
  1120. } catch (err) {
  1121. this.log(err);
  1122. ctx.body = this.ajaxErrorBody(err, '书签数据错误');
  1123. }
  1124. }
  1125. async listLoad(ctx) {
  1126. try {
  1127. const data = JSON.parse(ctx.request.body.data);
  1128. if (!data.tid || !data.lid || !data.type) throw '数据错误';
  1129. const responseData = {
  1130. err: 0,
  1131. msg: '',
  1132. data: { ledgerAuditConst: auditConst.ledger, stageAuditConst: auditConst.stage },
  1133. };
  1134. responseData.data.category = await this.ctx.service.category.getAllCategory(this.ctx.session.sessionProject.id);
  1135. // 获取用户权限
  1136. const accountInfo = await this.ctx.service.projectAccount.getDataById(this.ctx.session.sessionUser.accountId);
  1137. const userPermission = accountInfo !== undefined && accountInfo.permission !== '' ? JSON.parse(accountInfo.permission) : null;
  1138. const tenderList = await this.ctx.service.tender.getList('', userPermission);
  1139. for (const t of tenderList) {
  1140. if (t.ledger_status === auditConst.ledger.status.checked) {
  1141. t.lastStage = await this.ctx.service.stage.getLastestStage(t.id, false);
  1142. }
  1143. }
  1144. if (data.type === 'ledger') {
  1145. responseData.data.tenders = tenderList.filter(x => {
  1146. return x.ledger_status === auditConst.ledger.status.checked;
  1147. });
  1148. const history = await this.ctx.service.sumLoadHistory.getLedgerHistory(data.tid, data.lid);
  1149. if (history) responseData.data.history = { tenders: history.tenders, load_time: history.load_time, type: 'ledger' };
  1150. } else if (data.type === 'stage') {
  1151. responseData.data.tenders = tenderList.filter(x => {
  1152. return x.ledger_status === auditConst.ledger.status.checked && !!x.lastStage;
  1153. });
  1154. let history = await this.ctx.service.sumLoadHistory.getStageHistory(data.tid, data.lid);
  1155. if (history) {
  1156. responseData.data.history = { tenders: history.tenders, load_time: history.load_time, type: 'stage' };
  1157. } else {
  1158. history = await this.ctx.service.sumLoadHistory.getReviseHistory(data.tid, data.lid);
  1159. if (history) {
  1160. responseData.data.history = { tenders: history.tenders, load_time: history.load_time, type: 'revise' };
  1161. } else {
  1162. history = await this.ctx.service.sumLoadHistory.getLedgerHistory(data.tid, data.lid);
  1163. if (history) responseData.data.history = { tenders: history.tenders, load_time: history.load_time, type: 'ledger' };
  1164. }
  1165. }
  1166. } else if (data.type === 'revise') {
  1167. responseData.data.tenders = tenderList.filter(x => {
  1168. return x.ledger_status === auditConst.ledger.status.checked;
  1169. });
  1170. let history = await this.ctx.service.sumLoadHistory.getReviseHistory(data.tid, data.lid);
  1171. if (history) {
  1172. responseData.data.history = { tenders: history.tenders, load_time: history.load_time, type: 'revise' };
  1173. } else {
  1174. history = await this.ctx.service.sumLoadHistory.getLedgerHistory(data.tid, data.lid);
  1175. if (history) responseData.data.history = { tenders: history.tenders, load_time: history.load_time, type: 'ledger' };
  1176. }
  1177. }
  1178. ctx.body = responseData;
  1179. } catch(err) {
  1180. this.log(err);
  1181. this.ajaxErrorBody(err, '获取标段列表错误');
  1182. }
  1183. }
  1184. async listLoad2(ctx) {
  1185. try {
  1186. const data = JSON.parse(ctx.request.body.data);
  1187. if (!data.type) throw '数据错误';
  1188. const responseData = {
  1189. err: 0,
  1190. msg: '',
  1191. data: { ledgerAuditConst: auditConst.ledger, stageAuditConst: auditConst.stage },
  1192. };
  1193. responseData.data.category = await this.ctx.service.category.getAllCategory(this.ctx.session.sessionProject.id);
  1194. // 获取用户权限
  1195. const accountInfo = await this.ctx.service.projectAccount.getDataById(this.ctx.session.sessionUser.accountId);
  1196. const userPermission = accountInfo !== undefined && accountInfo.permission !== '' ? JSON.parse(accountInfo.permission) : null;
  1197. const tenderList = await this.ctx.service.tender.getList('', userPermission);
  1198. for (const t of tenderList) {
  1199. if (t.ledger_status === auditConst.ledger.status.checked) {
  1200. t.lastStage = await this.ctx.service.stage.getLastestStage(t.id, false);
  1201. t.lastCheckedStage = await this.ctx.service.stage.getLastestCompleteStage(t.id);
  1202. }
  1203. }
  1204. if (data.type === 'all') {
  1205. responseData.data.tenders = tenderList;
  1206. } else if (data.type === 'ledger-checked') {
  1207. responseData.data.tenders = tenderList.filter(x => {
  1208. return x.ledger_status === auditConst.ledger.status.checked;
  1209. });
  1210. } else if (data.type === 'stage') {
  1211. responseData.data.tenders = tenderList.filter(x => {
  1212. return x.ledger_status === auditConst.ledger.status.checked && !!x.lastStage;
  1213. });
  1214. } else if (data.type === 'stage-checked') {
  1215. responseData.data.tenders = tenderList.filter(x => {
  1216. return x.ledger_status === auditConst.ledger.status.checked && !!x.lastCheckedStage;
  1217. });
  1218. }
  1219. ctx.body = responseData;
  1220. } catch(err) {
  1221. this.log(err);
  1222. this.ajaxErrorBody(err, '获取标段列表错误');
  1223. }
  1224. }
  1225. async sumLoad(ctx) {
  1226. try {
  1227. const data = JSON.parse(ctx.request.body.data);
  1228. if (!data.lid || !data.type || !data.tenders) throw '数据错误';
  1229. switch (data.type) {
  1230. case 'ledger':
  1231. const refreshData = await this.ctx.service.ledger.sumLoad(data.lid, data.tenders);
  1232. ctx.body = {err: 0, msg: '', data: refreshData};
  1233. break;
  1234. case 'revise':
  1235. const reviseData = await this.ctx.service.reviseBills.sumLoad(data.lid, this.ctx.revise.id, data.tenders);
  1236. ctx.body = {err: 0, msg: '', data: reviseData};
  1237. break;
  1238. case 'stage':
  1239. const stageData = await this.ctx.service.stageBills.sumLoad(data.lid, data.tenders, data.cover);
  1240. stageData.import_change = { target: { import_lid: data.lid } };
  1241. stageData.import_change.data = await this.ctx.service.stageImportChange.getLeafXmjStageImportData(ctx.stage, data.lid);
  1242. await ctx.service.stage.updateCheckCalcFlag(ctx.stage, true);
  1243. await ctx.service.stage.updateCacheTime(ctx.stage.id);
  1244. ctx.body = {err: 0, msg: '', data: stageData};
  1245. break;
  1246. default:
  1247. throw '数据错误';
  1248. }
  1249. } catch (err) {
  1250. this.log(err);
  1251. ctx.ajaxErrorBody(err, '导入数据失败');
  1252. }
  1253. }
  1254. /**
  1255. * 地图数据设置(Ajax)
  1256. *
  1257. * @param ctx
  1258. * @return {Promise<void>}
  1259. */
  1260. async saveMap(ctx) {
  1261. try {
  1262. const data = JSON.parse(ctx.request.body.data);
  1263. if (!data) {
  1264. throw '提交数据错误';
  1265. }
  1266. // 判断修改权限
  1267. if (ctx.session.sessionUser.is_admin === 0) {
  1268. throw '你没有权限修改概况设置';
  1269. }
  1270. let info = '';
  1271. let result = '';
  1272. switch (data.type) {
  1273. case 'select-map':
  1274. result = await ctx.service.tender.saveTenderData(ctx.tender.id, { had_map: data.value });
  1275. if (!result) {
  1276. throw '修改概况设置失败';
  1277. }
  1278. info = result;
  1279. break;
  1280. case 'add-map':
  1281. result = await ctx.service.tenderMap.addMap(ctx.tender.id, data.name);
  1282. info = await ctx.service.tenderMap.getDataById(result.insertId);
  1283. break;
  1284. case 'del-map':
  1285. await ctx.service.tenderMap.deleteById(data.id);
  1286. break;
  1287. case 'save-map':
  1288. await ctx.service.tenderMap.saveMap(data.mapData);
  1289. break;
  1290. default:break;
  1291. }
  1292. ctx.body = { err: 0, msg: '', data: info };
  1293. } catch (err) {
  1294. this.log(err);
  1295. ctx.body = this.ajaxErrorBody(err, '保存概况设置失败');
  1296. }
  1297. }
  1298. /**
  1299. * 上传静态图
  1300. *
  1301. * @param {object} ctx - egg全局变量
  1302. * @return {void}
  1303. */
  1304. async uploadMap(ctx) {
  1305. const responseData = {
  1306. err: 0, msg: '', data: null,
  1307. };
  1308. try {
  1309. const stream = await ctx.getFileStream();
  1310. const create_time = Date.parse(new Date()) / 1000;
  1311. const fileInfo = path.parse(stream.filename);
  1312. const filepath = `app/public/upload/${ctx.tender.id}/map/${create_time}${fileInfo.ext}`;
  1313. await ctx.app.fujianOss.put(ctx.app.config.fujianOssFolder + filepath, stream);
  1314. await sendToWormhole(stream);
  1315. const tenderInfo = await ctx.service.tender.getDataById(ctx.tender.id);
  1316. const oldMapPic = tenderInfo && tenderInfo.map_pic ? tenderInfo.map_pic : null;
  1317. const result = await ctx.service.tender.update({ map_pic: filepath }, { id: ctx.tender.id });
  1318. if (result) {
  1319. // 移除旧的map_pic oss
  1320. if (oldMapPic) {
  1321. await ctx.app.fujianOss.delete(ctx.app.config.fujianOssFolder + oldMapPic);
  1322. }
  1323. responseData.data = { map_pic: filepath };
  1324. } else {
  1325. throw '添加数据库失败';
  1326. }
  1327. } catch (err) {
  1328. this.log(err);
  1329. responseData.err = 1;
  1330. responseData.msg = err;
  1331. }
  1332. ctx.body = responseData;
  1333. }
  1334. async loadData(ctx) {
  1335. try {
  1336. const data = JSON.parse(ctx.request.body.data);
  1337. const filter = data.filter.split(';');
  1338. const responseData = { err: 0, msg: '', data: {} };
  1339. const where = { tid: this.ctx.tender.id };
  1340. for (const f of filter) {
  1341. switch (f) {
  1342. case 'change':
  1343. where.valid = 1;
  1344. if (ctx.session.sessionProject.page_show.isOnlyChecked) where.status = auditConst.flow.status.checked;
  1345. responseData.data[f] = await ctx.service.change.getAllDataByCondition({
  1346. columns: [ 'cid', 'code', 'name', 'selected' ],
  1347. where,
  1348. orders: [['in_time', 'desc']],
  1349. });
  1350. break;
  1351. case 'change_project':
  1352. if (ctx.session.sessionProject.page_show.isOnlyChecked) where.status = auditConst.changeProject.status.checked;
  1353. responseData.data[f] = await ctx.service.changeProject.getAllDataByCondition({
  1354. columns: [ 'id', 'code', 'name', 'selected' ],
  1355. where,
  1356. orders: [['in_time', 'desc']],
  1357. });
  1358. break;
  1359. case 'change_apply':
  1360. if (ctx.session.sessionProject.page_show.isOnlyChecked) where.status = auditConst.changeApply.status.checked;
  1361. responseData.data[f] = await ctx.service.changeApply.getAllDataByCondition({
  1362. columns: [ 'id', 'code', 'name', 'selected' ],
  1363. where,
  1364. orders: [['in_time', 'desc']],
  1365. });
  1366. break;
  1367. case 'change_plan':
  1368. if (ctx.session.sessionProject.page_show.isOnlyChecked) where.status = auditConst.changeApply.status.checked;
  1369. responseData.data[f] = await ctx.service.changePlan.getAllDataByCondition({
  1370. columns: [ 'id', 'code', 'name', 'selected' ],
  1371. where,
  1372. orders: [['in_time', 'desc']],
  1373. });
  1374. break;
  1375. case 'advance':
  1376. const advance = await ctx.service.advance.getAllDataByCondition({
  1377. columns: [ 'id', 'order', 'status', 'selected', 'type' ],
  1378. where,
  1379. orders: [['type', 'asc'], ['create_time', 'desc']],
  1380. });
  1381. advance.forEach(x => {
  1382. x.orderStr = `第${x.order}期`;
  1383. x.statusStr = auditConst.advance.statusString[x.status];
  1384. x.typeStr = advanceConst.typeColMap[x.type].text;
  1385. });
  1386. responseData.data[f] = advance;
  1387. break;
  1388. case 'pm_deal':
  1389. if (accountPermission.PermissionCheck.viewPmData(this.ctx.session.sessionUser.permission)) {
  1390. const selects = await this.ctx.service.project.getPmDealCache(this.ctx.session.sessionProject.id);
  1391. const pm = require('../lib/pm');
  1392. const tenders = await pm.dealCatagory(ctx, this.ctx.session.sessionProject.code);
  1393. tenders.forEach(x => {
  1394. x.selected = selects.indexOf(x.bidsectionId + '') >= 0;
  1395. });
  1396. responseData.data[f] = tenders;
  1397. } else {
  1398. throw '您无权查看该数据';
  1399. }
  1400. break;
  1401. default:
  1402. throw '未知数据类型';
  1403. }
  1404. }
  1405. ctx.body = responseData;
  1406. } catch (err) {
  1407. ctx.log(err);
  1408. ctx.ajaxErrorBody(err, '加载数据错误');
  1409. }
  1410. }
  1411. async saveRelaData(ctx) {
  1412. try {
  1413. const data = JSON.parse(ctx.request.body.data);
  1414. const responseData = { err: 0, msg: '', data: {} };
  1415. if (data.change) await this.ctx.service.change.updateChangeSelect(data.change);
  1416. if (data.change_apply) await this.ctx.service.changeApply.defaultUpdateRows(data.change_apply);
  1417. if (data.change_project) await this.ctx.service.changeProject.defaultUpdateRows(data.change_project);
  1418. if (data.change_plan) await this.ctx.service.changePlan.defaultUpdateRows(data.change_plan);
  1419. if (data.advance) await this.ctx.service.advance.defaultUpdateRows(data.advance);
  1420. if (data.pm_deal) await this.ctx.service.project.setPmDealCache(this.ctx.session.sessionProject.id, data.pm_deal);
  1421. ctx.body = responseData;
  1422. } catch (err) {
  1423. ctx.log(err);
  1424. ctx.ajaxErrorBody(err, '保存数据错误');
  1425. }
  1426. }
  1427. }
  1428. return TenderController;
  1429. };