Questioning Container Queries
Earlier last month, Ethan wrote about the need for container queries, a request that is frequently raised in any discussion about the future of responsive design. Ethan’s article is well worth a read, especially if you’re unfamiliar with the issue, but here’s a brief summary: the dimensions of the viewport are a poor proxy for judging layout requirements for components that can exist in a variety of contexts. If we could query the dimensions of a component’s parent container instead, our layouts would be more robust, and our code easier to maintain.
The need for this feature has long been evident. In fact, my erstwhile colleague Andy Hume wrote about this problem as far back as 2011. The growing adoption of pattern-driven approaches and modularised design has only seen this need grow more urgent. Ethan is just one of many voices calling for this – or something like this – to be implemented by browser vendors.
But, I’m not so sure.
Returning to the canon
Much has been written since Ethan first introduced the world to responsive design. While some articles now appear dated, others still stand up to scrutiny. A New Canon is one such piece. Based on an earlier talk, Mark Boulton suggested we move away from traditional graphic design approaches that derive layouts inwards from a fixed canvas, and instead look to create those that work outwards, from the content:
We stop trying to create a page where there isn’t one, and we welcome what makes the web, weblike: fluidity. We start creating the connectedness Tschichold talked about by looking at what is knowable; our content.
Relating our layouts to the viewport rather than an arbitrarily-sized fixed container is a step in the right direction, but it still amounts to a canvas-in approach. While container queriers would allow us to relate components to their immediate parents, the prescription remains outside-in, privileging containers over the content that sits within them.
Two years after he wrote about container queries, at Responsive Day Out in 2013, Andy said:
Media queries are the biggest distraction in responsive design. In my view, if we’re doing responsive design right, if the specs and browsers evolve in the correct way, in five years time we won’t be using media queries to implement responsive layouts.
Four years on and I’d say his prediction wasn’t far off. Andy went on to speculate that container queries and web components would herald the end of media queries. In fact, fluid typography – a technique that cleverly combines viewport units with calc()
– was the first nail in the coffin. I believe their eventual abandonment will come from an increased proficiency with new CSS layout methods.
We have the technology
In the almost three decade history of the web, a vast array of hacks – tables, absolute positioning, floats – have been employed to help us construct layouts, yet only recently have we been given instruments designed specifically for the job. With Flexbox, we have a powerful tool for defining one-dimensional layouts, and with the arrival of Grid, we now have a second tool that can help us create layouts in two dimensions.
Without needing hacks, frameworks that abstract away their complexity will become redundant. The ability to define layouts independent of source order will mean we can discard our 16-column comfort blankets. I share the concerns of Jen and Rachel, who fear web developers will port their existing thinking over to these new layout methods. To do so would be sorry waste of a set of impressively well-considered specifications.
I’ve written previously about how support for logical properties is built right into Flexbox (and Box Alignment) specifications. No more having to replace every occurrence of float: left
with float: right
to support right-to-left languages: just change the value of a dir
attribute, and you’re good to go.
Grid builds further on this less assumptive approach. It allows us to query the nature of a grid container’s content, with functions like fit-content()
and values like min-content
and max-content
. It feels like we’ve barely scratched the surface of what’s possible.
Rebuilding the recirculation module
As a way of demonstration, let’s return to one of the components Ethan described, The Toast’s recirculation module:
- By default, the list of links appears as a single column layout on small screens.
- As the screen gets wider, it moves to a two-column layout.
- We hit a three-column layout at the middle range.
- But then, once we reach the widescreen layout, the list moves into the right-hand sidebar. Once that happens, the list reverts to a two-column layout.
With Flexbox, we can build components such that their layout will adapt to the available space. Using Grid, we can name areas of a grid container, and then assign components to those areas:
View this code example on CodePen.
There’s a lot going on in this example, so here are the key parts:
-
I’ve set the
.ranked-list
container to use Flexbox for its internal layout (display: flex
), and have instructed it to wrap any child flex items (flex-wrap: wrap
). -
I’ve then set each flex item (
.ranked-list > li
) to grow and shrink with a basis of 33% (flex: 1 1 33%
). This basically says to the browser, each item should take up 33% of the containers width, but if that’s not possible, grow wider (or narrower) to fill the remaining space once you’ve redistributed the items how you see fit. -
Within each
li
, lies a.summary
item. I’ve given this amin-width
(in this case the width of the image plus some padding) which will be factored into the above calculation.
The layout of this component is therefore entirely dependent on how much space it has available to it. That addresses points 1-3. But how can we use CSS to move this component to another part of the layout when the viewport is much wider? Enter Grid Areas:
-
First, I set
.container
to use Grid for its internal layout (display: grid
), and created two named grid areas,content
andsidebar
, and placed one above the other. For wider viewports, within a media query1 I place these areas next to each other instead. -
I then assigned
.content
to thecontent
grid area, and.sidebar
to thesidebar
grid area. Done! I can change the position of those gird areas later, and the components assigned to them will relocate accordingly.
Grid is all about the container, yet the containers we’re manipulating aren’t defined by any markup; they exist solely within our CSS. It’s a very different approach to anything that has come before, and has the potential to become even more powerful as future revisions to the specification are developed.
In the past I have suggested we embrace the medium’s unpredictable nature and use its constraints to guide how we design for it. With Flexbox and Grid, those constraints are loosening.
And now for something completely different
The final reason I question the need for container queries is that a change in layout sometimes requires a change in behaviour, too. If accomplishing this involves restructuring the DOM, we’re essentially swapping one component for another.
On a recent e-commerce project I worked on, this scenario occurred on a number of occasions. On product listings, results could be narrowed by selecting options from a series of facets: size, colour, type etc. On narrow viewports, we decided to show these options within a model dialog, with facets presented within a series of disclosures; on wider viewports however, these facets appeared within a set of menus above the results:
Another example appeared on the product item page. On narrower viewports, a gallery of images was presented as a swipeable carousel. On larger viewports, the increased screen estate meant not only could the selected image be shown larger, but that image could be tapped on to reveal a full-screen overlay within which a user could zoom-in to see more detail. Granted, both of these examples adapted their design based on the size of the viewport, but conceptually the same changes in behaviour might apply if their size was dependant on the dimensions of a parent container.
This particular project was implemented using React, so we chose to render the narrow component on the server, and then if necessary, swap it out with the wider version on the client. Therefore, any viewport queries existed within the realm of JavaScript, not CSS.
So to recap, here are three reasons why I’m not sold on the need for container queries:
- While no longer canvas-in, they still provide an inwards approach when we should be designing from the content out.
- New CSS layout primitives give us tremendous power to make component-level layout adjustments already. We should build on existing specifications and concepts rather than define new ones.
- Often, when components are resized, they need to change their behaviour as well as their presentation; container queries would not help us in that respect, so perhaps other tools are better suited to this job.
This may be a minority opinion, and it’s one I’m happy to be challenged on. After all, wiser minds have advocated for container queries. This is not the first time I have found myself at odds with perceived wisdom; maybe I’m missing a use case for which they would provide the most suitable solution.
In my mind, container queries appear to be yesterday’s answer to today’s problems. I’d much rather we use the great new tools we have, and embrace a future that’s finally here.
-
I initially thought I could use a combination of auto-placement and implicitly-created grid tracks to achieve this layout without the need for media queries – is this possible? ↩︎