tender_controller.js 76 KB

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