google-apps-script - 在多个条件下一次移动多行
问题描述
我想用 for 循环一次遍历每一行。当行在 P 列中具有值“完成”并且在 M 列中具有“更新”值时,我想将该行移动到完成表,如果该行在 P 中只有“完成”,我希望它移动到控制表。
function done() {
var ss = SpreadsheetApp.getActiveSpreadsheet();
var s = ss.getActiveSheet();
var numColumns = s.getLastColumn();
for(var i = 1; i < 50; i++) {
var cP = s.getRange(i, 13).getValue();
var cM = s.getRange(i, 10).getValue();
if (cP == "Done" && cM == "Update") {
var targetSheet = ss.getSheetByName("Done Sheet"); // Logger.log(targetSheet); —> no output
var target = targetSheet.getRange(targetSheet.getLastRow() + 1, 1);
s.getRange(i, 1, 1, numColumns).moveTo(target);
s.deleteRow(i);
}
else if (cP == "Done") {
var targetSheet = ss.getSheetByName("Control Sheet");
var target = targetSheet.getRange(targetSheet.getLastRow() + 1, 1);
s.getRange(i, 1, 1, numColumns).moveTo(target);
s.deleteRow(i);
}
}
}
解决方案
请检查此代码是否适合您:
function done() {
var ss = SpreadsheetApp.getActiveSpreadsheet();
var s = ss.getSheetByName('Main Sheet'); // ss.getActiveSheet(); <= better call by name
var numColumns = s.getLastColumn();
var values = s.getDataRange().getValues(); // values keeps an array from the whole Sheet
var doneSheet = ss.getSheetByName("Done Sheet");
var ctrlSheet = ss.getSheetByName("Control Sheet");
var rows_to_delete = []; // will receive the indexes to delete
var doneSheetInputs = []; // will receive data to be inputed on doneSheet
var ctrlSheetInputs = []; // will receive data to be inputed on ctrlSheet
for(var i = 0; i < values.length; i++) {
var cP = values[i][15]; //s.getRange(i, 13).getValue();
var cM = values[i][12]; //s.getRange(i, 10).getValue();
if (cP == "Done" && cM == "Update") {
doneSheetInputs.push(values[i]);
rows_to_delete.push(i + 1); // +1 because of difference in Array versus Google Sheets index
Logger.log('Moving row ' + (i + 1) + ' to DONE');
}
else if (cP == "Done") {
ctrlSheetInputs.push(values[i]);
rows_to_delete.push(i + 1);
Logger.log('Moving row ' + (i + 1) + ' to CONTROL');
}
}
Logger.log('doneSheetInputs has ' + doneSheetInputs.length + ' entries');
if (doneSheetInputs.length > 0) {
doneSheet.getRange(doneSheet.getLastRow() + 1, 1, doneSheetInputs.length, doneSheetInputs[0].length).setValues(doneSheetInputs);
}
Logger.log('ctrlSheetInputs has ' + ctrlSheetInputs.length + ' entries');
if (ctrlSheetInputs.length > 0) {
ctrlSheet.getRange(ctrlSheet.getLastRow() + 1, 1, ctrlSheetInputs.length, ctrlSheetInputs[0].length).setValues(ctrlSheetInputs);
}
Logger.log('rows_to_delete has ' + rows_to_delete.length + ' entries');
var rev = rows_to_delete.reverse(); // delete from bottom to top
for (var i = 0; i < rev.length; i++) {
s.deleteRow(rev[i]);
}
}
不建议调用getRange().getValues()
这么多次,特别是在Google Apps 脚本最佳实践for
中描述的长语句中。我总是尝试在进行任何迭代之前调用并在代码末尾以批处理操作返回值。getValues
编辑:发现 P 和 M 列的索引是错误的 - 没有检查出来。这是一个工作示例:https ://docs.google.com/spreadsheets/d/1n68KqmI7HongEzJ_MSwmeQUqlAExjvCu2epEjEBZrlg/edit#gid=0 (如果需要,请向我发送访问请求)。
推荐阅读
- html - 窗口调整大小时的动画定位
- python - 我正在尝试使用并发期货进行线程处理,但似乎有些期货没有返回,它们会无限期地运行
- javascript - Angular中的条件更改检测策略?
- android - 更改 PlaceAutoCompleteFragment 提示时出现 NullPointerException
- php - 更新数据时,RouteCollection.php 第 251 行中的 MethodNotAllowedHttpException laravel
- android - 颤动动画在某些设备上运行非常慢?
- c++ - 如何仅使用 c++ 中的循环打印此代码的垂直直方图?
- java - 计算recyclerview中物品的总价
- react-native - FlatList 标头中有多个元素
- scala - 了解 Spark 流作业的 JVM 指标