/** * Created by Mai on 2017/6/21. */ var projectObj = { project: null, mainSpread: null, mainController: null, gljSpreed: null, beforeMainTreeSelectedChange: function (pre) { /*if (node) { subViewObj.saveComments(node); }*/ }, // CSL, 2018-02-09 用于测试显示。使用示例:projectObj.testDisplay(‘总额’, 100); testDisplay: function (caption, value) { let c = ""; if (caption) c = caption + ": "; let s = ` ${c}${value}`; $("#testDisplay").html(s); }, selectedChangeNowTime: 0, selectedChangeDelayTime: 20, treeSelectedChanged: function (node) { let me = this; let project = projectObj.project; let mainSheet = projectObj.mainController.sheet; let init = true; //设置选中行底色和恢复前选中行底色 let refreshNodes = [node]; if (!project.mainTree.preSelected) { refreshNodes.push(project.mainTree.items[0]); } else { refreshNodes.push(project.mainTree.preSelected); init = !(project.mainTree.preSelected == node); } mbzm_obj.nodeChanged = init; project.mainTree.preSelected = node; projectObj.setNodesStyle(projectObj.mainController.sheet, refreshNodes); function delayOpr(callback) { let nowTime = Date.now(); me.selectedChangeNowTime = nowTime; setTimeout(function () { if (nowTime - me.selectedChangeNowTime == 0) { callback(); } }, me.selectedChangeDelayTime); } delayOpr(function () { if ($("#linkComments").hasClass("active")) { subViewObj.loadComments(node); } let nodeID = node ? node.getID() : null; gljOprObj.mainTreeSelectedChange = gljOprObj.selectedNodeId != nodeID; if (init) subObj.initNavItem(node); if (calcTools.isBill(node)) subObj.setTZJNRValue(node); /* 2018-11-9 在NavItem里设置了默认显示的item,在里执行了click这个操作所以这两个操作不用重复执行了 gljOprObj.showDataIfRationSelect(node); if (activeSubSheetIsCalcProgram()) calcProgramObj.refreshCalcProgram(node, 3);*/ }); }, refreshBaseActn: function (tree) { let setButtonValid = function (valid, btn) { if (!projectReadOnly && valid) { btn.removeClass("disabled"); } else { btn.addClass("disabled"); } }; let selected = tree.selected, that = projectObj; const validNodes = projectObj.mainController.getValidNodesWithinSelection(); const canUpLevel = function (nodes) { if (!nodes.length || projectObj.project.isBillsLocked()) { return false; } const firstNode = nodes[0]; const lastNode = nodes[nodes.length - 1]; if (firstNode.depth() <= 1) { // 首节点是树结构的第一/二层节点,灰显。 return false; } if (firstNode.sourceType !== commonConstants.SourceType.BILLS) { // 首节点是定额/量价/工料机,灰显。 return false; } else { if (lastNode.data.type == billType.BILL && lastNode.nextSibling) { // 末节点是清单有后兄弟 if (lastNode.data.calcBase) { // 有基数计算 return false; } if (lastNode.children.length > 0 && lastNode.children[0].sourceType !== commonConstants.SourceType.BILLS) { // 有子项,并且子项不是清单 return false; } } } return true; }; const canDownLevel = function (nodes) { if (!nodes.length || projectObj.project.isBillsLocked()) { return false; } const firstNode = nodes[0]; if (firstNode.depth() === 0) { // 首节点是树结构的第一层节点,灰显。 return false; } if (firstNode.sourceType !== commonConstants.SourceType.BILLS) { // 首节点是定额/量价/工料机,灰显。 return false; } else { const preNode = firstNode.preSibling; if (!preNode) { // 首节点无前兄弟,灰显 return false; } else if (preNode.data.calcBase) { // 首节点前兄弟有基数计算 return false; } if (preNode.children.length > 0) { //前兄弟有子项,子项是分项,灰显。 if (firstNode.data.type == billType.BILL && preNode.children[0].sourceType !== commonConstants.SourceType.BILLS) { // 首节点是清单,首节点前兄弟子项不是清单 return false; } } } return true; }; const canUpMove = function (nodes) { if (!nodes.length) { return false; } const firstNode = nodes[0]; if (firstNode && firstNode.preSibling) { if (firstNode.sourceType == commonConstants.SourceType.BILLS) { if (projectObj.project.isBillsLocked()) { return false; } if (firstNode.data.type == billType.DXFY && firstNode.data.isAdd !== 1) { return false; } } return true; } return false; }; const canDownMove = function (nodes) { if (!nodes.length) { return false; } const lastNode = nodes[nodes.length - 1]; if (lastNode && lastNode.nextSibling) { if (lastNode.sourceType == commonConstants.SourceType.BILLS) { if (projectObj.project.isBillsLocked()) { return false; } if (lastNode.data.type == billType.DXFY && lastNode.data.isAdd !== 1) { return false; } } return true; } return false; }; setButtonValid(projectObj.project.Ration.canAdd(selected), $("#insertRation")); setButtonValid(ifCanDelete(), $("#delete")); setButtonValid(canUpLevel(validNodes), $("#upLevel")); setButtonValid(canDownLevel(validNodes), $("#downLevel")); setButtonValid(canUpMove(validNodes), $("#upMove")); setButtonValid(canDownMove(validNodes), $("#downMove")); setButtonValid(!projectObj.project.isBillsLocked(), $("#ZLFB_btn")); }, checkCommonField: function (editingText, colSetting) { let value; if (colSetting.data.decimal) { value = Number(editingText); if (number_util.isNumber(value)) { if (colSetting.data.decimal) { value = value.toDecimal(colSetting.data.decimal); } } else { value = null; alert("当前输入的数据类型不正确,请重新输入。"); } } else { value = editingText; } return value; }, checkFormulaValidField: function (editingText, colSetting) { let value = Number(editingText); if (isNaN(value)) return value; if (!value) { try { let exp = new Expression(""); exp.Expression(editingText); value = Number(exp.Evaluate()); } catch (error) { value = null; } } if (value != undefined || value != null) { if (colSetting.data.decimal) { value = value.toDecimal(colSetting.data.decimal); } } else if (editingText && editingText !== "") { value = null; alert("当前输入的数据类型不正确,请重新输入。"); } return value; }, checkSpreadEditingText: function (editingText, colSetting) { if (colSetting.data.field === "contain" || colSetting.data.field === "feesIndex.common.unitFee" || colSetting.data.field === "feesIndex.common.totalFee") { //colSetting.data.field === 'quantity' return this.checkFormulaValidField(editingText, colSetting); } else if (colSetting.data.field === "programID") { return this.project.calcProgram.compiledTemplateMaps[editingText]; } else if (colSetting.data.field === "subType") { if (typeof editingText !== "number") return volumePriceMaps[editingText]; else return editingText; } else { return this.checkCommonField(editingText, colSetting); } }, getVisibleRows: function (sRow, count) { let rst = []; let nodes = projectObj.project.mainTree.items; for (let i = sRow, len = nodes.length; i < len; i++) { if (count <= 0) { break; } if (nodes[i].visible) { rst.push(i); count--; } } return rst; }, //根据列标段设置焦点单元格 setActiveCell(field, moveScroll) { projectObj.mainSpread.focus(true); let mainSheet = projectObj.mainSpread.getActiveSheet(); TREE_SHEET_HELPER.massOperationSheet(mainSheet, function () { let fieldCol = colSettingObj.getColByField(field); if (fieldCol) { if (moveScroll) { mainSheet.showColumn(fieldCol, GC.Spread.Sheets.HorizontalPosition.center); } mainSheet.setActiveCell(projectObj.project.mainTree.selected.serialNo(), fieldCol); } }); }, //获取粘贴更改的单元格(粘贴时,跳过隐藏行) checkSpreadChangedCells: function (info) { let rst = { refreshRows: [], changedCells: [] }; //获取显示行号 let vRows = projectObj.getVisibleRows(info.row, info.rowCount); let vCount = vRows.length; let lastRow = null, count = 0; for (let cell of info.changedCells) { cell.text = info.sheet.getCell(cell.row, cell.col).text(); if (cell.row !== lastRow) { count++; lastRow = cell.row; } if (count <= vCount) { if (vRows[count - 1] !== cell.row) { rst.refreshRows.push(cell.row); } rst.changedCells.push({ row: vRows[count - 1], col: cell.col, text: cell.text }); } else { rst.refreshRows.push(cell.row); } } rst.refreshRows = Array.from(new Set(rst.refreshRows)); return rst; }, updateBillsCode: function (node, value) { let project = projectObj.project, me = this; let stdMatchCode, formatCode, matchs; let searchStdBillsAndUpdate = function (stdCode, formatCode) { let orgCode = node.data.code ? node.data.code.substr(0, 9) : ""; if (projectObj.project.projectInfo.engineeringInfo.bill_lib.length === 0 || updateCodeCheck(value, stdCode === orgCode)) { normalUpdate(node, value, stdCode === orgCode); } else if (projectObj.project.projectInfo.engineeringInfo.bill_lib.length > 0) { let libId = projectObj.project.projectInfo.engineeringInfo.bill_lib[0].id; CommonAjax.post("/stdBillsEditor/getStdBillsByCode", { userId: userID, billsLibId: libId, code: stdCode }, function (data) { if (data) { function sortItems(serialItems, items) { for (let item of items) { for (let serialItem of serialItems) { if (item.id === serialItem.id) { item.serialNo = serialItem.serialNo; } } } items.sort(function (a, b) { let rst = 0; if (a.serialNo > b.serialNo) { rst = 1; } else if (a.serialNo < b.serialNo) { rst = -1; } return rst; }); } function updateBeforeInsert(node, data) { node.data.name = data.name; if (node.data.type == billType.BX) { //从清单库中找到标准清单的话,要把补项改成分项 node.data.type = billType.FX; } } if (/\//.test(data.unit)) { let existB = projectObj.project.Bills.sameStdCodeBillsData(data.code); if (existB) { updateBeforeInsert(node, data); data.unit = existB.unit; project.Bills.replaceBills(node.source, data, formatCode); projectObj.mainController.refreshTreeNode([node], false); } else { ConfirmModal.stdBillsUnit.check( data, function (std) { updateBeforeInsert(node, data); project.Bills.replaceBills(node.source, std, formatCode); projectObj.mainController.refreshTreeNode([node], false); }, function () { projectObj.mainController.refreshTreeNode([node], false); } ); } } else { updateBeforeInsert(node, data); project.Bills.replaceBills(node.source, data, formatCode); projectObj.mainController.refreshTreeNode([node], false); } } else { normalUpdate(node, value); } }); } }; //分部分项、措施项目才匹配 let withinValidFixed = false; let matchFixedFlags = [fixedFlag.SUB_ENGINERRING, fixedFlag.MEASURE]; let rootNode = getRootFixedNode(node); if (rootNode && rootNode.data.flagsIndex && rootNode.data.flagsIndex.fixed && matchFixedFlags.includes(rootNode.data.flagsIndex.fixed.flag)) { withinValidFixed = true; } if ( withinValidFixed && (node.data.type == billType.FX || node.data.type == billType.BX || (node.data.type == billType.BILL && node.source.children.length == 0)) ) { //是分项、补项或者叶子清单的情况下才需要查找替换 if (value && value.length === 9) { //&& /^[\d]+$/.test(value) 去掉全数字判断 07-31 zhang stdMatchCode = value; formatCode = project.Bills.newFormatCode(stdMatchCode); searchStdBillsAndUpdate(stdMatchCode, formatCode); return; } else if (value && value.length === 12) { //&& /^[\d]+$/.test(value) 去掉全数字判断 07-31 zhang stdMatchCode = value.substr(0, 9); matchs = project.Bills.sameStdCode(stdMatchCode, node.data.code); if (matchs.indexOf(value) === -1) { searchStdBillsAndUpdate(stdMatchCode, value); return; } else { hintBox.infoBox( "系统提示", "已存在该编码的清单,是否继续?", 2, function () { searchStdBillsAndUpdate(stdMatchCode, value); }, function () { me.mainController.refreshTreeNode([node], false); }, ["确定", "取消"] ); return; } } } normalUpdate(node, value); function normalUpdate(billnode, codeValue, onNeedCheck) { //在标准库中没有找到清单时改分项为补项再更新 let toBX = false; if (onNeedCheck !== true && billnode.data.type == billType.FX) { billnode.data.type = billType.BX; toBX = true; } project.Bills.updateField(billnode.source, "code", codeValue, toBX); me.mainController.refreshTreeNode([billnode], false); } function updateCodeCheck(value, sameStdCode) { //如果前9位相同,只改前三位,则只更新编号就行了,不用做其它处理 return sameStdCode && value && value.length == 12; } }, updateRationCode: function (node, value) { if (!isDef(node.data.code) && (!isDef(value) || value.toString().trim() == "")) { return; } if (projectObj.project.projectInfo.engineeringInfo.ration_lib.length === 0) { alert("当前项目无定额库,请添加定额库。"); this.mainController.refreshTreeNode([node], false); } else { let libId = parseInt($("#stdRationLibSelect").val()); CommonAjax.post("/complementaryRation/api/getRationItem", { user_id: userID, rationRepId: libId, code: value }, function (data) { if (data) { projectObj.project.Ration.replaceRation(node.source, data); projectObj.project.ration_glj.addRationGLJ(node.source, data); } else { alert('当前库中找不到定额"' + value + '"'); } projectObj.mainController.refreshTreeNode([node], false); }); } }, updateCode: function (node, value) { let project = projectObj.project; if (node.sourceType === project.Bills.getSourceType()) { this.updateBillsCode(node, value); // 新清单不适合实时计算,下面套什么还不能确定,无数量计算也无意义 } else if (node.sourceType === project.Ration.getSourceType()) { if (value && typeof value == "string") { //小写转换成大写 value = value.toUpperCase(); } project.Ration.updateRationCodes([{ node: node, value: value }]); // this.updateRationCode(node, value); // 新套定额适合实时计算 // 这里因异步问题暂时缺少工料机价格。该过程移到:ration_glj.js的refreshAfterSave方法中。 } }, updateNodeField: function (node, value, filedID, callback) { if (node.data[filedID] == value) { return; } let data = { type: node.sourceType, data: { ID: node.data.ID, }, }; data.data[filedID] = value; $.bootstrapLoading.start(); projectObj.project.updateNodes([data], function () { node.data[filedID] = value; if (callback) { callback(); } $.bootstrapLoading.end(); }); }, updateCellValue: function (node, value, colSetting, editingText) { let project = projectObj.project, fieldName = colSetting.data.field; if (node.sourceType == ModuleNames.ration_glj) { project.ration_glj.updateFromMainSpread(value, node, fieldName); } else if (calcTools.isGljRation(node)) { if (fieldName === "feesIndex.common.unitFee") fieldName = "marketUnitFee"; gljOprObj.updateRationTypeGLJ(value, node, fieldName, editingText); } else if (fieldName === "remark") { projectObj.updateNodeField(node, value, "remark"); } else if (fieldName === "feeRate" && value == null) { project.FeeRate.cleanFeeRateID(node); } else if (value !== calcFees.getFee(node.data, fieldName) || fieldName == "quantity") { //工程量需要进行转换,所以做特殊处理 if (fieldName === "code" && value != "" && !calcTools.isVolumePrice(node)) { projectObj.updateCode(node, value); } else if (fieldName === "unit" && node.sourceType == ModuleNames.bills) { node.updateData[fieldName] = value; node.changed = true; if (node.data.quantity) { node.updateData.quantity = scMathUtil.roundForObj(node.data.quantity, getDecimal("quantity", node)); //修改清单单位的时候清单工程量要重新4舍5入; } project.calcProgram.calcAndSave(node); } else if (fieldName === "feeRate") { project.FeeRate.updateFeeRateFromBills(value, node, fieldName); } else if (fieldName === "contain") { //编辑含量 project.Ration.updateContain(value, node); } else if ( fieldName === "quantity" || fieldName === "marketUnitFee" || fieldName === "feesIndex.common.unitFee" || fieldName === "feesIndex.common.totalFee" || fieldName === "calcBase" || fieldName === "programID" || fieldName === "subType" || fieldName === "isSubcontract" ) { if (fieldName === "quantity") { quantityEditObj.checkingAndUpdate(editingText, node); return; } else if (fieldName === "marketUnitFee" || fieldName === "feesIndex.common.unitFee" || fieldName === "feesIndex.common.totalFee") { if (isNaN(value)) { // 输入的是无效的字符 alert("当前输入的数据类型不正确,请重新输入。"); projectObj.mainController.refreshTreeNode([node]); return; } // if (value) { if (fieldName === "feesIndex.common.unitFee") { value = parseFloat(value).toDecimal(decimalObj.decimal("unitPrice", node)); //当前是量价/人材机,综合单价可输入,输入的值读取到市场单价,重算综合单价。 if (node.sourceType == ModuleNames.ration && node.data.type != rationType.ration) { fieldName = "marketUnitFee"; } else if (node.sourceType == ModuleNames.bills) { node.updateData.calcFlag = treeNodeCalcFlag.customUnitPrice; } } else if (fieldName === "feesIndex.common.totalFee") { value = scMathUtil.roundForObj(value, getDecimal("totalPrice", node)); node.updateData.calcFlag = treeNodeCalcFlag.customTotalPrice; } // } } else if (fieldName === "calcBase") { //zhong if (value === undefined || value === null || value.toString().trim() === "") { value = ""; } if (value === node.data.calcBase) { return; } node.data.userCalcBase = value; project.calcBase.calculate(node); if (!project.calcBase.success) { let activeCell = projectObj.mainSpread.getActiveSheet().getSelections()[0]; projectObj.mainController.refreshTreeNode([node]); return; } else if (value === "") { //删除清单基数,要清空费用 calcTools.clearFees(node); } // if (value) {value = parseFloat(value).toDecimal(decimalObj.decimal("totalPrice", node))}; } //计算基数赋值要经过解析和标准化,已在calculate里赋值 if (fieldName !== "calcBase") { calcTools.setFieldValue(node, fieldName, value); projectObj.mainController.refreshTreeNode([node]); // 量价,输入单价x,反算单价x1,第二次输入单价x,单价不再反算。(实际反算了,没刷新) } project.calcProgram.calcAndSave(node); gljOprObj.showRationGLJSheetData(); } else { //不涉及计算的,只改updateData的值就好了 if (node.sourceType === project.Bills.getSourceType()) { if (fieldName === "maxPrice" || fieldName === "minPrice") { if (value == null) { value = ""; } else { value = scMathUtil.roundToString(value, getDecimal("unitPrice", node)); } } project.Bills.updateField(node.source, fieldName, value, false); } node.updateData[fieldName] = value; node.changed = true; if (colSetting.data.wordWrap) { this.mainSpread.getActiveSheet().autoFitRow(node.serialNo()); } project.calcProgram.calcAndSave(node); // projectObj.mainController.refreshTreeNode([node]); } } else { projectObj.mainController.refreshTreeNode([node], false); } }, mainSpreadLeaveCell: function (sender, info) { let colSetting = projectObj.mainController.setting.cols[info.col]; projectObj.lastCol = colSetting; projectObj.lastCell = { row: info.row, col: info.col }; if (colSetting.data.field === "quantity") { SheetDataHelper.hideMoreButton(); } }, //repaint 动态下拉框 mainSpreadEnterCell: function (sender, info) { let colSetting = projectObj.mainController.setting.cols[info.col]; if ( colSetting.data.field === "unit" || (projectObj.lastCol && projectObj.lastCol.data.field === "unit") || colSetting.data.field === "subType" || (projectObj.lastCol && projectObj.lastCol.data.field === "subType") || colSetting.data.field === "programID" || (projectObj.lastCol && projectObj.lastCol.data.field === "programID") ) { let rects = []; rects.push(info.sheet.getCellRect(info.row, info.col)); if (projectObj.lastCell) { rects.push(info.sheet.getCellRect(projectObj.lastCell.row, projectObj.lastCell.col)); } for (let rect of rects) { info.sheet.repaint(rect); } } }, //滚动造价书鼠标时,工程量的更多按钮跟着移动 TopRowChanged: function (type, info) { let colSetting = projectObj.mainController.setting.cols[info.sheet.getActiveColumnIndex()]; if (colSetting.data.field === "quantity") { SheetDataHelper.moveMoreButton(info.sheet); } }, mainSpreadEditStarting: function (sender, info) { let project = projectObj.project; let node = project.mainTree.items[info.row]; let fieldName = projectObj.mainController.setting.cols[info.col].data.field; let value = info.sheet.getValue(info.row, info.col); if (fieldName === "code" && node && calcTools.isRationItem(node) && isDef(node.data.code) && isDef(value)) { if (isDef(node.data.prefix) && node.data.prefix !== rationPrefix.none) { value = value.replace(new RegExp(node.data.prefix), ""); } value = value.replace(new RegExp(rationPrefix.replace), ""); info.sheet.setValue(info.row, info.col, value); } if (node && fieldName == "quantity" && (node.data.quantityEXP !== null || node.data.quantityEXP !== undefined)) { info.sheet.setValue(info.row, info.col, node.data.quantityEXP); } }, onCellDoubleClick: function (sender, info) { let project = projectObj.project; if (info.row && info.col) { let node = project.mainTree.items[info.row]; let fieldName = projectObj.mainController.setting.cols[info.col].data.field; if (fieldName == "marketUnitFee") { if (gljOprObj.hasComposition(node.data, true)) { alert("当前人材机的市场价由组成物计算得出,不可直接修改。"); return; } } if (fieldName == "code" && node) { projectObj.showBillOrRationLib(node); } } }, showBillOrRationLib(node) { let BILLS = projectObj.project.Bills; let code = node.data.code ? node.data.code : ""; if (node.sourceType == ModuleNames.bills) { //当清单是“分部分项工程”、“措施项目工程”时,要展开清单规则节点 if (BILLS.isFXorBX(node) || (node.data.type == billType.BILL && BILLS.isMeasure(node))) { //是分项或补项,是清单并且属于措施项目节点 if (!billsGuidance.bills.tree) { doAfterLoadGuidance = function () { billsGuidance.locateAtBills(code); doAfterLoadGuidance = null; }; } else { billsGuidance.locateAtBills(code); } if (!$("#zy").is(":visible")) { $("#stdBillsGuidanceTab").click(); } } } if (node.sourceType == ModuleNames.ration) { //在定额编码中双击,如果右侧定额库没有展开,则自动展开。 if (node.data.type != rationType.ration) return; //只有定额类型才需要自动展开,其它都不用 let libID = node.data.libID; if (node.data.from === "cpt") { libID = node.data.fromUser ? `${rationLibObj.compleRationLibId}*${node.data.fromUser}` : rationLibObj.compleRationLibId; } if (!rationLibObj.tree) { if (isDef(libID)) sessionStorage.setItem("stdRationLib", libID); rationLibObj.doAfterGetRationTree = function () { this.locateAtRation(libID, code); this.doAfterGetRationTree = null; }; } else { if ($("#stdRationLibSelect").select().val() != libID) { let libOpts = $("#stdRationLibSelect").find("option"); for (let libOpt of libOpts) { if ($(libOpt).val() == libID) { $(libOpt).prop("selected", "selected"); break; } } $("#stdRationLibSelect").change(); rationLibObj.doAfterGetRationTree = function () { this.locateAtRation(libID, code); this.doAfterGetRationTree = null; }; } else { rationLibObj.locateAtRation(node.data.libID, code); } } if (!$("#de").is(":visible")) $("#stdRationTab").click(); } }, onColumnWidthChanged: function (sender, info) { // CSL,2018.07.25 if (info.colList.includes(4) || info.colList.includes(5)) { let items = projectObj.project.mainTree.items; let sheet = projectObj.mainSpread.getActiveSheet(); for (let i = 0; i < items.length; i++) { let need = false; if (items[i].data.itemCharacterText) { sheet.getCell(i, 4).wordWrap(true); need = true; } if (items[i].data.jobContentText) { sheet.getCell(i, 5).wordWrap(true); need = true; } if (need) sheet.autoFitRow(i); } } }, onClipboardPasting: function (sender, info) { let rIdx = info.sheet.getActiveRowIndex(); let node = projectObj.project.mainTree.items[rIdx]; if (!node.parent) { info.cancel = true; hintBox.infoBox("系统提示", "大项费用不允许粘贴!", 1); return false; } let colName = projectObj.project.projSetting.main_tree_col.cols[info.cellRange.col].data.field; if (node.sourceType === projectObj.project.Ration.getSourceType() && colName == "code") { info.cancel = true; function getNodeType(node) { let t = null; if (calcTools.isRationItem(node)) t = 1; else if (calcTools.isBill(node)) { if (calcTools.isBill_FB(node)) t = 2; else if (calcTools.isBill_FX(node) || calcTools.isBill_BX(node)) t = 3; else if (calcTools.isBill_BILL(node)) t = 4; } return t; } function getPasteTextArr(info) { let copyText = info.pasteData.text.trim(); if (!copyText) return null; let spliter = ""; if (copyText.includes("\r\n")) { spliter = "\r\n"; } else if (copyText.includes("\n")) { // 兼容Unix、Mac 等系统回车换行符 spliter = "\n"; } else { spliter = "\r"; } let rows = copyText.split(spliter); let rstArr = []; for (let row of rows) { row = row.trim(); row = row.split("\t"); rstArr.push(row); } return rstArr; } function getPasteNodes(node, nodeType, count) { let nodesArr = []; let curNode = null; for (let i = rIdx; i < rIdx + count; i++) { curNode = projectObj.project.mainTree.items[i]; if (curNode && calcTools.isSameTypeNode(curNode, node)) { nodesArr.push(curNode); } else { curNode = projectObj.project.mainTree.items[i - 1]; break; } } if (nodesArr.length < count) { if (projectObj.project.mainTree.selected != curNode) projectObj.project.mainTree.selected = curNode; let add = count - nodesArr.length; for (let i = 0; i < add; i++) { switch (nodeType) { case 1: curNode = projectObj.project.Ration.addNewRationFast(rationType.ration); break; case 2: curNode = ProjectController.addFB(projectObj.project, projectObj.mainController); break; case 3: curNode = ProjectController.addFX(projectObj.project, projectObj.mainController); break; case 4: curNode = ProjectController.addBills(projectObj.project, projectObj.mainController); break; } if (!curNode) { hintBox.infoBox("系统提示", "插入新结点出错,粘贴失败!", 1); return nodesArr; } else { nodesArr.push(curNode); if (projectObj.project.mainTree.selected != curNode) projectObj.project.mainTree.selected = curNode; } } } return nodesArr; } function doPaste(nodes, nodeType, datas) { if (nodeType == 1) { // 批量粘贴定额有严重的异步问题,这里要和清单分开处理 let updateRationCodes = []; for (let i = 0; i < datas.length; i++) { let ptNode = nodes[i]; let code = datas[i][0]; // 编号去掉“换、借、补”等多余的字 code = code.replace(new RegExp(rationPrefix.complementary), ""); code = code.replace(new RegExp(rationPrefix.borrow), ""); code = code.replace(new RegExp(rationPrefix.replace), ""); updateRationCodes.push({ node: ptNode, value: code }); } projectObj.project.Ration.updateRationCodes(updateRationCodes); } else { for (let i = 0; i < datas.length; i++) { let ptNode = nodes[i]; let code = datas[i][0]; if (!(ptNode.data && ptNode.data.code && ptNode.data.code.sameText(code))) { projectObj.updateCode(ptNode, code); } } } projectObj.mainController.refreshTreeNode(nodes); } let tagType = getNodeType(node); if (!tagType) { hintBox.infoBox("系统提示", "该树结点类型不支持从项目编码列粘贴!", 1); return false; } let datas = getPasteTextArr(info); if (!datas) return; let pasteNodes = getPasteNodes(node, tagType, datas.length); doPaste(pasteNodes, tagType, datas); } }, mainSpreadEditEnded: function (sender, info) { let project = projectObj.project; let node = project.mainTree.items[info.row]; project.withinBillsLocked(node); let colSetting = projectObj.mainController.setting.cols[info.col]; let fieldName = projectObj.mainController.setting.cols[info.col].data.field; //提前处理 if (fieldName === "code" || fieldName === "name" || fieldName === "unit") { info.editingText = info.editingText ? info.editingText.toString().replace(new RegExp(/[\r,\n]/g), "") : info.editingText; } // 检查输入类型等 let value = projectObj.checkSpreadEditingText(info.editingText, colSetting); projectObj.updateCellValue(node, value, colSetting, info.editingText); // 自动行高 const autoHeight = project.property.displaySetting.autoHeight; if (autoHeight) { projectObj.mainSpread.getActiveSheet().autoFitRow(info.row); } }, msClipboardChanged: function (sender, info) { let colSettings = projectObj.mainController.setting.cols; let cDatas = sheetCommonObj.getTableData(info.sheet, colSettings); sheetCommonObj.copyTextToClipboard(cDatas); return; }, mainSpreadRangeChanged: function (sender, info) { let project = projectObj.project, setting = projectObj.mainController.setting; let changedObj = projectObj.checkSpreadChangedCells(info); let refreshNodes = []; for (let row of changedObj.refreshRows) { refreshNodes.push(project.mainTree.items[row]); } let updateRationCodes = []; //更新定额编码时要用同步的方式 if (changedObj.changedCells.length > 0) { for (let changedCell of changedObj.changedCells) { let node = project.mainTree.items[changedCell.row]; let colSetting = setting.cols[changedCell.col]; let value = projectObj.checkSpreadEditingText(changedCell.text, colSetting); if (colSetting.data.field == "code" && node.sourceType == project.Ration.getSourceType() && node.data.type == rationType.ration) { //如果是更新定额的编码 updateRationCodes.push({ node: node, value: value }); } else { projectObj.updateCellValue(node, value, colSetting, changedCell.text); } } } if (updateRationCodes.length > 0) { project.Ration.updateRationCodes(updateRationCodes); } if (refreshNodes.length > 0) { projectObj.mainController.refreshTreeNode(refreshNodes); } }, checkMainSpread: function () { if (!this.mainSpread) { this.mainSpread = SheetDataHelper.createNewSpread($("#billsSpread")[0]); sheetCommonObj.spreadDefaultStyle(this.mainSpread); this.mainSpread.getActiveSheet().selectionPolicy(GC.Spread.Sheets.SelectionPolicy.muliRange); this.mainSpread.getActiveSheet().name("mainSheet"); this.mainSpread.getActiveSheet().options.isProtected = true; this.mainSpread.getActiveSheet().options.frozenlineColor = "#ababab"; } }, refreshMainSpread: function () { if (this.mainSpread) { this.mainSpread.refresh(); } }, loadProjectData: function () { var that = this; this.project = PROJECT.createNew(scUrlUtil.GetQueryString("project"), userID); let startTime = +new Date(); $.bootstrapLoading.start(); this.project.loadDatas(function (err) { projectInfoObj.refreshTotalPriceSpan(); // 工具栏中的总造价信息 let mTime = +new Date(); //projectInfoObj.showProjectInfo(that.project.projectInfo); //快速列设置 if (!colSettingObj.getVisible("itemCharacterText")) { switchTznrHtml(true); } if (colSettingObj.getVisible("itemCharacterText")) { switchTznrHtml(false); } if (!err) { that.project.projectGLJ.calcQuantity(true); //计算分部分项和技术措施项目消耗量; let quantityTime = +new Date(); // that.project.property = projectObj.project.projectInfo.property; if (typeof overwriteRationCalcBases === "function") overwriteRationCalcBases(that.project.property.taxType); if (typeof overwriteFeeTypesWhenHasProperty === "function") { overwriteFeeTypesWhenHasProperty(that.project.property.engineeringName); that.project.calcProgram.compileAllTemps(); } //that.project.calcProgram.compileAllTemps(); that.project.calcBase.init(that.project); // that.project.calcFields = JSON.parse(JSON.stringify(cpFeeTypes)); // that.project.initCalcFields(); let str = JSON.stringify(that.project.projSetting.main_tree_col); that.project.projSetting.mainGridSetting = JSON.parse(str); that.project.projSetting.mainGridSetting.frozenCols = 4; TREE_SHEET_HELPER.initSetting($("#billsSpread")[0], that.project.projSetting.mainGridSetting); that.project.projSetting.mainGridSetting.setAutoFitRow = MainTreeCol.getEvent("setAutoFitRow"); that.project.projSetting.mainGridSetting.cols.forEach(function (col) { col.data.splitFields = col.data.field.split("."); if (col.data.getText && Object.prototype.toString.apply(col.data.getText) === "[object String]") { col.data.getText = MainTreeCol.getEvent(col.data.getText); } if (col.readOnly && Object.prototype.toString.apply(col.readOnly) === "[object String]") { col.readOnly = MainTreeCol.getEvent(col.readOnly); } if (col.data.cellType && Object.prototype.toString.apply(col.data.cellType) === "[object String]") { let getCellType = MainTreeCol.getEvent(col.data.cellType); col.data.cellType = getCellType; } // if (col.data.digit && Object.prototype.toString.apply(col.data.digit) === "[object String]") { // col.data.decimal = that.project.getDecimal(col.data.digit); // col.data.formatter = MainTreeCol.getNumberFormatter(col.data.decimal); // } if (col.data.field === "code") { col.data.formatter = "@"; } col.setAutoHeight = MainTreeCol.getEvent("setAutoHeight"); col.editChecking = MainTreeCol.getEvent("editChecking"); // 根据配置设置自动行高,在这里先做个标记,然后对每个单元格单独配置 if (col.data.field === "name" || col.data.field === "itemCharacterText" || col.data.field === "jobContentText" || col.data.field === "adjustState") { col.data.autoHeight = true; col.showHint = true; } if (col.data.field === "quantity" || col.data.field === "remark") { col.showHint = true; } // 综合单价、综合合价,小数部分应补0对齐。 CSL // if (col.data.field.hasSubStr("common")){ if (col.data.field.hasSubStr(".totalFee")) col.data.formatter = MainTreeCol.getNumberFormatter(decimalObj.ration.totalPrice, true); else if (col.data.field.hasSubStr(".unitFee")) col.data.formatter = MainTreeCol.getNumberFormatter(decimalObj.ration.unitPrice, true); // } }); that.mainController = TREE_SHEET_CONTROLLER.createNew( that.project.mainTree, that.mainSpread.getActiveSheet(), that.project.projSetting.mainGridSetting ); that.mainController.bind("refreshBaseActn", that.refreshBaseActn); that.mainController.bind(TREE_SHEET_CONTROLLER.eventName.beforeTreeSelectedChange, that.beforeMainTreeSelectedChange); that.mainController.bind(TREE_SHEET_CONTROLLER.eventName.treeSelectedChanged, that.treeSelectedChanged); gljCol.initGljCol( that.project.projSetting.glj_col ? that.project.projSetting.glj_col.showAdjustPrice : false, that.project.property.tenderSetting.showTenderFields ? that.project.property.tenderSetting.showTenderFields : false, true ); that.mainController.showTreeData(); subObj.initSubSpread(); //初始化主界面下方的表格 if (!projectReadOnly) { that.mainSpreadEscKey(that.mainSpread, that.mainSpreadEditStarting, that.mainSpreadEditEnded); sheetCommonObj.bindEnterKey(that.mainSpread, that.mainSpreadEnterKey); } setTimeout(function () { that.mainSpread.getActiveSheet().startEdit(); //这两句需要挺多时间,而又需要在editend 事件前触发,而这些又不影响计算,所以这里用异步的方法 that.mainSpread.getActiveSheet().endEdit(); that.mainSpread.bind(GC.Spread.Sheets.Events.EditEnded, that.mainSpreadEditEnded); that.mainSpread.bind(GC.Spread.Sheets.Events.EditStarting, that.mainSpreadEditStarting); }, 100); that.mainSpread.bind(GC.Spread.Sheets.Events.SelectionChanged, that.amountAreaNumber); //选中统计专用,节点相关操作在treeSelectedChanged里些 that.mainSpread.bind(GC.Spread.Sheets.Events.LeaveCell, that.mainSpreadLeaveCell); that.mainSpread.bind(GC.Spread.Sheets.Events.EnterCell, that.mainSpreadEnterCell); that.mainSpread.bind(GC.Spread.Sheets.Events.RangeChanged, that.mainSpreadRangeChanged); that.mainSpread.bind(GC.Spread.Sheets.Events.ClipboardChanged, that.msClipboardChanged); that.mainSpread.bind(GC.Spread.Sheets.Events.ButtonClicked, that.onButtonClick); that.mainSpread.bind(GC.Spread.Sheets.Events.CellDoubleClick, that.onCellDoubleClick); that.mainSpread.bind(GC.Spread.Sheets.Events.ColumnWidthChanged, that.onColumnWidthChanged); that.mainSpread.bind(GC.Spread.Sheets.Events.ClipboardPasting, that.onClipboardPasting); //工程量悬浮窗移动事件 that.mainSpread.bind(GC.Spread.Sheets.Events.TopRowChanged, that.TopRowChanged); //let loadOtherStartTime = +new Date(); //if(!projectReadOnly){ that.loadMainSpreadContextMenu(); //} socketObject.connect("main", { projectReadOnly: !!projectReadOnly, user: { ...projectObj.project.projectInfo.opener } }); //连接socket服务器 let endTime = +new Date(); console.log(`时间——${endTime - startTime}`); that.project.projectMarkChecking(); //是否需要重新进行造价计算 autoFlashHeight(); that.loadFocusLocation(); projectObj.refreshMainSpread(); //定位到会话中的选项 let mainTabFocus = sessionStorage.getItem("mainTab") ? sessionStorage.getItem("mainTab") : "#tab_zaojiashu"; $(mainTabFocus).click(); if (projectReadOnly) { sheetCommonObj.disableSpread(that.mainSpread); } if (projectObj.project.projectInfo.lastFileVer != VERSION) projectObj.project.calcProgram.doTenderCalc(); // 打开标段后,进行一次全部计算,包括调价计算。 $.bootstrapLoading.end(); } else { } const onceAlert = getLocalCache(commonConstants.StorageKey.ONCE_MAIN_LOADED); if (onceAlert) { alert(onceAlert); removeLocalCache(commonConstants.StorageKey.ONCE_MAIN_LOADED); } }); }, //mainSpread有一些单元格进入编辑状态后,会动态改变值,公用的bindEscKey方法不适用 mainSpreadEscKey: function (workBook, editStarting = null, editEnded = null) { workBook.commandManager().register("myEsc", function () { let sheet = workBook.getActiveSheet(); if (editStarting) { sheet.unbind(GC.Spread.Sheets.Events.EditStarting); } if (editEnded) { sheet.unbind(GC.Spread.Sheets.Events.EditEnded); } let row = sheet.getActiveRowIndex(); let col = sheet.getActiveColumnIndex(); let orgV = sheet.getValue(row, col); let node = projectObj.project.mainTree.items[row]; let field = colSettingObj.getFieldByCol(col); //进入编辑状态后改变了单元格值的时候特殊处理 if (isDef(field) && field === "code") { if ( node && node.sourceType === projectObj.project.Ration.getSourceType() && node.data.type === rationType.ration && isDef(node.data.code) && isDef(node.data.prefix) && node.data.prefix !== rationPrefix.none ) { orgV = node.data.prefix + orgV; } } else if (isDef(field) && field === "quantity") { orgV = isDef(node) && isDef(node.data.quantity) ? node.data.quantity : ""; } if (!isDef(orgV)) { orgV = ""; } if (sheet.isEditing()) { sheet.endEdit(); sheet.setValue(row, col, orgV); } if (editStarting) { sheet.bind(GC.Spread.Sheets.Events.EditStarting, editStarting); } if (editEnded) { sheet.bind(GC.Spread.Sheets.Events.EditEnded, editEnded); } }); workBook.commandManager().setShortcutKey(null, GC.Spread.Commands.Key.esc, false, false, false, false); workBook.commandManager().setShortcutKey("myEsc", GC.Spread.Commands.Key.esc, false, false, false, false); }, mainSpreadEnterKey: function () { let me = projectObj; let sheet = me.mainSpread.getActiveSheet(); let orgRow = sheet.getActiveRowIndex(); let orgCol = sheet.getActiveColumnIndex(); let orgField = colSettingObj.getFieldByCol(orgCol); let selected = me.project.mainTree.items[orgRow]; if (sheet.isEditing()) { sheet.endEdit(); } //正常情况,焦点应该是在下一行的同一列 let newRow = orgRow + 1, newCol = orgCol; //getColByField getVisible都不提出来,减少调用次数 if (selected) { //在分项/补项的编码单元格回车,焦点应跳动至本行的工程量单元格 if ( orgField && orgField === "code" && selected.sourceType === me.project.Bills.getSourceType() && [billType.FX, billType.BX].includes(selected.data.type) ) { let quantityCol = colSettingObj.getColByField("quantity"); let quantityVisible = colSettingObj.getVisible("quantity"); if (quantityCol !== null && quantityVisible !== false) { newRow = orgRow; newCol = quantityCol; } } else if ( orgField && orgField === "quantity" && selected.sourceType === me.project.Bills.getSourceType() && [billType.FX, billType.BX].includes(selected.data.type) ) { //在分项/补项的工程量单元格回车,如果其下有定额/量价/人材机,焦点应跳动至其下第一条定额/量价/人材机的工程量,如果是空行,则跳到至第一条定额xx的编码; if (selected.children.length !== 0 && !(isDef(selected.children[0].data.code) && selected.children[0].data.code.toString().trim() !== "")) { let codeCol = colSettingObj.getColByField("code"); let codeVisible = colSettingObj.getVisible("code"); newRow = orgRow + 1; if (codeCol !== null && codeVisible !== false) { newCol = codeCol; } } //如果其下没有定额/量价/人材机,则自动在其下插入一条定额空行,在其后自动插入一条分项空行,焦点跳动至定额空行的编码单元格。 else if (selected.children.length === 0) { //分项没有后兄弟,且没有锁定清单时在其后插入分项 if ( !selected.nextSibling && !(projectObj.project.projectInfo.property.lockBills && projectObj.project.withinBillsLocked(projectObj.project.mainTree.selected)) ) { ProjectController.addFX(me.project, me.mainController); //将焦点行设置成原本的行,以插入定额 me.mainController.setTreeSelected(me.project.mainTree.items[orgRow]); } /* 2021-04-26 只插入清单不插入定额 //插入定额 me.project.Ration.addNewRation(null, rationType.ration, function () { newRow = sheet.getActiveRowIndex(); let codeCol = colSettingObj.getColByField('code'); let codeVisible = colSettingObj.getVisible('code'); if(codeCol !== null && codeVisible !== false){ newCol = codeCol; } sheet.setActiveCell(newRow, newCol); },true,null,false); */ } } //在定额/量价/人材机的编码单元格回车,焦点应跳动至本行的工程量单元格 else if (orgField && orgField === "code" && selected.sourceType === me.project.Ration.getSourceType()) { let quantityCol = colSettingObj.getColByField("quantity"); let quantityVisible = colSettingObj.getVisible("quantity"); if (quantityCol !== null && quantityVisible !== false) { newRow = orgRow; newCol = quantityCol; } } //在定额/量价/人材机的工程量单元格回车 else if (orgField && orgField === "quantity" && selected.sourceType === me.project.Ration.getSourceType()) { let nextSibling = selected.nextSibling; //如果其后有定额/量价/人材机(空行:编码为空不算),焦点应跳动至下一行定额/量价/人材机的工程量,不需处理 //如果其后没有定额/量价/人材机(空行也没有),则自动在其后插入一行定额空行,焦点跳动至定额空行的编码单元格 /* if(!nextSibling){ let codeCol = colSettingObj.getColByField('code'); let codeVisible = colSettingObj.getVisible('code'); me.project.Ration.addNewRation(null, rationType.ration, function () { newRow = sheet.getActiveRowIndex(); if(codeCol !== null && codeVisible !== false){ newCol = codeCol; } sheet.setActiveCell(newRow, newCol); }, true,null,false); } */ //如果其后有定额空行,焦点跳动至定额空行的编码单元格 if (nextSibling && !(isDef(nextSibling.data.code) && nextSibling.data.code.toString().trim() !== "")) { let codeCol = colSettingObj.getColByField("code"); let codeVisible = colSettingObj.getVisible("code"); if (codeCol !== null && codeVisible !== false) { newRow = orgRow + 1; newCol = codeCol; } } } //在清单的编码单元格回车,如果清单是叶子节点,焦点跳动至本行的工程量单元格 else if ( orgField && orgField === "code" && selected.sourceType === me.project.Bills.getSourceType() && selected.data.type === billType.BILL && selected.source.children.length === 0 ) { let quantityCol = colSettingObj.getColByField("quantity"); let quantityVisible = colSettingObj.getVisible("quantity"); if (quantityCol && quantityVisible) { newRow = orgRow; newCol = quantityCol; } } } //设置焦点 sheet.setActiveCell(newRow, newCol); //触发表格事件 me.mainController.setTreeSelected(me.project.mainTree.items[newRow]); //SelectionChanged me.mainSpreadLeaveCell({ type: "LeaveCell" }, { sheet: sheet, sheetName: sheet.name(), cancel: false, row: orgRow, col: orgCol }); me.mainSpreadEnterCell({ type: "EnterCell" }, { sheet: sheet, sheetName: sheet.name(), cancel: false, row: newRow, col: newCol }); }, isInsertEquipmentVisable: function (selected) { //浙江不管是预算或者工程量清单,都是隐藏,overwrite文件重写 if (this.project.property.valuationType !== "ration") { //属于预算项目的情况下,在固定清单可见 //属于的固定清单 let belongFlag = cbTools.getBelongFlag(selected); if (belongFlag && belongFlag === fixedFlag.EQUIPMENT_ACQUISITION_FEE) { return true; } return false; } return true; }, //取安全生产费 getSaveProductionCostCode: function () { //公路云里18的清单范本安全生产费是102-3的;养护云里浙江养护的安全生产费是102-4。 return "102-4"; }, // 注册自定义插入清单数量 registerFlexibleInsertBillMenu: function (name) { const project = projectObj.project; const insertBillsHtml = `${name} 行`; return sheetCommonObj.registerInputContextMenuItem("insertBills", insertBillsHtml, "fa-sign-in", async function () { if (project.mainTree.selected.data.type == billType.DXFY) { if (project.mainTree.selected.data.calcBase && project.mainTree.selected.data.calcBase != "") { alert("当前有基数计算,不能插入子项。"); return; } } try { const number = +$("#insert-bills-number").val(); if (!number) { return; } $.bootstrapLoading.start(); const postData = ProjectController.getBillsPostData(number); const newNodes = await ProjectController.addBillsByData(postData, true); projectObj.mainController.setTreeSelected(newNodes[0]); projectObj.selectColAndFocus(project.mainTree.selected); projectObj.refreshBaseActn(project.mainTree); } catch (err) { console.log(err); if (!$("hintBox_form").is(":visible")) { alert(err); } } finally { $.bootstrapLoading.end(); } }); }, // 注册自定义插入定额数量 registerFlexibleInsertRatoinMenu: function () { const project = projectObj.project; const insertRationHtml = `插入定额 行`; return sheetCommonObj.registerInputContextMenuItem("insertRation", insertRationHtml, "fa-sign-in", async function () { try { const number = +$("#insert-ration-number").val(); if (!number) { return; } $.bootstrapLoading.start(); const newData = []; for (let i = 0; i < number; i++) { newData.push({ itemQuery: null, rationType: rationType.ration }); } await project.Ration.addMultiRation(newData); projectObj.setActiveCell("code", true); projectObj.refreshBaseActn(project.mainTree); } catch (err) { console.log(err); if (!$("hintBox_form").is(":visible")) { alert(err); } } finally { $.bootstrapLoading.end(); } }); }, loadMainSpreadContextMenu: function () { var project = this.project, spread = this.mainSpread, controller = this.mainController; let insertBillsName = project.projectInfo.property && [commonConstants.ValuationType.BOQ, commonConstants.ValuationType.BILL_BUDGET].includes(project.projectInfo.property.valuationType) ? "插入清单" : "插入项目节"; //右键“插入清单”改文字为“插入项目节”,工程量清单中保持不变。 $.contextMenu({ selector: "#billsSpread", selectableSubMenu: true, build: function ($trigger, e) { projectObj.cusRationCombCheck(spread.getActiveSheet()); var target = SheetDataHelper.safeRightClickSelection($trigger, e, spread); controller.setTreeSelected(controller.tree.items[target.row]); return target.hitTestType === GC.Spread.Sheets.SheetArea.viewport || target.hitTestType === GC.Spread.Sheets.SheetArea.rowHeader; }, items: { insertRootBill: { name: "插入大项费用", icon: "fa-sign-in", disabled: function () { return projectReadOnly || (projectObj.project.isBillsLocked() && project.withinBillsLocked(project.mainTree.selected)); //return project.mainTree.selected ? project.mainTree.selected.sourceType !== project.Bills.getSourceType() : false; }, callback: function (key, opt) { ProjectController.addRootBill(project, controller); }, visible: function (key, opt) { return project.mainTree.selected && project.mainTree.selected.parent == null; }, }, insertFB: { name: "插入分部", icon: "fa-sign-in", disabled: function () { if (projectReadOnly) { return true; } let selected = project.mainTree.selected; if (projectObj.project.isBillsLocked() == false && selected && selected.sourceType == project.Bills.getSourceType()) { if (selected.data.type == billType.FB) { return false; } if (isFlag(selected.data) && selected.data.flagsIndex.fixed.flag == fixedFlag.SUB_ENGINERRING) { //焦点行是分部分项工程 if (selected.children.length > 0) { return selected.children[0].data.type == billType.FX || selected.children[0].data.type == billType.BX; //焦点行是分部分项工程,且子项是分项或补项 } else { return false; } } } return true; //除了清单,其它类型都只读 }, callback: function (key, opt) { ProjectController.addFB(project, controller); projectObj.selectColAndFocus(project.mainTree.selected); }, visible: function (key, opt) { if (project.mainTree.selected) { return project.Bills.isFBFX(project.mainTree.selected); //不属于分部分项的话隐藏 } else { return false; } }, }, insertFX: { name: "插入分项", icon: "fa-sign-in", disabled: function () { if (projectReadOnly) { return true; } let selected = project.mainTree.selected; if (projectObj.project.isBillsLocked() == false && selected && selected.sourceType == project.Bills.getSourceType()) { if (selected.data.type == billType.FX || selected.data.type == billType.BX) { //焦点行是分项,有效显示 return false; } if (selected.data.type == billType.FB) { //点行是分部, if (selected.children.length > 0) { //且有子项,子项是分部,灰显。 return selected.children[0].data.type == billType.FB; } else { return false; } } if (isFlag(selected.data) && selected.data.flagsIndex.fixed.flag == fixedFlag.SUB_ENGINERRING) { //焦点行是分部分项工程 if (selected.children.length > 0) { return selected.children[0].data.type == billType.FB; //焦点行是分部分项工程,且子项是分部时灰显 } else { return false; } } } return true; //除了清单,其它类型都只读 }, callback: function (key, opt) { ProjectController.addFX(project, controller); projectObj.selectColAndFocus(project.mainTree.selected); }, visible: function (key, opt) { if (project.mainTree.selected) { return project.Bills.isFBFX(project.mainTree.selected); //不属于分部分项的话隐藏 } else { return false; } }, }, insertBills: { type: projectObj.registerFlexibleInsertBillMenu(insertBillsName), disabled: function () { if (projectReadOnly) { return true; } let selected = project.mainTree.selected; if ( !(projectObj.project.isBillsLocked() == true && project.withinBillsLocked(selected)) && selected && selected.sourceType === project.Bills.getSourceType() ) { return false; } return true; }, visible: function (key, opt) { if (project.mainTree.selected) { return project.Bills.isFBFX(project.mainTree.selected) == true ? false : true; } else { return false; } }, }, spr1: "--------", insertRation: { type: projectObj.registerFlexibleInsertRatoinMenu(), disabled: function () { if (projectReadOnly) { return true; } // var selected = project.mainTree.selected; // return project.Ration.addRationChecking(selected); // Vincent, 2018-01-02 // 工具栏要加按钮,且不能隐藏。菜单可以隐藏,两者又必须统一,所以启用新规则。怕以后又要改回来,所以保留。 CSL, 2018-01-02 return !project.Ration.canAdd(project.mainTree.selected); }, visible: function (key, opt) { let rst = true; if (compilationName === "内蒙古高速公路日常养护估算(2021)") return false; return rst; }, }, insertGLJ: { name: "插入工料机", icon: "fa-sign-in", disabled: function () { if (projectReadOnly) { return true; } // var selected = project.mainTree.selected; // return project.Ration.addRationChecking(selected); // Vincent, 2018-01-02 return !project.Ration.canAdd(project.mainTree.selected); }, callback: function (key, opt) { let selected = project.mainTree.selected; if (selected.data.calcBase && selected.data.calcBase != "") { alert("当前有基数计算,不能插入定额/量价/工料机。"); return; } getGLJData("insert"); // ProjectController.addRation(project, controller, rationType.volumePrice); }, visible: function (key, opt) { let rst = true; if (compilationName === "内蒙古高速公路日常养护估算(2021)") return false; return rst; }, }, insertLJ: { name: "插入量价", //插入量价不需要自动定位到编号列 icon: "fa-sign-in", disabled: function () { if (projectReadOnly) { return true; } return !project.Ration.canAdd(project.mainTree.selected); }, visible: function (key, opt) { let rst = true; if (compilationName === "内蒙古高速公路日常养护估算(2021)") rst = false; return rst; }, callback: function (key, opt) { /* project.Ration.addNewRation(null,rationType.volumePrice,function (newNode) { projectObj.selectColAndFocus(newNode,null); },true);*/ }, items: { insertLabour: { name: "人工", icon: "fa-sign-in", callback: function (key) { project.Ration.insertVolumePrice(gljType.LABOUR); }, }, insertMaterial: { name: "材料", icon: "fa-sign-in", callback: function (key) { project.Ration.insertVolumePrice(gljType.GENERAL_MATERIAL); }, }, insertMachine: { name: "机械", icon: "fa-sign-in", callback: function (key) { project.Ration.insertVolumePrice(gljType.GENERAL_MACHINE); }, }, }, }, insertMaterial: { name: "按清单名称插入材料", //插入量价不需要自动定位到编号列 icon: "fa-sign-in", disabled: function () { if (projectReadOnly) { return true; } return !project.Ration.canAdd(project.mainTree.selected); }, visible: function (key, opt) { let rst = true; if (compilationName === "内蒙古高速公路日常养护估算(2021)") return false; return rst; }, callback: function (key, opt) { //名称取清单名称,单位取清单单位 let selected = project.mainTree.selected; let billNode = selected; if (selected.sourceType == project.Ration.getSourceType()) billNode = selected.parent; let ext = { name: billNode.data.name, unit: billNode.data.unit }; project.Ration.insertVolumePrice(gljType.GENERAL_MATERIAL, ext); }, }, insertEquipment: { name: "插入设备", icon: "fa-sign-in", disabled: function () { if (projectReadOnly) { return true; } return !project.Ration.canAdd(project.mainTree.selected); }, callback: function (key, opt) { let selected = project.mainTree.selected; if (selected.data.calcBase && selected.data.calcBase != "") { alert("当前有基数计算,不能插入定额/量价/工料机。"); return; } getGLJData("insertEquipment"); // ProjectController.addRation(project, controller, rationType.volumePrice); }, visible: function (key, opt) { //2018-11-08 新需求,这个按钮先隐藏,有需要再放开 let selected = project.mainTree.selected; return projectObj.isInsertEquipmentVisable(selected); }, }, spr2: "--------", calc_installation_fee: { name: "计取安装费用", icon: "fa-sign-in", disabled: function () { if (projectReadOnly) { return true; } return false; }, callback: function (key, opt) { installationFeeObj.showCalcInstallSettingDiv(); }, visible: function (key, opt) { return projectObj.project.isInstall(); }, }, cleanzmhs: { name: "清空定额调整", icon: "fa-remove", callback: function (key, opt) { let selected = project.mainTree.selected; projectObj.project.Ration.updateRationCodes([{ node: selected, value: selected.data.code }], true); }, visible: function (key, opt) { let selected = project.mainTree.selected; if (selected && selected.sourceType == ModuleNames.ration) { return true; } return false; }, disabled: function () { let selected = project.mainTree.selected; if (projectReadOnly) { return true; } if (selected && selected.data.type != rationType.ration) { return true; } return false; }, }, setBookMark: { name: "设置书签批注", icon: "fa-flag", disabled: function () { if (projectReadOnly || !project.mainTree.selected || project.mainTree.selected.sourceType == ModuleNames.ration_glj) { return true; } return false; }, items: { cancelMark: { name: "取消书签批注", icon: "fa-remove", disabled: function () { const selected = project.mainTree.selected; return !(selected && selected.data.bookmarkBackground); }, callback: function () { locateObject.setOrCancelBookmark(project.mainTree.selected, null); }, }, markE2F2C5: { name: function () { return locateObject.getDescribeByColor("E2F2C5"); }, icon: "fa-square annotate-color-1", callback: function () { locateObject.setOrCancelBookmark(project.mainTree.selected, "E2F2C5"); }, }, markF9E2CF: { name: function () { return locateObject.getDescribeByColor("F9E2CF"); }, icon: "fa-square annotate-color-2", callback: function () { locateObject.setOrCancelBookmark(project.mainTree.selected, "F9E2CF"); }, }, markF2EFD9: { name: function () { return locateObject.getDescribeByColor("F2EFD9"); }, icon: "fa-square annotate-color-3", callback: function () { locateObject.setOrCancelBookmark(project.mainTree.selected, "F2EFD9"); }, }, markF5D1DA: { name: function () { return locateObject.getDescribeByColor("F5D1DA"); }, icon: "fa-square annotate-color-4", callback: function () { locateObject.setOrCancelBookmark(project.mainTree.selected, "F5D1DA"); }, }, markE3E3E3: { name: function () { return locateObject.getDescribeByColor("E3E3E3"); }, icon: "fa-square annotate-color-5", callback: function () { locateObject.setOrCancelBookmark(project.mainTree.selected, "E3E3E3"); }, }, markB6F3F2: { name: function () { return locateObject.getDescribeByColor("B6F3F2"); }, icon: "fa-square annotate-color-6", callback: function () { locateObject.setOrCancelBookmark(project.mainTree.selected, "B6F3F2"); }, }, markECE0F5: { name: function () { return locateObject.getDescribeByColor("ECE0F5"); }, icon: "fa-square annotate-color-7", callback: function () { locateObject.setOrCancelBookmark(project.mainTree.selected, "ECE0F5"); }, }, }, }, delete: { name: "删除", icon: "fa-remove", disabled: function () { return !ifCanDelete(); }, callback: function () { var selected = project.mainTree.selected; if (selected.sourceType == project.Bills.getSourceType() && selected.data.type == billType.FB && selected.children.length <= 0) { //选中的是分部,并且没有子项,直接删除 project.Bills.deleteSelectedNodes(); //project.Bills.deleteSelectedNode(); } else { $("#delete_row").modal({ show: true }); //弹出删除提示框; } }, }, calculateAll_RationContent: { name: "造价计算", icon: "fa-calculator", callback: function () { $.bootstrapLoading.start(); setTimeout(function () { project.calcProgram.calcAllNodesAndSave(); $.bootstrapLoading.end(); }, 100); }, disabled: function () { if (projectReadOnly) { return true; } }, visible: function (key, opt) { //2019-11-11 新需求重新开放右键“造价计算”。 return true; }, }, spr4: "--------", copyBlock: { name: "复制整块", icon: "fa-copy", disabled: function () { if (projectReadOnly) { return true; } let selection = projectObj.mainSpread.getActiveSheet().getSelections()[0]; let firstNode = projectObj.project.mainTree.items[selection.row]; //当多选的情况,用mainTree.selected判断不正确,要用第一个选中的节点 for (let i = 0; i < selection.rowCount; i++) { //多选的时候判断所有与第一个节点同级的节点 let temNode = projectObj.project.mainTree.items[selection.row + i]; if (firstNode.getParentID() == temNode.getParentID()) { if (BlockController.copyBtnDisable(temNode) == true) { return true; } } } return false; }, callback: function () { $.bootstrapLoading.start(); //let selected = project.mainTree.selected; let selections = projectObj.mainSpread.getActiveSheet().getSelections(); setTimeout(function () { BlockController.copyBlock(selections[0]); $.bootstrapLoading.end(); }, 100); }, }, pasteBlock: { name: "粘贴整块", icon: "fa-paste", disabled: function () { if (projectReadOnly) { return true; } return BlockController.pasteBtnDisable(project.mainTree.selected); }, callback: function () { BlockController.pasteBlock(project.mainTree.selected); }, }, recoverDeletedNodes: { name: "恢复删除节点", icon: "fa-undo", disabled: function () { return BlockController.recoverBlockDisabled(); }, callback: function () { BlockController.recoverBlock(); }, }, editEngineer: { name: "编辑工程内容", icon: "fa-edit", disabled: function () { return projectReadOnly; }, visible: function (key, opt) { let selected = project.mainTree.selected; return selected.sourceType == ModuleNames.bills && project.Bills.isEngineerEst(selected); //当焦点行是“专业工程暂估价”时,右键可见并有效。 }, callback: function () { let node = project.mainTree.selected; //project.Bills.getNodeByFlag(project.mainTree.selected,fixedFlag.ENGINEERING_ESITIMATE); if (node) { projectObj.editContent(node, "engineeringContent"); } }, }, editService: { name: "编辑服务内容", icon: "fa-edit", disabled: function () { return projectReadOnly; }, visible: function (key, opt) { let selected = project.mainTree.selected; return selected.sourceType == ModuleNames.bills && project.Bills.isTotalService(selected); //当焦点行是“总承包服务费”时,右键可见并有效。 }, callback: function () { let node = project.mainTree.selected; //project.Bills.getNodeByFlag(project.mainTree.selected,fixedFlag.TURN_KEY_CONTRACT); if (node) { projectObj.editContent(node, "serviceContent"); } }, }, editAccording: { name: "编辑签证及索赔依据", icon: "fa-edit", disabled: function () { return projectReadOnly; }, visible: function (key, opt) { let selected = project.mainTree.selected; return selected.sourceType == ModuleNames.bills && project.Bills.isClaimVisa(selected); //当焦点行是“签证及索赔计价”时,右键可见并有效。 }, callback: function () { let node = project.mainTree.selected; //project.Bills.getNodeByFlag(project.mainTree.selected,fixedFlag.CLAIM_VISA); if (node) { projectObj.editContent(node, "claimVisa"); } }, }, replaceMaterial: { name: "智能材料替换", icon: "fa-edit", disabled: function (key, opt) { if (projectReadOnly) { return true; } let selected = project.mainTree.selected; return selected.sourceType == ModuleNames.bills ? !(project.Bills.isFXorBX(selected) || selected.source.children.length == 0) : true; //是分项、补项或叶子清单才有效; }, callback: function () { MaterialController.replaceMaterial([project.mainTree.selected]); }, visible: function (key, opt) { //2018-11-15 暂时隐藏 return false; }, }, createBlocks: { name: "生成组价模板", icon: "fa-puzzle-piece", disabled: function (key, opt) { if (projectReadOnly) { return true; } let selected = project.mainTree.selected; return selected.sourceType != ModuleNames.bills; }, callback: function () { blockLibObj.checkShow(); }, visible: function (key, opt) { let rst = G_SHOW_BLOCK_LIB; if (compilationName === "内蒙古高速公路日常养护估算(2021)") rst = false; return rst; // return G_SHOW_BLOCK_LIB; }, }, }, }); }, // 计算node及node的所有父项 converseCalculateBills: function (node) { projectObj.project.calcProgram.calcAndSave(node); }, // 获取上次退出时的焦点位置 loadFocusLocation: function (orow, ocol) { const projectId = scUrlUtil.GetQueryString("project"); let row = gljUtil.isDef(orow) ? orow : getLocalCache("lastRow:" + projectId); let col = gljUtil.isDef(ocol) ? ocol : getLocalCache("lastCol:" + projectId); if (row == null || col == null) { //默认焦点定位到造价书的第一行“分项”。 col = 1; row = 0; for (let i = 0; i < this.mainController.tree.items.length; i++) { let node = this.mainController.tree.items[i]; if (node && node.sourceType == ModuleNames.bills && (node.data.type === billType.FX || node.data.type === billType.BX)) { //默认字位到第一条分项或补项 row = i; break; } } } row = parseInt(row); col = parseInt(col); const sheet = this.mainSpread.getActiveSheet(); sheet.setSelection(row, col, 1, 1); row = row < this.mainController.tree.items.length ? row : 0; this.mainController.setTreeSelected(this.mainController.tree.items[row]); //触发树节点选中事件 sheet.showRow(row, GC.Spread.Sheets.VerticalPosition.center); }, // 选中区域合计数字 amountAreaNumber: function (e, info) { if (info.newSelections === undefined || info.newSelections.length <= 0) { return false; } const selectedArea = info.newSelections[0]; const sheet = info.sheet; if (selectedArea.colCount <= 1 && selectedArea.rowCount <= 1) { return false; } //获取最左最右坐标 let maxCell = { row: selectedArea.row, col: selectedArea.col + selectedArea.colCount }; let maxCellX = sheet.getCellRect(maxCell.row, maxCell.col).x ? sheet.getCellRect(maxCell.row, maxCell.col).x : sheet.getCellRect(maxCell.row, maxCell.col - 1).x; let minCell = { row: selectedArea.row, col: selectedArea.col }; let minCellX = sheet.getCellRect(minCell.row, minCell.col).x ? sheet.getCellRect(minCell.row, minCell.col).x : sheet.getCellRect(minCell.row, minCell.col + 1).x; // 获取鼠标位置 let x = window.event.clientX; let y = window.event.clientY; // 匹配数字或小数 const regular = /^([0-9]+[.]{1}[0-9]+)$|^([1-9]{1}\d*)$/; // 小数点最高位数 let max = 0; let total = 0; let counter = 0; let notNumber = false; //如果有非数字,则不显示 for (let col = selectedArea.col; col < selectedArea.colCount + selectedArea.col; col++) { for (let row = selectedArea.row; row < selectedArea.rowCount + selectedArea.row; row++) { const value = sheet.getCell(row, col).text(); if (!regular.test(value)) { if (value) notNumber = true; continue; } counter++; // 获取当前数据小数位数 let pointPosition = value.toString().indexOf("."); pointPosition = pointPosition < 0 ? pointPosition : pointPosition + 1; const current = pointPosition > 0 ? value.toString().substring(pointPosition, value.toString().length).length : 0; max = current > max ? current : max; total += parseFloat(value); } } // 如果不为0则悬浮显示 if (notNumber === false && total > 0 && counter > 1) { const div = $( '
合计: