CSS Container Queries represent the most significant advancement in responsive design since media queries. While media queries respond to the viewport size, container queries respond to the size of a parent container, enabling truly reusable, context-aware components. This guide covers everything from basic syntax to advanced patterns, helping you build components that adapt to their container rather than the entire viewport.
What Are Container Queries?
Container queries allow you to apply styles to an element based on the size of its containing element rather than the viewport. This means a component can adapt its layout whether it is placed in a narrow sidebar, a wide main content area, or a full-width hero section — all without JavaScript or media queries.
/* The Problem: A card component in different contexts */
/* In main content (800px wide) → horizontal layout */
/* In sidebar (300px wide) → vertical layout */
/* In footer (200px wide) → minimal layout */
/* With media queries, you cannot distinguish these contexts
because they all exist at the same viewport width.
With container queries, each container context triggers
different styles automatically. */Container Queries vs Media Queries
Media queries have been the foundation of responsive design for over a decade. However, they have a fundamental limitation: they only respond to the viewport size, not the actual space available to a component.
| Feature | Media Queries | Container Queries |
|---|---|---|
| Responds to | Viewport size | Container size |
| Scope | Global (page-level) | Local (component-level) |
| Reusability | Breaks in different contexts | Works in any context |
| Use case | Page layout | Component layout |
| Units | vw, vh, vmin, vmax | cqw, cqh, cqi, cqb |
| Browser support | Universal | All modern browsers (2023+) |
Understanding Containment
Container queries require containment — you must explicitly define which elements are containers. Containment tells the browser that the internal layout of an element is independent of its surroundings, enabling the browser to optimize rendering and calculate container dimensions.
Containment Types
/* container-type values */
/* inline-size: Containment on the inline axis only (width) */
/* Most common — use this in most cases */
.card-wrapper {
container-type: inline-size;
}
/* size: Containment on both inline and block axes (width + height) */
/* Use when you need to query both dimensions */
.widget-wrapper {
container-type: size;
}
/* normal (default): No containment — cannot be queried */
.regular-div {
container-type: normal; /* This is the default */
}
/* IMPORTANT: container-type: size prevents auto-height behavior.
The element will not grow to fit its content on the block axis.
Use inline-size unless you specifically need height queries. */Container Query Syntax
The basic syntax involves two parts: defining a container with container-type, and writing queries with @container.
/* Step 1: Define a container */
.card-container {
container-type: inline-size;
}
/* Step 2: Write container queries */
.card {
/* Default styles (smallest) */
display: grid;
gap: 12px;
padding: 16px;
}
.card__image {
width: 100%;
aspect-ratio: 16/9;
border-radius: 8px;
object-fit: cover;
}
.card__title {
font-size: 1rem;
font-weight: 700;
}
/* When container is at least 400px wide */
@container (min-width: 400px) {
.card {
grid-template-columns: 200px 1fr;
align-items: center;
}
.card__title {
font-size: 1.25rem;
}
}
/* When container is at least 600px wide */
@container (min-width: 600px) {
.card {
grid-template-columns: 280px 1fr;
gap: 24px;
padding: 24px;
}
.card__title {
font-size: 1.5rem;
}
.card__description {
display: block; /* Show description only in wide containers */
}
}Named Containers
You can name containers to target specific ancestors instead of the nearest container. This is useful when you have nested containers and need to query a specific one.
/* Name containers to target specific ancestors */
.sidebar {
container-type: inline-size;
container-name: sidebar;
}
.main-content {
container-type: inline-size;
container-name: main;
}
/* Query a specific named container */
@container sidebar (max-width: 300px) {
.widget {
font-size: 0.875rem;
}
}
@container main (min-width: 800px) {
.article-card {
grid-template-columns: 1fr 1fr;
}
}
/* Without a name, @container queries the NEAREST ancestor
with containment. Named queries skip to the named container. */The container Shorthand
The container shorthand property combines container-type and container-name in a single declaration.
/* Shorthand: container: <name> / <type> */
.sidebar {
container: sidebar / inline-size;
}
/* Equivalent to: */
.sidebar {
container-name: sidebar;
container-type: inline-size;
}
/* Multiple names (for querying by either name) */
.widget-area {
container: widget-area panel / inline-size;
}Container Query Units
Container queries introduce new CSS units relative to the container dimensions. These units work similarly to viewport units (vw, vh) but are relative to the query container.
| Unit | Description | Equivalent |
|---|---|---|
| cqw | 1% of container width | Similar to vw for viewport |
| cqh | 1% of container height | Similar to vh for viewport |
| cqi | 1% of container inline size | Width in horizontal writing |
| cqb | 1% of container block size | Height in horizontal writing |
| cqmin | Smaller of cqi and cqb | Similar to vmin |
| cqmax | Larger of cqi and cqb | Similar to vmax |
/* Fluid typography with container query units */
.card-title {
/* Font size scales with container width */
/* Minimum 1rem, maximum 2rem, scales at 5cqi */
font-size: clamp(1rem, 5cqi, 2rem);
}
.card-padding {
/* Responsive padding based on container */
padding: clamp(12px, 4cqi, 32px);
}
.hero-text {
/* Large text that scales with its container */
font-size: clamp(2rem, 8cqi, 5rem);
line-height: 1.1;
}Practical Examples
Responsive Card Component
A card component that adapts its layout based on the available space in its container.
/* Responsive card component */
.card-container {
container: card / inline-size;
}
.card {
display: flex;
flex-direction: column;
gap: 12px;
padding: 16px;
border-radius: 12px;
background: var(--card-bg);
border: 1px solid var(--border-color);
}
.card__image {
width: 100%;
aspect-ratio: 16 / 9;
object-fit: cover;
border-radius: 8px;
}
.card__meta {
display: none; /* Hidden in small containers */
}
.card__actions {
display: flex;
gap: 8px;
flex-wrap: wrap;
}
/* Medium card (400px+) — side-by-side layout */
@container card (min-width: 400px) {
.card {
flex-direction: row;
align-items: flex-start;
}
.card__image {
width: 180px;
flex-shrink: 0;
aspect-ratio: 1;
}
.card__meta {
display: flex;
gap: 12px;
font-size: 0.875rem;
color: var(--text-muted);
}
}
/* Large card (700px+) — featured layout */
@container card (min-width: 700px) {
.card {
flex-direction: column;
padding: 0;
overflow: hidden;
}
.card__image {
width: 100%;
height: 300px;
border-radius: 0;
}
.card__content {
padding: 24px;
}
.card__title {
font-size: 1.75rem;
}
}Responsive Navigation
A navigation component that switches between horizontal and vertical layouts based on its container width.
/* Navigation that adapts to its container */
.nav-container {
container: nav / inline-size;
}
.nav {
display: flex;
flex-direction: column;
gap: 4px;
}
.nav__item {
padding: 8px 12px;
border-radius: 6px;
font-size: 0.875rem;
}
.nav__item-icon {
display: none;
}
/* Horizontal layout when container is wide enough */
@container nav (min-width: 500px) {
.nav {
flex-direction: row;
gap: 8px;
}
.nav__item {
white-space: nowrap;
}
}
/* Show icons when there is more space */
@container nav (min-width: 700px) {
.nav__item-icon {
display: inline-flex;
margin-right: 6px;
}
.nav__item {
padding: 10px 16px;
}
}Adaptive Grid Items
Grid items that change their internal layout based on the grid column width.
/* Grid items that adapt to their column width */
.grid {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
gap: 24px;
}
/* Each grid item is a container */
.grid__item {
container: grid-item / inline-size;
}
.product {
text-align: center;
padding: 16px;
}
.product__price {
font-size: 1.5rem;
font-weight: 700;
}
.product__details {
display: none;
}
@container grid-item (min-width: 350px) {
.product {
display: grid;
grid-template-columns: 120px 1fr;
text-align: left;
gap: 16px;
}
.product__details {
display: block;
}
}Style Container Queries
Style container queries let you query the computed style values of a container, not just its dimensions. This allows you to create variations based on custom properties or other style values.
/* Style container queries — query computed style values */
/* Define a container with a custom property */
.theme-container {
container-type: normal; /* No size containment needed for style queries */
--theme: light;
}
.theme-container.dark {
--theme: dark;
}
/* Query the custom property value */
@container style(--theme: dark) {
.card {
background: #1a1a2e;
color: #e0e0e0;
border-color: #333;
}
.button {
background: #4a90d9;
color: white;
}
}
@container style(--theme: light) {
.card {
background: #ffffff;
color: #333;
border-color: #e0e0e0;
}
}
/* Practical use: component variants via CSS properties */
.alert-container {
--variant: info;
}
@container style(--variant: error) {
.alert {
background: #fef2f2;
border-left: 4px solid #ef4444;
color: #991b1b;
}
}
@container style(--variant: success) {
.alert {
background: #f0fdf4;
border-left: 4px solid #22c55e;
color: #166534;
}
}Nesting Container Queries
Container queries can be nested to create complex responsive behavior based on multiple ancestor containers.
/* Nested container queries */
.page {
container: page / inline-size;
}
.sidebar {
container: sidebar / inline-size;
}
/* Different behavior based on multiple container sizes */
@container page (min-width: 1200px) {
.sidebar {
width: 350px;
}
}
@container sidebar (max-width: 250px) {
.sidebar-widget {
font-size: 0.875rem;
padding: 8px;
}
.sidebar-widget__title {
font-size: 1rem;
}
}
@container sidebar (min-width: 300px) {
.sidebar-widget {
padding: 16px;
}
.sidebar-widget__chart {
display: block;
}
}Design Patterns
Intrinsic Design with Container Queries
Combine container queries with modern CSS layout techniques like CSS Grid auto-fit for truly intrinsic responsive design.
/* Intrinsic design: Grid + Container Queries */
.auto-grid {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(280px, 1fr));
gap: 24px;
}
.auto-grid > * {
container-type: inline-size;
}
/* Items automatically adapt as the grid reflows */
/* No breakpoints needed — the grid column width drives the layout */
@container (min-width: 400px) {
.feature-card {
grid-template-columns: auto 1fr;
}
}
@container (min-width: 500px) {
.feature-card__description {
font-size: 1.125rem;
}
}Dashboard Widget Pattern
A common pattern for dashboards where widgets adapt based on the grid cell they occupy.
/* Dashboard widget pattern */
.dashboard {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(300px, 1fr));
gap: 20px;
}
.dashboard__cell {
container: widget / inline-size;
}
.widget {
padding: 16px;
border-radius: 12px;
background: var(--widget-bg);
}
.widget__chart {
height: 150px;
}
.widget__table {
display: none; /* Hidden in small widgets */
}
@container widget (min-width: 400px) {
.widget__chart {
height: 200px;
}
.widget__stats {
display: grid;
grid-template-columns: repeat(3, 1fr);
gap: 12px;
}
}
@container widget (min-width: 600px) {
.widget {
padding: 24px;
}
.widget__table {
display: table;
}
.widget__chart {
height: 280px;
}
}Browser Support
Container queries have excellent browser support as of 2026. All major browsers support both size and style container queries.
Fallback Strategies
For environments that may not support container queries, use @supports to provide graceful fallbacks.
/* Graceful fallback with @supports */
.card-container {
/* Fallback: use media queries */
}
@supports (container-type: inline-size) {
.card-container {
container-type: inline-size;
}
}
/* Fallback styles using media queries */
@media (min-width: 768px) {
.card {
grid-template-columns: 200px 1fr;
}
}
/* Container query override (when supported) */
@container (min-width: 400px) {
.card {
grid-template-columns: 200px 1fr;
}
}Performance Considerations
Container queries are designed to be performant, but understanding containment helps you avoid common pitfalls.
- Containment creates a new stacking context — be aware of z-index implications.
- Avoid setting container-type on deeply nested elements unnecessarily.
- Use container-type: inline-size instead of size when you only need width-based queries.
- Container queries do not cause additional layout passes compared to equivalent media query setups.
- Use named containers to explicitly control which container is being queried.
Best Practices
- Use container queries for component-level responsiveness and media queries for page-level layout.
- Start with container-type: inline-size — you rarely need size (which includes block dimension).
- Name your containers for clarity, especially with nested containers.
- Use container query units (cqw, cqh) for fluid typography and spacing within containers.
- Combine container queries with CSS Grid and Flexbox for maximum flexibility.
- Design components from smallest to largest — mobile-first works for containers too.
- Use @supports for fallback styles when supporting older browsers.
Migrating from Media Queries
You do not need to replace all media queries with container queries. Use this decision guide to determine which approach to use:
/* Decision Guide: Container Query vs Media Query */
/* USE CONTAINER QUERIES when:
✅ Component is reused in different layout contexts
✅ Component goes in sidebars, modals, grids, etc.
✅ You need component-level responsive behavior
✅ The component's layout depends on available space
USE MEDIA QUERIES when:
✅ Changing the overall page layout
✅ Showing/hiding entire sections
✅ Adjusting global typography or spacing
✅ Changing navigation from desktop to mobile
✅ Print stylesheets
USE BOTH when:
✅ Page layout switches at viewport breakpoints (media query)
✅ Components within the layout adapt to their containers (container query)
*/Conclusion
CSS Container Queries fundamentally change how we think about responsive design. Instead of designing pages that respond to viewport sizes, we can now design components that respond to their container context. This makes components truly reusable across different layout contexts without modification. Start by identifying components in your design system that need to adapt based on available space rather than viewport width — cards, navigation, sidebars, and dashboard widgets are prime candidates. Combine container queries with modern CSS layout techniques like Grid and Flexbox, and use container query units for fluid sizing within containers. The result is a more modular, maintainable, and truly responsive design system.
FAQ
What is the difference between container queries and media queries?
Media queries respond to the viewport (browser window) size. Container queries respond to the size of a parent container element. This makes components responsive to their context, not the entire page.
Do I need to set container-type on every parent element?
No. Only set container-type on elements you want to use as query containers. The @container rule will query the nearest ancestor with containment. You only need containment on the elements you actually want to query against.
Can I use container queries with CSS Grid and Flexbox?
Yes. Container queries work seamlessly with CSS Grid and Flexbox. The container wrapping element participates in grid or flex layout normally, and child elements can use @container to respond to the container size.
What is the difference between container-type: inline-size and size?
inline-size creates containment only on the inline axis (width in horizontal writing modes). size creates containment on both inline and block axes (width and height). Use inline-size in most cases, as size containment can interfere with auto-height behavior.