tender_controller.js 88 KB

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