project_view.js 48 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043
  1. /**
  2. * Created by Mai on 2017/6/21.
  3. */
  4. var projectObj = {
  5. project: null,
  6. mainSpread: null,
  7. mainController: null,
  8. gljSpreed:null,
  9. beforeMainTreeSelectedChange: function (node) {
  10. /*if (node) {
  11. subViewObj.saveComments(node);
  12. }*/
  13. },
  14. treeSelectedChanged: function (node) {
  15. let project = projectObj.project;
  16. subViewObj.loadComments(node);
  17. gljOprObj.showDataIfRationSelect(node);
  18. // CSL.2017.07.25
  19. if (activeSubSheetIs(subSheetIndex.ssiCalcProgram)) {
  20. calcProgramObj.showData(node);
  21. };
  22. //zhong 2017-9-1 特征及内容
  23. if(pageCCOprObj.active){
  24. pageCCOprObj.mainActiveCell = projectObj.mainSpread.getActiveSheet().getSelections()[0];//mainSpread焦点单元格
  25. if(node.sourceType === project.Bills.getSourceType()){
  26. pageCCOprObj.setCacheAndShow(node);
  27. }
  28. else{
  29. pageCCOprObj.clearData();
  30. }
  31. }
  32. },
  33. refreshBaseActn: function (tree) {
  34. let setButtonValid = function (valid, btn) {
  35. if (valid) {
  36. btn.removeClass('disabled');
  37. } else {
  38. btn.addClass('disabled');
  39. }
  40. };
  41. let selected = tree.selected, that = projectObj;
  42. let canUpLevel = function (node) {
  43. if (selected && selected.depth() > 0 && selected.canUpLevel()) {
  44. if (selected.sourceType === that.project.Bills.getSourceType()) {
  45. return (!selected.nextSibling) || (selected.children.length === 0) || (selected.source.children.length > 0);
  46. } else {
  47. return false;
  48. }
  49. } else {
  50. return false;
  51. }
  52. };
  53. let canDownLevel = function (node) {
  54. if (selected && selected.depth() > 0 && selected.canDownLevel()) {
  55. if (selected.sourceType === that.project.Bills.getSourceType()) {
  56. return (selected.preSibling.children.length === 0) || (selected.preSibling.source.children.length > 0);
  57. } else {
  58. return false;
  59. }
  60. } else {
  61. return false;
  62. }
  63. };
  64. let canDelete = function (node) {
  65. if (selected) {
  66. if (selected.sourceType === that.project.Bills.getSourceType()) {
  67. return !(selected.data.flagsIndex && selected.data.flagsIndex.fixed && selected.data.flagsIndex.fixed.flag > 0);
  68. } else {
  69. return true;
  70. }
  71. } else {
  72. return false;
  73. }
  74. };
  75. setButtonValid(canUpLevel(selected), $('#upLevel'));
  76. setButtonValid(canDownLevel(selected), $('#downLevel'));
  77. setButtonValid(selected && (selected.depth() > 0) && selected.canUpMove(), $('#upMove'));
  78. setButtonValid(selected && (selected.depth() > 0) && selected.canDownMove(), $('#downMove'));
  79. setButtonValid(canDelete(selected), $('#delete'));
  80. },
  81. checkCommonField: function (editingText, colSetting) {
  82. let value;
  83. if (colSetting.data.decimal) {
  84. value = Number(editingText);
  85. if (number_util.isNumber(value)) {
  86. if (colSetting.data.decimal) {
  87. value = value.toDecimal(colSetting.data.decimal);
  88. }
  89. } else {
  90. value = null;
  91. alert('当前输入的数据类型不正确,请重新输入。');
  92. }
  93. } else {
  94. value = editingText;
  95. }
  96. return value;
  97. },
  98. checkFormulaValidField: function (editingText, colSetting) {
  99. let value = Number(editingText);
  100. if (!value) {
  101. try {
  102. let exp = new Expression('');
  103. exp.Expression(editingText);
  104. value = Number(exp.Evaluate());
  105. } catch (error) {
  106. value = null;
  107. }
  108. }
  109. if (value!=undefined||value!=null) {
  110. if (colSetting.data.decimal) {
  111. value = value.toDecimal(colSetting.data.decimal);
  112. }
  113. } else if (editingText && editingText !== '') {
  114. value = null;
  115. alert('当前输入的数据类型不正确,请重新输入。');
  116. }
  117. return value;
  118. },
  119. checkSpreadEditingText: function (editingText, colSetting) {
  120. if (colSetting.data.field === 'quantity'||colSetting.data.field === 'contain') {
  121. return this.checkFormulaValidField(editingText, colSetting);
  122. }
  123. else if (colSetting.data.field === 'programID') {
  124. return this.project.calcProgram.compiledTemplateMaps[editingText];
  125. }
  126. else if (colSetting.data.field === 'subType') {
  127. if (typeof(editingText) !== "number")
  128. return volumePriceMaps[editingText]
  129. else
  130. return editingText;
  131. }
  132. else {
  133. return this.checkCommonField(editingText, colSetting);
  134. }
  135. },
  136. /*updateAndReCalculate: function (node, fieldName, value) {
  137. let project = projectObj.project, calc = new BillsCalcHelper(project), nodes = [];
  138. let getNodes = function (node) {
  139. let cur = node, nodes = [];
  140. while (cur) {
  141. nodes.push(cur);
  142. cur = cur.parent;
  143. }
  144. return nodes;
  145. }
  146. if (value) {
  147. value = value.toDecimal(projectObj.project.Decimal.common.quantity);
  148. }
  149. if (node.sourceType === projectObj.project.Bills.getSourceType()) {
  150. calcFees.setFee(node.data, fieldName, value);
  151. calc.calcNode(node, true);
  152. nodes = getNodes(node);
  153. project.Bills.updateNodes(nodes, true);
  154. }
  155. else if (node.sourceType === projectObj.project.VolumePrice.getSourceType()) {
  156. project.beginUpdate('VolumePrice_QuantityChange');
  157. project.VolumePrice.updateField(node.source, fieldName, value, false);
  158. calc.calcNode(node.parent, true);
  159. nodes = getNodes(node.parent);
  160. project.Bills.updateNodes(nodes, false);
  161. project.endUpdate();
  162. nodes.push(node);
  163. }
  164. else if (node.sourceType === projectObj.project.Ration.getSourceType()) {
  165. project.beginUpdate('Ration_QuantityChange');
  166. calcFees.setFee(node.data, fieldName, value);
  167. node.data.gljList = project.ration_glj.getGljArrByRation(node.data.ID);
  168. // calcProgram.calculate的传参必须是cacheNode类型,故无法将计算放在Ration模块中。
  169. project.calcProgram.calculate(node);
  170. project.Ration.updateRation(node.source, false); // 加入待存储队列
  171. calc.calcNode(node.parent, true); //
  172. nodes = getNodes(node.parent);
  173. project.Bills.updateNodes(nodes, false);
  174. project.endUpdate();
  175. nodes.push(node);
  176. }
  177. this.mainController.refreshTreeNode(nodes, false);
  178. calc = null;
  179. },*/
  180. updateBillsCode: function (node, value) {
  181. let project = projectObj.project;
  182. let stdMatchCode, formatCode, matchs;
  183. let searchStdBillsAndUpdate = function (stdCode, formatCode) {
  184. let orgCode = node.data.code.substr(0, 9);
  185. if (stdCode === orgCode || projectInfoObj.projectInfo.engineeringInfo.bill_lib.length === 0) {
  186. project.Bills.updateField(node.source, 'code', formatCode, true);
  187. projectObj.mainController.refreshTreeNode([node], false);
  188. } else if (projectInfoObj.projectInfo.engineeringInfo.bill_lib.length > 0) {
  189. let libId = projectInfoObj.projectInfo.engineeringInfo.bill_lib[0].id;
  190. CommonAjax.post('/stdBillsEditor/getStdBillsByCode', {userId: userID, billsLibId: libId, code: stdCode}, function (data) {
  191. if (data) {
  192. data.itemCharacter = pageCCOprObj.safeItemCharater(data.itemCharacter);
  193. if (/\//.test(data.unit)) {
  194. ConfirmModal.stdBillsUnit.check(data, function (std) {
  195. project.Bills.replaceBills(node.source, std, formatCode);
  196. projectObj.mainController.refreshTreeNode([node], false);
  197. }, function () {
  198. projectObj.mainController.refreshTreeNode([node], false);
  199. });
  200. } else {
  201. project.Bills.replaceBills(node.source, data, formatCode);
  202. projectObj.mainController.refreshTreeNode([node], false);
  203. }
  204. } else {
  205. project.Bills.updateField(node.source, 'code', formatCode, true);
  206. projectObj.mainController.refreshTreeNode([node], false);
  207. }
  208. });
  209. }
  210. }
  211. if (value&&value.length === 9 && /^[\d]+$/.test(value)) {
  212. stdMatchCode = value;
  213. formatCode = project.Bills.newFormatCode(stdMatchCode);
  214. searchStdBillsAndUpdate(stdMatchCode, formatCode);
  215. } else if (value&&value.length === 12 && /^[\d]+$/.test(value)) {
  216. stdMatchCode = value.substr(0, 9);
  217. matchs = project.Bills.sameStdCode(stdMatchCode, node.data.code);
  218. if (matchs.indexOf(value) === -1) {
  219. searchStdBillsAndUpdate(stdMatchCode, value);
  220. } else if (confirm('已存在该编码的清单,是否继续?')) {
  221. formatCode = project.Bills.newFormatCode(stdMatchCode, node.data.code);
  222. searchStdBillsAndUpdate(stdMatchCode, formatCode);
  223. }
  224. } else {
  225. project.Bills.updateField(node.source, 'code', value, true);
  226. this.mainController.refreshTreeNode([node], false);
  227. }
  228. },
  229. updateRationCode: function (node, value) {
  230. if (/[\w]{2}[\d]{4}/.test(value)) {
  231. if (projectInfoObj.projectInfo.engineeringInfo.ration_lib.length === 0) {
  232. alert('当前项目无定额库,请添加定额库。');
  233. this.mainController.refreshTreeNode([node], false);
  234. } else {
  235. let libId = projectInfoObj.projectInfo.engineeringInfo.ration_lib[0].id;
  236. CommonAjax.postRationLib('/rationRepository/api/matchRation', {user_id: userID, rationLibId: libId, code: value}, function (data) {
  237. if (data) {
  238. projectObj.project.Ration.replaceRation(node.source, data);
  239. projectObj.project.ration_glj.addRationGLJ(node.source, data);
  240. } else {
  241. alert('当前库中找不到定额"' + value + '"');
  242. }
  243. projectObj.mainController.refreshTreeNode([node], false);
  244. });
  245. }
  246. } else {
  247. alert('输入的定额编码有误,请检查。');
  248. this.mainController.refreshTreeNode([node], false);
  249. }
  250. },
  251. updateCode: function (node, value) {
  252. let project = projectObj.project;
  253. if (node.sourceType === project.Bills.getSourceType()) {
  254. this.updateBillsCode(node, value); // 新清单不适合实时计算,下面套什么还不能确定,无数量计算也无意义
  255. } else if (node.sourceType === project.Ration.getSourceType()) {
  256. this.updateRationCode(node, value); // 新套定额适合实时计算
  257. // 这里因异步问题暂时缺少工料机价格。该过程移到:ration_glj.js的refreshAfterSave方法中。
  258. /*project.calcProgram.calcAndSave(node);
  259. if (activeSubSheetIs(subSheetIndex.ssiCalcProgram)) {
  260. calcProgramObj.showData(node, false);
  261. };*/
  262. }
  263. },
  264. updateCellValue: function (node, value, colSetting) {
  265. let project = projectObj.project, fieldName = colSetting.data.field;
  266. if(node.sourceType==project.ration_glj.getSourceType()){
  267. project.ration_glj.updateFromMainSpread(value,node,fieldName);
  268. }else if(treeNodeTools.isGljRation(node)){
  269. gljOprObj.updateRationTypeGLJ(value,node,fieldName);
  270. } else if (value !== calcFees.getFee(node.data, fieldName)||fieldName == 'quantity') {//工程量需要进行转换,所以做特殊处理
  271. if (fieldName === 'code' && !treeNodeTools.isVolumePrice(node)) {
  272. projectObj.updateCode(node, value);
  273. }
  274. else if(fieldName ==='feeRate'){
  275. project.FeeRate.updateFeeRateFromBills(value,node,fieldName);
  276. }else if(fieldName ==='contain'){//编辑含量
  277. project.Ration.updateContain(value,node);
  278. }
  279. else if (fieldName === 'quantity' || fieldName === 'marketUnitFee' || fieldName === 'programID' ||
  280. fieldName === 'subType' || fieldName === 'calcBase' || fieldName === 'feesIndex.common.unitFee'){
  281. if (fieldName === 'quantity') {
  282. project.quantity_detail.editMainTreeNodeQuantity(value,node,fieldName);
  283. return;
  284. }
  285. else if (fieldName === 'marketUnitFee' || fieldName === 'feesIndex.common.unitFee') {
  286. if (value) {value = parseFloat(value).toDecimal(decimalObj.decimal("unitPrice", node))};
  287. }
  288. else if (fieldName === 'calcBase') {
  289. //zhong
  290. if(value === undefined ||value === null || value.toString().trim() === ''){
  291. value = '';
  292. }
  293. node.data.userCalcBase = value;
  294. project.calcBase.calculate(node);
  295. if(!project.calcBase.success){
  296. let activeCell = projectObj.mainSpread.getActiveSheet().getSelections()[0];
  297. projectObj.mainSpread.getActiveSheet().setValue(activeCell.row, activeCell.col, node.data.calcBase? node.data.calcBase: '');
  298. return;
  299. }
  300. // if (value) {value = parseFloat(value).toDecimal(decimalObj.decimal("totalPrice", node))};
  301. };
  302. node.changed = true;
  303. if (fieldName == 'feesIndex.common.unitFee'){
  304. treeNodeTools.initFeeField(node, 'common');
  305. node.data.feesIndex.common.unitFee = value;
  306. }
  307. else if(fieldName !== 'calcBase'){
  308. node.data[fieldName] = value;
  309. }
  310. project.calcProgram.calcAndSave(node);
  311. gljOprObj.showRationGLJSheetData();
  312. } else if (node.sourceType === project.Bills.getSourceType()&&fieldName === 'unit'){//修改清单单位的时候清单工程量要重新4舍5入
  313. node.data[fieldName] = value;
  314. node.changed = true;
  315. if(node.data.quantity){
  316. node.data.quantity =scMathUtil.roundForObj(node.data.quantity,getDecimal("quantity",node));
  317. }
  318. project.calcProgram.calcAndSave(node);
  319. }
  320. else {
  321. if (node.sourceType === project.Bills.getSourceType()) {
  322. project.Bills.updateField(node.source, fieldName, value, true);
  323. }
  324. else if (node.sourceType === project.Ration.getSourceType()) {
  325. project.Ration.updateField(node.source, fieldName, value, true);
  326. };
  327. if (colSetting.data.wordWrap) {
  328. this.mainSpread.getActiveSheet().autoFitRow(node.serialNo());
  329. }
  330. projectObj.mainController.refreshTreeNode([node]);
  331. }
  332. } else if(value==null && fieldName ==='feeRate'){
  333. project.FeeRate.cleanFeeRateID(node);
  334. } else {
  335. projectObj.mainController.refreshTreeNode([node], false);
  336. }
  337. },
  338. mainSpreadLeaveCell: function (sender, info) {
  339. let colSetting = projectObj.mainController.setting.cols[info.col];
  340. projectObj.lastCol = colSetting;
  341. },
  342. mainSpreadEnterCell: function (sender, info) {
  343. let colSetting = projectObj.mainController.setting.cols[info.col];
  344. if(colSetting.data.field === 'unit' || projectObj.lastCol.data.field === 'unit'||colSetting.data.field ==='subType'|| projectObj.lastCol.data.field === 'subType'){
  345. info.sheet.repaint();
  346. }
  347. },
  348. mainSpreadEditEnded: function (sender, info) {
  349. let project = projectObj.project;
  350. let node = project.mainTree.items[info.row];
  351. let colSetting = projectObj.mainController.setting.cols[info.col];
  352. let fieldName = projectObj.mainController.setting.cols[info.col].data.field;
  353. // 检查输入类型等
  354. let value = projectObj.checkSpreadEditingText(info.editingText, colSetting);
  355. projectObj.updateCellValue(node, value, colSetting);
  356. // 自动行高
  357. const autoHeight = project.property.displaySetting.autoHeight;
  358. if (autoHeight) {
  359. projectObj.mainSpread.getActiveSheet().autoFitRow(info.row);
  360. }
  361. },
  362. mainSpreadRangeChanged: function (sender, info) {
  363. let project = projectObj.project, setting = projectObj.mainController.setting;
  364. if (info.changedCells.length > 0) {
  365. for (let changedCell of info.changedCells) {
  366. let cell = info.sheet.getCell(changedCell.row, changedCell.col);
  367. let node = project.mainTree.items[changedCell.row];
  368. let colSetting = setting.cols[changedCell.col];
  369. let value = projectObj.checkSpreadEditingText(cell.text(), colSetting);
  370. projectObj.updateCellValue(node, value, colSetting);
  371. }
  372. }
  373. },
  374. checkMainSpread: function () {
  375. if (!this.mainSpread) {
  376. this.mainSpread = SheetDataHelper.createNewSpread($('#billsSpread')[0]);
  377. this.mainSpread.getActiveSheet().selectionPolicy(GC.Spread.Sheets.SelectionPolicy.muliRange);
  378. }
  379. },
  380. refreshMainSpread: function () {
  381. if (this.mainSpread) {
  382. this.mainSpread.refresh();
  383. }
  384. },
  385. loadProjectData: function () {
  386. var that = this;
  387. this.project = PROJECT.createNew(scUrlUtil.GetQueryString('project'), userID);
  388. let startTime = +new Date();
  389. console.log("开始加载-----"+startTime);
  390. this.project.loadDatas(function (err) {
  391. if (!err) {
  392. that.project.projectGLJ.calcQuantity();//计算分部分项和技术措施项目消耗量;
  393. that.project.property = projectInfoObj.projectInfo.property;
  394. //that.project.calcProgram.compileAllTemps();
  395. that.project.calcBase.init(that.project);
  396. that.project.calcFields = JSON.parse(JSON.stringify(cpFeeTypes));
  397. // that.project.initCalcFields();
  398. let str = JSON.stringify(that.project.projSetting.main_tree_col);
  399. that.project.projSetting.mainGridSetting = JSON.parse(str);
  400. that.project.projSetting.mainGridSetting.frozenCols = 4;
  401. TREE_SHEET_HELPER.initSetting($('#billsSpread')[0], that.project.projSetting.mainGridSetting);
  402. const autoHeight = that.project.property.displaySetting !== undefined ?
  403. that.project.property.displaySetting.autoHeight : false;
  404. that.project.projSetting.mainGridSetting.cols.forEach(function (col) {
  405. // for test. 后端没有绑定,暂时写死用于测试。
  406. /* if (col.data.field == '' && col.head.titleNames[0] == "取费专业") {
  407. col.data.field = 'programID';
  408. col.data.getText = 'getText.calcProgramName';
  409. col.data.cellType = 'cellType.calcProgramName';
  410. };*/
  411. col.data.splitFields = col.data.field.split('.');
  412. if (col.data.getText && Object.prototype.toString.apply(col.data.getText) === "[object String]") {
  413. col.data.getText = MainTreeCol.getEvent(col.data.getText);
  414. }
  415. if (col.readOnly && Object.prototype.toString.apply(col.readOnly) === "[object String]") {
  416. col.readOnly = MainTreeCol.getEvent(col.readOnly);
  417. }
  418. if (col.data.cellType && Object.prototype.toString.apply(col.data.cellType) === "[object String]") {
  419. let getCellType = MainTreeCol.getEvent(col.data.cellType);
  420. col.data.cellType = getCellType;
  421. }
  422. // if (col.data.digit && Object.prototype.toString.apply(col.data.digit) === "[object String]") {
  423. // col.data.decimal = that.project.getDecimal(col.data.digit);
  424. // col.data.formatter = MainTreeCol.getNumberFormatter(col.data.decimal);
  425. // }
  426. if (col.data.field === 'code') {
  427. col.data.formatter = '@';
  428. }
  429. // 根据配置设置自动行高
  430. if (col.data.field === 'name' || col.data.field === 'itemCharacterText' ||
  431. col.data.field === 'jobContentText' || col.data.field === 'adjustState') {
  432. if (!autoHeight) {
  433. col.showHint = true;
  434. col.data.wordWrap = false;
  435. } else {
  436. col.showHint = false;
  437. col.data.wordWrap = true;
  438. }
  439. }
  440. if(col.data.field ==='quantity'){
  441. col.showHint = true;
  442. }
  443. // for test digit. CSLAAAAA
  444. if (col.data.field.hasSubStr("totalFee"))
  445. col.data.formatter = MainTreeCol.getNumberFormatter(decimalObj.ration.totalPrice, false)
  446. else if (col.data.field.hasSubStr("unitFee"))
  447. col.data.formatter = MainTreeCol.getNumberFormatter(decimalObj.ration.unitPrice, false)
  448. else if (col.data.field == "quantity")
  449. col.data.formatter = MainTreeCol.getNumberFormatter(decimalObj.ration.quantity, false);
  450. });
  451. that.mainController = TREE_SHEET_CONTROLLER.createNew(that.project.mainTree, that.mainSpread.getActiveSheet(), that.project.projSetting.mainGridSetting);
  452. that.mainController.showTreeData();
  453. that.mainController.bind('refreshBaseActn', that.refreshBaseActn);
  454. that.mainController.bind(TREE_SHEET_CONTROLLER.eventName.beforeTreeSelectedChange, that.beforeMainTreeSelectedChange);
  455. that.mainController.bind(TREE_SHEET_CONTROLLER.eventName.treeSelectedChanged, that.treeSelectedChanged);
  456. that.mainSpread.bind(GC.Spread.Sheets.Events.LeaveCell, that.mainSpreadLeaveCell);
  457. that.mainSpread.bind(GC.Spread.Sheets.Events.EnterCell, that.mainSpreadEnterCell);
  458. that.mainSpread.bind(GC.Spread.Sheets.Events.EditEnded, that.mainSpreadEditEnded);
  459. that.mainSpread.bind(GC.Spread.Sheets.Events.RangeChanged, that.mainSpreadRangeChanged);
  460. that.mainSpread.bind(GC.Spread.Sheets.Events.SelectionChanged, that.amountAreaNumber);
  461. that.loadMainSpreadContextMenu();
  462. that.loadFocusLocation();
  463. let endTime = +new Date();
  464. console.log("加载完成-----"+endTime);
  465. }
  466. else {
  467. }
  468. });
  469. },
  470. loadMainSpreadContextMenu: function () {
  471. var project = this.project, spread = this.mainSpread, controller = this.mainController;
  472. $.contextMenu({
  473. selector: '#billsSpread',
  474. build: function ($trigger, e) {
  475. var target = SheetDataHelper.safeRightClickSelection($trigger, e, spread);
  476. controller.setTreeSelected(controller.tree.items[target.row]);
  477. return target.hitTestType === GC.Spread.Sheets.SheetArea.viewport || target.hitTestType === GC.Spread.Sheets.SheetArea.rowHeader;
  478. },
  479. items: {
  480. "insertRootBill": {
  481. name: "插入大项费用",
  482. icon: 'fa-sign-in',
  483. disabled: function () {
  484. //return project.mainTree.selected ? project.mainTree.selected.sourceType !== project.Bills.getSourceType() : false;
  485. },
  486. callback: function (key, opt) {
  487. ProjectController.addRootBill(project, controller);
  488. cbTools.refreshFormulaNodes();
  489. },
  490. visible: function(key, opt){
  491. return project.mainTree.selected&&project.mainTree.selected.parent==null;
  492. }
  493. },
  494. "insertFB": {
  495. name: "插入分部",
  496. icon: 'fa-sign-in',
  497. disabled: function () {
  498. let selected = project.mainTree.selected;
  499. if(selected.sourceType==project.Bills.getSourceType()){
  500. if(selected.data.type==billType.FB){
  501. return false;
  502. }
  503. if(isFlag(selected.data)&&selected.data.flagsIndex.fixed.flag==fixedFlag.SUB_ENGINERRING){//焦点行是分部分项工程
  504. if(selected.children.length>0){
  505. return selected.children[0].data.type==billType.FX;//焦点行是分部分项工程,且子项是分项
  506. }
  507. }
  508. }
  509. return true;//除了清单,其它类型都只读
  510. },
  511. callback: function (key, opt) {
  512. ProjectController.addFB(project, controller);
  513. cbTools.refreshFormulaNodes();
  514. },
  515. visible: function(key, opt){
  516. return project.Bills.isFBFX(project.mainTree.selected );//不属于分部分项的话隐藏
  517. }
  518. },
  519. "insertFX": {
  520. name: "插入分项",
  521. icon: 'fa-sign-in',
  522. disabled: function () {
  523. let selected = project.mainTree.selected;
  524. if(selected.sourceType==project.Bills.getSourceType()){
  525. if(selected.data.type==billType.FX){//焦点行是分项,有效显示
  526. return false
  527. }
  528. if(selected.data.type==billType.FB){//点行是分部,
  529. if(selected.children.length>0){//且有子项,子项是分部,灰显。
  530. return selected.children[0].data.type == billType.FB
  531. }else {
  532. return false;
  533. }
  534. }
  535. if(isFlag(selected.data)&&selected.data.flagsIndex.fixed.flag==fixedFlag.SUB_ENGINERRING){//焦点行是分部分项工程
  536. if(selected.children.length>0){
  537. return selected.children[0].data.type==billType.FB;//焦点行是分部分项工程,且子项是分部时灰显
  538. }else {
  539. return false
  540. }
  541. }
  542. }
  543. return true;//除了清单,其它类型都只读
  544. },
  545. callback: function (key, opt) {
  546. ProjectController.addFX(project, controller);
  547. cbTools.refreshFormulaNodes();
  548. },
  549. visible: function(key, opt){
  550. return project.Bills.isFBFX(project.mainTree.selected );//不属于分部分项的话隐藏
  551. }
  552. },
  553. "insertBills": {
  554. name: "插入清单",
  555. icon: 'fa-sign-in',
  556. disabled: function () {
  557. return project.mainTree.selected ? project.mainTree.selected.sourceType !== project.Bills.getSourceType() : false;
  558. },
  559. callback: function (key, opt) {
  560. if(project.mainTree.selected.data.type == billType.DXFY){
  561. if(project.mainTree.selected.data.calcBase&&project.mainTree.selected.data.calcBase!=""){
  562. alert("当前有基数计算不能插入子项。");
  563. return;
  564. }
  565. }
  566. ProjectController.addBills(project, controller);
  567. cbTools.refreshFormulaNodes();
  568. },
  569. visible: function(key, opt){
  570. return project.Bills.isFBFX(project.mainTree.selected)==true?false:true;//不属于分部分项的话隐藏
  571. }
  572. },
  573. "insertRation": {
  574. name: "插入定额",
  575. icon: 'fa-sign-in',
  576. disabled: function () {
  577. var selected = project.mainTree.selected;
  578. return project.Ration.addRationChecking(selected); // Vincent, 2018-01-02
  579. },
  580. callback: function (key, opt) {
  581. ProjectController.addRation(project, controller, rationType.ration);
  582. cbTools.refreshFormulaNodes();
  583. },
  584. visible: function(key, opt){
  585. var selected = project.mainTree.selected;
  586. if(selected.sourceType == ModuleNames.bills&&selected.data.type == billType.DXFY){
  587. return false
  588. }else {
  589. return true
  590. }
  591. }
  592. },
  593. "insertLJ": {
  594. name: "插入量价",
  595. icon: 'fa-sign-in',
  596. disabled: function () {
  597. var selected = project.mainTree.selected;
  598. return project.Ration.addRationChecking(selected); // Vincent, 2018-01-02
  599. },
  600. callback: function (key, opt) {
  601. ProjectController.addRation(project, controller, rationType.volumePrice);
  602. cbTools.refreshFormulaNodes();
  603. },
  604. visible: function(key, opt){
  605. var selected = project.mainTree.selected;
  606. if(selected.sourceType == ModuleNames.bills&&selected.data.type == billType.DXFY){
  607. return false
  608. }else {
  609. return true
  610. }
  611. }
  612. },
  613. "insertGLJ": {
  614. name: "插入工料机",
  615. icon: 'fa-sign-in',
  616. disabled: function () {
  617. var selected = project.mainTree.selected;
  618. return project.Ration.addRationChecking(selected); // Vincent, 2018-01-02
  619. },
  620. callback: function (key, opt) {
  621. var selected = project.mainTree.selected;
  622. if(selected.sourceType == ModuleNames.bills){
  623. if(selected.data.type == billType.FX||selected.data.type ==billType.BILL){
  624. if(selected.data.calcBase&&selected.data.calcBase!=""){
  625. alert("当前有基数计算不能插入子项。");
  626. return;
  627. }
  628. }
  629. }
  630. getGLJData('insert');// ProjectController.addRation(project, controller, rationType.volumePrice);
  631. cbTools.refreshFormulaNodes();
  632. },
  633. visible: function(key, opt){
  634. var selected = project.mainTree.selected;
  635. if(selected.sourceType == ModuleNames.bills&&selected.data.type == billType.DXFY){
  636. return false
  637. }else {
  638. return true
  639. }
  640. }
  641. },
  642. "spr1": '--------',
  643. "delete": {
  644. name: '删除',
  645. icon: 'fa-remove',
  646. disabled: function () {
  647. var selected = project.mainTree.selected;
  648. return !selected;
  649. },
  650. callback: function () {
  651. var selected = controller.tree.selected, parent = selected.parent;
  652. if (selected) {
  653. if (selected.sourceType === project.Bills.getSourceType()) {
  654. project.Bills.deleteBills(selected.source);
  655. controller.delete();
  656. } else if (selected.sourceType === project.Ration.getSourceType()) {
  657. project.Ration.delete(selected.source);
  658. controller.delete();
  659. };
  660. projectObj.converseCalculateBills(parent);
  661. cbTools.refreshFormulaNodes();
  662. }
  663. }
  664. },
  665. "spr2":'--------',
  666. "calculateAll_RationContent": {
  667. name: '造价计算',
  668. callback: function () {
  669. let changedNodes = project.calcProgram.calcAllNodes();
  670. project.calcProgram.saveNodes(changedNodes);
  671. }
  672. }
  673. }
  674. });
  675. },
  676. // 计算node及node的所有父项
  677. converseCalculateBills: function (node) {
  678. /* if (node) {
  679. let calc = new BillsCalcHelper(this.project);
  680. calc.calcNode(node, true);
  681. let cur = node, nodes = [];
  682. while (cur) {
  683. nodes.push(cur);
  684. cur = cur.parent;
  685. }
  686. this.mainController.refreshTreeNode(nodes, false);
  687. this.project.Bills.updateNodes(nodes, true);
  688. calc = null;
  689. }*/
  690. projectObj.project.calcProgram.calcAndSave(node);
  691. },
  692. // 计算全部清单
  693. /* calculateAll: function () {
  694. let calc = new BillsCalcHelper(this.project);
  695. calc.calcAll();
  696. this.mainController.showTreeData();
  697. this.project.Bills.updateAll();
  698. calc = null;
  699. }*/
  700. // 获取上次退出时的焦点位置
  701. loadFocusLocation: function() {
  702. const projectId = scUrlUtil.GetQueryString('project');
  703. let row = getLocalCache('lastRow:' + projectId);
  704. let col = getLocalCache('lastCol:' + projectId);
  705. if (row !== null && col !== null) {
  706. row = parseInt(row);
  707. col = parseInt(col);
  708. const sheet = this.mainSpread.getActiveSheet();
  709. sheet.setSelection(row, col, 1, 1);
  710. this.mainController.setTreeSelected(this.mainController.tree.items[row]);//触发树节点选中事件
  711. }
  712. },
  713. // 选中区域合计数字
  714. amountAreaNumber: function(e, info) {
  715. if (info.newSelections === undefined || info.newSelections.length <= 0) {
  716. return false;
  717. }
  718. const selectedArea = info.newSelections[0];
  719. const sheet = info.sheet;
  720. if (selectedArea.colCount <= 1 && selectedArea.rowCount <= 1) {
  721. return false;
  722. }
  723. // 获取鼠标位置
  724. let x = window.event.clientX;
  725. let y = window.event.clientY;
  726. // 匹配数字或小数
  727. const regular = /^([0-9]+[.]{1}[0-9]+)$|^([1-9]{1}\d*)$/;
  728. // 小数点最高位数
  729. let max = 0;
  730. let total = 0;
  731. let counter = 0;
  732. for (let col = selectedArea.col; col < (selectedArea.colCount + selectedArea.col); col++) {
  733. for (let row = selectedArea.row; row < (selectedArea.rowCount + selectedArea.row); row++) {
  734. const value = sheet.getCell(row, col).text();
  735. if (!regular.test(value)) {
  736. continue;
  737. }
  738. counter++;
  739. // 获取当前数据小数位数
  740. let pointPosition = value.toString().indexOf(".");
  741. pointPosition = pointPosition < 0 ? pointPosition : pointPosition + 1;
  742. const current = pointPosition > 0 ? value.toString().substring(pointPosition, value.toString().length).length : 0;
  743. max = current > max ? current : max;
  744. total += parseFloat(value);
  745. }
  746. }
  747. // 如果不为0则悬浮显示
  748. if (total > 0 && counter > 1) {
  749. const div = $('<div id="total-tips"><p>合计: <input type="text" id="total" readonly="readonly" style="border: none;"/></p><p><a href="javascript:void(0);">复制</a></p></div>');
  750. div.css({
  751. position: "absolute",
  752. border: "1px #C0C0C0 solid",
  753. background: "#fff",
  754. boxShadow: "1px 2px 5px rgba(0,0,0,0.4)",
  755. });
  756. div.children("p").css({
  757. textAlign: "center",
  758. padding: 8,
  759. marginBottom: 2,
  760. });
  761. div.children("p").first().css({
  762. borderBottom: "1px #C0C0C0 solid"
  763. });
  764. const totalString = total.toFixed(max);
  765. // input长度
  766. const inputWidth = totalString.length * 8;
  767. // 计算是否会超出显示范围
  768. const baseWidth = 48;
  769. const baseHeight = 81;
  770. const canvasWidth = $("#billsSpreadvp_vp").width();
  771. const canvasHeight = $("#billsSpreadvp_vp").height();
  772. x = x + baseWidth + inputWidth > canvasWidth ? x - baseWidth - inputWidth : x;
  773. y = y + baseHeight > canvasHeight ? y - baseHeight : y;
  774. div.css({
  775. left: x,
  776. top: y,
  777. });
  778. div.children().children("#total").width(inputWidth).val(totalString);
  779. $("body").append(div);
  780. // 用于判断是否要关闭窗体
  781. setTimeout(function() {
  782. isTotalShowing = true;
  783. }, 200);
  784. };
  785. }
  786. };
  787. // 点击合计框中的复制
  788. $("body").on("click", "#total-tips a", function() {
  789. const totalElement = $(this).parent().siblings("p").find("#total");
  790. totalElement.select();
  791. document.execCommand("Copy");
  792. $(this).text("已复制");
  793. });
  794. // 合计框点击其他位置关闭
  795. let isTotalShowing = false;
  796. $("body").on("click", function(e) {
  797. if (isTotalShowing && !$(e.target).is("#total-tips") && !$(e.target).is("#total-tips p") && !$(e.target).is("#total-tips p a")) {
  798. $("#total-tips").remove();
  799. isTotalShowing = false;
  800. }
  801. });
  802. $('#insert').click(function () {
  803. var controller = projectObj.mainController, project = projectObj.project;
  804. var selected = controller.tree.selected;
  805. if (!selected || selected.sourceType === project.Bills.getSourceType()) {
  806. ProjectController.addBills(project, controller);
  807. } else if (selected.sourceType === project.Ration.getSourceType()) {
  808. ProjectController.addRation(project, controller, selected.data.type);
  809. };
  810. cbTools.refreshFormulaNodes();
  811. });
  812. $('#delete').click(function () {
  813. var controller = projectObj.mainController, project = projectObj.project;
  814. var selected = controller.tree.selected, parent = selected.parent;
  815. if (selected) {
  816. if (selected.sourceType === project.Bills.getSourceType()) {
  817. project.Bills.deleteBills(selected.source);
  818. controller.delete();
  819. } else if (selected.sourceType === project.Ration.getSourceType()) {
  820. project.Ration.delete(selected.source);
  821. controller.delete();
  822. }else if(selected.sourceType==ModuleNames.ration_glj){
  823. project.ration_glj.updataOrdelete(selected.source);
  824. };
  825. projectObj.converseCalculateBills(parent);
  826. };
  827. cbTools.refreshFormulaNodes();
  828. });
  829. $('#upLevel').click(function () {
  830. var controller = projectObj.mainController, project = projectObj.project;
  831. var selected = controller.tree.selected, orgParent = selected.parent;
  832. if (selected && selected.sourceType === project.Bills.getSourceType()) {
  833. project.Bills.upLevelBills(selected.source);
  834. controller.upLevel();
  835. projectObj.converseCalculateBills(orgParent);
  836. }
  837. });
  838. $('#downLevel').click(function () {
  839. var controller = projectObj.mainController, project = projectObj.project;
  840. var selected = controller.tree.selected;
  841. if (selected && selected.sourceType === project.Bills.getSourceType()) {
  842. project.Bills.downLevelBills(selected.source);
  843. controller.downLevel();
  844. projectObj.converseCalculateBills(selected.parent);
  845. }
  846. });
  847. $('#upMove').click(function () {
  848. var controller = projectObj.mainController, project = projectObj.project;
  849. var selected = controller.tree.selected, pre = selected.preSibling, preSerialNo;
  850. if (selected.sourceType === project.Bills.getSourceType()) {
  851. project.Bills.upMoveBills(selected.source);
  852. controller.upMove();
  853. } else if (selected.sourceType === project.Ration.getSourceType()) {
  854. project.Ration.changePos(selected.source, selected.preSibling.source);
  855. controller.upMove();
  856. };
  857. });
  858. $('#downMove').click(function () {
  859. var controller = projectObj.mainController, project = projectObj.project;
  860. var selected = controller.tree.selected, next, nextSerialNo;
  861. if (selected) {
  862. if (selected.sourceType === project.Bills.getSourceType()) {
  863. project.Bills.downMoveBills(selected.source);
  864. controller.downMove();
  865. } else if (selected.sourceType === project.Ration.getSourceType()) {
  866. project.Ration.changePos(selected.source, selected.nextSibling.source);
  867. controller.downMove();
  868. };
  869. }
  870. });
  871. $('#poj-set').on('show.bs.modal', function () {
  872. let setCalcFlag = function (obj, val, curFlag) {
  873. obj.val(val);
  874. if (val === curFlag) {
  875. obj.attr('checked', true);
  876. } else {
  877. obj.removeAttr('checked');
  878. }
  879. }
  880. if (projectObj.project) {
  881. // let mode = projectObj.project.projSetting.billsCalcMode;
  882. // let settingConst = projectObj.project.projSetting.settingConst;
  883. let ft = projectObj.project.property.billsCalcMode ? projectObj.project.property.billsCalcMode : leafBillGetFeeType.rationContent;
  884. setCalcFlag($('#rationContent'), leafBillGetFeeType.rationContent, ft);
  885. setCalcFlag($('#rationPriceConverse'), leafBillGetFeeType.rationPriceConverse, ft);
  886. setCalcFlag($('#rationPrice'), leafBillGetFeeType.rationPrice, ft);
  887. setCalcFlag($('#billsPrice'), leafBillGetFeeType.billsPrice, ft);
  888. // mode = projectObj.project.projSetting.zanguCalcType;
  889. let zg = projectObj.project.property.zanguCalcMode ? projectObj.project.property.zanguCalcMode : zanguCalcType.common;
  890. setCalcFlag($('#zangu_common'), zanguCalcType.common, zg);
  891. setCalcFlag($('#zangu_gatherMaterial'), zanguCalcType.gatherMaterial, zg);
  892. }
  893. });
  894. $('#property_ok').click(function () {
  895. let project = projectObj.project,
  896. projectID = project.ID(),
  897. properties = {},
  898. labourCoes = {},
  899. rations = [],
  900. bills = [],
  901. mixDatas = {
  902. projectID: projectID,
  903. updateType: 'update',
  904. properties: properties,
  905. labourCoes: labourCoes,
  906. rations: rations,
  907. bills: bills
  908. },
  909. reCalcBills= false,
  910. reCalcRations= false;
  911. let b = parseInt($("input[name='calcFlag']:checked").val());
  912. if (b !== project.property.billsCalcMode) {
  913. properties['property.billsCalcMode'] = b;
  914. project.property.billsCalcMode = b;
  915. reCalcBills = true;
  916. };
  917. let zg = parseInt($("input[name='zangu']:checked").val());
  918. if (zg !== project.property.zanguCalcMode) {
  919. properties['property.zanguCalcMode'] = zg;
  920. project.property.zanguCalcMode = zg;
  921. reCalcBills = true;
  922. };
  923. //基本信息
  924. if(basicInfoView.toUpdate(basicInfoView.orgDatas, basicInfoView.datas)){
  925. properties['property.basicInformation'] = basicInfoView.toSaveDatas(basicInfoView.datas);
  926. }
  927. //工程特征
  928. if(projFeatureView.toUpdate(projFeatureView.orgDatas, projFeatureView.datas)){
  929. properties['property.projectFeature'] = projFeatureView.toSaveDatas(projFeatureView.datas);
  930. }
  931. //清单工程量精度
  932. let newBillsDecimalDatas = billsDecimalView.toBillsDecimalDatas(billsDecimalView.cache);
  933. if(billsDecimalView.toUpdate(billsQuanDecimal.datas, newBillsDecimalDatas)){
  934. billsQuanDecimal.datas = newBillsDecimalDatas;
  935. reCalcBills = true;
  936. properties['property.billsQuantityDecimal'] = newBillsDecimalDatas;
  937. }
  938. //小数位数
  939. let updateDecimal = m_getDecimalData($('input', '#poj-settings-decimal'));
  940. if(toUpdateDecimal(decimalObj, updateDecimal)) {
  941. let frChanged = decimalObj.feeRate != updateDecimal.feeRate;
  942. setDecimal(decimalObj, updateDecimal);
  943. if (frChanged) {
  944. project.calcProgram.compileAllTemps()
  945. }
  946. reCalcRations = true;
  947. reCalcBills = true;
  948. properties['property.decimal'] = updateDecimal;
  949. }
  950. // 呈现选项
  951. projDisplayView.updateChecking(properties);
  952. if(calcOptions.updateChecking(properties)){
  953. reCalcRations = true;
  954. }
  955. // 人工系数
  956. if (labourCoeView.needSave()){
  957. let libID = $("#std_labour_coe_files").children("option:selected").val();
  958. let libName = $("#std_labour_coe_files").children("option:selected").text();
  959. labourCoes.updateData = {libID: libID, libName: libName, newItemArr: labourCoeView.needUpdateDatas};
  960. project.labourCoe.refreshData(labourCoes.updateData); // 全编译所用到的人工系数来自project.labourCoe,所以必须先刷新project.labourCoe
  961. project.calcProgram.compileAllTemps();
  962. reCalcRations = true;
  963. reCalcBills = true;
  964. };
  965. // 重新计算树节点
  966. let changedNodes = [];
  967. if (reCalcRations) // 定额动,全部计算。
  968. changedNodes = project.calcProgram.calcAllNodes(calcAllType.catAll)
  969. else if (reCalcBills) // 清单动,可以只计算清单。
  970. changedNodes = project.calcProgram.calcAllNodes(calcAllType.catBills);
  971. if (changedNodes.length > 0) {
  972. for (let node of changedNodes){
  973. let data = {
  974. // projectID: projectID,
  975. ID: node.data.ID,
  976. quantity: node.data.quantity,
  977. calcBase: node.data.calcBase,
  978. calcBaseValue: node.data.calcBaseValue,
  979. marketUnitFee: node.data.marketUnitFee,
  980. marketTotalFee: node.data.marketTotalFee,
  981. fees: node.data.fees,
  982. feeRate: node.data.feeRate,
  983. feeRateID: node.data.feeRateID
  984. };
  985. if (node.sourceType == 'ration'){
  986. rations.push(data);
  987. }
  988. else if (node.sourceType == 'bills'){
  989. bills.push(data);
  990. };
  991. };
  992. };
  993. // for test.
  994. /* console.log(mixDatas);
  995. return; */
  996. function hasMixData() {
  997. return Object.keys(mixDatas.properties).length > 0 ||
  998. mixDatas.labourCoes.updateData || mixDatas.rations.length > 0 || mixDatas.bills.length > 0;
  999. }
  1000. if(hasMixData()){
  1001. CommonAjax.post('/pm/api/updateMixDatas', {user_id: userID, mixDataArr: mixDatas}, function (rstData) {
  1002. /* if (changedNodes.length > 0) {
  1003. for (let node of changedNodes){delete node.changed};
  1004. };
  1005. if (mixDatas.labourCoes.updateData) labourCoeView.refresh();*/
  1006. window.location.href = '/main?project=' + projectID;
  1007. });
  1008. }
  1009. });