template_controller.js 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217
  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. if (!ctx.subProject.page_show.posCalcDetail) throw '该功能已关闭';
  56. const renderData = {
  57. validColInfo: ctx.service.calcTmpl.TemplateRela.posCalc.ValidColInfo,
  58. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.template.posCalc),
  59. };
  60. renderData.templateList = await ctx.service.calcTmpl.getAllTemplate(ctx.subProject.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. // if (!ctx.subProject.page_show.cost) throw '该功能已关闭';
  74. const renderData = {
  75. validColInfo: ctx.service.calcTmpl.TemplateRela.cost.ValidColInfo,
  76. jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.template.cost),
  77. };
  78. renderData.templateList = await ctx.service.calcTmpl.getAllTemplate(ctx.subProject.id, 'cost');
  79. await ctx.service.calcTmpl.checkTemplateUsed(renderData.templateList, 'cost');
  80. await this.layout('template/cost.ejs', renderData, 'template/cost_modal.ejs');
  81. } catch (err) {
  82. ctx.log(err);
  83. ctx.postError(err, '查看模板数据错误');
  84. ctx.redirect(ctx.request.header.referer);
  85. }
  86. }
  87. // ------------ 以下方法为所有模板共用 --------------
  88. async load(ctx) {
  89. try {
  90. const data = JSON.parse(ctx.request.body.data);
  91. const filter = data.filter ? data.filter.split(';') : [];
  92. const result = {};
  93. for (const f of filter) {
  94. switch(f) {
  95. case 'detail':
  96. result[f] = await this.ctx.service.calcTmpl.getTemplate(data.id, data.type);
  97. break;
  98. default:
  99. throw '未知数据类型';
  100. }
  101. }
  102. ctx.body = { err: 0, msg: '', data: result };
  103. } catch (error) {
  104. ctx.log(error);
  105. ctx.ajaxErrorBody(error, '加载数据失败');
  106. }
  107. }
  108. async saveTemplate(ctx) {
  109. try {
  110. const data = JSON.parse(ctx.request.body.data);
  111. const result = await ctx.service.calcTmpl.saveTemplate(data);
  112. ctx.body = { err: 0, msg: '', data: result };
  113. } catch (err) {
  114. ctx.log(err);
  115. ctx.ajaxErrorBody(err, '修改数据失败');
  116. }
  117. }
  118. async preview(ctx) {
  119. try {
  120. const data = JSON.parse(ctx.request.body.data);
  121. const spreadSetting = this.ctx.service.calcTmpl.calcSpreadCache(data.type, data.col_set, data.multi_header);
  122. const testData = await this.ctx.service.calcTmpl.getCalcTestData(data.col_set, data.type, 3);
  123. const calc = this.ctx.service.calcTmpl._getCalcExpr(spreadSetting.cols);
  124. ctx.body = { err: 0, msg: '', data: { spreadSetting, testData, calc } };
  125. } catch (error) {
  126. ctx.log(error);
  127. ctx.ajaxErrorBody(error, '预览失败');
  128. }
  129. }
  130. async reCalcTemplate(ctx) {
  131. try {
  132. const templateId = ctx.query.tid;
  133. await ctx.service.calcTmpl.reCalcTemplate(templateId, true);
  134. } catch(error) {
  135. ctx.log(error);
  136. ctx.postError(error, '重算模板失败');
  137. }
  138. ctx.redirect(ctx.request.header.referer);
  139. }
  140. async exportTemplate(ctx) {
  141. try {
  142. const template = await ctx.service.calcTmpl.getTemplate(ctx.query.tid);
  143. if (!template) throw '计算模板不存在';
  144. if (template.spid !== ctx.subProject.id) throw '您无权导出该模板数据';
  145. const Cpd = require('../lib/crypt').cpd;
  146. const cpd = new Cpd();
  147. const filename = `${template.name}.ctd`;
  148. const filepath = path.join(this.ctx.app.baseDir, 'temp', filename);
  149. await cpd.encrypt({ type: template.type, col_set: template.col_set, multi_header: template.multi_header }, filepath);
  150. const userAgent = (ctx.request.header['user-agent'] || '').toLowerCase();
  151. let disposition = '';
  152. if (userAgent.indexOf('msie') >= 0 || userAgent.indexOf('chrome') >= 0) {
  153. disposition = 'attachment; filename=' + encodeURIComponent(filename);
  154. } else if (userAgent.indexOf('firefox') >= 0) {
  155. disposition = 'attachment; filename*="utf8\'\'' + encodeURIComponent(filename) + '"';
  156. } else {
  157. /* safari等其他非主流浏览器只能自求多福了 */
  158. disposition = 'attachment; filename=' + new Buffer(filename).toString('binary');
  159. }
  160. ctx.response.set({
  161. 'Content-Type': 'application/octet-stream',
  162. 'Content-Disposition': disposition,
  163. });
  164. ctx.body = await fs.readFileSync(filepath);
  165. } catch(err) {
  166. ctx.log(err);
  167. ctx.postError(err, '导出模板数据失败');
  168. }
  169. }
  170. async importTemplate(ctx) {
  171. const template = await ctx.service.calcTmpl.getTemplate(ctx.query.tid);
  172. if (!template) throw '计算模板不存在';
  173. const create_time = Date.parse(new Date()) / 1000;
  174. const filepath = path.join(this.ctx.app.baseDir, 'temp', `计算模板${create_time}.ctd`);
  175. let stream, index = 0;
  176. try {
  177. const parts = ctx.multipart({ autoFields: true });
  178. stream = await parts();
  179. while (stream !== undefined) {
  180. if (!stream.filename) throw '未发现上传文件!';
  181. // 保存文件
  182. await ctx.helper.saveStreamFile(stream, filepath);
  183. await sendToWormhole(stream);
  184. ++index;
  185. if (Array.isArray(parts.field.size) && index < parts.field.size.length) {
  186. stream = await parts();
  187. } else {
  188. stream = undefined;
  189. }
  190. }
  191. const Cpd = require('../lib/crypt').cpd;
  192. const cpd = new Cpd();
  193. const data = await cpd.decrypt(filepath);
  194. if (template.type !== data.type) throw '导入的模板文件与当前模板类型不一致,不可导入';
  195. const result =await ctx.service.calcTmpl.saveTemplate({ update: { id: template.id, col_set: data.col_set, multi_header: data.multi_header } });
  196. ctx.body = { err: 0, mgs: '', data: result };
  197. } catch (err) {
  198. // 失败需要消耗掉stream 以防卡死
  199. if (stream) await sendToWormhole(stream);
  200. ctx.log(err);
  201. ctx.ajaxErrorBody(err, '导入模板文件失败');
  202. }
  203. }
  204. // ---------------------------------------------------
  205. }
  206. return TemplateController;
  207. };