tender_controller.js 74 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508
  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.add(lastStage.contract_tp, lastStage.qc_tp);
  386. tender.end_contract_tp = ctx.helper.add(lastStage.contract_tp, lastStage.pre_contract_tp);
  387. tender.end_qc_tp = ctx.helper.add(lastStage.qc_tp, lastStage.pre_qc_tp);
  388. tender.end_gather_tp = ctx.helper.add(tender.end_contract_tp, tender.end_qc_tp);
  389. tender.pre_gather_tp = ctx.helper.add(lastStage.pre_contract_tp, lastStage.pre_qc_tp);
  390. tender.yf_tp = lastStage.yf_tp;
  391. const change_tp = await ctx.service.change.getChangeTp(tender.id);
  392. tender.change_tp = change_tp;
  393. tender.qc_ratio = ctx.helper.mul(ctx.helper.div(tender.end_qc_tp, ctx.tender.info.deal_param.contractPrice, 2), 100);
  394. tender.sum = ctx.helper.add(tender.total_price, tender.change_tp);
  395. tender.pre_ratio = ctx.helper.mul(ctx.helper.div(tender.pre_gather_tp, tender.sum, 2), 100);
  396. tender.cur_ratio = ctx.helper.mul(ctx.helper.div(tender.gather_tp, tender.sum, 2), 100);
  397. tender.other_tp = ctx.helper.sub(ctx.helper.sub(tender.sum, tender.pre_gather_tp), tender.gather_tp);
  398. tender.other_ratio = Math.max(0, 100 - tender.pre_ratio - tender.cur_ratio);
  399. tender.end_yf_tp = ctx.helper.add(lastStage.yf_tp, lastStage.pre_yf_tp);
  400. tender.end_sf_tp = ctx.helper.add(lastStage.sf_tp, lastStage.pre_sf_tp);
  401. 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);
  402. if (lastStage.status === auditConst.stage.status.uncheck) {
  403. const status_name = await this.ctx.service.projectAccount.getAccountInfoById(lastStage.user_id);
  404. lastStage.status_users = status_name ? status_name.name : '';
  405. lastStage.auditors = [];
  406. } else {
  407. lastStage.status = lastStage.status === auditConst.stage.status.checkNoPre ? auditConst.stage.status.checking : lastStage.status;
  408. let cur;
  409. if (lastStage.status === auditConst.stage.status.checked) {
  410. cur = await this.ctx.service.stageAudit.getLastestAuditor(lastStage.id, lastStage.times, auditConst.stage.status.checked);
  411. } else if (lastStage.status === auditConst.stage.status.checking) {
  412. cur = await this.ctx.service.stageAudit.getCurAuditor(lastStage.id, lastStage.times);
  413. } else {
  414. cur = await this.ctx.service.stageAudit.getAuditorByStatus(lastStage.id, lastStage.status, lastStage.times);
  415. }
  416. // const status_name = await this.ctx.service.stageAudit.getAuditorByStatus(lastStage.id, lastStage.status, lastStage.times);
  417. lastStage.status_users = cur ? cur.name : '';
  418. const times = lastStage.status === auditConst.stage.status.checkNo ? lastStage.times - 1 : lastStage.times;
  419. lastStage.auditors = await ctx.service.stageAudit.getFinalAuditGroup(lastStage.id, times);
  420. }
  421. } else {
  422. if (bCalcTp) {
  423. const sum = await this.ctx.service.ledger.addUp({ tender_id: tender.id/* , is_leaf: true*/ });
  424. tender.total_price = sum.total_price;
  425. tender.deal_tp = sum.deal_tp;
  426. }
  427. if (tender.ledger_status !== auditConst.ledger.status.uncheck) {
  428. const status_name = await this.ctx.service.ledgerAudit.getStatusName(tender.id, tender.ledger_times);
  429. tender.status_users = status_name ? status_name.name : '';
  430. const times = tender.ledger_status === auditConst.ledger.status.checkNo ? tender.ledger_times - 1 : tender.ledger_times;
  431. tender.auditors = await ctx.service.ledgerAudit.getFinalAuditGroup(tender.id, times);
  432. } else {
  433. const status_name = await this.ctx.service.projectAccount.getAccountInfoById(tender.user_id);
  434. tender.status_users = status_name ? status_name.name : '';
  435. }
  436. }
  437. const tiModel = new tenderInfoModel(ctx);
  438. const gclChapter = await tiModel.gatherChapter();
  439. const monthProgress = [];
  440. for (const s of stages) {
  441. if (s.s_time) {
  442. let progress = monthProgress.find(function(x) {
  443. return x.month === s.s_time;
  444. });
  445. if (!progress) {
  446. progress = { month: s.s_time };
  447. monthProgress.push(progress);
  448. }
  449. progress.tp = ctx.helper.add(ctx.helper.add(progress.tp, s.contract_tp), s.qc_tp);
  450. }
  451. }
  452. monthProgress.sort(function(x, y) {
  453. return Date.parse(x.month) - Date.parse(y.month);
  454. });
  455. let sum = 0;
  456. for (const p of monthProgress) {
  457. p.ratio = ctx.helper.mul(ctx.helper.div(p.tp, tender.sum, 4), 100);
  458. sum = ctx.helper.add(sum, p.tp);
  459. p.end_tp = sum;
  460. p.end_ratio = ctx.helper.mul(ctx.helper.div(p.end_tp, tender.sum, 4), 100);
  461. }
  462. const revise = await ctx.service.ledgerRevise.getLastestRevise(tender.id);
  463. const tenders = await ctx.service.tender.getList('', null, 1);
  464. const categoryData = await ctx.service.category.getAllCategory(ctx.session.sessionProject.id);
  465. // 变更图表数据
  466. const change_done_total = await ctx.service.change.getCountByStatus2(tender.id, auditConst.filter.status.checked);
  467. const change_doing_total = await ctx.service.change.getCountByStatus2(tender.id, auditConst.filter.status.checking);
  468. const change_uncheck_total = await ctx.service.change.getCountByStatus2(tender.id, auditConst.filter.status.uncheck);
  469. const change_status_total = [
  470. { num: change_uncheck_total, name: '待上报' },
  471. { num: change_doing_total, name: '审批中' },
  472. { num: change_done_total, name: '已完成' },
  473. ];
  474. const change_common_total = await ctx.service.change.getCountByQuality(tender.id, changeConst.quality.common.value);
  475. const change_more_total = await ctx.service.change.getCountByQuality(tender.id, changeConst.quality.more.value);
  476. const change_great_total = await ctx.service.change.getCountByQuality(tender.id, changeConst.quality.great.value);
  477. const change_quality_total = [
  478. { num: change_common_total, name: '一般变更' },
  479. { num: change_more_total, name: '较大变更' },
  480. { num: change_great_total, name: '重大变更' },
  481. ];
  482. // 调差最新期数据
  483. const materials = await ctx.service.material.getValidMaterials(ctx.tender.id);
  484. let materialData = null;
  485. if (materials && materials.length > 0) {
  486. materialData = materials[0];
  487. materialData.curAuditor = await ctx.service.materialAudit.getAuditorByStatus(materialData.id, materialData.status, materialData.times);
  488. const times = materialData.status === auditConst.material.status.checkNo ? materialData.times - 1 : materialData.times;
  489. materialData.auditors = materialData.status === auditConst.material.status.uncheck ? [] : await ctx.service.materialAudit.getFinalAuditGroup(materialData.id, times);
  490. }
  491. // 修订完成数目
  492. // const reviseNum = await ctx.service.ledgerRevise.count({ tid: tender.id, status: auditConst.revise.status.checked });
  493. // 计量完成概况
  494. // tender.total_price
  495. const stage_total = [
  496. { num: tender.end_contract_tp ? tender.end_contract_tp : 0, name: '合同完成' },
  497. { num: tender.end_qc_tp ? tender.end_qc_tp : 0, name: '变更完成' },
  498. { num: tender.undone_tp ? tender.undone_tp : 0, name: '未完成' },
  499. ];
  500. // 地图
  501. const tenderALLInfo = await ctx.service.tender.getDataById(tender.id);
  502. const hadMap = tenderALLInfo.had_map === 0 ? 1 : tenderALLInfo.had_map;// 0为初始值,因为默认可能会变化,所以暂时把0都默认为1。共三种模式坐标模式1,图片模式2,无图模式3。
  503. const tenderMapList = await ctx.service.tenderMap.getAllDataByCondition({ where: { tid: tender.id } });
  504. // 默认坐标,否则则取办事处坐标
  505. const projectData = await ctx.service.project.getDataById(ctx.session.sessionProject.id);
  506. let map_json = {
  507. province: mapConst.map[0].province,
  508. lng: mapConst.map[0].lng,
  509. lat: mapConst.map[0].lat,
  510. level: 15,
  511. };
  512. if (projectData.map_json) {
  513. map_json = JSON.parse(projectData.map_json);
  514. } else {
  515. const mapInfo = ctx.helper._.find(mapConst.map, { office: projectData.manager_office });
  516. if (mapInfo) {
  517. map_json.province = mapInfo.province;
  518. map_json.lng = mapInfo.lng;
  519. map_json.lat = mapInfo.lat;
  520. }
  521. }
  522. const renderData = {
  523. tenders,
  524. categoryData,
  525. tender,
  526. revise,
  527. tenderInfo: ctx.tender.info,
  528. tenderMenu: this.menu.tenderMenu,
  529. preUrl: '/tender/' + ctx.tender.id,
  530. cooperation: ctx.session.sessionUser.cooperation,
  531. lastStage,
  532. stages: stages.reverse(),
  533. monthProgress,
  534. audit: auditConst,
  535. change_status_total,
  536. change_quality_total,
  537. materialData,
  538. // reviseNum,
  539. stage_total,
  540. hadMap,
  541. map_pic: tenderALLInfo.map_pic,
  542. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.tender.tenderInfo),
  543. gclChapter,
  544. tenderMapList,
  545. map_json,
  546. fujianOssPath: ctx.app.config.fujianOssPath,
  547. };
  548. if (ctx.session.sessionUser.is_admin) {
  549. renderData.tourists = await ctx.service.tenderTourist.getTourists(tender.id);
  550. for (const t of renderData.tourists) {
  551. t.permission = await ctx.service.tenderTourist.getTouristPermission(t);
  552. }
  553. // 获取所有项目参与者
  554. const accountList = await ctx.service.projectAccount.getAllDataByCondition({
  555. where: { project_id: ctx.session.sessionProject.id, enable: 1 },
  556. columns: ['id', 'name', 'company', 'role', 'enable', 'is_admin', 'account_group', 'mobile'],
  557. });
  558. const accountGroupList = accountGroup.map((item, idx) => {
  559. const groupList = accountList.filter(item => item.account_group === idx);
  560. return { groupName: item, groupList };
  561. });
  562. renderData.accountList = accountList;
  563. renderData.accountGroup = accountGroupList;
  564. }
  565. if (ctx.session.sessionProject.page_show.xxjd && ctx.session.sessionUser.is_admin) {
  566. // 投资进度内容
  567. renderData.scheduleAuditList = await ctx.service.scheduleAudit.getAllDataByCondition({ where: { tid: tender.id } });
  568. renderData.scPermission = scheduleConst.permission;
  569. }
  570. await this.layout('tender/detail.ejs', renderData, 'tender/detail_modal.ejs');
  571. } catch (error) {
  572. this.log(error);
  573. this.ctx.redirect('/list');
  574. }
  575. }
  576. /**
  577. * 保存标段属性等(Ajax)
  578. *
  579. * @param ctx
  580. * @return {Promise<void>}
  581. */
  582. async saveTenderInfo(ctx) {
  583. try {
  584. const data = JSON.parse(ctx.request.body.data);
  585. if (!data) {
  586. throw '提交数据错误';
  587. }
  588. // 针对查阅所有标段者但非原报和审批人提示
  589. const times = ctx.tender.data.ledger_status === auditConst.ledger.status.checkNo ? ctx.tender.data.ledger_times - 1 : ctx.tender.data.ledger_times;
  590. const auditors = await this.service.ledgerAudit.getAuditors(ctx.tender.id, times);
  591. const auditorsId = ctx.helper._.map(auditors, 'audit_id');
  592. const stageAuditors = await this.service.stageAudit.getAllAuditors(ctx.tender.id);
  593. const stageAUditorsId = ctx.helper._.map(stageAuditors, 'aid');
  594. const accountId = ctx.session.sessionUser.accountId;
  595. if (auditorsId.indexOf(accountId) === -1 && ctx.tender.data.user_id !== accountId &&
  596. stageAUditorsId.indexOf(accountId) === -1) {
  597. throw '您无权修改标段设置内容';
  598. }
  599. if (ctx.tender.data.ledger_status === auditConst.ledger.status.checked) {
  600. if (data.deal_param) {
  601. const lastStage = await this.ctx.service.stage.getLastestStage(ctx.tender.id, true);
  602. if (lastStage) {
  603. if (lastStage.order > 1 || (lastStage.status === auditConst.stage.status.checked || lastStage.status === auditConst.stage.status.checking)) throw '第一期上报后不可修改合同参数';
  604. if (lastStage.user_id !== ctx.session.sessionUser.accountId) throw '仅原报可修改合同参数';
  605. }
  606. }
  607. }
  608. if (data.decimal) {
  609. if (ctx.tender.data.user_id !== ctx.session.sessionUser.accountId) throw '仅原报可修改小数位数';
  610. await ctx.service.tenderInfo.saveDecimal(ctx.tender.id, data.decimal, ctx.tender.info.decimal);
  611. } else if (data.precision) {
  612. if (ctx.tender.data.user_id !== ctx.session.sessionUser.accountId) throw '仅原报可修改清单精度';
  613. await ctx.service.tenderInfo.savePrecision(ctx.tender.id,
  614. data.precision, ctx.tender.info.precision, ctx.tender.info.decimal);
  615. } else {
  616. await ctx.service.tenderInfo.saveTenderInfo(ctx.tender.id, data);
  617. }
  618. ctx.body = { err: 0, msg: '', data: JSON.parse(ctx.request.body.data) };
  619. } catch (err) {
  620. this.log(err);
  621. ctx.body = this.ajaxErrorBody(err, '保存标段设置失败');
  622. }
  623. }
  624. async saveTenderInfo2(ctx) {
  625. try {
  626. const data = JSON.parse(ctx.request.body.data);
  627. if (!data || (!data.ledger_check && !data.fun_rela)) throw '提交数据错误';
  628. if (!ctx.session.sessionUser.is_admin) throw '您无权修改该内容';
  629. const updateData = {};
  630. if (data.ledger_check) updateData.ledger_check = data.ledger_check;
  631. if (data.fun_rela) updateData.fun_rela = data.fun_rela;
  632. await ctx.service.tenderInfo.saveTenderInfo(ctx.tender.id, data);
  633. ctx.body = { err: 0, msg: '', data: JSON.parse(ctx.request.body.data) };
  634. } catch (err) {
  635. this.log(err);
  636. ctx.body = this.ajaxErrorBody(err, '保存失败');
  637. }
  638. }
  639. /**
  640. * 设置标段计量类型并调整到标段概况(Get)
  641. *
  642. * @param ctx
  643. * @return {Promise<void>}
  644. */
  645. async tenderType(ctx) {
  646. try {
  647. const tenderId = ctx.params.id,
  648. type = ctx.query.type;
  649. if (!tenderId) {
  650. throw '当前未打开标段';
  651. }
  652. await ctx.service.tender.checkTender(tenderId);
  653. if (!ctx.tender) {
  654. throw '标段数据错误';
  655. }
  656. if (!ctx.tender.measure_type) {
  657. await ctx.service.tender.setTenderType(ctx.tender, parseInt(type));
  658. }
  659. ctx.redirect('/tender/' + tenderId);
  660. } catch (error) {
  661. ctx.helper.log(error);
  662. this.postError(error, '设置标段类型错误');
  663. ctx.redirect('/list');
  664. }
  665. }
  666. /**
  667. * 标段协作办公
  668. *
  669. * @param {Object} ctx - egg全局变量
  670. * @return {void}
  671. */
  672. async tenderCooperation(ctx) {
  673. const tenderId = ctx.params.id;
  674. try {
  675. if (!ctx.session.sessionUser.cooperation) {
  676. throw '权限不足';
  677. }
  678. const tender = await ctx.service.tender.getDataById(tenderId);
  679. const user = await ctx.service.projectAccount.getDataById(ctx.session.sessionUser.accountId);
  680. // 获取已参与协作用户列表
  681. const cooperationArray = [];
  682. if (tender.cooperation !== null && tender.cooperation !== '') {
  683. const cooperationList = JSON.parse(tender.cooperation);
  684. for (const cl in cooperationList) {
  685. const clArray = [];
  686. for (const audit of cooperationList[cl]) {
  687. const userInfo = await ctx.service.projectAccount.getDataById(audit);
  688. clArray.push(userInfo);
  689. }
  690. cooperationArray[cl] = clArray;
  691. }
  692. }
  693. const renderData = {
  694. user,
  695. tender,
  696. tenderMenu: this.menu.tenderMenu,
  697. preUrl: '/tender/' + tenderId,
  698. tenderPermissionList: accountPermission.tenderPermissionList,
  699. cooperationArray,
  700. };
  701. await this.layout('tender/cooperation.ejs', renderData, 'tender/cooperationModal.ejs');
  702. } catch (error) {
  703. ctx.helper.log(error);
  704. this.ctx.redirect('/tender/' + tenderId);
  705. }
  706. }
  707. /**
  708. * 添加标段操作
  709. *
  710. * @param {Object} ctx - egg全局变量
  711. * @return {void}
  712. */
  713. async add(ctx) {
  714. try {
  715. const rule = ctx.service.tender.rule('add');
  716. ctx.helper.validate(rule);
  717. const result = ctx.service.tender.add(ctx.request.body);
  718. if (!result) {
  719. throw '新增标段失败';
  720. }
  721. } catch (error) {
  722. ctx.helper.log(error);
  723. this.setMessage(error.toString(), this.messageType.ERROR);
  724. }
  725. ctx.redirect(ctx.request.header.referer);
  726. }
  727. /**
  728. * 保存标段操作
  729. *
  730. * @param {Object} ctx - egg全局变量
  731. * @return {void}
  732. */
  733. async save(ctx) {
  734. let id = ctx.request.body.tenderId;
  735. id = parseInt(id);
  736. try {
  737. if (isNaN(id) || id < 0) {
  738. throw '参数错误';
  739. }
  740. // 获取数据规则
  741. const rule = ctx.service.tender.rule('save');
  742. ctx.validate(rule);
  743. const result = await ctx.service.tender.save(ctx.request.body, id);
  744. if (!result) {
  745. throw '保存标段数据失败';
  746. }
  747. this.setMessage('保存标段数据成功', this.messageType.SUCCESS);
  748. } catch (error) {
  749. this.postError(error, '保存标段数据失败');
  750. }
  751. ctx.redirect(ctx.request.header.referer);
  752. }
  753. /**
  754. * 删除标段
  755. *
  756. * @param {Object} ctx -egg全局变量
  757. * @return {void}
  758. */
  759. async delete(ctx) {
  760. let id = ctx.request.body.tenderId;
  761. id = parseInt(id);
  762. try {
  763. if (isNaN(id) || id <= 0) {
  764. throw '参数错误';
  765. }
  766. const result = ctx.service.tender.deleteTenderById(id);
  767. if (!result) {
  768. throw '删除标段失败';
  769. }
  770. this.setMessage('删除标段成功', this.messageType.SUCCESS);
  771. } catch (error) {
  772. this.postError(error, '删除标段失败');
  773. }
  774. ctx.redirect(ctx.request.header.referer);
  775. }
  776. async rule(ctx) {
  777. const responseData = {
  778. err: 0,
  779. msg: '',
  780. data: {},
  781. };
  782. try {
  783. const tenderId = ctx.session.sessionUser.tenderId;
  784. if (!tenderId) {
  785. throw '当前未打开标段';
  786. }
  787. const data = JSON.parse(ctx.request.body.data);
  788. if (!data.rule || !JSON.parse(data.data).length || !codeRuleConst.ruleField[data.rule]) {
  789. throw '请选择组件再添加';
  790. }
  791. if (!data.connector) {
  792. throw '请选择连接符';
  793. }
  794. const updateData = {
  795. id: tenderId,
  796. };
  797. if (data.type) {
  798. const tenderData = await ctx.service.tender.getDataById(tenderId);
  799. const c_code_rules = tenderData.c_code_rules ? JSON.parse(tenderData.c_code_rules) : {};
  800. c_code_rules[data.type + '_rule'] = JSON.parse(data.data);
  801. c_code_rules[data.type + '_rule_first'] = 0;
  802. c_code_rules[data.type + '_connector'] = data.connector;
  803. updateData.c_code_rules = JSON.stringify(c_code_rules);
  804. } else {
  805. updateData[codeRuleConst.ruleField[data.rule]] = data.data;
  806. updateData.c_connector = data.connector;
  807. updateData.c_rule_first = 0;
  808. }
  809. const result = await ctx.service.tender.db.update(ctx.service.tender.tableName, updateData);
  810. if (result.affectedRows !== 1) {
  811. throw '更新规则失败';
  812. }
  813. } catch (err) {
  814. ctx.helper.log(err);
  815. responseData.err = 1;
  816. responseData.msg = err.toString();
  817. }
  818. ctx.body = responseData;
  819. }
  820. async ruleFirst(ctx) {
  821. const responseData = {
  822. err: 0,
  823. msg: '',
  824. data: {},
  825. };
  826. try {
  827. const tenderId = ctx.session.sessionUser.tenderId;
  828. if (!tenderId) {
  829. throw '当前未打开标段';
  830. }
  831. const updateData = {
  832. id: tenderId,
  833. };
  834. const data = JSON.parse(ctx.request.body.data);
  835. if (data && data.type) {
  836. const tenderData = await ctx.service.tender.getDataById(tenderId);
  837. const c_code_rules = tenderData.c_code_rules ? JSON.parse(tenderData.c_code_rules) : {};
  838. c_code_rules[data.type + '_rule_first'] = 0;
  839. updateData.c_code_rules = JSON.stringify(c_code_rules);
  840. } else {
  841. updateData.c_rule_first = 0;
  842. }
  843. const result = await ctx.service.tender.db.update(ctx.service.tender.tableName, updateData);
  844. if (result.affectedRows !== 1) {
  845. throw '更新规则失败';
  846. }
  847. } catch (err) {
  848. ctx.helper.log(err);
  849. responseData.err = 1;
  850. responseData.msg = err.toString();
  851. }
  852. ctx.body = responseData;
  853. }
  854. async shenpiSet(ctx) {
  855. if (ctx.session.sessionUser.is_admin === 0) {
  856. ctx.request.headers.referer ? ctx.redirect(ctx.request.headers.referer) : ctx.redirect('/list');
  857. }
  858. // 获取所有项目参与者
  859. const accountList = await ctx.service.projectAccount.getAllDataByCondition({
  860. where: { project_id: ctx.session.sessionProject.id, enable: 1 },
  861. columns: ['id', 'name', 'company', 'role', 'enable', 'is_admin', 'account_group', 'mobile'],
  862. });
  863. const accountGroupList = accountGroup.map((item, idx) => {
  864. const groupList = accountList.filter(item => item.account_group === idx);
  865. return { groupName: item, groupList };
  866. });
  867. // 获取固定审批流 or 固定终审
  868. for (const sp of shenpiConst.sp_lc) {
  869. sp.status = ctx.tender.info.shenpi ? ctx.tender.info.shenpi[sp.code] : shenpiConst.sp_status.sqspr;
  870. if (sp.status === shenpiConst.sp_status.gdspl) {
  871. sp.auditList = await ctx.service.shenpiAudit.getAuditList(ctx.tender.id, sp.type, sp.status);
  872. } else if (sp.status === shenpiConst.sp_status.gdzs) {
  873. sp.audit = await ctx.service.shenpiAudit.getAudit(ctx.tender.id, sp.type, sp.status);
  874. }
  875. }
  876. const tenders = await ctx.service.tender.getList('', null, 1);
  877. const removeTenders = [];
  878. for (const tender of tenders) {
  879. const shenpiInfo = await ctx.service.tenderInfo.getTenderShenpiInfo(tender.id);
  880. if (!shenpiInfo) {
  881. removeTenders.push(tender.id);
  882. } else {
  883. tender.shenpiInfo = shenpiInfo;
  884. // 获取所有的固定审批流或固定终审
  885. const shenpiauditList = {};
  886. for (const shenpi in tender.shenpiInfo) {
  887. if (tender.shenpiInfo[shenpi] === shenpiConst.sp_status.gdspl) {
  888. const shenpiList = await ctx.service.shenpiAudit.getAllDataByCondition({ where: { tid: tender.id, sp_type: shenpiConst.sp_type[shenpi], sp_status: tender.shenpiInfo[shenpi] } });
  889. const shenpiIdList = ctx.helper._.map(shenpiList, 'audit_id');
  890. shenpiauditList[shenpi] = shenpiIdList.length ? shenpiIdList : null;
  891. } else if (tender.shenpiInfo[shenpi] === shenpiConst.sp_status.gdzs) {
  892. const shenpiInfo = await ctx.service.shenpiAudit.getDataByCondition({ tid: tender.id, sp_type: shenpiConst.sp_type[shenpi], sp_status: tender.shenpiInfo[shenpi] });
  893. shenpiauditList[shenpi] = shenpiInfo && shenpiInfo.audit_id ? [shenpiInfo.audit_id] : null;
  894. }
  895. }
  896. tender.shenpiauditList = shenpiauditList;
  897. }
  898. }
  899. if (removeTenders.length > 0) {
  900. ctx.helper._.remove(tenders, function(n) {
  901. return removeTenders.indexOf(n.id) !== -1;
  902. });
  903. }
  904. const categoryData = await ctx.service.category.getAllCategory(ctx.session.sessionProject.id);
  905. // 是否修订中
  906. const lastRevise = await ctx.service.ledgerRevise.getLastestRevise(ctx.tender.id);
  907. const revising = (lastRevise && lastRevise.status !== auditConst.revise.status.checked) || false;
  908. const cooperationNum = await ctx.service.ledgerCooperation.count({ tid: ctx.tender.id, status: 1 });
  909. const renderData = {
  910. shenpi: shenpiConst,
  911. accountList,
  912. accountGroup: accountGroupList,
  913. tenders,
  914. categoryData,
  915. auditConst,
  916. revising,
  917. measureType,
  918. cooperationNum,
  919. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.tender.shenpi),
  920. };
  921. await this.layout('tender/shenpi.ejs', renderData, 'tender/shenpi_modal.ejs');
  922. }
  923. async saveTenderInfoShenpi(ctx) {
  924. try {
  925. const data = JSON.parse(ctx.request.body.data);
  926. if (!data) {
  927. throw '提交数据错误';
  928. }
  929. // 判断修改权限
  930. if (ctx.session.sessionUser.is_admin === 0) {
  931. throw '你没有权限修改审批流程';
  932. }
  933. // let postData = {};
  934. // if (!ctx.tender.info.shenpi) {
  935. // for (const sp of shenpiConst.sp_lc) {
  936. // if (sp.code === data.code) {
  937. // postData[sp.code] = data.status;
  938. // } else {
  939. // postData[sp.code] = shenpiConst.sp_status.sqspr;
  940. // }
  941. // }
  942. // } else {
  943. const postData = ctx.tender.info.shenpi;
  944. postData[data.code] = data.status;
  945. if (data.code === shenpiConst.sp_lc[shenpiConst.sp_type.stage - 1].code) {
  946. const status = parseInt(data.status) === shenpiConst.sp_status.gdspl ? 1 : 0;
  947. await ctx.service.ledgerCooperation.changeAllStatus(status);
  948. }
  949. // }
  950. await ctx.service.tenderInfo.saveTenderInfo(ctx.tender.id, { shenpi: postData });
  951. let auditList = [];
  952. if (data.status === shenpiConst.sp_status.gdspl) {
  953. auditList = await ctx.service.shenpiAudit.getAuditList(ctx.tender.id, shenpiConst.sp_type[data.code], data.status);
  954. } else if (data.status === shenpiConst.sp_status.gdzs) {
  955. auditList = await ctx.service.shenpiAudit.getAudit(ctx.tender.id, shenpiConst.sp_type[data.code], data.status);
  956. }
  957. ctx.body = { err: 0, msg: '', data: auditList };
  958. } catch (err) {
  959. this.log(err);
  960. ctx.body = this.ajaxErrorBody(err, '保存审批流程设置失败');
  961. }
  962. }
  963. async saveShenpiAudit(ctx) {
  964. try {
  965. const data = JSON.parse(ctx.request.body.data);
  966. if (!data) {
  967. throw '提交数据错误';
  968. }
  969. // 判断修改权限
  970. if (ctx.session.sessionUser.is_admin === 0) {
  971. throw '你没有权限修改审批流程';
  972. }
  973. let info = '';
  974. switch (data.type) {
  975. case 'add':
  976. const result = await ctx.service.shenpiAudit.addAudit(data);
  977. if (result) {
  978. throw '添加审批人失败';
  979. }
  980. break;
  981. case 'del':
  982. await ctx.service.shenpiAudit.removeAudit(data);
  983. break;
  984. case 'copy2ot':
  985. await ctx.service.shenpiAudit.copyAudit2otherTender(data);
  986. break;
  987. case 'copy2os':
  988. await ctx.service.shenpiAudit.copyAudit2otherShenpi(data);
  989. break;
  990. case 'pwd':
  991. info = await ctx.service.ledgerCooperation.save(data);
  992. break;
  993. case 'company':
  994. info = await ctx.service.ledgerCooperation.saveCompany(data);
  995. break;
  996. default:break;
  997. }
  998. ctx.body = { err: 0, msg: '', data: info };
  999. } catch (err) {
  1000. this.log(err);
  1001. ctx.body = this.ajaxErrorBody(err, '保存审批流程设置失败');
  1002. }
  1003. }
  1004. /**
  1005. * 签名,上传图片 (Ajax)
  1006. * @param ctx
  1007. * @return {Promise<void>}
  1008. */
  1009. async saveCooperateSign(ctx) {
  1010. try {
  1011. const stream = await ctx.getFileStream();
  1012. const create_time = Date.parse(new Date()) / 1000;
  1013. const id = stream.fields.id;
  1014. const fileInfo = path.parse(stream.filename);
  1015. const fileName = path.join('public/upload', ctx.tender.id.toString(), 'sign', 'signImg_' + create_time + fileInfo.ext);
  1016. await ctx.helper.saveStreamFile(stream, path.join(this.app.baseDir, 'app', fileName));
  1017. if (stream) {
  1018. await sendToWormhole(stream);
  1019. }
  1020. await ctx.service.ledgerCooperation.saveSign(id, fileName);
  1021. ctx.body = { err: 0, msg: '', data: fileName };
  1022. } catch (err) {
  1023. this.log(err);
  1024. ctx.body = { err: 1, msg: err.toString(), data: null };
  1025. }
  1026. }
  1027. /**
  1028. * 拷贝设置
  1029. * @param {object} ctx - 上下文
  1030. */
  1031. async copyTender(ctx) {
  1032. try {
  1033. const id = ctx.tender.data.id;
  1034. const { id: copy_id = '', type: typeArr = [] } = JSON.parse(ctx.request.body.data);
  1035. await ctx.service.tenderInfo.copyTenderHandler(id, copy_id, typeArr);
  1036. ctx.body = { err: 0, msg: '' };
  1037. } catch (error) {
  1038. this.log(error);
  1039. ctx.body = this.ajaxErrorBody(error, '保存审批流程设置失败');
  1040. }
  1041. }
  1042. /**
  1043. * 游客账号设置
  1044. * @param {object} ctx - 上下文
  1045. */
  1046. async saveTourist(ctx) {
  1047. try {
  1048. const data = JSON.parse(ctx.request.body.data);
  1049. if (!data) {
  1050. throw '提交数据错误';
  1051. }
  1052. // 判断修改权限
  1053. if (ctx.session.sessionUser.is_admin === 0) {
  1054. throw '你没有权限修改游客账号';
  1055. }
  1056. let info = '';
  1057. switch (data.type) {
  1058. case 'add':
  1059. const result = await ctx.service.tenderTourist.addAudit(data);
  1060. if (!result) {
  1061. throw '添加审批人失败';
  1062. }
  1063. info = result;
  1064. break;
  1065. case 'del':
  1066. await ctx.service.tenderTourist.removeAudit(data);
  1067. break;
  1068. case 'permission':
  1069. await ctx.service.tenderTourist.setPermission(data);
  1070. break;
  1071. default:break;
  1072. }
  1073. ctx.body = { err: 0, msg: '', data: info };
  1074. } catch (err) {
  1075. this.log(err);
  1076. ctx.body = this.ajaxErrorBody(err, '保存游客账号设置失败');
  1077. }
  1078. }
  1079. /**
  1080. * 获取部位明细数据(Ajax)
  1081. *
  1082. * @param ctx
  1083. * @return {Promise<void>}
  1084. */
  1085. async loadLedgerData(ctx) {
  1086. try {
  1087. const ledgerData = await ctx.service.ledger.getData(ctx.tender.id);
  1088. const posData = ctx.tender.data.measure_type === measureType.tz.value
  1089. ? await ctx.service.pos.getPosData({ tid: ctx.tender.id }) : [];
  1090. const convert = new billsPosConvert(ctx);
  1091. convert.loadData(ledgerData, posData, []);
  1092. const result = await convert.convert();
  1093. const ledgerCooperationList = await ctx.service.ledgerCooperation.getAllDataByCondition({ where: { tid: ctx.tender.id, status: 1 } });
  1094. ctx.body = { err: 0, msg: '', data: { ledgerList: result, ledgerCooperationList } };
  1095. } catch (err) {
  1096. this.log(err);
  1097. ctx.body = { err: 1, msg: err.toString(), data: [] };
  1098. }
  1099. }
  1100. async billsTag(ctx) {
  1101. try {
  1102. const isValidTourist = ctx.tender.isTourist && ctx.tender.touristPermission.tag;
  1103. if (ctx.stage) {
  1104. const isAuditor = ctx.stage.users.indexOf(this.ctx.session.sessionUser.accountId) >= 0;
  1105. if (!isAuditor && !isValidTourist) throw '您无权进行该操作';
  1106. } else if (ctx.revise) {
  1107. const isAuditor = ctx.revise.reviseUsers.indexOf(this.ctx.session.sessionUser.accountId) >= 0;
  1108. if (!isAuditor && !isValidTourist) throw '您无权进行该操作';
  1109. } else {
  1110. const isAuditor = ctx.tender.ledgerUsers.indexOf(this.ctx.session.sessionUser.accountId) >= 0;
  1111. if (!isAuditor && !isValidTourist) throw '您无权进行该操作';
  1112. }
  1113. const data = JSON.parse(ctx.request.body.data);
  1114. const result = await ctx.service.ledgerTag.update(data);
  1115. ctx.body = { err: 0, msg: '', data: result };
  1116. } catch (err) {
  1117. this.log(err);
  1118. ctx.body = this.ajaxErrorBody(err, '书签数据错误');
  1119. }
  1120. }
  1121. async listLoad(ctx) {
  1122. try {
  1123. const data = JSON.parse(ctx.request.body.data);
  1124. if (!data.tid || !data.lid || !data.type) throw '数据错误';
  1125. const responseData = {
  1126. err: 0,
  1127. msg: '',
  1128. data: { ledgerAuditConst: auditConst.ledger, stageAuditConst: auditConst.stage },
  1129. };
  1130. responseData.data.category = await this.ctx.service.category.getAllCategory(this.ctx.session.sessionProject.id);
  1131. // 获取用户权限
  1132. const accountInfo = await this.ctx.service.projectAccount.getDataById(this.ctx.session.sessionUser.accountId);
  1133. const userPermission = accountInfo !== undefined && accountInfo.permission !== '' ? JSON.parse(accountInfo.permission) : null;
  1134. const tenderList = await this.ctx.service.tender.getList('', userPermission);
  1135. for (const t of tenderList) {
  1136. if (t.ledger_status === auditConst.ledger.status.checked) {
  1137. t.lastStage = await this.ctx.service.stage.getLastestStage(t.id, false);
  1138. }
  1139. }
  1140. if (data.type === 'ledger') {
  1141. responseData.data.tenders = tenderList.filter(x => {
  1142. return x.ledger_status === auditConst.ledger.status.checked;
  1143. });
  1144. const history = await this.ctx.service.sumLoadHistory.getLedgerHistory(data.tid, data.lid);
  1145. if (history) responseData.data.history = { tenders: history.tenders, load_time: history.load_time, type: 'ledger' };
  1146. } else if (data.type === 'stage') {
  1147. responseData.data.tenders = tenderList.filter(x => {
  1148. return x.ledger_status === auditConst.ledger.status.checked && !!x.lastStage;
  1149. });
  1150. let history = await this.ctx.service.sumLoadHistory.getStageHistory(data.tid, data.lid);
  1151. if (history) {
  1152. responseData.data.history = { tenders: history.tenders, load_time: history.load_time, type: 'stage' };
  1153. } else {
  1154. history = await this.ctx.service.sumLoadHistory.getReviseHistory(data.tid, data.lid);
  1155. if (history) {
  1156. responseData.data.history = { tenders: history.tenders, load_time: history.load_time, type: 'revise' };
  1157. } else {
  1158. history = await this.ctx.service.sumLoadHistory.getLedgerHistory(data.tid, data.lid);
  1159. if (history) responseData.data.history = { tenders: history.tenders, load_time: history.load_time, type: 'ledger' };
  1160. }
  1161. }
  1162. } else if (data.type === 'revise') {
  1163. responseData.data.tenders = tenderList.filter(x => {
  1164. return x.ledger_status === auditConst.ledger.status.checked;
  1165. });
  1166. let history = await this.ctx.service.sumLoadHistory.getReviseHistory(data.tid, data.lid);
  1167. if (history) {
  1168. responseData.data.history = { tenders: history.tenders, load_time: history.load_time, type: 'revise' };
  1169. } else {
  1170. history = await this.ctx.service.sumLoadHistory.getLedgerHistory(data.tid, data.lid);
  1171. if (history) responseData.data.history = { tenders: history.tenders, load_time: history.load_time, type: 'ledger' };
  1172. }
  1173. }
  1174. ctx.body = responseData;
  1175. } catch(err) {
  1176. this.log(err);
  1177. this.ajaxErrorBody(err, '获取标段列表错误');
  1178. }
  1179. }
  1180. async listLoad2(ctx) {
  1181. try {
  1182. const data = JSON.parse(ctx.request.body.data);
  1183. if (!data.type) throw '数据错误';
  1184. const responseData = {
  1185. err: 0,
  1186. msg: '',
  1187. data: { ledgerAuditConst: auditConst.ledger, stageAuditConst: auditConst.stage },
  1188. };
  1189. responseData.data.category = await this.ctx.service.category.getAllCategory(this.ctx.session.sessionProject.id);
  1190. // 获取用户权限
  1191. const accountInfo = await this.ctx.service.projectAccount.getDataById(this.ctx.session.sessionUser.accountId);
  1192. const userPermission = accountInfo !== undefined && accountInfo.permission !== '' ? JSON.parse(accountInfo.permission) : null;
  1193. const tenderList = await this.ctx.service.tender.getList('', userPermission);
  1194. for (const t of tenderList) {
  1195. if (t.ledger_status === auditConst.ledger.status.checked) {
  1196. t.lastStage = await this.ctx.service.stage.getLastestStage(t.id, false);
  1197. t.lastCheckedStage = await this.ctx.service.stage.getLastestCompleteStage(t.id);
  1198. }
  1199. }
  1200. if (data.type === 'all') {
  1201. responseData.data.tenders = tenderList;
  1202. } else if (data.type === 'ledger-checked') {
  1203. responseData.data.tenders = tenderList.filter(x => {
  1204. return x.ledger_status === auditConst.ledger.status.checked;
  1205. });
  1206. } else if (data.type === 'stage') {
  1207. responseData.data.tenders = tenderList.filter(x => {
  1208. return x.ledger_status === auditConst.ledger.status.checked && !!x.lastStage;
  1209. });
  1210. } else if (data.type === 'stage-checked') {
  1211. responseData.data.tenders = tenderList.filter(x => {
  1212. return x.ledger_status === auditConst.ledger.status.checked && !!x.lastCheckedStage;
  1213. });
  1214. }
  1215. ctx.body = responseData;
  1216. } catch(err) {
  1217. this.log(err);
  1218. this.ajaxErrorBody(err, '获取标段列表错误');
  1219. }
  1220. }
  1221. async sumLoad(ctx) {
  1222. try {
  1223. const data = JSON.parse(ctx.request.body.data);
  1224. if (!data.lid || !data.type || !data.tenders) throw '数据错误';
  1225. switch (data.type) {
  1226. case 'ledger':
  1227. const refreshData = await this.ctx.service.ledger.sumLoad(data.lid, data.tenders);
  1228. ctx.body = {err: 0, msg: '', data: refreshData};
  1229. break;
  1230. case 'revise':
  1231. const reviseData = await this.ctx.service.reviseBills.sumLoad(data.lid, this.ctx.revise.id, data.tenders);
  1232. ctx.body = {err: 0, msg: '', data: reviseData};
  1233. break;
  1234. case 'stage':
  1235. const stageData = await this.ctx.service.stageBills.sumLoad(data.lid, data.tenders, data.cover);
  1236. stageData.import_change = { target: { import_lid: data.lid } };
  1237. stageData.import_change.data = await this.ctx.service.stageImportChange.getLeafXmjStageImportData(ctx.stage, data.lid);
  1238. await ctx.service.stage.updateCheckCalcFlag(ctx.stage, true);
  1239. await ctx.service.stage.updateCacheTime(ctx.stage.id);
  1240. ctx.body = {err: 0, msg: '', data: stageData};
  1241. break;
  1242. default:
  1243. throw '数据错误';
  1244. }
  1245. } catch (err) {
  1246. this.log(err);
  1247. ctx.ajaxErrorBody(err, '导入数据失败');
  1248. }
  1249. }
  1250. /**
  1251. * 地图数据设置(Ajax)
  1252. *
  1253. * @param ctx
  1254. * @return {Promise<void>}
  1255. */
  1256. async saveMap(ctx) {
  1257. try {
  1258. const data = JSON.parse(ctx.request.body.data);
  1259. if (!data) {
  1260. throw '提交数据错误';
  1261. }
  1262. // 判断修改权限
  1263. if (ctx.session.sessionUser.is_admin === 0) {
  1264. throw '你没有权限修改概况设置';
  1265. }
  1266. let info = '';
  1267. let result = '';
  1268. switch (data.type) {
  1269. case 'select-map':
  1270. result = await ctx.service.tender.saveTenderData(ctx.tender.id, { had_map: data.value });
  1271. if (!result) {
  1272. throw '修改概况设置失败';
  1273. }
  1274. info = result;
  1275. break;
  1276. case 'add-map':
  1277. result = await ctx.service.tenderMap.addMap(ctx.tender.id, data.name);
  1278. info = await ctx.service.tenderMap.getDataById(result.insertId);
  1279. break;
  1280. case 'del-map':
  1281. await ctx.service.tenderMap.deleteById(data.id);
  1282. break;
  1283. case 'save-map':
  1284. await ctx.service.tenderMap.saveMap(data.mapData);
  1285. break;
  1286. default:break;
  1287. }
  1288. ctx.body = { err: 0, msg: '', data: info };
  1289. } catch (err) {
  1290. this.log(err);
  1291. ctx.body = this.ajaxErrorBody(err, '保存概况设置失败');
  1292. }
  1293. }
  1294. /**
  1295. * 上传静态图
  1296. *
  1297. * @param {object} ctx - egg全局变量
  1298. * @return {void}
  1299. */
  1300. async uploadMap(ctx) {
  1301. const responseData = {
  1302. err: 0, msg: '', data: null,
  1303. };
  1304. try {
  1305. const stream = await ctx.getFileStream();
  1306. const create_time = Date.parse(new Date()) / 1000;
  1307. const fileInfo = path.parse(stream.filename);
  1308. const filepath = `app/public/upload/${ctx.tender.id}/map/${create_time}${fileInfo.ext}`;
  1309. await ctx.app.fujianOss.put(ctx.app.config.fujianOssFolder + filepath, stream);
  1310. await sendToWormhole(stream);
  1311. const tenderInfo = await ctx.service.tender.getDataById(ctx.tender.id);
  1312. const oldMapPic = tenderInfo && tenderInfo.map_pic ? tenderInfo.map_pic : null;
  1313. const result = await ctx.service.tender.update({ map_pic: filepath }, { id: ctx.tender.id });
  1314. if (result) {
  1315. // 移除旧的map_pic oss
  1316. if (oldMapPic) {
  1317. await ctx.app.fujianOss.delete(ctx.app.config.fujianOssFolder + oldMapPic);
  1318. }
  1319. responseData.data = { map_pic: filepath };
  1320. } else {
  1321. throw '添加数据库失败';
  1322. }
  1323. } catch (err) {
  1324. this.log(err);
  1325. responseData.err = 1;
  1326. responseData.msg = err;
  1327. }
  1328. ctx.body = responseData;
  1329. }
  1330. async loadData(ctx) {
  1331. try {
  1332. const data = JSON.parse(ctx.request.body.data);
  1333. const filter = data.filter.split(';');
  1334. const responseData = { err: 0, msg: '', data: {} };
  1335. const where = { tid: this.ctx.tender.id };
  1336. for (const f of filter) {
  1337. switch (f) {
  1338. case 'change':
  1339. where.valid = 1;
  1340. if (ctx.session.sessionProject.page_show.isOnlyChecked) where.status = auditConst.flow.status.checked;
  1341. responseData.data[f] = await ctx.service.change.getAllDataByCondition({
  1342. columns: [ 'cid', 'code', 'name', 'selected' ],
  1343. where,
  1344. orders: [['in_time', 'desc']],
  1345. });
  1346. break;
  1347. case 'change_project':
  1348. if (ctx.session.sessionProject.page_show.isOnlyChecked) where.status = auditConst.changeProject.status.checked;
  1349. responseData.data[f] = await ctx.service.changeProject.getAllDataByCondition({
  1350. columns: [ 'id', 'code', 'name', 'selected' ],
  1351. where,
  1352. orders: [['in_time', 'desc']],
  1353. });
  1354. break;
  1355. case 'change_apply':
  1356. if (ctx.session.sessionProject.page_show.isOnlyChecked) where.status = auditConst.changeApply.status.checked;
  1357. responseData.data[f] = await ctx.service.changeApply.getAllDataByCondition({
  1358. columns: [ 'id', 'code', 'name', 'selected' ],
  1359. where,
  1360. orders: [['in_time', 'desc']],
  1361. });
  1362. break;
  1363. case 'change_plan':
  1364. if (ctx.session.sessionProject.page_show.isOnlyChecked) where.status = auditConst.changeApply.status.checked;
  1365. responseData.data[f] = await ctx.service.changePlan.getAllDataByCondition({
  1366. columns: [ 'id', 'code', 'name', 'selected' ],
  1367. where,
  1368. orders: [['in_time', 'desc']],
  1369. });
  1370. break;
  1371. case 'advance':
  1372. const advance = await ctx.service.advance.getAllDataByCondition({
  1373. columns: [ 'id', 'order', 'status', 'selected', 'type' ],
  1374. where,
  1375. orders: [['type', 'asc'], ['create_time', 'desc']],
  1376. });
  1377. advance.forEach(x => {
  1378. x.orderStr = `第${x.order}期`;
  1379. x.statusStr = auditConst.advance.statusString[x.status];
  1380. x.typeStr = advanceConst.typeColMap[x.type].text;
  1381. });
  1382. responseData.data[f] = advance;
  1383. break;
  1384. case 'pm_deal':
  1385. if (accountPermission.PermissionCheck.viewPmData(this.ctx.session.sessionUser.permission)) {
  1386. const selects = await this.ctx.service.project.getPmDealCache(this.ctx.session.sessionProject.id);
  1387. // todo 接入项目管理系统
  1388. const pm = require('../lib/pm');
  1389. const tenders = await pm.dealCatagory(ctx, this.ctx.session.sessionProject.code);
  1390. tenders.forEach(x => {
  1391. x.selected = selects.indexOf(x.id + '') >= 0;
  1392. });
  1393. responseData.data[f] = tenders;
  1394. } else {
  1395. throw '您无权查看该数据';
  1396. }
  1397. break;
  1398. default:
  1399. throw '未知数据类型';
  1400. }
  1401. }
  1402. ctx.body = responseData;
  1403. } catch (err) {
  1404. ctx.log(err);
  1405. ctx.ajaxErrorBody(err, '加载数据错误');
  1406. }
  1407. }
  1408. async saveRelaData(ctx) {
  1409. try {
  1410. const data = JSON.parse(ctx.request.body.data);
  1411. const responseData = { err: 0, msg: '', data: {} };
  1412. if (data.change) await this.ctx.service.change.updateChangeSelect(data.change);
  1413. if (data.change_apply) await this.ctx.service.changeApply.defaultUpdateRows(data.change_apply);
  1414. if (data.change_project) await this.ctx.service.changeProject.defaultUpdateRows(data.change_project);
  1415. if (data.change_plan) await this.ctx.service.changePlan.defaultUpdateRows(data.change_plan);
  1416. if (data.advance) await this.ctx.service.advance.defaultUpdateRows(data.advance);
  1417. if (data.pm_deal) await this.ctx.service.project.setPmDealCache(this.ctx.session.sessionProject.id, data.pm_deal);
  1418. ctx.body = responseData;
  1419. } catch (err) {
  1420. ctx.log(err);
  1421. ctx.ajaxErrorBody(err, '保存数据错误');
  1422. }
  1423. }
  1424. }
  1425. return TenderController;
  1426. };