blob: 4e9317229a61f06f58be44e9b2f85d658388f1b1 [file] [log] [blame]
/**
* @license
* Copyright Google Inc. All Rights Reserved.
*
* Use of this source code is governed by an MIT-style license that can be
* found in the LICENSE file at https://angular.io/license
*/
(function (factory) {
if (typeof module === "object" && typeof module.exports === "object") {
var v = factory(require, exports);
if (v !== undefined) module.exports = v;
}
else if (typeof define === "function" && define.amd) {
define("@angular/core/schematics/migrations/static-queries/angular/ng_query_visitor", ["require", "exports", "typescript", "@angular/core/schematics/utils/ng_decorators", "@angular/core/schematics/utils/typescript/class_declaration", "@angular/core/schematics/utils/typescript/property_name", "@angular/core/schematics/migrations/static-queries/angular/directive_inputs", "@angular/core/schematics/migrations/static-queries/angular/query-definition"], factory);
}
})(function (require, exports) {
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
const ts = require("typescript");
const ng_decorators_1 = require("@angular/core/schematics/utils/ng_decorators");
const class_declaration_1 = require("@angular/core/schematics/utils/typescript/class_declaration");
const property_name_1 = require("@angular/core/schematics/utils/typescript/property_name");
const directive_inputs_1 = require("@angular/core/schematics/migrations/static-queries/angular/directive_inputs");
const query_definition_1 = require("@angular/core/schematics/migrations/static-queries/angular/query-definition");
/**
* Visitor that can be used to determine Angular queries within given TypeScript nodes.
* Besides resolving queries, the visitor also records class relations and searches for
* Angular input setters which can be used to analyze the timing usage of a given query.
*/
class NgQueryResolveVisitor {
constructor(typeChecker) {
this.typeChecker = typeChecker;
/** Resolved Angular query definitions. */
this.resolvedQueries = new Map();
/** Maps a class declaration to its class metadata. */
this.classMetadata = new Map();
}
visitNode(node) {
switch (node.kind) {
case ts.SyntaxKind.PropertyDeclaration:
this.visitPropertyDeclaration(node);
break;
case ts.SyntaxKind.ClassDeclaration:
this.visitClassDeclaration(node);
break;
case ts.SyntaxKind.GetAccessor:
case ts.SyntaxKind.SetAccessor:
this.visitAccessorDeclaration(node);
break;
}
ts.forEachChild(node, n => this.visitNode(n));
}
visitPropertyDeclaration(node) {
this._recordQueryDeclaration(node, node, property_name_1.getPropertyNameText(node.name));
}
visitAccessorDeclaration(node) {
this._recordQueryDeclaration(node, null, property_name_1.getPropertyNameText(node.name));
}
visitClassDeclaration(node) {
this._recordClassInputSetters(node);
this._recordClassInheritances(node);
}
_recordQueryDeclaration(node, property, queryName) {
if (!node.decorators || !node.decorators.length) {
return;
}
const ngDecorators = ng_decorators_1.getAngularDecorators(this.typeChecker, node.decorators);
const queryDecorator = ngDecorators.find(({ name }) => name === 'ViewChild' || name === 'ContentChild');
// Ensure that the current property declaration is defining a query.
if (!queryDecorator) {
return;
}
const queryContainer = class_declaration_1.findParentClassDeclaration(node);
// If the query is not located within a class declaration, skip this node.
if (!queryContainer) {
return;
}
const sourceFile = node.getSourceFile();
const newQueries = this.resolvedQueries.get(sourceFile) || [];
this.resolvedQueries.set(sourceFile, newQueries.concat({
name: queryName,
type: queryDecorator.name === 'ViewChild' ? query_definition_1.QueryType.ViewChild : query_definition_1.QueryType.ContentChild,
node,
property,
decorator: queryDecorator,
container: queryContainer,
}));
}
_recordClassInputSetters(node) {
const resolvedInputNames = directive_inputs_1.getInputNamesOfClass(node, this.typeChecker);
if (resolvedInputNames) {
const classMetadata = this._getClassMetadata(node);
classMetadata.ngInputNames = resolvedInputNames;
this.classMetadata.set(node, classMetadata);
}
}
_recordClassInheritances(node) {
const baseTypes = class_declaration_1.getBaseTypeIdentifiers(node);
if (!baseTypes || baseTypes.length !== 1) {
return;
}
const superClass = baseTypes[0];
const baseClassMetadata = this._getClassMetadata(node);
// We need to resolve the value declaration through the resolved type as the base
// class could be declared in different source files and the local symbol won't
// contain a value declaration as the value is not declared locally.
const symbol = this.typeChecker.getTypeAtLocation(superClass).getSymbol();
if (symbol && symbol.valueDeclaration && ts.isClassDeclaration(symbol.valueDeclaration)) {
const extendedClass = symbol.valueDeclaration;
const classMetadataExtended = this._getClassMetadata(extendedClass);
// Record all classes that derive from the given class. This makes it easy to
// determine all classes that could potentially use inherited queries statically.
classMetadataExtended.derivedClasses.push(node);
this.classMetadata.set(extendedClass, classMetadataExtended);
// Record the super class of the current class.
baseClassMetadata.superClass = extendedClass;
this.classMetadata.set(node, baseClassMetadata);
}
}
_getClassMetadata(node) {
return this.classMetadata.get(node) || { derivedClasses: [], superClass: null, ngInputNames: [] };
}
}
exports.NgQueryResolveVisitor = NgQueryResolveVisitor;
});
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ng_query_visitor.js","sourceRoot":"","sources":["../../../../../../../../../packages/core/schematics/migrations/static-queries/angular/ng_query_visitor.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google Inc. All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport * as ts from 'typescript';\n\nimport {ResolvedTemplate} from '../../../utils/ng_component_template';\nimport {getAngularDecorators} from '../../../utils/ng_decorators';\nimport {findParentClassDeclaration, getBaseTypeIdentifiers} from '../../../utils/typescript/class_declaration';\nimport {getPropertyNameText} from '../../../utils/typescript/property_name';\n\nimport {getInputNamesOfClass} from './directive_inputs';\nimport {NgQueryDefinition, QueryType} from './query-definition';\n\n\n/** Resolved metadata of a given class. */\nexport interface ClassMetadata {\n  /** List of class declarations that derive from the given class. */\n  derivedClasses: ts.ClassDeclaration[];\n  /** Super class of the given class. */\n  superClass: ts.ClassDeclaration|null;\n  /** List of property names that declare an Angular input within the given class. */\n  ngInputNames: string[];\n  /** Component template that belongs to that class if present. */\n  template?: ResolvedTemplate;\n}\n\n/** Type that describes a map which can be used to get a class declaration's metadata. */\nexport type ClassMetadataMap = Map<ts.ClassDeclaration, ClassMetadata>;\n\n/**\n * Visitor that can be used to determine Angular queries within given TypeScript nodes.\n * Besides resolving queries, the visitor also records class relations and searches for\n * Angular input setters which can be used to analyze the timing usage of a given query.\n */\nexport class NgQueryResolveVisitor {\n  /** Resolved Angular query definitions. */\n  resolvedQueries = new Map<ts.SourceFile, NgQueryDefinition[]>();\n\n  /** Maps a class declaration to its class metadata. */\n  classMetadata: ClassMetadataMap = new Map();\n\n  constructor(public typeChecker: ts.TypeChecker) {}\n\n  visitNode(node: ts.Node) {\n    switch (node.kind) {\n      case ts.SyntaxKind.PropertyDeclaration:\n        this.visitPropertyDeclaration(node as ts.PropertyDeclaration);\n        break;\n      case ts.SyntaxKind.ClassDeclaration:\n        this.visitClassDeclaration(node as ts.ClassDeclaration);\n        break;\n      case ts.SyntaxKind.GetAccessor:\n      case ts.SyntaxKind.SetAccessor:\n        this.visitAccessorDeclaration(node as ts.AccessorDeclaration);\n        break;\n    }\n\n    ts.forEachChild(node, n => this.visitNode(n));\n  }\n\n  private visitPropertyDeclaration(node: ts.PropertyDeclaration) {\n    this._recordQueryDeclaration(node, node, getPropertyNameText(node.name));\n  }\n\n  private visitAccessorDeclaration(node: ts.AccessorDeclaration) {\n    this._recordQueryDeclaration(node, null, getPropertyNameText(node.name));\n  }\n\n  private visitClassDeclaration(node: ts.ClassDeclaration) {\n    this._recordClassInputSetters(node);\n    this._recordClassInheritances(node);\n  }\n\n  private _recordQueryDeclaration(\n      node: ts.Node, property: ts.PropertyDeclaration|null, queryName: string|null) {\n    if (!node.decorators || !node.decorators.length) {\n      return;\n    }\n\n    const ngDecorators = getAngularDecorators(this.typeChecker, node.decorators);\n    const queryDecorator =\n        ngDecorators.find(({name}) => name === 'ViewChild' || name === 'ContentChild');\n\n    // Ensure that the current property declaration is defining a query.\n    if (!queryDecorator) {\n      return;\n    }\n\n    const queryContainer = findParentClassDeclaration(node);\n\n    // If the query is not located within a class declaration, skip this node.\n    if (!queryContainer) {\n      return;\n    }\n\n    const sourceFile = node.getSourceFile();\n    const newQueries = this.resolvedQueries.get(sourceFile) || [];\n\n    this.resolvedQueries.set(sourceFile, newQueries.concat({\n      name: queryName,\n      type: queryDecorator.name === 'ViewChild' ? QueryType.ViewChild : QueryType.ContentChild,\n      node,\n      property,\n      decorator: queryDecorator,\n      container: queryContainer,\n    }));\n  }\n\n  private _recordClassInputSetters(node: ts.ClassDeclaration) {\n    const resolvedInputNames = getInputNamesOfClass(node, this.typeChecker);\n\n    if (resolvedInputNames) {\n      const classMetadata = this._getClassMetadata(node);\n\n      classMetadata.ngInputNames = resolvedInputNames;\n      this.classMetadata.set(node, classMetadata);\n    }\n  }\n\n  private _recordClassInheritances(node: ts.ClassDeclaration) {\n    const baseTypes = getBaseTypeIdentifiers(node);\n\n    if (!baseTypes || baseTypes.length !== 1) {\n      return;\n    }\n\n    const superClass = baseTypes[0];\n    const baseClassMetadata = this._getClassMetadata(node);\n\n    // We need to resolve the value declaration through the resolved type as the base\n    // class could be declared in different source files and the local symbol won't\n    // contain a value declaration as the value is not declared locally.\n    const symbol = this.typeChecker.getTypeAtLocation(superClass).getSymbol();\n\n    if (symbol && symbol.valueDeclaration && ts.isClassDeclaration(symbol.valueDeclaration)) {\n      const extendedClass = symbol.valueDeclaration;\n      const classMetadataExtended = this._getClassMetadata(extendedClass);\n\n      // Record all classes that derive from the given class. This makes it easy to\n      // determine all classes that could potentially use inherited queries statically.\n      classMetadataExtended.derivedClasses.push(node);\n      this.classMetadata.set(extendedClass, classMetadataExtended);\n\n      // Record the super class of the current class.\n      baseClassMetadata.superClass = extendedClass;\n      this.classMetadata.set(node, baseClassMetadata);\n    }\n  }\n\n  private _getClassMetadata(node: ts.ClassDeclaration): ClassMetadata {\n    return this.classMetadata.get(node) || {derivedClasses: [], superClass: null, ngInputNames: []};\n  }\n}\n"]}