145 lines
5.7 KiB
TypeScript
145 lines
5.7 KiB
TypeScript
|
/*---------------------------------------------------------------------------------------------
|
||
|
* Copyright (c) Microsoft Corporation. All rights reserved.
|
||
|
* Licensed under the MIT License. See License.txt in the project root for license information.
|
||
|
*--------------------------------------------------------------------------------------------*/
|
||
|
|
||
|
import { TextDocument, SemanticTokenData } from './languageModes';
|
||
|
import * as ts from 'typescript';
|
||
|
|
||
|
export function getSemanticTokenLegend() {
|
||
|
if (tokenTypes.length !== TokenType._) {
|
||
|
console.warn('TokenType has added new entries.');
|
||
|
}
|
||
|
if (tokenModifiers.length !== TokenModifier._) {
|
||
|
console.warn('TokenModifier has added new entries.');
|
||
|
}
|
||
|
return { types: tokenTypes, modifiers: tokenModifiers };
|
||
|
}
|
||
|
|
||
|
export function getSemanticTokens(jsLanguageService: ts.LanguageService, currentTextDocument: TextDocument, fileName: string): SemanticTokenData[] {
|
||
|
//https://ts-ast-viewer.com/#code/AQ0g2CmAuwGbALzAJwG4BQZQGNwEMBnQ4AQQEYBmYAb2C22zgEtJwATJVTRxgcwD27AQAp8AGmAAjAJS0A9POB8+7NQ168oscAJz5wANXwAnLug2bsJmAFcTAO2XAA1MHyvgu-UdOeWbOw8ViAAvpagocBAA
|
||
|
|
||
|
let resultTokens: SemanticTokenData[] = [];
|
||
|
const collector = (node: ts.Node, typeIdx: number, modifierSet: number) => {
|
||
|
resultTokens.push({ start: currentTextDocument.positionAt(node.getStart()), length: node.getWidth(), typeIdx, modifierSet });
|
||
|
};
|
||
|
collectTokens(jsLanguageService, fileName, { start: 0, length: currentTextDocument.getText().length }, collector);
|
||
|
|
||
|
return resultTokens;
|
||
|
}
|
||
|
|
||
|
function collectTokens(jsLanguageService: ts.LanguageService, fileName: string, span: ts.TextSpan, collector: (node: ts.Node, tokenType: number, tokenModifier: number) => void) {
|
||
|
|
||
|
const program = jsLanguageService.getProgram();
|
||
|
if (program) {
|
||
|
const typeChecker = program.getTypeChecker();
|
||
|
|
||
|
function visit(node: ts.Node) {
|
||
|
if (!node || !ts.textSpanIntersectsWith(span, node.pos, node.getFullWidth())) {
|
||
|
return;
|
||
|
}
|
||
|
if (ts.isIdentifier(node)) {
|
||
|
let symbol = typeChecker.getSymbolAtLocation(node);
|
||
|
if (symbol) {
|
||
|
if (symbol.flags & ts.SymbolFlags.Alias) {
|
||
|
symbol = typeChecker.getAliasedSymbol(symbol);
|
||
|
}
|
||
|
let typeIdx = classifySymbol(symbol);
|
||
|
if (typeIdx !== undefined) {
|
||
|
let modifierSet = 0;
|
||
|
if (node.parent) {
|
||
|
const parentTypeIdx = tokenFromDeclarationMapping[node.parent.kind];
|
||
|
if (parentTypeIdx === typeIdx && (<ts.NamedDeclaration>node.parent).name === node) {
|
||
|
modifierSet = 1 << TokenModifier.declaration;
|
||
|
}
|
||
|
}
|
||
|
const decl = symbol.valueDeclaration;
|
||
|
const modifiers = decl ? ts.getCombinedModifierFlags(decl) : 0;
|
||
|
const nodeFlags = decl ? ts.getCombinedNodeFlags(decl) : 0;
|
||
|
if (modifiers & ts.ModifierFlags.Static) {
|
||
|
modifierSet |= 1 << TokenModifier.static;
|
||
|
}
|
||
|
if (modifiers & ts.ModifierFlags.Async) {
|
||
|
modifierSet |= 1 << TokenModifier.async;
|
||
|
}
|
||
|
if ((modifiers & ts.ModifierFlags.Readonly) || (nodeFlags & ts.NodeFlags.Const) || (symbol.getFlags() & ts.SymbolFlags.EnumMember)) {
|
||
|
modifierSet |= 1 << TokenModifier.readonly;
|
||
|
}
|
||
|
collector(node, typeIdx, modifierSet);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
ts.forEachChild(node, visit);
|
||
|
}
|
||
|
const sourceFile = program.getSourceFile(fileName);
|
||
|
if (sourceFile) {
|
||
|
visit(sourceFile);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
function classifySymbol(symbol: ts.Symbol) {
|
||
|
const flags = symbol.getFlags();
|
||
|
if (flags & ts.SymbolFlags.Class) {
|
||
|
return TokenType.class;
|
||
|
} else if (flags & ts.SymbolFlags.Enum) {
|
||
|
return TokenType.enum;
|
||
|
} else if (flags & ts.SymbolFlags.TypeAlias) {
|
||
|
return TokenType.type;
|
||
|
} else if (flags & ts.SymbolFlags.Type) {
|
||
|
if (flags & ts.SymbolFlags.Interface) {
|
||
|
return TokenType.interface;
|
||
|
} if (flags & ts.SymbolFlags.TypeParameter) {
|
||
|
return TokenType.typeParameter;
|
||
|
}
|
||
|
}
|
||
|
const decl = symbol.valueDeclaration || symbol.declarations && symbol.declarations[0];
|
||
|
return decl && tokenFromDeclarationMapping[decl.kind];
|
||
|
}
|
||
|
|
||
|
export const enum TokenType {
|
||
|
class, enum, interface, namespace, typeParameter, type, parameter, variable, property, function, member, _
|
||
|
}
|
||
|
|
||
|
export const enum TokenModifier {
|
||
|
declaration, static, async, readonly, _
|
||
|
}
|
||
|
|
||
|
const tokenTypes: string[] = [];
|
||
|
tokenTypes[TokenType.class] = 'class';
|
||
|
tokenTypes[TokenType.enum] = 'enum';
|
||
|
tokenTypes[TokenType.interface] = 'interface';
|
||
|
tokenTypes[TokenType.namespace] = 'namespace';
|
||
|
tokenTypes[TokenType.typeParameter] = 'typeParameter';
|
||
|
tokenTypes[TokenType.type] = 'type';
|
||
|
tokenTypes[TokenType.parameter] = 'parameter';
|
||
|
tokenTypes[TokenType.variable] = 'variable';
|
||
|
tokenTypes[TokenType.property] = 'property';
|
||
|
tokenTypes[TokenType.function] = 'function';
|
||
|
tokenTypes[TokenType.member] = 'member';
|
||
|
|
||
|
const tokenModifiers: string[] = [];
|
||
|
tokenModifiers[TokenModifier.async] = 'async';
|
||
|
tokenModifiers[TokenModifier.declaration] = 'declaration';
|
||
|
tokenModifiers[TokenModifier.readonly] = 'readonly';
|
||
|
tokenModifiers[TokenModifier.static] = 'static';
|
||
|
|
||
|
const tokenFromDeclarationMapping: { [name: string]: TokenType } = {
|
||
|
[ts.SyntaxKind.VariableDeclaration]: TokenType.variable,
|
||
|
[ts.SyntaxKind.Parameter]: TokenType.parameter,
|
||
|
[ts.SyntaxKind.PropertyDeclaration]: TokenType.property,
|
||
|
[ts.SyntaxKind.ModuleDeclaration]: TokenType.namespace,
|
||
|
[ts.SyntaxKind.EnumDeclaration]: TokenType.enum,
|
||
|
[ts.SyntaxKind.EnumMember]: TokenType.property,
|
||
|
[ts.SyntaxKind.ClassDeclaration]: TokenType.class,
|
||
|
[ts.SyntaxKind.MethodDeclaration]: TokenType.member,
|
||
|
[ts.SyntaxKind.FunctionDeclaration]: TokenType.function,
|
||
|
[ts.SyntaxKind.MethodSignature]: TokenType.member,
|
||
|
[ts.SyntaxKind.GetAccessor]: TokenType.property,
|
||
|
[ts.SyntaxKind.PropertySignature]: TokenType.property,
|
||
|
[ts.SyntaxKind.InterfaceDeclaration]: TokenType.interface,
|
||
|
[ts.SyntaxKind.TypeAliasDeclaration]: TokenType.type,
|
||
|
[ts.SyntaxKind.TypeParameter]: TokenType.typeParameter
|
||
|
};
|