如何从 CSS 选择器中提取类名

How to extract class names from a CSS selector?

本文关键字:提取 选择器 CSS      更新时间:2023-09-26

故事:

我目前正在构建一个 ESLint 规则,以警告在 CSS 选择器定位器中使用面向引导布局和角度的技术类。目前,我在字符串方法中使用了一个简单的子字符串:

for (var i = 0; i < prohibitedClasses.length; i++) {
  if (node.arguments[0].value.indexOf(prohibitedClasses[i]) >= 0) {
    context.report({
      node: node,
      message: 'Unexpected Bootstrap class "' + prohibitedClasses[i] + '" inside a CSS selector'
    })
  }

但事实证明,它并不可靠。例如,它会在 CSS 选择器上抛出错误 2 次.col-sm-offset-11报告要使用的col-sm-offset-1col-sm-offset-11。我可以想象它可以很容易地在使用多个伪类的更复杂的选择器上中断。


问题:

从 CSS 选择器中提取类名的最可靠方法是什么?


以下是我们应该涵盖的示例测试列表(有待改进):

.col-sm-push-4                 // -> ['col-sm-push-4']
.myclass.col-lg-pull-8         // -> ['myclass', 'col-lg-pull-8']
[class*='col-md-offset-4']     // -> []
[class$=col-md-offset-11]      // -> []
[class~="col-md-10"] .myclass  // -> ['col-md-10', 'myclass']
.col-md-10,.col-md-11          // -> ['col-md-10', 'col-md-11']

请注意,我们需要跳过^=$=*=部分类过滤器值,留下~=(感谢您的评论)。

有一个

专门为问题包设计的node-css-selector-parser缺少"如何使用它"部分来提取类名。填补空白,这是我如何将其应用于问题的方法。

使用 node-css-selector-parser ,我们可以解析一个 CSS 选择器,并根据结果类型分析与点一起使用的类名(例如 .myclass ) 和属性选择器中使用的类名(例如 [class*=test] ):

// setup up CSS selector parser
var CssSelectorParser = require('css-selector-parser').CssSelectorParser
var parser = new CssSelectorParser()
parser.registerSelectorPseudos('has', 'contains')
parser.registerNestingOperators('>', '+', '~')
parser.registerAttrEqualityMods('^', '$', '*', '~')
parser.enableSubstitutes()
function extractClassNames (rule) {
  var classNames = []
  // extract class names defined with ".", e.g. .myclass
  if (rule.classNames) {
    classNames.push.apply(classNames, rule.classNames)
  }
  // extract class names defined in attributes, e.g. [class*=myclass]
  if (rule.attrs) {
    rule.attrs.forEach(function (attr) {
      if (attr.name === 'class') {
        classNames.push(attr.value)
      }
    })
  }
  return classNames
}
module.exports = function (cssSelector) {
  try {
    var result = parser.parse(cssSelector)
  } catch (err) {
    // ignore parsing errors - we don't want it to fail miserably on a target machine during a ESLint run
    console.log('Parsing CSS selector: "' + cssSelector + '". ' + err)
    return []
  }
  // handling empty inputs
  if (!result) {
    return []
  }
  var classNames = []
  if (result.type === 'ruleSet') {
    var rule = result.rule
    while (rule) {
      classNames.push.apply(classNames, extractClassNames(rule))
      rule = rule.rule
    }
  } else if (result.type === 'selectors' && result.selectors) {
    result.selectors.forEach(function (selector) {
      var srule = selector.rule
      while (srule) {
        classNames.push.apply(classNames, extractClassNames(srule))
        srule = srule.rule;
      }
    })
  }
  return classNames
}

(使用standard代码样式 - 因此,例如,行尾没有;

事实证明,这对我有用,希望能帮助其他人解决类似的问题。请注意,在这种状态下,此代码还将提取传递到^=$=*=的分部类值,理想情况下需要跳过这些值。