material_controller.js 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369
  1. /**
  2. * Created by zhang on 2018/9/12.
  3. */
  4. let MaterialController = {
  5. moduleName: 'subRcj', //模块名
  6. subSpreadDivWidth:"",
  7. showItemText:false,
  8. //规则相关映射
  9. rule1: function (node) {
  10. let itemText = node.data.itemCharacterText;
  11. let name = this.getKeyString(itemText,"材质及规格:");
  12. if(name){
  13. return{name:name,specs:""}//规格型号变成空的
  14. }
  15. return null;
  16. },
  17. rule2:function (node) {
  18. let itemText = node.data.itemCharacterText;
  19. let name = this.getKeyString(itemText,"混凝土种类:");
  20. let specs = this.getKeyString(itemText,"混凝土强度等级:");
  21. if(name||specs){
  22. let doc ={};
  23. if(name) doc["name"] = name;
  24. if(specs) doc["specs"] = specs;
  25. return doc;
  26. }
  27. return null
  28. },
  29. replaceMaterial:function(nodes){
  30. let me = this;
  31. this.getMaterial(nodes,function (result) {
  32. if(!_.isEmpty(result)){
  33. me.startReplace(nodes,result);
  34. }
  35. });
  36. },
  37. startReplace:function (nodes,result) {//其实应该是批量修改工料机属性,与替换工料机不同
  38. let me = this,updateData = [];
  39. for(let n of nodes){
  40. let code = n.data.code;
  41. if(code.length >= 9){
  42. let r_list = this.eachNode(n,result[code.substr(0,9)]);
  43. if(r_list.length > 0) updateData.push(...r_list);
  44. }
  45. }
  46. me.postReplace(updateData);
  47. },
  48. postReplace:function (updateData) {
  49. let me = this;
  50. if(updateData.length == 0) return;
  51. $.bootstrapLoading.start();
  52. CommonAjax.post("/material/replace",updateData,function(result){
  53. $.bootstrapLoading.end();
  54. me.updateCacheAfterReplace(result)
  55. })
  56. },
  57. updateCacheAfterReplace:function (result){
  58. let nodes = [];
  59. for(let data of result){
  60. let node = projectObj.project.ration_glj.refreshRationNode(data.name,data.adjustState,data.rationID);//刷新定额名称和子目调整状态
  61. if(node) nodes.push(node);
  62. if(data.ration_gljs.length > 0) this.refreshRationGLJ(data.ration_gljs);
  63. }
  64. if(nodes.length >0){
  65. projectObj.project.projectGLJ.loadData(function () {
  66. projectObj.project.calcProgram.calcNodesAndSave(nodes,async function(){
  67. installationFeeObj.calcInstallationFee();
  68. await OVER_HEIGHT.reCalcOverHeightFee();
  69. await itemIncreaseFeeObj.calcItemIncreaseFeeByNodes(nodes);
  70. });
  71. gljOprObj.refreshView();
  72. })
  73. }
  74. },
  75. refreshRationGLJ:function(ration_gljs){
  76. let ration_glj_model = projectObj.project.ration_glj;
  77. for(let rg of ration_gljs){
  78. let glj = ration_glj_model.refreshByID(rg.ID,rg.doc);
  79. ration_glj_model.refreshTreeNodeIfNeeded(glj);//刷新造价书上的树节点(如果需要)
  80. }
  81. },
  82. eachNode:function (node,item) {
  83. let replaceDatas =[];
  84. if(item && item.bills.rule){
  85. let replace_property = this["rule"+item.bills.rule](node);//按清单设置的规则获取要修改的属性
  86. if(replace_property){
  87. replaceDatas = this.getReplaceData(node.data.ID,item.materialMap,replace_property);
  88. }
  89. }
  90. return replaceDatas;
  91. },
  92. getReplaceData:function (billsItemID,materialMap,replace_property) {
  93. let list = [];
  94. let replace_glj_list = this.findMatchRationGLJ(billsItemID,materialMap);//取出需要替换的工料机和对替换的材料内容
  95. for(let r of replace_glj_list){
  96. let doc = this.getDoc(r.glj,r.material,replace_property);//获取要修改的字段
  97. list.push({glj:gljOprObj.setGLJPrice(r.glj),doc:doc});
  98. }
  99. return list;
  100. },
  101. getDoc:function (glj,material,replace_property) {//取定额工料机中需要修改的内容
  102. let doc = {},keyList = ['name','specs','type','unit'];
  103. for(let key of keyList){
  104. if(glj[key] != material[key]) doc[key] = material[key];
  105. }
  106. for(let rkey in replace_property){
  107. doc[rkey] = replace_property[rkey]
  108. }
  109. return doc;
  110. },
  111. findMatchRationGLJ:function (billsItemID,materialMap) {//查找清单下匹配的需要替换的定额工料机
  112. let replaceList=[];
  113. for(let g of projectObj.project.ration_glj.datas){
  114. if(g.billsItemID == billsItemID && materialMap[g.original_code]) {
  115. replaceList.push({glj:g,material:materialMap[g.original_code]});
  116. }
  117. }
  118. return replaceList;
  119. },
  120. getMaterial:function (nodes,callback) {
  121. let data = [],materialMap= null;
  122. for(let n of nodes){
  123. let code = n.data.code;
  124. if(code.length >= 9 && n.data.billsLibId){
  125. let billCode = code.substr(0,9);
  126. data.push({code:billCode,billsLibId:n.data.billsLibId});
  127. }
  128. }
  129. if(data.length>0){
  130. //data 按编码去重
  131. let dataMap = _.indexBy(data,'code');
  132. CommonAjax.post("/material/getMaterial",dataMap,function (result) {
  133. callback(result);
  134. })
  135. }else {
  136. callback(null);
  137. }
  138. },
  139. getKeyString:function (itemText,matchStr) {//截取关键数据
  140. itemText = itemText.replace(/:/g, ":");//中文字符转换为英文字符
  141. itemText = itemText.replace(matchStr,"@$@");//用特殊符号取代关键字,方便截取
  142. let index = itemText.indexOf("@$@");
  143. if(index == -1) return null;
  144. let temString = itemText.substr(index+3);
  145. let strArray = temString.split(/\n/);
  146. let keyString = this.trim(strArray[0]);
  147. return keyString===""?null:keyString;
  148. },
  149. trim:function (str) {
  150. return str.replace(/(^\s*)|(\s*$)/g, "");
  151. },
  152. //--------------2018-10-08新加替换表格-----------------
  153. setting:{
  154. header:[
  155. {headerName: "项目特征", headerWidth: 120, dataCode: "character", dataType: "String"},
  156. {headerName: "内容", headerWidth: 240, dataCode: "context", dataType: "String"},
  157. {headerName: "名称", headerWidth: 50, dataCode: "name", dataType: "String",hAlign: "center",cellType:"replaceButton"},
  158. {headerName: "规格", headerWidth: 50, dataCode: "specs", dataType: "String",hAlign: "center",cellType:"replaceButton"},
  159. ],
  160. view: {
  161. lockColumns: [0,1,2,3],
  162. rowHeaderWidth:25
  163. }
  164. },
  165. spread:null,
  166. sheet:null,
  167. datas:[],
  168. showReplaceDiv:function (node) {
  169. if(!($('#linkGLJ').hasClass('active'))){
  170. return;
  171. }
  172. $("#subSpread").addClass("ration_glj_spread");
  173. $('#itemTextDiv').show();
  174. $("#replaceM").show();
  175. $("#rgResize").show();// 拖动调整大小功能添加
  176. // 2018-11-07 需求变更,这个项目特征替换表格暂时隐藏不删除,以后可能还要用,先把项目初始化这句注释掉
  177. // if(node) this.showReplaceSpread(node);//如果不是只刷新页面
  178. this.showItemCharacterText(node);//只显示文本
  179. refreshSubSpread();
  180. },
  181. hideReplaceDiv:function(){
  182. $("#subSpread").removeClass("ration_glj_spread");
  183. $("#subSpread").css("width",""); //左右拖动调整表格大小的时候会设置css属性,所以隐藏这个div的时候也要把这个属性给去掉
  184. $("#replaceM").hide();
  185. //refreshSubSpread(); //提升焦点变换性能 2019年4月12日
  186. },
  187. showReplaceSpread:function(node){
  188. $("#replaceM").addClass("ovf-hidden");
  189. this.initSpread();
  190. this.showData(node);
  191. },
  192. initSpread:function () {
  193. if(!this.spread){
  194. this.spread = SheetDataHelper.createNewSpread($("#replaceM")[0]);
  195. sheetCommonObj.spreadDefaultStyle(this.spread);
  196. this.initSheet();
  197. }else {
  198. this.spread.refresh();
  199. }
  200. },
  201. initSheet:function(){
  202. this.sheet = this.spread .getSheet(0);
  203. sheetCommonObj.initSheet( this.sheet, this.setting);
  204. this.spread.bind(GC.Spread.Sheets.Events.ButtonClicked, this.onReplaceButtonClick);
  205. this.sheet.name('materialReplace');
  206. },
  207. showData:function(node){
  208. this.datas = [];
  209. let selected = node?node:projectObj.project.mainTree.selected;
  210. let parent = selected.parent;
  211. if(selected&&parent){
  212. this.getItems(parent.data.itemCharacterText);
  213. }
  214. sheetCommonObj.showData(this.sheet, this.setting,this.datas);
  215. this.sheet.setRowCount(this.datas.length);
  216. },
  217. showItemCharacterText:function (node, selector = '#itemCharacterText') {
  218. let text="";
  219. let selected = node?node:projectObj.project.mainTree.selected;
  220. let parent = selected.parent;
  221. if(selector != '#itemCharacterText'){//如果是清单精灵里显示的,parent就是本身
  222. parent = selected
  223. $("#xmtz_billID").val(selected.data.ID);
  224. }
  225. if(selected&&parent) text = parent.data.itemCharacterText;
  226. if(selector != '#itemCharacterText'){
  227. $(selector).val(text);
  228. }else{
  229. text = parent.data.itemCharacterText?parent.data.itemCharacterText.replace(/\n/g,"<br>"):"";
  230. text = text.replace(/\s/g,"&nbsp;");
  231. $(selector).html(text);
  232. }
  233. /* if(text == "") {//为空的时候不显示
  234. $("#replaceM").hide();
  235. $("#subSpread").removeClass("ration_glj_spread");
  236. $('#subSpread').css('width','');
  237. return;
  238. }*/
  239. //$('#replaceM').css('margin-top',0);
  240. //this.toggleItemInit();
  241. },
  242. //初始化人材机和项目特征文本两个区域的宽度,改变窗口大小时调用此方法,实时刷新
  243. initItemWidth: function () {
  244. //总宽度
  245. let totalWidth = $('#subItems').width();
  246. //人材机和项目特征文本比例
  247. const openWidth = 30;//打开项目特征工具条
  248. let textVisible = $('#replaceText').is(':visible'),
  249. //默认比例
  250. textPercent = '15%',
  251. rcjPercent = '85%';
  252. //文本没显示,则打开工具条固定30px转换百分比
  253. if (!textVisible) {
  254. textPercent = openWidth / totalWidth;
  255. rcjPercent = 1 - textPercent;
  256. textPercent = textPercent * 100 + '%';
  257. rcjPercent = rcjPercent * 100 + '%';
  258. }
  259. //设置特征及内容与排版规则的比例
  260. if ($('#replaceM').is(':visible')) {//显示工具条
  261. $('#subSpread').css('width', rcjPercent);
  262. $('#itemTextDiv').css('width', textPercent);
  263. //打开了项目特征,则继续加载人材机表和项目特征文本具体比例
  264. if (this.showItemText === true) {
  265. let rg_sideResizeEles = MaterialController.getSideResize();
  266. SlideResize.loadHorizonWidth(rg_sideResizeEles.eleObj.module,
  267. [rg_sideResizeEles.eleObj.resize], [rg_sideResizeEles.eleObj.left, rg_sideResizeEles.eleObj.right]);
  268. }
  269. } else {//不显示工具条
  270. $('#subSpread').css('width', '100%');
  271. }
  272. },
  273. toggleItemInit:function () {
  274. if(this.showItemText == true){
  275. $("#replaceM").removeClass("ovf-hidden");
  276. $("#openItemText").hide();
  277. $("#replaceText").show();
  278. $("#closeItemText").show();
  279. $("#rgResize").show();
  280. this.initItemWidth();
  281. }else {
  282. $("#replaceM").addClass("ovf-hidden");
  283. $("#openItemText").show();
  284. $("#replaceText").hide();
  285. $("#closeItemText").hide();
  286. $("#rgResize").hide();
  287. this.initItemWidth();
  288. }
  289. },
  290. getItems:function(characterText){
  291. if(!characterText) return;
  292. let strArray = characterText.split(/\n/);
  293. for(let tem of strArray){
  294. tem = tem.replace(/:/g, ":");//中文字符转换为英文字符
  295. if(tem.indexOf(":") != -1){//有:号才提取
  296. let proArr = tem.split(":");
  297. //如果内容为空的话也不提取
  298. let context = this.trim(proArr[1]);
  299. if(context!="") this.datas.push({character:proArr[0],context:context,name:"换",specs:"换"})
  300. }
  301. }
  302. },
  303. onReplaceButtonClick:function(e, args){
  304. //项目只读
  305. if(projectReadOnly) {
  306. return;
  307. }
  308. let me = MaterialController;
  309. let sheet = args.sheet, row = args.row, col = args.col;
  310. let dataCode = me.setting.header[col].dataCode, item = me.datas[row], doc = {};
  311. if(dataCode && item){
  312. doc[dataCode] = item.context;
  313. me.replaceByClick(doc);
  314. }
  315. },
  316. replaceByClick:function (tem) {
  317. let ration_glj = gljOprObj.getSelectedRationGlj();
  318. let doc = {};
  319. if(ration_glj && !ration_glj.isMixRatio){//有选中并且不是组成物
  320. if(gljOprObj.isExtraType(ration_glj.type)) return; //如果是“企业管理费”“利润”“一般风险费”类型,不能替换
  321. for(let key in tem){//检查属性是否做了改变,如果没有改变不用提交替换
  322. if(ration_glj[key] != tem[key]) doc[key] = tem[key]
  323. }
  324. if(!_.isEmpty(doc)) this.postReplace([{glj:ration_glj,doc:doc}]);
  325. }
  326. },
  327. getSideResize: function () {
  328. let rg_sideResizeEles = {};
  329. rg_sideResizeEles.eleObj = {
  330. module: this.moduleName,
  331. resize: $('#rgResize'),
  332. parent: $('#subItems'),
  333. left: $('#subSpread'),
  334. right: $('#itemTextDiv')
  335. };
  336. rg_sideResizeEles.limit = {
  337. min: 100,
  338. max: `$('#subItems').width()-100`
  339. };
  340. return rg_sideResizeEles;
  341. }
  342. };
  343. let rg_sideResizeEles = MaterialController.getSideResize();
  344. SlideResize.horizontalSlide(rg_sideResizeEles.eleObj, rg_sideResizeEles.limit, function(){
  345. MaterialController.subSpreadDivWidth = $('#subSpread').css('width');
  346. refreshSubSpread();
  347. });
  348. $(function (){
  349. $("#openItemText").click(function () {
  350. MaterialController.showItemText = true;
  351. MaterialController.toggleItemInit();
  352. refreshSubSpread();
  353. });
  354. $("#closeItemText").click(function () {
  355. MaterialController.showItemText = false;
  356. MaterialController.toggleItemInit();
  357. refreshSubSpread();
  358. });
  359. })