quantity_detail.js 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517
  1. /**
  2. * Created by Mai on 2017/4/1.
  3. */
  4. var quantity_detail = {
  5. createNew: function (project) {
  6. // 用户定义private方法
  7. var tools = {};
  8. // 所有通过this访问的属性,都不应在此单元外部进行写入操作
  9. var quantity_detail = function (proj) {
  10. this.gljTree = cacheTree.createNew(this);
  11. // this.project = proj;
  12. this.datas = [];
  13. var sourceType = ModuleNames.quantity_detail;
  14. this.getSourceType = function () {
  15. return sourceType;
  16. }
  17. proj.registerModule(ModuleNames.quantity_detail, this);
  18. this.temList=[];
  19. };
  20. // prototype用于定义public方法
  21. quantity_detail.prototype.loadData = function (datas) {
  22. this.datas = datas;
  23. };
  24. // 提交数据后返回数据处理
  25. quantity_detail.prototype.doAfterUpdate = function(err, data){
  26. if(!err){
  27. if(data.updateTpye=='ut_update'){
  28. this.refreshAfterUpdate(data);
  29. }else if(data.updateTpye=='ut_delete'){
  30. this.refreshAfterDelete(data);
  31. } else {
  32. this.refreshAfterSave(data);
  33. }
  34. }else {
  35. alert(err.message);
  36. this.refreshSheetData();
  37. }
  38. };
  39. quantity_detail.prototype.refreshAfterSave=function(data){
  40. var me = this;
  41. if(data.hasOwnProperty('resort')){
  42. this.resortData(data.doc,1);
  43. _.forEach(data.update_task,function (item) {
  44. me.refreshEachItme(item.query,item.doc);
  45. })
  46. this.datas.push(data.doc);
  47. }else {
  48. this.datas.push(data);
  49. }
  50. this.refreshSheetData();
  51. };
  52. quantity_detail.prototype.resortData=function(data,req){
  53. for(var i =0;i<gljOprObj.detailData.length;i++){
  54. var item = gljOprObj.detailData[i];
  55. if(item.seq>=data.seq){
  56. item.seq=item.seq+req;
  57. }
  58. }
  59. };
  60. quantity_detail.prototype.refreshAfterUpdate=function(data){
  61. var me = this;
  62. var filter_object;
  63. if(data.hasOwnProperty('refreshList')){
  64. _.forEach(data.refreshList,function (item) {
  65. filter_object= me.refreshEachItme(item.query,item.doc);
  66. })
  67. }else {
  68. filter_object = me.refreshEachItme(data.query,data.doc);
  69. }
  70. var showList = _.filter(this.datas,filter_object);
  71. gljOprObj.detailData=showList;
  72. gljOprObj.detailData=_.sortBy(gljOprObj.detailData,'seq');
  73. this.refreshSheetData();
  74. };
  75. quantity_detail.prototype.refreshEachItme = function(query,doc){
  76. var detail_list = this.datas;
  77. var detail_index= _.findIndex(detail_list,function(detail){
  78. return detail.ID==query.ID;
  79. })
  80. _.forEach(doc, function(n, key) {
  81. detail_list[detail_index][key] = n;
  82. });
  83. var filter_object;
  84. if(detail_list[detail_index].hasOwnProperty('rationID')){
  85. filter_object={'rationID':detail_list[detail_index].rationID};
  86. }else {
  87. filter_object={'billID':detail_list[detail_index].billID};
  88. }
  89. return filter_object;
  90. };
  91. quantity_detail.prototype.refreshAfterDelete=function(data){
  92. var me = this;
  93. if(data.doc.seq != gljOprObj.detailData.length - 1){
  94. this.resortData(data.doc,-1);
  95. }
  96. _.forEach(data.update_task,function (item) {
  97. me.refreshEachItme(item.query,item.doc);
  98. });
  99. _.remove(this.datas,{ID:data.doc.ID});
  100. this.refreshSheetData();
  101. };
  102. quantity_detail.prototype.refreshSheetData=function () {
  103. gljOprObj.showQuantityDetailData();
  104. };
  105. quantity_detail.prototype.saveQuantityDetail=function (args,dataCode,selected) {
  106. var me = this;
  107. var doc={};
  108. var selected = selected?selected:projectObj.project.mainTree.selected;
  109. if(selected.sourceType==ModuleNames.ration){
  110. doc.rationID=selected.data.ID;
  111. }
  112. if(selected.sourceType==ModuleNames.bills){
  113. doc.billID=selected.data.ID;
  114. }
  115. doc.projectID = selected.data.projectID;
  116. doc[dataCode]=args.editingText;
  117. doc.seq=args.row;
  118. if(dataCode=='regex'){
  119. if(!this.regexChecking(args.editingText)||!this.referenceChecking(args.editingText,args.row,doc)){
  120. gljOprObj.showQuantityDetailData();
  121. return;
  122. }
  123. doc.refreshQuantity=true;
  124. if(!selected.data.hasOwnProperty('isFromDetail')||selected.data.isFromDetail==0){
  125. var c = confirm("确定要使用工程量明细替换原工程量吗?");
  126. if(!c){
  127. doc.refreshQuantity=false;
  128. }
  129. }
  130. }
  131. var url="";
  132. $.bootstrapLoading.start();
  133. if(args.hasOwnProperty("insertRecode")){//右键插入或者是通过直接编辑保存
  134. url = "/quantity_detail/insertRecode";
  135. }else{
  136. url = "/quantity_detail/save";
  137. }
  138. var callback = function (data) {
  139. if(doc.refreshQuantity==false){//清空数据
  140. me.cleanQuantityDetail();
  141. }else {
  142. data.newRecord?me.refreshAfterSave(data.newRecord):me.refreshAfterSave(data);
  143. data.node?gljOprObj.refreshTreeNode(data.node):"";
  144. //gljOprObj.detailSheet.setActiveCell(0,0);
  145. //gljOprObj.detailSheet.clearSelection();
  146. }
  147. $.bootstrapLoading.end();
  148. }
  149. CommonAjax.post(url,doc,callback,function () {
  150. $.bootstrapLoading.end();
  151. });
  152. };
  153. quantity_detail.prototype.insertQuantityDetail = function (row) {
  154. var args = {
  155. row:row,
  156. editingText:1
  157. }
  158. if(row < gljOprObj.detailData.length){
  159. args.insertRecode = true;
  160. }
  161. this.saveQuantityDetail(args,'isSummation');
  162. };
  163. quantity_detail.prototype.deleteQuantityDetail = function (row) {
  164. var me = this;
  165. var deleteable = this.checkReference(row);
  166. if(deleteable){
  167. var recode = gljOprObj.detailData[row];
  168. $.bootstrapLoading.start();
  169. var callback=function (result) {
  170. me.refreshAfterDelete(result.data);
  171. result.node?gljOprObj.refreshTreeNode(result.node):"";
  172. $.bootstrapLoading.end();
  173. }
  174. CommonAjax.post("/quantity_detail/deleteRecode",recode,callback,function () {
  175. $.bootstrapLoading.end();
  176. });
  177. }else {
  178. alert("当前行已被引用,不可删除。");
  179. }
  180. };
  181. quantity_detail.prototype.checkReference = function (row) {
  182. var deleteable = true;
  183. for(var i =0;i<gljOprObj.detailData.length;i++){
  184. var item = gljOprObj.detailData[i];
  185. if(_.includes(item.referenceIndexs,row+1)){
  186. deleteable = false;
  187. break;
  188. }
  189. }
  190. return deleteable;
  191. };
  192. quantity_detail.prototype.moveDown = function (row) {
  193. this.swapRow(row);
  194. };
  195. quantity_detail.prototype.moveUp = function (row) {
  196. this.swapRow(row-1);
  197. };
  198. quantity_detail.prototype.swapRow = function (preRow) {
  199. var me = this;
  200. var update_task = [];
  201. var a_row = gljOprObj.detailData[preRow];//
  202. var b_row = gljOprObj.detailData[preRow +1];//
  203. var temA = a_row.seq;
  204. var temB = b_row.seq;
  205. a_row.seq = temB;
  206. update_task.push({query:{ID:a_row.ID,projectID:a_row.projectID},doc:{seq:a_row.seq}});
  207. b_row.seq = temA;
  208. update_task.push({query:{ID:b_row.ID,projectID:b_row.projectID},doc:{seq:b_row.seq}});
  209. gljOprObj.detailData.forEach(function (item) {
  210. if(_.includes(item.referenceIndexs,temA+1)||_.includes(item.referenceIndexs,temB+1)){
  211. var regex = item.regex;
  212. for (var i=0;i<item.referenceIndexs.length;i++){
  213. if(item.referenceIndexs[i]==temA+1){
  214. regex = me.replaceAll('C'+item.referenceIndexs[i],'B'+(temB+1),regex);
  215. regex = me.replaceAll('c'+item.referenceIndexs[i],'b'+(temB+1),regex);
  216. item.referenceIndexs[i]=temB+1;
  217. }else if(item.referenceIndexs[i]==temB+1){
  218. regex = me.replaceAll('C'+item.referenceIndexs[i],'B'+(temA+1),regex);
  219. regex = me.replaceAll('c'+item.referenceIndexs[i],'b'+(temA+1),regex);
  220. item.referenceIndexs[i]=temA+1;
  221. }
  222. }
  223. regex = me.replaceAll('B','C',regex);
  224. regex = me.replaceAll('b','c',regex);
  225. update_task.push({query:{ID:item.ID,projectID:item.projectID},doc:{regex:regex,referenceIndexs:item.referenceIndexs}});
  226. }
  227. })
  228. me.commonUpdate("/quantity_detail/swapRow",update_task);
  229. };
  230. quantity_detail.prototype.replaceAll=function(FindText, RepText,str) {
  231. let regExp = new RegExp(FindText, "g");
  232. return str.replace(regExp, RepText);
  233. };
  234. quantity_detail.prototype.updateQuantityDetail=function (args,dataCode,recode,selected) {
  235. var doc ={};
  236. var query={
  237. ID:recode.ID,
  238. projectID:recode.projectID
  239. };
  240. var selected = selected?selected:projectObj.project.mainTree.selected;
  241. doc[dataCode]=args.editingText;
  242. if (dataCode == 'regex') {
  243. if(recode.hasOwnProperty('rationID')){
  244. query.rationID=recode.rationID;
  245. }else {
  246. query.billID = recode.billID
  247. }
  248. query.refreshQuantity=true;
  249. if(!selected.data.hasOwnProperty('isFromDetail')||selected.data.isFromDetail==0){
  250. var c = confirm("确定要使用工程量明细替换原工程量吗?");
  251. if(!c){
  252. //query.refreshQuantity=false;
  253. this.cleanQuantityDetail(selected,true);
  254. return;
  255. }
  256. }
  257. query.index = args.row;
  258. this.updateQuantityRegex(query,doc,args)
  259. }else {
  260. this.normalUpdate(query,doc);
  261. }
  262. };
  263. quantity_detail.prototype.updateQuantityRegex=function(query,doc,args){
  264. var needupdate = false;
  265. if(args.editingText==null){
  266. needupdate =true;
  267. }else {
  268. args.editingText = _.trim(args.editingText,/\r\n/);
  269. if(this.regexChecking(args.editingText)&&this.referenceChecking(args.editingText,args.row,doc)){
  270. needupdate = true;
  271. }
  272. }
  273. if(needupdate){
  274. this.commonUpdate("/quantity_detail/updateRegex",{query:query,doc:doc});
  275. }else {
  276. var sheet = subSpread.getActiveSheet();
  277. sheet.getCell(args.row,args.col).value(gljOprObj.detailData[args.row].regex);
  278. }
  279. };
  280. quantity_detail.prototype.isSummationUpdate=function (args,detailList,newval) {
  281. var query={
  282. ID:detailList[args.row].ID,
  283. projectID:detailList[args.row].projectID
  284. };
  285. var selected = projectObj.project.mainTree.selected;
  286. query.refreshQuantity=true;
  287. if(!selected.data.hasOwnProperty('isFromDetail')||selected.data.isFromDetail==0){
  288. var c = confirm("确定要使用工程量明细替换原工程量吗?");
  289. if(!c){
  290. query.refreshQuantity=false;
  291. }
  292. }
  293. if(detailList[args.row].hasOwnProperty('rationID')){
  294. query.rationID=detailList[args.row].rationID;
  295. }else {
  296. query.billID = detailList[args.row].billID
  297. }
  298. var doc={
  299. isSummation:newval
  300. };
  301. this.normalUpdate(query,doc);
  302. };
  303. quantity_detail.prototype.commonUpdate = function (url,postData) {
  304. var me = this;
  305. $.bootstrapLoading.start();
  306. var callback = function (data) {
  307. me.refreshAfterUpdate(data);
  308. data.node?gljOprObj.refreshTreeNode(data.node):"";
  309. $.bootstrapLoading.end();
  310. }
  311. CommonAjax.post(url,postData,callback,function () {
  312. $.bootstrapLoading.end();
  313. });
  314. };
  315. quantity_detail.prototype.normalUpdate=function(query,doc){
  316. var url = "/quantity_detail/update";
  317. this.commonUpdate(url,{query:query,doc:doc});
  318. };
  319. quantity_detail.prototype.regexChecking=function(text){
  320. var regex=/^[0-9Cc\+\-\*\^/\(\)\.]*$/g;
  321. if(!regex.test(text)){
  322. alert("输入了非法字符,请重新输入!")
  323. return false;
  324. }else {
  325. return true;
  326. }
  327. };
  328. quantity_detail.prototype.referenceChecking=function (text,row,doc) {
  329. text = text.toUpperCase();
  330. //text= this.replaceSqr(text);
  331. var me = this;
  332. var refReg = /C\d+/g;
  333. var self ='C'+(row+1);
  334. var refList = text.match(refReg);
  335. var invalidate = _.includes(refList,self);
  336. var referenceIndexs = [];
  337. var indexOut = false;
  338. _.forEach(refList,function (item) {
  339. var ref_index = parseInt(item.substring(1));
  340. if(ref_index>gljOprObj.detailData.length){
  341. indexOut=true;
  342. return;
  343. }else {
  344. referenceIndexs.push(ref_index);
  345. }
  346. });
  347. if(indexOut){
  348. alert("引用有误,请重新输入!");
  349. return false;
  350. }
  351. referenceIndexs=_.uniq(referenceIndexs);
  352. doc.referenceIndexs = referenceIndexs;
  353. this.temList = referenceIndexs;
  354. invalidate=this.getAllReferenceList((row+1),referenceIndexs);
  355. if(invalidate){
  356. alert("计算式中产生了循环引用,请重新输入!");
  357. return false;
  358. }
  359. return true;
  360. };
  361. quantity_detail.prototype.getAllReferenceList=function(original,refList){
  362. var me =this;
  363. var invalidate=false;
  364. _.forEach(refList,function (item) {
  365. if(me.getReferenceList(item,original)){
  366. invalidate=true;
  367. }
  368. })
  369. return invalidate;
  370. };
  371. quantity_detail.prototype.getReferenceList=function(item,original) {
  372. var invalidate =false;
  373. if(gljOprObj.detailData.length>=item){
  374. var recode = gljOprObj.detailData[item - 1];
  375. if (recode.referenceIndexs.length > 0) {
  376. if(_.includes(recode.referenceIndexs,original)){
  377. invalidate = true;
  378. return invalidate;
  379. }
  380. this.temList = this.temList.concat(recode.referenceIndexs);
  381. _.forEach(recode.referenceIndex, function (item) {
  382. if(this.getReferenceList(item,original)){
  383. invalidate = true;
  384. }
  385. })
  386. }
  387. }
  388. return invalidate;
  389. }
  390. quantity_detail.prototype.replaceSqr = function(text) {
  391. var squarRegex = /\([^\^]+\)\^\d+/g;
  392. var sqararr = text.match(squarRegex);
  393. var squarRegex2 = /C[0-9]+\^\d+|[0-9]+([.]{1}[0-9]+){0,1}\^\d+/g; //匹配没有括号的
  394. var sqararr2=text.match(squarRegex2);
  395. if(sqararr){
  396. text=converSqrByArr(sqararr,text);
  397. }
  398. if(sqararr2){
  399. text=converSqrByArr(sqararr2,text);
  400. }
  401. return text;
  402. };
  403. quantity_detail.prototype.converSqrByArr = function (sqararr,text) {
  404. var temp = text;
  405. sqararr.forEach(function (item) {
  406. var arr = item.split('\^');
  407. var y = parseInt(arr[1]);
  408. var x_arr = [];
  409. for (var i = 0; i < y; i++) {
  410. x_arr.push(arr[0]);
  411. }
  412. var temStr = x_arr.join('*');
  413. temp = temp.replace(item, temStr);
  414. });
  415. return temp;
  416. };
  417. quantity_detail.prototype.deleteByRation = function(ration){
  418. var detail_list = this.datas;
  419. var newList =_.filter(detail_list,(d)=>{
  420. return d.rationID!=ration.ID;
  421. });
  422. if(newList!=undefined){
  423. this.datas = newList;
  424. }
  425. };
  426. quantity_detail.prototype.deleteByBills=function(deleteData){
  427. var detail_list = this.datas;
  428. var billIDList = [];
  429. for(var i=0;i<deleteData.length;i++){
  430. if(deleteData[i].type=='delete'){
  431. billIDList.push(deleteData[i].data.ID);
  432. }
  433. }
  434. var newList =_.filter(detail_list,(d)=>{
  435. return !_.includes(billIDList,d.billID);
  436. });
  437. if(newList!=undefined){
  438. this.datas = newList;
  439. }
  440. };
  441. quantity_detail.prototype.cleanQuantityDetail = function (node,needSave) {
  442. node =node?node:projectObj.project.mainTree.selected;
  443. var query={projectID:node.data.projectID};
  444. if(node.sourceType === project.Bills.getSourceType()){
  445. query.billID = node.data.ID;
  446. this.deleteByBills([{type:'delete',data:node.data}]);
  447. }else if(node.sourceType === project.Ration.getSourceType()){
  448. this.deleteByRation(node.data);
  449. query.rationID = node.data.ID;
  450. }
  451. if(needSave===true){
  452. query.refreshQuantity=false;
  453. CommonAjax.post("/quantity_detail/save",query);
  454. }
  455. gljOprObj.detailData=[];
  456. sheetCommonObj.showData(gljOprObj.detailSheet,gljOprObj.detailSetting,[]);
  457. };
  458. quantity_detail.prototype.quantityEditChecking = function(value,node,fieldName){
  459. var validate = true;
  460. if(fieldName=='quantity'){
  461. if(node.data.hasOwnProperty('isFromDetail')&&node.data.isFromDetail==1){
  462. var c = confirm('已有工程量明细,是否清空明细表,采用手工输入的表达式?')
  463. if(c){
  464. validate = true;
  465. }else {
  466. validate = false;
  467. }
  468. }
  469. }
  470. return validate;
  471. };
  472. quantity_detail.prototype.autoTransformQuantity = function(value,node){
  473. let data = node.data;
  474. let option = optionsOprObj.getOption(optionsOprObj.optionsTypes.GENERALOPTS,'rationQuanACToRationUnit');
  475. console.log(option);
  476. if(option==true&&node.sourceType === project.Ration.getSourceType()&&data.unit) {//还需加入判读是否转换
  477. let times = parseInt(data.unit);
  478. if (isNaN(times)) {
  479. times = 1
  480. }
  481. value = value / times;
  482. }
  483. return value;
  484. };
  485. quantity_detail.prototype.getDecimal=function (node) {
  486. var decimal = 3;
  487. if(node.sourceType === project.Bills.getSourceType()){
  488. decimal = billsQuanDecimal.decimal(node.data.unit);
  489. }else {
  490. decimal = decimalObj.ration.quantity
  491. }
  492. return;
  493. };
  494. return new quantity_detail(project);
  495. }
  496. };