sub_proj_controller.js 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468
  1. 'use strict';
  2. /**
  3. *
  4. *
  5. * @author Mai
  6. * @date
  7. * @version
  8. */
  9. const auditConst = require('../const/audit');
  10. const accountGroup = require('../const/account_group').group;
  11. const sendToWormhole = require('stream-wormhole');
  12. const path = require('path');
  13. module.exports = app => {
  14. class SubProjController extends app.BaseController {
  15. async _loadCount(projectList){
  16. for (const p of projectList) {
  17. p.tender_count = await this.ctx.service.tender.count({ spid: p.id });
  18. p.contract_count = await this.ctx.service.contract.count({ spid: p.id });
  19. p.file_count = await this.ctx.service.file.count({ spid: p.id, is_deleted: 0 });
  20. }
  21. }
  22. /**
  23. * 概算投资
  24. *
  25. * @param ctx
  26. * @returns {Promise<void>}
  27. */
  28. async index(ctx) {
  29. try {
  30. const renderData = {
  31. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.subProject.list),
  32. auditConst,
  33. };
  34. renderData.budgetStd = await ctx.service.budgetStd.getDataByProjectId(ctx.session.sessionProject.id);
  35. renderData.projectList = await ctx.service.subProject.getSubProject(ctx.session.sessionProject.id, ctx.session.sessionUser.accountId, ctx.session.sessionUser.is_admin);
  36. await this._loadCount(renderData.projectList);
  37. renderData.tenderList = await ctx.service.tender.getManageTenderList(ctx.session.sessionProject.id);
  38. const accountList = await ctx.service.projectAccount.getAllDataByCondition({
  39. columns: ['id', 'name', 'company', 'role', 'enable', 'is_admin', 'account_group', 'mobile'],
  40. where: { project_id: ctx.session.sessionProject.id }
  41. });
  42. renderData.accountList = accountList;
  43. const unitList = await ctx.service.constructionUnit.getAllDataByCondition({ where: { pid: ctx.session.sessionProject.id } });
  44. renderData.accountGroup = unitList.map(item => {
  45. const groupList = accountList.filter(item1 => item1.company === item.name);
  46. return { groupName: item.name, groupList };
  47. }).filter(x => { return x.groupList.length > 0; });
  48. renderData.permissionConst = ctx.service.subProjPermission.PermissionConst;
  49. renderData.companys = await this.ctx.service.constructionUnit.getAllDataByCondition({ where: { pid: ctx.session.sessionProject.id } });
  50. renderData.templates = await this.ctx.service.filingTemplateList.getAllTemplate(ctx.session.sessionProject.id);
  51. await this.layout('sub_proj/index.ejs', renderData, 'sub_proj/modal.ejs');
  52. } catch (err) {
  53. console.log(err);
  54. ctx.log(err);
  55. ctx.session.postError = err.toString();
  56. ctx.redirect(this.menu.menu.dashboard.url);
  57. }
  58. }
  59. async addFolder(ctx) {
  60. try {
  61. const data = JSON.parse(ctx.request.body.data);
  62. if (!data.name || data.name.length > 100) throw '文件夹名称有误';
  63. const result = await ctx.service.subProject.addFolder(data);
  64. await this._loadCount(result);
  65. ctx.body = { err: 0, msg: '', data: result };
  66. } catch (err) {
  67. ctx.log(err);
  68. ctx.ajaxErrorBody(err, '新建文件夹失败');
  69. }
  70. }
  71. async addProj(ctx) {
  72. try {
  73. const data = JSON.parse(ctx.request.body.data);
  74. if (!data.name || data.name.length > 100) throw '项目名称有误';
  75. const result = await ctx.service.subProject.addSubProject(data);
  76. await this._loadCount(result);
  77. ctx.body = { err: 0, msg: '', data: result };
  78. } catch (err) {
  79. ctx.log(err);
  80. ctx.ajaxErrorBody(err, '新建项目失败');
  81. }
  82. }
  83. async dragTo(ctx) {
  84. try {
  85. const data = JSON.parse(ctx.request.body.data);
  86. if (!data.drag_id || !data.drop_id) throw '提交数据错误';
  87. const result = await ctx.service.subProject.dragTo(data);
  88. await this._loadCount(result);
  89. ctx.body = { err: 0, msg: '', data: result };
  90. } catch (err) {
  91. ctx.log(err);
  92. ctx.ajaxErrorBody(err, '调整所属文件夹失败');
  93. }
  94. }
  95. async move(ctx) {
  96. try {
  97. const data = JSON.parse(ctx.request.body.data);
  98. if (!data.id || !data.type) throw '提交数据错误';
  99. const result = await ctx.service.subProject.move(data);
  100. await this._loadCount(result);
  101. ctx.body = { err: 0, msg: '', data: result };
  102. } catch (err) {
  103. ctx.log(err);
  104. ctx.ajaxErrorBody(err, '调整所属文件夹失败');
  105. }
  106. }
  107. async del(ctx) {
  108. try {
  109. const data = JSON.parse(ctx.request.body.data);
  110. if (!data.id) throw '参数有误';
  111. const result = await ctx.service.subProject.del(data.id);
  112. ctx.body = { err: 0, msg: '', data: result };
  113. } catch(err) {
  114. ctx.log(err);
  115. ctx.ajaxErrorBody(err, '删除项目失败');
  116. }
  117. }
  118. async save(ctx) {
  119. try {
  120. const data = JSON.parse(ctx.request.body.data);
  121. if (!data.id) throw '参数有误';
  122. let result = null;
  123. if (data.name !== undefined) {
  124. if (!data.name || data.name.length > 100) throw '项目名称有误';
  125. result = await ctx.service.subProject.save({ id: data.id, name: data.name });
  126. } else if (data.rela_tender !== undefined) {
  127. result = await ctx.service.subProject.setRelaTender({ id: data.id, rela_tender: data.rela_tender });
  128. } else if (data.std_id !== undefined) {
  129. result = await ctx.service.subProject.setBudgetStd({ id: data.id, std_id: data.std_id });
  130. } else if (data.management !== undefined) {
  131. result = await ctx.service.subProject.setManagement({ id: data.id, management: data.management, filingTemplate: data.filingTemplate });
  132. }
  133. ctx.body = { err: 0, msg: '', data: { update: [result] } };
  134. } catch(err) {
  135. ctx.log(err);
  136. ctx.ajaxErrorBody(err, '保存数据失败');
  137. }
  138. }
  139. async refresh(ctx) {
  140. try {
  141. const data = JSON.parse(ctx.request.body.data);
  142. if (!data.id) throw '参数有误';
  143. const result = await ctx.service.subProject.refreshManagementPermission({ id: data.id });
  144. ctx.body = { err: 0, msg: '', data: result };
  145. } catch(err) {
  146. ctx.log(err);
  147. ctx.ajaxErrorBody(err, '保存数据失败');
  148. }
  149. }
  150. async rela(ctx) {
  151. try {
  152. const id = ctx.query.id;
  153. const projectList = await ctx.service.subProject.getSubProject(this.ctx.session.sessionProject.id, this.ctx.session.sessionUser.accountId, true);
  154. const otherProj = projectList.filter(x => { return x.id !== id || !x.rela_tender || x.is_folder });
  155. const _ = ctx.helper._;
  156. const otherRela = _.map(_.map(otherProj, 'rela_tender').join(',').split(','), _.toInteger);
  157. const tenderList = await ctx.service.tender.getList4Select('stage');
  158. ctx.body = {
  159. err: 0,
  160. msg: '',
  161. data: tenderList.filter(x => { return otherRela.indexOf(x.id) === -1})
  162. .map(y => { return {id: y.id, name: y.name, lastStageOrder: y.lastStage.order, lastStageStatus: auditConst.stage.statusString[y.lastStage.status], category: y.category}}),
  163. };
  164. } catch (err) {
  165. ctx.log(err);
  166. ctx.postError(err, '获取数据失败');
  167. }
  168. }
  169. async saveRela(ctx) {
  170. try {
  171. const data = JSON.parse(ctx.request.body.data);
  172. if (!data.id || data.rela_tender === undefined) throw '参数有误';
  173. const permission = ctx.session.sessionUser.is_admin
  174. ? ctx.service.subProjPermission.adminPermission
  175. : await ctx.service.subProjPermission.getSubProjectUserPermission(data.id, ctx.session.sessionUser.accountId);
  176. if (!permission || permission.manage_permission.indexOf(ctx.service.subProjPermission.PermissionConst.manage.rela.value) < 0) throw '您无权进行该操作';
  177. const result = await ctx.service.subProject.setRelaTender({ id: data.id, rela_tender: data.rela_tender || '' });
  178. ctx.body = { err: 0, msg: '', data: { update: [result] } };
  179. } catch(err) {
  180. ctx.log(err);
  181. ctx.ajaxErrorBody(err, '保存数据失败');
  182. }
  183. }
  184. async member(ctx) {
  185. try {
  186. const data = JSON.parse(ctx.request.body.data);
  187. const member = await ctx.service.subProjPermission.getPermission(data.id);
  188. ctx.body = { err: 0, msg: '', data: member };
  189. } catch (err) {
  190. ctx.log(err);
  191. ctx.ajaxErrorBody(err, '查询项目成员失败');
  192. }
  193. }
  194. async memberSave(ctx) {
  195. try {
  196. const data = JSON.parse(ctx.request.body.data);
  197. if (!data.id) throw '参数有误';
  198. await ctx.service.subProjPermission.savePermission(data.id, data.member);
  199. ctx.body = { err: 0, msg: '', data: '' };
  200. } catch (err) {
  201. ctx.log(err);
  202. ctx.ajaxErrorBody(err, '保存数据失败');
  203. }
  204. }
  205. async info(ctx) {
  206. try {
  207. const info = await this.ctx.service.subProjInfo.getInfo(ctx.subProject.id);
  208. info.lx_tp = await this.ctx.service.budgetGu.getSumTp(ctx.subProject.budget_id);
  209. info.cb_tp = await this.ctx.service.budgetGai.getSumTp(ctx.subProject.budget_id);
  210. info.sgt_tp = await this.ctx.service.budgetYu.getSumTp(ctx.subProject.budget_id);
  211. info.zbys_tp = await this.ctx.service.budgetZb.getSumTp(ctx.subProject.budget_id);
  212. const renderData = {
  213. info,
  214. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.subProject.info),
  215. };
  216. await this.layout('sub_proj/info.ejs', renderData);
  217. } catch (err) {
  218. ctx.log(err);
  219. }
  220. }
  221. async dataIndex(ctx) {
  222. try {
  223. const info = await this.ctx.service.subProjInfo.getInfo(ctx.subProject.id);
  224. const renderData = {
  225. info,
  226. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.subProject.data),
  227. };
  228. await this.layout('sub_proj/data_index.ejs', renderData);
  229. } catch (err) {
  230. ctx.log(err);
  231. }
  232. }
  233. async saveInfo(ctx) {
  234. try {
  235. const data = JSON.parse(ctx.request.body.data);
  236. const result = await ctx.service.subProjInfo.saveInfo(ctx.subProject.id, data);
  237. ctx.body = { err: 0, msg: '', data: result };
  238. } catch(err) {
  239. ctx.log(err);
  240. }
  241. }
  242. async progress(ctx) {
  243. try {
  244. const fileReferenceList = await ctx.service.subProject.getFileReference(ctx.subProject, ctx.service.subProject.FileReferenceType.info_progress);
  245. const renderData = {
  246. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.subProject.progress),
  247. fileReferenceList,
  248. };
  249. await this.layout('sub_proj/progress.ejs', renderData, 'sub_proj/progress_modal.ejs');
  250. } catch (err) {
  251. ctx.log(err);
  252. ctx.postError(err, '查看阶段进度')
  253. }
  254. }
  255. async load(ctx) {
  256. try {
  257. const data = JSON.parse(ctx.request.body.data);
  258. const filter = data.filter.split(';');
  259. const result = {};
  260. for (const f of filter) {
  261. switch(f) {
  262. case 'progress':
  263. result[f] = await ctx.service.subProjProgress.getData(ctx.subProject);
  264. break;
  265. case 'progress_file':
  266. result[f] = await ctx.service.subProjFile.getData(ctx.subProject.id, 'progress');
  267. break;
  268. case 'push':
  269. result[f] = await ctx.service.subProjPush.getData(ctx.subProject.id);
  270. break;
  271. case 'push_file':
  272. result[f] = await ctx.service.subProjFile.getData(ctx.subProject.id, 'push');
  273. break;
  274. default:
  275. continue;
  276. }
  277. }
  278. ctx.body = { err: 0, msg: '', data: result };
  279. } catch(err) {
  280. ctx.log(err);
  281. ctx.ajaxErrorBody(err, '获取阶段进度数据有误');
  282. }
  283. }
  284. async _progressBase(subProj, type, data) {
  285. if (isNaN(data.id) || data.id <= 0) throw '数据错误';
  286. if (type !== 'add') {
  287. if (isNaN(data.count) || data.count <= 0) data.count = 1;
  288. }
  289. switch (type) {
  290. case 'add':
  291. return await this.ctx.service.subProjProgress.addProgressNode(subProj.id, data.id, data.count);
  292. case 'delete':
  293. return await this.ctx.service.subProjProgress.delete(subProj.id, data.id, data.count);
  294. case 'up-move':
  295. return await this.ctx.service.subProjProgress.upMoveNode(subProj.id, data.id, data.count);
  296. case 'down-move':
  297. return await this.ctx.service.subProjProgress.downMoveNode(subProj.id, data.id, data.count);
  298. case 'up-level':
  299. return await this.ctx.service.subProjProgress.upLevelNode(subProj.id, data.id, data.count);
  300. case 'down-level':
  301. return await this.ctx.service.subProjProgress.downLevelNode(subProj.id, data.id, data.count);
  302. }
  303. }
  304. async progressUpdate(ctx) {
  305. try {
  306. const data = JSON.parse(ctx.request.body.data);
  307. if (!data.postType || !data.postData) throw '数据错误';
  308. const responseData = { err: 0, msg: '', data: {} };
  309. switch (data.postType) {
  310. case 'add':
  311. case 'delete':
  312. case 'up-move':
  313. case 'down-move':
  314. case 'up-level':
  315. case 'down-level':
  316. responseData.data = await this._progressBase(ctx.subProject, data.postType, data.postData);
  317. break;
  318. case 'update':
  319. responseData.data = await this.ctx.service.subProjProgress.updateInfos(ctx.subProject.id, data.postData);
  320. break;
  321. case 'add-std':
  322. responseData.data = await this.ctx.service.subProjProgress.addStdNode(ctx.subProject, data.postData.id, data.postData.stdData);
  323. break;
  324. default:
  325. throw '未知操作';
  326. }
  327. ctx.body = responseData;
  328. } catch (err) {
  329. this.log(err);
  330. ctx.body = this.ajaxErrorBody(err, '数据错误');
  331. }
  332. }
  333. async push(ctx) {
  334. try {
  335. const renderData = {
  336. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.subProject.push),
  337. };
  338. await this.layout('sub_proj/push.ejs', renderData, 'sub_proj/push_modal.ejs');
  339. } catch (err) {
  340. ctx.log(err);
  341. }
  342. }
  343. async pushUpdate(ctx) {
  344. try {
  345. const data = JSON.parse(ctx.request.body.data);
  346. const result = await ctx.service.subProjPush.updateDatas(data);
  347. ctx.body = { err: 0, msg: '', data: result };
  348. } catch (error) {
  349. ctx.helper.log(error);
  350. ctx.body = this.ajaxErrorBody(error, '提交数据失败,请重试');
  351. }
  352. }
  353. /**
  354. * 上传附件
  355. * @param {Object} ctx - egg全局变量
  356. * @return {void}
  357. */
  358. async uploadFile(ctx) {
  359. let stream;
  360. try {
  361. const parts = ctx.multipart({autoFields: true});
  362. let index = 0;
  363. const create_time = Date.parse(new Date()) / 1000;
  364. let stream = await parts();
  365. const user = await ctx.service.projectAccount.getDataById(ctx.session.sessionUser.accountId);
  366. const rela_id = parts.field.rela_id;
  367. const uploadfiles = [];
  368. while (stream !== undefined) {
  369. if (!stream.filename) throw '未发现上传文件!';
  370. const fileInfo = path.parse(stream.filename);
  371. const filepath = `sp/progress/${ctx.subProject.id}/${ctx.moment().format('YYYYMMDD')}/${create_time + '_' + index + fileInfo.ext}`;
  372. // 保存文件
  373. await ctx.app.fujianOss.put(ctx.app.config.fujianOssFolder + filepath, stream);
  374. await sendToWormhole(stream);
  375. // 插入到stage_pay对应的附件列表中
  376. uploadfiles.push({
  377. rela_id,
  378. filename: fileInfo.name,
  379. fileext: fileInfo.ext,
  380. filesize: Array.isArray(parts.field.size) ? parts.field.size[index] : parts.field.size,
  381. filepath,
  382. });
  383. ++index;
  384. if (Array.isArray(parts.field.size) && index < parts.field.size.length) {
  385. stream = await parts();
  386. } else {
  387. stream = undefined;
  388. }
  389. }
  390. const result = await ctx.service.subProjFile.addFiles(ctx.subProject.id, ctx.request.url.split('/')[3], uploadfiles, user);
  391. ctx.body = {err: 0, msg: '', data: result};
  392. } catch (error) {
  393. ctx.log(error);
  394. // 失败需要消耗掉stream 以防卡死
  395. if (stream) await sendToWormhole(stream);
  396. ctx.body = this.ajaxErrorBody(error, '上传附件失败,请重试');
  397. }
  398. }
  399. async deleteFile(ctx) {
  400. try{
  401. const data = JSON.parse(ctx.request.body.data);
  402. if (!data) throw '缺少参数';
  403. const result = await ctx.service.subProjFile.delFiles(data);
  404. ctx.body = { err: 0, msg: '', data: result };
  405. } catch(error) {
  406. this.log(error);
  407. ctx.ajaxErrorBody(error, '删除附件失败');
  408. }
  409. }
  410. async noPermission(ctx) {
  411. ctx.controllerName = ctx.params.block;
  412. await this.layout('sub_proj/no_permission.ejs');
  413. }
  414. /**
  415. * 保存列设置
  416. * @param ctx
  417. * @returns {Promise<void>}
  418. */
  419. async colSet(ctx) {
  420. try {
  421. const colType = ctx.request.body.col_type;
  422. const colSet = JSON.parse(ctx.request.body.col_set);
  423. await ctx.service.projectColSet.setProjectColSet(ctx.subProject.project_id, ctx.subProject.id, colType, colSet);
  424. ctx.redirect(ctx.request.header.referer);
  425. } catch (err) {
  426. ctx.redirect(ctx.request.header.referer);
  427. }
  428. }
  429. }
  430. return SubProjController;
  431. };