rpt_construct_data_util.js 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498
  1. /**
  2. * Created by Tony on 2017/7/14.
  3. * 报表数据提取class,是协助报表模板里指标字段自主提取数据的工具类
  4. */
  5. let JV = require('../rpt_component/jpc_value_define');
  6. let $JE = require('../rpt_component/jpc_rte');
  7. let consts = require('../../../modules/main/models/project_consts');
  8. let fsUtil = require("../../../public/fsUtil");
  9. let treeUtil = require('../../../public/treeUtil');
  10. let projectConst = consts.projectConst;
  11. let projectConstList = consts.projectConstList;
  12. class Rpt_Common{
  13. initialize(rpt_tpl, currentDataObj) {
  14. this.template = rpt_tpl;
  15. this.currentDataObj = currentDataObj;
  16. };
  17. Multiply(val1, val2, fixFormat) {
  18. let rst = [], maxLen = val1.length, minLen = val2.length;
  19. if (minLen > maxLen) {
  20. maxLen = maxLen + minLen; minLen = maxLen - minLen; maxLen = maxLen - minLen;
  21. }
  22. for (let i = 0; i < maxLen; i++) {
  23. let value = ((i < val1.length)?val1[i]:val1[minLen - 1]) * ((i < val2.length)?val2[i]:val2[minLen - 1]);
  24. if (fixFormat) value = value.toFixed(fixFormat);
  25. rst.push(value);
  26. }
  27. return rst;
  28. };
  29. Divide(val1, val2, fixFormat) {
  30. let rst = [], maxLen = val1.length, minLen = val2.length;
  31. if (minLen > maxLen) {
  32. maxLen = maxLen + minLen; minLen = maxLen - minLen; maxLen = maxLen - minLen;
  33. }
  34. for (let i = 0; i < maxLen; i++) {
  35. let value = ((i < val1.length)?val1[i]:val1[minLen - 1]) / ((i < val2.length)?val2[i]:val2[minLen - 1]);
  36. if (fixFormat) value = value.toFixed(fixFormat);
  37. rst.push(value);
  38. }
  39. return rst;
  40. };
  41. Plus(val1, val2, fixFormat) {
  42. let rst = [], maxLen = val1.length, minLen = val2.length;
  43. if (minLen > maxLen) {
  44. maxLen = maxLen + minLen; minLen = maxLen - minLen; maxLen = maxLen - minLen;
  45. }
  46. for (let i = 0; i < maxLen; i++) {
  47. let value = ((i < val1.length)?val1[i]:val1[minLen - 1]) + ((i < val2.length)?val2[i]:val2[minLen - 1]);
  48. if (fixFormat) value = value.toFixed(fixFormat);
  49. rst.push(value);
  50. }
  51. return rst;
  52. };
  53. Minus(val1, val2, fixFormat) {
  54. let rst = [], maxLen = val1.length, minLen = val2.length;
  55. if (minLen > maxLen) {
  56. maxLen = maxLen + minLen; minLen = maxLen - minLen; maxLen = maxLen - minLen;
  57. }
  58. for (let i = 0; i < maxLen; i++) {
  59. let value = ((i < val1.length)?val1[i]:val1[minLen - 1]) - ((i < val2.length)?val2[i]:val2[minLen - 1]);
  60. if (fixFormat) value = value.toFixed(fixFormat);
  61. rst.push(value);
  62. }
  63. return rst;
  64. };
  65. }
  66. class Rpt_Data_Extractor {
  67. constructor () {
  68. this.COMMON = new Rpt_Common();
  69. };
  70. initialize(tpl) {
  71. this.rptTpl = tpl;
  72. };
  73. //-- 根据报表模板映射指标(非离散指标)的定义,罗列出所有需要用到的data对象key,作为数据请求的过滤依据
  74. getDataRequestFilter() {
  75. let rst = [];
  76. let tpl = this.rptTpl;
  77. let pri_func_chk_filter = function (field) {
  78. for (let key of projectConstList) {
  79. if (rst.indexOf(key) < 0) {
  80. if (field[JV.PROP_FIELD_EXP_MAP]) {
  81. if (field[JV.PROP_FIELD_EXP_MAP].indexOf('.' + key + '.') >= 0) {
  82. rst.push(key);
  83. if (key === projectConst.RATION_GLJ && (rst.indexOf(projectConst.PROJECTGLJ) < 0)) {
  84. rst.push(projectConst.PROJECTGLJ);
  85. }
  86. if (key === projectConst.PROJECTGLJ && (rst.indexOf(projectConst.RATION_GLJ) < 0)) {
  87. rst.push(projectConst.RATION_GLJ);
  88. }
  89. }
  90. }
  91. }
  92. }
  93. };
  94. let pri_setup_filter = function (FIELD_LIST_KEY) {
  95. if (tpl[JV.NODE_FIELD_MAP][FIELD_LIST_KEY]) {
  96. for (let field of tpl[JV.NODE_FIELD_MAP][FIELD_LIST_KEY]) {
  97. pri_func_chk_filter(field);
  98. }
  99. }
  100. };
  101. pri_setup_filter(JV.NODE_DISCRETE_FIELDS);
  102. pri_setup_filter(JV.NODE_MASTER_FIELDS);
  103. pri_setup_filter(JV.NODE_DETAIL_FIELDS);
  104. pri_setup_filter(JV.NODE_MASTER_FIELDS_EX);
  105. pri_setup_filter(JV.NODE_DETAIL_FIELDS_EX);
  106. return rst;
  107. };
  108. //--- 装配数据(把收集到的数据,依据报表模板的指示,预处理(如:排序、过滤、合计)及装配到相关指标) ---//
  109. assembleData(rawDataObj) {
  110. let $PROJECT = {"COMMON": null, "MAIN": {}, "DETAIL": {}};
  111. let tpl = this.rptTpl;
  112. this.COMMON.initialize(tpl, rawDataObj);
  113. $PROJECT.COMMON = this.COMMON;
  114. $PROJECT.MAIN["myOwnRawDataObj"] = rawDataObj.prj._doc;
  115. $PROJECT.MAIN.getProperty = ext_mainGetPropety;
  116. $PROJECT.MAIN.getFee = ext_mainGetFee;
  117. if (tpl[JV.NODE_MAP_DATA_HANDLE_INFO]) {
  118. for (let preHandle of tpl[JV.NODE_MAP_DATA_HANDLE_INFO]) {
  119. let srcData = getModuleDataByKey(rawDataObj.prjData, preHandle[JV.PROP_DATA_KEY]);
  120. switch(preHandle[JV.PROP_HANDLE_TYPE]) {
  121. case JV.PROP_HANDLE_TYPE_SORT:
  122. sortData(srcData, preHandle);
  123. break;
  124. case JV.PROP_HANDLE_TYPE_FILTER:
  125. filterData(srcData, preHandle, rawDataObj.prjData);
  126. break;
  127. case JV.PROP_HANDLE_TYPE_SUM:
  128. summaryData(srcData, preHandle, rawDataObj.prjData);
  129. break;
  130. default:
  131. break;
  132. }
  133. }
  134. }
  135. for (let item of rawDataObj.prjData) {
  136. setupFunc($PROJECT.DETAIL, item.moduleName, item);
  137. }
  138. let rptDataObj = {};
  139. rptDataObj[JV.DATA_DISCRETE_DATA] = [];
  140. rptDataObj[JV.DATA_MASTER_DATA] = [];
  141. rptDataObj[JV.DATA_DETAIL_DATA] = [];
  142. rptDataObj[JV.DATA_MASTER_DATA_EX] = [];
  143. rptDataObj[JV.DATA_DETAIL_DATA_EX] = [];
  144. assembleFields(tpl[JV.NODE_FIELD_MAP][JV.NODE_DISCRETE_FIELDS], rptDataObj[JV.DATA_DISCRETE_DATA], $PROJECT);
  145. // console.log(JV.DATA_DISCRETE_DATA);
  146. // console.log(rptDataObj[JV.DATA_DISCRETE_DATA]);
  147. assembleFields(tpl[JV.NODE_FIELD_MAP][JV.NODE_MASTER_FIELDS], rptDataObj[JV.DATA_MASTER_DATA], $PROJECT);
  148. // console.log(JV.DATA_MASTER_DATA);
  149. // console.log(rptDataObj[JV.DATA_MASTER_DATA]);
  150. assembleFields(tpl[JV.NODE_FIELD_MAP][JV.NODE_DETAIL_FIELDS], rptDataObj[JV.DATA_DETAIL_DATA], $PROJECT);
  151. // console.log(JV.DATA_DETAIL_DATA);
  152. // console.log(rptDataObj[JV.DATA_DETAIL_DATA]);
  153. assembleFields(tpl[JV.NODE_FIELD_MAP][JV.NODE_MASTER_FIELDS_EX], rptDataObj[JV.DATA_MASTER_DATA_EX], $PROJECT);
  154. // console.log(JV.DATA_MASTER_DATA_EX);
  155. // console.log(rptDataObj[JV.DATA_MASTER_DATA_EX]);
  156. assembleFields(tpl[JV.NODE_FIELD_MAP][JV.NODE_DETAIL_FIELDS_EX], rptDataObj[JV.DATA_DETAIL_DATA_EX], $PROJECT);
  157. console.log(JV.DATA_DETAIL_DATA_EX);
  158. console.log(rptDataObj[JV.DATA_DETAIL_DATA_EX]);
  159. return rptDataObj;
  160. };
  161. }
  162. function getModuleDataByKey(prjData, key) {
  163. let rst = null;
  164. for (let item of prjData) {
  165. if (item.moduleName === key) {
  166. rst = item;
  167. break;
  168. }
  169. }
  170. return rst;
  171. }
  172. function summaryData(sourceData, handleCfg, prjData){
  173. let rstArr = [], tempRstArr = [];
  174. let curParentPrjData = {};
  175. for (let item of sourceData.data) {
  176. if (item._doc) {
  177. tempRstArr.push(item._doc);
  178. } else {
  179. tempRstArr.push(item);
  180. }
  181. }
  182. let private_get_grp_key = function (item) {
  183. let keys = [];
  184. for (let cfg of handleCfg[JV.PROP_SUM_GROUP_KEYS]) {
  185. if (typeof cfg === "string") {
  186. keys.push(item[cfg]);
  187. } else {
  188. if (!curParentPrjData[cfg["seeking_parent"]]) curParentPrjData[cfg["seeking_parent"]] = getModuleDataByKey(prjData, cfg["seeking_parent"]);
  189. for (let pDataItem of curParentPrjData[cfg["seeking_parent"]].data) {
  190. let data = (pDataItem._doc)?pDataItem._doc:pDataItem;
  191. if (item[cfg["seeking_key"]] === data[cfg["parent_key"]]) {
  192. keys.push(data[cfg["parent_grp_key"]]);
  193. break;
  194. }
  195. }
  196. }
  197. }
  198. return ( "grp_key_" + keys.join('_'));
  199. }
  200. let sumObj = {};
  201. for (let dtl of tempRstArr) {
  202. let grpKey = private_get_grp_key(dtl);
  203. if (sumObj[grpKey] === null || sumObj[grpKey] === undefined) {
  204. sumObj[grpKey] = dtl;
  205. rstArr.push(dtl);
  206. } else {
  207. for (let sumKey of handleCfg[JV.PROP_SUM_SUM_KEYS]) {
  208. if (dtl[sumKey]) {
  209. sumObj[grpKey][sumKey] += dtl[sumKey];
  210. }
  211. }
  212. }
  213. }
  214. delete sourceData.data;
  215. sourceData.data = rstArr;
  216. // fsUtil.wirteObjToFile(sourceData.data, "D:/GitHome/ConstructionCost/tmp/sumRst.js");
  217. }
  218. function filterData(sourceData, handleCfg, prjData) {
  219. let rstArr = [], tempRstArr = [];
  220. for (let item of sourceData.data) {
  221. if (item._doc) {
  222. tempRstArr.push(item._doc);
  223. } else {
  224. tempRstArr.push(item);
  225. }
  226. }
  227. let private_chkVal = function (src, dest, compStr) {
  228. let rst = true;
  229. switch (compStr) {
  230. case "==" :
  231. rst = (src == dest);
  232. break;
  233. case "===" :
  234. rst = (src === dest);
  235. break;
  236. case ">" :
  237. rst = (src > dest);
  238. break;
  239. case ">=" :
  240. rst = (src >= dest);
  241. break;
  242. case "<" :
  243. rst = (src < dest);
  244. break;
  245. case "<=" :
  246. rst = (src <= dest);
  247. break;
  248. case "!=" :
  249. rst = (src != dest);
  250. break;
  251. case "!==" :
  252. rst = (src !== dest);
  253. break;
  254. default:
  255. rst = true;
  256. }
  257. return rst;
  258. }
  259. for (let item of tempRstArr) {
  260. let compRst = true;
  261. let curComparePrjData = null;
  262. for (let cfg of handleCfg[JV.PROP_FILTER_KEY]) {
  263. if (cfg[JV.PROP_FILTER_COMPARE_VAL]) {
  264. //比较key值
  265. compRst = private_chkVal(item[cfg.key], cfg[JV.PROP_FILTER_COMPARE_VAL], cfg[JV.PROP_FILTER_CONDITION]);
  266. } else if (cfg[JV.PROP_FILTER_COMPARE_OBJ] && cfg[JV.PROP_FILTER_COMPARE_OBJ_KEY]){
  267. //通过其他对象来过滤
  268. if (!curComparePrjData) {
  269. curComparePrjData = getModuleDataByKey(prjData, cfg[JV.PROP_FILTER_COMPARE_OBJ]);
  270. }
  271. for (let data of curComparePrjData.data) {
  272. compRst = private_chkVal(item[cfg.key], data[cfg[JV.PROP_FILTER_COMPARE_OBJ_KEY]], cfg[JV.PROP_FILTER_CONDITION]);
  273. if (compRst) break;
  274. }
  275. }
  276. }
  277. if (compRst) {
  278. rstArr.push(item);
  279. }
  280. }
  281. delete sourceData.data;
  282. sourceData.data = rstArr;
  283. // fsUtil.wirteObjToFile(sourceData.data, "D:/GitHome/ConstructionCost/tmp/filteredRst.js");
  284. }
  285. function sortData(sourceData, sortCfg) {
  286. let rst = sourceData.data, tempRstArr = [];
  287. let sortType = sortCfg[JV.PROP_SORT_TYPE];
  288. for (let item of sourceData.data) {
  289. if (item._doc) {
  290. tempRstArr.push(item._doc);
  291. } else {
  292. tempRstArr.push(item);
  293. }
  294. }
  295. switch (sortType) {
  296. case "tree":
  297. rst = treeUtil.buildTreeNodeDirectly(tempRstArr);
  298. let destArr = [];
  299. treeUtil.getFlatArray(rst, destArr);
  300. delete sourceData.data;
  301. sourceData.data = destArr;
  302. // fsUtil.wirteObjToFile(sourceData.data, "D:/GitHome/ConstructionCost/tmp/sortedAndFlattedRst.js");
  303. break;
  304. case "normal":
  305. tempRstArr.sort(function(a, b){
  306. let compRst = 0;
  307. for (let comp of sortCfg[JV.PROP_SORT_KEYS]) {
  308. let reverse = (comp.order === 'ascend')?1:(-1);
  309. if (a[comp.key] > b[comp.key]) {
  310. compRst = reverse;
  311. break;
  312. } else if (a[comp.key] < b[comp.key]) {
  313. compRst = -reverse;
  314. break;
  315. }
  316. }
  317. return compRst;
  318. });
  319. delete sourceData.data;
  320. sourceData.data = tempRstArr;
  321. // fsUtil.wirteObjToFile(sourceData.data, "D:/GitHome/ConstructionCost/tmp/normalSortedRst.js");
  322. break;
  323. default:
  324. //
  325. }
  326. return rst;
  327. }
  328. function setupFunc(obj, prop, ownRawObj) {
  329. obj[prop] = {};
  330. obj[prop]["myOwnRawDataObj"] = ownRawObj;
  331. obj[prop].getProperty = ext_getPropety;
  332. obj[prop].getFee = ext_getFee;
  333. obj[prop].getPropertyByForeignId = ext_getPropertyByForeignId;
  334. }
  335. function assembleFields(fieldList, rstDataArr, $PROJECT) {
  336. if (fieldList) {
  337. for (let field of fieldList) {
  338. shielded_exec_env($PROJECT, field, rstDataArr);
  339. }
  340. }
  341. }
  342. function shielded_exec_env($PROJECT, $ME, rptDataItemObj) {
  343. if ($ME[JV.PROP_FIELD_EXP_MAP]) {
  344. rptDataItemObj.push(eval($ME[JV.PROP_FIELD_EXP_MAP]));
  345. }
  346. }
  347. function ext_mainGetPropety(propKey) {
  348. let rst = [], parentObj = this;
  349. let dtObj = parentObj["myOwnRawDataObj"];
  350. if (propKey && dtObj) {
  351. if (dtObj.hasOwnProperty("property")) {
  352. rst.push(dtObj["property"][propKey]);
  353. } else {
  354. rst.push(dtObj[propKey]);
  355. }
  356. }
  357. return rst;
  358. }
  359. function ext_getPropety(propKey) {
  360. let rst = [], parentObj = this;
  361. let dtObj = parentObj["myOwnRawDataObj"];
  362. if (propKey && dtObj) {
  363. for (let dItem of dtObj.data) {
  364. let doc = (dItem._doc === null || dItem._doc === undefined)?dItem:dItem._doc;
  365. if (doc.hasOwnProperty("property")) {
  366. rst.push(doc["property"][propKey]);
  367. } else if (doc.hasOwnProperty(propKey)) {
  368. rst.push(doc[propKey]);
  369. } else {
  370. rst.push('');
  371. }
  372. }
  373. }
  374. return rst;
  375. }
  376. function ext_mainGetFee(feeKey, dtlFeeKey) {
  377. let rst = [];
  378. let parentObj = this;
  379. let dtObj = parentObj["myOwnRawDataObj"];
  380. if ((dtObj) && (feeKey)) {
  381. if (dtObj.hasOwnProperty("fees")) {
  382. for (let fee of dtObj["fees"]) {
  383. if (fee["fieldName"] === feeKey) {
  384. if (dtlFeeKey) {
  385. rst.push(fee[dtlFeeKey]);
  386. } else {
  387. rst.push(fee["unitFee"]);
  388. }
  389. break;
  390. }
  391. }
  392. } else if (dtObj.hasOwnProperty(feeKey)) {
  393. rst.push(dtObj[feeKey]);
  394. } else {
  395. //
  396. }
  397. }
  398. return rst;
  399. }
  400. function ext_getFee(feeKey, dtlFeeKey) {
  401. let rst = [], parentObj = this;
  402. let dtObj = parentObj["myOwnRawDataObj"];
  403. if (feeKey && dtObj) {
  404. for (let dItem of dtObj.data) {
  405. let hasValue = false;
  406. if (dItem.hasOwnProperty("fees")) {
  407. for (let fee of dItem["fees"]) {
  408. if (fee["fieldName"] === feeKey) {
  409. if (dtlFeeKey) {
  410. rst.push(fee[dtlFeeKey]);
  411. } else {
  412. rst.push(fee["unitFee"]);
  413. }
  414. hasValue = true;
  415. break;
  416. }
  417. }
  418. } else if (dItem.hasOwnProperty(feeKey)) {
  419. hasValue = true;
  420. rst.push(dItem[feeKey]);
  421. } else {
  422. hasValue = true;
  423. rst.push(0);
  424. }
  425. if (!hasValue) {
  426. rst.push(0);
  427. }
  428. }
  429. }
  430. return rst;
  431. }
  432. function ext_getPropertyByForeignId(foreignIdVal, adHocIdKey, propKey) {
  433. let rst = [], parentObj = this;
  434. let IdKey = (adHocIdKey)?adHocIdKey:"ID";
  435. let dtObj = parentObj["myOwnRawDataObj"];
  436. if (foreignIdVal !== null && foreignIdVal !== undefined) {
  437. let isFound = false;
  438. if (foreignIdVal instanceof Array) {
  439. for (let idVal of foreignIdVal) {
  440. isFound = false;
  441. for (let i = 0; i < dtObj.data.length; i++) {
  442. let item = (dtObj.data[i]._doc)?dtObj.data[i]._doc:dtObj.data[i];
  443. if (item[IdKey] === idVal) {
  444. let splitPKey = propKey.split(".");
  445. if (splitPKey.length > 1) {
  446. let rstP = null;
  447. for (let i = 0; i < splitPKey.length; i++) {
  448. if (i === 0) {
  449. rstP = item[splitPKey[i]];
  450. } else {
  451. rstP = rstP[splitPKey[i]];
  452. }
  453. }
  454. rst.push(rstP);
  455. } else {
  456. rst.push(item[propKey]);
  457. }
  458. isFound = true;
  459. break;
  460. }
  461. }
  462. // if (!isFound) rst.push[null];
  463. }
  464. } else {
  465. for (let item of dtObj.data) {
  466. if (item[IdKey] === foreignIdVal) {
  467. rst.push(item[propKey]);
  468. isFound = true;
  469. break;
  470. }
  471. }
  472. // if (!isFound) rst.push[null];
  473. }
  474. }
  475. return rst;
  476. }
  477. export default Rpt_Data_Extractor;