:has()セレクター
問題
CSSには、子要素に基づいて親要素を選択する方法がありませんでした。開発者は、入力が無効な場合にフォームグループをハイライトしたり、画像の有無に基づいてカードレイアウトを変更したりするなど、親子の状態関係にJavaScriptでクラスをトグルすることに頼っていました。AIエージェントが:has()を使うことはほとんどなく、代わりにこれらのパターンにJavaScriptベースのソリューションを提案します。
解決方法
:has()関係擬似クラスは、指定されたセレクターリストに一致する要素を少なくとも1つ含む要素を選択します。「親セレクター」として機能しますが、はるかに強力です:子、兄弟、子孫など、あらゆる相対的な位置を見て条件的にスタイルを適用できます。
コード例
基本的な親の選択
/* Style a card differently when it contains an image */
.card:has(img) {
grid-template-rows: 200px 1fr;
}
.card:has(img) .card-body {
padding-top: 0;
}
フォームバリデーションのスタイリング
JavaScriptなしで入力の有効性に基づいてフォームグループにスタイルを適用します。
/* Highlight the entire field group when input is invalid */
.field-group:has(:user-invalid) {
border-left: 3px solid red;
background: #fff5f5;
}
.field-group:has(:user-invalid) .error-message {
display: block;
}
/* Style label when its sibling input is focused */
.field-group:has(input:focus) label {
color: blue;
font-weight: bold;
}
<div class="field-group">
<label for="email">Email</label>
<input type="email" id="email" required />
<span class="error-message">Please enter a valid email</span>
</div>
数量クエリ
JavaScriptなしで子要素の数に基づいてレイアウトを適応させます。
/* Switch to grid layout when a list has 5 or more items */
.item-list:has(> :nth-child(5)) {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(200px, 1fr));
}
/* Single-column layout for fewer items */
.item-list:not(:has(> :nth-child(5))) {
display: flex;
flex-direction: column;
}
/* Style based on even/odd number of children */
.grid:has(> :last-child:nth-child(even)) {
/* Even number of children */
grid-template-columns: repeat(2, 1fr);
}
.grid:has(> :last-child:nth-child(odd)) {
/* Odd number of children */
grid-template-columns: repeat(3, 1fr);
}
兄弟の状態に基づくスタイリング
/* Change page layout when a sidebar checkbox is checked */
body:has(#sidebar-toggle:checked) .main-content {
margin-left: 0;
}
body:has(#sidebar-toggle:checked) .sidebar {
transform: translateX(-100%);
}
他のセレクターとの組み合わせ
/* Style a navigation item that contains the current page link */
nav li:has(> a[aria-current="page"]) {
background: #e0e7ff;
border-radius: 4px;
}
/* Style a table row that has an empty cell */
tr:has(td:empty) {
opacity: 0.6;
}
:has()と直接子結合子の使用
パフォーマンス向上のために直接子結合子>を使用しましょう。ブラウザの検索をすべての子孫ではなく直接の子要素に限定します。
/* Preferred: direct child (faster) */
.container:has(> .alert) {
border: 2px solid red;
}
/* Avoid when possible: descendant (slower on large DOMs) */
.container:has(.alert) {
border: 2px solid red;
}
ブラウザサポート
- Chrome 105+
- Safari 15.4+
- Firefox 121+
- Edge 105+
グローバルサポートは96%を超えています。機能検出は@supports selector(:has(*))で利用可能です。
AIがよくやるミス
:has()で解決できる問題にJavaScriptのクラストグルを提案する:has()の存在を知らず回避策を推奨する:has()内で直接子>の方がパフォーマンスが良い場合に子孫セレクターを使用する:has()と:not()を逆ロジックのために組み合わせない(例:.card:not(:has(img))):has()が子孫だけでなく兄弟も見ることができることを忘れる(例:h2:has(+ p)):has()のポリフィルを試みる — リアルタイムのDOM認識が必要で、効率的なポリフィルはできない
使い分け
- 子の状態に基づく親のスタイリング(フォームバリデーション、コンテンツ対応レイアウト)
- 子要素の数に基づいてレイアウトを適応させる数量クエリ
- JavaScriptなしの状態駆動スタイリング(チェックボックスハック、フォーカス管理)
- コンテンツの有無に基づく条件的なコンポーネントスタイリング
ライブプレビュー
:has(:focus) — 子のフォーカスに親が反応する
:has(:checked) — チェックボックスでスタイルを切り替える