UNPKG

rbxts-transformer-t

Version:

TypeScript transformer which converts TypeScript types to t entities

249 lines (172 loc) 5.76 kB
import ts from "typescript"; import path from "path"; /** * Concatenates two arrays and removes duplicates */ export function mergeArrays<T>(array1: T[], array2: T[]): T[] { return [...new Set([...array1, ...array2])] } /** * Separates an array into two arrays: * the first array contains elements matched * by predicate as true, the second as false. */ export function separateArray<T>(array: T[], predicate: (element: T) => boolean): [T[], T[]] { const nextIteration = (array: T[], onTrue: T[], onFalse: T[]): [T[], T[]] => { if (array.length === 0) return [onTrue, onFalse] const [head, ...tail] = array if (predicate(head)) return nextIteration(tail, [...onTrue, head], onFalse) return nextIteration(tail, onTrue, [...onFalse, head]) } return nextIteration(array, [], []) } /** * Checks if type is an EnumItem */ export function isEnum(type: ts.Type): boolean { return (<any>type.aliasSymbol)?.parent?.escapedName === "Enum"; } /** * Checks if ts.Type is a function */ export function isFunctionType(type: ts.Type): boolean { return type.getCallSignatures().length !== 0 } /** * Checks if ts.Type is of ts.TupleType */ export function isTupleType(type: ts.Type, typeChecker: ts.TypeChecker): type is ts.TupleType { return (<any>typeChecker).isTupleType(type) } /** * Checks if ts.Type is of Array type */ export function isArrayType(type: ts.Type, typeChecker: ts.TypeChecker): type is ts.GenericType { return (<any>typeChecker).isArrayType(type) } /** * Checks if ts.Type is of Map type */ export function isMapType(type: ts.Type): type is ts.GenericType { return type.symbol?.getName() === "Map" } /** * Checks if ts.Type is of Map type */ export function isBrickColorType(type: ts.Type): type is ts.GenericType { return type.symbol?.getName() === "BrickColor" } /** * Converts array of ts.Types to array of ts.Expressions */ /** * Checks if ts.Type being is of ts.ObjectType */ export function isObjectType(type: ts.Type): type is ts.ObjectType { return type.symbol?.getName() === "__type" } /** * Checks if property declaration matched is optional */ export function isOptionalPropertyDeclaration(prop: ts.Symbol): boolean { const property = prop.valueDeclaration as ts.PropertyDeclaration return property.questionToken !== undefined } /** * Checks if type is a literal */ export function isLiteral(typeChecker: ts.TypeChecker): (type: ts.Type) => boolean { return (type: ts.Type): boolean => type.isLiteral() || ["true", "false"].includes(typeChecker.typeToString(type)) } export function isCustomEnum(type: ts.Type) { return type.symbol?.valueDeclaration?.kind === ts.SyntaxKind.EnumDeclaration } /** * Extracts property type and name from ts.Symbol */ export function extractProperty(prop: ts.Symbol, typeChecker: ts.TypeChecker): { name: string, type: ts.Type } { const declaration = prop.valueDeclaration const type = (<any>prop).type === undefined ? typeChecker.getTypeFromTypeNode((<any>declaration).type) : (<any>prop).type.target ?? (<any>prop).type const name = String(prop.escapedName) return { name, type } } /** * Builds name for property. * If it is simple, use just string. * For complex names wrap it into ts.StringLiteral */ export function buildPropertyName(name: string): string | ts.StringLiteral { if (name.match(/^[a-zA-Z_]+[\w_]+$/) !== null) return name return ts.factory.createStringLiteral(name) } /** * Represents function that maps single node */ type MappingFunction = { (node: ts.Node, program: ts.Program): [ts.Node, boolean] } /** * Predicate accepts ts.Node or it"s subtypes */ type NodePredicate<T extends ts.Node = ts.Node> = (node: T) => boolean /** * Returns mapping function that replaces * nodes that satisfy the passed predicate * with a replacement node passed but skips the first appearance */ export function getNodeReplacer(predicate: NodePredicate, replacement: ts.Node): MappingFunction { let replaced = false return (node: ts.Node) => { if (replaced && predicate(node)) return [replacement, replaced] if (replaced || !predicate(node)) return [node, replaced] replaced = true return [node, replaced] } } /** * Returns aliased symbol of ts.Symbol or * this symbol itself if it is aliased */ export function getAliasedSymbol(symbol: ts.Symbol, typeChecker: ts.TypeChecker): ts.Symbol { try { return typeChecker.getAliasedSymbol(symbol) } catch (_) { return symbol } } /** * Returns aliased symbol of ts.Identifier */ export function getAliasedSymbolOfNode(node: ts.Identifier, typeChecker: ts.TypeChecker): ts.Symbol | undefined { const symbol = typeChecker.getSymbolAtLocation(node) if (symbol === undefined) return undefined return getAliasedSymbol(symbol, typeChecker) } /** * Checks is symbol belongs to declaration * with name "name" located in file filePath */ export function isSymbolOf(symbol: ts.Symbol, name: string, filePath: string): boolean { if (symbol.escapedName !== name) return false return [symbol.valueDeclaration, ...symbol.declarations!].filter(Boolean) // TODO .some(declaration => path.join(declaration!.getSourceFile().fileName) === filePath) } /** * Returns id of ts.Type */ export function getTypeId(type: ts.Type): number { return (<any>type).id } /** * Merges two objects */ export function mergeObjects<K extends string | number, V>(obj1: Record<K, V>, obj2: Record<K, V>): Record<K, V> { return { ...obj1, ...obj2 } }