CSS 容器查询是自媒体查询以来响应式设计中最重要的进步。媒体查询响应视口大小,而容器查询响应父容器的大小,实现了真正可复用、上下文感知的组件。本指南涵盖从基本语法到高级模式的所有内容。
什么是容器查询?
容器查询允许你根据包含元素的大小而不是视口来应用样式。组件可以根据放置位置自适应布局。
/* 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. */容器查询 vs 媒体查询
媒体查询只响应视口大小,而不是组件实际可用的空间。容器查询解决了这个根本性限制。
| 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+) |
理解包含
容器查询需要包含 — 你必须显式定义哪些元素是容器。
包含类型
/* 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-type 定义容器,用 @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 */
}
}命名容器
可以命名容器以定位特定的祖先元素,而不是最近的容器。
/* 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. */container 简写
container 简写属性将 container-type 和 container-name 合并在一个声明中。
/* 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;
}容器查询单位
容器查询引入了相对于容器尺寸的新 CSS 单位,类似于视口单位。
| 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;
}实际示例
响应式卡片组件
根据容器可用空间自适应布局的卡片组件。
/* 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;
}
}响应式导航
根据容器宽度切换水平和垂直布局的导航组件。
/* 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;
}
}自适应网格项
根据网格列宽度改变内部布局的网格项。
/* 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 — 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;
}
}嵌套容器查询
容器查询可以嵌套以创建基于多个祖先容器的复杂响应行为。
/* 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;
}
}设计模式
固有响应式设计
结合容器查询和 CSS Grid auto-fit 实现真正的固有响应式设计。
/* 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 */
.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;
}
}浏览器支持
容器查询在 2026 年有出色的浏览器支持。
回退策略
使用 @supports 提供优雅的回退。
/* 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;
}
}性能考虑
了解包含有助于避免常见陷阱。
- 包含创建新的堆叠上下文。
- 避免在深层嵌套元素上不必要地设置 container-type。
- 仅需要宽度查询时使用 inline-size。
- 容器查询不会导致额外的布局过程。
- 使用命名容器明确控制查询目标。
最佳实践
- 组件级使用容器查询,页面级使用媒体查询。
- 从 inline-size 开始。
- 命名容器以提高清晰度。
- 使用容器查询单位实现流式排版。
- 结合 CSS Grid 和 Flexbox。
- 从最小到最大设计组件。
- 使用 @supports 提供回退样式。
从媒体查询迁移
不需要替换所有媒体查询。使用此决策指南来确定使用哪种方法。
/* 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)
*/总结
CSS 容器查询从根本上改变了我们对响应式设计的思考方式。我们现在可以设计响应容器上下文的组件。结合现代 CSS 布局技术,构建更模块化、可维护和真正响应式的设计系统。
常见问题
容器查询和媒体查询有什么区别?
媒体查询响应视口大小,容器查询响应父容器元素的大小。
需要在每个父元素上设置 container-type 吗?
不需要。只在你想用作查询容器的元素上设置。
可以与 CSS Grid 和 Flexbox 一起使用吗?
可以。容器查询与它们无缝配合。
inline-size 和 size 有什么区别?
inline-size 仅在行内轴上创建包含,size 在两个轴上都创建包含。大多数情况下使用 inline-size。