扩展解剖
在上一主题中,你成功运行了一个基本扩展。它在底层是如何工作的呢?
Hello World
扩展完成了 3 件事
- 注册了
onCommand
激活事件:onCommand:helloworld.helloWorld
,因此当用户运行Hello World
命令时,扩展会被激活。注意: 从 VS Code 1.74.0 开始,在
package.json
的commands
部分声明的命令在调用时会自动激活扩展,而无需在activationEvents
中显式添加onCommand
条目。 - 使用
contributes.commands
贡献点 将命令Hello World
呈现在命令面板中,并将其绑定到命令 IDhelloworld.helloWorld
。 - 使用
commands.registerCommand
VS Code API 将一个函数绑定到已注册的命令 IDhelloworld.helloWorld
。
理解这三个概念对于编写 VS Code 扩展至关重要
- 激活事件:扩展激活时触发的事件。
- 贡献点:你在
package.json
扩展清单 中进行的静态声明,用于扩展 VS Code。 - VS Code API:你可以在扩展代码中调用的 JavaScript API 集。
通常,你的扩展会结合使用贡献点和 VS Code API 来扩展 VS Code 的功能。扩展能力概述 主题能帮助你为扩展找到合适的贡献点和 VS Code API。
让我们仔细看看 Hello World
示例的源代码,了解这些概念是如何应用的。
扩展文件结构
.
├── .vscode
│ ├── launch.json // Config for launching and debugging the extension
│ └── tasks.json // Config for build task that compiles TypeScript
├── .gitignore // Ignore build output and node_modules
├── README.md // Readable description of your extension's functionality
├── src
│ └── extension.ts // Extension source code
├── package.json // Extension manifest
├── tsconfig.json // TypeScript configuration
你可以阅读更多关于配置文件的信息
但是,让我们专注于 package.json
和 extension.ts
,它们对于理解 Hello World
扩展至关重要。
扩展清单
每个 VS Code 扩展都必须有一个 package.json
作为其扩展清单。package.json
包含 Node.js 字段(如 scripts
和 devDependencies
)以及 VS Code 特定的字段(如 publisher
、activationEvents
和 contributes
)。你可以在扩展清单参考中找到所有 VS Code 特定字段的描述。以下是一些最重要的字段
name
和publisher
:VS Code 使用<publisher>.<name>
作为扩展的唯一 ID。例如,Hello World 示例的 ID 是vscode-samples.helloworld-sample
。VS Code 使用此 ID 来唯一标识你的扩展。main
:扩展入口点。activationEvents
和contributes
:激活事件 和 贡献点。engines.vscode
:指定扩展所依赖的最低 VS Code API 版本。
{
"name": "helloworld-sample",
"displayName": "helloworld-sample",
"description": "HelloWorld example for VS Code",
"version": "0.0.1",
"publisher": "vscode-samples",
"repository": "https://github.com/microsoft/vscode-extension-samples/helloworld-sample",
"engines": {
"vscode": "^1.51.0"
},
"categories": ["Other"],
"activationEvents": [],
"main": "./out/extension.js",
"contributes": {
"commands": [
{
"command": "helloworld.helloWorld",
"title": "Hello World"
}
]
},
"scripts": {
"vscode:prepublish": "npm run compile",
"compile": "tsc -p ./",
"watch": "tsc -watch -p ./"
},
"devDependencies": {
"@types/node": "^8.10.25",
"@types/vscode": "^1.51.0",
"tslint": "^5.16.0",
"typescript": "^3.4.5"
}
}
注意:如果你的扩展针对的是 VS Code 1.74 之前的版本,则必须在
activationEvents
中显式列出onCommand:helloworld.helloWorld
。
扩展入口文件
扩展入口文件导出了两个函数:activate
和 deactivate
。activate
在注册的激活事件发生时执行。deactivate
让你有机会在扩展停用前进行清理。对于许多扩展来说,可能不需要显式的清理,deactivate
方法可以删除。但是,如果扩展需要在 VS Code 关闭、扩展被禁用或卸载时执行操作,则在此方法中进行。
VS Code 扩展 API 在 @types/vscode 类型定义中声明。vscode
类型定义的版本由 package.json
中 engines.vscode
字段的值控制。vscode
类型为你提供了 IntelliSense、转到定义以及代码中的其他 TypeScript 语言特性。
// The module 'vscode' contains the VS Code extensibility API
// Import the module and reference it with the alias vscode in your code below
import * as vscode from 'vscode';
// this method is called when your extension is activated
// your extension is activated the very first time the command is executed
export function activate(context: vscode.ExtensionContext) {
// Use the console to output diagnostic information (console.log) and errors (console.error)
// This line of code will only be executed once when your extension is activated
console.log('Congratulations, your extension "helloworld-sample" is now active!');
// The command has been defined in the package.json file
// Now provide the implementation of the command with registerCommand
// The commandId parameter must match the command field in package.json
let disposable = vscode.commands.registerCommand('helloworld.helloWorld', () => {
// The code you place here will be executed every time your command is executed
// Display a message box to the user
vscode.window.showInformationMessage('Hello World!');
});
context.subscriptions.push(disposable);
}
// this method is called when your extension is deactivated
export function deactivate() {}