1. Introduction
This is currently an early draft of the things that are new in level 5. The features in Level 3 and Level 4 are still defined in [css-conditional-3] and [css-conditional-4] and have not yet been copied here.
CSS Conditional Level 5 extends the @supports rule and supports query syntax to allow testing for supported font technologies.
It also adds an @when rule, which generalizes the concept of a conditional rule. Anything that can be expressed in an existing conditional rule can be expressed in @when by wrapping it in an appropriate function to declare what kind of condition it is. This allow authors to easily combine multiple types of queries, such as media queries and supports queries, in a single boolean expression. Without this, authors must rely on nesting separate conditional rules, which is harder to read and write, presupposes the conditions are to be conjoined with the “and” boolean relation (with no easy way to indicate anything else), and restricts their utility in the proposed conditional rule chains.
It also adds @else rules, which immediately follow other conditional rules and automatically qualify their conditions as the inverse of the immediately preceding rule’s conditions, such that only the first matching rule in a conditional rule chain is applied.
It also adds Container Queries. They are conceptually similar to Media Queries, but allow testing aspects of elements within the document (such as box dimensions or computed styles), rather than on the document as a whole.
2. Extensions to the @supports rule
This level of the specification extends the <supports-feature> syntax as follows:
<supports-feature> = <supports-selector-fn> | <supports-font-tech-fn> | <supports-font-format-fn> | <supports-decl> <supports-font-tech-fn> = font-tech( <font-tech> ) <supports-font-format-fn> = font-format( <font-format> )
Tests
- <supports-font-tech-fn>
-
The result is true if the UA supports the font tech provided as an argument to the function.
- <supports-font-format-fn>
-
The result is true if the UA supports the font format provided as an argument to the function.
2.1. Extensions to the definition of support
A CSS processor is considered to support a font tech when it is capable of utilizing the specified CSS Fonts 4 § 11.1 Font tech in layout and rendering.
A CSS processor is considered to support a font format when it is capable of utilizing the specified CSS Fonts 4 § 11.2 Font formats in layout and rendering, and this format is not specified as a <string>.
3. Generalized Conditional Rules: the @when rule
The @when at-rule is a conditional group rule that generalizes the individual conditional group rules such as @media and @supports. It is defined as:
@when <boolean-condition>{ <rule-list>}
Where <boolean-condition> is a boolean algebra a la Media Queries 4 § 3 Syntax, but with media() and supports() functions as leaves.
Define "boolean algebra, with X as leaves" in a generic way in Conditional, so all the conditional rules can reference it directly, rather than having to redefine boolean algebra on their own.
The media() and supports() functions are defined as:
media () =media ( [ <mf-plain> | <mf-boolean> | <mf-range>] ) supports () =supports ( <declaration>)
A media() or supports() function is associated the boolean result that its contained condition is associated with.
4. Chained Conditionals: the @else rule
Usually, conditional group rules are independent; each one has a separate condition evaluated without direct reference to any other rule, and decides whether or not to apply its contained rules based solely on its condition.
This is fine for simple conditions, but makes it difficult to write a collection of conditionals that are meant to be mutually exclusive: authors have to very carefully craft their conditions to not activate when the other rules are meant to, and make sure the collection of conditionals don’t accidentally all exclude some situation which is then left unstyled.
The @else rule is a conditional group rule used to form conditional rule chains, which associate multiple conditional group rules and guarantee that only the first one that matches will evaluate its condition as true. It is defined as:
@else <boolean-condition>?{ <rule-list>}
@else is interpreted identically to @when. If its <boolean-condition> is omitted, it’s treated as having a condition that’s always true.
A conditional rule chain is a series of consecutive conditional group rules, starting with a conditional group rule other than @else, followed by zero or more @else rules. There cannot be anything between the successive conditional group rules other than whitespace and/or comments; any other token “breaks” the chain.
Should we require that only the last @else in a chain can have an omitted condition? It’s not uncommon for me, when debugging code, to short-circuit an if-else chain by setting one of them to "true"; I presume that would be similarly useful in CSS? It’s still pretty easy to see you’ve done something wrong if you omit the condition accidentally.
Within a conditional rule chain, the conditions of each conditional group rule are evaluated in order. If one of them is true, the conditions of all following conditional group rules in the chain evaluate to false, regardless of their stated condition.
An @else rule that is not part of a conditional rule chain is invalid and must be ignored.
@when media ( width >=400 px ) andmedia ( pointer: fine) andsupports ( display: flex) { /* A */ } @else supports ( caret-color: pink) andsupports ( background:double-rainbow ()) { /* B */ } @else { /* C */ }
Exactly one of the preceding rules will be chosen, even though the second rule doesn’t exclude large widths, fine points, or flexbox support, and the last rule doesn’t specify anything at all.
To achieve the same result without conditional rule chains, you’d need to write:
@media ( width >=400 px ) and( pointer: fine) { @supports ( display: flex) { /* A */ } @supports not( display: flex) { @supports ( caret-color: pink) and( background:double-rainbow ()) { /* B */ } @supports not(( caret-color: pink) and( background:double-rainbow ())) { /* C */ } } } @media not(( width >=400 px ) and( pointer: fine)) { @supports ( caret-color: pink) and( background:double-rainbow ()) { /* B */ } @supports not(( caret-color: pink) and( background:double-rainbow ())) { /* C */ } }
This is simultaneously hard to read, requires significant duplication of both conditions and contents, and is very difficult to write correctly. If the conditions got any more complicated (which is not unusual in real-world content), the example would get significantly worse.
The fallback has no test condition, so will always be chosen unless one of the earlier conditions succeeds.
@when font-tech ( color-COLRv1) andfont-tech ( variations) { @font-face { font-family : icons; src : url ( icons-gradient-var.woff2 ); } } @else font-tech ( color-SVG) { @font-face { font-family : icons; src : url ( icons-gradient.woff2 ); } } @else font-tech ( color-COLRv0) { @font-face { font-family : icons; src : url ( icons-flat.woff2 ); } } @else { @font-face { font-family : icons; src : url ( icons-fallback.woff2 ); } }
Notice that in this example, the variable color font is only downloaded if COLRv1 is supported and font variations are also supported.
Notice too that only one of the available options will be downloaded; this would not be the case without @when and @else, as the next example shows.
The fallback might still be used for some characters; for example, if the color font supports only Latin, while the fallback supports Latin and Greek.
@font-face { font-family : icons; src : url ( icons-fallback.woff2 ); @supports font-tech ( color-COLRv1) { @font-face { font-family : icons; src : url ( icons-gradient-var.woff2 ); } }
5. Container Queries
While media queries provide a method to query aspects of the user agent or device environment that a document is being displayed in (such as viewport dimensions or user preferences), container queries allow testing aspects of elements within the document (such as box dimensions or computed styles).
By default, all elements are query containers for the purpose of container style queries, and can be established as query containers for container size queries and container scroll-state queries by specifying the additional query types using the container-type property (or the container shorthand). Style rules applying to a query container’s flat tree descendants can be conditioned by querying against it, using the @container conditional group rule.
main, aside{ container : my-layout / inline-size; } .media-object{ display : grid; grid-template : 'img' auto'content' auto /100 % ; } @container my-layout( inline-size >45 em ) { .media-object{ grid-template : 'img content' auto / auto1 fr ; } }
Media objects in the main and sidebar areas will each respond to their own container context.
For the ::part() and ::slotted() pseudo-element selectors, which represent real elements in the DOM tree, query containers can be established by flat tree ancestors of those elements. For other pseudo-elements, query containers can be established by inclusive flat tree ancestors of their originating element.
-
::before, ::after, ::marker, and ::backdrop query their originating elements
-
::first-letter and ::first-line query their originating elements, even if the fictional tag sequence may push the
::first-line
past other elements for the purpose of inheritance and rendering -
::slotted() selectors can query containers inside the shadow tree, including the slot itself
-
::slotted()::before selectors can query the slotted shadow host child
-
::part() selectors can query containers inside the shadow tree
-
::placeholder and ::file-selector-button can query the input element, but do not expose any internal containers if the input element is implemented using a shadow tree
< style > # container { width : 100 px ; container-type : inline - size ; } @ container ( inline-size < 150px ) { # inner :: before { content : "BEFORE" ; } } </ style > < div id = container > < span id = inner ></ span > </ div >
< div id = host style = "width:200px" > < template shadowroot = open > < style > # container { width : 100 px ; container-type : inline - size ; } @ container ( inline-size < 150px ) { :: slotted ( span ) { color : green ; } } </ style > < div id = container > < slot /> </ div > </ template > < span id = slotted > Green</ span > </ div >
5.1. Creating Query Containers: the container-type property
Name: | container-type |
---|---|
Value: | normal | [ [ size | inline-size ] || scroll-state ] |
Initial: | normal |
Applies to: | all elements |
Inherited: | no |
Percentages: | n/a |
Computed value: | specified keyword |
Canonical order: | per grammar |
Animation type: | not animatable |
The container-type property establishes the element as a query container for certain types of queries. For size container queries, which require certain types of containment, elements are explicitly made query containers through this property. For other types of query containers any element can be a query container, such as for container style queries.
Values have the following meanings:
- size
- Establishes a query container for container size queries on both the inline and block axis. Applies style containment and size containment to the principal box, and establishes an independent formatting context.
- inline-size
- Establishes a query container for container size queries on the container’s own inline axis. Applies style containment and inline-size containment to the principal box, and establishes an independent formatting context.
- scroll-state
- Establishes a query container for container scroll-state queries
- normal
- The element is not a query container for any container size queries or container scroll-state queries, but remains a query container for container style queries.
aside, main{ container-type : inline-size; } h2{ font-size : 1.2 em ; } @container ( width >40 em ) { h2{ font-size : 1.5 em ; } }
The 40em value used in the query condition is relative to the computed value of font-size on the relevant query container.
@container style ( --cards: small) { article{ border : thin solid silver; border-radius : 0.5 em ; padding : 1 em ; } }
#sticky{ container-type : scroll-state; position : sticky; } @container scroll-state ( stuck: top) { #sticky-child{ background-color : lime; } }
5.2. Naming Query Containers: the container-name property
Name: | container-name |
---|---|
Value: | none | <custom-ident>+ |
Initial: | none |
Applies to: | all elements |
Inherited: | no |
Percentages: | n/a |
Computed value: | the keyword none, or an ordered list of identifiers |
Canonical order: | per grammar |
Animation type: | not animatable |
The container-name property specifies a list of query container names. These names can be used by @container rules to filter which query containers are targeted. Container names are not tree-scoped names.
- none
- The query container has no query container name.
- <custom-ident>
- Specifies a query container name as an identifier. The keywords none, and, not, and or are excluded from this <custom-ident>.
main{ container-type : size; container-name : my-page-layout; } .my-component{ container-type : inline-size; container-name : my-component-library; } @container my-page-layout( block-size >12 em ) { .card{ margin-block : 2 em ; } } @container my-component-library( inline-size >30 em ) { .card{ margin-inline : 2 em ; } }
It is also possible to query for a container only based on its name.
@container my-page-layout{ .card{ padding : 1 em ; } }
5.3. Creating Named Containers: the container shorthand
Name: | container |
---|---|
Value: | <'container-name'> [ / <'container-type'> ]? |
Initial: | see individual properties |
Applies to: | see individual properties |
Inherited: | see individual properties |
Percentages: | see individual properties |
Computed value: | see individual properties |
Animation type: | see individual properties |
Canonical order: | per grammar |
Tests
- animation-container-size.html (live test) (source)
- animation-container-type-dynamic.html (live test) (source)
- animation-nested-animation.html (live test) (source)
- animation-nested-transition.html (live test) (source)
- aspect-ratio-feature-evaluation.html (live test) (source)
- at-container-parsing.html (live test) (source)
- at-container-serialization.html (live test) (source)
- at-container-style-parsing.html (live test) (source)
- at-container-style-serialization.html (live test) (source)
- auto-scrollbars.html (live test) (source)
- backdrop-invalidation.html (live test) (source)
- calc-evaluation.html (live test) (source)
- canvas-as-container-001.html (live test) (source)
- canvas-as-container-002.html (live test) (source)
- canvas-as-container-003.html (live test) (source)
- canvas-as-container-004.html (live test) (source)
- canvas-as-container-005.html (live test) (source)
- canvas-as-container-006.html (live test) (source)
- change-display-in-container.html (live test) (source)
- chrome-legacy-skip-recalc.html (live test) (source)
- column-spanner-in-container.html (live test) (source)
- conditional-container-status.html (live test) (source)
- container-computed.html (live test) (source)
- container-for-cue.html (live test) (source)
- container-for-shadow-dom.html (live test) (source)
- container-inheritance.html (live test) (source)
- container-inner-at-rules.html (live test) (source)
- container-inside-multicol-with-table.html (live test) (source)
- container-longhand-animation-type.html (live test) (source)
- container-name-computed.html (live test) (source)
- container-name-invalidation.html (live test) (source)
- container-name-parsing.html (live test) (source)
- container-name-tree-scoped.html (live test) (source)
- container-nested.html (live test) (source)
- container-parsing.html (live test) (source)
- container-selection-unknown-features.html (live test) (source)
- container-selection.html (live test) (source)
- container-size-invalidation-after-load.html (live test) (source)
- container-size-invalidation.html (live test) (source)
- container-size-nested-invalidation.html (live test) (source)
- container-size-shadow-invalidation.html (live test) (source)
- container-type-computed.html (live test) (source)
- container-type-containment.html (live test) (source)
- container-type-invalidation.html (live test) (source)
- container-type-layout-invalidation.html (live test) (source)
- container-type-parsing.html (live test) (source)
- container-units-animation.html (live test) (source)
- container-units-basic.html (live test) (source)
- container-units-computational-independence.html (live test) (source)
- container-units-content-box.html (live test) (source)
- container-units-gradient-invalidation.html (live test) (source)
- container-units-gradient.html (live test) (source)
- container-units-in-at-container-dynamic.html (live test) (source)
- container-units-in-at-container-fallback.html (live test) (source)
- container-units-in-at-container.html (live test) (source)
- container-units-ineligible-container.html (live test) (source)
- container-units-invalidation.html (live test) (source)
- container-units-media-queries.html (live test) (source)
- container-units-rule-cache.html (live test) (source)
- container-units-selection.html (live test) (source)
- container-units-shadow.html (live test) (source)
- container-units-sharing-via-rule-node.html (live test) (source)
- container-units-small-viewport-fallback.html (live test) (source)
- container-units-svglength.html (live test) (source)
- container-units-typed-om.html (live test) (source)
- counters-flex-circular.html (live test) (source)
- counters-in-container-dynamic.html (live test) (source)
- counters-in-container.html (live test) (source)
- br-crash.html (live test) (source)
- canvas-as-container-crash.html (live test) (source)
- chrome-bug-1289718-000-crash.html (live test) (source)
- chrome-bug-1289718-001-crash.html (live test) (source)
- chrome-bug-1346969-crash.html (live test) (source)
- chrome-bug-1362391-crash.html (live test) (source)
- chrome-bug-1429955-crash.html (live test) (source)
- chrome-bug-1505250-crash.html (live test) (source)
- chrome-bug-346264227-crash.html (live test) (source)
- chrome-bug-372358471-crash.html (live test) (source)
- chrome-custom-highlight-crash.html (live test) (source)
- chrome-layout-root-crash.html (live test) (source)
- chrome-quotes-crash.html (live test) (source)
- chrome-remove-insert-evaluator-crash.html (live test) (source)
- columns-in-table-001-crash.html (live test) (source)
- columns-in-table-002-crash.html (live test) (source)
- container-in-canvas-crash.html (live test) (source)
- container-type-change-chrome-legacy-crash.html (live test) (source)
- dialog-backdrop-crash.html (live test) (source)
- dirty-rowgroup-crash.html (live test) (source)
- flex-in-columns-000-crash.html (live test) (source)
- flex-in-columns-001-crash.html (live test) (source)
- flex-in-columns-002-crash.html (live test) (source)
- flex-in-columns-003-crash.html (live test) (source)
- focus-inside-content-visibility-crash.html (live test) (source)
- force-sibling-style-crash.html (live test) (source)
- grid-in-columns-000-crash.html (live test) (source)
- grid-in-columns-001-crash.html (live test) (source)
- grid-in-columns-002-crash.html (live test) (source)
- grid-in-columns-003-crash.html (live test) (source)
- iframe-init-crash.html (live test) (source)
- inline-multicol-inside-container-crash.html (live test) (source)
- inline-with-columns-000-crash.html (live test) (source)
- inline-with-columns-001-crash.html (live test) (source)
- input-column-group-container-crash.html (live test) (source)
- input-placeholder-inline-size-crash.html (live test) (source)
- marker-gcs-after-disconnect-crash.html (live test) (source)
- math-block-container-child-crash.html (live test) (source)
- mathml-container-type-crash.html (live test) (source)
- orthogonal-replaced-crash.html (live test) (source)
- pseudo-container-crash.html (live test) (source)
- remove-dom-child-change-style.html (live test) (source)
- reversed-ol-crash.html (live test) (source)
- size-change-during-transition-crash.html (live test) (source)
- svg-layout-root-crash.html (live test) (source)
- svg-resource-in-container-crash.html (live test) (source)
- svg-text-crash.html (live test) (source)
- table-in-columns-000-crash.html (live test) (source)
- table-in-columns-001-crash.html (live test) (source)
- table-in-columns-002-crash.html (live test) (source)
- table-in-columns-003-crash.html (live test) (source)
- table-in-columns-004-crash.html (live test) (source)
- table-in-columns-005-crash.html (live test) (source)
- top-layer-crash.html (live test) (source)
- top-layer-nested-crash.html (live test) (source)
- custom-layout-container-001.https.html (live test) (source)
- custom-property-style-queries.html (live test) (source)
- custom-property-style-query-change.html (live test) (source)
- deep-nested-inline-size-containers.html (live test) (source)
- dialog-backdrop-create.html (live test) (source)
- dialog-backdrop-remove.html (live test) (source)
- display-contents-dynamic-style-queries.html (live test) (source)
- display-contents.html (live test) (source)
- display-in-container.html (live test) (source)
- display-none.html (live test) (source)
- fieldset-legend-change.html (live test) (source)
- flex-basis-with-container-type.html (live test) (source)
- font-relative-calc-dynamic.html (live test) (source)
- font-relative-units-dynamic.html (live test) (source)
- font-relative-units.html (live test) (source)
- fragmented-container-001.html (live test) (source)
- get-animations.html (live test) (source)
- grid-container.html (live test) (source)
- grid-item-container.html (live test) (source)
- idlharness.html (live test) (source)
- iframe-in-container-invalidation.html (live test) (source)
- iframe-invalidation.html (live test) (source)
- ineligible-containment.html (live test) (source)
- inheritance-from-container.html (live test) (source)
- inline-size-and-min-width.html (live test) (source)
- inline-size-bfc-floats.html (live test) (source)
- inline-size-containment-vertical-rl.html (live test) (source)
- inline-size-containment.html (live test) (source)
- inner-first-line-non-matching.html (live test) (source)
- layout-dependent-focus.html (live test) (source)
- multicol-container-001.html (live test) (source)
- multicol-inside-container.html (live test) (source)
- nested-query-containers.html (live test) (source)
- nested-size-style-container-invalidation.html (live test) (source)
- never-match-container.html (live test) (source)
- no-layout-containment-abspos-dynamic.html (live test) (source)
- no-layout-containment-abspos.html (live test) (source)
- no-layout-containment-baseline.html (live test) (source)
- no-layout-containment-fixedpos-dynamic.html (live test) (source)
- no-layout-containment-fixedpos.html (live test) (source)
- no-layout-containment-scroll.html (live test) (source)
- no-layout-containment-subgrid-crash.html (live test) (source)
- orthogonal-wm-container-query.html (live test) (source)
- percentage-padding-orthogonal.html (live test) (source)
- pseudo-elements-001.html (live test) (source)
- pseudo-elements-002.html (live test) (source)
- pseudo-elements-002b.html (live test) (source)
- pseudo-elements-003.html (live test) (source)
- pseudo-elements-004.html (live test) (source)
- pseudo-elements-005.html (live test) (source)
- pseudo-elements-006.html (live test) (source)
- pseudo-elements-007.html (live test) (source)
- pseudo-elements-008.html (live test) (source)
- pseudo-elements-009.html (live test) (source)
- pseudo-elements-010.html (live test) (source)
- pseudo-elements-011.html (live test) (source)
- pseudo-elements-012.html (live test) (source)
- pseudo-elements-013.html (live test) (source)
- query-content-box.html (live test) (source)
- query-evaluation-style.html (live test) (source)
- query-evaluation.html (live test) (source)
- reattach-container-with-dirty-child.html (live test) (source)
- registered-color-style-queries.html (live test) (source)
- resize-while-content-visibility-hidden.html (live test) (source)
- at-container-scrollable-parsing.html (live test) (source)
- at-container-scrollable-serialization.html (live test) (source)
- at-container-snapped-parsing.html (live test) (source)
- at-container-snapped-serialization.html (live test)