tender_controller.js 75 KB

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