tender_controller.js 70 KB

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