7

目前undo(),Spreadsheet/Sheet/Range 类中没有 Google Apps 脚本的功能。在 Issue Tracker 上打开了几个问题,我现在只能找到一个(我不知道 Triaged 是什么意思):这里

有人建议使用 DriveApp 和修订历史的解决方法,但我环顾四周并没有发现任何东西(也许它被埋没了?)。无论如何,undo()对于这么多不同的操作来说,一个函数是非常必要的。我只能想到一种解决方法,但我无法让它工作(数据存储的方式,我什至不知道它是否可能)。这是一些伪 -

function onOpen () {
  // Get all values in the sheet(s)
  // Stringify this/each (matrix) using JSON.stringify
  // Store this/each stringified value as a Script or User property (character limits, ignore for now)
}

function onEdit () {
  // Get value of edited cell
  // Compare to some value (restriction, desired value, etc.)
  // If value is not what you want/expected, then:
  // -----> get the stringified value and parse it back into an object (matrix)
  // -----> get the old data of the current cell location (column, row)
  // -----> replace current cell value with the old data
  // -----> notifications, coloring cell, etc, whatever else you want
  // If the value IS what you expected, then:
  // -----> update the 'undoData' by getting all values and re-stringifying them
  //        and storing them as a new Script/User property
}

基本上,当电子表格打开时,将所有值存储为脚本/用户属性,并且仅在满足某些单元格条件(打开)时才引用它们。当您要撤消时,获取存储在当前单元格位置的旧数据,并将当前单元格的值替换为旧数据。如果不需要撤消该值,则更新存储的数据以反映对电子表格所做的更改。

到目前为止,我的代码已经失败,我认为这是因为当对象被字符串化和存储(例如,它没有正确解析)时,嵌套数组结构丢失了。如果有人写过这种功能,请分享。否则,有关如何编写此内容的建议将很有帮助。

编辑:这些文件是令人难以置信的静态。行/列的数量不会改变,数据的位置也不会改变。如果可能的话,为临时修订历史实现一个get-all-data/store-all-data类型的函数实际上会满足我的需要。

4

3 回答 3

9

当我需要保护工作表但允许通过侧边栏进行编辑时,我遇到了类似的问题。我的解决方案是准备两张纸(一张隐藏)。如果您编辑第一个工作表,这将触发 onEdit 过程并重新加载第二个工作表中的值。如果您取消隐藏并编辑第二个工作表,它会从第一个工作表重新加载。完美地工作,并且非常有趣地删除大量数据并观看它自我修复!

于 2014-10-31T15:22:08.947 回答
5

只要您不添加或删除行和列,您就可以将行号和列号作为您存储在 ScriptDb 中的历史值的索引。

function onEdit(e) {
  // Exit if outside validation range
  // Column 3 (C) for this example
  var row = e.range.getRow();
  var col = e.range.getColumn();
  if (col !== 3) return;
  if (row <= 1) return; // skip headers

  var db = ScriptDb.getMyDb();

  // Query database for history on this cell
  var dbResult = db.query({type:"undoHistory",
                       row:row,
                       col:col});
  if (dbResult.getSize() > 0) {
    // Found historic value
    var historicObject = dbResult.next();
  }
  else {
    // First change for this cell; seed historic value
    historicObject = db.save({type:"undoHistory",
                              row:row,
                              col:col,
                              value:''});
  }

  // Validate the change.
  if (valueValid(e.value,row,col)) {
    // update script db with this value
    historicObject.value = e.value;
    db.save(historicObject);
  }
  else {
    // undo the change.
    e.range.getSheet()
           .getRange(row,col)
           .setValue(historicObject.value);
  }
}

您需要提供一个验证数据值的函数。同样,在此示例中,我们只关心一列中的数据,因此验证非常简单。例如,如果您需要对不同的列执行不同类型的验证,那么您可以switchcol参数上。

/**
 * Test validity of edited value. Return true if it
 * checks out, false if it doesn't.
 */
function valueValid( value, row, col ) {
  var valid = false;

  // Simple validation rule: must be a number between 1 and 5.
  if (value >= 1 && value <= 5)
    valid = true;

  return valid;
}

合作

此撤消功能适用于协作编辑的电子表格,尽管在脚本数据库中存储历史值存在竞争条件。如果多个用户同时对一个单元格进行第一次编辑,则数据库最终可能会出现多个表示该单元格的对象。在随后的更改中,使用query()和选择仅选择第一个结果可确保仅选择其中一个倍数。

如果这成为一个问题,可以通过将函数包含在 Lock 中来解决。

于 2013-08-23T16:12:32.927 回答
1

当用户选择多个单元格时,修改了组中的答案以允许范围:

我使用了我称之为“双张纸”的东西。

/**
 * Test function for onEdit. Passes an event object to simulate an edit to
 * a cell in a spreadsheet.
 * Check for updates: https://stackoverflow.com/a/16089067/1677912
 */
function test_onEdit() {
  onEdit({
    user : Session.getActiveUser().getEmail(),
    source : SpreadsheetApp.getActiveSpreadsheet(),
    range : SpreadsheetApp.getActiveSpreadsheet().getActiveCell(),
    value : SpreadsheetApp.getActiveSpreadsheet().getActiveCell().getValue(),
    authMode : "LIMITED"
  });
}


function onEdit() {
  // This script prevents cells from being updated. When a user edits a cell on the master sheet,
  // it is checked against the same cell on a helper sheet. If the value on the helper sheet is
  // empty, the new value is stored on both sheets.
  // If the value on the helper sheet is not empty, it is copied to the cell on the master sheet,
  // effectively undoing the change.
  // The exception is that the first few rows and the first few columns can be left free to edit by
  // changing the firstDataRow and firstDataColumn variables below to greater than 1.
  // To create the helper sheet, go to the master sheet and click the arrow in the sheet's tab at
  // the tab bar at the bottom of the browser window and choose Duplicate, then rename the new sheet
  // to Helper.
  // To change a value that was entered previously, empty the corresponding cell on the helper sheet,
  // then edit the cell on the master sheet.
  // You can hide the helper sheet by clicking the arrow in the sheet's tab at the tab bar at the
  // bottom of the browser window and choosing Hide Sheet from the pop-up menu, and when necessary,
  // unhide it by choosing View > Hidden sheets > Helper.
  // See https://productforums.google.com/d/topic/docs/gnrD6_XtZT0/discussion

  // modify these variables per your requirements
  var masterSheetName = "Master" // sheet where the cells are protected from updates
  var helperSheetName = "Helper" // sheet where the values are copied for later checking

  var ss = SpreadsheetApp.getActiveSpreadsheet();
  var masterSheet = ss.getActiveSheet();
  if (masterSheet.getName() != masterSheetName) return;

  var masterRange = masterSheet.getActiveRange();

  var helperSheet = ss.getSheetByName(helperSheetName);
  var helperRange = helperSheet.getRange(masterRange.getA1Notation());
  var newValue = masterRange.getValues();
  var oldValue = helperRange.getValues();
  Logger.log("newValue " + newValue);
  Logger.log("oldValue " + oldValue);
      Logger.log(typeof(oldValue));
  if (oldValue == "" || isEmptyArrays(oldValue)) {
    helperRange.setValues(newValue);
  } else {
    Logger.log(oldValue);
    masterRange.setValues(oldValue);

  }
}

// In case the user pasted multiple cells this will be checked
function isEmptyArrays(oldValues) {
  if(oldValues.constructor === Array && oldValues.length > 0) {
    for(var i=0;i<oldValues.length;i++) {
      if(oldValues[i].length > 0 && (oldValues[i][0] != "")) {
          return false; 
      }
    }
  }
  return true;
}
于 2017-09-18T23:24:34.953 回答