使用特定于应用程序的 API 模型

本文介绍如何使用 API 模型在 Excel、OneNote、PowerPoint、Visio 和 Word 中生成加载项。 本文介绍核心概念,这些概念是使用基于承诺的 API 的基础。

注意

Outlook 或 Project 客户端不支持此模型。 使用 通用 API 模型 处理这些应用程序。 有关完整的平台可用性说明,请参阅 Office 客户端应用程序和平台可用性的 Office 加载项组

提示

本页中的示例使用 Excel JavaScript API,但概念也适用于 OneNote、PowerPoint、Visio 和 Word JavaScript API。 有关演示如何在各种 Office 应用程序中使用这些概念和其他概念的完整代码示例,请参阅 Office 加载项代码示例

基于承诺的 API 的异步性质

Office 加载项是显示在 Office 应用程序(如 Excel)中的 Web 视图控件内的网站。 此控件嵌入在基于桌面的 Office 应用程序(如 Windows 上的 Office)上的 Office 应用程序中,在 Office 网页版中的 HTML iFrame 内运行。 出于性能方面的考虑,Office.js API 无法跨所有平台与 Office 应用程序同步交互。 因此,Office.js sync() API 调用返回 Office 请求的读取或写入操作时要解决的"承诺"问题。 此外,还可以将多个操作(例如设置属性或调用方法)排队,并且只要一次呼叫 sync(),就可以作为一批命令运行这些操作,而不是针对每个操作发送单独的请求。 以下各节介绍如何使用 API 和 run()sync()实现此操作。

*.run 函数

Excel.runOneNote.runPowerPoint.runWord.run 执行一个函数,指定针对 Excel、Word 和 OneNote 要执行的操作。 *.run 会自动创建可用于与 Office 对象交互的请求上下文。 当 *.run ,将做出承诺,并自动发布运行时分配的任何对象。

以下示例显示了如何使用 Excel.run。 OneNote、PowerPoint、Visio 和 Word 也使用相同的模式。

Excel.run(function (context) {
    // Add your Excel JS API calls here that will be batched and sent to the workbook.
    console.log('Your code goes here.');
}).catch(function (error) {
    // Catch and log any errors that occur within `Excel.run`.
    console.log('error: ' + error);
    if (error instanceof OfficeExtension.Error) {
        console.log('Debug info: ' + JSON.stringify(error.debugInfo));
    }
});

请求上下文

Office 应用程序和加载项在不同过程中运行。 由于加载项使用不同的运行时环境,因此需要一个 RequestContext 对象才能将加载项连接到 Office 中的对象,例如工作表、区域、段落和表。 调用 RequestContext 时,此对象作为 *.run

代理对象

声明并用于基于承诺的 API 的 Office JavaScript 对象是代理对象。 调用的任何方法或在代理对象上设置或加载的属性都只是添加到挂起命令的队列中。 调用请求 sync() (例如 context.sync())上的方法时,排队的命令将调用 Office 应用程序并运行。 这些 API 在根本上以批处理为中心。 您可以根据对请求上下文希望排入多达数个更改,然后调用 sync() 方法,以运行排队命令的批处理。

例如,以下代码片段声明本地 JavaScript Excel.Range 对象 selectedRange引用 Excel 工作簿中的选定区域,然后针对该对象设置一些属性。 对象 selectedRange 代理对象,因此在调用加载项之前,不会在 Excel 文档中反映已设置的属性和在该对象上调用 context.sync()

const selectedRange = context.workbook.getSelectedRange();
selectedRange.format.fill.color = "#4472C4";
selectedRange.format.font.color = "white";
selectedRange.format.autofitColumns();

性能提示:最小化创建代理对象的数量

避免重复创建同一个代理对象。 如果多个操作需要同一个代理对象,则改为创建一次并将其分配给一个变量,然后在代码中使用该变量。

// BAD: Repeated calls to .getRange() to create the same proxy object.
worksheet.getRange("A1").format.fill.color = "red";
worksheet.getRange("A1").numberFormat = "0.00%";
worksheet.getRange("A1").values = [[1]];

// GOOD: Create the range proxy object once and assign to a variable.
const range = worksheet.getRange("A1");
range.format.fill.color = "red";
range.numberFormat = "0.00%";
range.values = [[1]];

// ALSO GOOD: Use a "set" method to immediately set all the properties
// without even needing to create a variable!
worksheet.getRange("A1").set({
    numberFormat: [["0.00%"]],
    values: [[1]],
    format: {
        fill: {
            color: "red"
        }
    }
});

sync()

调用 sync() 上下文的方法可同步 Office 文档中代理对象和对象之间的状态。 该 sync() 在请求上下文中排入队列的任何命令,并检索应在代理对象上加载的任何属性的值。 方法 sync() 异步执行,并返回 承诺,该 sync() 完成时完成。

下面的示例显示一个批处理函数,定义本地 JavaScript 代理对象 (selectedRange),加载该对象的属性,然后使用 JavaScript 形式调用 context.sync() 以在 Excel 文档中的代理对象和对象间同步状态。

await Excel.run(async (context) => {
    const selectedRange = context.workbook.getSelectedRange();
    selectedRange.load('address');
    await context.sync();
    console.log('The selected range is: ' + selectedRange.address);
});

在上一示例中,已设置 selectedRange,并且将在调用 context.sync() 时加载其 address 属性。

由于 sync() 是异步操作,因此在脚本继续运行之前,应始终 Promise 同步对象,以确保 sync() 操作完成。 如果使用 TypeScript 或 ES6+ JavaScript, await 调用 context.sync() ,而不是返回承诺。

性能提示:减少同步呼叫数

在 Excel JavaScript API 中,sync() 是唯一的异步操作,在某些情况下可能会很慢,尤其是对于 Excel 网页版。 若要优化性能,在调用之前,通过尽可能多地将更改加入队列来最大程度减少调用 sync() 的次数。 有关使用 sync()优化性能,请参阅 循环使用 context.sync 方法

load()

必须显式加载属性才能读取代理对象的属性,才能使用 Office 文档中的数据填充代理对象,然后调用 context.sync()。 例如,如果创建代理对象来引用选定的区域,然后希望读取所选区域的 address 属性,需要首先加载 address 属性,然后才可以读取它。 若要请求加载代理对象的属性,调用 load() 的方法并指定要加载的属性。 以下示例显示了为 Range.address 加载的 myRange

await Excel.run(async (context) => {
    const sheetName = 'Sheet1';
    const rangeAddress = 'A1:B2';
    const myRange = context.workbook.worksheets.getItem(sheetName).getRange(rangeAddress);

    myRange.load('address');
    await context.sync();
      
    console.log (myRange.address);   // ok
    //console.log (myRange.values);  // not ok as it was not loaded

    console.log('done');
});

注意

如果只是调用代理对象或设置属性,则无需调用代理 load() 方法。 只有在想要读取代理对象上的属性时 load() 代理方法才必需。

类似于对代理对象设置属性或调用方法的请求,加载代理对象属性的请求会被添加到请求上下文的挂起命令队列中,将在下一次调用 sync() 方法时运行。 必要时,可以将请求上下文中尽可能多的 load() 调用排入队列。

标量和导航属性

属性分为两种类别:标量导航。 标量属性是可分配的类型,如字符串、整数和 JSON 结构。 导航属性是分配了字段的只读对象和对象集合,而不是直接分配属性。 例如,Worksheet 对象上的 nameposition 成员是标量属性,而 protectiontables 是导航属性。

加载项可使用导航属性作为加载特定标量属性的路径。 以下代码会按照 load 对象使用的字体的名称将向上排队 Excel.Range 命令,而无需加载任何其他信息。

someRange.load("format/font/name")

还可通过遍历路径来设置导航属性的标量属性。 例如,通过使用"另一种" Excel.RangesomeRange.format.font.size = 10;。 设置属性前无需加载属性。

请注意,一个对象下的某些“属性”可能与另一个对象同名。 例如, format 是对象下 Excel.Range 属性, format 值本身也是一个对象。 因此,如果你进行 range.load("format")等呼叫,这相当于 range.format.load() (一个空的 load() 语句)。 若要避免这种情况,代码应仅加载对象树中的“叶节点”。

从集合加载

使用集合时,使用 load 集合加载集合中每个对象的属性。 与对该集合中的单个对象使用 load 完全相同。

下面的示例代码演示为“示例”工作表中的每个图表加载和记录的 name 属性。

await Excel.run(async (context) => {
    const sheet = context.workbook.worksheets.getItem("Sample");
    const chartCollection = sheet.charts;

    // Load the name property on every chart in the chart collection.
    chartCollection.load("name");
    await context.sync();

    chartCollection.items.forEach((chart) => {
        console.log(chart.name);
    });
});

通常不会在 items 参数中包含集合的 load 属性。 如果加载任何项属性,则加载所有项。 但是,如果要循环访问集合中的项,但不需要加载项的任何特定属性,则需要 loaditems 属性。

下面的示例代码演示为“示例”工作表中的每个图表设置的 name 属性。

await Excel.run(async (context) => {
    const sheet = context.workbook.worksheets.getItem("Sample");
    const chartCollection = sheet.charts;

    // Load the items property from the chart collection to set properties on individual charts.
    chartCollection.load("items");
    await context.sync();

    chartCollection.items.forEach((chart, index) => {
        chart.name = `Sample chart ${index}`;
    });
});

如果在不 load() 参数的情况下调用对象(或集合)上的标量方法,将加载该对象或集合对象的所有标量属性。 加载不需要的数据会降低加载项的加载速度。 应始终显式指定要加载的属性。

重要

无参数 load 语句返回的数据量可能超过该服务的大小限制。 为了降低较旧加载项的风险,load 不会在明确请求它们之前返回某些属性。 此类加载操作中排除了以下属性。

  • Excel.Range.numberFormatCategories

ClientResult

基于承诺的 API 中返回类型 API 的方法与现代方法load/sync模式。 举个例子,Excel.TableCollection.getCount获取集合中的表的数量。 getCount 返回一ClientResult<number>,这意味着返回的value中的 ClientResult 属性是一个数字。 在调用 context.sync() 之前,脚本无法访问此值。

以下代码获取 Excel 工作簿中的表总数,并对此数目的日志记录到控制台。

const tableCount = context.workbook.tables.getCount();

// This sync call implicitly loads tableCount.value.
// Any other ClientResult values are loaded too.
await context.sync();

// Trying to log the value before calling sync would throw an error.
console.log (tableCount.value);

set()

在具有嵌套导航属性的对象上设置属性可能很麻烦。 除了使用上述导航路径设置单个属性, object.set() 基于承诺的 JavaScript API 中的对象上可用的另一种方法。 使用此方法,可以通过传递相同 Office.js 类型的另一个对象或 JavaScript 对象(其属性结构类似于调用该方法的对象的属性)一次设置对象的多个属性。

下面的代码示例设置区域的多个格式属性,具体方法是调用 set() 方法,并传入 JavaScript 对象,其中包含可反映 Range 对象中属性结构的属性名称和类型。 此示例假定区域 B2:E2 中有数据。

await Excel.run(async (context) => {
    const sheet = context.workbook.worksheets.getItem("Sample");
    const range = sheet.getRange("B2:E2");
    range.set({
        format: {
            fill: {
                color: '#4472C4'
            },
            font: {
                name: 'Verdana',
                color: 'white'
            }
        }
    });
    range.format.autofitColumns();

    await context.sync();
});

某些属性不能直接设置

尽管可写的属性,但某些属性不能设置。 这些属性是必须将设置为单个对象的父属性的一部分。 这是因为父属性依赖于具有特定逻辑关系的子属性。 必须使用对象文字表示法设置这些父属性来设置整个对象,而不是设置该对象的单个子问题。 PageLayout 中可找到此示例。 必须使用单个 PageLayoutZoomOptions 对象设置 zoom 属性,如此处所示。

// PageLayout.zoom.scale must be set by assigning PageLayout.zoom to a PageLayoutZoomOptions object.
sheet.pageLayout.zoom = { scale: 200 };

在上一示例中,无法直接分配zoom为值:sheet.pageLayout.zoom.scale = 200;。 该语句会引发错误, zoom 加载错误。 即使 zoom ,该比例也会生效。 所有上下文操作 zoom、刷新加载项中的代理对象并覆盖本地设置的值。

此行为与 Range.format导航属性。 你可以使用对象导航设置 format 属性,如此处所示。

// This will set the font size on the range during the next `content.sync()`.
range.format.font.size = 10;

可通过检查其只读修改者,识别不能直接设置其子问题的属性。 所有只读属性可直接设置其非只读子问题。 必须在该级别 PageLayout.zoom 可编写属性,如属性。 摘要:

  • 只读属性:可通过导航设置子项目。
  • 可写的属性:无法通过导航设置子项目(必须设置为初始父对象分配的一部分)。

*OrNullObject 方法与属性

当所需对象不存在时,某些配件方法和属性将引发异常。 例如,如果尝试通过指定工作簿未包含的工作表名称获取 Excel 工作表,则 getItem() 会引发 ItemNotFound 异常。 特定于应用程序的库为代码提供了一种方法,用于测试文档实体是否存在,而无需异常处理代码。 此操作是通过使用多种 *OrNullObject 和属性实现的。 如果指定的项目不存在,这些变体将返回其 isNullObject 被设置为 true值的对象,而不是引发异常。

例如,可以在集合(如 Worksheets)上调用 getItemOrNullObject() 方法,尝试从集合中检索某个项。 方法 getItemOrNullObject() 返回指定项目(如果存在);否则,将返回其属性 isNullObjecttrue。 然后,代码可评估此属性,以确定该对象是否存在。

注意

这些 *OrNullObject 变体永远不会返回值 JavaScript null。 它们返回普通 Office 代理对象。 如果对象表示的实体不存在,则对象的 isNullObject 属性设置为 true。 不要测试返回的对象为 nullity 或 fality。 它从 nullfalseundefined

以下代码示例尝试使用以下方法检索名为"数据"的 Excel getItemOrNullObject()。 如果具有该名称的工作表不存在,将创建一个新工作表。 请注意,该代码不会加载 isNullObject 属性。 当调用此属性时,Office context.sync 加载,因此不需要使用 dataSheet.load('isNullObject')等内容显式加载。

await Excel.run(async (context) => {
    let dataSheet = context.workbook.worksheets.getItemOrNullObject("Data");
    
    await context.sync();
    
    if (dataSheet.isNullObject) {
        dataSheet = context.workbook.worksheets.add("Data");
    }
    
    // Set `dataSheet` to be the second worksheet in the workbook.
    dataSheet.position = 1;
});

应用程序撤消堆栈

处理特定于应用程序的 API 时,将清除应用程序的撤消堆栈。 这意味着不能撤消在外接程序执行任何操作之前所做的更改(除非该加载项仅使用通用 API 或不与文件交互)。 加载项所做的更改也是如此。

另请参阅