ts-json-schema-generator
Version:
Generate JSON schema from your Typescript sources
42 lines (38 loc) • 1.79 kB
text/typescript
import ts from "typescript";
import type { NodeParser } from "../NodeParser.js";
import { Context } from "../NodeParser.js";
import type { SubNodeParser } from "../SubNodeParser.js";
import type { BaseType } from "../Type/BaseType.js";
export class ExpressionWithTypeArgumentsNodeParser implements SubNodeParser {
public constructor(
protected typeChecker: ts.TypeChecker,
protected childNodeParser: NodeParser,
) {}
public supportsNode(node: ts.ExpressionWithTypeArguments): boolean {
return node.kind === ts.SyntaxKind.ExpressionWithTypeArguments;
}
public createType(node: ts.ExpressionWithTypeArguments, context: Context): BaseType {
const typeSymbol = this.typeChecker.getSymbolAtLocation(node.expression)!;
if (typeSymbol.flags & ts.SymbolFlags.Alias) {
const aliasedSymbol = this.typeChecker.getAliasedSymbol(typeSymbol);
return this.childNodeParser.createType(
aliasedSymbol.declarations![0],
this.createSubContext(node, context),
);
} else if (typeSymbol.flags & ts.SymbolFlags.TypeParameter) {
return context.getArgument(typeSymbol.name);
} else {
return this.childNodeParser.createType(typeSymbol.declarations![0], this.createSubContext(node, context));
}
}
protected createSubContext(node: ts.ExpressionWithTypeArguments, parentContext: Context): Context {
const subContext = new Context(node);
if (node.typeArguments?.length) {
node.typeArguments.forEach((typeArg) => {
const type = this.childNodeParser.createType(typeArg, parentContext);
subContext.pushArgument(type);
});
}
return subContext;
}
}