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' });

ベストプラクティスまとめ

  1. ネイティブネストを優先:新規プロジェクトで直接使用し、ビルド依存を削減
  2. 詳細度の違いに注意:ネイティブネストの:is()ラッピングはSassと異なる詳細度を生成する可能性
  3. &を明示的に使用:疑似クラスと疑似要素は&で親セレクタを参照必須
  4. ネスト深度は3-4レベルに制限:深すぎるネストは詳細度を上げ、オーバーライドを困難にする
  5. @layerと組み合わせ:カスケードレイヤーでグローバル優先度を管理、ネストでコンポーネント内部構造を管理

ブラウザローカルツールを無料で試す →

#CSS嵌套#原生CSS#预处理替代#选择器#样式架构