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:
Pete Bacon Darwin 2020-09-27 11:54:37 +01:00 committed by atscott
parent 1d6e67478e
commit 65997c0649
1 changed files with 61 additions and 47 deletions

View File

@ -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;