refactor(ngcc): simplify and break up ES2015 functions with helpers (#38959)
The protected helper functions can then be overridden by subclasses of the Esm2015ReflectionHost. PR Close #38959
This commit is contained in:
parent
1d6e67478e
commit
65997c0649
|
@ -132,15 +132,8 @@ export class Esm2015ReflectionHost extends TypeScriptReflectionHost implements N
|
||||||
if (symbol !== undefined) {
|
if (symbol !== undefined) {
|
||||||
return symbol;
|
return symbol;
|
||||||
}
|
}
|
||||||
|
const innerDeclaration = this.getInnerDeclarationFromAliasOrInner(declaration);
|
||||||
if (declaration.parent !== undefined && isNamedVariableDeclaration(declaration.parent)) {
|
return this.getClassSymbolFromInnerDeclaration(innerDeclaration);
|
||||||
const variableValue = this.getVariableValue(declaration.parent);
|
|
||||||
if (variableValue !== null) {
|
|
||||||
declaration = variableValue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return this.getClassSymbolFromInnerDeclaration(declaration);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -243,13 +236,7 @@ export class Esm2015ReflectionHost extends TypeScriptReflectionHost implements N
|
||||||
clazz.name.text} to be a class declaration.`);
|
clazz.name.text} to be a class declaration.`);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (classSymbol.adjacent !== undefined) {
|
return this.getAdjacentNameOfClassSymbol(classSymbol);
|
||||||
return this.getNameFromClassSymbolDeclaration(
|
|
||||||
classSymbol, classSymbol.adjacent.valueDeclaration);
|
|
||||||
} else {
|
|
||||||
return this.getNameFromClassSymbolDeclaration(
|
|
||||||
classSymbol, classSymbol.implementation.valueDeclaration);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private getNameFromClassSymbolDeclaration(
|
private getNameFromClassSymbolDeclaration(
|
||||||
|
@ -430,21 +417,8 @@ export class Esm2015ReflectionHost extends TypeScriptReflectionHost implements N
|
||||||
*/
|
*/
|
||||||
findClassSymbols(sourceFile: ts.SourceFile): NgccClassSymbol[] {
|
findClassSymbols(sourceFile: ts.SourceFile): NgccClassSymbol[] {
|
||||||
const classes: NgccClassSymbol[] = [];
|
const classes: NgccClassSymbol[] = [];
|
||||||
this.getModuleStatements(sourceFile).forEach(statement => {
|
this.getModuleStatements(sourceFile)
|
||||||
if (ts.isVariableStatement(statement)) {
|
.forEach(statement => this.addClassSymbolsFromStatement(classes, statement));
|
||||||
statement.declarationList.declarations.forEach(declaration => {
|
|
||||||
const classSymbol = this.getClassSymbol(declaration);
|
|
||||||
if (classSymbol) {
|
|
||||||
classes.push(classSymbol);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
} else if (ts.isClassDeclaration(statement)) {
|
|
||||||
const classSymbol = this.getClassSymbol(statement);
|
|
||||||
if (classSymbol) {
|
|
||||||
classes.push(classSymbol);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
return classes;
|
return classes;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -569,6 +543,41 @@ export class Esm2015ReflectionHost extends TypeScriptReflectionHost implements N
|
||||||
|
|
||||||
///////////// Protected Helpers /////////////
|
///////////// Protected Helpers /////////////
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Extract all the "classes" from the `statement` and add them to the `classes` array.
|
||||||
|
*/
|
||||||
|
protected addClassSymbolsFromStatement(classes: NgccClassSymbol[], statement: ts.Statement):
|
||||||
|
void {
|
||||||
|
if (ts.isVariableStatement(statement)) {
|
||||||
|
statement.declarationList.declarations.forEach(declaration => {
|
||||||
|
const classSymbol = this.getClassSymbol(declaration);
|
||||||
|
if (classSymbol) {
|
||||||
|
classes.push(classSymbol);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} else if (ts.isClassDeclaration(statement)) {
|
||||||
|
const classSymbol = this.getClassSymbol(statement);
|
||||||
|
if (classSymbol) {
|
||||||
|
classes.push(classSymbol);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Compute the inner declaration node of a "class" from the given `declaration` node.
|
||||||
|
*
|
||||||
|
* @param declaration a node that is either an inner declaration or an alias of a class.
|
||||||
|
*/
|
||||||
|
protected getInnerDeclarationFromAliasOrInner(declaration: ts.Node): ts.Node {
|
||||||
|
if (declaration.parent !== undefined && isNamedVariableDeclaration(declaration.parent)) {
|
||||||
|
const variableValue = this.getVariableValue(declaration.parent);
|
||||||
|
if (variableValue !== null) {
|
||||||
|
declaration = variableValue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return declaration;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A class may be declared as a top level class declaration:
|
* A class may be declared as a top level class declaration:
|
||||||
*
|
*
|
||||||
|
@ -611,7 +620,7 @@ export class Esm2015ReflectionHost extends TypeScriptReflectionHost implements N
|
||||||
protected getClassSymbolFromOuterDeclaration(declaration: ts.Node): NgccClassSymbol|undefined {
|
protected getClassSymbolFromOuterDeclaration(declaration: ts.Node): NgccClassSymbol|undefined {
|
||||||
// Return a class symbol without an inner declaration if it is a regular "top level" class
|
// Return a class symbol without an inner declaration if it is a regular "top level" class
|
||||||
if (isNamedClassDeclaration(declaration) && isTopLevel(declaration)) {
|
if (isNamedClassDeclaration(declaration) && isTopLevel(declaration)) {
|
||||||
return this.createClassSymbol(declaration, null);
|
return this.createClassSymbol(declaration.name, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Otherwise, an outer class declaration must be an initialized variable declaration:
|
// Otherwise, an outer class declaration must be an initialized variable declaration:
|
||||||
|
@ -620,12 +629,11 @@ export class Esm2015ReflectionHost extends TypeScriptReflectionHost implements N
|
||||||
}
|
}
|
||||||
|
|
||||||
const innerDeclaration = getInnerClassDeclaration(skipClassAliases(declaration));
|
const innerDeclaration = getInnerClassDeclaration(skipClassAliases(declaration));
|
||||||
if (innerDeclaration !== null) {
|
if (innerDeclaration === null) {
|
||||||
return this.createClassSymbol(declaration, innerDeclaration);
|
return undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return this.createClassSymbol(declaration.name, innerDeclaration);
|
||||||
return undefined;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -681,7 +689,7 @@ export class Esm2015ReflectionHost extends TypeScriptReflectionHost implements N
|
||||||
return undefined;
|
return undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
return this.createClassSymbol(outerDeclaration, declaration);
|
return this.createClassSymbol(outerDeclaration.name, declaration);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -695,10 +703,10 @@ export class Esm2015ReflectionHost extends TypeScriptReflectionHost implements N
|
||||||
* @returns the `NgccClassSymbol` representing the class, or undefined if a `ts.Symbol` for any of
|
* @returns the `NgccClassSymbol` representing the class, or undefined if a `ts.Symbol` for any of
|
||||||
* the declarations could not be resolved.
|
* the declarations could not be resolved.
|
||||||
*/
|
*/
|
||||||
protected createClassSymbol(outerDeclaration: ClassDeclaration, innerDeclaration: ts.Node|null):
|
protected createClassSymbol(outerDeclaration: ts.Identifier, innerDeclaration: ts.Node|null):
|
||||||
NgccClassSymbol|undefined {
|
NgccClassSymbol|undefined {
|
||||||
const declarationSymbol =
|
const declarationSymbol =
|
||||||
this.checker.getSymbolAtLocation(outerDeclaration.name) as ClassSymbol | undefined;
|
this.checker.getSymbolAtLocation(outerDeclaration) as ClassSymbol | undefined;
|
||||||
if (declarationSymbol === undefined) {
|
if (declarationSymbol === undefined) {
|
||||||
return undefined;
|
return undefined;
|
||||||
}
|
}
|
||||||
|
@ -716,13 +724,9 @@ export class Esm2015ReflectionHost extends TypeScriptReflectionHost implements N
|
||||||
name: declarationSymbol.name,
|
name: declarationSymbol.name,
|
||||||
declaration: declarationSymbol,
|
declaration: declarationSymbol,
|
||||||
implementation: implementationSymbol,
|
implementation: implementationSymbol,
|
||||||
|
adjacent: this.getAdjacentSymbol(declarationSymbol, implementationSymbol),
|
||||||
};
|
};
|
||||||
|
|
||||||
let adjacent = this.getAdjacentSymbol(declarationSymbol, implementationSymbol);
|
|
||||||
if (adjacent !== null) {
|
|
||||||
classSymbol.adjacent = adjacent;
|
|
||||||
}
|
|
||||||
|
|
||||||
return classSymbol;
|
return classSymbol;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1166,8 +1170,7 @@ export class Esm2015ReflectionHost extends TypeScriptReflectionHost implements N
|
||||||
|
|
||||||
const outerDeclaration = classSymbol.declaration.valueDeclaration;
|
const outerDeclaration = classSymbol.declaration.valueDeclaration;
|
||||||
const innerDeclaration = classSymbol.implementation.valueDeclaration;
|
const innerDeclaration = classSymbol.implementation.valueDeclaration;
|
||||||
const adjacentDeclaration =
|
const adjacentDeclaration = this.getAdjacentNameOfClassSymbol(classSymbol).parent;
|
||||||
this.getAdjacentNameOfClass((classSymbol.declaration.valueDeclaration)).parent;
|
|
||||||
const matchesClass = (identifier: ts.Identifier) => {
|
const matchesClass = (identifier: ts.Identifier) => {
|
||||||
const decl = this.getDeclarationOfIdentifier(identifier);
|
const decl = this.getDeclarationOfIdentifier(identifier);
|
||||||
return decl !== null &&
|
return decl !== null &&
|
||||||
|
@ -2017,6 +2020,16 @@ export class Esm2015ReflectionHost extends TypeScriptReflectionHost implements N
|
||||||
}
|
}
|
||||||
return reflectEnumAssignment(innerExpression);
|
return reflectEnumAssignment(innerExpression);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private getAdjacentNameOfClassSymbol(classSymbol: NgccClassSymbol): ts.Identifier {
|
||||||
|
if (classSymbol.adjacent !== undefined) {
|
||||||
|
return this.getNameFromClassSymbolDeclaration(
|
||||||
|
classSymbol, classSymbol.adjacent.valueDeclaration);
|
||||||
|
} else {
|
||||||
|
return this.getNameFromClassSymbolDeclaration(
|
||||||
|
classSymbol, classSymbol.implementation.valueDeclaration);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
///////////// Exported Helpers /////////////
|
///////////// Exported Helpers /////////////
|
||||||
|
@ -2305,6 +2318,7 @@ function isInitializedVariableClassDeclaration(node: ts.Node):
|
||||||
node is InitializedVariableClassDeclaration {
|
node is InitializedVariableClassDeclaration {
|
||||||
return isNamedVariableDeclaration(node) && node.initializer !== undefined;
|
return isNamedVariableDeclaration(node) && node.initializer !== undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Handle a variable declaration of the form
|
* Handle a variable declaration of the form
|
||||||
*
|
*
|
||||||
|
@ -2364,7 +2378,7 @@ export function skipClassAliases(node: InitializedVariableClassDeclaration): ts.
|
||||||
* @param expression the node that represents the class whose declaration we are finding.
|
* @param expression the node that represents the class whose declaration we are finding.
|
||||||
* @returns the declaration of the class or `null` if it is not a "class".
|
* @returns the declaration of the class or `null` if it is not a "class".
|
||||||
*/
|
*/
|
||||||
function getInnerClassDeclaration(expression: ts.Expression):
|
export function getInnerClassDeclaration(expression: ts.Expression):
|
||||||
ClassDeclaration<ts.ClassExpression|ts.ClassDeclaration|ts.FunctionDeclaration>|null {
|
ClassDeclaration<ts.ClassExpression|ts.ClassDeclaration|ts.FunctionDeclaration>|null {
|
||||||
if (ts.isClassExpression(expression) && hasNameIdentifier(expression)) {
|
if (ts.isClassExpression(expression) && hasNameIdentifier(expression)) {
|
||||||
return expression;
|
return expression;
|
||||||
|
|
Loading…
Reference in New Issue