.gitignore added
[dotfiles/.git] / .config / coc / extensions / node_modules / coc-prettier / node_modules / stylelint / lib / rules / selector-max-class / index.js
diff --git a/.config/coc/extensions/node_modules/coc-prettier/node_modules/stylelint/lib/rules/selector-max-class/index.js b/.config/coc/extensions/node_modules/coc-prettier/node_modules/stylelint/lib/rules/selector-max-class/index.js
new file mode 100644 (file)
index 0000000..6c1dd7d
--- /dev/null
@@ -0,0 +1,88 @@
+"use strict";
+
+const isStandardSyntaxRule = require("../../utils/isStandardSyntaxRule");
+const isStandardSyntaxSelector = require("../../utils/isStandardSyntaxSelector");
+const parseSelector = require("../../utils/parseSelector");
+const report = require("../../utils/report");
+const resolvedNestedSelector = require("postcss-resolve-nested-selector");
+const ruleMessages = require("../../utils/ruleMessages");
+const validateOptions = require("../../utils/validateOptions");
+
+const ruleName = "selector-max-class";
+
+const messages = ruleMessages(ruleName, {
+  expected: (selector, max) =>
+    `Expected "${selector}" to have no more than ${max} ${
+      max === 1 ? "class" : "classes"
+    }`
+});
+
+function rule(max) {
+  return (root, result) => {
+    const validOptions = validateOptions(result, ruleName, {
+      actual: max,
+      possible: [
+        function(max) {
+          return typeof max === "number" && max >= 0;
+        }
+      ]
+    });
+    if (!validOptions) {
+      return;
+    }
+
+    function checkSelector(selectorNode, ruleNode) {
+      const count = selectorNode.reduce((total, childNode) => {
+        // Only traverse inside actual selectors and :not()
+        if (childNode.type === "selector" || childNode.value === ":not") {
+          checkSelector(childNode, ruleNode);
+        }
+
+        return (total += childNode.type === "class" ? 1 : 0);
+      }, 0);
+
+      if (
+        selectorNode.type !== "root" &&
+        selectorNode.type !== "pseudo" &&
+        count > max
+      ) {
+        report({
+          ruleName,
+          result,
+          node: ruleNode,
+          message: messages.expected(selectorNode, max),
+          word: selectorNode
+        });
+      }
+    }
+
+    root.walkRules(ruleNode => {
+      if (!isStandardSyntaxRule(ruleNode)) {
+        return;
+      }
+      if (!isStandardSyntaxSelector(ruleNode.selector)) {
+        return;
+      }
+      if (
+        ruleNode.nodes.some(
+          node => ["rule", "atrule"].indexOf(node.type) !== -1
+        )
+      ) {
+        // Skip unresolved nested selectors
+        return;
+      }
+
+      ruleNode.selectors.forEach(selector => {
+        resolvedNestedSelector(selector, ruleNode).forEach(resolvedSelector => {
+          parseSelector(resolvedSelector, result, ruleNode, container =>
+            checkSelector(container, ruleNode)
+          );
+        });
+      });
+    });
+  };
+}
+
+rule.ruleName = ruleName;
+rule.messages = messages;
+module.exports = rule;