template_controller.js 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271
  1. 'use strict';
  2. /**
  3. *
  4. *
  5. * @author Mai
  6. * @date
  7. * @version
  8. */
  9. const path = require('path');
  10. const fs = require('fs');
  11. const sendToWormhole = require('stream-wormhole');
  12. module.exports = app => {
  13. class TemplateController extends app.BaseController {
  14. /**
  15. * 下载 各种模板
  16. * @param ctx
  17. * @returns {Promise<void>}
  18. */
  19. async download(ctx) {
  20. const file = ctx.params.file;
  21. if (file) {
  22. try {
  23. let fileName;
  24. switch (file) {
  25. case '导入分项清单EXCEL格式.xlsx':
  26. fileName = path.join(this.app.baseDir, 'app', 'public', 'files', 'template', 'ledger', '导入分项清单EXCEL格式.xlsx');
  27. break;
  28. case '导入工程量清单EXCEL格式.xls':
  29. fileName = path.join(this.app.baseDir, 'app', 'public', 'files', 'template', 'ledger', '导入工程量清单EXCEL格式.xls');
  30. break;
  31. case '估概预算示例EXCEL格式.xlsx':
  32. fileName = path.join(this.app.baseDir, 'app', 'public', 'files', 'template', file);
  33. break;
  34. case '控制价示例EXCEL格式.xlsx':
  35. fileName = path.join(this.app.baseDir, 'app', 'public', 'files', 'template', file);
  36. break;
  37. default:
  38. throw '参数错误'
  39. }
  40. ctx.body = await fs.readFileSync(fileName);
  41. } catch (err) {
  42. this.log(err);
  43. if (err.stack) {
  44. ctx.body = '您下载的示例文件不存在';
  45. } else {
  46. ctx.body = err;
  47. }
  48. }
  49. } else {
  50. ctx.body = '参数错误';
  51. }
  52. }
  53. async posCalc(ctx) {
  54. try {
  55. const renderData = {
  56. validColInfo: ctx.service.calcTmpl.TemplateRela.posCalc.ValidColInfo,
  57. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.template.posCalc),
  58. };
  59. renderData.folderList = await ctx.service.calcTmplFolder.getData(ctx.session.sessionProject.id + '-' + 'posCalc');
  60. renderData.templateList = await ctx.service.calcTmpl.getAllTemplate(ctx.session.sessionProject.id, 'posCalc');
  61. await ctx.service.calcTmpl.checkTemplateUsed(renderData.templateList, 'posCalc');
  62. const specList = await ctx.service.stdExtraList.getList(0);
  63. renderData.specList = specList.map(x => { return { value: x.id, text: x.name }; });
  64. await this.layout('template/pos_calc.ejs', renderData, 'template/pos_calc_modal.ejs');
  65. } catch (err) {
  66. ctx.log(err);
  67. ctx.postError(err, '查看模板数据错误');
  68. ctx.redirect(ctx.request.header.referer);
  69. }
  70. }
  71. async cost(ctx) {
  72. try {
  73. const renderData = {
  74. validColInfo: ctx.service.calcTmpl.TemplateRela.cost.ValidColInfo,
  75. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.template.cost),
  76. };
  77. renderData.templateList = await ctx.service.calcTmpl.getAllTemplate(ctx.session.sessionProject.id, 'cost');
  78. await ctx.service.calcTmpl.checkTemplateUsed(renderData.templateList, 'cost');
  79. await this.layout('template/cost.ejs', renderData, 'template/cost_modal.ejs');
  80. } catch (err) {
  81. ctx.log(err);
  82. ctx.postError(err, '查看模板数据错误');
  83. ctx.redirect(ctx.request.header.referer);
  84. }
  85. }
  86. async saveFolder(ctx) {
  87. try {
  88. const data = JSON.parse(ctx.request.body.data);
  89. const responseData = { err: 0, msg: '', data: {} };
  90. if (!data.postType || !data.postData || !data.target) throw '数据错误';
  91. const masterId = `${this.ctx.session.sessionProject.id}-${data.target}`;
  92. switch (data.postType) {
  93. case 'add':
  94. responseData.data = await this.ctx.service.calcTmplFolder.addNode(masterId, data.postData.id);
  95. break;
  96. case 'delete':
  97. responseData.data = await this.ctx.service.calcTmplFolder.delete(masterId, data.postData.id, data.postData.count);
  98. break;
  99. case 'up-move':
  100. responseData.data = await this.ctx.service.calcTmplFolder.upMoveNode(masterId, data.postData.id, data.postData.count);
  101. break;
  102. case 'down-move':
  103. responseData.data = await this.ctx.service.calcTmplFolder.downMoveNode(masterId, data.postData.id, data.postData.count);
  104. break;
  105. case 'up-level':
  106. responseData.data = await this.ctx.service.calcTmplFolder.upLevelNode(masterId, data.postData.id, data.postData.count);
  107. break;
  108. case 'down-level':
  109. responseData.data = await this.ctx.service.calcTmplFolder.downLevelNode(masterId, data.postData.id, data.postData.count);
  110. break;
  111. case 'update':
  112. responseData.data = await this.ctx.service.calcTmplFolder.updateInfos(masterId, data.postData);
  113. break;
  114. default:
  115. throw '未知操作';
  116. }
  117. ctx.body = responseData;
  118. } catch (err) {
  119. this.log(err);
  120. ctx.body = this.ajaxErrorBody(err, '数据错误');
  121. }
  122. }
  123. // ------------ 以下方法为所有模板共用 --------------
  124. async load(ctx) {
  125. try {
  126. const data = JSON.parse(ctx.request.body.data);
  127. const filter = data.filter ? data.filter.split(';') : [];
  128. const result = {};
  129. for (const f of filter) {
  130. switch(f) {
  131. case 'detail':
  132. const detail = await this.ctx.service.calcTmpl.getTemplate(data.id, data.type);
  133. const user = await this.ctx.service.projectAccount.getDataById(detail.create_user_id);
  134. detail.user_name = user.name;
  135. detail.user_company = user.company;
  136. detail.user_role = user.role;
  137. detail.create_time_str = this.ctx.moment(detail.create_time).format('YYYY-MM-DD HH:mm:ss');
  138. result[f] = detail;
  139. break;
  140. default:
  141. throw '未知数据类型';
  142. }
  143. }
  144. ctx.body = { err: 0, msg: '', data: result };
  145. } catch (error) {
  146. ctx.log(error);
  147. ctx.ajaxErrorBody(error, '加载数据失败');
  148. }
  149. }
  150. async saveTemplate(ctx) {
  151. try {
  152. const data = JSON.parse(ctx.request.body.data);
  153. const result = await ctx.service.calcTmpl.saveTemplate(data);
  154. ctx.body = { err: 0, msg: '', data: result };
  155. } catch (err) {
  156. ctx.log(err);
  157. ctx.ajaxErrorBody(err, '修改数据失败');
  158. }
  159. }
  160. async preview(ctx) {
  161. try {
  162. const data = JSON.parse(ctx.request.body.data);
  163. const spreadSetting = this.ctx.service.calcTmpl.calcSpreadCache(data.type, data.col_set, data.multi_header);
  164. const testData = await this.ctx.service.calcTmpl.getCalcTestData(data.col_set, data.type, 3);
  165. const calc = this.ctx.service.calcTmpl._getCalcExpr(spreadSetting.cols);
  166. ctx.body = { err: 0, msg: '', data: { spreadSetting, testData, calc } };
  167. } catch (error) {
  168. ctx.log(error);
  169. ctx.ajaxErrorBody(error, '预览失败');
  170. }
  171. }
  172. async reCalcTemplate(ctx) {
  173. try {
  174. const templateId = ctx.query.tid;
  175. await ctx.service.calcTmpl.reCalcTemplate(templateId, true);
  176. } catch(error) {
  177. ctx.log(error);
  178. ctx.postError(error, '重算模板失败');
  179. }
  180. ctx.redirect(ctx.request.header.referer);
  181. }
  182. async exportTemplate(ctx) {
  183. try {
  184. const template = await ctx.service.calcTmpl.getTemplate(ctx.query.tid);
  185. if (!template) throw '计算模板不存在';
  186. if (template.project_id !== ctx.session.sessionProject.id) throw '您无权导出该模板数据';
  187. const Cpd = require('../lib/crypt').cpd;
  188. const cpd = new Cpd();
  189. const filename = `${template.name}.ctd`;
  190. const filepath = path.join(this.ctx.app.baseDir, 'temp', filename);
  191. await cpd.encrypt({ type: template.type, col_set: template.col_set, multi_header: template.multi_header }, filepath);
  192. const userAgent = (ctx.request.header['user-agent'] || '').toLowerCase();
  193. let disposition = '';
  194. if (userAgent.indexOf('msie') >= 0 || userAgent.indexOf('chrome') >= 0) {
  195. disposition = 'attachment; filename=' + encodeURIComponent(filename);
  196. } else if (userAgent.indexOf('firefox') >= 0) {
  197. disposition = 'attachment; filename*="utf8\'\'' + encodeURIComponent(filename) + '"';
  198. } else {
  199. /* safari等其他非主流浏览器只能自求多福了 */
  200. disposition = 'attachment; filename=' + new Buffer(filename).toString('binary');
  201. }
  202. ctx.response.set({
  203. 'Content-Type': 'application/octet-stream',
  204. 'Content-Disposition': disposition,
  205. });
  206. ctx.body = await fs.readFileSync(filepath);
  207. } catch(err) {
  208. ctx.log(err);
  209. ctx.postError(err, '导出模板数据失败');
  210. }
  211. }
  212. async importTemplate(ctx) {
  213. const template = await ctx.service.calcTmpl.getTemplate(ctx.query.tid);
  214. if (!template) throw '计算模板不存在';
  215. const create_time = Date.parse(new Date()) / 1000;
  216. const filepath = path.join(this.ctx.app.baseDir, 'temp', `计算模板${create_time}.ctd`);
  217. let stream, index = 0;
  218. try {
  219. const parts = ctx.multipart({ autoFields: true });
  220. stream = await parts();
  221. while (stream !== undefined) {
  222. if (!stream.filename) throw '未发现上传文件!';
  223. // 保存文件
  224. await ctx.helper.saveStreamFile(stream, filepath);
  225. await sendToWormhole(stream);
  226. ++index;
  227. if (Array.isArray(parts.field.size) && index < parts.field.size.length) {
  228. stream = await parts();
  229. } else {
  230. stream = undefined;
  231. }
  232. }
  233. const Cpd = require('../lib/crypt').cpd;
  234. const cpd = new Cpd();
  235. const data = await cpd.decrypt(filepath);
  236. if (template.type !== data.type) throw '导入的模板文件与当前模板类型不一致,不可导入';
  237. const result =await ctx.service.calcTmpl.saveTemplate({ update: { id: template.id, col_set: data.col_set, multi_header: data.multi_header } });
  238. ctx.body = { err: 0, mgs: '', data: result };
  239. } catch (err) {
  240. // 失败需要消耗掉stream 以防卡死
  241. if (stream) await sendToWormhole(stream);
  242. ctx.log(err);
  243. ctx.ajaxErrorBody(err, '导入模板文件失败');
  244. }
  245. }
  246. // ---------------------------------------------------
  247. async saveLedgerTemplate(ctx) {
  248. try {
  249. const data = JSON.parse(ctx.request.body.data);
  250. const result = await ctx.service.ledgerTemplate.saveTemplate(data);
  251. ctx.body = { err: 0, msg: '', data: result };
  252. } catch (err) {
  253. ctx.log(err);
  254. ctx.ajaxErrorBody(err, '修改数据失败');
  255. }
  256. }
  257. }
  258. return TemplateController;
  259. };