Container queries
Container queries help make a UI module responsive. A container is an ancestor element that contains the element that you are trying to size. It usually contains some sort of sizing or background color that defines the region of the page.
There are two kinds of container queries:
- container size queries: Modify the styles of elements based on the width of their container element
- container style queries: Modify styles of elements based on a custom property of the container
In some cases, a module does not fit nicely on the page at specific screen sizes, which might cause you to create multiple media queries to address any issues. This violates a key principle of modular design: design your module without concern for the context that you might use it in. You can replace the multiple media queries with container queries.
Container size queries
- Define the container
- Query the container with the
@container
rule
This example gives the container a name and specifies that you will query it based on its inline size or width:
/* 1. Define the container */
container-name: layout;
container-type: inline-size;
/* equivalent to this: */
container: layout / inline-size;
After you define the query, you use the @container
syntax to query its width and size any element within the container accordingly, much like a @media
query:
@layer layout {
.l-page {
margin-inline: 1rem;
}
.l-page > * { /* define the container */
container: layout / inline-size;
}
...
}
@layer modules {
...
@container (width >= 450px) { /* create the container query */
.media {
display: flex;
gap: 1.5rem;
}
.media__image {
align-self: start;
margin-inline: revert;
}
}
}
Container types
Container types work because CSS uses containment to isolate a subsection of the DOM from the rest of the DOM.
- This means that you cannot use a container query to alter the size of the container being queried. This prevents this scenario: You’re targeting an
h1
in a container <= 300px, then the font increases the container to 301px and then the container query no longer applies - To prevent this, you have set a value for the
container-type
for the element to one of these values:normal
: default, the element is not a containerinline-size
: most practical usage, lets you query a container only on its inline size (width), not height. The height is determined by the contents of the container. Normally, the element width fills the available space, so this is the width you need to query against. For flex items, you need to setflex-basis
orflex-grow
or the container width is0
.size
: has full-size containment in both block and inline directions. The height of the container is not determined by the contents. You need to setheight
ormin-height
explicitly, or use grid or flexbox to define the height. For absolute- or fix-positioned elements, you can establish the height with theinset
property. If you don’t use any of these, the height is0
and you will have issues.
Container names
Assign a container name so you can choose which container to query. You can assign unique names, or you can reuse the same name for multiple containers.
- You can reuse a name like
layout
. This usually applies to the container you want to query, because a container query queries against the closest ancestor element assigned a container name. - Assign multiple names, depending on the use case
- (Bad idea) Assign no name, and the browser looks up the DOM and queries the first container it finds. Can’t use
container
shorthand if you don’t assign a name.
/* multiple names, one query */
container: layout sidebar / inline-size;
Containers and modules
Good practice is to make a container when a module has a containing element that might house other modules. This approach is almost always better than working with @media
queries, but you have to be consistent.
This example makes a container out of a flex item, so it adds a flex-grow
value
<aside>
<h3>Running tips</h3>
<div>
<img />
<div class="media__body">
<!-------- container -------->
<h4>Change it up</h4>
<p>
Don't run the same every time you hit the road. Vary your pace, and vary
the distance of your runs.
</p>
</div>
</div>
</aside>
@layer modules {
.media {
padding: 1.5rem;
background-color: #eee;
border-radius: 5px;
}
.media__image {
margin-inline: auto;
}
.media__body {
container: layout / inline-size;
flex-grow: 1;
}
...;
}
Container units
Containers have their own units that are similar to viewport width and height:
cqw
: 1% of container widthcqh
: 1% of container heightcqi
: 1% of container inline sizecqb
: 1% of container block sizecqmin
: Smaller ofcqi
orcqb
cqmax
: Larger ofcqi
orcqb
You can’t use block-directional units (cqh
and cqb
) to query inline-size
containers.
This example uses container units to size images and fonts within a container. There are no container queries because you aren’t applying styles based on size, you are just using the size of the container to apply styles:
@layer modules {
...
/* image is 30% container height */
.race-detail > img {
inline-size: 100%;
max-block-size: 30cqi;
object-fit: cover;
}
/* font is 3.5% of container width, w min and max */
.race-detail__body > h3 {
font-size: clamp(1rem, 3.5cqi, 1.8rem);
font-family: Helvetica, Arial, sans-serif;
}
}
Container style queries
Container style queries respond to a container based on aspects of its style, such as custom property values that set a dark theme. All styles that inherit this custom property apply the styles in this container query:
@container style(--color-theme: dark) {
/* styles */
}
You do not have to explicitly define containers to use style queries:
- good if you want a module to appear differently, depending on the context
- not widely adopted, best if used as progressive enhancement
These rulesets change the behavior of the media object when it is in a sidebar. The layout module says, “For any l-page
element with a child element with the aside
class, set the sidebar
custom property to true
.” Then the container query says, “For any module where the sidebar
custom property is true
, apply the styles that match these selectors”:
@layer layout {
.l-page > aside {
--sidebar: true;
}
}
@layer modules {
@container style(--sidebar: true) {
.race-detail {
margin-block: 1em;
display: flex;
}
.race-detail > img {
flex: 0 0 30cqi;
}
.race-detail__body {
margin-block-start: 0;
}
}
}
These styles help you maintain modularity, because now you can apply styles to specific elements in context, regardless of what page they are on. For example, you could create these styles with this rule:
.l-page > aside .race-detail {...}
But that couples the styles to pages where the element is nested in .l-page
.
Dark mode with style queries
You can use data attributes to apply light and dark theme styles:
- Use a
prefers-color-scheme
media query to set a--color-theme
custom property. - Add buttons that add the light and dark theme data attributes.
- Use container style queries to adjust specific styles depending on which custom property is applied to the page.
@layer base {
:root {
--color-theme: light;
}
@media (prefers-color-scheme: dark) {
:root {
--color-theme: dark;
}
}
[data-theme="light"] {
--color-theme: light;
}
[data-theme="dark"] {
--color-theme: dark;
}
@container style(--color-theme: dark) {
body {
background-color: #555;
color: white;
color-scheme: dark;
}
}
}
@layer modules {
/* ... */
.race-detail {
container: layout / inline-size;
border: 1px solid #ccc;
}
@container style(--color-theme: dark) {
.race-detail {
background-color: #333;
}
}
/* ... */
.media {
padding: 1.5rem;
background-color: #eee;
border-radius: 5px;
}
@container style(--color-theme: dark) {
.media {
background-color: #5f5f5f;
}
}
/* ... */
}
Here is the Javascript to set the appropriate data-theme
attribute:
const lightThemeButton = document.querySelector("#light-theme");
const darkThemeButton = document.querySelector("#dark-theme");
lightThemeButton.addEventListener("click", () => {
document.documentElement.setAttribute("data-theme", "light");
});
darkThemeButton.addEventListener("click", () => {
document.documentElement.setAttribute("data-theme", "dark");
});