UNPKG

@angular/core

Version:

Angular - the core framework

127 lines • 20 kB
/** * @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"]}