条件格式脚本-设置迭代

时间:2018-11-07 07:48:39

标签: google-sheets formatting conditional

我有一个用于条件格式设置规则的自定义公式。我正在尝试编写一个脚本,该脚本检查一列(“地雷”表上的B列)上的多个值(大约50个),并且如果单元格等于特定字符串(M1,M2或M3),则使用指定的公式用于条件格式的格式将应用于“日历视图”表。我目前拥有的代码是:

function myFunction() {
var ss = SpreadsheetApp.getActiveSpreadsheet();
var sheet = ss.getSheetByName("Calendar View");
sheet.getRange("C4:NC50").clearFormat();
var range = sheet.getRange("C4:NC4");
var rule = SpreadsheetApp.newConditionalFormatRule()
    .whenFormulaSatisfied('=AND(indirect("Mine!$B5")="M1", C$2>=indirect("Mine!$C5"), C$2<indirect("Mine!$D5"))')
    .setBackground("#FF0000")
    .setRanges([range])
    .build();
var rules = sheet.getConditionalFormatRules();
rules.push(rule);
sheet.setConditionalFormatRules(rules);
}

如何在.whenFormulaSatisfied上输入迭代方法,例如:

.whenFormulaSatisfied('=AND(indirect("Mine!$B6")="M1", C$2>=indirect("Mine!$C6"), C$2<indirect("Mine!$D6"))')
.whenFormulaSatisfied('=AND(indirect("Mine!$B7")="M1", C$2>=indirect("Mine!$C7"), C$2<indirect("Mine!$D7"))')
.whenFormulaSatisfied('=AND(indirect("Mine!$B8")="M1", C$2>=indirect("Mine!$C8"), C$2<indirect("Mine!$D8"))')
............

这是我正在处理的工作表: https://docs.google.com/spreadsheets/d/1Af84aHaG0VjXmtaWc0-uAdGFrX1LozRNLQLMatSOqgU/edit?usp=sharing

1 个答案:

答案 0 :(得分:0)

发问者的方法面临一些挑战-首先是动态标识每个属性的开始日期和结束日期,其次是创建多达50个独立的条件格式设置规则。众所周知,电子表格的性能会受到许多条件格式设置规则的影响。

我建议使用稍微不同的方法。
1)在Mine上获取数据并构建日历。
2)将值放在预订的日期字段中。
3)为日历应用单个条件格式设置规则。

用于标识预定日期的方法是在各个单元格中插入一个名义值。然后应用规则.whenCellNotEmpty()而不是指定特定值。此外,该代码还设置了背景以及字体颜色的格式,以便隐藏任何数据。

还要注意:在脚本的开头,该代码同时删除了内容和格式。

function so_53185335() {
    // build the spreadsheet app and set source and target sheets
    var ss = SpreadsheetApp.getActiveSpreadsheet();
    var calSheet = ss.getSheetByName("Calendar View");
    var dataSheet = ss.getSheetByName("Mine");

    // get the last rows and start rows for both sheets 
    var lrMine = dataSheet.getLastRow();
    var lrCal = calSheet.getLastRow();
    var dataRowStart = 5;
    var calRowStart = 4;

    // clear formats and data from Calendar  
    calSheet.getRange(calRowStart, 2, lrCal, 366).clear({
        contentsOnly: true,
        formatOnly: true
    });

    // get Mine rows with data, define the range and  get data
    var dataRows = lrMine - dataRowStart + 1;
    //Logger.log("Mine: number of data rows "+dataRows);// DEBUG  
    var dataRange = dataSheet.getRange(dataRowStart, 2, dataRows, 3);
    //Logger.log("data range is "+dataRange.getA1Notation());// DEBUG
    var dataValues = dataRange.getValues();

    //set some variables for use in loop  
    var i = 0; // counter
    var z = 0; // counter
    var calstartCol = 3; // equals first day of the year
    var calrow = 0; // counter row for Calendar sheet
    var calArray = [];
    var masterArray = [];

    // loop through the rows in Mine

    for (i = 0; i < dataRows; i++) {
        // test for value 
        if (dataValues[i][0] === "M1" || dataValues[i][0] === "M2" || dataValues[i][0] === "M3") {
            //Logger.log("Match: i="+i+", value = "+dataValues[i][0]);//DEBUG
            calArray = [];
            masterArray = [];
            calrow = calrow + 1;

            // calculate the start day (as a day in the year)
            var now = new Date(dataValues[i][1]);
            var start = new Date(now.getFullYear(), 0, 0);
            var diff = (now - start) + ((start.getTimezoneOffset() - now.getTimezoneOffset()) * 60 * 1000);
            var oneDay = 1000 * 60 * 60 * 24;
            var startday = Math.floor(diff / oneDay);

            // calculate the end day (as a day in the year)
            var fnow = new Date(dataValues[i][2]);
            var fstart = new Date(fnow.getFullYear(), 0, 0);
            var fdiff = (fnow - fstart) + ((fstart.getTimezoneOffset() - fnow.getTimezoneOffset()) * 60 * 1000);
            var foneDay = 1000 * 60 * 60 * 24;
            var endday = Math.floor(fdiff / foneDay);
            var nod = endday - startday + 1;

            // assign the value for the Property
            var cell = calSheet.getRange(calstartCol + calrow, 2);
            cell.setValue(dataValues[i][0]);

            // create an array of values for booked dates; just insert the number "1"
            for (z = 1; z < nod + 1; z++) {
                calArray.push(1);
            }
            masterArray.push(calArray);

            // Assign the values for booked dates
            var cell = calSheet.getRange(calstartCol + calrow, startday + 2, 1, nod);
            cell.setValue(masterArray);
        }

    }

    // create and apply a single Conditional forma rule for the data range on Calendar
    var range = calSheet.getRange(calRowStart, calstartCol, calstartCol + calrow, 366);
    var rule = SpreadsheetApp.newConditionalFormatRule()
        .whenCellNotEmpty()
        .setFontColor("#FF0000")
        .setBackground("#FF0000")
        .setRanges([range])
        .build();
    var rules = calSheet.getConditionalFormatRules();
    rules.push(rule);
    calSheet.setConditionalFormatRules(rules);

}

日历看起来像这样。


enter image description here