rpt_main.js 41 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914
  1. /**
  2. * Created by Tony on 2017/6/26.
  3. */
  4. 'use strict'
  5. const PRE_PAGE_OFFSET = 150;
  6. const NEXT_PAGE_OFFSET = 160;
  7. const FIRST_PAGE_OFFSET = 50;
  8. const LAST_PAGE_OFFSET = 60;
  9. const WAIT_TIME_EXPORT = 300000;
  10. let fontSuffixMapObj = {"表标题": "title", "列标题": "column", "正文内容": "content", "合计": "summary", "表眉/表脚": "header_footer"};
  11. let rptTplObj = {
  12. hasInitialized: false,
  13. pdfFont: {'SmartSimsun': [], 'simhei': [], 'simkai': []},
  14. iniPage: function() {
  15. let me = this;
  16. if (!me.hasInitialized) {
  17. zTreeOprObj.getReportTemplateTree();
  18. zTreeOprObj.selectedPrjIDs = [];
  19. me.hasInitialized = true;
  20. zTreeOprObj.canvas = document.getElementById("rptCanvas");
  21. // canvas.onclick = canvasOprObj.canvasOnClick;
  22. // canvas.onmousemove = canvasOprObj.canvasOnMouseMove;
  23. dynamicLoadJs('/public/jspdf/Arial Narrow-normal.js');
  24. dynamicLoadJs('/public/jspdf/Arial Narrow-bold.js');
  25. dynamicLoadJs('/public/jspdf/Arial Narrow-italic.js');
  26. dynamicLoadJs('/public/jspdf/Arial Narrow-bolditalic.js');
  27. rptControlObj.loadPDFFonts();
  28. // dynamicLoadJs('/public/jspdf/SmartSimsun-normal.js');
  29. // dynamicLoadJs('/public/jspdf/SmartSimsun-bold.js', me.pdfFontSimsunCallBack);
  30. }
  31. },
  32. pdfFontSimsunCallBack: function() {
  33. rptTplObj.pdfFont['SmartSimsun'].push('normal');
  34. rptTplObj.pdfFont['SmartSimsun'].push('bold');
  35. },
  36. pdfFontSimkaiCallBack: function() {
  37. rptTplObj.pdfFont['simkai'].push('normal');
  38. rptTplObj.pdfFont['simkai'].push('bold');
  39. },
  40. pdfFontSimheiCallBack: function() {
  41. rptTplObj.pdfFont['simhei'].push('normal');
  42. rptTplObj.pdfFont['simhei'].push('bold');
  43. }
  44. }
  45. let zTreeOprObj = {
  46. treeObj: null,
  47. prjFolderTreeObj: null,
  48. currentNode: null,
  49. checkedRptTplNodes: null,
  50. currentRptPageRst: null,
  51. defReportPageCfg: null,
  52. currentPage: 1,
  53. maxPages: 0,
  54. canvas: null,
  55. selectedPrjIDs: [],
  56. countChkedRptTpl: function () {
  57. let me = zTreeOprObj;
  58. if (me.treeObj) {
  59. me.checkedRptTplNodes = [];
  60. let chkNodes = me.treeObj.getCheckedNodes(true), cnt = 0, hasCurrentNode = false;
  61. for (let node of chkNodes) {
  62. if (node.nodeType === TPL_TYPE_TEMPLATE) {
  63. cnt++;
  64. me.checkedRptTplNodes.push(node);
  65. if (me.currentNode === node) hasCurrentNode = true;
  66. }
  67. }
  68. if (!hasCurrentNode && cnt === 0 && me.currentNode !== null) {
  69. //这里根据实际需求再做处理
  70. cnt++;
  71. me.checkedRptTplNodes.push(me.currentNode);
  72. }
  73. $("#print_div").find("span").each(function(cIdx,elementSpan){
  74. elementSpan.innerText = cnt;
  75. });
  76. $("#export_div").find("span").each(function(cIdx,elementSpan){
  77. elementSpan.innerText = cnt;
  78. });
  79. }
  80. },
  81. getReportTemplateTree: function() {
  82. let me = zTreeOprObj;
  83. let private_remove_hide_item = function (items, nlv) {
  84. if (items && items.length > 0) {
  85. for (let i = items.length - 1; i >= 0; i--) {
  86. if (!(items[i].released) && items[i].nodeType === 2) {
  87. items.splice(i, 1);
  88. } else {
  89. if (items[i].items && items[i].items.length > 0) {
  90. private_remove_hide_item(items[i].items, nlv + 1);
  91. if (items[i].items.length === 0 && nlv > 0) {
  92. items.splice(i, 1);
  93. }
  94. }
  95. }
  96. }
  97. }
  98. };
  99. let nodeLv = 0;
  100. private_remove_hide_item(TOP_TREE_NODES, nodeLv);
  101. zTreeHelper.createTreeDirectly(TOP_TREE_NODES, rpt_tpl_setting, "rptTplTree", me);
  102. me.treeObj.expandAll(true);
  103. // let topNodes = me.treeObj.getNodes();
  104. // for (let topLvItem of topNodes) {
  105. // me.treeObj.expandNode(topLvItem, true, false, false);
  106. // if (topLvItem.items && topLvItem.items.length > 0) {
  107. // for (let secTopLvItem of topLvItem.items) {
  108. // me.treeObj.expandNode(secTopLvItem, true, false, false);
  109. // }
  110. // }
  111. // }
  112. me.refreshNodes();
  113. },
  114. getCustomerCfg: function() {
  115. let me = zTreeOprObj;
  116. me.defReportPageCfg = {};
  117. Object.assign(me.defReportPageCfg, CUST_CFG);
  118. me.defReportPageCfg.margins = {};
  119. Object.assign(me.defReportPageCfg.margins, CUST_CFG.margins);
  120. me.defReportPageCfg.fonts = [];
  121. for (let fi = 0; fi < CUST_CFG.fonts.length; fi++) {
  122. me.defReportPageCfg.fonts.push({});
  123. Object.assign(me.defReportPageCfg.fonts[fi], CUST_CFG.fonts[fi]);
  124. }
  125. },
  126. iniFontCfgDom: function (cfg) {
  127. for (let font of cfg.fonts) {
  128. let domArrs = [];
  129. let fontPropSuffix = fontSuffixMapObj[font.CfgDispName];
  130. domArrs.push("<div class='row mb-1'>");
  131. //1. label
  132. domArrs.push("<div class='col-3'>" + font.CfgDispName + "</div>");
  133. //2. font name
  134. domArrs.push("<div class='col-3'>");
  135. domArrs.push("<select class='form-control input-sm' id='fontName_" + fontPropSuffix + "' onchange='rptControlObj.changeFontMain(\"" + font.CfgDispName + "\", \"Name\", this)'>");
  136. domArrs.push("<option>宋体</option>");
  137. // <option>楷体</option><option>黑体</option> // 因导出PDF调整到前端输出,字体文件大,支持的字体要谨慎些,不轻易多加字体
  138. domArrs.push("</select>");
  139. domArrs.push("</div>");
  140. //3. font height
  141. domArrs.push("<div class='col-3'>");
  142. domArrs.push("<input class='form-control input-sm' id='fontHeight_" + fontPropSuffix + "' type='number' value='30' step='1' min='6' max='66' " +
  143. "onchange='rptControlObj.changeFontMain(\"" + font.CfgDispName + "\", \"FontHeight\", this)' " +
  144. "onkeyup='rptControlObj.changeFontMain(\"" + font.CfgDispName + "\", \"FontHeight\", this)'>");
  145. domArrs.push("</div>");
  146. //4. font bold italic underline
  147. domArrs.push("<div class='col-3'>");
  148. domArrs.push("<a id='font_bold_" + fontPropSuffix + "' class='btn btn-sm btn-outline-secondary' title='加粗' onclick='rptControlObj.changeFontAdhoc(\"" + font.CfgDispName + "\", \"FontBold\", this)'><i class='fa fa-bold'></i></a>");
  149. domArrs.push("<a id='font_italic_" + fontPropSuffix + "' class='btn btn-sm btn-outline-secondary' title='斜体' onclick='rptControlObj.changeFontAdhoc(\"" + font.CfgDispName + "\", \"FontItalic\", this)'><i class='fa fa-italic'></i></a>");
  150. domArrs.push("<a id='font_underline_" + fontPropSuffix + "' class='btn btn-sm btn-outline-secondary' title='下划线' onclick='rptControlObj.changeFontAdhoc(\"" + font.CfgDispName + "\", \"FontUnderline\", this)'><i class='fa fa-underline'></i></a>");
  151. domArrs.push("</div>");
  152. //
  153. domArrs.push("</div>");
  154. $(domArrs.join("")).insertBefore($("#font_cfg_blank_flag"));
  155. }
  156. },
  157. renderRptCfg: function (cfg) {
  158. this.renderMargin(cfg);
  159. this.renderFormat(cfg);
  160. },
  161. renderMargin: function (cfg) {
  162. $("#elementMargin_Left")[0].value = cfg.margins.Left;
  163. $("#elementMargin_Right")[0].value = cfg.margins.Right;
  164. $("#elementMargin_Top")[0].value = cfg.margins.Top;
  165. $("#elementMargin_Bottom")[0].value = cfg.margins.Bottom;
  166. },
  167. renderFormat: function (cfg) {
  168. for (let font of cfg.fonts) {
  169. let fontPropSuffix = fontSuffixMapObj[font.CfgDispName];
  170. document.getElementById("fontName_" + fontPropSuffix).value = font.Name;
  171. document.getElementById("fontHeight_" + fontPropSuffix).value = font.FontHeight;
  172. document.getElementById("font_bold_" + fontPropSuffix).className = (font.FontBold === "T")?"btn btn-sm btn-outline-secondary active":"btn btn-sm btn-outline-secondary";
  173. document.getElementById("font_italic_" + fontPropSuffix).className = (font.FontItalic === "T")?"btn btn-sm btn-outline-secondary active":"btn btn-sm btn-outline-secondary";
  174. document.getElementById("font_underline_" + fontPropSuffix).className = (font.FontUnderline === "T")?"btn btn-sm btn-outline-secondary active":"btn btn-sm btn-outline-secondary";
  175. }
  176. document.getElementById("cfg_border_thick").value = cfg.borderThick;
  177. document.getElementById("cfg_rpt_vertical_line").checked = cfg.showVerticalLine;
  178. document.getElementById("cfg_rpt_fill_zero").checked = cfg.fillZero;
  179. document.getElementById("cfg_rpt_narrow").checked = cfg.isNarrow;
  180. },
  181. extractRptCfg: function (cfg) {
  182. cfg.margins.Left = $("#elementMargin_Left")[0].value;
  183. cfg.margins.Right = $("#elementMargin_Right")[0].value;
  184. cfg.margins.Top = $("#elementMargin_Top")[0].value;
  185. cfg.margins.Bottom = $("#elementMargin_Bottom")[0].value;
  186. for (let font of cfg.fonts) {
  187. let fontPropSuffix = fontSuffixMapObj[font.CfgDispName];
  188. font.Name = document.getElementById("fontName_" + fontPropSuffix).value;
  189. font.FontHeight = document.getElementById("fontHeight_" + fontPropSuffix).value;
  190. font.FontBold = (document.getElementById("font_bold_" + fontPropSuffix).className === "btn btn-sm btn-outline-secondary active")?"T":"F";
  191. font.FontItalic = (document.getElementById("font_italic_" + fontPropSuffix).className === "btn btn-sm btn-outline-secondary active")?"T":"F";
  192. font.FontUnderline = (document.getElementById("font_underline_" + fontPropSuffix).className === "btn btn-sm btn-outline-secondary active")?"T":"F";
  193. }
  194. cfg.showVerticalLine = document.getElementById("cfg_rpt_vertical_line").checked;
  195. cfg.isNarrow = document.getElementById("cfg_rpt_narrow").checked;
  196. cfg.fillZero = document.getElementById("cfg_rpt_fill_zero").checked;
  197. cfg.borderThick = document.getElementById("cfg_border_thick").value;
  198. },
  199. refreshNodes: function() {
  200. let me = this;
  201. let private_setupIsParent = function(node){
  202. node.isParent = (node.nodeType === RT.NodeType.NODE || node.level === 0);
  203. if (node.items && node.items.length) {
  204. for (let i = 0; i < node.items.length; i++) {
  205. private_setupIsParent(node.items[i]);
  206. }
  207. }
  208. };
  209. let topNodes = me.treeObj.getNodes();
  210. for (let i = 0; i < topNodes.length; i++) {
  211. private_setupIsParent(topNodes[i]);
  212. }
  213. me.treeObj.refresh();
  214. },
  215. onCheck: function(event, treeId, treeNode) {
  216. zTreeOprObj.countChkedRptTpl();
  217. rptCustomObj.showMaterialSelect();
  218. if (treeNode.isParent) {
  219. zTreeOprObj.treeObj.expandNode(treeNode, true, true, false);
  220. }
  221. },
  222. onClick: function(event,treeId,treeNode) {
  223. let me = zTreeOprObj;
  224. if (treeNode && treeNode.nodeType === TPL_TYPE_TEMPLATE && treeNode.refId > 0) {
  225. me.currentNode = treeNode;
  226. let params = {};
  227. let pageSize = rptControlObj.getCurrentPageSize();
  228. params.pageSize = pageSize;
  229. params.rpt_tpl_id = treeNode.refId;
  230. params.project_id = PROJECT_ID;
  231. params.tender_id = TENDER_ID;
  232. params.stage_id = getStageId();
  233. params.stage_status = getStageStatus();
  234. params.stage_order = getStageOrder();
  235. params.stage_times = getStageTimes();
  236. params.material_order = getMaterialOrder();
  237. params.closeWatermark = getCloseWatermark();
  238. params.custCfg = CUST_CFG;
  239. const gather_select = customSelects.gather_select.find(function (x) {
  240. return x.id === treeNode.refId;
  241. });
  242. if (gather_select) {
  243. rptCustomObj.init(gather_select.custom_define, customSelects.stageFlow, gather_select);
  244. return;
  245. }
  246. const stage_select = customSelects.stage_select.find(function (x) {
  247. return x.id === treeNode.refId;
  248. });
  249. if (stage_select) {
  250. rptCustomObj.init(stage_select.custom_define, customSelects.stageFlow, stage_select);
  251. return;
  252. }
  253. me.requestNormalReport(params);
  254. me.countChkedRptTpl();
  255. rptCustomObj.showMaterialSelect();
  256. }
  257. },
  258. changePageSize: function(dom) {
  259. let me = zTreeOprObj,
  260. targetDom = document.getElementById("btnRptPageSize");
  261. let tmpStr = targetDom.innerHTML.trim();
  262. targetDom.innerHTML = dom.innerHTML.trim();
  263. dom.innerHTML = tmpStr;
  264. me.changeCfg();
  265. },
  266. changeOrientation: function(dom) {
  267. let me = zTreeOprObj,
  268. targetDom = document.getElementById("btnRptOrientation");
  269. let tmpStr = targetDom.innerHTML.trim();
  270. targetDom.innerHTML = dom.innerHTML.trim();
  271. dom.innerHTML = tmpStr;
  272. me.changeCfg();
  273. },
  274. changeCfg: function() {
  275. let me = zTreeOprObj;
  276. if (me.currentNode) {
  277. let params = {};
  278. params.pageSize = rptControlObj.getCurrentPageSize();
  279. params.orientation = rptControlObj.getCurrentOrientation();
  280. params.custCfg = CUST_CFG;
  281. params.rpt_tpl_id = me.currentNode.refId;
  282. params.project_id = PROJECT_ID;
  283. params.tender_id = TENDER_ID;
  284. params.stage_id = getStageId();
  285. params.stage_status = getStageStatus();
  286. params.stage_order = getStageOrder();
  287. params.stage_times = getStageTimes();
  288. params.material_order = getMaterialOrder();
  289. // me.requestNormalReport(params);
  290. const gather_select = customSelects.gather_select.find(function (x) {
  291. return x.id === me.currentNode.refId;
  292. });
  293. if (gather_select) {
  294. rptCustomObj.init(gather_select.custom_define, customSelects.stageFlow, gather_select);
  295. return;
  296. }
  297. const stage_select = customSelects.stage_select.find(function (x) {
  298. return x.id === me.currentNode.refId;
  299. });
  300. if (stage_select) {
  301. rptCustomObj.init(stage_select.custom_define, customSelects.stageFlow, stage_select);
  302. return;
  303. }
  304. me.requestNormalReport(params);
  305. me.countChkedRptTpl();
  306. rptCustomObj.showMaterialSelect();
  307. }
  308. },
  309. resetAfter: function (pageRst) {
  310. let size = pageRst[JV.NODE_PAGE_INFO][JV.NODE_PAGE_SIZE].slice(0);
  311. if (size[0] > size[1]) {
  312. document.getElementById("btnRptOrientation").innerHTML = "横向";
  313. document.getElementById("hrefRptOrientation").innerHTML = "纵向";
  314. } else {
  315. document.getElementById("btnRptOrientation").innerHTML = "纵向";
  316. document.getElementById("hrefRptOrientation").innerHTML = "横向";
  317. }
  318. },
  319. _parseRoleRelList: function(org_rel_content) {
  320. const rst = JSON.parse(org_rel_content);
  321. for (const role_rel of rst) {
  322. if (role_rel.sign_date !== null && role_rel.sign_date !== undefined && role_rel.sign_date.length >= 8) {
  323. role_rel.sign_date = new Date(role_rel.sign_date);
  324. }
  325. }
  326. return rst;
  327. },
  328. requestNormalReport: function (params) {
  329. let me = zTreeOprObj;
  330. $.bootstrapLoading.start();
  331. CommonAjax.postXsrfEx("/tender/report_api/getReport", params, 60000, true, getCookie('csrfToken'),
  332. function(result){
  333. $.bootstrapLoading.end();
  334. let pageRst = result.data;
  335. STAGE_AUDIT = result.stageAudit;
  336. STAGE_AUDIT_ORG = result.stageAuditOrg;
  337. STAGE_FLOW = result.stageFlow;
  338. //stageAuditOrg
  339. if (result.signatureRelInfo && result.signatureRelInfo.length > 0) {
  340. CURRENT_ROLE_REL_ID = result.signatureRelInfo[0].id;
  341. ROLE_REL_LIST = me._parseRoleRelList(result.signatureRelInfo[0].rel_content);
  342. rptSignatureHelper.originalRoleRelList = me._parseRoleRelList(result.signatureRelInfo[0].rel_content);
  343. if (current_stage_status === 3) {
  344. rptSignatureHelper.mergeSignDate(pageRst, ROLE_REL_LIST, true);
  345. rptSignatureHelper.mergeSignature(pageRst, ROLE_REL_LIST);
  346. rptSignatureHelper.mergeSignAudit(pageRst, ROLE_REL_LIST, STAGE_AUDIT);
  347. }
  348. } else {
  349. CURRENT_ROLE_REL_ID = -1;
  350. ROLE_REL_LIST = [];
  351. }
  352. // if (ROLE_REL_LIST)
  353. let canvas = zTreeOprObj.canvas;
  354. if (pageRst && pageRst.items && pageRst.items.length > 0) {
  355. me.resetAfter(pageRst);
  356. me.currentRptPageRst = pageRst;
  357. me.maxPages = pageRst.items.length;
  358. me.currentPage = 1;
  359. me.displayPageValue();
  360. let size = JpcCanvasOutput.getReportSizeInPixel(me.currentRptPageRst, getScreenDPI());
  361. canvas.width = size[0] + 20;
  362. if (size[1] > size[0]) {
  363. canvas.height = size[1] + 100;
  364. } else {
  365. canvas.height = size[1] + 50;
  366. }
  367. // zTreeOprObj.resetESignature(zTreeOprObj.currentRptPageRst);
  368. rptSignatureHelper.buildSelectableAccount();
  369. rptSignatureHelper.buildSelectableAccountUsed();
  370. rptSignatureHelper.buildRoleDom(ROLE_LIST);
  371. me.showPage(1, canvas);
  372. } else {
  373. //返回了无数据表
  374. JpcCanvasOutput.cleanCanvas(canvas);
  375. JpcCanvasOutput.drawPageBorder(me.currentRptPageRst, canvas, getScreenDPI());
  376. }
  377. rptCustomObj.init(result.customDefine, result.stageFlow, result.customSelect);
  378. try {
  379. if (is_debug && result.debugInfo) {
  380. console.log('含有key的debug信息:');
  381. for (const k in result.debugInfo.key) {
  382. console.log(k + ':', ...result.debugInfo.key[k]);
  383. }
  384. //console.log(result.debugInfo.key);
  385. console.log('其他debug信息:');
  386. for (const di of result.debugInfo.other) {
  387. console.log(...di);
  388. }
  389. }
  390. } catch(err) {
  391. }
  392. }, function(err){
  393. $.bootstrapLoading.end();
  394. }, function(ex){
  395. $.bootstrapLoading.end();
  396. }
  397. );
  398. },
  399. scaleReport: function (accScale) {
  400. let me = zTreeOprObj;
  401. let canvas = zTreeOprObj.canvas;
  402. if (accScale !== 0) {
  403. JpcCanvasOutput.scaleFactor += accScale;
  404. if (JpcCanvasOutput.scaleFactor < 0.5) JpcCanvasOutput.scaleFactor = 0.5;
  405. if (JpcCanvasOutput.scaleFactor > 1.5) JpcCanvasOutput.scaleFactor = 1.5;
  406. } else {
  407. JpcCanvasOutput.scaleFactor = 1;
  408. }
  409. document.getElementById("btnNormalScale").innerText = (JpcCanvasOutput.scaleFactor * 100) + '%';
  410. me.showPage(me.currentPage, canvas);
  411. },
  412. requestPrjFolderCommon: function () {
  413. //
  414. },
  415. showPage: function (pageNum, canvas) {
  416. let me = zTreeOprObj;
  417. if (pageNum >= 1 && pageNum <= me.maxPages) {
  418. me.currentPage = pageNum;
  419. JpcCanvasOutput.cleanCanvas(canvas);
  420. JpcCanvasOutput.drawPageBorder(me.currentRptPageRst, canvas, getScreenDPI());
  421. JpcCanvasOutput.drawToCanvas(me.currentRptPageRst, canvas, me.currentPage);
  422. }
  423. me.displayPageValue();
  424. },
  425. displayPageValue: function() {
  426. let me = zTreeOprObj;
  427. $("#rpt_page_num")[0].value = me.currentPage + "/" + me.maxPages;
  428. }
  429. };
  430. let canvasOprObj = {
  431. canvasOnMouseMove: function (event) {
  432. if (zTreeOprObj.currentNode) {
  433. let x = event.offsetX - JpcCanvasOutput.offsetX, canvas = event.originalTarget;
  434. if (!(canvas)) canvas = event.target; //chrome浏览器不认event.originalTarget,只认event.target或event.currentTarget
  435. if (x < FIRST_PAGE_OFFSET) {
  436. canvas.style.cursor = "url(/web/building_saas/img/FirstPageSimple.cur), auto";
  437. } else if (x < PRE_PAGE_OFFSET) {
  438. canvas.style.cursor = "url(/web/building_saas/img/PreviousPageSimple.cur), auto";
  439. } else if ((canvas.width - x) < LAST_PAGE_OFFSET) {
  440. canvas.style.cursor = "url(/web/building_saas/img/LastPageSimple.cur), auto";
  441. } else if ((canvas.width - x) < NEXT_PAGE_OFFSET) {
  442. canvas.style.cursor = "url(/web/building_saas/img/NextPageSimple.cur), auto";
  443. } else {
  444. canvas.style.cursor = "";
  445. }
  446. }
  447. },
  448. canvasOnClick: function(event){
  449. if (zTreeOprObj.currentNode) {
  450. let x = event.offsetX - JpcCanvasOutput.offsetX, canvas = event.originalTarget;
  451. if (!(canvas)) canvas = event.target; //chrome浏览器不认event.originalTarget,只认event.target或event.currentTarget
  452. if (x < FIRST_PAGE_OFFSET) {
  453. zTreeOprObj.showPage(1, canvas);
  454. } else if (x < PRE_PAGE_OFFSET) {
  455. zTreeOprObj.showPage(zTreeOprObj.currentPage - 1, canvas);
  456. } else if ((canvas.width - x) < LAST_PAGE_OFFSET) {
  457. zTreeOprObj.showPage(zTreeOprObj.maxPages, canvas);
  458. } else if ((canvas.width - x) < NEXT_PAGE_OFFSET) {
  459. zTreeOprObj.showPage(zTreeOprObj.currentPage + 1, canvas);
  460. }
  461. }
  462. }
  463. };
  464. let rptControlObj = {
  465. currentOutputType: "Excel",
  466. currentDownloadUrl: null,
  467. currentDownloadIdx: 0,
  468. isLoading: false,
  469. getCurrentPageSize: function() {
  470. // let rst = "A4";
  471. let rst = document.getElementById("btnRptPageSize").innerHTML.trim();
  472. //btnRptPageSize
  473. return rst;
  474. },
  475. getCurrentOrientation: function() {
  476. // let rst = "横向";
  477. let rst = document.getElementById("btnRptOrientation").innerHTML.trim();
  478. return rst;
  479. },
  480. changeType: function(newType) {
  481. let me = rptControlObj;
  482. let excelDom = document.getElementById("EXCEL_TYPE");
  483. let pdfDom = document.getElementById("PDF_TYPE");
  484. if (newType === "Excel") {
  485. excelDom.className = "btn btn-block btn-primary";
  486. pdfDom.className = "btn btn-block btn-outline-secondary";
  487. me.currentOutputType = newType;
  488. } else if (newType === "PDF") {
  489. excelDom.className = "btn btn-block btn-outline-secondary";
  490. pdfDom.className = "btn btn-block btn-primary";
  491. me.currentOutputType = newType;
  492. } else {
  493. //me.currentOutputType = newType;
  494. }
  495. },
  496. getTplIdsCommon: function (refRptTplIds, rpt_names) {
  497. for (let node of zTreeOprObj.checkedRptTplNodes) {
  498. if (node.hasOwnProperty('flags') && node.flags.hasOwnProperty('reportType') && node['flags']['reportType'] !== 'NA') {
  499. // 未来可能会有这些处理,目前空着
  500. } else {
  501. refRptTplIds.push(node.refId);
  502. if (rpt_names) rpt_names.push(node.name);
  503. }
  504. }
  505. },
  506. creatCommonExportParam: function (refRptTplIds) {
  507. let rst = {};
  508. rst.rpt_ids = refRptTplIds;
  509. rst.pageSize = rptControlObj.getCurrentPageSize();
  510. rst.orientation = ((zTreeOprObj.checkedRptTplNodes.length > 1)?null:rptControlObj.getCurrentOrientation());
  511. rst.project_id = PROJECT_ID;
  512. rst.tender_id = TENDER_ID;
  513. rst.stage_id = getStageId();
  514. rst.stage_status = getStageStatus();
  515. rst.stage_order = getStageOrder();
  516. rst.stage_times = getStageTimes();
  517. rst.material_order = getMaterialOrder();
  518. rst.custCfg = CUST_CFG;
  519. rst.closeWatermark = getCloseWatermark();
  520. return rst;
  521. },
  522. getAllInOneBook: async function () {
  523. if (zTreeOprObj.checkedRptTplNodes && zTreeOprObj.checkedRptTplNodes.length > 0) {
  524. let me = rptControlObj;
  525. let refRptTplIds = [], rpt_sheet_names = [];
  526. rptControlObj.getTplIdsCommon(refRptTplIds, rpt_sheet_names);
  527. let params = rptControlObj.creatCommonExportParam(refRptTplIds);
  528. await rptCustomObj.getCustomSelect(params);
  529. params.rpt_names = rpt_sheet_names;
  530. params.rptName = TENDER_NAME;
  531. let chkNodes = zTreeOprObj.treeObj.getCheckedNodes(true);
  532. if (chkNodes.length > 0) {
  533. delete params.orientation; // 打印时有勾选的话,不需要提供方向
  534. }
  535. CommonAjax.postXsrfEx("/tender/report_api/createExcelFilesInOneBook", params, WAIT_TIME_EXPORT, true, getCookie('csrfToken'), function(result){
  536. if (result) {
  537. let uuIdUrls = [];
  538. let uuIdUrl = "/getFileByUUID/" + result.data[0].uuid + "/" + stringUtil.replaceAll(result.data[0].reportName, "#", "_") + "/xlsx";
  539. uuIdUrls.push(uuIdUrl);
  540. downloadReport(uuIdUrls);
  541. } else {
  542. //
  543. }
  544. }, null, null
  545. );
  546. }
  547. },
  548. getAllIndividualExcelBook: async function () {
  549. let me = rptControlObj;
  550. if (zTreeOprObj.checkedRptTplNodes && zTreeOprObj.checkedRptTplNodes.length > 0) {
  551. let refRptTplIds = [];
  552. let rpt_names = [];
  553. rptControlObj.getTplIdsCommon(refRptTplIds, rpt_names);
  554. let params = rptControlObj.creatCommonExportParam(refRptTplIds);
  555. await rptCustomObj.getCustomSelect(params);
  556. params.isOneSheet = true;
  557. params.rpt_names = rpt_names;
  558. params.rptName = 'All';
  559. let chkNodes = zTreeOprObj.treeObj.getCheckedNodes(true);
  560. if (chkNodes.length > 0) {
  561. delete params.orientation; // 打印时有勾选的话,不需要提供方向
  562. }
  563. CommonAjax.postXsrfEx("/tender/report_api/createExcelFiles", params, WAIT_TIME_EXPORT, true, getCookie('csrfToken'), function(result){
  564. if (result) {
  565. let uuIdUrls = [];
  566. for (let uuIdObj of result.data) {
  567. let uuIdUrl = "/getFileByUUID/" + uuIdObj.uuid + "/" + stringUtil.replaceAll(uuIdObj.reportName, "#", "_") + "/xlsx";
  568. uuIdUrls.push(uuIdUrl);
  569. }
  570. downloadReport(uuIdUrls);
  571. } else {
  572. //
  573. }
  574. }, null, null
  575. );
  576. }
  577. },
  578. checkAndGetExcel: function () {
  579. if (zTreeOprObj.treeObj) {
  580. let chkNodes = zTreeOprObj.treeObj.getCheckedNodes(true);
  581. if (chkNodes.length > 0) {
  582. $("#show_excel_output_cfg").trigger("click");
  583. } else {
  584. rptControlObj.getAllIndividualExcelBook();
  585. }
  586. }
  587. },
  588. getExcel: function () {
  589. let me = rptControlObj;
  590. if ($("#excelExportType_AllInOneBook")[0].checked) {
  591. me.getAllInOneBook();
  592. } else if ($("#excelExportType_IndividualBook")[0].checked) {
  593. me.getAllIndividualExcelBook();
  594. }
  595. },
  596. getPdfFontCallbackLight: function(fontProperty) {
  597. rptTplObj.pdfFont['SmartSimsun'].push(fontProperty);
  598. if (rptTplObj.pdfFont['SmartSimsun'].length === 2) {
  599. rptControlObj.isLoading = false;
  600. }
  601. },
  602. getPdfFontCallback: function(fontProperty) {
  603. let me = rptControlObj;
  604. if (rptTplObj.pdfFont['SmartSimsun'].indexOf(fontProperty) < 0) {
  605. rptTplObj.pdfFont['SmartSimsun'].push(fontProperty);
  606. }
  607. if (rptTplObj.pdfFont['SmartSimsun'].length === 2) {
  608. $.bootstrapLoading.end();
  609. me.getPDFEx();
  610. }
  611. },
  612. loadPDFFonts: function () {
  613. //这里尝试下异步加载字体文件
  614. let me = rptControlObj;
  615. me.isLoading = true;
  616. // dynamicLoadJs('/public/jspdf/SmartSimsun-normal.js', 'normal', me.getPdfFontCallbackLight);
  617. // dynamicLoadJs('/public/jspdf/SmartSimsun-bold.js', 'bold', me.getPdfFontCallbackLight);
  618. dynamicLoadJs('https://d2.smartcost.com.cn/cach/SmartSimsun-normal.js', 'normal', me.getPdfFontCallbackLight);
  619. dynamicLoadJs('https://d2.smartcost.com.cn/cach/SmartSimsun-bold.js', 'bold', me.getPdfFontCallbackLight);
  620. },
  621. getPDFPre: function () {
  622. let me = rptControlObj;
  623. if (me.isLoading) {
  624. $.bootstrapLoading.start();
  625. console.log('fonts are loading...');
  626. setTimeout(me.getPDFPre, 3000); //延时3秒
  627. } else {
  628. $.bootstrapLoading.end(); //根据测试,需要先做一次清理,防止在极端情况下出现阻塞
  629. if (rptTplObj.pdfFont['SmartSimsun'].length === 2) {
  630. me.getPDFEx();
  631. } else {
  632. $.bootstrapLoading.start();
  633. // dynamicLoadJs('/public/jspdf/SmartSimsun-normal.js',"normal", me.getPdfFontCallback);
  634. // dynamicLoadJs('/public/jspdf/SmartSimsun-bold.js',"bold", me.getPdfFontCallback);
  635. dynamicLoadJs('https://d2.smartcost.com.cn/cach/SmartSimsun-normal.js', 'normal', me.getPdfFontCallback);
  636. dynamicLoadJs('https://d2.smartcost.com.cn/cach/SmartSimsun-bold.js', 'bold', me.getPdfFontCallback);
  637. }
  638. }
  639. },
  640. getPDFEx: async function () {
  641. let me = rptControlObj;
  642. if (zTreeOprObj.checkedRptTplNodes && zTreeOprObj.checkedRptTplNodes.length > 0 && PAGE_SHOW['closeExportPdf'] !== 1) {
  643. let refRptTplIds = [];
  644. let rpt_names = [];
  645. rptControlObj.getTplIdsCommon(refRptTplIds, rpt_names);
  646. const signatureRelArr = [];
  647. if (refRptTplIds.length > 0) {
  648. let params = rptControlObj.creatCommonExportParam(refRptTplIds);
  649. await rptCustomObj.getCustomSelect(params);
  650. delete params.orientation; // 打印时有勾选的话,不需要提供方向
  651. $.bootstrapLoading.start();
  652. CommonAjax.postXsrfEx("/tender/report_api/getMultiReports", params, WAIT_TIME_EXPORT, true, getCookie('csrfToken'),
  653. function(result){
  654. // closeWaitingView();
  655. $.bootstrapLoading.end();
  656. STAGE_AUDIT = result.stageAudit;
  657. let pageSize = rptControlObj.getCurrentPageSize();
  658. for (const signatureRel of result.signatureRelInfo) {
  659. signatureRelArr.push(JSON.parse(signatureRel.rel_content));
  660. }
  661. downloadPDFReport(result.data, pageSize, rpt_names, signatureRelArr, result.signatureRelInfo, refRptTplIds, STAGE_AUDIT);
  662. /*
  663. for (let idx = 0; idx < result.data.length; idx++) {
  664. let pageData = result.data[idx];
  665. // if (current_stage_status === 3) {
  666. // rptSignatureHelper.mergeSignDate(pageData);
  667. // rptSignatureHelper.mergeSignature(pageData);
  668. // rptSignatureHelper.mergeSignAudit(pageRst, ROLE_REL_LIST, STAGE_AUDIT);
  669. // }
  670. // 备注:因多表的原因,无需merge电子签名,在下面处理
  671. let singleSignatureRelArr = [];
  672. for (let rIdx = 0; rIdx < result.signatureRelInfo.length; rIdx++) {
  673. if (result.signatureRelInfo[rIdx].rpt_id === refRptTplIds[idx]) {
  674. // singleSignatureRelArr.push(signatureRelArr[rIdx]);
  675. singleSignatureRelArr = signatureRelArr[rIdx]; // 有些报表可能没有签名
  676. break;
  677. }
  678. }
  679. // JpcJsPDFHelper.outputAsPdf(pageData, pageSize, rpt_names[idx], signatureRelArr);
  680. JpcJsPDFHelper.outputAsPdf(pageData, pageSize, rpt_names[idx], singleSignatureRelArr, STAGE_AUDIT); // 精确控制签名
  681. }
  682. //*/
  683. },
  684. function(failRst){
  685. // closeWaitingView();
  686. $.bootstrapLoading.end();
  687. console.log(failRst);
  688. },
  689. function(exceptionRst){
  690. // closeWaitingView();
  691. $.bootstrapLoading.end();
  692. console.log(exceptionRst);
  693. }
  694. );
  695. } else {
  696. //这个分支是为了减少请求,用户已经点过的表,又没有勾选,那么就直接导出成PDF
  697. let pageSize = rptControlObj.getCurrentPageSize();
  698. let pageData = zTreeOprObj.currentRptPageRst;
  699. signatureRelArr.push(ROLE_REL_LIST);
  700. // closeWaitingView();
  701. $.bootstrapLoading.end();
  702. JpcJsPDFHelper.outputAsPdf(pageData, pageSize, rpt_names[0], signatureRelArr);
  703. }
  704. }
  705. },
  706. firstPage: function(dom) {
  707. zTreeOprObj.showPage(1, zTreeOprObj.canvas);
  708. },
  709. prePage: function(dom) {
  710. zTreeOprObj.showPage(zTreeOprObj.currentPage - 1, zTreeOprObj.canvas);
  711. },
  712. nextPage: function(dom) {
  713. zTreeOprObj.showPage(zTreeOprObj.currentPage + 1, zTreeOprObj.canvas);
  714. },
  715. lastPage: function(dom) {
  716. let me = zTreeOprObj;
  717. zTreeOprObj.showPage(me.maxPages, zTreeOprObj.canvas);
  718. },
  719. onKeydown: function (event, dom) {
  720. let me = zTreeOprObj, keyPressed = null;
  721. if (window.event) {
  722. keyPressed = window.event.keyCode; // IE/Chrome
  723. } else {
  724. keyPressed = event.which; // Firefox
  725. }
  726. if (keyPressed === 13) {
  727. let pageNum = 1;
  728. try {
  729. pageNum = parseInt(dom.value);
  730. } catch (e) {
  731. pageNum = 1;
  732. }
  733. if (pageNum < 1) {
  734. pageNum = 1;
  735. } else if (pageNum > me.maxPages) {
  736. pageNum = me.maxPages;
  737. }
  738. zTreeOprObj.showPage(pageNum, zTreeOprObj.canvas);
  739. return false;
  740. }
  741. },
  742. changeMargin: function(marginPropStr, marginDom) {
  743. CUST_CFG.margins[marginPropStr] = marginDom.value;
  744. },
  745. changeFontMain: function(CfgDispName, fontProperty, fontDom) {
  746. for (let font of CUST_CFG.fonts) {
  747. if (font["CfgDispName"] === CfgDispName) {
  748. font[fontProperty] = fontDom.value;
  749. break;
  750. }
  751. }
  752. },
  753. changeBorderWeight: function (dom) {
  754. CUST_CFG.borderThick = dom.value;
  755. },
  756. changeCfgOption: function (optStr, dom) {
  757. CUST_CFG[optStr] = dom.checked;
  758. },
  759. changeFontAdhoc: function(CfgDispName, fontProperty, fontDom) {
  760. for (let font of CUST_CFG.fonts) {
  761. if (font["CfgDispName"] === CfgDispName) {
  762. if (font[fontProperty] === 'T') {
  763. font[fontProperty] = 'F';
  764. fontDom.className = "btn btn-sm btn-outline-secondary";
  765. } else {
  766. font[fontProperty] = 'T';
  767. fontDom.className = "btn btn-sm btn-outline-secondary active";
  768. }
  769. break;
  770. }
  771. }
  772. },
  773. restoreMargine: function () {
  774. zTreeOprObj.renderMargin(zTreeOprObj.defReportPageCfg);
  775. zTreeOprObj.extractRptCfg(CUST_CFG);
  776. },
  777. restoreFormat: function () {
  778. zTreeOprObj.renderFormat(zTreeOprObj.defReportPageCfg);
  779. zTreeOprObj.extractRptCfg(CUST_CFG);
  780. },
  781. restoreCustCFG: function () {
  782. let me = this;
  783. zTreeOprObj.renderRptCfg(zTreeOprObj.defReportPageCfg);
  784. zTreeOprObj.extractRptCfg(CUST_CFG);
  785. // me.saveCustCfg();
  786. },
  787. saveCustCfg: function() {
  788. let params = {};
  789. params.custCfg = CUST_CFG;
  790. CommonAjax.postEx("report_tpl_api/saveCustomerCfg", params, 20000, true, function(result){
  791. // alert("Save successfully!");
  792. $("#update_msg_response")[0].style.color = "green";
  793. $("#update_msg_response")[0].innerHTML = " (保存成功!)";
  794. setTimeout(function(){
  795. $("#update_msg_response")[0].innerHTML = "";
  796. }, 1000);
  797. }, function (failRst) {
  798. $("#update_msg_response")[0].style.color = "red";
  799. $("#update_msg_response")[0].innerHTML = " (保存失败!)";
  800. setTimeout(function(){
  801. $("#update_msg_response")[0].innerHTML = "";
  802. }, 1000);
  803. }, null
  804. );
  805. },
  806. confirmCfgChange: function() {
  807. zTreeOprObj.changeCfg();
  808. }
  809. };
  810. function downloadPDFReport(pageDataArr, pageSize, rpt_names, signatureRelArr, signatureRelInfo, refRptTplIds, STAGE_AUDIT) {
  811. rptControlObj.currentDownloadIdx = 0;
  812. const private_download = function() {
  813. if (rptControlObj.currentDownloadIdx < pageDataArr.length) {
  814. let singleSignatureRelArr = [];
  815. for (let rIdx = 0; rIdx < signatureRelInfo.length; rIdx++) {
  816. // for (const rptId of refRptTplIds) {
  817. // if (signatureRelInfo[rIdx].rpt_id === rptId) {
  818. // singleSignatureRelArr = signatureRelArr[rIdx]; // 有些报表可能没有签名
  819. // break;
  820. // }
  821. // }
  822. let rptId = refRptTplIds[rptControlObj.currentDownloadIdx];
  823. if (signatureRelInfo[rIdx].rpt_id === rptId) {
  824. singleSignatureRelArr = signatureRelArr[rIdx]; // 有些报表可能没有签名
  825. break;
  826. }
  827. // if (signatureRelInfo[rIdx].rpt_id === refRptTplIds[idx]) {
  828. // singleSignatureRelArr = signatureRelArr[rIdx]; // 有些报表可能没有签名
  829. // break;
  830. // }
  831. }
  832. let pageData = pageDataArr[rptControlObj.currentDownloadIdx];
  833. let rptName = rpt_names[rptControlObj.currentDownloadIdx];
  834. rptControlObj.currentDownloadIdx++;
  835. JpcJsPDFHelper.outputAsPdf(pageData, pageSize, rptName, singleSignatureRelArr, STAGE_AUDIT); // 精确控制签名
  836. if (rptControlObj.currentDownloadIdx < pageDataArr.length) setTimeout(private_download, 2000);
  837. }
  838. }
  839. private_download();
  840. }
  841. function downloadReport(urls) {
  842. //考虑到多个报表下载,一些浏览器(如chrome)不允许一下子下载多个文件,得缓缓处理,统一在这处理
  843. rptControlObj.currentDownloadUrl = null;
  844. rptControlObj.currentDownloadIdx = 0;
  845. const private_download = function() {
  846. if (rptControlObj.currentDownloadIdx >= 0 && rptControlObj.currentDownloadIdx < urls.length) {
  847. rptControlObj.currentDownloadUrl = urls[rptControlObj.currentDownloadIdx];
  848. rptControlObj.currentDownloadIdx++;
  849. window.location = rptControlObj.currentDownloadUrl;
  850. if (rptControlObj.currentDownloadIdx < urls.length) setTimeout(private_download, 2000);
  851. }
  852. }
  853. private_download();
  854. }
  855. function dynamicLoadJs(url, type, callback) {
  856. let head = document.getElementsByTagName('head')[0];
  857. let script = document.createElement('script');
  858. script.type = 'text/javascript';
  859. script.src = url;
  860. if(callback) {
  861. script.onload = script.onreadystatechange = function (event) {
  862. callback(type);
  863. script.onload = script.onreadystatechange = null;
  864. };
  865. }
  866. head.appendChild(script);
  867. }
  868. function getStageStatus() {
  869. return current_stage_status;
  870. }
  871. function getStageId() {
  872. return current_stage_id;
  873. }
  874. function getStageOrder() {
  875. return CUR_ORDER;
  876. }
  877. function getMaterialOrder() {
  878. return parseInt($('#material-select').attr('m-order'));
  879. }
  880. function getStageTimes() {
  881. return current_stage_times;
  882. }
  883. function getCloseWatermark() {
  884. return PAGE_SHOW['closeWatermark'];
  885. }