project_view.js 49 KB

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