原生CSS嵌套简介

原生,css,嵌套,简介 · 浏览次数 : 147

小编点评

**CSS嵌套的替代方案** 使用 CSS 预处理器可以为嵌套选择器提供替代方案,简化代码并提高效率。 * **使用`&`符号:** 可以将子选择器嵌套在父选择器中,例如:`.parent1 .child1,.parent2 .child1 {}`。 * **使用更复杂的选择器:** 例如:`.parent1, .parent2 { .child1 {color: red; } .child2 {color: green; &:hover { color: blue; } }}`。 * **使用媒体查询:** 可以使用媒体查询对选择器进行动态调整,例如:`p { color: cyan; @media (min-width: 800px) { color: purple; }}`。 **其他好处:** * **代码可读性:**使用嵌套选择器可以使代码更加易于理解。 * **性能优化:**嵌套选择器可以减少样式解析的次数,提高性能。 * **可扩展性:**可以使用嵌套选择器创建更复杂的选择器。 **注意:** * 嵌套选择器在 CSS 1.1 和更高版本中是支持的。 * 在使用嵌套选择器时,确保选择器的顺序合理。 * 在一些情况下,嵌套选择器可能会产生优先级冲突。

正文

嵌套是使用Sass等CSS预处理器的核心原因之一。现在,该功能已经以类似的语法出现在标准浏览器CSS中。你能否在构建系统时放弃对预处理器的依赖?

CSS嵌套可以节省输入时间,并使语法更易于阅读和维护。迄今为止,你必须像这样键入完整的选择器路径:

.parent1 .child1,
.parent2 .child1 {
  color: red;
}

.parent1 .child2,
.parent2 .child2 {
  color: green;
}

.parent1 .child2:hover,
.parent2 .child2:hover {
  color: blue;
}

现在,你可以将子选择器嵌套在父选择器中,比如:

.parent1, .parent2 {

  .child1 {
    color: red;
  }

  .child2 {
    color: green;

    &:hover {
      color: blue;
    }
  }

}

你可以嵌套任意层级的选择器,但要注意不要超过两到三级。嵌套层级没有技术层面的限制,但是会让代码更难阅读,并且让最终CSS变得很冗长。

直到2023年四月,暂没有浏览器支持CSS嵌套语法。你需要使用 CSS 预处理器(如 Sass、Less 或 PostCSS)进行构建步骤,以便将嵌套代码转换为常规的全选择器语法。嵌套功能现已在Chrome 112+和Safari 16.5+中实现,Firefox将在2023年晚些时候提供支持。

image.png

CSS原生嵌套规则

你可以将任何选择器嵌套到另一个选择器中,但必须以符号开头,如&.(用于HTML class),#(用于HTML id),@(用于媒体查询),:::*+~>,或 [。换句话说,它不能直接引用HTML元素。下面的代码无效,<p>选择器不会被解析:

.parent1 {

  /* FAILS */
  p {
    color: blue;
  }

}

修复该问题最简单的方法就是使用&,其与Sass相同的方式引用当前选择器:

.parent1 {

  /* WORKS */
  & p {
    color: blue;
  }

}

或者,可以这么解决:

  • > p - 但这将仅对 .parent1 的直接子元素进行样式调整
  • :is(p) - 但是:is()使用最优先选择器的优先级
  • :where(p) - 但是:where()的优先级为0

在这个简单的示例中,它们都可以工作,但在以后使用更复杂的样式表时,你可能会遇到优先级问题。

&还允许你在父选择器上定位伪元素和伪类。例如:

p.my-element {

  &::after {}

  &:hover {}

  &:target {}

}

如果你不使用&,你的目标将是选择器的所有子元素,而不是p.my-element本身。(在Sass中也会出现同样的情况)。

需要注意的是,你可以在选择器的任何位置使用&,比如:

.child1 {

  .parent3 & {
    color: red;
  }

}

这会转换为下列非嵌套语法:

.parent3 .child1 { color: red; }

你甚至可以在一个选择器中使用多个&符:

ul {

  & li & {
    color: blue;
  }

}

这会作用于嵌套的<ul>元素(ul li ul),如果你不想被逼疯我建议还是不要这么使用了!

最后,你可以嵌套媒体查询。下面的代码为段落元素应用了cyan颜色,除非浏览器宽度至少为800px,否则将变为purple

p {

  color: cyan;

  @media (min-width: 800px) {
    color: purple;
  }

}

CSS原生嵌套问题

原生嵌套在:is()中包裹父选择器,这可能会导致与Sass输出的差异。

考虑下列的嵌套代码:

.parent1, #parent2 {
  .child1 {}
}

在浏览器中解析时,这实际上变成了以下内容:

:is(.parent1, #parent2) .child1 {}

.parent1中的.child1元素的优先级为101,因为:is()使用了其最优先选择器的优先级--在本例中,是#parent2 ID。

Sass编译的代码与此相同:

.parent1 .child1,
#parent2 .child1 {
}

在本例中,.parent1 中的 .child1 元素的特异性为 002,因为它匹配两个类(#parent2 被忽略)。它的选择器比原生选项的优先级低,在级联中被覆盖的可能性更大。

你可能还会遇到一个更微妙的问题。考虑下列代码:

.parent .child {

  .grandparent & {}

}

原生CSS等价于:

.grandparent :is(.parent .child) {}

这与下列排序错误的HTML元素相匹配:

<div class="parent">
  <div class="grandparent">
    <div class="child">MATCH</div>
  </div>
</div>

由于CSS解析器会执行以下操作,因此MATCH会改变样式:

  1. 在DOM层次结构中的任意位置,找到所有类为child的元素,同时祖先元素的类为parent
  2. 在找到包含MATCH的元素后,解析器会检查该元素是否有一个祖先为grandparent的元素--同样是在DOM层次结构中的任意位置。找到后,解析器会相应地对该元素应用样式。

在Sass中不是这种情况,最终会编译成这样:

.grandparent .parent .child {}

上面的HTML没有样式化,因为元素的类没有遵循严格的grandparentparentchild顺序。

最后,Sass使用字符串替换,所以像下面这样的声明是有效的,可以匹配任何具有outer-space类的元素:

.outer {
  &-space { color: black; }
}

而原生CSS会忽略&-space选择器。

还需要CSS预处理器吗

在短期内,现有的CSS预处理器仍然必不可少。

Sass开发团队已经宣布,他们将支持.css文件中的原生CSS嵌套,并按原样输出代码。他们将一如既往地编译嵌套的SCSS代码,以避免破坏现有代码库,但当全球浏览器支持率达到98%时,他们将开始输出:is()选择器。

我猜测PostCSS插件等预处理器目前会扩展嵌套代码,但当浏览器支持率越来越高时,就会移除该功能。

当然,使用预处理器还有其他很好的理由--比如将部分代码捆绑到单一文件中,以及精简代码。但如果嵌套是你唯一需要的功能,那么你当然可以考虑在小型项目中使用本地CSS。

总结

CSS嵌套是最有用、最实用的预处理器功能之一。浏览器供应商努力创造了一个本地CSS版本,其相似性足以让Web开发人员满意。虽然两者之间存在细微差别,而且在使用(过于)复杂的选择器时可能会遇到不寻常的优先级问题,但很少有代码库需要进行彻底修改。

原生嵌套可能会让你重新考虑对CSS预处理器的需求,但它们仍能提供其他好处。Sass和类似工具仍然是大多数开发人员工具包的重要组成部分。

以上就是本文的全部内容。如果对你有所帮助,欢迎点赞、收藏、转发~

与原生CSS嵌套简介相似的内容:

原生CSS嵌套简介

> 嵌套是使用Sass等CSS预处理器的核心原因之一。现在,该功能已经以类似的语法出现在标准浏览器CSS中。你能否在构建系统时放弃对预处理器的依赖? CSS嵌套可以节省输入时间,并使语法更易于阅读和维护。迄今为止,你必须像这样键入完整的选择器路径: ```css .parent1 .child1,

云间玉兔,自出机抒,从零开始制作Web插件网页特效小兔子组件(小挂件widget),基于原生CSS/NPM

著意登楼瞻玉兔,何人张幕遮银阙?又到了一年一度的网页小挂件环节,以往我们都是集成别人开源的组件,但所谓熟读唐诗三百首,不会做诗也会吟,熟读了别人的东西,做几首打油诗也是可以的,但若不能自出机抒,却也成不了大事,所以本次我们从零开始制作属于自己的网页小挂件,博君一晒。 玉兔主题元素绘制 成本最低的绘制

【现代 CSS】标准滚动条控制规范 scrollbar-color 和 scrollbar-width

Chrome 在 121 版本开始,原生支持了两个滚动条样式相关的样式 scrollbar-color 和 scrollbar-width。 要知道,在此前,虽然有 ::-webkit-scrollbar 规范可以控制滚动条,可是,::-webkit-scrollbar 是非标准特性,在 MDN 文

WebKit Inside: CSS 样式表解码字符集

CSS 样式表引入有3种方式: 外部样式表、内部样式表、行内样式,不同的引入方式,解码样式表的字符集原理不一样。 外部样式表 外部样式表由 link 标签引入,当 WebKit 解析到 link 标签时就会构造 CachedCSSStyleSheet 对象。这个对象持有 CachedResource

Vue学习笔记(七):绑定css样式

1 绑定class样式¶ vue为HTML绑定css中的class样式是通过v-bind实现的。 1.1 绑定单个class¶ 把需要绑定的样式class名赋值给一遍变量,然后通过变量v-bind绑定class属性,绑定后的class并不会覆盖原来的class属性,而是与原来的class进行叠加。如

掉了两根头发后,我悟了!vue3的scoped原来是这样避免样式污染(下)

上篇文章中我们讲了使用scoped后,vue是如何给CSS选择器添加对应的属性选择器[data-v-x]。这篇文章我们来接着讲vue是如何给html增加自定义属性data-v-x

掉了两根头发后,我悟了!vue3的scoped原来是这样避免样式污染(下)

上篇文章中我们讲了使用scoped后,vue是如何给CSS选择器添加对应的属性选择器[data-v-x]。这篇文章我们来接着讲vue是如何给html增加自定义属性data-v-x

架构设计(四):CDN

架构设计(四):CDN 作者:Grey 原文地址: 博客园:架构设计(四):CDN CSDN:架构设计(四):CDN CDN 全称 Content delivery network ,即:内容分发网络。 CDN 是一个地理上分散的服务器网络,主要用于提供静态内容。如:图片、视频、CSS、JavaSc

掉了两根头发后,我悟了!vue3的scoped原来是这样避免样式污染(上)

前言 众所周知,在vue中使用scoped可以避免父组件的样式渗透到子组件中。使用了scoped后会给html增加自定义属性data-v-x,同时会给组件内CSS选择器添加对应的属性选择器[data-v-x]。这篇我们来讲讲vue是如何给CSS选择器添加对应的属性选择器[data-v-x]。注:本文

【动画进阶】极具创意的鼠标交互动画

最近,群里在讨论这么一个有趣的交互效果,来源于:vueflow.dev: 通过审查元素,发现原效果借助了 Canvas 实现。 思索了一番,觉得这个效果利用 CSS 配合部分 Javascript 代码完全也是可以做到的。 于是动手尝试了一番,最终完美的复刻了该效果: 过程中还是有非常多有意思的技巧