Google Apps 脚本 - 将匿名函数传递给 Menu.addItem

Ari*_*rad 3 javascript eval google-apps-script


我正在为一所学校构建一个容器绑定的谷歌应用程序脚本。学校要求每一个书面项目都有“学校标题”。学校使用 AG 块作为句点。我的目标是会有一个菜单“学校标题”,打开时有子菜单“A 块”、“B 块”、“C 块”,并且在每个子菜单中每个班级都会有一个选项,其中将该类的标题插入到文档的标题中

这是我的代码:

var BLOCKS = "abcdefg";
var CLASSES = ["English", "History", "Science", "Writing", "Latin", "Math", "Study Skills"];
var FUNCTION_NAMES;
var global = {};

init();

/**
 * The onOpen function runs automatically when the Google Docs document is
 * opened. Use it to add custom menus to Google Docs that allow the user to run
 * custom scripts. For more information, please consult the following two
 * resources.
 *
 * Extending Google Docs developer guide:
 *     https://developers.google.com/apps-script/guides/docs
 *
 * Document service reference documentation:
 *     https://developers.google.com/apps-script/reference/document/
 */
function onOpen() {
  init();
  // Add a menu with some items, some separators, and a sub-menu.
  var menu = DocumentApp.getUi().createMenu('School Heading')
  for(var i = 0; i < BLOCKS.length; i++){
    block = BLOCKS[i];
    Logger.log("Block: " + block)
    menu = menu.addSubMenu(DocumentApp.getUi().createMenu(block + " Block")
                           .addItem('English', 'eng' + block)
                           .addItem('History', 'his' + block)
                           .addItem('Science', 'sci' + block)
                           .addItem('Writing', 'wri' + block)
                           .addItem('Latin', 'lat' + block)
                           .addItem('Math', 'mat'  + block) 
                           .addItem('Study Skills', 'stu' + block));
    defineFunctions(block, this);
  }
  menu.addToUi();
}

function getFunc(class,block){
  return function(){
    createHeading(class,block);
  }
}

function defineFunctions(block, global){
  Logger.log(FUNCTION_NAMES)
  for(var i = 0; i < FUNCTION_NAMES.length; i++){
    var funcName = FUNCTION_NAMES[i] + block;
    eval("function " + funcName + " () { createHeading('"+ CLASSES[i] + "', '" + block + "'); }");
  }
}

function createHeading(class, block){
  var header = DocumentApp.getActiveDocument().getHeader();
  if(!header){
    header = DocumentApp.getActiveDocument().addHeader();
  }
  header.insertParagraph(0, "Name\n{class}, Block {block}".replace("{class}", class).replace("{block}", block)).setAlignment(DocumentApp.HorizontalAlignment.RIGHT);
}

function init(){
  if(!Array.isArray(BLOCKS)){
    BLOCKS = BLOCKS.toUpperCase().split("");
  }
  if(!Array.isArray(CLASSES)){
    CLASSES = CLASSES.split("\n");
  }
  if(!Array.isArray(FUNCTION_NAMES) || FUNCTION_NAMES.length !== CLASSES.length){
    FUNCTION_NAMES = [];
    for(var i = 0; i < CLASSES.length; i++){
      FUNCTION_NAMES.push(CLASSES[i].toLowerCase().substring(0,3));
    }
  }
}
Run Code Online (Sandbox Code Playgroud)

当我选择学校标题 > A 块 > 英语时,我收到“找不到脚本函数 engA”

我的问题是,为什么 eval 不起作用,我是否可以将匿名函数传递给 Menu.addItem?

pat*_*tt0 5

我相信 eval 可能有效,但不能保证它与调用菜单时执行的脚本实例相同。在您准备好调用菜单之前,Google Apps 脚本无法将所有这些函数保留在内存中,因此到那时,您就拥有了一个干净的状态,所有这些函数都不再被引用。

现阶段无法传递匿名函数/参数,它被归类为增强功能:https://code.google.com/p/google-apps-script-issues/issues/detail ?id=477 不多但在这方面取得了进展。

为什么不使用侧边栏而不是菜单,您可以在其中创建带有选择器的 UI,基于该选择器可以插入正确的标题。一旦你有了这个机制,你就可以使用侧边栏来处理其他很酷的东西。

https://developers.google.com/apps-script/guides/docs