@nlabs/lex
Version:
142 lines (141 loc) • 16.4 kB
JavaScript
/**
* Copyright (c) 2025-Present, Nitrogen Labs, Inc.
* Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.
*
* PostCSS 8-compatible version of postcss-for plugin
* Original: https://github.com/antyakushev/postcss-for
*/ import postcssSimpleVars from 'postcss-simple-vars';
const postcssFor = (opts = {})=>{
const options = {
nested: opts.nested !== false
};
const iterStack = [];
const parentsHaveIterator = (rule, param)=>{
if (rule.parent === null) {
return false;
}
if (rule.parent.type === 'root') {
return false;
}
if (rule.parent.type !== 'atrule' || !rule.parent.params) {
return false;
}
const parentIterVar = rule.parent.params.split(/\s+/)[0];
if (!parentIterVar) {
return false;
}
if (parentIterVar === param) {
return true;
}
if (iterStack.indexOf(param) !== -1) {
return true;
}
return parentsHaveIterator(rule.parent, param);
};
const manageIterStack = (rule)=>{
if (rule.parent && rule.parent.type !== 'root') {
const parentIterVar = rule.parent.type === 'atrule' && rule.parent.params ? rule.parent.params.split(/\s+/)[0] : null;
if (parentIterVar && iterStack.indexOf(parentIterVar) === -1) {
iterStack.splice(0, iterStack.length);
} else if (parentIterVar) {
const parentIndex = iterStack.indexOf(parentIterVar);
if (parentIndex !== -1) {
iterStack.splice(parentIndex + 1, iterStack.length - parentIndex - 1);
}
}
} else {
iterStack.splice(0, iterStack.length);
}
const currentIterVar = rule.params.split(/\s+/)[0];
if (currentIterVar) {
iterStack.push(currentIterVar);
}
};
const checkNumber = (rule)=>(param)=>{
if (isNaN(Number(param)) || !param.match(/^-?\d+\.?\d*$/)) {
if (param.indexOf('$') !== -1) {
if (!parentsHaveIterator(rule, param)) {
throw rule.error('External variable (not from a parent for loop) cannot be used as a range parameter', {
plugin: 'postcss-for'
});
}
} else {
throw rule.error('Range parameter should be a number', {
plugin: 'postcss-for'
});
}
}
};
const checkParams = (rule, params)=>{
if (!params[0]?.startsWith('$') || params[1] !== 'from' || params[3] !== 'to' || params[5] && params[5] !== 'by') {
throw rule.error('Wrong loop syntax', {
plugin: 'postcss-for'
});
}
[
params[2],
params[4],
params[6] || '0'
].forEach(checkNumber(rule));
};
const unrollLoop = (rule)=>{
const params = rule.params.split(/\s+/);
checkParams(rule, params);
const iterator = params[0].slice(1);
const index = +params[2];
const top = +params[4];
const dir = top < index ? -1 : 1;
const by = +(params[6] || 1) * dir;
const value = {};
for(let i = index; i * dir <= top * dir; i = i + by){
const content = rule.clone();
value[iterator] = i;
const simpleVarsPlugin = postcssSimpleVars({
only: value
});
if (simpleVarsPlugin.prepare) {
const prepared = simpleVarsPlugin.prepare({});
if (prepared.Once) {
prepared.Once(content, {});
}
} else if (typeof simpleVarsPlugin === 'function') {
simpleVarsPlugin(content);
}
if (options.nested) {
processLoops(content);
}
if (rule.parent) {
rule.parent.insertBefore(rule, content.nodes);
}
}
if (rule.parent) {
rule.remove();
}
};
const processLoops = (css)=>{
css.walkAtRules((rule)=>{
if (rule.name === 'for') {
unrollLoop(rule);
}
});
};
const processOriginalLoops = (css)=>{
css.walkAtRules((rule)=>{
if (rule.name === 'for') {
if (rule.parent) {
manageIterStack(rule);
}
unrollLoop(rule);
}
});
};
return {
Once (root) {
processOriginalLoops(root);
},
postcssPlugin: 'postcss-for'
};
};
postcssFor.postcss = true;
export default postcssFor;
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../src/utils/postcss/postcss-for.ts"],"sourcesContent":["/**\n * Copyright (c) 2025-Present, Nitrogen Labs, Inc.\n * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.\n *\n * PostCSS 8-compatible version of postcss-for plugin\n * Original: https://github.com/antyakushev/postcss-for\n */\nimport postcss from 'postcss';\nimport postcssSimpleVars from 'postcss-simple-vars';\n\ninterface PostcssForOptions {\n  nested?: boolean;\n}\n\nconst postcssFor = (opts: PostcssForOptions = {}) => {\n  const options = {\n    nested: opts.nested !== false\n  };\n\n  const iterStack: string[] = [];\n\n  const parentsHaveIterator = (rule: postcss.AtRule, param: string): boolean => {\n    if(rule.parent === null) {\n      return false;\n    }\n    if(rule.parent.type === 'root') {\n      return false;\n    }\n    if(rule.parent.type !== 'atrule' || !rule.parent.params) {\n      return false;\n    }\n\n    const parentIterVar = rule.parent.params.split(/\\s+/)[0];\n    if(!parentIterVar) {\n      return false;\n    }\n    if(parentIterVar === param) {\n      return true;\n    }\n    if(iterStack.indexOf(param) !== -1) {\n      return true;\n    }\n    return parentsHaveIterator(rule.parent as postcss.AtRule, param);\n  };\n\n  const manageIterStack = (rule: postcss.AtRule) => {\n    if(rule.parent && rule.parent.type !== 'root') {\n      const parentIterVar = rule.parent.type === 'atrule' && rule.parent.params\n        ? rule.parent.params.split(/\\s+/)[0]\n        : null;\n      if(parentIterVar && iterStack.indexOf(parentIterVar) === -1) {\n        iterStack.splice(0, iterStack.length);\n      } else if(parentIterVar) {\n        const parentIndex = iterStack.indexOf(parentIterVar);\n        if(parentIndex !== -1) {\n          iterStack.splice(parentIndex + 1, iterStack.length - parentIndex - 1);\n        }\n      }\n    } else {\n      iterStack.splice(0, iterStack.length);\n    }\n    const currentIterVar = rule.params.split(/\\s+/)[0];\n    if(currentIterVar) {\n      iterStack.push(currentIterVar);\n    }\n  };\n\n  const checkNumber = (rule: postcss.AtRule) => (param: string) => {\n    if(isNaN(Number(param)) || !param.match(/^-?\\d+\\.?\\d*$/)) {\n      if(param.indexOf('$') !== -1) {\n        if(!parentsHaveIterator(rule, param)) {\n          throw rule.error('External variable (not from a parent for loop) cannot be used as a range parameter', {\n            plugin: 'postcss-for'\n          });\n        }\n      } else {\n        throw rule.error('Range parameter should be a number', {\n          plugin: 'postcss-for'\n        });\n      }\n    }\n  };\n\n  const checkParams = (rule: postcss.AtRule, params: string[]) => {\n    if(\n      !params[0]?.startsWith('$') ||\n      params[1] !== 'from' ||\n      params[3] !== 'to' ||\n      (params[5] && params[5] !== 'by')\n    ) {\n      throw rule.error('Wrong loop syntax', {\n        plugin: 'postcss-for'\n      });\n    }\n\n    [params[2], params[4], params[6] || '0'].forEach(checkNumber(rule));\n  };\n\n  const unrollLoop = (rule: postcss.AtRule) => {\n    const params = rule.params.split(/\\s+/);\n\n    checkParams(rule, params);\n\n    const iterator = params[0].slice(1);\n    const index = +params[2];\n    const top = +params[4];\n    const dir = top < index ? -1 : 1;\n    const by = (+(params[6] || 1)) * dir;\n\n    const value: Record<string, number> = {};\n    for(let i = index; i * dir <= top * dir; i = i + by) {\n      const content = rule.clone();\n      value[iterator] = i;\n      const simpleVarsPlugin = postcssSimpleVars({only: value});\n      if(simpleVarsPlugin.prepare) {\n        const prepared = simpleVarsPlugin.prepare({} as any);\n        if(prepared.Once) {\n          prepared.Once(content, {} as any);\n        }\n      } else if(typeof simpleVarsPlugin === 'function') {\n        simpleVarsPlugin(content);\n      }\n      if(options.nested) {\n        processLoops(content);\n      }\n      if(rule.parent) {\n        rule.parent.insertBefore(rule, content.nodes);\n      }\n    }\n    if(rule.parent) {\n      rule.remove();\n    }\n  };\n\n  const processLoops = (css: postcss.Container) => {\n    css.walkAtRules((rule) => {\n      if(rule.name === 'for') {\n        unrollLoop(rule);\n      }\n    });\n  };\n\n  const processOriginalLoops = (css: postcss.Root) => {\n    css.walkAtRules((rule) => {\n      if(rule.name === 'for') {\n        if(rule.parent) {\n          manageIterStack(rule);\n        }\n        unrollLoop(rule);\n      }\n    });\n  };\n\n  return {\n    Once(root) {\n      processOriginalLoops(root);\n    },\n    postcssPlugin: 'postcss-for'\n  };\n};\n\npostcssFor.postcss = true;\n\nexport default postcssFor;\n\n"],"names":["postcssSimpleVars","postcssFor","opts","options","nested","iterStack","parentsHaveIterator","rule","param","parent","type","params","parentIterVar","split","indexOf","manageIterStack","splice","length","parentIndex","currentIterVar","push","checkNumber","isNaN","Number","match","error","plugin","checkParams","startsWith","forEach","unrollLoop","iterator","slice","index","top","dir","by","value","i","content","clone","simpleVarsPlugin","only","prepare","prepared","Once","processLoops","insertBefore","nodes","remove","css","walkAtRules","name","processOriginalLoops","root","postcssPlugin","postcss"],"mappings":"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"}