@angular/core
Version:
Angular - the core framework
127 lines • 20 kB
JavaScript
/**
* @license
* Copyright Google LLC 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
*/
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
(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/utils/ng_component_template", ["require", "exports", "path", "typescript", "@angular/core/schematics/utils/line_mappings", "@angular/core/schematics/utils/ng_decorators", "@angular/core/schematics/utils/typescript/functions", "@angular/core/schematics/utils/typescript/property_name"], factory);
}
})(function (require, exports) {
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.NgComponentTemplateVisitor = void 0;
const path_1 = require("path");
const typescript_1 = __importDefault(require("typescript"));
const line_mappings_1 = require("@angular/core/schematics/utils/line_mappings");
const ng_decorators_1 = require("@angular/core/schematics/utils/ng_decorators");
const functions_1 = require("@angular/core/schematics/utils/typescript/functions");
const property_name_1 = require("@angular/core/schematics/utils/typescript/property_name");
/**
* Visitor that can be used to determine Angular templates referenced within given
* TypeScript source files (inline templates or external referenced templates)
*/
class NgComponentTemplateVisitor {
constructor(typeChecker, _basePath, _tree) {
this.typeChecker = typeChecker;
this._basePath = _basePath;
this._tree = _tree;
this.resolvedTemplates = [];
}
visitNode(node) {
if (node.kind === typescript_1.default.SyntaxKind.ClassDeclaration) {
this.visitClassDeclaration(node);
}
typescript_1.default.forEachChild(node, n => this.visitNode(n));
}
visitClassDeclaration(node) {
if (!node.decorators || !node.decorators.length) {
return;
}
const ngDecorators = (0, ng_decorators_1.getAngularDecorators)(this.typeChecker, node.decorators);
const componentDecorator = ngDecorators.find(dec => dec.name === 'Component');
// In case no "@Component" decorator could be found on the current class, skip.
if (!componentDecorator) {
return;
}
const decoratorCall = componentDecorator.node.expression;
// In case the component decorator call is not valid, skip this class declaration.
if (decoratorCall.arguments.length !== 1) {
return;
}
const componentMetadata = (0, functions_1.unwrapExpression)(decoratorCall.arguments[0]);
// Ensure that the component metadata is an object literal expression.
if (!typescript_1.default.isObjectLiteralExpression(componentMetadata)) {
return;
}
const sourceFile = node.getSourceFile();
const sourceFileName = sourceFile.fileName;
// Walk through all component metadata properties and determine the referenced
// HTML templates (either external or inline)
componentMetadata.properties.forEach(property => {
if (!typescript_1.default.isPropertyAssignment(property)) {
return;
}
const propertyName = (0, property_name_1.getPropertyNameText)(property.name);
// In case there is an inline template specified, ensure that the value is statically
// analyzable by checking if the initializer is a string literal-like node.
if (propertyName === 'template' && typescript_1.default.isStringLiteralLike(property.initializer)) {
// Need to add an offset of one to the start because the template quotes are
// not part of the template content.
// The `getText()` method gives us the original raw text.
// We could have used the `text` property, but if the template is defined as a backtick
// string then the `text` property contains a "cooked" version of the string. Such cooked
// strings will have converted CRLF characters to only LF. This messes up string
// replacements in template migrations.
// The raw text returned by `getText()` includes the enclosing quotes so we change the
// `content` and `start` values accordingly.
const content = property.initializer.getText().slice(1, -1);
const start = property.initializer.getStart() + 1;
this.resolvedTemplates.push({
filePath: sourceFileName,
container: node,
content,
inline: true,
start: start,
getCharacterAndLineOfPosition: pos => typescript_1.default.getLineAndCharacterOfPosition(sourceFile, pos + start)
});
}
if (propertyName === 'templateUrl' && typescript_1.default.isStringLiteralLike(property.initializer)) {
const templateDiskPath = (0, path_1.resolve)((0, path_1.dirname)(sourceFileName), property.initializer.text);
// TODO(devversion): Remove this when the TypeScript compiler host is fully virtual
// relying on the devkit virtual tree and not dealing with disk paths. This is blocked on
// providing common utilities for schematics/migrations, given this is done in the
// Angular CDK already:
// https://github.com/angular/components/blob/3704400ee67e0190c9783e16367587489c803ebc/src/cdk/schematics/update-tool/utils/virtual-host.ts.
const templateDevkitPath = (0, path_1.relative)(this._basePath, templateDiskPath);
// In case the template does not exist in the file system, skip this
// external template.
if (!this._tree.exists(templateDevkitPath)) {
return;
}
const fileContent = this._tree.read(templateDevkitPath).toString();
const lineStartsMap = (0, line_mappings_1.computeLineStartsMap)(fileContent);
this.resolvedTemplates.push({
filePath: templateDiskPath,
container: node,
content: fileContent,
inline: false,
start: 0,
getCharacterAndLineOfPosition: pos => (0, line_mappings_1.getLineAndCharacterFromPosition)(lineStartsMap, pos),
});
}
});
}
}
exports.NgComponentTemplateVisitor = NgComponentTemplateVisitor;
});
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ng_component_template.js","sourceRoot":"","sources":["../../../../../../../packages/core/schematics/utils/ng_component_template.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC 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 {Tree} from '@angular-devkit/schematics';\nimport {existsSync, readFileSync} from 'fs';\nimport {dirname, relative, resolve} from 'path';\nimport ts from 'typescript';\n\nimport {computeLineStartsMap, getLineAndCharacterFromPosition} from './line_mappings';\nimport {getAngularDecorators} from './ng_decorators';\nimport {unwrapExpression} from './typescript/functions';\nimport {getPropertyNameText} from './typescript/property_name';\n\nexport interface ResolvedTemplate {\n  /** Class declaration that contains this template. */\n  container: ts.ClassDeclaration;\n  /** File content of the given template. */\n  content: string;\n  /** Start offset of the template content (e.g. in the inline source file) */\n  start: number;\n  /** Whether the given template is inline or not. */\n  inline: boolean;\n  /** Path to the file that contains this template. */\n  filePath: string;\n  /**\n   * Gets the character and line of a given position index in the template.\n   * If the template is declared inline within a TypeScript source file, the line and\n   * character are based on the full source file content.\n   */\n  getCharacterAndLineOfPosition: (pos: number) => {\n    character: number, line: number\n  };\n}\n\n/**\n * Visitor that can be used to determine Angular templates referenced within given\n * TypeScript source files (inline templates or external referenced templates)\n */\nexport class NgComponentTemplateVisitor {\n  resolvedTemplates: ResolvedTemplate[] = [];\n\n  constructor(public typeChecker: ts.TypeChecker, private _basePath: string, private _tree: Tree) {}\n\n  visitNode(node: ts.Node) {\n    if (node.kind === ts.SyntaxKind.ClassDeclaration) {\n      this.visitClassDeclaration(node as ts.ClassDeclaration);\n    }\n\n    ts.forEachChild(node, n => this.visitNode(n));\n  }\n\n  private visitClassDeclaration(node: ts.ClassDeclaration) {\n    if (!node.decorators || !node.decorators.length) {\n      return;\n    }\n\n    const ngDecorators = getAngularDecorators(this.typeChecker, node.decorators);\n    const componentDecorator = ngDecorators.find(dec => dec.name === 'Component');\n\n    // In case no \"@Component\" decorator could be found on the current class, skip.\n    if (!componentDecorator) {\n      return;\n    }\n\n    const decoratorCall = componentDecorator.node.expression;\n\n    // In case the component decorator call is not valid, skip this class declaration.\n    if (decoratorCall.arguments.length !== 1) {\n      return;\n    }\n\n    const componentMetadata = unwrapExpression(decoratorCall.arguments[0]);\n\n    // Ensure that the component metadata is an object literal expression.\n    if (!ts.isObjectLiteralExpression(componentMetadata)) {\n      return;\n    }\n\n    const sourceFile = node.getSourceFile();\n    const sourceFileName = sourceFile.fileName;\n\n    // Walk through all component metadata properties and determine the referenced\n    // HTML templates (either external or inline)\n    componentMetadata.properties.forEach(property => {\n      if (!ts.isPropertyAssignment(property)) {\n        return;\n      }\n\n      const propertyName = getPropertyNameText(property.name);\n\n      // In case there is an inline template specified, ensure that the value is statically\n      // analyzable by checking if the initializer is a string literal-like node.\n      if (propertyName === 'template' && ts.isStringLiteralLike(property.initializer)) {\n        // Need to add an offset of one to the start because the template quotes are\n        // not part of the template content.\n        // The `getText()` method gives us the original raw text.\n        // We could have used the `text` property, but if the template is defined as a backtick\n        // string then the `text` property contains a \"cooked\" version of the string. Such cooked\n        // strings will have converted CRLF characters to only LF. This messes up string\n        // replacements in template migrations.\n        // The raw text returned by `getText()` includes the enclosing quotes so we change the\n        // `content` and `start` values accordingly.\n        const content = property.initializer.getText().slice(1, -1);\n        const start = property.initializer.getStart() + 1;\n        this.resolvedTemplates.push({\n          filePath: sourceFileName,\n          container: node,\n          content,\n          inline: true,\n          start: start,\n          getCharacterAndLineOfPosition: pos =>\n              ts.getLineAndCharacterOfPosition(sourceFile, pos + start)\n        });\n      }\n      if (propertyName === 'templateUrl' && ts.isStringLiteralLike(property.initializer)) {\n        const templateDiskPath = resolve(dirname(sourceFileName), property.initializer.text);\n        // TODO(devversion): Remove this when the TypeScript compiler host is fully virtual\n        // relying on the devkit virtual tree and not dealing with disk paths. This is blocked on\n        // providing common utilities for schematics/migrations, given this is done in the\n        // Angular CDK already:\n        // https://github.com/angular/components/blob/3704400ee67e0190c9783e16367587489c803ebc/src/cdk/schematics/update-tool/utils/virtual-host.ts.\n        const templateDevkitPath = relative(this._basePath, templateDiskPath);\n\n        // In case the template does not exist in the file system, skip this\n        // external template.\n        if (!this._tree.exists(templateDevkitPath)) {\n          return;\n        }\n\n        const fileContent = this._tree.read(templateDevkitPath)!.toString();\n        const lineStartsMap = computeLineStartsMap(fileContent);\n\n        this.resolvedTemplates.push({\n          filePath: templateDiskPath,\n          container: node,\n          content: fileContent,\n          inline: false,\n          start: 0,\n          getCharacterAndLineOfPosition: pos => getLineAndCharacterFromPosition(lineStartsMap, pos),\n        });\n      }\n    });\n  }\n}\n"]}