123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209 |
- const topologicalSort = require("./topologicalSort");
- const matchImports = /^(.+?)\s+from\s+(?:"([^"]+)"|'([^']+)'|(global))$/;
- const icssImport = /^:import\((?:"([^"]+)"|'([^']+)')\)/;
- const VISITED_MARKER = 1;
- /**
- * :import('G') {}
- *
- * Rule
- * composes: ... from 'A'
- * composes: ... from 'B'
- * Rule
- * composes: ... from 'A'
- * composes: ... from 'A'
- * composes: ... from 'C'
- *
- * Results in:
- *
- * graph: {
- * G: [],
- * A: [],
- * B: ['A'],
- * C: ['A'],
- * }
- */
- function addImportToGraph(importId, parentId, graph, visited) {
- const siblingsId = parentId + "_" + "siblings";
- const visitedId = parentId + "_" + importId;
- if (visited[visitedId] !== VISITED_MARKER) {
- if (!Array.isArray(visited[siblingsId])) {
- visited[siblingsId] = [];
- }
- const siblings = visited[siblingsId];
- if (Array.isArray(graph[importId])) {
- graph[importId] = graph[importId].concat(siblings);
- } else {
- graph[importId] = siblings.slice();
- }
- visited[visitedId] = VISITED_MARKER;
- siblings.push(importId);
- }
- }
- module.exports = (options = {}) => {
- let importIndex = 0;
- const createImportedName =
- typeof options.createImportedName !== "function"
- ? (importName /*, path*/) =>
- `i__imported_${importName.replace(/\W/g, "_")}_${importIndex++}`
- : options.createImportedName;
- const failOnWrongOrder = options.failOnWrongOrder;
- return {
- postcssPlugin: "postcss-modules-extract-imports",
- prepare() {
- const graph = {};
- const visited = {};
- const existingImports = {};
- const importDecls = {};
- const imports = {};
- return {
- Once(root, postcss) {
- // Check the existing imports order and save refs
- root.walkRules((rule) => {
- const matches = icssImport.exec(rule.selector);
- if (matches) {
- const [, /*match*/ doubleQuotePath, singleQuotePath] = matches;
- const importPath = doubleQuotePath || singleQuotePath;
- addImportToGraph(importPath, "root", graph, visited);
- existingImports[importPath] = rule;
- }
- });
- root.walkDecls(/^composes$/, (declaration) => {
- const multiple = declaration.value.split(",");
- const values = [];
- multiple.forEach((value) => {
- const matches = value.trim().match(matchImports);
- if (!matches) {
- values.push(value);
- return;
- }
- let tmpSymbols;
- let [
- ,
- /*match*/ symbols,
- doubleQuotePath,
- singleQuotePath,
- global,
- ] = matches;
- if (global) {
- // Composing globals simply means changing these classes to wrap them in global(name)
- tmpSymbols = symbols.split(/\s+/).map((s) => `global(${s})`);
- } else {
- const importPath = doubleQuotePath || singleQuotePath;
- let parent = declaration.parent;
- let parentIndexes = "";
- while (parent.type !== "root") {
- parentIndexes =
- parent.parent.index(parent) + "_" + parentIndexes;
- parent = parent.parent;
- }
- const { selector } = declaration.parent;
- const parentRule = `_${parentIndexes}${selector}`;
- addImportToGraph(importPath, parentRule, graph, visited);
- importDecls[importPath] = declaration;
- imports[importPath] = imports[importPath] || {};
- tmpSymbols = symbols.split(/\s+/).map((s) => {
- if (!imports[importPath][s]) {
- imports[importPath][s] = createImportedName(s, importPath);
- }
- return imports[importPath][s];
- });
- }
- values.push(tmpSymbols.join(" "));
- });
- declaration.value = values.join(", ");
- });
- const importsOrder = topologicalSort(graph, failOnWrongOrder);
- if (importsOrder instanceof Error) {
- const importPath = importsOrder.nodes.find((importPath) =>
- // eslint-disable-next-line no-prototype-builtins
- importDecls.hasOwnProperty(importPath)
- );
- const decl = importDecls[importPath];
- throw decl.error(
- "Failed to resolve order of composed modules " +
- importsOrder.nodes
- .map((importPath) => "`" + importPath + "`")
- .join(", ") +
- ".",
- {
- plugin: "postcss-modules-extract-imports",
- word: "composes",
- }
- );
- }
- let lastImportRule;
- importsOrder.forEach((path) => {
- const importedSymbols = imports[path];
- let rule = existingImports[path];
- if (!rule && importedSymbols) {
- rule = postcss.rule({
- selector: `:import("${path}")`,
- raws: { after: "\n" },
- });
- if (lastImportRule) {
- root.insertAfter(lastImportRule, rule);
- } else {
- root.prepend(rule);
- }
- }
- lastImportRule = rule;
- if (!importedSymbols) {
- return;
- }
- Object.keys(importedSymbols).forEach((importedSymbol) => {
- rule.append(
- postcss.decl({
- value: importedSymbol,
- prop: importedSymbols[importedSymbol],
- raws: { before: "\n " },
- })
- );
- });
- });
- },
- };
- },
- };
- };
- module.exports.postcss = true;
|