>:
* Properties can be directly used,
applying to the same elements as the parent rule
(when the [=conditional group rule=] matches)
* [=Style rules=] are [=nested style rules=],
with their [=nesting selector=] taking its definition
from the nearest ancestor [=style rule=].
Issue: Do @media/etc gain a .style
attribute
to expose their properties,
or do we pretend properties are wrapped in a ''& {...}'' rule,
so we can just reflect them in the .cssRules
attribute?
(Issue 7850)
For example, the following conditional nestings are valid:
/* Properties can be directly used */
.foo {
display: grid;
@media (orientation: landscape) {
grid-auto-flow: column;
}
}
/* equivalent to
.foo {
display: grid;
@media (orientation: landscape) {
& {
grid-auto-flow: column;
}
}
}
*/
/* finally equivalent to
.foo { display: grid; }
@media (orientation: landscape) {
.foo {
grid-auto-flow: column;
}
}
*/
/* Conditionals can be further nested */
.foo {
display: grid;
@media (orientation: landscape) {
grid-auto-flow: column;
@media (min-width > 1024px) {
max-inline-size: 1024px;
}
}
}
/* equivalent to
.foo { display: grid; }
@media (orientation: landscape) {
.foo {
grid-auto-flow: column;
}
}
@media (orientation: landscape) and (min-width > 1024px) {
.foo {
max-inline-size: 1024px;
}
}
*/
/* Example nesting Cascade Layers */
html {
@layer base {
block-size: 100%;
@layer support {
& body {
min-block-size: 100%;
}
}
}
}
/* equivalent to
@layer base {
html { block-size: 100%; }
}
@layer base.support {
html body { min-block-size: 100%; }
}
*/
/* Example nesting Scoping */
.card {
inline-size: 40ch;
aspect-ratio: 3/4;
@scope (&) {
:scope {
border: 1px solid white;
}
}
}
/* equivalent to
.card { inline-size: 40ch; aspect-ratio: 3/4; }
@scope (.card) {
:scope { border-block-end: 1px solid white; }
}
*/
Mixing Nesting Rules and Declarations {#mixing}
-----------------------------------------------
When a style rule contains both declarations
and [=nested style rules=] or [=nested conditional group rules=],
all three can be arbitrarily mixed.
However, the relative order of declarations
vs other rules
is not preserved in any way.
For example,
in the following code:
article {
color: green;
& { color: blue; }
color: red;
}
/* equivalent to */
article {
color: green;
color: red;
& { color: blue; }
}
For the purpose of determining the [[css-cascade-4#cascade-sort|Order Of Appearance]],
[=nested style rules=] and [=nested conditional group rules=]
are considered to come after their parent rule.
For example:
article {
color: blue;
& { color: red; }
}
Both declarations have the same specificity (0,0,1),
but the nested rule is considered to come after its parent rule,
so the ''color: red'' declarations wins the cascade.
On the other hand, in this example:
article {
color: blue;
:where(&) { color: red; }
}
The '':where()'' pseudoclass reduces the specificity of the [=nesting selector=] to 0,
so the ''color: red'' declaration now has a specificity of (0,0,0),
and loses to the ''color: blue'' declaration
before "Order Of Appearance" comes into consideration.
Nesting Selector: the ''&'' selector {#nest-selector}
=====================================================
When using a nested style rule,
one must be able to refer to the elements matched by the parent rule;
that is, after all, the entire point of nesting.
To accomplish that,
this specification defines a new selector,
the nesting selector,
written as & (U+0026 AMPERSAND).
When used in the selector of a nested style rule,
the nesting selector represents the elements matched by the parent rule.
When used in any other context,
it represents nothing.
(That is, it's valid, but matches no elements.)
The
nesting selector can be desugared
by replacing it with the parent style rule's selector,
wrapped in an '':is()'' selector.
For example,
a, b {
& c { color: blue; }
}
is equivalent to
:is(a, b) c { color: blue; }
The [=nesting selector=] cannot represent pseudo-elements
(identical to the behavior of the '':is()'' pseudo-class).
For example, in the following style rule:
.foo, .foo::before, .foo::after {
color: red;
&:hover { color: blue; }
}
the ''&'' only represents the elements matched by ''.foo'';
in other words, it's equivalent to:
.foo, .foo::before, .foo::after {
color: red;
}
.foo:hover {
color: blue;
}
Issue: We'd like to relax this restriction,
but need to do so simultaneously for both '':is()'' and ''&'',
since they're intentionally built on the same underlying mechanisms.
(Issue 7433)
The specificity of the nesting selector
is equal to the largest specificity among the complex selectors
in the parent style rule's selector list
(identical to the behavior of '':is()'').
For example, given the following style rules:
#a, b {
& c { color: blue; }
}
.foo c { color: red; }
Then in a DOM structure like
Blue text
The text will be blue, rather than red.
The specificity of the ''&''
is the larger of the specificities of ''#a'' ([1,0,0])
and
b ([0,0,1]),
so it's [1,0,0],
and the entire ''& c'' selector thus has specificity [1,0,1],
which is larger than the specificity of ''.foo c'' ([0,1,1]).
Notably, this is
different than the result you'd get
if the nesting were manually expanded out
into non-nested rules,
since the ''color: blue'' declaration would then be matching
due to the ''b c'' selector ([0,0,2])
rather than ''#a c'' ([1,0,1]).
Why is the specificity different than non-nested rules?
The [=nesting selector=] intentionally uses the same specificity rules
as the '':is()'' pseudoclass,
which just uses the largest specificity among its arguments,
rather than tracking which selector actually matched.
This is required for performance reasons;
if a selector has multiple possible specificities,
depending on how precisely it was matched,
it makes selector matching much more complicated and slower.
That skirts the question, tho:
why do we define ''&'' in terms of '':is()''?
Some non-browser implementations of Nesting-like functionality
do not desugar to '':is()'',
largely because they predate the introduction of '':is()'' as well.
Instead, they desugar directly;
however, this comes with its own significant problems,
as some (reasonably common) cases can accidentally produce massive selectors,
due to the exponential explosion of possibilities.
.a1, .a2, .a3 {
.b1, .b3, .b3 {
.c1, .c2, .c3 {
...;
}
}
}
/* naively desugars to */
.a1 .b1 .c1,
.a1 .b1 .c2,
.a1 .b1 .c3,
.a1 .b2 .c1,
.a1 .b2 .c2,
.a1 .b2 .c3,
.a1 .b3 .c1,
.a1 .b3 .c2,
.a1 .b3 .c3,
.a2 .b1 .c1,
.a2 .b1 .c2,
.a2 .b1 .c3,
.a2 .b2 .c1,
.a2 .b2 .c2,
.a2 .b2 .c3,
.a2 .b3 .c1,
.a2 .b3 .c2,
.a2 .b3 .c3,
.a3 .b1 .c1,
.a3 .b1 .c2,
.a3 .b1 .c3,
.a3 .b2 .c1,
.a3 .b2 .c2,
.a3 .b2 .c3,
.a3 .b3 .c1,
.a3 .b3 .c2,
.a3 .b3 .c3 {...}
Here, three levels of nesting,
each with three selectors in their lists,
produced 27 desugared selectors.
Adding more selectors to the lists,
adding more levels of nesting,
or making the nested rules more complex
can make a relatively small rule
expand into multiple megabytes of selectors
(or much, much more!).
Some CSS tools avoid the worst of this
by heuristically discarding some variations,
so they don't have to output as much
but are still probably correct,
but that's not an option available to UAs.
Desugaring with '':is()'' instead eliminates this problem entirely,
at the cost of making specificity slightly less useful,
which was judged a reasonable trade-off.
The nesting selector is allowed anywhere in a compound selector,
even before a type selector,
violating the normal restrictions on ordering within a compound selector.
For example, ''&div'' is a valid nesting selector,
meaning "whatever the parent rules matches,
but only if it's also a <{div}> element".
It could also be written as ''div&'' with the same meaning,
but that wouldn't be valid to start a [=nested style rule=]
(but it could be used somewhere other than the very start of the selector).
CSSOM {#cssom}
==============
Modifications to {{CSSStyleRule}} {#cssom-style}
---------------------------------------------
CSS style rules gain the ability to have nested rules:
partial interface CSSStyleRule {
[SameObject] readonly attribute CSSRuleList cssRules;
unsigned long insertRule(CSSOMString rule, optional unsigned long index = 0);
undefined deleteRule(unsigned long index);
};
The cssRules attribute
must return a {{CSSRuleList}} object for the [=CSSRule/child CSS rules=].
The insertRule(rule, index) method
must return the result of
invoking [=insert a CSS rule=] rule
into the [=CSSRule/child CSS rules=] at index.
The deleteRule(index) method
must [=remove a CSS rule=] from the [=CSSRule/child CSS rules=] at index.
Note: Serialization of {{CSSStyleRule}} with nested rules
are already well-defined by [[CSSOM]],
via [=serialize a CSS rule=].
Whether we attach .cssRules to CSSStyleRule,
as described here,
or to CSSStyleDeclaration
(to be consistent with future work
allowing `style` attributes to have nested rules)
is still an open question.
Also, attaching it to .style
means existing code setting
rule.style=""
and expecting
(
Issue 7850)