首页 > 解决方案 > 谷歌表格单元格重新计算

问题描述

我正在创建一个棋盘游戏,我决定为此选择 Google 表格。我已将我的问题简化为一个由一张纸和一个脚本组成的最小示例。


情况

以下几点参考我的骰子表:


问题

问题是,当我更改C列中的骰子面时,DICEFACE不会重新计算公式。就在我创建屏幕截图之前,我在单元格C2中添加了,4后缀,您可以看到单元格N2中没有。但是,如果我重新保存脚本文件更改E2:I2中的骰子则会立即进行重新计算。4Code.gs

我很确定我知道问题出在哪里:因为我正在遍历脚本中的单元格,所以工作表应用程序本身看不到C列中的单元格与K2K10中的公式之间的引用链接。查看我的工作表,单元格引用可能类似于:

K4  <-- E2:I2   <-- B2, B3 (C is not here)
K10 <-- E10:I10 <-- B4, B5 (C is not here)

我的符号的意思A <-- BIf there's a change in range B, update cell A.


问题

修改骰子面后,我应该更改什么以使自动重新计算立即发生?如果这是不可能的,那么完成我的任务的最佳方法是什么?

标签: google-apps-scriptgoogle-sheetsgoogle-sheets-formulacustom-function

解决方案


问题是,当我更改C列中的骰子面时,DICEFACE不会重新计算公式。

DIFACE 是一个自定义函数,当打开电子表格并且自定义函数参数值发生变化时,会重新计算自定义函数。

考虑到上述情况,为了最大限度地减少对自定义函数的更改,请添加第二个参数作为触发器。

更改正则表达式

/=\w+\((.*)\)/i

/=\w+\((.*),.*\)/i

然后通过以下方式调用您的自定义函数

=DICEFACES(E2:I2,C2)

或者

=DICEFACES(E2:I2,C2:C5)


我使用逗号假设它使用的是 Google 表格参数分隔符,但某些电子表格可以使用分号。


修改版的OP自定义函数

/**
 * Returns a matrix of dice faces corresponding to the dices in the provided range.
 *
 * @param {Array} unused_ref_to_range_containing_dices Reference to range. i.e. E2:I2
 * @param {String|Number|Date|Array} ref_as_trigger Reference to a range used as trigger. i.e. C2 or C2:C5
 * @return array
 * @customfunction
 */
function DICEFACES(unused_ref_to_range_containing_dices,ref_as_trigger)
{
  var app  = SpreadsheetApp;
  var spr  = app.getActiveSheet();

  // In the end this array will hold the dice faces. For example two
  // 1d6 dices would result in [[1,2,3,4,5,6],[1,2,3,4,5,6]].
  //
  var Dices = [];

  // The the formula inside the active cell (i.e. the cell on which
  // we are calling this function). This is a string like:
  //
  // "=DICEFACES(E2:I2)"
  //
  var active_formula = spr.getActiveRange().getFormula();

  // Set item_range to the one pointed to by the formula. This could
  // be a range like E2:I2.
  //
  var item_range = spr.getRange(active_formula.match(/=\w+\((.*),.*\)/i)[1]); // CHANGED

  // Loop over dice cells in the item_range.
  //
  for (var i = 1; i <= item_range.getNumColumns(); i++)
  {
    // "=B2", "=B3", ...
    //
    var dice_formula = item_range.getCell(1, i).getFormula();

    // As soon as we encounter an empty formula, we skip (i.e. there are
    // no more dices).
    //
    if (dice_formula == "")
    {
      break;
    }

    // A reference to the cell containing the dice image. We don't really
    // need the image, the dice faces are of greater importance to us.
    //
    var dice_cell = spr.getRange(dice_formula.substr(1));

    // Move one column to the right prior to the dice_cell and retreive
    // the value of the cell. This is a string like "1,2,3,4,5,6".
    //
    var dice_csv = dice_cell.offset(0, 1).getValue();

    // Convert the CSV string to a javascript array like [1,2,3,4,5,6]
    // and push it to Dices.
    //
    Dices.push(dice_csv.split(",").map(Number));
  }
  return Dices;
}

推荐阅读