UNPKG

@mui/codemod

Version:
134 lines (124 loc) 5.04 kB
"use strict"; // This codemod attempts to fix the color imports breaking change introduced in // https://github.com/mui/material-ui/releases/tag/v1.0.0-alpha.21 // List of colors that are in the `common` module const commonColors = ['black', 'white', 'transparent', 'fullBlack', 'darkBlack', 'lightBlack', 'minBlack', 'faintBlack', 'fullWhite', 'darkWhite', 'lightWhite']; /** * Break down `colorIdentifier` into its `palette` and `hue` * e.g. lightBlue600 -> [lightBlue, 600] * @param {string} colorIdentifier */ function colorAccent(colorIdentifier) { const [, palette, hue] = colorIdentifier.match(/([A-Za-z]+?)(A?\d+)?$/); return { palette, hue }; } /** * Return color module path * @param {string} colorPalette */ function colorImportPath(colorPalette) { return commonColors.includes(colorPalette) ? 'common' : colorPalette; } /** * Replace all expressions that use identifier to access color palettes. * e.g. colors.amber100 -> colors.amber['100'] * @param {sting} identifier * @param {jscodeshift_api_object} j * @param {jscodeshift_ast_object} root */ function transformMemberExpressions(identifier, j, root) { // replace all expressions using `identifier` to access color palettes root.find(j.MemberExpression).forEach(path => { if (path.node.object.name !== identifier) { return; } const colorProperty = path.node.property.name; const { palette, hue } = colorAccent(colorProperty); const colorModuleName = colorImportPath(palette); const property = hue || palette; path.node.property = hue || colorModuleName === 'common' ? j.memberExpression(j.identifier(colorModuleName), /^[_|a-z]/i.test(property) ? j.identifier(property) : j.literal(property)) : j.identifier(colorModuleName); }); } /** * Replace all member imports. * e.g. import { red, blue } from 'material-ui/styles/colors' * @param {jscodeshift_api_object} j * @param {jscodeshift_ast_object} root * @param {string} importPath * @param {string} targetPath */ function transformMemberImports(j, root, importPath, targetPath) { // find member imports root.find(j.ImportDeclaration, { source: { value: importPath } }).forEach(importDeclaration => { const memberImportSpecifiers = importDeclaration.node.specifiers.filter(specifier => specifier.type === 'ImportSpecifier'); if (memberImportSpecifiers.length) { j(importDeclaration).replaceWith(() => { const importDeclarations = []; const assignmentExpressions = []; memberImportSpecifiers.forEach(memberSpecifier => { const { palette, hue } = colorAccent(memberSpecifier.imported.name); const colorModuleName = colorImportPath(palette); const modulePath = `${targetPath}/${colorModuleName}`; const colorIdentifier = j.identifier(colorModuleName); // import color module (if not already imported) if (!importDeclarations.map(p => p.source.value).includes(modulePath)) { importDeclarations.push(j.importDeclaration([j.importDefaultSpecifier(colorIdentifier)], j.literal(modulePath))); } // conditional assignment expression if (hue || colorModuleName === 'common') { const property = hue || palette; assignmentExpressions.push(j.variableDeclaration('const', [j.variableDeclarator(j.identifier(memberSpecifier.local.name), j.memberExpression(colorIdentifier, /^[_|a-z]/i.test(property) ? j.identifier(property) : j.literal(property)))])); } }); return importDeclarations.concat(assignmentExpressions); }); } }); } /** * Replace all namespace imports. * e.g. import * as colors from 'material-ui/styles/colors' * @param {jscodeshift_api_object} j * @param {jscodeshift_ast_object} root * @param {string} importPath * @param {string} targetPath */ function transformNamespaceImports(j, root, importPath, targetPath) { // find namespace imports root.find(j.ImportDeclaration, { source: { value: importPath } }).forEach(importDeclaration => { const namespaceImportSpecifier = importDeclaration.node.specifiers.find(specifier => specifier.type === 'ImportNamespaceSpecifier'); if (namespaceImportSpecifier) { j(importDeclaration).replaceWith(j.importDeclaration([j.importNamespaceSpecifier(j.identifier(namespaceImportSpecifier.local.name))], j.literal(targetPath))); transformMemberExpressions(namespaceImportSpecifier.local.name, j, root); } }); } module.exports = function transformer(fileInfo, api, options = {}) { const j = api.jscodeshift; const root = j(fileInfo.source); const importPath = options.importPath || 'material-ui/styles/colors'; const targetPath = options.targetPath || '@material-ui/core/colors'; // transforms transformMemberImports(j, root, importPath, targetPath); transformNamespaceImports(j, root, importPath, targetPath); return root.toSource({ quote: 'single' }); };