Jelajahi Sumber

Merge branch 'budget' of http://192.168.1.41:3000/SmartCost/ConstructionCost into budget

# Conflicts:
#	config/gulpConfig.js
#	web/building_saas/main/html/main.html
zhangweicheng 4 tahun lalu
induk
melakukan
0585071b37

+ 1 - 0
config/gulpConfig.js

@@ -183,6 +183,7 @@ module.exports = {
         'web/building_saas/main/js/views/mbzm_view.js',
         'web/building_saas/main/js/views/tender_price_view.js',
         'web/building_saas/main/js/views/equipment_purchase_view.js',
+        'web/building_saas/main/js/views/gather_fees_view.js',
         'web/building_saas/main/js/views/billsElf.js',
         'web/building_saas/main/js/views/sub_view.js',
         'web/building_saas/main/js/views/fee_rate_view.js',

+ 26 - 0
web/building_saas/main/html/gather_fees.html

@@ -0,0 +1,26 @@
+<!DOCTYPE html>
+<html lang="zh-CN">
+
+<head>
+    <meta charset="utf-8">
+    <meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no">
+    <meta http-equiv="x-ua-compatible" content="ie=edge">
+    <script src="../../../../public/web/common_util.js"></script>
+</head>
+
+<body>
+    <div class="toolsbar px-1" style="height:30px;width:100%;line-height:30px;font-size:15px;">费用汇总</div>
+    <div class="container-fluid">
+        <div class="row">
+            <div class="col-lg-2 p-0">
+                <div class="grid main-data-full" id="gfMainSpread"></div>
+            </div>
+            <div class="col-lg-10 p-0">
+                <div class="grid main-data-full" id="gfDetailSpread"></div>
+            </div>
+        </div>
+    </div>
+</body>
+
+
+</html>

+ 9 - 4
web/building_saas/main/html/main.html

@@ -74,8 +74,9 @@
         <li class="nav-item"><a data-toggle="tab" href="#fee_rates" id="tab_fee_rate" role="tab">费率</a></li>
         <li class="nav-item"><a data-toggle="tab" href="#calc_program_manage" id="tab_calc_program_manage" role="tab"
             style="display:none">总计算程序</a></li>
-        <!-- <li class="nav-item"><a data-toggle="tab" href="#tender_price" id="tab_tender_price" role="tab">调价</a></li> -->
-        <li class="nav-item"><a data-toggle="tab" href="#equipment_purchase" id="tab_equipment_purchase" role="tab">设备购置</a></li>
+        <li class="nav-item"><a data-toggle="tab" href="#tender_price" id="tab_tender_price" role="tab" style="display:none">调价</a></li>
+     <li class="nav-item"><a data-toggle="tab" href="#equipment_purchase" id="tab_equipment_purchase" role="tab">设备购置</a></li>
+              <li class="nav-item"><a data-toggle="tab" href="#gather_fees" id="tab_gather_fees" role="tab">费用汇总</a></li>
         <li class="nav-item"><a data-toggle="tab" href="#reports" role="tab" id="tab_report"
             onclick="rptTplObj.iniPage();">报表</a></li>
         <li class="nav-item"><a data-toggle="tab" href="#index" id="tab_index" role="tab" style="display:none">指标信息</a>
@@ -991,9 +992,12 @@
         <div class="tab-pane" id="tender_price" role="tabpanel">
           <%include tender_price.html %>
         </div>
-        <div class="tab-pane" id="equipment_purchase" role="tabpanel">
+         <div class="tab-pane" id="equipment_purchase" role="tabpanel">
           <%include equipment_purchase.html %>
         </div>
+        <div class="tab-pane" id="gather_fees" role="tabpanel">
+          <%include gather_fees.html %>
+        </div>
         <div class="tab-pane" id="calc_program_manage" role="tabpanel">
           <%include calc_program_manage.html %>
         </div>
@@ -2956,7 +2960,8 @@
   <script type="text/javascript" src="/web/building_saas/main/js/views/zmhs_view.js"></script>
   <script type="text/javascript" src="/web/building_saas/main/js/views/mbzm_view.js"></script>
   <script type="text/javascript" src="/web/building_saas/main/js/views/tender_price_view.js"></script>
-  <script type="text/javascript" src="/web/building_saas/main/js/views/equipment_purchase_view.js"></script>
+   <script type="text/javascript" src="/web/building_saas/main/js/views/equipment_purchase_view.js"></script>
+  <script type="text/javascript" src="/web/building_saas/main/js/views/gather_fees_view.js"></script>
   <script type="text/javascript" src="/web/building_saas/main/js/views/billsElf.js"></script>
   <script type="text/javascript" src="/web/building_saas/main/js/views/sub_view.js"></script>
   <script type="text/javascript" src="/web/building_saas/main/js/views/fee_rate_view.js"></script>

+ 66 - 25
web/building_saas/main/js/models/calc_program.js

@@ -1081,7 +1081,7 @@ let calcTools = {
     else if (tender == tenderTypes.ttReverseRation)
         return node.data.rationQuantityCoe;
     }
-   
+
 
 };
 
@@ -2075,7 +2075,7 @@ class CalcProgram {
             projectObj.project.projectInfo.lastFileVer = VERSION;
             // 批量树结点计算后,计算程序早已物是人非,所以这里要重新计算一下。警告:第二个参数千万不能改成3,否则死循环!
             if (activeSubSheetIsCalcProgram())
-                calcProgramObj.refreshCalcProgram(treeNodes.tree.selected, 2);
+                calcProgramObj.refreshCalcProgram(treeNodes[0].tree.selected, 2);
             $.bootstrapLoading.end();
         });
     };
@@ -2142,7 +2142,7 @@ class CalcProgram {
         calcNodes(tree.roots);
         me.calcFormulaNodes(changedNodes, tender);
         debugger;
-        if (tender){   
+        if (tender){
             for(let node of tree.items){
                 this.clearTenderCache(node);
             };
@@ -2716,30 +2716,71 @@ class CalcProgram {
     };
 
     doTenderCalc(callback) {
-    $.bootstrapLoading.start();
-    setTimeout(()=>{
-        let tender = calcTools.getTenderType();
-        if (tender == tenderTypes.ttReverseGLJ || tender == tenderTypes.ttReverseRation) {
-        // 调价计算必须依赖调价树。
-        if (!tender_obj.tenderTree){
-            tender_obj.createTree();
-            tender_obj.createTreeNodes();
-        }
-        // 反向调价+逼近原理:
-        // 清理调价缓存 → 分摊目标金额(从子往父处理满载、从父往子分摊) → 全局反算 → 逼近 → 全局正算、存储
-        this.reverseTenderInitDatas();
-        this.prepareForDistribute(tender_obj.tenderTree.roots[0]);
-        this.distributeTargetTotalFee(tender_obj.tenderTree.roots[0]);
-        this.calcAllNodes(calcAllType.catAll, tender);    // 先全局反算:得到每定额的coe、基础调后金额(误差大,需逼近)
-        this.reverseTenderApproach(callback, tender);     // 逼近上述基础调后金额
-        }
-
-        // 全局正算(用的是主树,无需调价树)
-        this.calcAllNodesAndSave(calcAllType.catAll, callback, tenderTypes.ttCalc);
-    })
+        $.bootstrapLoading.start();
+        setTimeout(()=>{
+            let tender = calcTools.getTenderType();
+            if (tender == tenderTypes.ttReverseGLJ || tender == tenderTypes.ttReverseRation) {
+            // 调价计算必须依赖调价树。
+            if (!tender_obj.tenderTree){
+                tender_obj.createTree();
+                tender_obj.createTreeNodes();
+            }
+            // 反向调价+逼近原理:
+            // 清理调价缓存 → 分摊目标金额(从子往父处理满载、从父往子分摊) → 全局反算 → 逼近 → 全局正算、存储
+            this.reverseTenderInitDatas();
+            this.prepareForDistribute(tender_obj.tenderTree.roots[0]);
+            this.distributeTargetTotalFee(tender_obj.tenderTree.roots[0]);
+            this.calcAllNodes(calcAllType.catAll, tender);    // 先全局反算:得到每定额的coe、基础调后金额(误差大,需逼近)
+            this.reverseTenderApproach(callback, tender);     // 逼近上述基础调后金额
+            }
+
+            // 全局正算(用的是主树,无需调价树)
+            this.calcAllNodesAndSave(calcAllType.catAll, callback, tenderTypes.ttCalc);
+        })
 
     };
-   
+
+    gatherRationFees(programID){
+        // $.bootstrapLoading.start();
+        // setTimeout(()=>{
+            // 先把符合的定额筛选出来,以提高速度。
+            let rations = []; 
+            for (const r of projectObj.project.Ration.datas) {
+                if (r.programID == programID) rations.push(r);
+            };    
+
+            let gatherObj = {};
+            gatherObj.programID = programID;
+            for (let ft of cpFeeTypes) {
+                let ftObj = {};
+                ftObj.name = ft.name;
+    
+                let sum_uf = 0, sum_tuf = 0, sum_tf = 0, sum_ttf = 0;
+                for (const r of rations) {
+                    let uf = 0, tuf = 0, tf = 0, ttf = 0;
+                    if (r.feesIndex && r.feesIndex[ft.type]) {
+                        uf = parseFloatPlus(r.feesIndex[ft.type].unitFee).toDecimal(decimalObj.bills.unitPrice);
+                        tuf = parseFloatPlus(r.feesIndex[ft.type].tenderUnitFee).toDecimal(decimalObj.bills.unitPrice);
+                        tf = parseFloatPlus(r.feesIndex[ft.type].totalFee).toDecimal(decimalObj.bills.totalPrice);
+                        ttf = parseFloatPlus(r.feesIndex[ft.type].tenderTotalFee).toDecimal(decimalObj.bills.totalPrice);
+                    };
+                    sum_uf = (sum_uf + uf).toDecimal(decimalObj.process);
+                    sum_tuf = (sum_tuf + tuf).toDecimal(decimalObj.process);
+                    sum_tf = (sum_tf + tf).toDecimal(decimalObj.process);
+                    sum_ttf = (sum_ttf + ttf).toDecimal(decimalObj.process);
+                };
+
+                ftObj.totalFee = sum_tf.toDecimal(decimalObj.bills.totalPrice);
+                ftObj.tenderTotalFee = sum_ttf.toDecimal(decimalObj.bills.totalPrice);
+                ftObj.unitFee = sum_uf.toDecimal(decimalObj.bills.unitPrice);
+                ftObj.tenderUnitFee = sum_tuf.toDecimal(decimalObj.bills.unitPrice);
+
+                gatherObj[ft.type] = ftObj;
+            };
+            return gatherObj;
+        // });
+        // $.bootstrapLoading.end();    
+    }
 };
 
 // export default analyzer;

+ 584 - 0
web/building_saas/main/js/views/gather_fees_view.js

@@ -0,0 +1,584 @@
+let gatherFeesView = {
+    datas: [],
+    mainSpread: null,
+    detailSpread: null,
+    mainSheet: null,
+    detailSheet: null,
+    mainSetting: {
+        header:[
+            // {headerName:"ID",headerWidth:80,dataCode:"ID", hAlign: "center"},
+            {headerName:"名称",headerWidth:300,dataCode:"name", dataType: "String"}
+        ],
+        view:{
+            comboBox:[],
+            lockColumns:[0,1],
+            colHeaderHeight: CP_Col_Width.colHeader,
+            rowHeaderWidth: CP_Col_Width.rowHeader
+        }
+    },
+    detailSetting: {
+        header:[
+            // {headerName:"ID",headerWidth:80,dataCode:"ID", hAlign: "center"},
+            {headerName:"费用代号",headerWidth:CP_Col_Width.code, dataCode:"code", dataType: "String"},
+            {headerName:"费用名称",headerWidth:CP_Col_Width.name, dataCode:"name", dataType: "String"},
+            {headerName:"计算基数",headerWidth:CP_Col_Width.dispExprUser, dataCode:"dispExprUser", dataType: "String"},
+            {headerName:"费率",headerWidth:CP_Col_Width.feeRate, dataCode:"feeRate", dataType: "Number",hAlign: "right",decimalField:"feeRate"},
+            {headerName: "金额", headerWidth: CP_Col_Width.totalFee, dataCode: "totalFee", dataType: "Number"},
+            {headerName:"费用类别",headerWidth:CP_Col_Width.displayFieldName, dataCode:"displayFieldName", dataType: "String", hAlign: "center"},
+            {headerName:"基数说明",headerWidth:CP_Col_Width.statement, dataCode:"statement", dataType: "String"},
+            {headerName:"备注",headerWidth:CP_Col_Width.memo, dataCode:"memo", dataType: "String"}
+        ],
+        view:{
+            comboBox:[],
+            lockColumns:[0,1,2,3,4,5,6,7],
+            colHeaderHeight: CP_Col_Width.colHeader,
+            rowHeaderWidth: CP_Col_Width.rowHeader
+        },
+        //callback 和disable都是和 cusButton配套使用的
+        callback:{
+            calcBase:function (hitinfo) {
+                calcBaseView.onCalcBaseButtonClick(hitinfo,'ration');
+            }
+        },
+        disable:{
+            calcBase:function (row,col) {
+                return projectReadOnly || !calcBaseView.ifEdit('ration', row)
+            }
+        }
+    },
+
+    buildSheet: function (){
+        let me = this;
+        me.datas = projectObj.project.calcProgram.datas.templates;
+        if (me.mainSpread) {
+            me.mainSpread.destroy();
+            me.mainSpread = null;
+        };
+        if (me.detailSpread) {
+            me.detailSpread.destroy();
+            me.detailSpread = null;
+        };
+        me.mainSpread = sheetCommonObj.buildSheet($('#gfMainSpread')[0], me.mainSetting, me.datas.length);
+        me.mainSheet = me.mainSpread.getSheet(0);
+        sheetCommonObj.spreadDefaultStyle(me.mainSpread);
+        me.detailSpread = sheetCommonObj.buildSheet($('#gfDetailSpread')[0], me.detailSetting, me.datas[0].calcItems.length);
+        me.detailSheet = me.detailSpread.getSheet(0);
+        sheetCommonObj.spreadDefaultStyle(me.detailSpread);
+        let arr = projectObj.project.calcProgram.compiledFeeTypeNames.slice();
+        // arr.delete('暂估费');
+        // let fieldName = new GC.Spread.Sheets.CellTypes.ComboBox();
+        // fieldName.items(arr);
+        // me.detailSheet.getRange(-1, 4, -1, 1).cellType(fieldName);
+
+        me.mainSheet.bind(GC.Spread.Sheets.Events.EnterCell, me.onMainEnterCell);
+        me.detailSheet.bind(GC.Spread.Sheets.Events.ValueChanged, me.onDetailValueChanged);
+        me.detailSheet.bind(GC.Spread.Sheets.Events.RangeChanged, me.onRangeChanged);
+        me.detailSheet.bind(GC.Spread.Sheets.Events.EnterCell, me.onDetailEnterCell);
+        me.detailSheet.bind(GC.Spread.Sheets.Events.ClipboardPasting, me.onClipboardPasting);
+        sheetCommonObj.showData(me.mainSheet, me.mainSetting, me.datas);
+        me.mainSheet.setRowCount(me.datas.length);
+
+        me.detailSheet.name('gatherfee_detail');
+        feeRateObject.setFeeRateCellCol(me.detailSheet, _.findIndex(me.detailSetting.header,{'dataCode':'feeRate'}));
+        sheetCommonObj.showData(me.detailSheet, me.detailSetting, me.datas[0].calcItems);
+        me.getfeeRateColor(me.datas[0].calcItems);
+        customRowHeader(me.detailSheet, me.datas[0].calcItems.length);
+        if(!projectReadOnly){
+            // me.loadMainContextMenu();
+            // me.loadDetailContextMenu();
+        }
+        else {
+            if(me.mainSetting.view.lockColumns){
+                me.mainSetting.view.lockColumns = null;
+            }
+            if(me.detailSetting.view.lockColumns){
+                me.detailSetting.view.lockColumns = null;
+            }
+            sheetCommonObj.disableSpread(me.mainSpread);
+            sheetCommonObj.disableSpread(me.detailSpread);
+        }
+    },
+    getStdCalcProgramFiles: function(){
+        function getStdCPFilesHtml(stdCPLibs) {
+            let result = '<option value="">请选择计算程序标准库</option>';
+            if (stdCPLibs.length <= 0) {
+                return result;
+            };
+
+            for (let lib of stdCPLibs){
+                result += '<option value='+ lib.id +'>'+ lib.name +'</option>';
+            };
+            return result;
+        };
+
+        let stdCPHtml = getStdCPFilesHtml(projectObj.project.projectInfo.engineeringInfo.program_lib);
+        $("#calcProgramFileSelect").html(stdCPHtml);
+    },
+    onMainEnterCell: function(sender, args) {
+        var me = gatherFeesView;
+        var row = args.sheet.getActiveRowIndex();
+
+        me.detailSpread.suspendPaint();
+        var dData = me.datas[row].calcItems;
+        me.detailSheet.setRowCount(dData.length, GC.Spread.Sheets.SheetArea.viewport);
+        let gatherObj = projectObj.project.calcProgram.gatherRationFees(me.datas[row].ID);
+        for (const d of dData) {
+           d['totalFee'] = gatherObj[d.fieldName]?.totalFee;
+        }
+        sheetCommonObj.showData(me.detailSheet, me.detailSetting, dData);
+        me.getfeeRateColor(dData);
+        customRowHeader(me.detailSheet, dData.length);
+        me.detailSpread.resumePaint();
+    },
+    onRangeChanged:function (sender,args) {
+        let me = gatherFeesView;
+        let editInfo= me.getSelectionInfo();
+        let dataCode = me.detailSetting.header[args.col].dataCode;
+        // if (args.action == GC.Spread.Sheets.RangeChangedAction.clear&&args.changedCells.length == 1) {//清除操作
+        if (args.action == GC.Spread.Sheets.RangeChangedAction.clear) {//清除操作
+            if (dataCode == 'feeRate') {
+                $.bootstrapLoading.start();
+                projectObj.project.FeeRate.updateFeeRateFromCalc(null, editInfo);
+            }
+            else if (dataCode == 'displayFieldName'){
+                $.bootstrapLoading.start();
+                for (let cell of args.changedCells){
+                    let curCalcItem = editInfo.template.calcItems[cell.row];
+                    curCalcItem.fieldName = '';
+                    curCalcItem.displayFieldName = '';
+                };
+
+                let data = {
+                    'projectID': projectObj.project.ID(),
+                    'ID': editInfo.template.ID,
+                    'calcItems': editInfo.template.calcItems
+                };
+                gatherFeesView.updateTemplate(data, function (rst) {
+                    if (rst){
+                        gatherFeesView.refreshDetailSheet();
+                        $.bootstrapLoading.end();
+                    }
+                });
+            }
+        }
+    },
+    onDetailValueChanged: function(sender, args) {
+        $.bootstrapLoading.start();
+        let me = gatherFeesView;
+        let editInfo= me.getSelectionInfo();
+        let curCalcItem = editInfo.calcItem;
+        let dataCode = me.detailSetting.header[args.col].dataCode;
+
+        if (dataCode == 'feeRate') {
+            projectObj.project.FeeRate.updateFeeRateFromCalc(args.newValue, editInfo);
+            $.bootstrapLoading.end();
+            return;
+        }
+
+        let template = me.getSelectionInfo().template;
+        if (dataCode == 'displayFieldName') {
+            if (curCalcItem.displayFieldName == args.newValue) {
+                $.bootstrapLoading.end();
+                return;
+            }
+
+            if (analyzer.fieldNameIsUsed(template, args.newValue)){
+                let sheet = me.detailSpread.getActiveSheet();
+                sheet.suspendEvent();
+                sheet.setValue(args.row, args.col, args.oldValue);
+                sheet.resumeEvent();
+                $.bootstrapLoading.end();
+                let s = hintBox.font(template.fieldNameTempUsed + 1);
+                hintBox.infoBox('系统提示', `“${args.newValue}” 已被第 ${s} 行使用,不允许重复选择!`, 1);
+                delete template.fieldNameTempUsed;
+                return;
+            }
+
+            curCalcItem.fieldName = projectObj.project.calcProgram.compiledFeeTypeMaps[args.newValue];
+        }
+        else if (dataCode == 'dispExprUser'){  // 除非直接改单元格,弹窗不会走这里
+            hintBox.infoBox('系统提示', '用户直接在sheet中修改了dispExprUser值,触发onDetailEditEnded事件。', 1);
+        };
+
+        curCalcItem[dataCode] = args.newValue;
+
+        if (dataCode == 'name'){
+            analyzer.refreshUsedCalcItemsStatement(template, curCalcItem);
+            let data = {
+                'projectID': projectObj.project.ID(),
+                'ID': template.ID,
+                'calcItems': template.calcItems
+            };
+            gatherFeesView.updateTemplate(data, function (rst) {
+                if (rst){
+                    gatherFeesView.refreshDetailSheet();
+                    $.bootstrapLoading.end();
+                }
+            });
+            if (activeSubSheetIsCalcProgram())
+                calcProgramObj.refreshCalcProgram(projectObj.project.mainTree.selected, 1);
+        }
+        else{
+            let data = {
+                'projectID': projectObj.project.ID(),
+                'templatesID': template.ID,
+                'calcItem': curCalcItem
+            };
+            me.saveCalcItem(data, function (rst) {
+                if (rst){
+                    let relationNodes = calcTools.getNodesByProgramID(template.ID);
+                    projectObj.project.calcProgram.calcNodesAndSave(relationNodes);
+                    $.bootstrapLoading.end();
+                }
+            });
+        }
+    },
+    onDetailEnterCell: function (sender, args) {
+        // if (args.col == 2)    // 加这句,切换单元格后,原单元格的三点图片不会消失。
+            gatherFeesView.detailSheet.repaint();   // 这句是为了触发sheetCommonObj → getCusButtonCellType → CusButtonCellType → paint方法。
+        // for test.
+        // let t = gatherFeesView.getSelectionInfo().template;
+        // let c = gatherFeesView.getSelectionInfo().calcItem;
+        // let lc = analyzer.calcItemLabourCoe(c);
+        // c.dispExpr = analyzer.getDispExpr(c.expression, t);
+        // c.dispExprUser = analyzer.getDispExprUser(c.dispExpr, lc);
+        // c.compiledExpr = analyzer.getCompiledExpr(c.expression, lc);
+        // let e = `ID:${c.ID} ${c.expression} ${c.dispExpr} ${c.dispExprUser} ${c.compiledExpr} ${c.custom}`;
+        // projectObj.testDisplay('', e);
+    },
+    onClipboardPasting: function (sender, args) {
+         args.cancel = true;
+    },
+    loadMainContextMenu: function () {
+        $.contextMenu({
+            selector: '#gfMainSpread',
+            build: function ($trigger, e) {
+                SheetDataHelper.safeRightClickSelection($trigger, e, gatherFeesView.mainSpread);
+            },
+            items: {
+                "copyTemplate": {
+                    name: "另存为",
+                    icon: 'copy',
+                    callback: function (key, opt) {
+                        $.bootstrapLoading.start();
+                        let template = gatherFeesView.getSelectionInfo().template;
+                        let idx = gatherFeesView.mainSpread.getActiveSheet().getActiveRowIndex();
+
+                        let newTemplate = {};
+                        newTemplate.ID = analyzer.templateMaxID() + 1;
+                        newTemplate.name = analyzer.templateNewName(template.name);
+                        newTemplate.custom = true;
+                        newTemplate.calcItems = [];
+                        $.extend(true, newTemplate.calcItems, template.calcItems);
+                        // 清理掉费率ID关联
+                        for (let ci of newTemplate.calcItems){
+                            if (ci.feeRateID || ci.feeRateID == null)
+                                delete ci.feeRateID;
+                        };
+
+                        let data = {
+                            'projectID': projectObj.project.ID(),
+                            'ID': newTemplate.ID,
+                            'name': newTemplate.name,
+                            'custom': newTemplate.custom,
+                            'calcItems': newTemplate.calcItems
+                        };
+
+                        gatherFeesView.addTemplate(data, function (rst) {
+                            if (rst){
+                                let ts = projectObj.project.calcProgram.templates;
+                                ts.push(newTemplate);
+                                projectObj.project.calcProgram.compileTemplateMaps();
+                                projectObj.project.calcProgram.compileTemplate(newTemplate);
+                                gatherFeesView.buildSheet();
+                                gatherFeesView.mainSheet.setSelection(ts.length - 1, 0, 1, 1);
+                                gatherFeesView.mainSheet.showRow(ts.length - 1, GC.Spread.Sheets.VerticalPosition.center);
+                                gatherFeesView.refreshDetailSheet();
+                                $.bootstrapLoading.end();
+                            }
+                        });
+                    }
+                },
+                "reNameTemplate": {
+                    name: "重命名...",
+                    icon: 'edit',
+                    disabled: function () {
+                        let custom = gatherFeesView.getSelectionInfo().template.custom;
+                        let canReName = custom ? custom : false;
+                        return !canReName;
+                    },
+                    callback: function (key, opt) {
+                        $.bootstrapLoading.start();
+                        let template = gatherFeesView.getSelectionInfo().template;
+                        let idx = gatherFeesView.mainSpread.getActiveSheet().getActiveRowIndex();
+
+                        let newName = '';
+                        hintBox.valueBox('重命名', template.name, function () {
+                            newName = hintBox.value;
+
+                            if (!newName){
+                                hintBox.error(`名称不能为空!`);
+                                return false;
+                            };
+
+                            if (newName == template.name) {
+                                $.bootstrapLoading.end();
+                                return;
+                            }
+
+                            if (analyzer.templateNameIsExist(newName)){
+                                hintBox.error(`“${newName}” 已存在,请重新输入!`);
+                                return false;
+                            };
+
+                            template.name = newName;
+                            let data = {
+                                'projectID': projectObj.project.ID(),
+                                'ID': template.ID,
+                                'name': template.name
+                            };
+                            gatherFeesView.updateTemplate(data, function (rst) {
+                                if (rst){
+                                    projectObj.project.calcProgram.compileTemplateMaps();
+                                    projectObj.mainController.refreshTreeNode(calcTools.getNodesByProgramID(template.ID));
+                                    sheetCommonObj.showData(gatherFeesView.mainSpread.getSheet(0), gatherFeesView.mainSetting, gatherFeesView.datas);
+                                    $.bootstrapLoading.end();
+                                }
+                            });
+                        });
+                    }
+                },
+                "spr1": '--------',
+                "deleteTemplate": {
+                    name: '删除',
+                    icon: 'fa-remove',
+                    disabled: function () {
+                        let custom = gatherFeesView.getSelectionInfo().template.custom;
+                        let canDelete = custom ? custom : false;
+                        return !canDelete;
+                    },
+                    callback: function () {
+                        $.bootstrapLoading.start();
+                        let template = gatherFeesView.getSelectionInfo().template;
+                        if (analyzer.templateIsUsed(template.ID)) {
+                            $.bootstrapLoading.end();
+                            hintBox.infoBox('系统提示', `计算模板“${template.name}”已被使用,不允许删除!`, 1);
+                            return;
+                        };
+                        hintBox.infoBox('系统提示', `确定要删除计算模板“${template.name}”吗?`, 2, cbYes);
+                        function cbYes() {
+                            let data = {
+                                'projectID': projectObj.project.ID(),
+                                'ID': template.ID
+                            };
+                            gatherFeesView.deleteTemplate(data, function (rst) {
+                                if (rst){
+                                    let idx = gatherFeesView.mainSheet.getActiveRowIndex();
+                                    projectObj.project.calcProgram.templates.splice(idx, 1);
+                                    projectObj.project.calcProgram.compileTemplateMaps();
+                                    gatherFeesView.buildSheet();
+                                    gatherFeesView.mainSheet.setSelection(idx - 1, 0, 1, 1);
+                                    gatherFeesView.mainSheet.showRow(idx - 1, GC.Spread.Sheets.VerticalPosition.center);
+                                    gatherFeesView.refreshDetailSheet();
+                                    $.bootstrapLoading.end();
+                                }
+                            });
+                        };
+                    }
+                }
+            }
+        });
+    },
+    loadDetailContextMenu: function () {
+        $.contextMenu({
+            selector: '#gfDetailSpread',
+            build: function ($triggerElement, event) {
+                SheetDataHelper.safeRightClickSelection($triggerElement, event, gatherFeesView.detailSpread);
+            },
+            items: {
+                "newCalcItem": {
+                    name: "插入行",
+                    icon: 'fa-sign-in',
+                    callback: function () {
+                        $.bootstrapLoading.start();
+                        let template = gatherFeesView.getSelectionInfo().template;
+                        let idx = gatherFeesView.detailSpread.getActiveSheet().getActiveRowIndex();
+
+                        let newItem = {};
+                        newItem.ID = analyzer.calcItemMaxID(template) + 1;
+                        // newItem.name = '新建';
+                        newItem.memo = '自定义';
+                        newItem.custom = true;
+                        newItem.expression = '0';                           // 这里必须得有值,否则程序内部公式解析会出现诸多问题
+                        template.calcItems.splice(idx + 1, 0, newItem);
+                        let data = {
+                            'projectID': projectObj.project.ID(),
+                            'ID': template.ID,
+                            'calcItems': template.calcItems
+                        };
+                        gatherFeesView.updateTemplate(data, function (rst) {
+                            if (rst){
+                                projectObj.project.calcProgram.compileTemplate(template);
+                                gatherFeesView.refreshDetailSheet();
+                                gatherFeesView.detailSpread.getActiveSheet().setSelection(idx + 1, 0, 1, 1);
+                                let relationNodes = calcTools.getNodesByProgramID(template.ID);
+                                projectObj.project.calcProgram.calcNodesAndSave(relationNodes);
+                            }
+                        });
+                        $.bootstrapLoading.end();
+                    }
+                },
+                "spr1": '--------',
+                "deleteCalcItem": {
+                    name: '删除行',
+                    icon: 'fa-remove',
+                    callback: function () {
+                        $.bootstrapLoading.start();
+                        let template = gatherFeesView.getSelectionInfo().template;
+                        let idx = gatherFeesView.detailSpread.getActiveSheet().getActiveRowIndex();
+                        let item = template.calcItems[idx];
+
+                        if (item.fieldName == 'common'){
+                            $.bootstrapLoading.end();
+                            hintBox.infoBox('系统提示', `费用类别为“工程造价”的行不允许删除!`, 1);
+                            return;
+                        };
+
+                        if (analyzer.calcItemIsUsed(template, item)){
+                            $.bootstrapLoading.end();
+                            let s = hintBox.font(item.tempUsed + 1);
+                            hintBox.infoBox('系统提示', `第 ${idx + 1} 行“${item.name}”已被第 ${s} 行引用,不允许删除!`, 1);
+                            delete item.tempUsed;
+                            return;
+                        };
+
+                        hintBox.infoBox('系统提示', `确定要删除计算规则“${item.name}”吗?`, 2, cbYes);
+                        function cbYes() {
+                            template.calcItems.splice(idx, 1);
+                            let data = {
+                                'projectID': projectObj.project.ID(),
+                                'ID': template.ID,
+                                'calcItems': template.calcItems
+                            };
+                            gatherFeesView.updateTemplate(data, function (rst) {
+                                if (rst){
+                                    projectObj.project.calcProgram.compileTemplate(template);
+                                    gatherFeesView.refreshDetailSheet();
+                                    let relationNodes = calcTools.getNodesByProgramID(template.ID);
+                                    projectObj.project.calcProgram.calcNodesAndSave(relationNodes);
+                                    $.bootstrapLoading.end();
+                                }
+                            });
+                        };
+                    }
+                }
+            }
+        });
+
+    },
+    saveCalcItem: function (data, callback) {//data
+        CommonAjax.post('/calcProgram/saveCalcItem', data,
+            function (result) {
+                if(callback){
+                    callback(result);
+                }
+            }
+        );
+    },
+    updateTemplate: function (data, callback) {
+        CommonAjax.post('/calcProgram/updateTemplate', data,
+            function (result) {
+                if(callback){
+                    callback(result);
+                }
+            }
+        );
+    },
+    addTemplate: function (data, callback) {
+        CommonAjax.post('/calcProgram/addTemplate', data,
+            function (result) {
+                if(callback){
+                    callback(result);
+                }
+            }
+        );
+    },
+    deleteTemplate: function (data, callback) {
+        $.bootstrapLoading.start();
+        CommonAjax.post('/calcProgram/deleteTemplate', data,
+            function (result) {
+                if(callback){
+                    callback(result);
+                }
+                $.bootstrapLoading.end();
+            }
+        );
+    },
+    getSelectionInfo:function () {
+        var templateIndex = this.mainSpread.getActiveSheet().getActiveRowIndex();
+        var dIndex = this.detailSpread.getActiveSheet().getActiveRowIndex();
+        var info = {
+            template:this.datas[templateIndex],
+            calcItem:this.datas[templateIndex].calcItems[dIndex]
+        }
+        return info;
+    },
+    refreshDetailSheet:function () {
+        var me = this;
+        if(me.mainSpread && me.detailSpread){
+            let mainRowIdx = me.mainSpread.getActiveSheet().getActiveRowIndex();
+            let calcItems = me.datas[mainRowIdx].calcItems;
+            let detailSheet = me.detailSpread.getActiveSheet();
+            detailSheet.setRowCount(calcItems.length);
+            sheetCommonObj.showData(detailSheet, me.detailSetting, calcItems);
+            me.getfeeRateColor(calcItems);
+            customRowHeader(detailSheet, calcItems.length);
+        }
+    },
+    getfeeRateColor: function (calcItems) {    // 有费率ID关联的变个色
+        var me = this;
+        for (let i = 0; i < calcItems.length; i++) {
+            if (calcItems[i].feeRateID != undefined && calcItems[i].feeRateID != null)
+                me.detailSheet.getCell(i, 3).foreColor("#0aa8ea")
+            else
+                me.detailSheet.getCell(i, 3).foreColor("black");
+        }
+    }
+};
+
+$(document).ready(function(){
+    $('#tab_gather_fees').on('shown.bs.tab', function (e) {
+        sessionStorage.setItem('mainTab', '#tab_gather_fees');
+        $(e.relatedTarget.hash).removeClass('active');
+        if (!gatherFeesView.mainSpread)
+            gatherFeesView.buildSheet()
+        else
+            gatherFeesView.mainSpread.refresh();
+
+        let count = gatherFeesView.datas[gatherFeesView.mainSheet.getActiveRowIndex()].calcItems.length;
+        gatherFeesView.detailSheet.setRowCount(count, GC.Spread.Sheets.SheetArea.viewport);
+    });
+
+    $("#calcProgramFileSelect").change(function() {
+        // 取标准库数据过来显示。
+        let libID = $(this).val();
+        if (libID == ''){
+            gatherFeesView.mainSpread.getSheet(0).setRowCount(0);
+            gatherFeesView.detailSpread.getSheet(0).setRowCount(0);
+            return false;
+        };
+
+        $.bootstrapLoading.start();
+        libID = parseFloat(libID);
+        let projectID = projectObj.project.ID();
+        CommonAjax.post('/calcProgram/updateTemplateFile', {"projectID": projectID, "libID": libID}, function (data) {
+            projectObj.project.calcProgram.datas.templates = data;
+            projectObj.project.calcProgram.compileAllTemps();
+            projectObj.project.calcProgram.calcAllNodesAndSave();
+            gatherFeesView.buildSheet();
+            $.bootstrapLoading.end();
+        });
+    });
+});
+
+