通用 API
在 Univer 中,根据文档的不同类型,所能调用的 Facade API 也有所不同。本章节将介绍在所有类型文档都适用的通用 Facade API。
概念
命令
Univer 中大多数的操作都会注册到命令系统,并通过命令系统来触发。这种统一的操作方式使得 Univer 可以很容易的实现撤销、重做、协同等功能。
如需了解更多设计细节请阅读 Univer 命令系统 。
命令系统
Univer 为用户提供了统一的命令系统,通过命令系统用户可以实现各种定制化的功能。
每一个命令对应了一个唯一的 ID。如果你正在寻找某个特定的命令 ID,可以参考 如何查找命令 ID。
监听命令
Univer 设计了两种监听命令的方式,分别是在命令执行前和在命令执行后:
onBeforeCommandExecute
:在命令执行前执行自定义逻辑。onCommandExecuted
:在命令执行后执行自定义逻辑。
监听命令执行前
在命令执行之前,我们向 FUniver.onBeforeCommandExecute
API 传入一个回调函数来注册自定义的预处理监听器。
当命令执行前,预处理监听器内逻辑会被执行。
const univerAPI = FUniver.newAPI(univer);
univerAPI.onBeforeCommandExecute((command)=>{
const { id, type, params } = command;
// 在命令执行前执行自定义逻辑
})
如果你想在命令执行前阻止命令执行,可以在监听器回调中 throw
一个异常。
const univerAPI = FUniver.newAPI(univer);
univerAPI.onBeforeCommandExecute((command)=>{
throw new Error('禁止编辑')
})
监听命令执行后
在命令执行之后,我们也可以向 FUniver.onCommandExecuted
API 传入一个回调函数来注册自定义的后处理监听器。
当命令执行后,后处理监听器内逻辑会被执行。
const univerAPI = FUniver.newAPI(univer);
univerAPI.onCommandExecuted((command)=>{
const { id, type, params } = command;
// 在命令执行后执行自定义逻辑
})
取消监听
注册命令监听器的方法会返回一个 IDisposable
对象,调用 IDisposable.dispose
可以销毁该监听器。
建议你及时销毁不再使用的监听器,有助于提高程序的健壮性。
const univerAPI = FUniver.newAPI(univer);
// 注册监听器
const disposable = univerAPI.onBeforeCommandExecute((command)=>{
// 在命令执行前执行自定义逻辑
})
// 示例:1 秒后取消监听
setTimeout(()=>{
// 取消监听
disposable.dispose();
}, 1000);
执行命令
如果你知道命令 ID 和所需要传递的参数,也可以通过 FUniver.executeCommand
方法来执行命令。
例如,我们可以通过 sheet.command.set-range-values
命令来设置单元格 A1 的值:
const univerAPI = FUniver.newAPI(univer);
// 执行命令
univerAPI.executeCommand('sheet.command.set-range-values', {
value: { v: "Hello, Univer!" },
range: { startRow: 0, startColumn: 0, endRow: 0, endColumn: 0 }
});
撤销和重做
撤销
univerAPI.executeCommand('univer.command.undo')
重做
univerAPI.executeCommand('univer.command.redo')
系统剪切板
从 0.2.12
开始,使用 FUniver.copy()
和 FUniver.paste()
方法读取和写入系统剪贴板。
复制与粘贴依赖浏览器原生 API ,当环境条件或者权限不满足时复制与粘贴功能将无法工作。
示例: Sheet 区域的复制与粘贴 0.2.12+
var sheet = univerAPI.getActiveWorkbook().getActiveSheet()
// 获取 A1 作为复制来源
var sourceRange = sheet.getRange(0,0)
// 设置 A1 的值
sourceRange.setValue('Hello, Univer!')
// 设置 A1 为激活选区
sheet.setActiveRange(sourceRange)
// 复制 A1 的值到系统剪贴板
univerAPI.copy(sourceRange)
// 获取 A2 作为粘贴目标
var targetRange = sheet.getRange(0,1)
// 设置 A2 为激活选区
sheet.setActiveRange(targetRange)
// 将剪贴板内容粘贴到当前选区,A2 的值变为 "Hello, Univer!"
univerAPI.paste()
0.2.12
之前的版本,可以使用命令系统实现调用剪贴板功能:
import { CopyCommand, PasteCommand } from '@univerjs/ui';
// 复制
univerAPI.executeCommand(CopyCommand.id)
// 粘贴
univerAPI.executeCommand(PasteCommand.id)
基础事件
如何使用:
const univerAPI = FUniver.newAPI(univer);
univerAPI.getHooks().onStarting(() => {
console.log('Starting');
})
完整基础事件列表
方法 | 描述 | 回调参数 | 返回值 |
---|---|---|---|
生命周期钩子 | |||
onStarting | 当生命周期阶段为 Starting 时触发。 | callback: () => void | IDisposable 用于取消订阅 |
onReady | 当生命周期阶段为 Ready 时触发。 | callback: () => void | IDisposable 用于取消订阅 |
onRendered | 当生命周期阶段为 Rendered 时触发。 | callback: () => void | IDisposable 用于取消订阅 |
onSteady | 当生命周期阶段为 Steady 时触发。 | callback: () => void | IDisposable 用于取消订阅 |
撤销/重做钩子 | |||
onBeforeUndo | 在执行撤销操作之前触发。 | callback: (action: IUndoRedoItem) => void | IDisposable 用于取消订阅 |
onUndo | 在执行撤销操作之后触发。 | callback: (action: IUndoRedoItem) => void | IDisposable 用于取消订阅 |
onBeforeRedo | 在执行重做操作之前触发。 | callback: (action: IUndoRedoItem) => void | IDisposable 用于取消订阅 |
onRedo | 在执行重做操作之后触发。 | callback: (action: IUndoRedoItem) => void | IDisposable 用于取消订阅 |
剪贴板钩子 | |||
onBeforeCopy | 在执行复制操作之前触发。 | callback: () => void | IDisposable 用于取消订阅 |
onCopy | 在执行复制操作之后触发。 | callback: () => void | IDisposable 用于取消订阅 |
onBeforePaste | 在执行粘贴操作之前触发。 | callback: () => void | IDisposable 用于取消订阅 |
onPaste | 在执行粘贴操作之后触发。 | callback: () => void | IDisposable 用于取消订阅 |
通用返回值
所有这些方法都会返回一个 IDisposable
对象,该对象可以用于取消事件订阅,当不再需要监听时使用。
UI
请参考下面文档来拓展 Univer 的 UI :
WebSocket
Facade 提供了一个便捷的 API createSocket
来创建 WebSocket,传入一个 URL 即可。
然后可以监听 open、message、close、error 事件,以及主动发送消息 send 方法和主动关闭 close 方法。
// URL 换成你自己 WebSocket 服务的地址
const ws = univerAPI.createSocket("ws://47.100.177.253:8449/ws");
ws.open$.subscribe(() => {
console.log('websocket opened')
ws.send('hello')
})
ws.message$.subscribe((message) => {
console.log('websocket message', message)
const content = JSON.parse(message.data).content
if (!content.includes('command')) {
return
}
const commandInfo = JSON.parse(content);
const { command, options } = commandInfo;
const { id, params } = command;
// 接受到协同数据,本地落盘
univerAPI.executeCommand(id, params, options)
});
ws.close$.subscribe(() => {
console.log("websocket closed");
});
ws.error$.subscribe((error) => {
console.log("websocket error", error);
});
univerAPI.onCommandExecuted((command, options) => {
// 仅同步本地 mutation
if (command.type !== 2 || options?.fromCollab || options?.onlyLocal || command.id === 'doc.mutation.rich-text-editing') {
return;
}
const commandInfo = JSON.stringify({ command, options: { fromCollab: true } })
ws.send(commandInfo);
})
注意:启动 Univer 的时候要确保有 unitID ,不指定 unitID 的话无法协同。
注册公式
使用 Facade API,可以方便快速的在当前 Univer 实例中注册自定义公式。
如下案例所示,使用 registerFunction
将一个 CUSTOMSUM
公式所需要的算法、名称、描述一次性注册到公式插件,执行之后就可以使用公式了。在任一空白单元格输入 =CUSTOMSUM
可以看到提示。
univerAPI.registerFunction({
calculate: [
[function (...variants) {
let sum = 0;
for(const variant of variants){
sum += Number(variant) || 0;
}
return sum;
}, 'CUSTOMSUM', '求参数的和'],
// ... 更多公式
]
})
如果需要卸载所注册的公式,可以调用 dispose
方法。
const functionDisposable = univerAPI.registerFunction({
// calculate
})
// 卸载所注册的公式
functionDisposable.dispose();
如果想要提供更完善的国际化内容和描述,还可以配置 locales
和 description
字段。如下所示。
const FUNCTION_NAMES_USER = {
CUSTOMSUM: 'CUSTOMSUM'
}
univerAPI.registerFunction({
locales: {
'zhCN': {
formulaCustom: {
CUSTOMSUM: {
description: '将单个值、单元格引用或是区域相加,或者将三者的组合相加。',
abstract: '求参数的和',
links: [
{
title: '教学',
url: 'https://support.microsoft.com/zh-cn/office/sum-%E5%87%BD%E6%95%B0-043e1c7d-7726-4e80-8f32-07b23e057f89',
},
],
functionParameter: {
number1: {
name: '数值1',
detail: '要相加的第一个数字。 该数字可以是 4 之类的数字,B6 之类的单元格引用或 B2:B8 之类的单元格范围。',
},
number2: {
name: '数值2',
detail: '这是要相加的第二个数字。 可以按照这种方式最多指定 255 个数字。',
},
},
},
// ... 更多公式
},
},
'enUS': {
formulaCustom: {
CUSTOMSUM: {
description: `You can add individual values, cell references or ranges or a mix of all three.`,
abstract: `Adds its arguments`,
links: [
{
title: 'Instruction',
url: 'https://support.microsoft.com/en-us/office/sum-function-043e1c7d-7726-4e80-8f32-07b23e057f89',
},
],
functionParameter: {
number1: {
name: 'number1',
detail: 'The first number you want to add. The number can be like 4, a cell reference like B6, or a cell range like B2:B8.',
},
number2: {
name: 'number2',
detail: 'This is the second number you want to add. You can specify up to 255 numbers in this way.',
},
},
},
}
}
},
description: [
{
functionName: FUNCTION_NAMES_USER.CUSTOMSUM,
aliasFunctionName: 'formulaCustom.CUSTOMSUM.aliasFunctionName',
functionType: 15,
description: 'formulaCustom.CUSTOMSUM.description',
abstract: 'formulaCustom.CUSTOMSUM.abstract',
functionParameter: [
{
name: 'formulaCustom.CUSTOMSUM.functionParameter.number1.name',
detail: 'formulaCustom.CUSTOMSUM.functionParameter.number1.detail',
example: 'A1:A20',
require: 1,
repeat: 0,
},
{
name: 'formulaCustom.CUSTOMSUM.functionParameter.number2.name',
detail: 'formulaCustom.CUSTOMSUM.functionParameter.number2.detail',
example: 'B2:B10',
require: 0,
repeat: 1,
},
],
},
// ... 更多公式
],
calculate: [
[function (...variants) {
let sum = 0;
for (const variant of variants) {
sum += Number(variant) || 0;
}
return sum;
}, FUNCTION_NAMES_USER.CUSTOMSUM],
// ... 更多公式
]
})
说明
locales
下可以设置多种语言,命名规则参考 LocaleType。可以在functionList
下添加多个公式的翻译。详细的字段说明请参考 如何在 Formula Engine 中添加公式 的部分。description
设置自定义公式的描述。calculate
编写计算公式的具体算法和名称映射。入参为使用公式时用户输入的内容,可能为数字、字符串、布尔值,或者一个数组。
如果想复用公式系统提供的算法,增强公式算法的能力,可以通过插件配置的方式注册自定义公式,详细教程请参考 自定义公式。