revise_compare.js 16 KB

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