UNPKG

41.2 kBJavaScriptView Raw
1import _ from 'lodash';
2import ts from 'typescript';
3import { heritage, modifier, overload, parametered } from './base';
4import * as declaration from './declaration';
5import * as node_ from './node';
6import * as reference from './reference';
7import * as symbol_ from './symbol';
8import * as type_ from './type_';
9import * as utils from './utils';
10export function isClassProperty(node) {
11 return ts.isPropertyDeclaration(node) || ts.isGetAccessorDeclaration(node) || ts.isSetAccessorDeclaration(node);
12}
13export function isClassInstanceProperty(node) {
14 return (isClassProperty(node) || ts.isParameterPropertyDeclaration(node, node.parent)) && !modifier.isStatic(node);
15}
16export function isClassInstanceMember(node) {
17 return (ts.isMethodDeclaration(node) || isClassInstanceProperty(node)) && !modifier.isStatic(node);
18}
19export function isClassStaticProperty(node) {
20 return (ts.isMethodDeclaration(node) || isClassProperty(node)) && modifier.isStatic(node);
21}
22export function isClassStaticMember(node) {
23 return (ts.isMethodDeclaration(node) || isClassProperty(node)) && modifier.isStatic(node);
24}
25export function isClassMember(node) {
26 return isClassInstanceMember(node) || isClassStaticMember(node);
27}
28export function getExtends(node) {
29 const extendsClause = heritage.getHeritageClauseByKind(node, ts.SyntaxKind.ExtendsKeyword);
30 if (extendsClause === undefined) {
31 return undefined;
32 }
33 const typeNodes = heritage.getTypeNodes(extendsClause);
34 return typeNodes.length === 0 ? undefined : typeNodes[0];
35}
36export function getExtendsOrThrow(node) {
37 return utils.throwIfNullOrUndefined(getExtends(node), 'extends expression');
38}
39export function getImplements(node) {
40 const implementsClause = heritage.getHeritageClauseByKind(node, ts.SyntaxKind.ImplementsKeyword);
41 if (implementsClause === undefined) {
42 return undefined;
43 }
44 return heritage.getTypeNodes(implementsClause);
45}
46export function getImplementsArray(node) {
47 return utils.getArray(getImplements(node));
48}
49export function getMembers(node) {
50 const members = [...node.members];
51 const implementationCtors = members.filter(ts.isConstructorDeclaration).filter((c) => overload.isImplementation(c));
52 for (const ctor of implementationCtors) {
53 let insertIndex = members.indexOf(ctor) + 1;
54 for (const param of parametered.getParameters(ctor)) {
55 if (ts.isParameterPropertyDeclaration(param, param.parent)) {
56 members.splice(insertIndex, 0, param);
57 insertIndex += 1;
58 }
59 }
60 }
61 return members.filter(isClassMember);
62}
63export function getConcreteMembers(node) {
64 return declaration.isAmbient(node)
65 ? []
66 : getMembers(node).filter((member) => {
67 if (ts.isMethodDeclaration(member)) {
68 return overload.isImplementation(member);
69 }
70 return true;
71 });
72}
73export function getInstanceProperties(node) {
74 return getMembers(node).filter(isClassInstanceProperty);
75}
76export function getInstanceMembers(node) {
77 return getMembers(node).filter(isClassInstanceMember);
78}
79export function getInstanceMethods(node) {
80 return getInstanceMembers(node).filter(ts.isMethodDeclaration);
81}
82export function getMethods(node) {
83 return getMembers(node).filter(ts.isMethodDeclaration);
84}
85export function getSetAccessors(node) {
86 return getMembers(node).filter(ts.isSetAccessor);
87}
88export function getInstanceMethod(node, name) {
89 return getInstanceMethods(node).find((method) => node_.getName(method) === name);
90}
91export function getConcreteInstanceProperties(node) {
92 return getConcreteMembers(node).filter(isClassInstanceProperty);
93}
94export function getConcreteInstanceMembers(node) {
95 return getConcreteMembers(node).filter(isClassInstanceMember);
96}
97export function getConcreteInstanceMethods(node) {
98 return getConcreteInstanceMembers(node).filter(ts.isMethodDeclaration);
99}
100export function getStaticProperties(node) {
101 return getMembers(node).filter(isClassStaticProperty);
102}
103export function getStaticMembers(node) {
104 return getMembers(node).filter(isClassStaticMember);
105}
106export function getConcreteStaticProperties(node) {
107 return getConcreteMembers(node).filter(isClassStaticProperty);
108}
109export function getConcreteStaticMembers(node) {
110 return getConcreteMembers(node).filter(isClassStaticMember);
111}
112export function getConcreteStaticMethods(node) {
113 return getConcreteStaticMembers(node).filter(ts.isMethodDeclaration);
114}
115export function getConstructors(node) {
116 return node.members.filter(ts.isConstructorDeclaration);
117}
118export function getConcreteConstructor(node) {
119 return getConstructors(node).find((ctor) => overload.isImplementation(ctor));
120}
121export function getFirstConcreteConstructor(typeChecker, node) {
122 const ctor = getConcreteConstructor(node);
123 if (ctor !== undefined) {
124 return ctor;
125 }
126 const baseClass = getBaseClass(typeChecker, node);
127 if (baseClass === undefined) {
128 return undefined;
129 }
130 return getFirstConcreteConstructor(typeChecker, baseClass);
131}
132function getDerivedClassesWorker(program, languageService, node, seen = new Set()) {
133 if (seen.has(node)) {
134 return [];
135 }
136 return reference
137 .findReferencesAsNodes(program, languageService, node)
138 .reduce((acc, ref) => {
139 const parent = node_.getParent(ref);
140 if (parent === undefined) {
141 return acc;
142 }
143 const clause = node_.getParent(parent);
144 if (clause === undefined || !ts.isHeritageClause(clause) || !heritage.isExtends(clause)) {
145 return acc;
146 }
147 const derived = node_.getFirstAncestorByKindOrThrow(clause, ts.SyntaxKind.ClassDeclaration);
148 return acc.concat(getDerivedClassesWorker(program, languageService, derived, seen));
149 }, [])
150 .concat([node]);
151}
152export function getDerivedClasses(program, languageService, node) {
153 const result = getDerivedClassesWorker(program, languageService, node);
154 return result.filter((value) => value !== node);
155}
156function getImplementorsWorker(program, languageService, node, seen = new Set()) {
157 if (seen.has(node)) {
158 return [];
159 }
160 return reference
161 .findReferencesAsNodes(program, languageService, node)
162 .reduce((acc, ref) => {
163 const parent = node_.getParent(ref);
164 if (parent === undefined) {
165 return acc;
166 }
167 const clause = node_.getParent(parent);
168 if (clause === undefined ||
169 !ts.isHeritageClause(clause) ||
170 (!heritage.isImplements(clause) && !heritage.isExtends(clause))) {
171 return acc;
172 }
173 let derived = node_.getFirstAncestorByKind(clause, ts.SyntaxKind.ClassDeclaration);
174 if (derived === undefined) {
175 derived = node_.getFirstAncestorByKindOrThrow(clause, ts.SyntaxKind.InterfaceDeclaration);
176 }
177 return acc.concat(getImplementorsWorker(program, languageService, derived, seen));
178 }, [])
179 .concat(ts.isClassDeclaration(node) ? [node] : []);
180}
181export function getImplementors(program, languageService, node) {
182 return getImplementorsWorker(program, languageService, node);
183}
184function getExtendorsWorker(program, languageService, node, seen = new Set()) {
185 if (seen.has(node)) {
186 return [];
187 }
188 return reference
189 .findReferencesAsNodes(program, languageService, node)
190 .reduce((acc, ref) => {
191 const parent = node_.getParent(ref);
192 if (parent === undefined) {
193 return acc;
194 }
195 const clause = node_.getParent(parent);
196 if (clause === undefined || !ts.isHeritageClause(clause) || !heritage.isExtends(clause)) {
197 return acc;
198 }
199 const derived = node_.getFirstAncestorByTestOrThrow(clause, ts.isClassDeclaration);
200 return acc.concat(getImplementorsWorker(program, languageService, derived, seen));
201 }, [])
202 .concat(ts.isClassDeclaration(node) ? [node] : []);
203}
204export function getExtendors(program, languageService, node) {
205 return getExtendorsWorker(program, languageService, node);
206}
207export function getBaseTypes(typeChecker, node) {
208 return type_.getBaseTypesArray(type_.getType(typeChecker, node));
209}
210export function getBaseTypesFlattened(typeChecker, node) {
211 function getBaseTypesWorker(type) {
212 if (type_.isIntersection(type)) {
213 return _.flatten(type_.getIntersectionTypesArray(type).map(getBaseTypesWorker));
214 }
215 const baseTypes = type_.getBaseTypesArray(type);
216 return [type].concat(_.flatten(baseTypes.map(getBaseTypesWorker)));
217 }
218 return _.flatten(getBaseTypes(typeChecker, node).map(getBaseTypesWorker));
219}
220export function getBaseClasses(typeChecker, node) {
221 const baseTypes = getBaseTypesFlattened(typeChecker, node);
222 return baseTypes
223 .map((type) => type_.getSymbol(type))
224 .filter(utils.notNull)
225 .map((symbol) => symbol_.getDeclarations(symbol))
226 .reduce((a, b) => a.concat(b), [])
227 .filter(ts.isClassDeclaration);
228}
229export function getBaseClass(typeChecker, node) {
230 const declarations = getBaseClasses(typeChecker, node);
231 return declarations.length === 1 ? declarations[0] : undefined;
232}
233
234//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["class_.ts"],"names":[],"mappings":"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","file":"neo-one-ts-utils/src/class_.js","sourcesContent":["import _ from 'lodash';\nimport ts from 'typescript';\nimport { heritage, modifier, overload, parametered } from './base';\nimport * as declaration from './declaration';\nimport * as node_ from './node';\nimport * as reference from './reference';\nimport * as symbol_ from './symbol';\nimport * as type_ from './type_';\nimport * as utils from './utils';\n\nexport type ClassPropertyType = ts.PropertyDeclaration | ts.GetAccessorDeclaration | ts.SetAccessorDeclaration;\nexport function isClassProperty(node: ts.Node): node is ClassPropertyType {\n  return ts.isPropertyDeclaration(node) || ts.isGetAccessorDeclaration(node) || ts.isSetAccessorDeclaration(node);\n}\n\nexport type ClassInstancePropertyType = ClassPropertyType | ts.ParameterPropertyDeclaration;\nexport function isClassInstanceProperty(node: ts.Node): node is ClassInstancePropertyType {\n  return (isClassProperty(node) || ts.isParameterPropertyDeclaration(node, node.parent)) && !modifier.isStatic(node);\n}\n\nexport type ClassInstanceMemberType = ts.MethodDeclaration | ClassInstancePropertyType;\nexport function isClassInstanceMember(node: ts.Node): node is ClassInstanceMemberType {\n  return (ts.isMethodDeclaration(node) || isClassInstanceProperty(node)) && !modifier.isStatic(node);\n}\n\nexport type ClassStaticPropertyType = ClassPropertyType;\nexport function isClassStaticProperty(node: ts.Node): node is ClassStaticPropertyType {\n  return (ts.isMethodDeclaration(node) || isClassProperty(node)) && modifier.isStatic(node);\n}\n\nexport type ClassStaticMemberType = ts.MethodDeclaration | ClassStaticPropertyType;\nexport function isClassStaticMember(node: ts.Node): node is ClassStaticMemberType {\n  return (ts.isMethodDeclaration(node) || isClassProperty(node)) && modifier.isStatic(node);\n}\n\nexport type ClassMemberType = ClassInstanceMemberType | ClassStaticMemberType;\nexport function isClassMember(node: ts.Node): node is ClassMemberType {\n  return isClassInstanceMember(node) || isClassStaticMember(node);\n}\n\nexport function getExtends(node: ts.ClassDeclaration | ts.ClassExpression): ts.ExpressionWithTypeArguments | undefined {\n  const extendsClause = heritage.getHeritageClauseByKind(node, ts.SyntaxKind.ExtendsKeyword);\n  if (extendsClause === undefined) {\n    return undefined;\n  }\n\n  const typeNodes = heritage.getTypeNodes(extendsClause);\n\n  return typeNodes.length === 0 ? undefined : typeNodes[0];\n}\n\nexport function getExtendsOrThrow(node: ts.ClassDeclaration | ts.ClassExpression): ts.ExpressionWithTypeArguments {\n  return utils.throwIfNullOrUndefined(getExtends(node), 'extends expression');\n}\n\nexport function getImplements(\n  node: ts.ClassDeclaration | ts.ClassExpression,\n): readonly ts.ExpressionWithTypeArguments[] | undefined {\n  const implementsClause = heritage.getHeritageClauseByKind(node, ts.SyntaxKind.ImplementsKeyword);\n  if (implementsClause === undefined) {\n    return undefined;\n  }\n\n  return heritage.getTypeNodes(implementsClause);\n}\n\nexport function getImplementsArray(\n  node: ts.ClassDeclaration | ts.ClassExpression,\n): readonly ts.ExpressionWithTypeArguments[] {\n  return utils.getArray(getImplements(node));\n}\n\nexport function getMembers(node: ts.ClassDeclaration | ts.ClassExpression): readonly ClassMemberType[] {\n  // tslint:disable-next-line readonly-array\n  const members: Array<ts.ClassElement | ts.ParameterPropertyDeclaration> = [...node.members];\n  const implementationCtors = members.filter(ts.isConstructorDeclaration).filter((c) => overload.isImplementation(c));\n  // tslint:disable-next-line no-loop-statement\n  for (const ctor of implementationCtors) {\n    // insert after the constructor\n    let insertIndex = members.indexOf(ctor) + 1;\n    // tslint:disable-next-line no-loop-statement\n    for (const param of parametered.getParameters(ctor)) {\n      if (ts.isParameterPropertyDeclaration(param, param.parent)) {\n        // tslint:disable-next-line no-array-mutation\n        members.splice(insertIndex, 0, param);\n        insertIndex += 1;\n      }\n    }\n  }\n\n  return members.filter(isClassMember);\n}\n\nexport function getConcreteMembers(node: ts.ClassDeclaration | ts.ClassExpression): readonly ClassMemberType[] {\n  return declaration.isAmbient(node)\n    ? []\n    : getMembers(node).filter((member) => {\n        if (ts.isMethodDeclaration(member)) {\n          return overload.isImplementation(member);\n        }\n\n        return true;\n      });\n}\n\nexport function getInstanceProperties(\n  node: ts.ClassDeclaration | ts.ClassExpression,\n): readonly ClassInstancePropertyType[] {\n  return getMembers(node).filter(isClassInstanceProperty);\n}\n\nexport function getInstanceMembers(node: ts.ClassDeclaration | ts.ClassExpression): readonly ClassInstanceMemberType[] {\n  return getMembers(node).filter(isClassInstanceMember);\n}\n\nexport function getInstanceMethods(node: ts.ClassDeclaration | ts.ClassExpression): readonly ts.MethodDeclaration[] {\n  return getInstanceMembers(node).filter(ts.isMethodDeclaration);\n}\n\nexport function getMethods(node: ts.ClassDeclaration | ts.ClassExpression): readonly ts.MethodDeclaration[] {\n  return getMembers(node).filter(ts.isMethodDeclaration);\n}\n\nexport function getSetAccessors(node: ts.ClassDeclaration | ts.ClassExpression): readonly ts.SetAccessorDeclaration[] {\n  return getMembers(node).filter(ts.isSetAccessor);\n}\n\nexport function getInstanceMethod(\n  node: ts.ClassDeclaration | ts.ClassExpression,\n  name: string,\n): ts.MethodDeclaration | undefined {\n  return getInstanceMethods(node).find((method) => node_.getName(method) === name);\n}\n\nexport function getConcreteInstanceProperties(\n  node: ts.ClassDeclaration | ts.ClassExpression,\n): readonly ClassInstancePropertyType[] {\n  return getConcreteMembers(node).filter(isClassInstanceProperty);\n}\n\nexport function getConcreteInstanceMembers(\n  node: ts.ClassDeclaration | ts.ClassExpression,\n): readonly ClassInstanceMemberType[] {\n  return getConcreteMembers(node).filter(isClassInstanceMember);\n}\n\nexport function getConcreteInstanceMethods(\n  node: ts.ClassDeclaration | ts.ClassExpression,\n): readonly ts.MethodDeclaration[] {\n  return getConcreteInstanceMembers(node).filter(ts.isMethodDeclaration);\n}\n\nexport function getStaticProperties(\n  node: ts.ClassDeclaration | ts.ClassExpression,\n): readonly ClassStaticPropertyType[] {\n  return getMembers(node).filter(isClassStaticProperty);\n}\n\nexport function getStaticMembers(node: ts.ClassDeclaration | ts.ClassExpression): readonly ClassStaticMemberType[] {\n  return getMembers(node).filter(isClassStaticMember);\n}\n\nexport function getConcreteStaticProperties(\n  node: ts.ClassDeclaration | ts.ClassExpression,\n): readonly ClassStaticPropertyType[] {\n  return getConcreteMembers(node).filter(isClassStaticProperty);\n}\n\nexport function getConcreteStaticMembers(\n  node: ts.ClassDeclaration | ts.ClassExpression,\n): readonly ClassStaticMemberType[] {\n  return getConcreteMembers(node).filter(isClassStaticMember);\n}\n\nexport function getConcreteStaticMethods(\n  node: ts.ClassDeclaration | ts.ClassExpression,\n): readonly ts.MethodDeclaration[] {\n  return getConcreteStaticMembers(node).filter(ts.isMethodDeclaration);\n}\n\nexport function getConstructors(node: ts.ClassDeclaration | ts.ClassExpression): readonly ts.ConstructorDeclaration[] {\n  return node.members.filter(ts.isConstructorDeclaration);\n}\n\nexport function getConcreteConstructor(\n  node: ts.ClassDeclaration | ts.ClassExpression,\n): ts.ConstructorDeclaration | undefined {\n  return getConstructors(node).find((ctor) => overload.isImplementation(ctor));\n}\n\nexport function getFirstConcreteConstructor(\n  typeChecker: ts.TypeChecker,\n  node: ts.ClassDeclaration,\n): ts.ConstructorDeclaration | undefined {\n  const ctor = getConcreteConstructor(node);\n  if (ctor !== undefined) {\n    return ctor;\n  }\n\n  const baseClass = getBaseClass(typeChecker, node);\n  if (baseClass === undefined) {\n    return undefined;\n  }\n\n  return getFirstConcreteConstructor(typeChecker, baseClass);\n}\n\nfunction getDerivedClassesWorker(\n  program: ts.Program,\n  languageService: ts.LanguageService,\n  node: ts.ClassDeclaration,\n  seen = new Set<ts.ClassDeclaration>(),\n): readonly ts.ClassDeclaration[] {\n  if (seen.has(node)) {\n    return [];\n  }\n\n  return reference\n    .findReferencesAsNodes(program, languageService, node)\n    .reduce<readonly ts.ClassDeclaration[]>((acc, ref) => {\n      const parent = node_.getParent(ref) as ts.Node | undefined;\n      if (parent === undefined) {\n        return acc;\n      }\n\n      const clause = node_.getParent(parent) as ts.Node | undefined;\n      if (clause === undefined || !ts.isHeritageClause(clause) || !heritage.isExtends(clause)) {\n        return acc;\n      }\n\n      const derived = node_.getFirstAncestorByKindOrThrow<ts.ClassDeclaration>(clause, ts.SyntaxKind.ClassDeclaration);\n\n      return acc.concat(getDerivedClassesWorker(program, languageService, derived, seen));\n    }, [])\n    .concat([node]);\n}\n\nexport function getDerivedClasses(\n  program: ts.Program,\n  languageService: ts.LanguageService,\n  node: ts.ClassDeclaration,\n): readonly ts.ClassDeclaration[] {\n  const result = getDerivedClassesWorker(program, languageService, node);\n\n  return result.filter((value) => value !== node);\n}\n\nfunction getImplementorsWorker(\n  program: ts.Program,\n  languageService: ts.LanguageService,\n  node: ts.ClassDeclaration | ts.InterfaceDeclaration,\n  seen = new Set<ts.ClassDeclaration | ts.InterfaceDeclaration>(),\n): readonly ts.ClassDeclaration[] {\n  if (seen.has(node)) {\n    return [];\n  }\n\n  return reference\n    .findReferencesAsNodes(program, languageService, node)\n    .reduce<readonly ts.ClassDeclaration[]>((acc, ref) => {\n      const parent = node_.getParent(ref) as ts.Node | undefined;\n      if (parent === undefined) {\n        return acc;\n      }\n\n      const clause = node_.getParent(parent) as ts.Node | undefined;\n      if (\n        clause === undefined ||\n        !ts.isHeritageClause(clause) ||\n        (!heritage.isImplements(clause) && !heritage.isExtends(clause))\n      ) {\n        return acc;\n      }\n\n      let derived: ts.ClassDeclaration | ts.InterfaceDeclaration | undefined = node_.getFirstAncestorByKind<\n        ts.ClassDeclaration\n      >(clause, ts.SyntaxKind.ClassDeclaration);\n      if (derived === undefined) {\n        derived = node_.getFirstAncestorByKindOrThrow<ts.InterfaceDeclaration>(\n          clause,\n          ts.SyntaxKind.InterfaceDeclaration,\n        );\n      }\n\n      return acc.concat(getImplementorsWorker(program, languageService, derived, seen));\n    }, [])\n    .concat(ts.isClassDeclaration(node) ? [node] : []);\n}\n\nexport function getImplementors(\n  program: ts.Program,\n  languageService: ts.LanguageService,\n  node: ts.InterfaceDeclaration,\n): readonly ts.ClassDeclaration[] {\n  return getImplementorsWorker(program, languageService, node);\n}\n\nfunction getExtendorsWorker(\n  program: ts.Program,\n  languageService: ts.LanguageService,\n  node: ts.ClassDeclaration,\n  seen = new Set<ts.ClassDeclaration>(),\n): readonly ts.ClassDeclaration[] {\n  if (seen.has(node)) {\n    return [];\n  }\n\n  return reference\n    .findReferencesAsNodes(program, languageService, node)\n    .reduce<readonly ts.ClassDeclaration[]>((acc, ref) => {\n      const parent = node_.getParent(ref) as ts.Node | undefined;\n      if (parent === undefined) {\n        return acc;\n      }\n\n      const clause = node_.getParent(parent) as ts.Node | undefined;\n      if (clause === undefined || !ts.isHeritageClause(clause) || !heritage.isExtends(clause)) {\n        return acc;\n      }\n\n      const derived: ts.ClassDeclaration | undefined = node_.getFirstAncestorByTestOrThrow(\n        clause,\n        ts.isClassDeclaration,\n      );\n\n      return acc.concat(getImplementorsWorker(program, languageService, derived, seen));\n    }, [])\n    .concat(ts.isClassDeclaration(node) ? [node] : []);\n}\n\nexport function getExtendors(\n  program: ts.Program,\n  languageService: ts.LanguageService,\n  node: ts.ClassDeclaration,\n): readonly ts.ClassDeclaration[] {\n  return getExtendorsWorker(program, languageService, node);\n}\n\nexport function getBaseTypes(\n  typeChecker: ts.TypeChecker,\n  node: ts.ClassDeclaration | ts.ClassExpression | ts.InterfaceDeclaration,\n): readonly ts.Type[] {\n  return type_.getBaseTypesArray(type_.getType(typeChecker, node));\n}\n\nexport function getBaseTypesFlattened(\n  typeChecker: ts.TypeChecker,\n  node: ts.ClassDeclaration | ts.ClassExpression | ts.InterfaceDeclaration,\n): readonly ts.Type[] {\n  function getBaseTypesWorker(type: ts.Type): readonly ts.Type[] {\n    if (type_.isIntersection(type)) {\n      return _.flatten(type_.getIntersectionTypesArray(type).map(getBaseTypesWorker));\n    }\n\n    const baseTypes = type_.getBaseTypesArray(type);\n\n    return [type].concat(_.flatten(baseTypes.map(getBaseTypesWorker)));\n  }\n\n  return _.flatten(getBaseTypes(typeChecker, node).map(getBaseTypesWorker));\n}\n\nexport function getBaseClasses(\n  typeChecker: ts.TypeChecker,\n  node: ts.ClassDeclaration | ts.ClassExpression,\n): readonly ts.ClassDeclaration[] {\n  const baseTypes = getBaseTypesFlattened(typeChecker, node);\n\n  return baseTypes\n    .map((type) => type_.getSymbol(type))\n    .filter(utils.notNull)\n    .map((symbol) => symbol_.getDeclarations(symbol))\n    .reduce<readonly ts.Declaration[]>((a, b) => a.concat(b), [])\n    .filter(ts.isClassDeclaration);\n}\n\nexport function getBaseClass(\n  typeChecker: ts.TypeChecker,\n  node: ts.ClassDeclaration | ts.ClassExpression,\n): ts.ClassDeclaration | undefined {\n  const declarations = getBaseClasses(typeChecker, node);\n\n  return declarations.length === 1 ? declarations[0] : undefined;\n}\n"]}