UNPKG

@angular/core

Version:

Angular - the core framework

119 lines • 17.6 kB
/** * @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/renderer-to-renderer2/util", ["require", "exports", "typescript"], factory); } })(function (require, exports) { "use strict"; Object.defineProperty(exports, "__esModule", { value: true }); const ts = require("typescript"); /** * Finds typed nodes (e.g. function parameters or class properties) that are referencing the old * `Renderer`, as well as calls to the `Renderer` methods. */ function findRendererReferences(sourceFile, typeChecker, rendererImport) { const typedNodes = new Set(); const methodCalls = new Set(); const forwardRefs = new Set(); const importSpecifier = findImportSpecifier(rendererImport.elements, 'Renderer'); const forwardRefImport = findCoreImport(sourceFile, 'forwardRef'); const forwardRefSpecifier = forwardRefImport ? findImportSpecifier(forwardRefImport.elements, 'forwardRef') : null; ts.forEachChild(sourceFile, function visitNode(node) { if ((ts.isParameter(node) || ts.isPropertyDeclaration(node)) && isReferenceToImport(typeChecker, node.name, importSpecifier)) { typedNodes.add(node); } else if (ts.isAsExpression(node) && isReferenceToImport(typeChecker, node.type, importSpecifier)) { typedNodes.add(node); } else if (ts.isCallExpression(node)) { if (ts.isPropertyAccessExpression(node.expression) && isReferenceToImport(typeChecker, node.expression.expression, importSpecifier)) { methodCalls.add(node); } else if ( // If we're dealing with a forwardRef that's returning a Renderer. forwardRefSpecifier && ts.isIdentifier(node.expression) && isReferenceToImport(typeChecker, node.expression, forwardRefSpecifier) && node.arguments.length) { const rendererIdentifier = findRendererIdentifierInForwardRef(typeChecker, node, importSpecifier); if (rendererIdentifier) { forwardRefs.add(rendererIdentifier); } } } ts.forEachChild(node, visitNode); }); return { typedNodes, methodCalls, forwardRefs }; } exports.findRendererReferences = findRendererReferences; /** Finds the import from @angular/core that has a symbol with a particular name. */ function findCoreImport(sourceFile, symbolName) { // Only look through the top-level imports. for (const node of sourceFile.statements) { if (!ts.isImportDeclaration(node) || !ts.isStringLiteral(node.moduleSpecifier) || node.moduleSpecifier.text !== '@angular/core') { continue; } const namedBindings = node.importClause && node.importClause.namedBindings; if (!namedBindings || !ts.isNamedImports(namedBindings)) { continue; } if (findImportSpecifier(namedBindings.elements, symbolName)) { return namedBindings; } } return null; } exports.findCoreImport = findCoreImport; /** Finds an import specifier with a particular name, accounting for aliases. */ function findImportSpecifier(elements, importName) { return elements.find(element => { const { name, propertyName } = element; return propertyName ? propertyName.text === importName : name.text === importName; }) || null; } exports.findImportSpecifier = findImportSpecifier; /** Checks whether a node is referring to an import spcifier. */ function isReferenceToImport(typeChecker, node, importSpecifier) { if (importSpecifier) { const nodeSymbol = typeChecker.getTypeAtLocation(node).getSymbol(); const importSymbol = typeChecker.getTypeAtLocation(importSpecifier).getSymbol(); return !!(nodeSymbol && importSymbol) && nodeSymbol.valueDeclaration === importSymbol.valueDeclaration; } return false; } /** Finds the identifier referring to the `Renderer` inside a `forwardRef` call expression. */ function findRendererIdentifierInForwardRef(typeChecker, node, rendererImport) { const firstArg = node.arguments[0]; if (ts.isArrowFunction(firstArg)) { // Check if the function is `forwardRef(() => Renderer)`. if (ts.isIdentifier(firstArg.body) && isReferenceToImport(typeChecker, firstArg.body, rendererImport)) { return firstArg.body; } else if (ts.isBlock(firstArg.body) && ts.isReturnStatement(firstArg.body.statements[0])) { // Otherwise check if the expression is `forwardRef(() => { return Renderer })`. const returnStatement = firstArg.body.statements[0]; if (returnStatement.expression && ts.isIdentifier(returnStatement.expression) && isReferenceToImport(typeChecker, returnStatement.expression, rendererImport)) { return returnStatement.expression; } } } return null; } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"util.js","sourceRoot":"","sources":["../../../../../../../../packages/core/schematics/migrations/renderer-to-renderer2/util.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\n/**\n * Finds typed nodes (e.g. function parameters or class properties) that are referencing the old\n * `Renderer`, as well as calls to the `Renderer` methods.\n */\nexport function findRendererReferences(\n    sourceFile: ts.SourceFile, typeChecker: ts.TypeChecker, rendererImport: ts.NamedImports) {\n  const typedNodes = new Set<ts.ParameterDeclaration|ts.PropertyDeclaration|ts.AsExpression>();\n  const methodCalls = new Set<ts.CallExpression>();\n  const forwardRefs = new Set<ts.Identifier>();\n  const importSpecifier = findImportSpecifier(rendererImport.elements, 'Renderer');\n  const forwardRefImport = findCoreImport(sourceFile, 'forwardRef');\n  const forwardRefSpecifier =\n      forwardRefImport ? findImportSpecifier(forwardRefImport.elements, 'forwardRef') : null;\n\n  ts.forEachChild(sourceFile, function visitNode(node: ts.Node) {\n    if ((ts.isParameter(node) || ts.isPropertyDeclaration(node)) &&\n        isReferenceToImport(typeChecker, node.name, importSpecifier)) {\n      typedNodes.add(node);\n    } else if (\n        ts.isAsExpression(node) && isReferenceToImport(typeChecker, node.type, importSpecifier)) {\n      typedNodes.add(node);\n    } else if (ts.isCallExpression(node)) {\n      if (ts.isPropertyAccessExpression(node.expression) &&\n          isReferenceToImport(typeChecker, node.expression.expression, importSpecifier)) {\n        methodCalls.add(node);\n      } else if (\n          // If we're dealing with a forwardRef that's returning a Renderer.\n          forwardRefSpecifier && ts.isIdentifier(node.expression) &&\n          isReferenceToImport(typeChecker, node.expression, forwardRefSpecifier) &&\n          node.arguments.length) {\n        const rendererIdentifier =\n            findRendererIdentifierInForwardRef(typeChecker, node, importSpecifier);\n        if (rendererIdentifier) {\n          forwardRefs.add(rendererIdentifier);\n        }\n      }\n    }\n\n    ts.forEachChild(node, visitNode);\n  });\n\n  return {typedNodes, methodCalls, forwardRefs};\n}\n\n/** Finds the import from @angular/core that has a symbol with a particular name. */\nexport function findCoreImport(sourceFile: ts.SourceFile, symbolName: string): ts.NamedImports|\n    null {\n  // Only look through the top-level imports.\n  for (const node of sourceFile.statements) {\n    if (!ts.isImportDeclaration(node) || !ts.isStringLiteral(node.moduleSpecifier) ||\n        node.moduleSpecifier.text !== '@angular/core') {\n      continue;\n    }\n\n    const namedBindings = node.importClause && node.importClause.namedBindings;\n\n    if (!namedBindings || !ts.isNamedImports(namedBindings)) {\n      continue;\n    }\n\n    if (findImportSpecifier(namedBindings.elements, symbolName)) {\n      return namedBindings;\n    }\n  }\n\n  return null;\n}\n\n/** Finds an import specifier with a particular name, accounting for aliases. */\nexport function findImportSpecifier(\n    elements: ts.NodeArray<ts.ImportSpecifier>, importName: string) {\n  return elements.find(element => {\n    const {name, propertyName} = element;\n    return propertyName ? propertyName.text === importName : name.text === importName;\n  }) ||\n      null;\n}\n\n/** Checks whether a node is referring to an import spcifier. */\nfunction isReferenceToImport(\n    typeChecker: ts.TypeChecker, node: ts.Node, importSpecifier: ts.ImportSpecifier|null): boolean {\n  if (importSpecifier) {\n    const nodeSymbol = typeChecker.getTypeAtLocation(node).getSymbol();\n    const importSymbol = typeChecker.getTypeAtLocation(importSpecifier).getSymbol();\n    return !!(nodeSymbol && importSymbol) &&\n        nodeSymbol.valueDeclaration === importSymbol.valueDeclaration;\n  }\n  return false;\n}\n\n/** Finds the identifier referring to the `Renderer` inside a `forwardRef` call expression. */\nfunction findRendererIdentifierInForwardRef(\n    typeChecker: ts.TypeChecker, node: ts.CallExpression,\n    rendererImport: ts.ImportSpecifier|null): ts.Identifier|null {\n  const firstArg = node.arguments[0];\n\n  if (ts.isArrowFunction(firstArg)) {\n    // Check if the function is `forwardRef(() => Renderer)`.\n    if (ts.isIdentifier(firstArg.body) &&\n        isReferenceToImport(typeChecker, firstArg.body, rendererImport)) {\n      return firstArg.body;\n    } else if (ts.isBlock(firstArg.body) && ts.isReturnStatement(firstArg.body.statements[0])) {\n      // Otherwise check if the expression is `forwardRef(() => { return Renderer })`.\n      const returnStatement = firstArg.body.statements[0] as ts.ReturnStatement;\n\n      if (returnStatement.expression && ts.isIdentifier(returnStatement.expression) &&\n          isReferenceToImport(typeChecker, returnStatement.expression, rendererImport)) {\n        return returnStatement.expression;\n      }\n    }\n  }\n\n  return null;\n}\n"]}