语义高亮指南
语义高亮是对 语法高亮指南 中所述的语法高亮的补充。Visual Studio Code 使用 TextMate 语法作为主要的标记化引擎。TextMate 语法以单个文件作为输入,并根据正则表达式表示的词法规则将其分解。
语义标记化允许语言服务器根据其在项目上下文中解析符号的知识提供额外的标记信息。主题可以选择使用语义标记来改进和优化语法的语法高亮。编辑器将语义标记的高亮应用在语法高亮之上。
以下是语义高亮可以增加的示例
没有语义高亮
有语义高亮
请注意基于语言服务符号理解的颜色差异
- 第 10 行:
languageModes
被着色为参数 - 第 11 行:
Range
和Position
被着色为类,document
被着色为参数。 - 第 13 行:
getFoldingRanges
被着色为函数。
语义标记提供者
要实现语义高亮,语言扩展可以按文档语言和/或文件名注册一个 semantic token provider
(语义标记提供者)。当需要语义标记时,编辑器会向提供者发出请求。
const tokenTypes = ['class', 'interface', 'enum', 'function', 'variable'];
const tokenModifiers = ['declaration', 'documentation'];
const legend = new vscode.SemanticTokensLegend(tokenTypes, tokenModifiers);
const provider: vscode.DocumentSemanticTokensProvider = {
provideDocumentSemanticTokens(
document: vscode.TextDocument
): vscode.ProviderResult<vscode.SemanticTokens> {
// analyze the document and return semantic tokens
const tokensBuilder = new vscode.SemanticTokensBuilder(legend);
// on line 1, characters 1-5 are a class declaration
tokensBuilder.push(
new vscode.Range(new vscode.Position(1, 1), new vscode.Position(1, 5)),
'class',
['declaration']
);
return tokensBuilder.build();
}
};
const selector = { language: 'java', scheme: 'file' }; // register for all Java documents from the local file system
vscode.languages.registerDocumentSemanticTokensProvider(selector, provider, legend);
语义标记提供者 API 有两种风格,以适应语言服务器的能力
-
DocumentSemanticTokensProvider
- 始终将完整文档作为输入。provideDocumentSemanticTokens
- 提供文档的所有标记。provideDocumentSemanticTokensEdits
- 提供文档的所有标记,作为对先前响应的增量。
-
DocumentRangeSemanticTokensProvider
- 仅在某个范围内工作。provideDocumentRangeSemanticTokens
- 提供文档范围内的所有标记。
提供者返回的每个标记都带有一个分类,该分类由一个标记类型、任意数量的标记修饰符和一种标记语言组成。
如上面的示例所示,提供者在 SemanticTokensLegend
中命名它将使用的类型和修饰符。这使得 provide
API 可以将标记类型和修饰符作为图例的索引返回。
语义标记分类
语义标记提供者的输出包含标记。每个标记都有一个范围和一个标记分类,描述该标记表示的语法元素类型。如果标记是嵌入式语言的一部分,分类还可以选择性地命名语言。
为了描述语法元素的类型,使用语义标记类型和修饰符。此信息类似于 语法高亮指南 中描述的 TextMate 范围,但我们希望提出一个专门且更清晰的分类系统。
VS Code 带有一组标准的语义标记类型和修饰符,供所有语义标记提供者使用。但是,语义标记提供者可以自由定义新的类型和修饰符,并创建标准类型的子类型。
标准标记类型和修饰符
标准类型和修饰符涵盖了许多语言使用的常见概念。虽然每种语言可能对某些类型和修饰符使用不同的术语,但通过遵循标准分类,主题作者可以定义跨语言工作的主题规则。
以下是 VS Code 预定义的标准语义标记类型和语义标记修饰符
标准标记类型
ID | 描述 |
---|---|
namespace |
用于声明或引用命名空间、模块或包的标识符。 |
class |
用于声明或引用类类型的标识符。 |
enum |
用于声明或引用枚举类型的标识符。 |
interface |
用于声明或引用接口类型的标识符。 |
struct |
用于声明或引用结构类型的标识符。 |
typeParameter |
用于声明或引用类型参数的标识符。 |
type |
用于声明或引用上述未涵盖的类型的标识符。 |
parameter |
用于声明或引用函数或方法参数的标识符。 |
variable |
用于声明或引用局部或全局变量的标识符。 |
property |
用于声明或引用成员属性、成员字段或成员变量的标识符。 |
enumMember |
用于声明或引用枚举属性、常量或成员的标识符。 |
decorator |
用于声明或引用装饰器和注解的标识符。 |
event |
用于声明事件属性的标识符。 |
function |
用于声明函数的标识符。 |
method |
用于声明成员函数或方法的标识符。 |
macro |
用于声明宏的标识符。 |
label |
用于声明标签的标识符。 |
comment |
表示注释的标记。 |
string |
表示字符串字面量的标记。 |
keyword |
表示语言关键字的标记。 |
number |
表示数字字面量的标记。 |
regexp |
表示正则表达式字面量的标记。 |
operator |
表示运算符的标记。 |
标准标记修饰符
ID | 描述 |
---|---|
declaration |
用于符号的声明。 |
definition |
用于符号的定义,例如在头文件中。 |
readonly |
用于只读变量和成员字段(常量)。 |
static |
用于类成员(静态成员)。 |
deprecated |
用于不应再使用的符号。 |
abstract |
用于抽象类型和成员函数。 |
async |
用于标记为 async 的函数。 |
modification |
用于变量被赋值的变量引用。 |
documentation |
用于文档中出现的符号。 |
defaultLibrary |
用于标准库一部分的符号。 |
除了标准类型和修饰符之外,VS Code 还定义了类型和修饰符到类似 TextMate 范围的映射。这在语义标记范围映射一节中有所介绍。
自定义标记类型和修饰符
如有必要,扩展可以在其 package.json
中通过 semanticTokenTypes
和 semanticTokenModifiers
贡献点声明新的类型和修饰符,或创建现有类型的子类型
{
"contributes": {
"semanticTokenTypes": [
{
"id": "templateType",
"superType": "type",
"description": "A template type."
}
],
"semanticTokenModifiers": [
{
"id": "native",
"description": "Annotates a symbol that is implemented natively"
}
]
}
}
在上面的示例中,一个扩展声明了一个新类型 templateType
和一个新修饰符 native
。通过将 type
命名为超类型,type
的主题样式规则也将适用于 templateType
{
"name": "Red Theme",
"semanticTokenColors": {
"type": "#ff0011"
}
}
上面显示的 semanticTokenColors
值 "#ff0011"
适用于 type
及其所有子类型,包括 templateType
。
除了自定义标记类型之外,扩展还可以定义如何将它们映射到 TextMate 范围。这在自定义映射一节中有所描述。请注意,自定义映射规则不会自动从超类型继承。相反,子类型需要重新定义映射,最好是映射到更具体的范围。
启用语义高亮
语义标记是否被计算和高亮取决于设置 editor.semanticHighlighting.enabled
。它可以有值 true
、false
和 configuredByTheme
。
true
和false
会为所有主题开启或关闭语义高亮。configuredByTheme
是默认值,它允许每个主题控制是否启用语义高亮。VS Code 附带的所有主题(例如,“深色+”默认主题)默认都启用了语义高亮。
依赖语义标记的语言扩展可以在其 package.json
中覆盖其语言的默认设置
{
"configurationDefaults": {
"[languageId]": {
"editor.semanticHighlighting.enabled": true
}
}
}
主题
主题是关于为标记分配颜色和样式。主题规则在颜色主题文件(JSON 格式)中指定。用户也可以在用户设置中自定义主题规则。
颜色主题中的语义着色
为了支持基于语义标记的高亮,颜色主题文件格式中添加了两个新属性。
属性 semanticHighlighting
定义主题是否准备好使用语义标记进行高亮。默认情况下它是 false,但我们鼓励所有主题启用它。当设置 editor.semanticHighlighting.enabled
设置为 configuredByTheme
时,此属性会被使用。
属性 semanticTokenColors
允许主题定义新的着色规则,这些规则与语义标记提供者发出的语义标记类型和修饰符匹配。
{
"name": "Red Theme",
"tokenColors": [
{
"scope": "comment",
"settings": {
"foreground": "#dd0000",
"fontStyle": "italic"
}
}
],
"semanticHighlighting": true,
"semanticTokenColors": {
"variable.readonly:java": "#ff0011"
}
}
variable.readonly:java
称为选择器,形式为 (*|tokenType)(.tokenModifier)*(:tokenLanguage)?
。
值描述了规则匹配时的样式。它可以是字符串,表示前景颜色,也可以是一个对象,形式为 { foreground: string, bold: boolean, italic: boolean, underline: boolean }
或 { foreground: string, fontStyle: string }
,如 tokenColors
中用于 TextMate 主题规则那样。
前景需要遵循 颜色格式 中描述的颜色格式。不支持透明度。
以下是选择器和样式的其他示例
"*.declaration": { "bold": true } // 所有声明都加粗
"class:java": { "foreground": "#0f0", "italic": true } // Java 中的类
如果没有规则匹配,或者主题没有 semanticTokenColors
部分(但启用了 semanticHighlighting
),VS Code 会使用语义标记范围映射为给定的语义标记评估 TextMate 范围。该范围将与主题的 tokenColors
中的 TextMate 主题规则进行匹配。
语义标记范围映射
为了使语义高亮适用于尚未定义任何特定语义规则的主题,并作为自定义标记类型和修饰符的后备,VS Code 维护了一个从语义标记选择器到 TextMate 范围的映射。
如果主题启用了语义高亮,但对于给定的语义标记没有包含规则,则会使用这些 TextMate 范围来查找 TextMate 主题规则。
预定义的 TextMate 范围映射
下表列出了当前预定义的映射。
语义标记选择器 | 后备 TextMate 范围 |
---|---|
namespace |
entity.name.namespace |
type |
entity.name.type |
type.defaultLibrary |
support.type |
struct |
storage.type.struct |
class |
entity.name.type.class |
class.defaultLibrary |
support.class |
interface |
entity.name.type.interface |
enum |
entity.name.type.enum |
function |
entity.name.function |
function.defaultLibrary |
support.function |
method |
entity.name.function.member |
macro |
entity.name.function.preprocessor |
variable |
variable.other.readwrite , entity.name.variable |
variable.readonly |
variable.other.constant |
variable.readonly.defaultLibrary |
support.constant |
parameter |
variable.parameter |
property |
variable.other.property |
property.readonly |
variable.other.constant.property |
enumMember |
variable.other.enummember |
event |
variable.other.event |
自定义 TextMate 范围映射
此映射可以通过扩展在其 package.json
中的 semanticTokenScopes
贡献点进行扩展。
扩展执行此操作有两种用例
-
定义自定义标记类型和标记修饰符的扩展在主题未定义用于添加的语义标记类型或修饰符的主题规则时提供 TextMate 范围作为后备。
{ "contributes": { "semanticTokenScopes": [ { "scopes": { "templateType": ["entity.name.type.template"] } } ] } }
-
TextMate 语法的提供者可以描述特定于语言的范围。这有助于包含特定于语言的主题规则的主题。
{ "contributes": { "semanticTokenScopes": [ { "language": "typescript", "scopes": { "property.readonly": ["variable.other.constant.property.ts"] } } ] } }
试用
我们有一个语义标记示例,它说明了如何创建语义标记提供者。
范围检查器工具允许您探索源文件中的语义标记以及它们匹配的主题规则。要查看语义标记,请在 TypeScript 文件上使用内置主题(例如,深色+)。