revise_compare.js 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462
  1. 'use strict';
  2. /**
  3. * 台账修订页面js
  4. *
  5. * @author Mai
  6. * @date
  7. * @version
  8. */
  9. const billsCompareField = [
  10. 'code', 'b_code', 'name', 'unit', 'unit_price', 'dgn_qty1', 'dgn_qty2', 'dgn_price',
  11. 'sgfh_qty', 'sgfh_tp', 'sjcl_qty', 'sjcl_tp', 'qtcl_qty', 'qtcl_tp', 'quantity', 'total_price',
  12. 'deal_qty', 'deal_tp'
  13. ];
  14. const posCompareField = [
  15. 'name', 'position', 'sgfh_qty', 'sjcl_qty', 'qtcl_qty', 'quantity'
  16. ];
  17. $(document).ready(() => {
  18. let searchLedger;
  19. autoFlashHeight();
  20. // 初始化spread
  21. const billsSpread = SpreadJsObj.createNewSpread($('#bills-spread')[0]);
  22. const billsSheet = billsSpread.getActiveSheet();
  23. sjsSettingObj.setFxTreeStyle(billsSpreadSetting, sjsSettingObj.FxTreeStyle.jz);
  24. if (thousandth) {
  25. sjsSettingObj.setTpThousandthFormat(billsSpreadSetting);
  26. sjsSettingObj.setTpColsThousandthFormat(billsSpreadSetting.extraCols);
  27. }
  28. billsSpreadSetting.getColor = function (sheet, data, row, col, defaultColor) {
  29. // 增
  30. if (data.differ.indexOf('add') >= 0) return '#ffddc5';
  31. // 删
  32. if (data.differ.indexOf('del') >= 0) return '#DCDCDC';
  33. // 结构变动
  34. if (data.differ.indexOf('tree') >= 0) return '#d0f6fd';
  35. // 修改计算或文字
  36. if (data.differ.indexOf('calc') >= 0 || data.differ.indexOf('pos-add') >= 0 || data.differ.indexOf('pos-del') >= 0 || data.differ.indexOf('pos-calc') >= 0) return '#f8d7da';
  37. if (data.differ.indexOf('info') >= 0 || data.differ.indexOf('pos-info') >= 0) return '#d4edda';
  38. // 层次结构
  39. if (data.level === 2) {
  40. return '#C4CAFB';
  41. } else if ((!data.b_code || data.b_code === '') && data.level > 2) {
  42. return '#DFE8F9';
  43. } else {
  44. return defaultColor;
  45. }
  46. };
  47. SpreadJsObj.initSheet(billsSheet, billsSpreadSetting);
  48. const posSpread = SpreadJsObj.createNewSpread($('#pos-spread')[0]);
  49. const posSheet = posSpread.getActiveSheet();
  50. sjsSettingObj.setGridSelectStyle(posSpreadSetting);
  51. if (thousandth) {
  52. sjsSettingObj.setTpThousandthFormat(posSpreadSetting);
  53. sjsSettingObj.setTpColsThousandthFormat(posSpreadSetting.extraCols);
  54. }
  55. posSpreadSetting.getColor = function (sheet, data, row, col, defaultColor) {
  56. if (!data || !data.differ) return defaultColor;
  57. // 增
  58. if (data.differ.indexOf('add') >= 0) return '#ffddc5';
  59. // 删
  60. if (data.differ.indexOf('del') >= 0) return '#DCDCDC';
  61. // 修改计算或文字
  62. if (data.differ.indexOf('calc') >= 0) return '#f8d7da';
  63. if (data.differ.indexOf('info') >= 0) return '#d4edda';
  64. };
  65. SpreadJsObj.initSheet(posSheet, posSpreadSetting);
  66. const posSearch = $.posSearch({selector: '#pos-search', searchSpread: posSpread});
  67. // 初始化 节点树结构
  68. const treeSetting = {
  69. id: 'ledger_id',
  70. pid: 'ledger_pid',
  71. order: 'order',
  72. level: 'level',
  73. full_path: 'full_path',
  74. rootId: -1,
  75. keys: ['id', 'tender_id', 'ledger_id'],
  76. calcFields: [
  77. 'org_sgfh_tp', 'org_sjcl_tp', 'org_qtcl_tp', 'org_total_price', 'org_deal_tp',
  78. 'new_sgfh_tp', 'new_sjcl_tp', 'new_qtcl_tp', 'new_total_price', 'new_deal_tp',
  79. 'contract_tp', 'qc_tp', 'pre_contract_tp', 'pre_qc_tp'
  80. ],
  81. findNode: function (tree, node, parent) {
  82. const sameId = tree.datas.find(x => {return x.id === node.id});
  83. if (sameId) {
  84. const pid = parent ? parent[tree.setting.id] : tree.setting.rootId;
  85. if (sameId[tree.setting.pid] !== pid) {
  86. sameId.changeParent = true;
  87. sameId.org_parent = parent;
  88. }
  89. return sameId;
  90. } else {
  91. const siblings = parent ? parent.children : tree.children;
  92. return siblings.find(function (x) {
  93. return node.b_code
  94. ? x.b_code === node.b_code && x.name === node.name && x.unit === node.unit && x.unit_price === node.unit_price
  95. : x.code === node.code && x.name === node.name;
  96. });
  97. }
  98. },
  99. loadInfo1: function (node, sourceNode, source) {
  100. for (const f of billsCompareField) {
  101. node['new_' + f] = sourceNode[f];
  102. }
  103. node.id = sourceNode.id;
  104. node.isNew = true;
  105. const posRange = source.pos.getLedgerPos(sourceNode.id);
  106. if (posRange && posRange.length > 0) {
  107. if (!node.pos) node.pos = [];
  108. for (const p of posRange) {
  109. let nP = _.find(node.pos, {id: p.id});
  110. if (!nP) {
  111. nP = {id: p.id};
  112. node.pos.push(nP);
  113. }
  114. for (const f of posCompareField) {
  115. nP['new_' + f] = p[f];
  116. }
  117. nP.isNew = true;
  118. }
  119. }
  120. },
  121. loadInfo2: function (node, sourceNode, source) {
  122. for (const f of billsCompareField) {
  123. node['org_' + f] = sourceNode[f];
  124. }
  125. node.isOrg = true;
  126. const posRange = source.pos.getLedgerPos(sourceNode.id);
  127. if (posRange && posRange.length > 0) {
  128. if (!node.pos) node.pos = [];
  129. for (const p of posRange) {
  130. let nP = _.find(node.pos, {id: p.id});
  131. if (!nP) {
  132. nP = {id: p.id};
  133. node.pos.push(nP);
  134. }
  135. for (const f of posCompareField) {
  136. nP['org_' + f] = p[f];
  137. }
  138. nP.isOrg = true;
  139. }
  140. }
  141. },
  142. afterLoad: function (tree) {
  143. for (const data of tree.datas) {
  144. data.differ = [];
  145. data.differ_str = [];
  146. for (const p of data.pos) {
  147. p.differ = [];
  148. p.differ_str = [];
  149. if (p.isNew && !p.isOrg) {
  150. p.differ.push('add');
  151. if (data.differ.indexOf('pos-add') === -1) data.differ.push('pos-add');
  152. p.differ_str.push('增');
  153. } else if (!p.isNew && p.isOrg) {
  154. p.differ.push('del');
  155. if (data.differ.indexOf('pos-del') === -1) data.differ.push('pos-del');
  156. p.differ_str.push('删');
  157. } else {
  158. const orgPosCalc = getCompare(p, compareFields.posCalc, 'org_', 0);
  159. const newPosCalc = getCompare(p, compareFields.posCalc, 'new_', 0);
  160. if (!_.isMatch(orgPosCalc, newPosCalc)) {
  161. p.differ.push('calc');
  162. if (data.differ.indexOf('pos-calc') === -1) data.differ.push('pos-calc');
  163. p.differ_str.push('量改');
  164. }
  165. const orgPosInfo = getCompare(p, compareFields.posInfo, 'org_', 0);
  166. const newPosInfo = getCompare(p, compareFields.posInfo, 'new_', 0);
  167. if (!_.isMatch(orgPosInfo, newPosInfo)) {
  168. p.differ.push('info');
  169. if (data.differ.indexOf('pos-info') === -1) data.differ.push('pos-info');
  170. p.differ_str.push('文改');
  171. }
  172. }
  173. }
  174. if (data.isNew && !data.isOrg) {
  175. data.differ.push('add');
  176. data.differ_str.push('增');
  177. } else if (!data.isNew && data.isOrg) {
  178. data.differ.push('del');
  179. data.differ_str.push('删');
  180. } else {
  181. if (data.changeParent) data.differ.push('tree');
  182. if (!data.children || data.children.length === 0) {
  183. const orgCalc = getCompare(data, compareFields.leafCalc, 'org_', 0);
  184. const newCalc = getCompare(data, compareFields.leafCalc, 'new_', 0);
  185. if (!_.isMatch(newCalc, orgCalc)) data.differ.push('calc');
  186. } else {
  187. const orgCalc = getCompare(data, compareFields.parentCalc, 'org_', 0);
  188. const newCalc = getCompare(data, compareFields.parentCalc, 'new_', 0);
  189. if (!_.isMatch(newCalc, orgCalc)) data.differ.push('calc');
  190. }
  191. const orgInfo = getCompare(data, compareFields.info, 'org_', '');
  192. const newInfo = getCompare(data, compareFields.info, 'new_', '');
  193. if (!_.isMatch(newInfo, orgInfo)) data.differ.push('info');
  194. if (data.differ.length > 0) data.differ_str.push('改');
  195. }
  196. }
  197. }
  198. };
  199. if (!isTz) {
  200. treeSetting.calcFields.push('deal_tp');
  201. }
  202. treeSetting.calcFun = function (node) {
  203. if (!node.children || node.children.length === 0) {
  204. // node.gather_qty = ZhCalc.add(node.contract_qty, node.qc_qty);
  205. // node.pre_gather_qty = ZhCalc.add(node.pre_contract_qty, node.pre_qc_qty);
  206. node.end_contract_qty = ZhCalc.add(node.contract_qty, node.pre_contract_qty);
  207. node.end_qc_qty = ZhCalc.add(node.qc_qty, node.pre_qc_qty);
  208. // node.end_gather_qty = ZhCalc.add(node.end_contract_qty, node.end_qc_qty);
  209. }
  210. // node.gather_tp = ZhCalc.add(node.contract_tp, node.qc_tp);
  211. // node.pre_gather_tp = ZhCalc.add(node.pre_contract_tp, node.pre_qc_tp);
  212. node.end_contract_tp = ZhCalc.add(node.contract_tp, node.pre_contract_tp);
  213. node.end_qc_tp = ZhCalc.add(node.qc_tp, node.pre_qc_tp);
  214. // node.end_gather_tp = ZhCalc.add(node.end_contract_tp, node.end_qc_tp);
  215. node.dgn_price = ZhCalc.round(ZhCalc.div(node.total_price, node.dgn_qty1), 2);
  216. };
  217. const billsTree = createNewPathTree('compare', treeSetting);
  218. // 清单 相关方法&绑定spreadjs事件
  219. const billsTreeSpreadObj = {
  220. selectionChanged: function (e, info) {
  221. if (info.newSelections) {
  222. if (!info.oldSelections || info.newSelections[0].row !== info.oldSelections[0].row) {
  223. posSpreadObj.loadCurPosData();
  224. posSearch.search($('#pos-keyword').val());
  225. }
  226. }
  227. },
  228. };
  229. billsSpread.bind(spreadNS.Events.SelectionChanged, billsTreeSpreadObj.selectionChanged);
  230. // 计量单元 相关方法&绑定spreadjs事件
  231. const posSpreadObj = {
  232. /**
  233. * 加载计量单元 根据当前台账选择节点
  234. */
  235. loadCurPosData: function () {
  236. const node = SpreadJsObj.getSelectObject(billsSheet);
  237. if (node) {
  238. SpreadJsObj.loadSheetData(posSheet, 'data', node.pos || []);
  239. } else {
  240. SpreadJsObj.loadSheetData(posSheet, 'data', []);
  241. }
  242. },
  243. };
  244. // 加载清单&计量单元数据
  245. postData('load', {filter: 'bills;pos;reviseBills;revisePos'}, function (result) {
  246. const tenderTreeSetting = {
  247. id: 'ledger_id',
  248. pid: 'ledger_pid',
  249. order: 'order',
  250. level: 'level',
  251. rootId: -1,
  252. keys: ['id', 'tender_id', 'ledger_id'],
  253. calcFields: ['deal_tp', 'sgfh_tp', 'sjcl_tp', 'qtcl_tp', 'total_price'],
  254. };
  255. const reviseLedger = {
  256. billsTree: createNewPathTree('ledger', tenderTreeSetting),
  257. pos: new PosData({ id: 'id', ledgerId: 'lid', }),
  258. };
  259. reviseLedger.billsTree.loadDatas(result.reviseBills);
  260. reviseLedger.pos.loadDatas(result.revisePos);
  261. const orgLedger = {
  262. billsTree: createNewPathTree('ledger', tenderTreeSetting),
  263. pos: new PosData({ id: 'id', ledgerId: 'lid', }),
  264. };
  265. orgLedger.billsTree.loadDatas(result.bills);
  266. orgLedger.pos.loadDatas(result.pos);
  267. billsTree.loadCompareData(reviseLedger, orgLedger);
  268. treeCalc.calculateAll(billsTree);
  269. SpreadJsObj.loadSheetData(billsSheet, SpreadJsObj.DataType.Tree, billsTree);
  270. posSpreadObj.loadCurPosData();
  271. }, null);
  272. $.divResizer({
  273. select: '#revise-resize',
  274. callback: function () {
  275. billsSpread.refresh();
  276. let bcontent = $(".bcontent-wrap") ? $(".bcontent-wrap").height() : 0;
  277. $(".sp-wrap").height(bcontent-30);
  278. posSpread.refresh();
  279. }
  280. });
  281. $.divResizer({
  282. select: '#revise-right-spr',
  283. callback: function () {
  284. billsSpread.refresh();
  285. if (posSpread) {
  286. posSpread.refresh();
  287. }
  288. if (searchLedger) {
  289. searchLedger.spread.refresh();
  290. }
  291. }
  292. });
  293. $.subMenu({
  294. menu: '#sub-menu', miniMenu: '#sub-mini-menu', miniMenuList: '#mini-menu-list',
  295. toMenu: '#to-menu', toMiniMenu: '#to-mini-menu',
  296. key: 'menu.1.0.0',
  297. miniHint: '#sub-mini-hint', hintKey: 'menu.hint.1.0.1',
  298. callback: function (info) {
  299. if (info.mini) {
  300. $('.panel-title').addClass('fluid');
  301. $('#sub-menu').removeClass('panel-sidebar');
  302. } else {
  303. $('.panel-title').removeClass('fluid');
  304. $('#sub-menu').addClass('panel-sidebar');
  305. }
  306. autoFlashHeight();
  307. billsSpread.refresh();
  308. if (posSpread) {
  309. posSpread.refresh();
  310. }
  311. if (searchLedger) {
  312. searchLedger.spread.refresh();
  313. }
  314. }
  315. });
  316. // 展开收起标准节点
  317. $('a', '#side-menu').bind('click', function (e) {
  318. e.preventDefault();
  319. const tab = $(this), tabPanel = $(tab.attr('content'));
  320. // 展开工具栏、切换标签
  321. if (!tab.hasClass('active')) {
  322. const close = $('.active', '#side-menu').length === 0;
  323. $('a', '#side-menu').removeClass('active');
  324. tab.addClass('active');
  325. $('.tab-content .tab-pane').removeClass('active');
  326. tabPanel.addClass('active');
  327. showSideTools(tab.hasClass('active'));
  328. if (tab.attr('content') === '#search') {
  329. if (!searchLedger) {
  330. searchLedger = $.billsSearch({
  331. selector: '#search',
  332. searchSpread: billsSpread,
  333. resultSpreadSetting: {
  334. cols: [
  335. {title: '项目节编号', field: 'code', hAlign: 0, width: 90, formatter: '@', readOnly: true},
  336. {title: '清单编号', field: 'b_code', hAlign: 0, width: 80, formatter: '@', readOnly: true},
  337. {title: '名称', field: 'name', width: 150, hAlign: 0, formatter: '@', readOnly: true},
  338. {title: '单位', field: 'unit', width: 50, hAlign: 1, formatter: '@', readOnly: true},
  339. ],
  340. emptyRows: 0,
  341. headRows: 1,
  342. headRowHeight: [32],
  343. headColWidth: [30],
  344. defaultRowHeight: 21,
  345. headerFont: '12px 微软雅黑',
  346. font: '12px 微软雅黑',
  347. selectedBackColor: '#fffacd',
  348. },
  349. afterLocated: function () {
  350. posSpreadObj.loadCurPosData();
  351. }
  352. });
  353. }
  354. searchLedger.spread.refresh();
  355. }
  356. }
  357. else {// 收起工具栏
  358. tab.removeClass('active');
  359. tabPanel.removeClass('active');
  360. showSideTools(tab.hasClass('active'));
  361. }
  362. billsSpread.refresh();
  363. if (posSpread) {
  364. posSpread.refresh();
  365. }
  366. });
  367. // 显示层次
  368. (function (select, sheet) {
  369. $(select).click(function () {
  370. if (!sheet.zh_tree) return;
  371. const tag = $(this).attr('tag');
  372. const tree = sheet.zh_tree;
  373. switch (tag) {
  374. case "1":
  375. case "2":
  376. case "3":
  377. case "4":
  378. case "5":
  379. tree.expandByLevel(parseInt(tag));
  380. SpreadJsObj.refreshTreeRowVisible(sheet);
  381. break;
  382. case "last":
  383. tree.expandByCustom(() => { return true; });
  384. SpreadJsObj.refreshTreeRowVisible(sheet);
  385. break;
  386. case "leafXmj":
  387. tree.expandToLeafXmj();
  388. SpreadJsObj.refreshTreeRowVisible(sheet);
  389. break;
  390. case "differ":
  391. tree.expandByCustom(function (x) {
  392. const posterity = tree.getPosterity(x);
  393. if (posterity.length === 0) return x.differ.length > 0;
  394. for (const p of posterity) {
  395. if (p.differ.length > 0) return true;
  396. }
  397. return false;
  398. });
  399. SpreadJsObj.refreshTreeRowVisible(sheet);
  400. }
  401. });
  402. })('a[name=showLevel]', billsSheet);
  403. // 加载计量数据
  404. $('#load-stage-ok').click(function (x) {
  405. postData('/tender/' + window.location.pathname.split('/')[2] + '/revise/load', {filter: 'stageBills;stagePos'}, function (result) {
  406. const col = billsSpreadSetting.cols.length;
  407. billsSpreadSetting.cols = billsSpreadSetting.cols.concat(billsSpreadSetting.extraCols);
  408. SpreadJsObj.reinitSheetHeader(billsSheet);
  409. for (const b of result.stageBills) {
  410. const node = billsTree.nodes.find(x => {return x.id === b.lid});
  411. if (!node) continue;
  412. for (const f of ['contract_qty', 'contract_tp', 'qc_qty', 'qc_tp', 'pre_contract_qty', 'pre_contract_tp', 'pre_qc_qty', 'pre_qc_tp']) {
  413. node[f] = b[f];
  414. }
  415. }
  416. treeCalc.calculateAll(billsTree);
  417. SpreadJsObj.reloadColData(billsSheet, col, billsSpreadSetting.extraCols.length);
  418. posSpreadSetting.cols = posSpreadSetting.cols.concat(posSpreadSetting.extraCols);
  419. SpreadJsObj.reinitSheetHeader(posSheet);
  420. // todo 加载并计算计量单元期数据
  421. for (const p of result.stagePos) {
  422. const node = billsTree.nodes.find(x => {return x.id === p.lid});
  423. if (!node) continue;
  424. const pos = node.pos.find(x => {return x.id === p.pid});
  425. if (!pos) continue;
  426. pos.contract_qty = p.contract_qty;
  427. pos.qc_qty = p.qc_qty;
  428. //pos.gather_qty = ZhCalc.add(pos.contract_qty, pos.qc_qty);
  429. pos.pre_contract_qty = p.pre_contract_qty;
  430. pos.pre_qc_qty = p.pre_qc_qty;
  431. //pos.pre_gather_qty = ZhCalc.add(pos.pre_contract_qty, pos.pre_qc_qty);
  432. pos.end_contract_qty = ZhCalc.add(pos.contract_qty, pos.pre_contract_qty);
  433. pos.end_qc_qty = ZhCalc.add(pos.qc_qty, pos.pre_qc_qty);
  434. //pos.end_gather_qty = ZhCalc.add(pos.gather_qty, pos.pre_gather_qty);
  435. }
  436. posSpreadObj.loadCurPosData();
  437. $('#load-stage-btn').hide();
  438. $('#load-stage').modal('hide');
  439. });
  440. });
  441. });