81 lines
3.7 KiB
JavaScript
81 lines
3.7 KiB
JavaScript
|
"use strict";
|
||
|
/**
|
||
|
* @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 };
|
||
|
};
|
||
|
Object.defineProperty(exports, "__esModule", { value: true });
|
||
|
exports.CodeBlock = void 0;
|
||
|
const typescript_1 = __importDefault(require("../../third_party/github.com/Microsoft/TypeScript/lib/typescript"));
|
||
|
const ast_utils_1 = require("../ast-utils");
|
||
|
const change_1 = require("../change");
|
||
|
/** Counter used to generate unique IDs. */
|
||
|
let uniqueIdCounter = 0;
|
||
|
/**
|
||
|
* Utility class used to generate blocks of code that
|
||
|
* can be inserted by the devkit into a user's app.
|
||
|
*/
|
||
|
class CodeBlock {
|
||
|
_imports = new Map();
|
||
|
// Note: the methods here are defined as arrow function so that they can be destructured by
|
||
|
// consumers without losing their context. This makes the API more concise.
|
||
|
/** Function used to tag a code block in order to produce a `PendingCode` object. */
|
||
|
code = (strings, ...params) => {
|
||
|
return {
|
||
|
expression: strings.map((part, index) => part + (params[index] || '')).join(''),
|
||
|
imports: this._imports,
|
||
|
};
|
||
|
};
|
||
|
/**
|
||
|
* Used inside of a code block to mark external symbols and which module they should be imported
|
||
|
* from. When the code is inserted, the required import statements will be produced automatically.
|
||
|
* @param symbolName Name of the external symbol.
|
||
|
* @param moduleName Module from which the symbol should be imported.
|
||
|
*/
|
||
|
external = (symbolName, moduleName) => {
|
||
|
if (!this._imports.has(moduleName)) {
|
||
|
this._imports.set(moduleName, new Map());
|
||
|
}
|
||
|
const symbolsPerModule = this._imports.get(moduleName);
|
||
|
if (!symbolsPerModule.has(symbolName)) {
|
||
|
symbolsPerModule.set(symbolName, `@@__SCHEMATIC_PLACEHOLDER_${uniqueIdCounter++}__@@`);
|
||
|
}
|
||
|
return symbolsPerModule.get(symbolName);
|
||
|
};
|
||
|
/**
|
||
|
* Produces the necessary rules to transform a `PendingCode` object into valid code.
|
||
|
* @param initialCode Code pending transformed.
|
||
|
* @param filePath Path of the file in which the code will be inserted.
|
||
|
*/
|
||
|
static transformPendingCode(initialCode, filePath) {
|
||
|
const code = { ...initialCode };
|
||
|
const rules = [];
|
||
|
code.imports.forEach((symbols, moduleName) => {
|
||
|
symbols.forEach((placeholder, symbolName) => {
|
||
|
rules.push((tree) => {
|
||
|
const recorder = tree.beginUpdate(filePath);
|
||
|
const sourceFile = typescript_1.default.createSourceFile(filePath, tree.readText(filePath), typescript_1.default.ScriptTarget.Latest, true);
|
||
|
// Note that this could still technically clash if there's a top-level symbol called
|
||
|
// `${symbolName}_alias`, however this is unlikely. We can revisit this if it becomes
|
||
|
// a problem.
|
||
|
const alias = (0, ast_utils_1.hasTopLevelIdentifier)(sourceFile, symbolName, moduleName)
|
||
|
? symbolName + '_alias'
|
||
|
: undefined;
|
||
|
code.expression = code.expression.replace(new RegExp(placeholder, 'g'), alias || symbolName);
|
||
|
(0, change_1.applyToUpdateRecorder)(recorder, [
|
||
|
(0, ast_utils_1.insertImport)(sourceFile, filePath, symbolName, moduleName, false, alias),
|
||
|
]);
|
||
|
tree.commitUpdate(recorder);
|
||
|
});
|
||
|
});
|
||
|
});
|
||
|
return { code, rules };
|
||
|
}
|
||
|
}
|
||
|
exports.CodeBlock = CodeBlock;
|