divide_view.js 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468
  1. let divideObj = {
  2. divideSetting:{//“类别”、“清单编号”、“名称”、“单位”、“工程量”、“单价”、“金额”、“预算价”、“取费类别”。
  3. header: [
  4. {headerName: "类别", headerWidth: 60, dataCode: "itemType", hAlign:'center', dataType: "String"},
  5. {headerName: "清单编号", headerWidth: 130, dataCode: "code", hAlign:'left', dataType: "String", formatter: "@"},
  6. {headerName: "名称", headerWidth: 160, dataCode: "name", dataType: "String",formatter: "@"},
  7. {headerName: "单位", headerWidth: 100, dataCode: "unit", dataType: "String",cellType:'comboBox',editable:true,options:['m', 'm2', 'm3', 'km', 't', 'kg', '台班', '工日', '昼夜', '元', '项', '处', '个', '件',
  8. '根', '组', '系统', '台', '套', '株', '丛', '缸', '支', '只', '块', '座', '对', '份', '樘', '攒', '榀']},
  9. {headerName: "工程量", headerWidth: 80, dataCode: "quantity", dataType: "Number",validator:"number",getText:"notZero"},
  10. {headerName: "单价", headerWidth: 100, dataCode: "unitPrice", dataType: "Number",validator:"number",getText:"notZero"},
  11. {headerName: "金额", headerWidth: 100, dataCode: "totalPrice", dataType: "Number",validator:"number",getText:"notZero"},
  12. {headerName: "预算价", headerWidth: 100, dataCode: "marketPrice", dataType: "Number",validator:"number",getText:"notZero"},
  13. {headerName: "取费类别", headerWidth: 60, dataCode: "programID", hAlign:'center', dataType: "String",cellType:'comboBox',editorValueType:true}
  14. ],
  15. view:{
  16. lockColumns: [0],
  17. colHeaderHeight:30
  18. },
  19. getText:{
  20. notZero:function (item,value) {
  21. return value == 0?"":value;
  22. }
  23. },
  24. treeCol:1
  25. },
  26. billsSetting:{
  27. header: [//“编号”、“名称”、“分摊比例”、“分摊金额”、“原始金额”、“合计”、“金额”。
  28. {headerName: "编号", headerWidth: 130, dataCode: "code", hAlign:'left', dataType: "String"},
  29. {headerName: "名称", headerWidth: 160, dataCode: "name", dataType: "String"},
  30. {headerName: "分摊比例", headerWidth: 80, dataCode: "divideRate", dataType: "Number",validator:"number"},
  31. {headerName: "分摊金额", headerWidth: 100, dataCode: "dividePrice", dataType: "Number",validator:"number"},
  32. {headerName: "原始金额", headerWidth: 100, dataCode: "originalPrice", dataType: "Number",validator:"number"},
  33. {headerName: "合计", headerWidth: 100, dataCode: "totalPrice", dataType: "Number",validator:"number"},
  34. {headerName: "金额", headerWidth: 60, dataCode: "price", dataType: "Number",validator:"number"}
  35. ],
  36. view:{
  37. lockColumns: [0,1,2,3,4,5,6],
  38. colHeaderHeight:30
  39. }
  40. },
  41. divideSpread:null,
  42. billsSpread:null,
  43. initSpread:function(){
  44. this.initDivideSpread();
  45. this.initBillsSpread();
  46. if(!projectReadOnly){
  47. this.initDivideRightClick();
  48. }else {
  49. disableSpread(this.divideSpread);
  50. disableSpread(this.billsSpread);
  51. }
  52. },
  53. showDatas:function(){
  54. this.showDivideDatas();
  55. this.showBillsDatas();
  56. },
  57. refreshViews:function(){
  58. if(!$('#divide_main').is(':visible')) return;
  59. if(this.divideSpread) this.divideSpread.refresh();
  60. if(this.billsSpread) this.billsSpread.refresh();
  61. },
  62. showDivideDatas:function(){
  63. if(!$('#divide_main_sheet').is(':visible')) return;
  64. this.divideDatas = getDivideDatas();
  65. this.divideSheet.setRowCount(this.divideDatas.length);
  66. let fheader= _.find(this.divideSetting.header,{'dataCode': "programID"})
  67. fheader.options = this.getProgramOptions();
  68. sheetCommonObj.showTreeData(this.divideSheet, this.divideSetting,this.divideDatas);
  69. function getDivideDatas(){
  70. let datas = [];
  71. let divideSetting = projectObj.project.divide_setting.datas
  72. let parentMap={};
  73. //divideSetting.divideList = [{ID:"1",code:"1",ParentID:-1,collapsed:false},{ID:"2",code:"2",ParentID:"1",collapsed:false}]
  74. divideSetting.divideList = _.sortByAll(divideSetting.divideList,['seq'])
  75. for(let d of divideSetting.divideList){
  76. parentMap[d.ParentID]?parentMap[d.ParentID].push(d):parentMap[d.ParentID]=[d]
  77. }
  78. if(parentMap[-1]&&parentMap[-1].length>0){
  79. for(let r of parentMap[-1]){
  80. r.collapsed = false;
  81. datas.push(r)
  82. if(parentMap[r.ID] && parentMap[r.ID].length > 0){
  83. for(let c of parentMap[r.ID]){
  84. datas.push(c);
  85. }
  86. }
  87. }
  88. }
  89. return datas;
  90. }
  91. },
  92. getProgramOptions:function(){
  93. let names = projectObj.project.calcProgram.compiledTemplateNames;
  94. let map = projectObj.project.calcProgram.compiledTemplateMaps;
  95. let options = [];
  96. for(let n of names){
  97. options.push({text:n,value:map[n]});
  98. }
  99. return options;
  100. },
  101. divideEditChecking:function(row,col){//return false表示不能编辑
  102. let me = this;
  103. let data = me.divideDatas[row],setting = me.divideSetting;
  104. let dataCode = setting.header[col].dataCode;
  105. if(data){
  106. if(data.itemType == "分摊项"){
  107. //金额,预算价,取费类别不可编辑
  108. if(dataCode =="totalPrice" || dataCode =="marketPrice" || dataCode =="programID" ) return false;
  109. //有子项时,单价不可编辑
  110. if(_.find(me.divideDatas,{ParentID:data.ID})) return false;
  111. }
  112. if( data.itemType == "量价" && dataCode =="totalPrice") return false//量价:“金额”不可编辑。
  113. if(data.itemType == "定额"){ //定额:“单位”、“单价”、“金额”、“预算价”
  114. return !(dataCode == "unit"||dataCode == "unitPrice"||dataCode == "totalPrice"||dataCode == "marketPrice")
  115. }
  116. return true
  117. }
  118. return false;
  119. },
  120. showBillsDatas:function(){
  121. if(!$('#divide_bills_sheet').is(':visible')) return;
  122. this.billsDatas = getBillsDatas();
  123. sheetCommonObj.showData(this.billsSheet, this.billsSetting,this.billsDatas);
  124. this.billsSheet.setRowCount(this.billsDatas.length);
  125. function getBillsDatas(){
  126. return [];
  127. }
  128. },
  129. getSelectedItem:function(){
  130. let me = this,data=null;
  131. let sel = me.divideSheet.getSelections()[0];
  132. let srow = sel.row == -1||sel.row == ""?0:sel.row;
  133. if(me.divideDatas.length>srow){
  134. data = me.divideDatas[srow];
  135. }
  136. return data;
  137. },
  138. addNewItems:async function(itemType,type){
  139. let newItem = {
  140. ID:uuid.v1(),
  141. itemType:itemType,
  142. ParentID:-1
  143. };
  144. if(type) newItem.type = type;
  145. let selected = this.getSelectedItem();
  146. if(itemType!="分摊项"){//当插入定额或者量价时:
  147. newItem.ParentID = selected.itemType == "分摊项"?selected.ID:selected.ParentID;
  148. }
  149. if(itemType == "量价") newItem['programID'] = projectObj.project.calcProgram.compiledTemplateMaps["费率为0"];
  150. let [seq,datas] = this.getNewSeqs(selected,newItem);
  151. newItem.seq = seq;
  152. datas.push({type:"add",doc:newItem});
  153. await this.updateItem(datas);
  154. },
  155. deleteItem:async function(){
  156. let selected = this.getSelectedItem();
  157. let subItems = [];
  158. if(selected.itemType == "分摊项"){
  159. //检查是否执行了分摊,如果已经执行了,要取消分摊才能删除
  160. // todo
  161. for(let s of this.divideDatas){
  162. if(s.ParentID == selected.ID) subItems.push({type:"delete",ID:s.ID});
  163. }
  164. }
  165. subItems.push({type:"delete",ID:selected.ID});
  166. await this.updateItem(subItems);
  167. },
  168. getNewSeqs:function(selected,newItem){
  169. let seq = 1,datas=[],refreshSeq = false;
  170. for(let d of this.divideDatas){
  171. if(d.ParentID == newItem.ParentID){
  172. if(refreshSeq == true){
  173. if(d.seq == seq || d.seq > seq) datas.push({ID:d.ID,type:'update',doc:{seq:d.seq+1}})
  174. }
  175. if(selected && d.ID == selected.ID){
  176. seq = d.seq + 1;
  177. refreshSeq = true;
  178. }
  179. }
  180. }
  181. return [seq,datas]
  182. },
  183. updateItem: async function(datas){
  184. let divideSetting = projectObj.project.divide_setting.datas
  185. try {
  186. $.bootstrapLoading.start();
  187. let result = ajaxPost("/divide/updateItem",{ID:divideSetting.ID,updateDatas:datas})
  188. for(let d of datas){
  189. if(d.type == "add"){
  190. divideSetting.divideList.push(d.doc);
  191. }
  192. if(d.type == "delete"){
  193. _.remove(divideSetting.divideList,{'ID':d.ID});
  194. }
  195. if(d.type == "update"){
  196. let item = _.find(divideSetting.divideList,{'ID':d.ID});
  197. if(item) gljUtil.updateProperty(item,d.doc);
  198. }
  199. }
  200. } catch (error) {
  201. console.log(error)
  202. }
  203. $.bootstrapLoading.end();
  204. this.showDatas();
  205. },
  206. initDivideSpread:function(){
  207. if(this.divideSpread) return;
  208. this.divideSpread = SheetDataHelper.createNewSpread($("#divide_main_sheet")[0]);
  209. sheetCommonObj.spreadDefaultStyle(this.divideSpread);
  210. this.divideSheet = this.divideSpread.getSheet(0);
  211. sheetCommonObj.initSheet(this.divideSheet, this.divideSetting, 2);
  212. this.divideSheet.bind(GC.Spread.Sheets.Events.EditStarting,this.onDivideEditStarting);
  213. this.divideSheet.bind(GC.Spread.Sheets.Events.ValueChanged,this.onDivideValueChange)
  214. /* ;
  215. this.divideSheet.bind(GC.Spread.Sheets.Events.RangeChanged, this.onCoeRangeChanged); */
  216. this.divideSheet.bind(GC.Spread.Sheets.Events.SelectionChanged, function (e,args) {
  217. args.sheet.repaint();
  218. });
  219. SheetDataHelper.protectdSheet(this.divideSheet);
  220. },
  221. onDivideEditStarting:function (sender, args) {
  222. let me = divideObj;
  223. let row = args.row;
  224. let col = args.col;
  225. if(me.divideEditChecking(row,col)==false){
  226. args.cancel = true;
  227. }
  228. },
  229. onDivideValueChange:async function(sender,info){
  230. let me = divideObj,row = info.row, col = info.col;
  231. let setting = me.divideSetting;
  232. let dataCode = setting.header[col].dataCode;
  233. let recode = me.divideDatas[row];
  234. let value = info.newValue;
  235. if(info.newValue === undefined ){
  236. return;
  237. }
  238. if(value && !sheetCommonObj.checkData(col,setting,value)) {
  239. alert('输入的数据类型不对,请重新输入!');
  240. return me.showDatas();
  241. }
  242. await me.updateItem([me.getDivideUpdateData(recode,dataCode,value)]);
  243. },
  244. getDivideUpdateData:function(recode,dataCode,value){
  245. let doc = {};
  246. if(recode.itemType == "分摊项"){
  247. if(dataCode == "quantity"){
  248. value = scMathUtil.roundForObj(value,3);
  249. let tu = scMathUtil.roundForObj(recode.unitPrice&&recode.unitPrice!=""?recode.unitPrice:0,getDecimal("bills.unitPrice"))
  250. doc["totalPrice"] = scMathUtil.roundForObj(value * tu,getDecimal("bills.totalPrice"))
  251. }
  252. if(dataCode == "unitPrice"){
  253. value = scMathUtil.roundForObj(value,getDecimal("bills.unitPrice"));
  254. let tq = scMathUtil.roundForObj(recode.quantity && recode.quantity!=""?recode.quantity:0,3)
  255. doc["totalPrice"] = scMathUtil.roundForObj(value * tq,getDecimal("bills.totalPrice"))
  256. }
  257. }
  258. doc[dataCode]=value;
  259. return {ID:recode.ID,type:'update',doc:doc}
  260. },
  261. initBillsSpread:function(){
  262. if(this.billsSpread) return;
  263. this.billsSpread = SheetDataHelper.createNewSpread($("#divide_bills_sheet")[0]);
  264. sheetCommonObj.spreadDefaultStyle(this.billsSpread);
  265. this.billsSheet = this.billsSpread.getSheet(0);
  266. sheetCommonObj.initSheet(this.billsSheet, this.billsSetting, 0);
  267. SheetDataHelper.protectdSheet(this.divideSheet);
  268. },
  269. initDivideRightClick:function(){
  270. //如果当前行是无组成物的“普通材料”、“绿化苗木”、“外购砼构件”、“商品混凝土”、“商品砂浆”,则右键“添加计算材料”按钮有效。
  271. let me = this;
  272. $.contextMenu({
  273. selector: '#divide_main_sheet',
  274. build: function ($trigger, e) {
  275. me.rightClickTarget = SheetDataHelper.safeRightClickSelection($trigger, e, me.divideSpread);
  276. return me.rightClickTarget.hitTestType === GC.Spread.Sheets.SheetArea.viewport ||
  277. me.rightClickTarget.hitTestType === GC.Spread.Sheets.SheetArea.rowHeader;
  278. },
  279. items: {
  280. "insertItem": {
  281. name: "插入分摊项",
  282. icon: 'fa-sign-in',
  283. disabled: function () {
  284. let selected = me.getSelectedItem();
  285. if(selected && selected.itemType !="分摊项") return true;
  286. return false;
  287. },
  288. callback: function (key, opt) {
  289. me.addNewItems("分摊项");
  290. }
  291. },
  292. "insertRation": {
  293. name: "插入定额",
  294. icon: 'fa-sign-in',
  295. disabled: function () {
  296. return me.rightClickTarget.row === undefined
  297. },
  298. callback: function (key, opt) {
  299. me.addNewItems("定额");
  300. }
  301. },
  302. "insertVolumePrice": {
  303. name: "插入量价",
  304. icon: 'fa-sign-in',
  305. disabled: function () {
  306. return me.rightClickTarget.row === undefined;
  307. },
  308. items:{
  309. insertLabour:{
  310. name: "人工",
  311. icon: 'fa-sign-in',
  312. callback:function(key){
  313. me.addNewItems("量价",1);
  314. }
  315. },
  316. insertMaterial:{
  317. name:"材料" ,
  318. icon: 'fa-sign-in',
  319. callback:function(key){
  320. me.addNewItems("量价",201);
  321. }
  322. },
  323. insertMachine:{
  324. name:"机械" ,
  325. icon: 'fa-sign-in',
  326. callback:function(key){
  327. me.addNewItems("量价",301);
  328. }
  329. }
  330. }
  331. },
  332. "removeItem": {
  333. name: "删除",
  334. icon: 'fa-trash-o',
  335. disabled: function () {
  336. return me.rightClickTarget.row === undefined;;
  337. },
  338. callback: function (key, opt) {
  339. me.deleteItem();
  340. }
  341. }
  342. }
  343. });
  344. },
  345. getDivideResizeEles:function () {
  346. let divideResizeEles = {};
  347. divideResizeEles.eleObj = {
  348. module: 'divide',
  349. resize: $('#divideResize'),
  350. top: $('#divide_top'),
  351. topSpread: $('#divide_main_sheet'),
  352. bottom: $('#divideBottom'),
  353. bottomSpread: [$('#sub_sheet')]
  354. };
  355. divideResizeEles.limit = {
  356. min: 150,
  357. max: `$(window).height()-$('.header').height()-$('#divideToolsBar').height()-150-5`,//5: resize.height()
  358. notTopSpread: 0,
  359. notBottomSpread: 0,
  360. bottomNav:`$('#divideBottom ul').height()`,
  361. totalHeight: `$(window).height()-$('.header').height()-$('#divideToolsBar').height()-5`
  362. };
  363. return divideResizeEles;
  364. }
  365. }
  366. function loadDivideHeight(){
  367. if(!$('#divide_main').is(':visible')) return;
  368. let me = divideObj;
  369. let divideResizeEles = me.getDivideResizeEles();
  370. SlideResize.loadVerticalHeight(divideResizeEles.eleObj.module, divideResizeEles.eleObj, divideResizeEles.limit, function () {
  371. me.refreshViews();
  372. });
  373. }
  374. $('#tab_divide').on('shown.bs.tab', function (e) {
  375. sessionStorage.setItem('mainTab', '#tab_divide');
  376. loadDivideHeight();
  377. divideObj.initSpread();
  378. divideObj.showDatas();
  379. })
  380. $('.divide-nav-link').bind('click', function () {//$('.side-tabs ul li a').bind 2018-11-23 使用更多标签,所以不能这样绑定事件了
  381. var tab = $(this), tabPanel = $(tab.attr('relaPanel'));
  382. if(tab.hasClass('disabled')){
  383. return;
  384. }
  385. let setActiveTab = tab;
  386. if (!(setActiveTab.hasClass('active')&&tabPanel.is(":visible"))) {
  387. $('.divide_side-tabs ul li a').removeClass('active');
  388. setActiveTab.addClass('active');
  389. showDivideSide(tabPanel, true, tab.attr('id'));
  390. } else {
  391. setActiveTab.removeClass('active');
  392. showDivideSide(tabPanel, false, tab.attr('id'));
  393. }
  394. divideObj.refreshViews();
  395. });
  396. //定额库上下拖动
  397. let dividRationLibResizeEles = {};
  398. dividRationLibResizeEles.eleObj = {
  399. module: 'divide_de',
  400. resize: $('#divide_deResize'),
  401. top: $('#divide_deTopDiv'),
  402. topSpread: $('#divide_stdRationChapter'),
  403. bottom: $('#divide_deBottomDiv'),
  404. bottomSpread: $('#divide_stdSectionRations')
  405. };
  406. dividRationLibResizeEles.limit = {
  407. min: 150,
  408. max: `$(window).height()-$('.header').height()-$('.toolsbar').height()-$('#deToolsBar').height()-150-5`,//5: resize.height()
  409. notTopSpread: 0,
  410. notBottomSpread: 0,
  411. totalHeight: `$(window).height()-$('.header').height()-$('.toolsbar').height()-$('#divide_deToolsBar').height()-5`
  412. };
  413. function showDivideSide(tabPanel, show, id){
  414. let divideSideResizeEles = {};
  415. divideSideResizeEles.eleObj = {
  416. module: 'divideRationLibTab',
  417. resize: $('#divideSideResize'),
  418. parent: $('#divideRow'),
  419. left: $('#divide_main'),
  420. right: $('#divide_side')
  421. };
  422. divideSideResizeEles.limit = {
  423. min: 150,
  424. max: `$('#divideRow').width()-150`
  425. };
  426. divideSideResizeEles.eleObj.module = id;
  427. if (show) {
  428. //刚打开各库时的默认比例
  429. divideSideResizeEles.eleObj.left.css('width', '66.666667%');
  430. divideSideResizeEles.eleObj.right.css('width', '33.333333%');
  431. SlideResize.setResizeWidth(divideSideResizeEles.eleObj.resize);
  432. $('.divide_side .tab-pane').hide();
  433. tabPanel.show();//locateTab要等div显示后才执行刷新操作
  434. //加载打开的库与主界面的宽度比
  435. SlideResize.loadHorizonWidth(id, [divideSideResizeEles.eleObj.resize], [divideSideResizeEles.eleObj.left, divideSideResizeEles.eleObj.right], function(){
  436. if (id === 'stdRationTab') {//加载定额库内部上下高度
  437. SlideResize.loadVerticalHeight(dividRationLibResizeEles.eleObj.module, dividRationLibResizeEles.eleObj, dividRationLibResizeEles.limit, function () {
  438. //sheetCommonObj.setColumnWidthByRate($('#divide_stdRationChapter').width() - 30, rationLibObj.rationChapterSpread, rationLibObj.rationChapterTreeSetting.cols);
  439. });
  440. }
  441. });
  442. } else {
  443. divideSideResizeEles.eleObj.left.css('width', '100%');
  444. divideSideResizeEles.eleObj.right.css('width', '0%');
  445. tabPanel.hide();
  446. }
  447. }