1 | "use strict";
|
---|
2 | /**
|
---|
3 | * @license
|
---|
4 | * Copyright Google LLC All Rights Reserved.
|
---|
5 | *
|
---|
6 | * Use of this source code is governed by an MIT-style license that can be
|
---|
7 | * found in the LICENSE file at https://angular.io/license
|
---|
8 | */
|
---|
9 | Object.defineProperty(exports, "__esModule", { value: true });
|
---|
10 | exports.migrateFileContent = void 0;
|
---|
11 | const config_1 = require("./config");
|
---|
12 | /** Possible pairs of comment characters in a Sass file. */
|
---|
13 | const commentPairs = new Map([['/*', '*/'], ['//', '\n']]);
|
---|
14 | /** Prefix for the placeholder that will be used to escape comments. */
|
---|
15 | const commentPlaceholderStart = '__<<ngThemingMigrationEscapedComment';
|
---|
16 | /** Suffix for the comment escape placeholder. */
|
---|
17 | const commentPlaceholderEnd = '>>__';
|
---|
18 | /**
|
---|
19 | * Migrates the content of a file to the new theming API. Note that this migration is using plain
|
---|
20 | * string manipulation, rather than the AST from PostCSS and the schematics string manipulation
|
---|
21 | * APIs, because it allows us to run it inside g3 and to avoid introducing new dependencies.
|
---|
22 | * @param fileContent Content of the file.
|
---|
23 | * @param oldMaterialPrefix Prefix with which the old Material imports should start.
|
---|
24 | * Has to end with a slash. E.g. if `@import '~@angular/material/theming'` should be
|
---|
25 | * matched, the prefix would be `~@angular/material/`.
|
---|
26 | * @param oldCdkPrefix Prefix with which the old CDK imports should start.
|
---|
27 | * Has to end with a slash. E.g. if `@import '~@angular/cdk/overlay'` should be
|
---|
28 | * matched, the prefix would be `~@angular/cdk/`.
|
---|
29 | * @param newMaterialImportPath New import to the Material theming API (e.g. `~@angular/material`).
|
---|
30 | * @param newCdkImportPath New import to the CDK Sass APIs (e.g. `~@angular/cdk`).
|
---|
31 | * @param excludedImports Pattern that can be used to exclude imports from being processed.
|
---|
32 | */
|
---|
33 | function migrateFileContent(fileContent, oldMaterialPrefix, oldCdkPrefix, newMaterialImportPath, newCdkImportPath, extraMaterialSymbols = {}, excludedImports) {
|
---|
34 | let { content, placeholders } = escapeComments(fileContent);
|
---|
35 | const materialResults = detectImports(content, oldMaterialPrefix, excludedImports);
|
---|
36 | const cdkResults = detectImports(content, oldCdkPrefix, excludedImports);
|
---|
37 | // Try to migrate the symbols even if there are no imports. This is used
|
---|
38 | // to cover the case where the Components symbols were used transitively.
|
---|
39 | content = migrateCdkSymbols(content, newCdkImportPath, placeholders, cdkResults);
|
---|
40 | content = migrateMaterialSymbols(content, newMaterialImportPath, materialResults, placeholders, extraMaterialSymbols);
|
---|
41 | content = replaceRemovedVariables(content, config_1.removedMaterialVariables);
|
---|
42 | // We can assume that the migration has taken care of any Components symbols that were
|
---|
43 | // imported transitively so we can always drop the old imports. We also assume that imports
|
---|
44 | // to the new entry points have been added already.
|
---|
45 | if (materialResults.imports.length) {
|
---|
46 | content = replaceRemovedVariables(content, config_1.unprefixedRemovedVariables);
|
---|
47 | content = removeStrings(content, materialResults.imports);
|
---|
48 | }
|
---|
49 | if (cdkResults.imports.length) {
|
---|
50 | content = removeStrings(content, cdkResults.imports);
|
---|
51 | }
|
---|
52 | return restoreComments(content, placeholders);
|
---|
53 | }
|
---|
54 | exports.migrateFileContent = migrateFileContent;
|
---|
55 | /**
|
---|
56 | * Counts the number of imports with a specific prefix and extracts their namespaces.
|
---|
57 | * @param content File content in which to look for imports.
|
---|
58 | * @param prefix Prefix that the imports should start with.
|
---|
59 | * @param excludedImports Pattern that can be used to exclude imports from being processed.
|
---|
60 | */
|
---|
61 | function detectImports(content, prefix, excludedImports) {
|
---|
62 | if (prefix[prefix.length - 1] !== '/') {
|
---|
63 | // Some of the logic further down makes assumptions about the import depth.
|
---|
64 | throw Error(`Prefix "${prefix}" has to end in a slash.`);
|
---|
65 | }
|
---|
66 | // List of `@use` namespaces from which Angular CDK/Material APIs may be referenced.
|
---|
67 | // Since we know that the library doesn't have any name collisions, we can treat all of these
|
---|
68 | // namespaces as equivalent.
|
---|
69 | const namespaces = [];
|
---|
70 | const imports = [];
|
---|
71 | const pattern = new RegExp(`@(import|use) +['"]${escapeRegExp(prefix)}.*['"].*;?\n`, 'g');
|
---|
72 | let match = null;
|
---|
73 | while (match = pattern.exec(content)) {
|
---|
74 | const [fullImport, type] = match;
|
---|
75 | if (excludedImports === null || excludedImports === void 0 ? void 0 : excludedImports.test(fullImport)) {
|
---|
76 | continue;
|
---|
77 | }
|
---|
78 | if (type === 'use') {
|
---|
79 | const namespace = extractNamespaceFromUseStatement(fullImport);
|
---|
80 | if (namespaces.indexOf(namespace) === -1) {
|
---|
81 | namespaces.push(namespace);
|
---|
82 | }
|
---|
83 | }
|
---|
84 | imports.push(fullImport);
|
---|
85 | }
|
---|
86 | return { imports, namespaces };
|
---|
87 | }
|
---|
88 | /** Migrates the Material symbols in a file. */
|
---|
89 | function migrateMaterialSymbols(content, importPath, detectedImports, commentPlaceholders, extraMaterialSymbols = {}) {
|
---|
90 | const initialContent = content;
|
---|
91 | const namespace = 'mat';
|
---|
92 | // Migrate the mixins.
|
---|
93 | const mixinsToUpdate = Object.assign(Object.assign({}, config_1.materialMixins), extraMaterialSymbols.mixins);
|
---|
94 | content = renameSymbols(content, mixinsToUpdate, detectedImports.namespaces, mixinKeyFormatter, getMixinValueFormatter(namespace));
|
---|
95 | // Migrate the functions.
|
---|
96 | const functionsToUpdate = Object.assign(Object.assign({}, config_1.materialFunctions), extraMaterialSymbols.functions);
|
---|
97 | content = renameSymbols(content, functionsToUpdate, detectedImports.namespaces, functionKeyFormatter, getFunctionValueFormatter(namespace));
|
---|
98 | // Migrate the variables.
|
---|
99 | const variablesToUpdate = Object.assign(Object.assign({}, config_1.materialVariables), extraMaterialSymbols.variables);
|
---|
100 | content = renameSymbols(content, variablesToUpdate, detectedImports.namespaces, variableKeyFormatter, getVariableValueFormatter(namespace));
|
---|
101 | if (content !== initialContent) {
|
---|
102 | // Add an import to the new API only if any of the APIs were being used.
|
---|
103 | content = insertUseStatement(content, importPath, namespace, commentPlaceholders);
|
---|
104 | }
|
---|
105 | return content;
|
---|
106 | }
|
---|
107 | /** Migrates the CDK symbols in a file. */
|
---|
108 | function migrateCdkSymbols(content, importPath, commentPlaceholders, detectedImports) {
|
---|
109 | const initialContent = content;
|
---|
110 | const namespace = 'cdk';
|
---|
111 | // Migrate the mixins.
|
---|
112 | content = renameSymbols(content, config_1.cdkMixins, detectedImports.namespaces, mixinKeyFormatter, getMixinValueFormatter(namespace));
|
---|
113 | // Migrate the variables.
|
---|
114 | content = renameSymbols(content, config_1.cdkVariables, detectedImports.namespaces, variableKeyFormatter, getVariableValueFormatter(namespace));
|
---|
115 | // Previously the CDK symbols were exposed through `material/theming`, but now we have a
|
---|
116 | // dedicated entrypoint for the CDK. Only add an import for it if any of the symbols are used.
|
---|
117 | if (content !== initialContent) {
|
---|
118 | content = insertUseStatement(content, importPath, namespace, commentPlaceholders);
|
---|
119 | }
|
---|
120 | return content;
|
---|
121 | }
|
---|
122 | /**
|
---|
123 | * Renames all Sass symbols in a file based on a pre-defined mapping.
|
---|
124 | * @param content Content of a file to be migrated.
|
---|
125 | * @param mapping Mapping between symbol names and their replacements.
|
---|
126 | * @param namespaces Names to iterate over and pass to getKeyPattern.
|
---|
127 | * @param getKeyPattern Function used to turn each of the keys into a regex.
|
---|
128 | * @param formatValue Formats the value that will replace any matches of the pattern returned by
|
---|
129 | * `getKeyPattern`.
|
---|
130 | */
|
---|
131 | function renameSymbols(content, mapping, namespaces, getKeyPattern, formatValue) {
|
---|
132 | // The null at the end is so that we make one last pass to cover non-namespaced symbols.
|
---|
133 | [...namespaces.slice(), null].forEach(namespace => {
|
---|
134 | Object.keys(mapping).forEach(key => {
|
---|
135 | const pattern = getKeyPattern(namespace, key);
|
---|
136 | // Sanity check since non-global regexes will only replace the first match.
|
---|
137 | if (pattern.flags.indexOf('g') === -1) {
|
---|
138 | throw Error('Replacement pattern must be global.');
|
---|
139 | }
|
---|
140 | content = content.replace(pattern, formatValue(mapping[key]));
|
---|
141 | });
|
---|
142 | });
|
---|
143 | return content;
|
---|
144 | }
|
---|
145 | /** Inserts an `@use` statement in a string. */
|
---|
146 | function insertUseStatement(content, importPath, namespace, commentPlaceholders) {
|
---|
147 | // If the content already has the `@use` import, we don't need to add anything.
|
---|
148 | if (new RegExp(`@use +['"]${importPath}['"]`, 'g').test(content)) {
|
---|
149 | return content;
|
---|
150 | }
|
---|
151 | // Sass will throw an error if an `@use` statement comes after another statement. The safest way
|
---|
152 | // to ensure that we conform to that requirement is by always inserting our imports at the top
|
---|
153 | // of the file. Detecting where the user's content starts is tricky, because there are many
|
---|
154 | // different kinds of syntax we'd have to account for. One approach is to find the first `@import`
|
---|
155 | // and insert before it, but the problem is that Sass allows `@import` to be placed anywhere.
|
---|
156 | let newImportIndex = 0;
|
---|
157 | // One special case is if the file starts with a license header which we want to preserve on top.
|
---|
158 | if (content.trim().startsWith(commentPlaceholderStart)) {
|
---|
159 | const commentStartIndex = content.indexOf(commentPlaceholderStart);
|
---|
160 | newImportIndex = content.indexOf(commentPlaceholderEnd, commentStartIndex + 1) +
|
---|
161 | commentPlaceholderEnd.length;
|
---|
162 | // If the leading comment doesn't end with a newline,
|
---|
163 | // we need to insert the import at the next line.
|
---|
164 | if (!commentPlaceholders[content.slice(commentStartIndex, newImportIndex)].endsWith('\n')) {
|
---|
165 | newImportIndex = Math.max(newImportIndex, content.indexOf('\n', newImportIndex) + 1);
|
---|
166 | }
|
---|
167 | }
|
---|
168 | return content.slice(0, newImportIndex) + `@use '${importPath}' as ${namespace};\n` +
|
---|
169 | content.slice(newImportIndex);
|
---|
170 | }
|
---|
171 | /** Formats a migration key as a Sass mixin invocation. */
|
---|
172 | function mixinKeyFormatter(namespace, name) {
|
---|
173 | // Note that adding a `(` at the end of the pattern would be more accurate, but mixin
|
---|
174 | // invocations don't necessarily have to include the parentheses. We could add `[(;]`,
|
---|
175 | // but then we won't know which character to include in the replacement string.
|
---|
176 | return new RegExp(`@include +${escapeRegExp((namespace ? namespace + '.' : '') + name)}`, 'g');
|
---|
177 | }
|
---|
178 | /** Returns a function that can be used to format a Sass mixin replacement. */
|
---|
179 | function getMixinValueFormatter(namespace) {
|
---|
180 | // Note that adding a `(` at the end of the pattern would be more accurate,
|
---|
181 | // but mixin invocations don't necessarily have to include the parentheses.
|
---|
182 | return name => `@include ${namespace}.${name}`;
|
---|
183 | }
|
---|
184 | /** Formats a migration key as a Sass function invocation. */
|
---|
185 | function functionKeyFormatter(namespace, name) {
|
---|
186 | const functionName = escapeRegExp(`${namespace ? namespace + '.' : ''}${name}(`);
|
---|
187 | return new RegExp(`(?<![-_a-zA-Z0-9])${functionName}`, 'g');
|
---|
188 | }
|
---|
189 | /** Returns a function that can be used to format a Sass function replacement. */
|
---|
190 | function getFunctionValueFormatter(namespace) {
|
---|
191 | return name => `${namespace}.${name}(`;
|
---|
192 | }
|
---|
193 | /** Formats a migration key as a Sass variable. */
|
---|
194 | function variableKeyFormatter(namespace, name) {
|
---|
195 | const variableName = escapeRegExp(`${namespace ? namespace + '.' : ''}$${name}`);
|
---|
196 | return new RegExp(`${variableName}(?![-_a-zA-Z0-9])`, 'g');
|
---|
197 | }
|
---|
198 | /** Returns a function that can be used to format a Sass variable replacement. */
|
---|
199 | function getVariableValueFormatter(namespace) {
|
---|
200 | return name => `${namespace}.$${name}`;
|
---|
201 | }
|
---|
202 | /** Escapes special regex characters in a string. */
|
---|
203 | function escapeRegExp(str) {
|
---|
204 | return str.replace(/([.*+?^=!:${}()|[\]\/\\])/g, '\\$1');
|
---|
205 | }
|
---|
206 | /** Removes all strings from another string. */
|
---|
207 | function removeStrings(content, toRemove) {
|
---|
208 | return toRemove
|
---|
209 | .reduce((accumulator, current) => accumulator.replace(current, ''), content)
|
---|
210 | .replace(/^\s+/, '');
|
---|
211 | }
|
---|
212 | /** Parses out the namespace from a Sass `@use` statement. */
|
---|
213 | function extractNamespaceFromUseStatement(fullImport) {
|
---|
214 | const closeQuoteIndex = Math.max(fullImport.lastIndexOf(`"`), fullImport.lastIndexOf(`'`));
|
---|
215 | if (closeQuoteIndex > -1) {
|
---|
216 | const asExpression = 'as ';
|
---|
217 | const asIndex = fullImport.indexOf(asExpression, closeQuoteIndex);
|
---|
218 | // If we found an ` as ` expression, we consider the rest of the text as the namespace.
|
---|
219 | if (asIndex > -1) {
|
---|
220 | return fullImport.slice(asIndex + asExpression.length).split(';')[0].trim();
|
---|
221 | }
|
---|
222 | // Otherwise the namespace is the name of the file that is being imported.
|
---|
223 | const lastSlashIndex = fullImport.lastIndexOf('/', closeQuoteIndex);
|
---|
224 | if (lastSlashIndex > -1) {
|
---|
225 | const fileName = fullImport.slice(lastSlashIndex + 1, closeQuoteIndex)
|
---|
226 | // Sass allows for leading underscores to be omitted and it technically supports .scss.
|
---|
227 | .replace(/^_|(\.import)?\.scss$|\.import$/g, '');
|
---|
228 | // Sass ignores `/index` and infers the namespace as the next segment in the path.
|
---|
229 | if (fileName === 'index') {
|
---|
230 | const nextSlashIndex = fullImport.lastIndexOf('/', lastSlashIndex - 1);
|
---|
231 | if (nextSlashIndex > -1) {
|
---|
232 | return fullImport.slice(nextSlashIndex + 1, lastSlashIndex);
|
---|
233 | }
|
---|
234 | }
|
---|
235 | else {
|
---|
236 | return fileName;
|
---|
237 | }
|
---|
238 | }
|
---|
239 | }
|
---|
240 | throw Error(`Could not extract namespace from import "${fullImport}".`);
|
---|
241 | }
|
---|
242 | /**
|
---|
243 | * Replaces variables that have been removed with their values.
|
---|
244 | * @param content Content of the file to be migrated.
|
---|
245 | * @param variables Mapping between variable names and their values.
|
---|
246 | */
|
---|
247 | function replaceRemovedVariables(content, variables) {
|
---|
248 | Object.keys(variables).forEach(variableName => {
|
---|
249 | // Note that the pattern uses a negative lookahead to exclude
|
---|
250 | // variable assignments, because they can't be migrated.
|
---|
251 | const regex = new RegExp(`\\$${escapeRegExp(variableName)}(?!\\s+:|:)`, 'g');
|
---|
252 | content = content.replace(regex, variables[variableName]);
|
---|
253 | });
|
---|
254 | return content;
|
---|
255 | }
|
---|
256 | /**
|
---|
257 | * Replaces all of the comments in a Sass file with placeholders and
|
---|
258 | * returns the list of placeholders so they can be restored later.
|
---|
259 | */
|
---|
260 | function escapeComments(content) {
|
---|
261 | const placeholders = {};
|
---|
262 | let commentCounter = 0;
|
---|
263 | let [openIndex, closeIndex] = findComment(content);
|
---|
264 | while (openIndex > -1 && closeIndex > -1) {
|
---|
265 | const placeholder = commentPlaceholderStart + (commentCounter++) + commentPlaceholderEnd;
|
---|
266 | placeholders[placeholder] = content.slice(openIndex, closeIndex);
|
---|
267 | content = content.slice(0, openIndex) + placeholder + content.slice(closeIndex);
|
---|
268 | [openIndex, closeIndex] = findComment(content);
|
---|
269 | }
|
---|
270 | return { content, placeholders };
|
---|
271 | }
|
---|
272 | /** Finds the start and end index of a comment in a file. */
|
---|
273 | function findComment(content) {
|
---|
274 | // Add an extra new line at the end so that we can correctly capture single-line comments
|
---|
275 | // at the end of the file. It doesn't really matter that the end index will be out of bounds,
|
---|
276 | // because `String.prototype.slice` will clamp it to the string length.
|
---|
277 | content += '\n';
|
---|
278 | for (const [open, close] of commentPairs.entries()) {
|
---|
279 | const openIndex = content.indexOf(open);
|
---|
280 | if (openIndex > -1) {
|
---|
281 | const closeIndex = content.indexOf(close, openIndex + 1);
|
---|
282 | return closeIndex > -1 ? [openIndex, closeIndex + close.length] : [-1, -1];
|
---|
283 | }
|
---|
284 | }
|
---|
285 | return [-1, -1];
|
---|
286 | }
|
---|
287 | /** Restores the comments that have been escaped by `escapeComments`. */
|
---|
288 | function restoreComments(content, placeholders) {
|
---|
289 | Object.keys(placeholders).forEach(key => content = content.replace(key, placeholders[key]));
|
---|
290 | return content;
|
---|
291 | }
|
---|
292 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"migration.js","sourceRoot":"","sources":["../../../../../../../../../src/material/schematics/ng-update/migrations/theming-api-v12/migration.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 {\n  materialMixins,\n  materialFunctions,\n  materialVariables,\n  cdkMixins,\n  cdkVariables,\n  removedMaterialVariables,\n  unprefixedRemovedVariables\n} from './config';\n\n/** The result of a search for imports and namespaces in a file. */\ninterface DetectImportResult {\n  imports: string[];\n  namespaces: string[];\n}\n\n/** Addition mixin and function names that can be updated when invoking migration directly. */\ninterface ExtraSymbols {\n  mixins?: Record<string, string>;\n  functions?: Record<string, string>;\n  variables?: Record<string, string>;\n}\n\n/** Possible pairs of comment characters in a Sass file. */\nconst commentPairs = new Map<string, string>([['/*', '*/'], ['//', '\\n']]);\n\n/** Prefix for the placeholder that will be used to escape comments. */\nconst commentPlaceholderStart = '__<<ngThemingMigrationEscapedComment';\n\n/** Suffix for the comment escape placeholder. */\nconst commentPlaceholderEnd = '>>__';\n\n/**\n * Migrates the content of a file to the new theming API. Note that this migration is using plain\n * string manipulation, rather than the AST from PostCSS and the schematics string manipulation\n * APIs, because it allows us to run it inside g3 and to avoid introducing new dependencies.\n * @param fileContent Content of the file.\n * @param oldMaterialPrefix Prefix with which the old Material imports should start.\n *   Has to end with a slash. E.g. if `@import '~@angular/material/theming'` should be\n *   matched, the prefix would be `~@angular/material/`.\n * @param oldCdkPrefix Prefix with which the old CDK imports should start.\n *   Has to end with a slash. E.g. if `@import '~@angular/cdk/overlay'` should be\n *   matched, the prefix would be `~@angular/cdk/`.\n * @param newMaterialImportPath New import to the Material theming API (e.g. `~@angular/material`).\n * @param newCdkImportPath New import to the CDK Sass APIs (e.g. `~@angular/cdk`).\n * @param excludedImports Pattern that can be used to exclude imports from being processed.\n */\nexport function migrateFileContent(fileContent: string,\n                                   oldMaterialPrefix: string,\n                                   oldCdkPrefix: string,\n                                   newMaterialImportPath: string,\n                                   newCdkImportPath: string,\n                                   extraMaterialSymbols: ExtraSymbols = {},\n                                   excludedImports?: RegExp): string {\n  let {content, placeholders} = escapeComments(fileContent);\n  const materialResults = detectImports(content, oldMaterialPrefix, excludedImports);\n  const cdkResults = detectImports(content, oldCdkPrefix, excludedImports);\n\n  // Try to migrate the symbols even if there are no imports. This is used\n  // to cover the case where the Components symbols were used transitively.\n  content = migrateCdkSymbols(content, newCdkImportPath, placeholders, cdkResults);\n  content = migrateMaterialSymbols(\n      content, newMaterialImportPath, materialResults, placeholders, extraMaterialSymbols);\n  content = replaceRemovedVariables(content, removedMaterialVariables);\n\n  // We can assume that the migration has taken care of any Components symbols that were\n  // imported transitively so we can always drop the old imports. We also assume that imports\n  // to the new entry points have been added already.\n  if (materialResults.imports.length) {\n    content = replaceRemovedVariables(content, unprefixedRemovedVariables);\n    content = removeStrings(content, materialResults.imports);\n  }\n\n  if (cdkResults.imports.length) {\n    content = removeStrings(content, cdkResults.imports);\n  }\n\n  return restoreComments(content, placeholders);\n}\n\n/**\n * Counts the number of imports with a specific prefix and extracts their namespaces.\n * @param content File content in which to look for imports.\n * @param prefix Prefix that the imports should start with.\n * @param excludedImports Pattern that can be used to exclude imports from being processed.\n */\nfunction detectImports(content: string, prefix: string,\n                       excludedImports?: RegExp): DetectImportResult {\n  if (prefix[prefix.length - 1] !== '/') {\n    // Some of the logic further down makes assumptions about the import depth.\n    throw Error(`Prefix \"${prefix}\" has to end in a slash.`);\n  }\n\n  // List of `@use` namespaces from which Angular CDK/Material APIs may be referenced.\n  // Since we know that the library doesn't have any name collisions, we can treat all of these\n  // namespaces as equivalent.\n  const namespaces: string[] = [];\n  const imports: string[] = [];\n  const pattern = new RegExp(`@(import|use) +['\"]${escapeRegExp(prefix)}.*['\"].*;?\\n`, 'g');\n  let match: RegExpExecArray | null = null;\n\n  while (match = pattern.exec(content)) {\n    const [fullImport, type] = match;\n\n    if (excludedImports?.test(fullImport)) {\n      continue;\n    }\n\n    if (type === 'use') {\n      const namespace = extractNamespaceFromUseStatement(fullImport);\n\n      if (namespaces.indexOf(namespace) === -1) {\n        namespaces.push(namespace);\n      }\n    }\n\n    imports.push(fullImport);\n  }\n\n  return {imports, namespaces};\n}\n\n/** Migrates the Material symbols in a file. */\nfunction migrateMaterialSymbols(content: string, importPath: string,\n                                detectedImports: DetectImportResult,\n                                commentPlaceholders: Record<string, string>,\n                                extraMaterialSymbols: ExtraSymbols = {}): string {\n  const initialContent = content;\n  const namespace = 'mat';\n\n  // Migrate the mixins.\n  const mixinsToUpdate = {...materialMixins, ...extraMaterialSymbols.mixins};\n  content = renameSymbols(content, mixinsToUpdate, detectedImports.namespaces, mixinKeyFormatter,\n    getMixinValueFormatter(namespace));\n\n  // Migrate the functions.\n  const functionsToUpdate = {...materialFunctions, ...extraMaterialSymbols.functions};\n  content = renameSymbols(content, functionsToUpdate, detectedImports.namespaces,\n    functionKeyFormatter, getFunctionValueFormatter(namespace));\n\n  // Migrate the variables.\n  const variablesToUpdate = {...materialVariables, ...extraMaterialSymbols.variables};\n  content = renameSymbols(content, variablesToUpdate, detectedImports.namespaces,\n    variableKeyFormatter, getVariableValueFormatter(namespace));\n\n  if (content !== initialContent) {\n    // Add an import to the new API only if any of the APIs were being used.\n    content = insertUseStatement(content, importPath, namespace, commentPlaceholders);\n  }\n\n  return content;\n}\n\n/** Migrates the CDK symbols in a file. */\nfunction migrateCdkSymbols(content: string, importPath: string,\n                           commentPlaceholders: Record<string, string>,\n                           detectedImports: DetectImportResult): string {\n  const initialContent = content;\n  const namespace = 'cdk';\n\n  // Migrate the mixins.\n  content = renameSymbols(content, cdkMixins, detectedImports.namespaces, mixinKeyFormatter,\n    getMixinValueFormatter(namespace));\n\n  // Migrate the variables.\n  content = renameSymbols(content, cdkVariables, detectedImports.namespaces, variableKeyFormatter,\n    getVariableValueFormatter(namespace));\n\n  // Previously the CDK symbols were exposed through `material/theming`, but now we have a\n  // dedicated entrypoint for the CDK. Only add an import for it if any of the symbols are used.\n  if (content !== initialContent) {\n    content = insertUseStatement(content, importPath, namespace, commentPlaceholders);\n  }\n\n  return content;\n}\n\n/**\n * Renames all Sass symbols in a file based on a pre-defined mapping.\n * @param content Content of a file to be migrated.\n * @param mapping Mapping between symbol names and their replacements.\n * @param namespaces Names to iterate over and pass to getKeyPattern.\n * @param getKeyPattern Function used to turn each of the keys into a regex.\n * @param formatValue Formats the value that will replace any matches of the pattern returned by\n *  `getKeyPattern`.\n */\nfunction renameSymbols(content: string,\n                       mapping: Record<string, string>,\n                       namespaces: string[],\n                       getKeyPattern: (namespace: string|null, key: string) => RegExp,\n                       formatValue: (key: string) => string): string {\n  // The null at the end is so that we make one last pass to cover non-namespaced symbols.\n  [...namespaces.slice(), null].forEach(namespace => {\n    Object.keys(mapping).forEach(key => {\n      const pattern = getKeyPattern(namespace, key);\n\n      // Sanity check since non-global regexes will only replace the first match.\n      if (pattern.flags.indexOf('g') === -1) {\n        throw Error('Replacement pattern must be global.');\n      }\n\n      content = content.replace(pattern, formatValue(mapping[key]));\n    });\n  });\n\n  return content;\n}\n\n/** Inserts an `@use` statement in a string. */\nfunction insertUseStatement(content: string, importPath: string, namespace: string,\n                            commentPlaceholders: Record<string, string>): string {\n  // If the content already has the `@use` import, we don't need to add anything.\n  if (new RegExp(`@use +['\"]${importPath}['\"]`, 'g').test(content)) {\n    return content;\n  }\n\n  // Sass will throw an error if an `@use` statement comes after another statement. The safest way\n  // to ensure that we conform to that requirement is by always inserting our imports at the top\n  // of the file. Detecting where the user's content starts is tricky, because there are many\n  // different kinds of syntax we'd have to account for. One approach is to find the first `@import`\n  // and insert before it, but the problem is that Sass allows `@import` to be placed anywhere.\n  let newImportIndex = 0;\n\n  // One special case is if the file starts with a license header which we want to preserve on top.\n  if (content.trim().startsWith(commentPlaceholderStart)) {\n    const commentStartIndex = content.indexOf(commentPlaceholderStart);\n    newImportIndex = content.indexOf(commentPlaceholderEnd, commentStartIndex + 1) +\n        commentPlaceholderEnd.length;\n    // If the leading comment doesn't end with a newline,\n    // we need to insert the import at the next line.\n    if (!commentPlaceholders[content.slice(commentStartIndex, newImportIndex)].endsWith('\\n')) {\n      newImportIndex = Math.max(newImportIndex, content.indexOf('\\n', newImportIndex) + 1);\n    }\n  }\n\n  return content.slice(0, newImportIndex) + `@use '${importPath}' as ${namespace};\\n` +\n         content.slice(newImportIndex);\n}\n\n/** Formats a migration key as a Sass mixin invocation. */\nfunction mixinKeyFormatter(namespace: string|null, name: string): RegExp {\n  // Note that adding a `(` at the end of the pattern would be more accurate, but mixin\n  // invocations don't necessarily have to include the parentheses. We could add `[(;]`,\n  // but then we won't know which character to include in the replacement string.\n  return new RegExp(`@include +${escapeRegExp((namespace ? namespace + '.' : '') + name)}`, 'g');\n}\n\n/** Returns a function that can be used to format a Sass mixin replacement. */\nfunction getMixinValueFormatter(namespace: string): (name: string) => string {\n  // Note that adding a `(` at the end of the pattern would be more accurate,\n  // but mixin invocations don't necessarily have to include the parentheses.\n  return name => `@include ${namespace}.${name}`;\n}\n\n/** Formats a migration key as a Sass function invocation. */\nfunction functionKeyFormatter(namespace: string|null, name: string): RegExp {\n  const functionName = escapeRegExp(`${namespace ? namespace + '.' : ''}${name}(`);\n  return new RegExp(`(?<![-_a-zA-Z0-9])${functionName}`, 'g');\n}\n\n/** Returns a function that can be used to format a Sass function replacement. */\nfunction getFunctionValueFormatter(namespace: string): (name: string) => string {\n  return name => `${namespace}.${name}(`;\n}\n\n/** Formats a migration key as a Sass variable. */\nfunction variableKeyFormatter(namespace: string|null, name: string): RegExp {\n  const variableName = escapeRegExp(`${namespace ? namespace + '.' : ''}$${name}`);\n  return new RegExp(`${variableName}(?![-_a-zA-Z0-9])`, 'g');\n}\n\n/** Returns a function that can be used to format a Sass variable replacement. */\nfunction getVariableValueFormatter(namespace: string): (name: string) => string {\n  return name => `${namespace}.$${name}`;\n}\n\n/** Escapes special regex characters in a string. */\nfunction escapeRegExp(str: string): string {\n  return str.replace(/([.*+?^=!:${}()|[\\]\\/\\\\])/g, '\\\\$1');\n}\n\n/** Removes all strings from another string. */\nfunction removeStrings(content: string, toRemove: string[]): string {\n  return toRemove\n    .reduce((accumulator, current) => accumulator.replace(current, ''), content)\n    .replace(/^\\s+/, '');\n}\n\n/** Parses out the namespace from a Sass `@use` statement. */\nfunction extractNamespaceFromUseStatement(fullImport: string): string {\n  const closeQuoteIndex = Math.max(fullImport.lastIndexOf(`\"`), fullImport.lastIndexOf(`'`));\n\n  if (closeQuoteIndex > -1) {\n    const asExpression = 'as ';\n    const asIndex = fullImport.indexOf(asExpression, closeQuoteIndex);\n\n    // If we found an ` as ` expression, we consider the rest of the text as the namespace.\n    if (asIndex > -1) {\n      return fullImport.slice(asIndex + asExpression.length).split(';')[0].trim();\n    }\n\n    // Otherwise the namespace is the name of the file that is being imported.\n    const lastSlashIndex = fullImport.lastIndexOf('/', closeQuoteIndex);\n\n    if (lastSlashIndex > -1) {\n      const fileName = fullImport.slice(lastSlashIndex + 1, closeQuoteIndex)\n        // Sass allows for leading underscores to be omitted and it technically supports .scss.\n        .replace(/^_|(\\.import)?\\.scss$|\\.import$/g, '');\n\n      // Sass ignores `/index` and infers the namespace as the next segment in the path.\n      if (fileName === 'index') {\n        const nextSlashIndex = fullImport.lastIndexOf('/', lastSlashIndex - 1);\n\n        if (nextSlashIndex > -1) {\n          return fullImport.slice(nextSlashIndex + 1, lastSlashIndex);\n        }\n      } else {\n        return fileName;\n      }\n    }\n  }\n\n  throw Error(`Could not extract namespace from import \"${fullImport}\".`);\n}\n\n/**\n * Replaces variables that have been removed with their values.\n * @param content Content of the file to be migrated.\n * @param variables Mapping between variable names and their values.\n */\nfunction replaceRemovedVariables(content: string, variables: Record<string, string>): string {\n  Object.keys(variables).forEach(variableName => {\n    // Note that the pattern uses a negative lookahead to exclude\n    // variable assignments, because they can't be migrated.\n    const regex = new RegExp(`\\\\$${escapeRegExp(variableName)}(?!\\\\s+:|:)`, 'g');\n    content = content.replace(regex, variables[variableName]);\n  });\n\n  return content;\n}\n\n\n/**\n * Replaces all of the comments in a Sass file with placeholders and\n * returns the list of placeholders so they can be restored later.\n */\nfunction escapeComments(content: string): {content: string, placeholders: Record<string, string>} {\n  const placeholders: Record<string, string> = {};\n  let commentCounter = 0;\n  let [openIndex, closeIndex] = findComment(content);\n\n  while (openIndex > -1 && closeIndex > -1) {\n    const placeholder = commentPlaceholderStart + (commentCounter++) + commentPlaceholderEnd;\n    placeholders[placeholder] = content.slice(openIndex, closeIndex);\n    content = content.slice(0, openIndex) + placeholder + content.slice(closeIndex);\n    [openIndex, closeIndex] = findComment(content);\n  }\n\n  return {content, placeholders};\n}\n\n/** Finds the start and end index of a comment in a file. */\nfunction findComment(content: string): [openIndex: number, closeIndex: number] {\n  // Add an extra new line at the end so that we can correctly capture single-line comments\n  // at the end of the file. It doesn't really matter that the end index will be out of bounds,\n  // because `String.prototype.slice` will clamp it to the string length.\n  content += '\\n';\n\n  for (const [open, close] of commentPairs.entries()) {\n    const openIndex = content.indexOf(open);\n\n    if (openIndex > -1) {\n      const closeIndex = content.indexOf(close, openIndex + 1);\n      return closeIndex > -1 ? [openIndex, closeIndex + close.length] : [-1, -1];\n    }\n  }\n\n  return [-1, -1];\n}\n\n/** Restores the comments that have been escaped by `escapeComments`. */\nfunction restoreComments(content: string, placeholders: Record<string, string>): string {\n  Object.keys(placeholders).forEach(key => content = content.replace(key, placeholders[key]));\n  return content;\n}\n"]} |
---|