rpt_construct_data_util.js 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435
  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(Projects) {
  14. this.Projects = Projects;
  15. };
  16. getSerialNo(fieldId, $CURRENT_RPT, $CURRENT_DATA){
  17. let itemSerialNoRec = $JE.F(fieldId, $CURRENT_RPT);
  18. if (itemSerialNoRec) {
  19. itemSerialNoRec[JV.PROP_AD_HOC_DATA] = [];
  20. for (var innerFmlIdx = 0; innerFmlIdx < $CURRENT_DATA[JV.DATA_DETAIL_DATA][0].length; innerFmlIdx++) {
  21. itemSerialNoRec[JV.PROP_AD_HOC_DATA][innerFmlIdx] = (innerFmlIdx + 1);
  22. }
  23. itemSerialNoRec = null;
  24. }
  25. };
  26. }
  27. class Rpt_Data_Extractor {
  28. constructor () {
  29. this.COMMON = new Rpt_Common();
  30. };
  31. initialize(tpl) {
  32. this.rptTpl = tpl;
  33. };
  34. getDataRequestFilter() {
  35. let rst = [];
  36. let tpl = this.rptTpl;
  37. let pri_func_chk_filter = function (field) {
  38. for (let key of projectConstList) {
  39. if (rst.indexOf(key) < 0) {
  40. if (field[JV.PROP_FIELD_EXP_MAP]) {
  41. if (field[JV.PROP_FIELD_EXP_MAP].indexOf('.' + key + '.') >= 0) {
  42. rst.push(key);
  43. }
  44. }
  45. }
  46. }
  47. };
  48. if (tpl[JV.NODE_FIELD_MAP][JV.NODE_DISCRETE_FIELDS]) {
  49. for (let field of tpl[JV.NODE_FIELD_MAP][JV.NODE_DISCRETE_FIELDS]) {
  50. pri_func_chk_filter(field);
  51. }
  52. }
  53. if (tpl[JV.NODE_FIELD_MAP][JV.NODE_MASTER_FIELDS]) {
  54. for (let field of tpl[JV.NODE_FIELD_MAP][JV.NODE_MASTER_FIELDS]) {
  55. pri_func_chk_filter(field);
  56. }
  57. }
  58. if (tpl[JV.NODE_FIELD_MAP][JV.NODE_DETAIL_FIELDS]) {
  59. for (let field of tpl[JV.NODE_FIELD_MAP][JV.NODE_DETAIL_FIELDS]) {
  60. pri_func_chk_filter(field);
  61. }
  62. }
  63. if (tpl[JV.NODE_FIELD_MAP][JV.NODE_MASTER_FIELDS_EX]) {
  64. for (let field of tpl[JV.NODE_FIELD_MAP][JV.NODE_MASTER_FIELDS_EX]) {
  65. pri_func_chk_filter(field);
  66. }
  67. }
  68. if (tpl[JV.NODE_FIELD_MAP][JV.NODE_DETAIL_FIELDS_EX]) {
  69. for (let field of tpl[JV.NODE_FIELD_MAP][JV.NODE_DETAIL_FIELDS_EX]) {
  70. pri_func_chk_filter(field);
  71. }
  72. }
  73. return rst;
  74. };
  75. //--- 装配数据(把收集到的数据,依据报表模板的指示,预处理(如:排序、过滤、合计)及装配到相关指标) ---//
  76. assembleData(rawDataObj) {
  77. let $PROJECT = {"COMMON": {}, "MAIN": {}, "DETAIL": {}};
  78. let tpl = this.rptTpl;
  79. $PROJECT.MAIN["myOwnRawDataObj"] = rawDataObj.prj._doc;
  80. $PROJECT.MAIN.getProperty = ext_mainGetPropety;
  81. $PROJECT.MAIN.getFee = ext_mainGetFee;
  82. $PROJECT.DETAIL.getRationPropertyByID = ext_getRationPropertyByID;
  83. if (tpl[JV.NODE_MAP_DATA_HANDLE_INFO]) {
  84. for (let preHandle of tpl[JV.NODE_MAP_DATA_HANDLE_INFO]) {
  85. let srcData = getModuleDataByKey(rawDataObj.prjData, preHandle[JV.PROP_DATA_KEY]);
  86. switch(preHandle[JV.PROP_HANDLE_TYPE]) {
  87. case JV.PROP_HANDLE_TYPE_SORT:
  88. sortData(srcData, preHandle);
  89. break;
  90. case JV.PROP_HANDLE_TYPE_FILTER:
  91. filterData(srcData, preHandle, rawDataObj.prjData);
  92. break;
  93. case JV.PROP_HANDLE_TYPE_SUM:
  94. summaryData(srcData, preHandle, rawDataObj.prjData);
  95. break;
  96. default:
  97. break;
  98. }
  99. }
  100. }
  101. for (let item of rawDataObj.prjData) {
  102. setupFunc($PROJECT.DETAIL, item.moduleName, item);
  103. }
  104. let rptDataObj = {};
  105. rptDataObj[JV.DATA_DISCRETE_DATA] = [];
  106. rptDataObj[JV.DATA_MASTER_DATA] = [];
  107. rptDataObj[JV.DATA_DETAIL_DATA] = [];
  108. rptDataObj[JV.DATA_MASTER_DATA_EX] = [];
  109. rptDataObj[JV.DATA_DETAIL_DATA_EX] = [];
  110. assembleFields(tpl[JV.NODE_FIELD_MAP][JV.NODE_DISCRETE_FIELDS], rptDataObj[JV.DATA_DISCRETE_DATA], $PROJECT);
  111. // console.log(JV.DATA_DISCRETE_DATA);
  112. // console.log(rptDataObj[JV.DATA_DISCRETE_DATA]);
  113. assembleFields(tpl[JV.NODE_FIELD_MAP][JV.NODE_MASTER_FIELDS], rptDataObj[JV.DATA_MASTER_DATA], $PROJECT);
  114. // console.log(JV.DATA_MASTER_DATA);
  115. // console.log(rptDataObj[JV.DATA_MASTER_DATA]);
  116. assembleFields(tpl[JV.NODE_FIELD_MAP][JV.NODE_DETAIL_FIELDS], rptDataObj[JV.DATA_DETAIL_DATA], $PROJECT);
  117. // console.log(JV.DATA_DETAIL_DATA);
  118. // console.log(rptDataObj[JV.DATA_DETAIL_DATA]);
  119. assembleFields(tpl[JV.NODE_FIELD_MAP][JV.NODE_MASTER_FIELDS_EX], rptDataObj[JV.DATA_MASTER_DATA_EX], $PROJECT);
  120. // console.log(JV.DATA_MASTER_DATA_EX);
  121. // console.log(rptDataObj[JV.DATA_MASTER_DATA_EX]);
  122. assembleFields(tpl[JV.NODE_FIELD_MAP][JV.NODE_DETAIL_FIELDS_EX], rptDataObj[JV.DATA_DETAIL_DATA_EX], $PROJECT);
  123. // console.log(JV.DATA_DETAIL_DATA_EX);
  124. // console.log(rptDataObj[JV.DATA_DETAIL_DATA_EX]);
  125. return rptDataObj;
  126. };
  127. }
  128. function getModuleDataByKey(prjData, key) {
  129. let rst = null;
  130. for (let item of prjData) {
  131. if (item.moduleName === key) {
  132. rst = item;
  133. break;
  134. }
  135. }
  136. return rst;
  137. }
  138. function summaryData(sourceData, handleCfg, prjData){
  139. let rstArr = [], tempRstArr = [];
  140. let curParentPrjData = {};
  141. for (let item of sourceData.data) {
  142. if (item._doc) {
  143. tempRstArr.push(item._doc);
  144. } else {
  145. tempRstArr.push(item);
  146. }
  147. }
  148. let private_get_grp_key = function (item) {
  149. let keys = [];
  150. for (let cfg of handleCfg[JV.PROP_SUM_GROUP_KEYS]) {
  151. if (typeof cfg === "string") {
  152. keys.push(item[cfg]);
  153. } else {
  154. if (!curParentPrjData[cfg["seeking_parent"]]) curParentPrjData[cfg["seeking_parent"]] = getModuleDataByKey(prjData, cfg["seeking_parent"]);
  155. for (let pDataItem of curParentPrjData[cfg["seeking_parent"]].data) {
  156. let data = (pDataItem._doc)?pDataItem._doc:pDataItem;
  157. if (item[cfg["seeking_key"]] === data[cfg["parent_key"]]) {
  158. keys.push(data[cfg["parent_grp_key"]]);
  159. break;
  160. }
  161. }
  162. }
  163. }
  164. return ( "grp_key_" + keys.join('_'));
  165. }
  166. let sumObj = {};
  167. for (let dtl of tempRstArr) {
  168. let grpKey = private_get_grp_key(dtl);
  169. if (sumObj[grpKey] === null || sumObj[grpKey] === undefined) {
  170. sumObj[grpKey] = dtl;
  171. rstArr.push(dtl);
  172. } else {
  173. for (let sumKey of handleCfg[JV.PROP_SUM_SUM_KEYS]) {
  174. if (dtl[sumKey]) {
  175. sumObj[grpKey][sumKey] += dtl[sumKey];
  176. }
  177. }
  178. }
  179. }
  180. delete sourceData.data;
  181. sourceData.data = rstArr;
  182. // fsUtil.wirteObjToFile(sourceData.data, "D:/GitHome/ConstructionCost/tmp/sumRst.js");
  183. }
  184. function filterData(sourceData, handleCfg, prjData) {
  185. let rstArr = [], tempRstArr = [];
  186. for (let item of sourceData.data) {
  187. if (item._doc) {
  188. tempRstArr.push(item._doc);
  189. } else {
  190. tempRstArr.push(item);
  191. }
  192. }
  193. let private_chkVal = function (src, dest, compStr) {
  194. let rst = true;
  195. switch (compStr) {
  196. case "==" :
  197. rst = (src == dest);
  198. break;
  199. case "===" :
  200. rst = (src === dest);
  201. break;
  202. case ">" :
  203. rst = (src > dest);
  204. break;
  205. case ">=" :
  206. rst = (src >= dest);
  207. break;
  208. case "<" :
  209. rst = (src < dest);
  210. break;
  211. case "<=" :
  212. rst = (src <= dest);
  213. break;
  214. case "!=" :
  215. rst = (src != dest);
  216. break;
  217. case "!==" :
  218. rst = (src !== dest);
  219. break;
  220. default:
  221. rst = true;
  222. }
  223. return rst;
  224. }
  225. for (let item of tempRstArr) {
  226. let compRst = true;
  227. let curComparePrjData = null;
  228. for (let cfg of handleCfg[JV.PROP_FILTER_KEY]) {
  229. if (cfg[JV.PROP_FILTER_COMPARE_VAL]) {
  230. //比较key值
  231. compRst = private_chkVal(item[cfg.key], cfg[JV.PROP_FILTER_COMPARE_VAL], cfg[JV.PROP_FILTER_CONDITION]);
  232. } else if (cfg[JV.PROP_FILTER_COMPARE_OBJ] && cfg[JV.PROP_FILTER_COMPARE_OBJ_KEY]){
  233. //通过其他对象来过滤
  234. if (!curComparePrjData) {
  235. curComparePrjData = getModuleDataByKey(prjData, cfg[JV.PROP_FILTER_COMPARE_OBJ]);
  236. }
  237. for (let data of curComparePrjData.data) {
  238. compRst = private_chkVal(item[cfg.key], data[cfg[JV.PROP_FILTER_COMPARE_OBJ_KEY]], cfg[JV.PROP_FILTER_CONDITION]);
  239. if (compRst) break;
  240. }
  241. }
  242. }
  243. if (compRst) {
  244. rstArr.push(item);
  245. }
  246. }
  247. delete sourceData.data;
  248. sourceData.data = rstArr;
  249. // fsUtil.wirteObjToFile(sourceData.data, "D:/GitHome/ConstructionCost/tmp/filteredRst.js");
  250. }
  251. function sortData(sourceData, sortCfg) {
  252. let rst = sourceData.data, tempRstArr = [];
  253. let sortType = sortCfg[JV.PROP_SORT_TYPE];
  254. for (let item of sourceData.data) {
  255. if (item._doc) {
  256. tempRstArr.push(item._doc);
  257. } else {
  258. tempRstArr.push(item);
  259. }
  260. }
  261. switch (sortType) {
  262. case "tree":
  263. rst = treeUtil.buildTreeNodeDirectly(tempRstArr);
  264. let destArr = [];
  265. treeUtil.getFlatArray(rst, destArr);
  266. delete sourceData.data;
  267. sourceData.data = destArr;
  268. // fsUtil.wirteObjToFile(sourceData.data, "D:/GitHome/ConstructionCost/tmp/sortedAndFlattedRst.js");
  269. break;
  270. case "normal":
  271. tempRstArr.sort(function(a, b){
  272. let compRst = 0;
  273. for (let comp of sortCfg[JV.PROP_SORT_KEYS]) {
  274. let reverse = (comp.order === 'ascend')?1:(-1);
  275. if (a[comp.key] > b[comp.key]) {
  276. compRst = reverse;
  277. break;
  278. } else if (a[comp.key] < b[comp.key]) {
  279. compRst = -reverse;
  280. break;
  281. }
  282. }
  283. return compRst;
  284. });
  285. delete sourceData.data;
  286. sourceData.data = tempRstArr;
  287. // fsUtil.wirteObjToFile(sourceData.data, "D:/GitHome/ConstructionCost/tmp/normalSortedRst.js");
  288. break;
  289. default:
  290. //
  291. }
  292. return rst;
  293. }
  294. function setupFunc(obj, prop, ownRawObj) {
  295. obj[prop] = {};
  296. obj[prop]["myOwnRawDataObj"] = ownRawObj;
  297. obj[prop].getProperty = ext_getPropety;
  298. obj[prop].getFee = ext_getFee;
  299. }
  300. function assembleFields(fieldList, rstDataArr, $PROJECT) {
  301. if (fieldList) {
  302. for (let field of fieldList) {
  303. shielded_exec_env($PROJECT, field, rstDataArr);
  304. }
  305. }
  306. }
  307. function shielded_exec_env($PROJECT, $ME, rptDataItemObj) {
  308. if ($ME[JV.PROP_FIELD_EXP_MAP]) {
  309. rptDataItemObj.push(eval($ME[JV.PROP_FIELD_EXP_MAP]));
  310. }
  311. }
  312. function ext_mainGetPropety(propKey) {
  313. let rst = [];
  314. let parentObj = this;
  315. //console.log(this);
  316. let dtObj = parentObj["myOwnRawDataObj"];
  317. if ((dtObj) && (propKey)) {
  318. if (dtObj.hasOwnProperty("property")) {
  319. rst.push(dtObj["property"][propKey]);
  320. } else {
  321. rst.push(dtObj[propKey]);
  322. }
  323. }
  324. return rst;
  325. }
  326. function ext_mainGetFee(feeKey) {
  327. let rst = [];
  328. let parentObj = this;
  329. let dtObj = parentObj["myOwnRawDataObj"];
  330. if ((dtObj) && (feeKey)) {
  331. if (dtObj.hasOwnProperty("fees")) {
  332. for (let fee of dtObj["fees"]) {
  333. if (fee["fieldName"] === feeKey) {
  334. rst.push(dtObj["fees"][feeKey]);
  335. break;
  336. }
  337. }
  338. } else if (dtObj.hasOwnProperty(feeKey)) {
  339. rst.push(dtObj[feeKey]);
  340. } else {
  341. //
  342. }
  343. }
  344. return rst;
  345. }
  346. function ext_getPropety(propKey) {
  347. let rst = [], parentObj = this;
  348. let dtObj = parentObj["myOwnRawDataObj"];
  349. if (propKey && dtObj) {
  350. for (let dItem of dtObj.data) {
  351. let doc = (dItem._doc === null || dItem._doc === undefined)?dItem:dItem._doc;
  352. if (doc.hasOwnProperty("property")) {
  353. rst.push(doc["property"][propKey]);
  354. } else if (doc.hasOwnProperty(propKey)) {
  355. rst.push(doc[propKey]);
  356. } else {
  357. rst.push('');
  358. }
  359. }
  360. }
  361. return rst;
  362. }
  363. function ext_getFee(feeKey) {
  364. let rst = [], parentObj = this;
  365. let dtObj = parentObj["myOwnRawDataObj"];
  366. if (feeKey && dtObj) {
  367. for (let dItem of dtObj.data) {
  368. if (dItem.hasOwnProperty("fees")) {
  369. for (let fee of dItem["fees"]) {
  370. if (fee["fieldName"] === feeKey) {
  371. rst.push(dItem["fees"][feeKey]);
  372. break;
  373. }
  374. }
  375. } else if (dItem.hasOwnProperty(feeKey)) {
  376. rst.push(dItem[feeKey]);
  377. } else {
  378. rst.push[0];
  379. }
  380. }
  381. }
  382. return rst;
  383. }
  384. function ext_getRationPropertyByID(IdVal, propKey) {
  385. let rst = [], me = this;
  386. if (IdVal !== null && IdVal !== undefined && me[projectConst.RATION]) {
  387. let isFound = false;
  388. if (IdVal instanceof Array) {
  389. for (let id of IdVal) {
  390. isFound = false;
  391. for (let item of me[projectConst.RATION]["myOwnRawDataObj"].data) {
  392. if (item.ID === id) {
  393. rst.push(item[propKey]);
  394. isFound = true;
  395. break;
  396. }
  397. }
  398. // if (!isFound) rst.push[null];
  399. }
  400. } else {
  401. for (let item of me[projectConst.RATION]["myOwnRawDataObj"].data) {
  402. if (item.ID === IdVal) {
  403. rst.push(item[propKey]);
  404. isFound = true;
  405. break;
  406. }
  407. }
  408. // if (!isFound) rst.push[null];
  409. }
  410. }
  411. return rst;
  412. }
  413. export default Rpt_Data_Extractor;