CSS伪类:is()嵌套在Sass/Less等预处理器中的写法对比

7次阅读

处理器需版本适配和配置调整才能支持 :is():sass v1.33+ 识别,旧版报错;less v4.1+ 需关闭严格模式;均推荐用插值 #{} 或顶层书写规避解析问题。

CSS伪类:is()嵌套在Sass/Less等预处理器中的写法对比

直接写 :is() 会报错,预处理器默认不解析

Sass 和 Less 默认把 :is() 当成普通伪类字符串处理,但实际它属于 css 原生语法,需要浏览器运行时解析。预处理器在编译阶段若遇到未识别的伪类(比如 :is():has()),可能直接抛错,或更隐蔽地——悄悄删掉整个选择器、甚至整条规则。

  • Less v4.1+ 支持原样输出 :is(),但需关闭严格模式(strictmath: false 不相关,关键是别用 javascriptEnabled 干预选择器)
  • Sass(Dart Sass)v1.33+ 才开始识别 :is(),旧版(如 Node Sass)完全不认,会报 Invalid CSS after "...": expected "{", was ":is("
  • 如果用 @extend%placeholder 引用了含 :is() 的选择器,Sass 旧版本会直接崩溃

Sass 中安全写法:用插值 #{} 绕过解析

不是所有地方都能直接敲 :is(),尤其当它嵌套在 & 或与其他伪类混用时。最稳的方式是用插值强制“逃逸”预处理器解析逻辑,让内容原样透出到 CSS。

  • ✅ 正确:&:hover, #{':is(ul, ol) li'}:focus { color: blue; }
  • ❌ 错误:&:hover, :is(ul, ol) li:focus { ... }(旧 Sass 报错;新 Sass 虽能编译,但嵌套层级可能错乱)
  • ⚠️ 注意:#{} 内不能有变量插值(如 #{':is(#{$list}, p) li'}),否则 Sass 会提前求值导致语法错误
  • 性能无影响——插值只是字符串拼接,不增加运行时开销

Less 中写 :is() 要关掉选择器校验

Less 默认会对选择器结构做轻量校验,遇到 :is() 这类新语法容易误判为非法。不改配置的话,常见现象是整条规则消失、无报错、无输出。

  • 必须在编译配置里加 math: 'always'(非必需)但关键是设 strictUnits: false 并确保没启用 javascriptEnabled
  • ✅ 推荐写法:.menu:hover, :is(nav, header) .logo { opacity: 1; }(Less v4.2+ 直接支持)
  • ❌ 避免嵌套写法:.box { &:is(:hover, :active) { ... } } —— Less 不支持伪类作为 & 后缀
  • 兼容性提示:即使 Less 编译成功,也要确认目标浏览器是否支持 :is()chrome 105+/firefox 103+/safari 15.4+)

真正麻烦的是和 @media@container 嵌套混用

预处理器对媒体查询内嵌套的支持本就参差不齐,再叠一层 :is(),很容易触发解析边界问题。这不是写法对错问题,而是当前工具链的解析深度限制。

立即学习前端免费学习笔记(深入)”;

  • Sass 在 @media (min-width: 768px) { :is(button, a):focus { ... } } 中通常没问题,但换成 @container (min-width: 400px) { ... } 就可能失败(Dart Sass v1.70+ 才初步支持)
  • Less 对 @container 根本无感知,会直接忽略整个块——此时 :is() 即使写对也白搭
  • 最保险的退路:把含 :is() 的规则抽到顶层,用注释标记来源,避免嵌套

预处理器对 :is() 的支持不是“开了就全好”,而是按版本、按上下文、按嵌套深度分层生效。你看到的“能编译”,不等于“能正确生成目标选择器”。

text=ZqhQzanResources