CSSネイティブネストリング実践:プリプロセッサを卒業するスタイルアーキテクチャの進化
前端工程(更新: 2026年6月19日)
CSSネイティブネストリング構文
CSS Nestingは、子セレクタを親セレクタの内側に直接ネストでき、プリプロセッサなしでスタイルのスコープ組織化を実現します。
.card {
padding: 1rem;
border-radius: 8px;
background: #fff;
.title {
font-size: 1.25rem;
font-weight: 600;
}
.body {
margin-top: 0.5rem;
color: #666;
}
&:hover {
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1);
}
}
コンパイル結果は以下と等価:
.card { padding: 1rem; border-radius: 8px; background: #fff; }
.card .title { font-size: 1.25rem; font-weight: 600; }
.card .body { margin-top: 0.5rem; color: #666; }
.card:hover { box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1); }
Sass/Lessネストリングとの重要な違い
| 特徴 | CSSネイティブネスト | Sass | Less |
|---|---|---|---|
| 実行時 | ブラウザネイティブ | コンパイル時 | コンパイル時 |
&の使用 |
明示的に使用必須 | 省略可能 | 省略可能 |
| ネストルール | &またはセレクタで開始必須 |
任意のネスト | 任意のネスト |
:is()ラッピング |
自動暗黙ラッピング | なし | なし |
| 詳細度への影響 | :is()が最高を取る |
影響なし | 影響なし |
@mediaネスト |
✅ | ✅ | ✅ |
@layerネスト |
✅ | ❌ | ❌ |
| 依存関係 | なし | Ruby/Dart | Node.js |
&ネストセレクタ詳解
基本的な使い方
.btn {
color: blue;
&:hover { color: darkblue; }
&:active { color: navy; }
&:focus-visible { outline: 2px solid blue; }
&.primary { background: blue; color: white; }
&.disabled { opacity: 0.5; pointer-events: none; }
}
BEMパターンの代替
.block {
background: #f5f5f5;
&__element {
padding: 1rem;
&--modifier {
padding: 2rem;
background: #e0e0e0;
}
}
&--featured {
border: 2px solid gold;
}
}
複合セレクタ内の&の位置
.card {
/* & が前(デフォルト) */
& .title { font-size: 1.25rem; }
/* & が後 — 逆参照 */
.wrapper & { margin: 0 auto; }
/* & が中間 */
.sidebar & .icon { width: 16px; }
}
:is()暗黙ラッピングと詳細度
CSSネイティブネストは内部的に:is()ラッピングを使用し、詳細度の計算に影響します。
詳細度の罠
/* ネイティブネスト */
#main .content {
.item { color: red; }
}
/* 等価:#main .content .item → 詳細度 (1,1,1) ✅ */
div .content {
.item { color: blue; }
}
/* 等価::is(div) .content .item → 詳細度 (0,1,1) ⚠️ */
/* Sassコンパイル結果と比較 */
div .content .item { color: blue; }
/* 詳細度 (0,2,1) — ネイティブネストとは異なる! */
詳細度ルール
:is(#main, .content) {
.item { color: red; }
}
/* :is()は引数の中で最も高い詳細度を取る → (1,0,0) */
/* 最終:#main .item または .content .item → 詳細度 (1,1,0) */
@layerカスケードレイヤー内のネスト
@layer base, components, utilities;
@layer base {
html {
font-size: 16px;
line-height: 1.6;
body {
margin: 0;
font-family: system-ui;
}
}
}
@layer components {
.card {
padding: 1rem;
border: 1px solid #ddd;
.title {
font-size: 1.25rem;
}
&:hover {
border-color: #999;
}
@media (min-width: 768px) {
padding: 1.5rem;
.title { font-size: 1.5rem; }
}
}
}
@layer utilities {
.text-center { text-align: center; }
}
カスケードレイヤーの優先度
レイヤーなしのスタイル > utilities > components > base
(後に宣言されたレイヤーが高い優先度、レイヤーなしのスタイルが最高)
メディアクエリのネスト
.container {
width: 100%;
padding: 1rem;
@media (min-width: 768px) {
width: 720px;
padding: 2rem;
.sidebar {
display: block;
width: 240px;
}
}
@media (min-width: 1024px) {
width: 960px;
.sidebar {
width: 300px;
}
}
}
実践:コンポーネントスタイルアーキテクチャ
.form-field {
display: flex;
flex-direction: column;
gap: 0.25rem;
label {
font-size: 0.875rem;
font-weight: 500;
color: #374151;
}
input, textarea, select {
padding: 0.5rem 0.75rem;
border: 1px solid #d1d5db;
border-radius: 6px;
font-size: 1rem;
&:focus {
outline: none;
border-color: #3b82f6;
box-shadow: 0 0 0 3px rgba(59, 130, 246, 0.15);
}
&:disabled {
background: #f3f4f6;
cursor: not-allowed;
}
}
.error-msg {
color: #ef4444;
font-size: 0.75rem;
&:empty { display: none; }
}
&.has-error {
input, textarea, select {
border-color: #ef4444;
&:focus {
box-shadow: 0 0 0 3px rgba(239, 68, 68, 0.15);
}
}
}
@media (prefers-color-scheme: dark) {
label { color: #d1d5db; }
input, textarea, select {
background: #1f2937;
border-color: #4b5563;
color: #f9fafb;
}
}
}
実践:テーマシステム
:root {
--color-primary: #3b82f6;
--color-surface: #ffffff;
--color-text: #1f2937;
--radius: 8px;
}
[data-theme="dark"] {
--color-primary: #60a5fa;
--color-surface: #1f2937;
--color-text: #f9fafb;
}
.button {
padding: 0.5rem 1rem;
background: var(--color-primary);
color: white;
border-radius: var(--radius);
border: none;
cursor: pointer;
&:hover { filter: brightness(1.1); }
&:active { filter: brightness(0.95); }
&--sm { padding: 0.25rem 0.5rem; font-size: 0.875rem; }
&--lg { padding: 0.75rem 1.5rem; font-size: 1.125rem; }
&--outline {
background: transparent;
border: 2px solid var(--color-primary);
color: var(--color-primary);
&:hover { background: var(--color-primary); color: white; }
}
}
移行戦略:プリプロセッサからネイティブCSSへ
段階的移行
/* フェーズ1:プリプロセッサを保持、ネイティブネストと併用 */
.container {
max-width: 1200px;
// Sassの単行コメントはネイティブCSSで無効、/* */に置き換え
.header {
/* コメントはブロックコメントに変更 */
display: flex;
}
}
/* フェーズ2:変数宣言を削除、CSSカスタムプロパティに変更 */
:root {
--spacing-sm: 0.5rem;
--spacing-md: 1rem;
--spacing-lg: 1.5rem;
}
/* フェーズ3:mixinを削除、@layer + ネイティブネストに変更 */
@layer reset {
* { margin: 0; box-sizing: border-box; }
}
移行時の注意点
| プリプロセッサ機能 | ネイティブCSS代替 | ステータス |
|---|---|---|
| ネスト | CSS Nesting | ✅ サポート済み |
| 変数 | CSS Custom Properties | ✅ サポート済み |
@mixin / @include |
直接の代替なし | ❌ リファクタリング必要 |
@extend |
直接の代替なし | ❌ リファクタリング必要 |
@function |
直接の代替なし | ❌ リファクタリング必要 |
| 数値演算 | calc() |
✅ サポート済み |
| カラー関数 | color-mix() / oklch() |
✅ 部分サポート |
@import |
@import / @use |
⚠️ ビルドツール必要 |
ブラウザ互換性
| ブラウザ | サポートバージョン | 備考 |
|---|---|---|
| Chrome 120+ | ✅ | 完全サポート |
| Firefox 117+ | ✅ | 完全サポート |
| Safari 17.2+ | ✅ | 完全サポート |
| Edge 120+ | ✅ | 完全サポート |
npm install postcss-nesting
import postcss from 'postcss';
import postcssNesting from 'postcss-nesting';
const result = await postcss([postcssNesting]).process(css, { from: 'style.css' });
ベストプラクティスまとめ
- ネイティブネストを優先:新規プロジェクトで直接使用し、ビルド依存を削減
- 詳細度の違いに注意:ネイティブネストの
:is()ラッピングはSassと異なる詳細度を生成する可能性 &を明示的に使用:疑似クラスと疑似要素は&で親セレクタを参照必須- ネスト深度は3-4レベルに制限:深すぎるネストは詳細度を上げ、オーバーライドを困難にする
@layerと組み合わせ:カスケードレイヤーでグローバル優先度を管理、ネストでコンポーネント内部構造を管理
ブラウザローカルツールを無料で試す →
#CSS嵌套#原生CSS#预处理替代#选择器#样式架构