@ngtools/webpack
Version:
Webpack plugin that AoT compiles your Angular components and modules.
231 lines • 33.6 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 __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
var desc = Object.getOwnPropertyDescriptor(m, k);
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
desc = { enumerable: true, get: function() { return m[k]; } };
}
Object.defineProperty(o, k2, desc);
}) : (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
o[k2] = m[k];
}));
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
Object.defineProperty(o, "default", { enumerable: true, value: v });
}) : function(o, v) {
o["default"] = v;
});
var __importStar = (this && this.__importStar) || function (mod) {
if (mod && mod.__esModule) return mod;
var result = {};
if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
__setModuleDefault(result, mod);
return result;
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.getResourceUrl = exports.replaceResources = exports.NG_COMPONENT_RESOURCE_QUERY = void 0;
const ts = __importStar(require("typescript"));
const inline_resource_1 = require("../loaders/inline-resource");
exports.NG_COMPONENT_RESOURCE_QUERY = 'ngResource';
function replaceResources(shouldTransform, getTypeChecker, inlineStyleFileExtension) {
return (context) => {
const typeChecker = getTypeChecker();
const resourceImportDeclarations = [];
const moduleKind = context.getCompilerOptions().module;
const nodeFactory = context.factory;
const visitNode = (node) => {
var _a;
if (ts.isClassDeclaration(node)) {
const decorators = ts.getDecorators(node);
if (!decorators || decorators.length === 0) {
return node;
}
return nodeFactory.updateClassDeclaration(node, [
...decorators.map((current) => visitDecorator(nodeFactory, current, typeChecker, resourceImportDeclarations, moduleKind, inlineStyleFileExtension)),
...((_a = ts.getModifiers(node)) !== null && _a !== void 0 ? _a : []),
], node.name, node.typeParameters, node.heritageClauses, node.members);
}
return ts.visitEachChild(node, visitNode, context);
};
return (sourceFile) => {
if (!shouldTransform(sourceFile.fileName)) {
return sourceFile;
}
const updatedSourceFile = ts.visitNode(sourceFile, visitNode);
if (resourceImportDeclarations.length) {
// Add resource imports
return context.factory.updateSourceFile(updatedSourceFile, ts.setTextRange(context.factory.createNodeArray([
...resourceImportDeclarations,
...updatedSourceFile.statements,
]), updatedSourceFile.statements));
}
return updatedSourceFile;
};
};
}
exports.replaceResources = replaceResources;
function visitDecorator(nodeFactory, node, typeChecker, resourceImportDeclarations, moduleKind, inlineStyleFileExtension) {
if (!isComponentDecorator(node, typeChecker)) {
return node;
}
if (!ts.isCallExpression(node.expression)) {
return node;
}
const decoratorFactory = node.expression;
const args = decoratorFactory.arguments;
if (args.length !== 1 || !ts.isObjectLiteralExpression(args[0])) {
// Unsupported component metadata
return node;
}
const objectExpression = args[0];
const styleReplacements = [];
// visit all properties
let properties = ts.visitNodes(objectExpression.properties, (node) => ts.isObjectLiteralElementLike(node)
? visitComponentMetadata(nodeFactory, node, styleReplacements, resourceImportDeclarations, moduleKind, inlineStyleFileExtension)
: node);
// replace properties with updated properties
if (styleReplacements.length > 0) {
const styleProperty = nodeFactory.createPropertyAssignment(nodeFactory.createIdentifier('styles'), nodeFactory.createArrayLiteralExpression(styleReplacements));
properties = nodeFactory.createNodeArray([...properties, styleProperty]);
}
return nodeFactory.updateDecorator(node, nodeFactory.updateCallExpression(decoratorFactory, decoratorFactory.expression, decoratorFactory.typeArguments, [nodeFactory.updateObjectLiteralExpression(objectExpression, properties)]));
}
function visitComponentMetadata(nodeFactory, node, styleReplacements, resourceImportDeclarations, moduleKind = ts.ModuleKind.ES2015, inlineStyleFileExtension) {
if (!ts.isPropertyAssignment(node) || ts.isComputedPropertyName(node.name)) {
return node;
}
const name = node.name.text;
switch (name) {
case 'moduleId':
return undefined;
case 'templateUrl':
const url = getResourceUrl(node.initializer);
if (!url) {
return node;
}
const importName = createResourceImport(nodeFactory, url, resourceImportDeclarations, moduleKind);
if (!importName) {
return node;
}
return nodeFactory.updatePropertyAssignment(node, nodeFactory.createIdentifier('template'), importName);
case 'styles':
case 'styleUrls':
if (!ts.isArrayLiteralExpression(node.initializer)) {
return node;
}
const isInlineStyle = name === 'styles';
const styles = ts.visitNodes(node.initializer.elements, (node) => {
if (!ts.isStringLiteral(node) && !ts.isNoSubstitutionTemplateLiteral(node)) {
return node;
}
let url;
if (isInlineStyle) {
if (inlineStyleFileExtension) {
const data = Buffer.from(node.text).toString('base64');
const containingFile = node.getSourceFile().fileName;
// app.component.ts.css?ngResource!=!@ngtools/webpack/src/loaders/inline-resource.js?data=...!app.component.ts
url =
`${containingFile}.${inlineStyleFileExtension}?${exports.NG_COMPONENT_RESOURCE_QUERY}` +
`!=!${inline_resource_1.InlineAngularResourceLoaderPath}?data=${encodeURIComponent(data)}!${containingFile}`;
}
else {
return nodeFactory.createStringLiteral(node.text);
}
}
else {
url = getResourceUrl(node);
}
if (!url) {
return node;
}
return createResourceImport(nodeFactory, url, resourceImportDeclarations, moduleKind);
});
// Styles should be placed first
if (isInlineStyle) {
styleReplacements.unshift(...styles);
}
else {
styleReplacements.push(...styles);
}
return undefined;
default:
return node;
}
}
function getResourceUrl(node) {
// only analyze strings
if (!ts.isStringLiteral(node) && !ts.isNoSubstitutionTemplateLiteral(node)) {
return null;
}
return `${/^\.?\.\//.test(node.text) ? '' : './'}${node.text}?${exports.NG_COMPONENT_RESOURCE_QUERY}`;
}
exports.getResourceUrl = getResourceUrl;
function isComponentDecorator(node, typeChecker) {
if (!ts.isDecorator(node)) {
return false;
}
const origin = getDecoratorOrigin(node, typeChecker);
if (origin && origin.module === '@angular/core' && origin.name === 'Component') {
return true;
}
return false;
}
function createResourceImport(nodeFactory, url, resourceImportDeclarations, moduleKind) {
const urlLiteral = nodeFactory.createStringLiteral(url);
if (moduleKind < ts.ModuleKind.ES2015) {
return nodeFactory.createCallExpression(nodeFactory.createIdentifier('require'), [], [urlLiteral]);
}
else {
const importName = nodeFactory.createIdentifier(`__NG_CLI_RESOURCE__${resourceImportDeclarations.length}`);
resourceImportDeclarations.push(nodeFactory.createImportDeclaration(undefined, nodeFactory.createImportClause(false, importName, undefined), urlLiteral));
return importName;
}
}
function getDecoratorOrigin(decorator, typeChecker) {
if (!ts.isCallExpression(decorator.expression)) {
return null;
}
let identifier;
let name = '';
if (ts.isPropertyAccessExpression(decorator.expression.expression)) {
identifier = decorator.expression.expression.expression;
name = decorator.expression.expression.name.text;
}
else if (ts.isIdentifier(decorator.expression.expression)) {
identifier = decorator.expression.expression;
}
else {
return null;
}
// NOTE: resolver.getReferencedImportDeclaration would work as well but is internal
const symbol = typeChecker.getSymbolAtLocation(identifier);
if (symbol && symbol.declarations && symbol.declarations.length > 0) {
const declaration = symbol.declarations[0];
let module;
if (ts.isImportSpecifier(declaration)) {
name = (declaration.propertyName || declaration.name).text;
module = declaration.parent.parent.parent.moduleSpecifier.text;
}
else if (ts.isNamespaceImport(declaration)) {
// Use the name from the decorator namespace property access
module = declaration.parent.parent.moduleSpecifier.text;
}
else if (ts.isImportClause(declaration)) {
name = declaration.name.text;
module = declaration.parent.moduleSpecifier.text;
}
else {
return null;
}
return { name, module };
}
return null;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"replace_resources.js","sourceRoot":"","sources":["../../../../../../../../packages/ngtools/webpack/src/transformers/replace_resources.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 * as ts from 'typescript';\nimport { InlineAngularResourceLoaderPath } from '../loaders/inline-resource';\n\nexport const NG_COMPONENT_RESOURCE_QUERY = 'ngResource';\n\nexport function replaceResources(\n  shouldTransform: (fileName: string) => boolean,\n  getTypeChecker: () => ts.TypeChecker,\n  inlineStyleFileExtension?: string,\n): ts.TransformerFactory<ts.SourceFile> {\n  return (context: ts.TransformationContext) => {\n    const typeChecker = getTypeChecker();\n    const resourceImportDeclarations: ts.ImportDeclaration[] = [];\n    const moduleKind = context.getCompilerOptions().module;\n    const nodeFactory = context.factory;\n\n    const visitNode: ts.Visitor = (node: ts.Node) => {\n      if (ts.isClassDeclaration(node)) {\n        const decorators = ts.getDecorators(node);\n\n        if (!decorators || decorators.length === 0) {\n          return node;\n        }\n\n        return nodeFactory.updateClassDeclaration(\n          node,\n          [\n            ...decorators.map((current) =>\n              visitDecorator(\n                nodeFactory,\n                current,\n                typeChecker,\n                resourceImportDeclarations,\n                moduleKind,\n                inlineStyleFileExtension,\n              ),\n            ),\n            ...(ts.getModifiers(node) ?? []),\n          ],\n          node.name,\n          node.typeParameters,\n          node.heritageClauses,\n          node.members,\n        );\n      }\n\n      return ts.visitEachChild(node, visitNode, context);\n    };\n\n    return (sourceFile: ts.SourceFile) => {\n      if (!shouldTransform(sourceFile.fileName)) {\n        return sourceFile;\n      }\n\n      const updatedSourceFile = ts.visitNode(sourceFile, visitNode);\n      if (resourceImportDeclarations.length) {\n        // Add resource imports\n        return context.factory.updateSourceFile(\n          updatedSourceFile,\n          ts.setTextRange(\n            context.factory.createNodeArray([\n              ...resourceImportDeclarations,\n              ...updatedSourceFile.statements,\n            ]),\n            updatedSourceFile.statements,\n          ),\n        );\n      }\n\n      return updatedSourceFile;\n    };\n  };\n}\n\nfunction visitDecorator(\n  nodeFactory: ts.NodeFactory,\n  node: ts.Decorator,\n  typeChecker: ts.TypeChecker,\n  resourceImportDeclarations: ts.ImportDeclaration[],\n  moduleKind?: ts.ModuleKind,\n  inlineStyleFileExtension?: string,\n): ts.Decorator {\n  if (!isComponentDecorator(node, typeChecker)) {\n    return node;\n  }\n\n  if (!ts.isCallExpression(node.expression)) {\n    return node;\n  }\n\n  const decoratorFactory = node.expression;\n  const args = decoratorFactory.arguments;\n  if (args.length !== 1 || !ts.isObjectLiteralExpression(args[0])) {\n    // Unsupported component metadata\n    return node;\n  }\n\n  const objectExpression = args[0] as ts.ObjectLiteralExpression;\n  const styleReplacements: ts.Expression[] = [];\n\n  // visit all properties\n  let properties = ts.visitNodes(objectExpression.properties, (node) =>\n    ts.isObjectLiteralElementLike(node)\n      ? visitComponentMetadata(\n          nodeFactory,\n          node,\n          styleReplacements,\n          resourceImportDeclarations,\n          moduleKind,\n          inlineStyleFileExtension,\n        )\n      : node,\n  );\n\n  // replace properties with updated properties\n  if (styleReplacements.length > 0) {\n    const styleProperty = nodeFactory.createPropertyAssignment(\n      nodeFactory.createIdentifier('styles'),\n      nodeFactory.createArrayLiteralExpression(styleReplacements),\n    );\n\n    properties = nodeFactory.createNodeArray([...properties, styleProperty]);\n  }\n\n  return nodeFactory.updateDecorator(\n    node,\n    nodeFactory.updateCallExpression(\n      decoratorFactory,\n      decoratorFactory.expression,\n      decoratorFactory.typeArguments,\n      [nodeFactory.updateObjectLiteralExpression(objectExpression, properties)],\n    ),\n  );\n}\n\nfunction visitComponentMetadata(\n  nodeFactory: ts.NodeFactory,\n  node: ts.ObjectLiteralElementLike,\n  styleReplacements: ts.Expression[],\n  resourceImportDeclarations: ts.ImportDeclaration[],\n  moduleKind: ts.ModuleKind = ts.ModuleKind.ES2015,\n  inlineStyleFileExtension?: string,\n): ts.ObjectLiteralElementLike | undefined {\n  if (!ts.isPropertyAssignment(node) || ts.isComputedPropertyName(node.name)) {\n    return node;\n  }\n\n  const name = node.name.text;\n  switch (name) {\n    case 'moduleId':\n      return undefined;\n\n    case 'templateUrl':\n      const url = getResourceUrl(node.initializer);\n      if (!url) {\n        return node;\n      }\n\n      const importName = createResourceImport(\n        nodeFactory,\n        url,\n        resourceImportDeclarations,\n        moduleKind,\n      );\n      if (!importName) {\n        return node;\n      }\n\n      return nodeFactory.updatePropertyAssignment(\n        node,\n        nodeFactory.createIdentifier('template'),\n        importName,\n      );\n    case 'styles':\n    case 'styleUrls':\n      if (!ts.isArrayLiteralExpression(node.initializer)) {\n        return node;\n      }\n\n      const isInlineStyle = name === 'styles';\n      const styles = ts.visitNodes(node.initializer.elements, (node) => {\n        if (!ts.isStringLiteral(node) && !ts.isNoSubstitutionTemplateLiteral(node)) {\n          return node;\n        }\n\n        let url;\n        if (isInlineStyle) {\n          if (inlineStyleFileExtension) {\n            const data = Buffer.from(node.text).toString('base64');\n            const containingFile = node.getSourceFile().fileName;\n            // app.component.ts.css?ngResource!=!@ngtools/webpack/src/loaders/inline-resource.js?data=...!app.component.ts\n            url =\n              `${containingFile}.${inlineStyleFileExtension}?${NG_COMPONENT_RESOURCE_QUERY}` +\n              `!=!${InlineAngularResourceLoaderPath}?data=${encodeURIComponent(\n                data,\n              )}!${containingFile}`;\n          } else {\n            return nodeFactory.createStringLiteral(node.text);\n          }\n        } else {\n          url = getResourceUrl(node);\n        }\n\n        if (!url) {\n          return node;\n        }\n\n        return createResourceImport(nodeFactory, url, resourceImportDeclarations, moduleKind);\n      });\n\n      // Styles should be placed first\n      if (isInlineStyle) {\n        styleReplacements.unshift(...styles);\n      } else {\n        styleReplacements.push(...styles);\n      }\n\n      return undefined;\n    default:\n      return node;\n  }\n}\n\nexport function getResourceUrl(node: ts.Node): string | null {\n  // only analyze strings\n  if (!ts.isStringLiteral(node) && !ts.isNoSubstitutionTemplateLiteral(node)) {\n    return null;\n  }\n\n  return `${/^\\.?\\.\\//.test(node.text) ? '' : './'}${node.text}?${NG_COMPONENT_RESOURCE_QUERY}`;\n}\n\nfunction isComponentDecorator(node: ts.Node, typeChecker: ts.TypeChecker): node is ts.Decorator {\n  if (!ts.isDecorator(node)) {\n    return false;\n  }\n\n  const origin = getDecoratorOrigin(node, typeChecker);\n  if (origin && origin.module === '@angular/core' && origin.name === 'Component') {\n    return true;\n  }\n\n  return false;\n}\n\nfunction createResourceImport(\n  nodeFactory: ts.NodeFactory,\n  url: string,\n  resourceImportDeclarations: ts.ImportDeclaration[],\n  moduleKind: ts.ModuleKind,\n): ts.Identifier | ts.Expression {\n  const urlLiteral = nodeFactory.createStringLiteral(url);\n\n  if (moduleKind < ts.ModuleKind.ES2015) {\n    return nodeFactory.createCallExpression(\n      nodeFactory.createIdentifier('require'),\n      [],\n      [urlLiteral],\n    );\n  } else {\n    const importName = nodeFactory.createIdentifier(\n      `__NG_CLI_RESOURCE__${resourceImportDeclarations.length}`,\n    );\n    resourceImportDeclarations.push(\n      nodeFactory.createImportDeclaration(\n        undefined,\n        nodeFactory.createImportClause(false, importName, undefined),\n        urlLiteral,\n      ),\n    );\n\n    return importName;\n  }\n}\n\ninterface DecoratorOrigin {\n  name: string;\n  module: string;\n}\n\nfunction getDecoratorOrigin(\n  decorator: ts.Decorator,\n  typeChecker: ts.TypeChecker,\n): DecoratorOrigin | null {\n  if (!ts.isCallExpression(decorator.expression)) {\n    return null;\n  }\n\n  let identifier: ts.Node;\n  let name = '';\n\n  if (ts.isPropertyAccessExpression(decorator.expression.expression)) {\n    identifier = decorator.expression.expression.expression;\n    name = decorator.expression.expression.name.text;\n  } else if (ts.isIdentifier(decorator.expression.expression)) {\n    identifier = decorator.expression.expression;\n  } else {\n    return null;\n  }\n\n  // NOTE: resolver.getReferencedImportDeclaration would work as well but is internal\n  const symbol = typeChecker.getSymbolAtLocation(identifier);\n  if (symbol && symbol.declarations && symbol.declarations.length > 0) {\n    const declaration = symbol.declarations[0];\n    let module: string;\n\n    if (ts.isImportSpecifier(declaration)) {\n      name = (declaration.propertyName || declaration.name).text;\n      module = (declaration.parent.parent.parent.moduleSpecifier as ts.Identifier).text;\n    } else if (ts.isNamespaceImport(declaration)) {\n      // Use the name from the decorator namespace property access\n      module = (declaration.parent.parent.moduleSpecifier as ts.Identifier).text;\n    } else if (ts.isImportClause(declaration)) {\n      name = (declaration.name as ts.Identifier).text;\n      module = (declaration.parent.moduleSpecifier as ts.Identifier).text;\n    } else {\n      return null;\n    }\n\n    return { name, module };\n  }\n\n  return null;\n}\n"]}
;