タグアーカイブ Web標準

2026年4月のBaseline新機能、contrast-color関数やsearch要素が利用可能に

2026年4月のBaseline月次ダイジェストが公開された。新たに利用可能になった機能として、CSSのcontrast-color()関数やJavaScriptのMath.sumPrecise()メソッドがある。合わせて、search要素やARIA属性リフレクションなど、すでに広く使える段階に達した機能も紹介されている。

今回のアップデートは、アクセシビリティ対応と開発効率の両面で重要な節目だ。ブラウザが自動的に最適な色を算出したり、セマンティックな構造をネイティブに解釈したりする機能が揃い、従来はカスタム実装に頼っていた領域が標準化されつつある。

この記事では、2026年4月のBaselineダイジェストの内容をもとに、新機能の具体的な使い方と、それが開発現場にもたらす変化を解説する。

Baselineとアクセシビリティをめぐる2026年の動向

Baselineとアクセシビリティをめぐる2026年の動向

web.devの記事では、A11y Upが公開した「Baseline and accessibility in 2026」という分析が紹介されている。この分析の核心は、アクセシビリティ対応をウェブ標準に委ねることで、開発の堅牢性と効率が大きく向上するという主張だ。

これまで多くの開発チームは、スクリーンリーダー対応やキーボードナビゲーションといったアクセシビリティ機能を、カスタムのJavaScript実装で再現してきた。しかし、そうした手作りのソリューションは往々にして壊れやすく、支援技術との相性問題を抱え、メンテナンスコストも高かった。

Baselineは、ある機能が主要ブラウザで相互運用可能になった時点を知らせる指標として機能する。この指標を活用すれば、開発者は標準機能への移行タイミングを判断しやすくなる。結果として、ブラウザが自動的に正しいセマンティクスをスクリーンリーダーに伝えてくれるため、開発者が手作業で調整する負担が減るというわけだ。

従来のアプローチ(カスタム実装依存)
開発者 手作りJSでアクセシビリティ実装 支援技術 誤動作・破損のリスク
⚠ メンテナンスコストが高く、壊れやすい
Baseline標準に則ったアプローチ
開発者 標準のsearch要素を使用 ブラウザ 自動でARIAロールを割り当て
✅ 堅牢でメンテナンスフリー

このデモが示すように、カスタム実装に頼る旧来の手法から、標準化された要素やAPIに移行することで、アクセシビリティの品質が安定し、開発者の負荷も低減する。

Baselineで新たに利用可能になった機能

Baselineで新たに利用可能になった機能

2026年4月の時点で、主要ブラウザ(Chrome、Firefox、Safari)すべてがサポートを開始し、Baseline newly available(新規利用可能)と位置づけられた機能が2つある。CSSのcontrast-color()関数と、JavaScriptのMath.sumPrecise()メソッドだ。

CSSのcontrast-color()関数

contrast-color()は、指定した背景色に対して最も読みやすい対照色(通常は黒か白)をブラウザが自動的に算出するCSS関数だ。動的なテーマエンジンやカスタマイズ可能なコンポーネントを扱う際、開発者がこれまで手作業で管理してきた「背景色に応じた文字色の切り替え」という負担を大幅に軽減する。

具体的な動作として、関数にベースとなる色を渡すと、ブラウザのエンジンがその色の輝度を評価し、最もコントラスト比が高い色を返す。これにより、ユーザーが好みの背景色を選べるUIでも、文字が読みにくくなる問題を自動的に回避できる。

.card-header {
  background-color: var(--dynamic-bg-color);
  /* 背景色に応じて自動的に最適な文字色が決まる */
  color: contrast-color(var(--dynamic-bg-color));
}
背景色 #1a1a2e(暗色)
contrast-color が自動で白文字を選択
コントラスト比 約15.3:1(WCAG AAA達成)
背景色 #fff8e1(明色)
contrast-color が自動で黒文字を選択
コントラスト比 約14.8:1(WCAG AAA達成)

上記のデモはcontrast-color()の概念を示したイメージだ。実際のブラウザでは、この関数が自動的に背景色を分析し、最も読みやすい文字色を適用する。中間的な明るさの背景色に対しては、ブラウザがどちらを選ぶか注意深く確認する必要があるが、大半のケースでは手動の分岐ロジックが不要になる。

Math.sumPrecise()メソッド

JavaScriptで浮動小数点数の合計を計算する際、従来のArray.prototype.reduce()や単純なループでは、丸め誤差が蓄積する問題があった。金融計算やテレメトリデータの集計といった、正確さが求められる場面ではこの誤差が致命的になることもある。

Math.sumPrecise()は、この問題に対処するために設計された静的メソッドだ。数値のイテラブル(配列など)を受け取り、精度を保ったまま安全に合計を返す。

// 従来の方法では浮動小数点誤差が発生する可能性がある
const values = [0.1, 0.2, 0.3, 0.4, 0.5, 0.6, 0.7, 0.8, 0.9, 1.0];
const preciseTotal = Math.sumPrecise(values);
// 誤差なく正確な合計値を返す

内部的には、標準化された高精度な加算アルゴリズム(Kahan summation algorithmや類似の手法)を用いて、丸め誤差を最小化する。ECサイトの売上集計や、センサーデータの分析など、正確性が重視されるシナリオで特に有効だ。

従来の Array.reduce() による加算
[0.1, 0.2, 0.3].reduce((a,b)=>a+b)
結果: 0.6000000000000001
通貨計算では誤差が問題になる
Math.sumPrecise() による加算
Math.sumPrecise([0.1, 0.2, 0.3])
結果: 0.6
正確で信頼できる

この関数を使うことで、フロントエンドでの計算結果に対する信頼性が一段上がる。特に数値の正確さがビジネス上の要件に直結するアプリケーションでは、導入を検討する価値が高い。

Baselineで広く利用可能になった機能

Baselineで広く利用可能になった機能

以下の機能は、すでに主要ブラウザで長期間サポートされ、Baseline widely available(広く利用可能)のステータスに達した。実質的にどのプロジェクトでも安心して採用できる段階だ。

search要素

HTMLのsearch要素は、検索フォームやフィルタリング機能といった、サイト内の検索体験を構成する要素群を明示的にラップするためのコンテナだ。従来はdivやformタグで代用されていたが、search要素を使うことでアクセシビリティ上の利点が生まれる。

具体的には、ブラウザがsearch要素に対して暗黙的にARIAランドマークロール「search」を割り当てる。これにより、form要素にrole=”search”を手動で付与する必要がなくなる。スクリーンリーダーのユーザーは、このランドマークを頼りに検索インターフェースへ素早く移動できる。

<search>
  <form action="/site-search">
    <label for="query">ドキュメントを検索</label>
    <input type="search" id="query" name="q">
    <button>実行</button>
  </form>
</search>
div 要素で検索エリアをマークアップ(非推奨)
<div>
<form role=”search”>…</form>
</div>
role属性の記述が必要。忘れるとアクセシビリティが損なわれる
search 要素でマークアップ(推奨)
<search>
<form>…</form>
</search>
ブラウザが暗黙的に role=”search” を付与。記述の手間と漏れがない

このシンプルな変更だけで、検索機能のアクセシビリティがワンランク向上する。既存のプロジェクトでも、該当するセクションをsearch要素に置き換えるリファクタリングを検討するとよい。

Web Authenticationの公開鍵アクセス

パスワードレス認証を実現するWeb Authentication(WebAuthn)APIにおいて、公開鍵情報の取り扱いが大幅に簡素化された。AuthenticatorAttestationResponseインターフェースに追加されたgetPublicKey()やgetPublicKeyAlgorithm()といったメソッドを使うことで、開発者が生のバイナリデータを手動で解析する必要がなくなった。

これまで公開鍵を抽出するには、CBOR(Concise Binary Object Representation)やDERエンコーディングといったバイナリ形式を手作業でパースする処理が必要だった。公開鍵の取り出しに失敗したり、アルゴリズムを誤認したりするリスクが常につきまとっていた。新しいメソッドはブラウザが直接プロパティとして公開鍵情報を提供するため、そのような低レイヤの処理が一切不要になる。

パスキー(Passkeys)の普及が加速する中、このAPIの安定化は認証フロー全体の信頼性を一段引き上げる要素だ。

String.prototype.isWellFormed()とtoWellFormed()

JavaScriptの文字列は内部的にUTF-16でエンコードされている。複雑な文字や絵文字の中には、サロゲートペアと呼ばれる2つの16ビットコード単位で表現されるものがある。文字列を途中で切断してしまうと、ペアの片方だけが残った「孤立サロゲート」という不正な文字が生まれる。

isWellFormed()は、文字列に孤立サロゲートが含まれていないかを真偽値で返すメソッドだ。toWellFormed()は、もし不正なサロゲートが見つかった場合、それをUnicodeの置換文字(U+FFFD)に置き換えた新しい文字列を返す。encodeURI()など、不正な文字列が渡されるとURIErrorをスローする関数にデータを渡す前に、これらのメソッドで検証と修正を行うのが主な用途だ。

const rawString = getUserInput();
// 不正な文字が混入していないか確認
if (!rawString.isWellFormed()) {
  // 問題があれば安全な形に修正してから処理を続行
  const cleanString = rawString.toWellFormed();
  const encoded = encodeURI(cleanString);
  // 安全にAPIリクエストなどを実行
}
不正な文字列の処理(従来)
不正文字列 encodeURI() URIError発生
アプリケーションがクラッシュするリスク
isWellFormed / toWellFormed で安全に処理
不正文字列 toWellFormed() 安全な文字列 encodeURI()
例外なく処理が完了

ユーザー入力や外部APIからのレスポンスを扱う場面では、予期せぬデータ不整合による例外発生を未然に防ぐ手段として、これらのメソッドが役立つ。

ARIA属性リフレクション

これまで、ARIA属性の値を更新するにはelement.setAttribute(‘aria-expanded’, ‘true’)のように、DOM属性を文字列で操作する必要があった。ARIA属性リフレクションは、この手順をオブジェクトプロパティへの代入に簡略化する。

ElementインターフェースがariaExpanded、ariaChecked、ariaHiddenといったプロパティを直接公開することで、ドット記法による読み書きが可能になった。これは単なるシンタックスシュガーではなく、UIフレームワークや状態管理ライブラリがアクセシビリティ状態をより正確に追跡し、スクリーンリーダーとの同期を保つうえで重要な基盤となる。

// トグルボタンのアクセシビリティ状態を簡潔に更新
toggleButton.ariaExpanded = toggleButton.ariaExpanded === "true" ? "false" : "true";
従来のsetAttributeによる操作(煩雑)
element.setAttribute(‘aria-expanded’, ‘true’)
element.getAttribute(‘aria-expanded’)
文字列操作のため、タイポや値の形式ミスのリスクがある
リフレクションによる操作(簡潔)
element.ariaExpanded = “true”
element.ariaExpanded
プロパティとして直感的にアクセス可能

ReactやVueのようなフレームワークで状態とARIA属性を紐付ける際、従来の文字列ベースの操作に比べてコードの見通しが格段に良くなる。特に複雑なUIコンポーネントを構築するチームにとって、採用するメリットは大きい。

この記事のポイント

  • contrast-color()関数で、背景色に応じた文字色の自動選択が可能になった
  • Math.sumPrecise()で浮動小数点数の正確な合計計算を実現
  • search要素が広く利用可能になり、アクセシビリティ対応が容易に
  • WebAuthnの公開鍵抽出がメソッド一発で完了するように簡略化
  • ARIA属性リフレクションで、状態管理と支援技術の同期が強化
JavaScriptの隔離実行を実現するShadowRealm APIとCSS設計への影響

JavaScriptの隔離実行を実現するShadowRealm APIとCSS設計への影響

TC39で策定が進む「ShadowRealm」APIはJavaScriptに新たな隔離実行の仕組みを持ち込む。グローバルスコープを汚染しないサンドボックス環境でコードを動かせるためサードパーティライブラリやテストコードの管理が大きく変わる可能性がある。

CSS設計の観点からもこのAPIは注目に値する。CSS-in-JSの実行や外部スクリプトによるスタイル競合といった問題に対してレルム(領域)レベルの分離が使えるようになればフロントエンド全体の堅牢性が一段上がるからだ。

本記事ではShadowRealmの基本的な考え方から具体的なAPIの使い方、そしてCSS設計との接点までを整理する。

JavaScriptのスレッドとレルム(領域)の基本

JavaScriptのスレッドとレルム(領域)の基本
従来の理解(誤解を含む)

「JavaScriptはシングルスレッド言語である」

より正確な捉え方

「ひとつのレルム(領域)はシングルスレッドだがJavaScriptアプリケーションは複数のレルムをまたいでマルチスレッド実行できる」

※Web Workersやiframeは別のレルムに属し専用のスレッドで動く。

上の図はよくある「JSはシングルスレッド」という言説が誤解を生む部分を示している。実際にはブラウザのタブひとつがひとつのレルムでありその中でJavaScriptはメインスレッドで動く。一方Web Workersは別のレルムでワーカースレッドを使いiframeもまた独自のレルムを持つ。

シングルスレッド言語の誤解

「JavaScriptはシングルスレッド」というのは「言語仕様としてマルチスレッドを提供していない」という意味では正しい。関数単位で別のスレッドにオフロードするといった仕組みはない。だがWeb Workersを使うとJavaScriptを別スレッドで実行できる。

ここで混乱が生まれる。言語がシングルスレッドでもアプリケーション全体ではマルチスレッド実行が可能だからだ。CSS-Tricksの記事ではこの点を「JavaScriptのレルムはシングルスレッド」と整理している。つまり実行環境の単位であるレルムに着目すれば言語の特性とアプリケーションの振る舞いを矛盾なく説明できる。

レルムとは何か

レルムとはコードが実行される環境そのものを指す。ブラウザタブが持つグローバルオブジェクト(window)や組み込みオブジェクト(ArrayObjectなど)はすべてそのレルムに紐づく。同じページ内のiframeであっても別のレルムでありwindowArrayも別物だ。

たとえば親ページで定義したグローバル関数はiframe内のレルムからは見えない。逆も同様だ。こうした隔離は意図しない干渉を防ぐ一方でレルム間の連携にはpostMessageなど特定の手段が必要になる。

グローバルスコープ汚染の現実とCSSへの影響

グローバルスコープ汚染の現実とCSSへの影響
Before(隔離なし)
広告スクリプトA
window.themeColor = '#ff0000';
自社スクリプトB
document.body.style.color = window.themeColor;
※意図しない赤色が適用される
After(ShadowRealmで隔離)
広告スクリプトA(ShadowRealm内)
globalThis.themeColor = '#ff0000';
自社スクリプトB(メインレルム)
document.body.style.color = 'initial';
※影響を受けない
■ 広告スクリプトA ■ 自社スクリプトB ※破線枠はShadowRealm内部を表す

JavaScriptのグローバルスコープは簡単に汚染される。変数のvar宣言の巻き上げや不用意なグローバル変数の追加に加えてサードパーティの広告タグやアナリティクスツール、A/Bテスト用スクリプトが暗黙のうちにグローバル空間へ値を書き込む。これがCSSにまで波及することがある。

サードパーティスクリプトがもたらすCSSの衝突

広告配信スクリプトがwindow.themeColorを書き換えればそれを参照している自前のCSS-in-JSロジックが意図しないスタイルを適用してしまう。また外部ウィジェットがページ全体のfont-sizeを動的に変更すればレイアウトが崩れる原因になる。

こうした問題はグローバルスコープを共有するからこそ起こる。完全に隔離されたレルムで外部コードを動かせれば変数の上書きやオブジェクトの改変は発生せず結果としてCSSの意図しない変化も防げる。

CSS-in-JSにおける隔離の必要性

CSS-in-JSライブラリではスタイルの計算結果をJavaScriptのオブジェクトや変数として管理するケースが多い。テーマ切り替えや動的スタイル生成にグローバル変数を使っていると外部スクリプトがそれらを上書きするリスクがある。ShadowRealmを使ってスタイル計算を隔離すれば外部からの干渉を受けない安全なスタイル生成が可能になる。

ShadowRealm APIの仕組み

ShadowRealm APIの仕組み

TC39で策定中のShadowRealmはコードを独立したグローバル環境で実行するためのAPIだ。このAPIには大きく分けてふたつの機能がある。evaluate()で任意のJavaScript文字列を評価する方法とimportValue()でモジュールを動的インポートしてエクスポートされた値だけを安全に受け取る方法だ。

基本的な使い方

// ShadowRealmインスタンスを作成
const shadow = new ShadowRealm();

// 外側のレルムでグローバル関数を定義
function globalFunction() {}

console.log( globalThis.globalFunction );
// 結果: function globalFunction()

// ShadowRealm内で同じ関数を評価しても未定義
console.log( shadow.evaluate( 'globalThis.globalFunction' ) );
// 結果: undefined
外側のレルム(メインスレッド)
globalThis.globalFunction → function globalFunction()
ShadowRealm内(同じメインスレッド上)
globalThis.globalFunction → undefined
※ShadowRealmは独自のグローバルオブジェクトと組み込みオブジェクトを持つがスレッドは共有する。

コードが示すようにShadowRealmインスタンス内部のグローバル環境は外側から完全に切り離されている。しかもこのコードはメインスレッド上でそのまま実行されるためスレッド間通信のコストや複雑さを伴わない。

CSS-Tricksの記事ではこの性質を「無限に使える使い捨てのクリーンルーム」と表現している。テストのモックデータが本番のグローバル変数と衝突する心配もなくサードパーティコードを安全に評価できる環境だ。

importValueによるモジュールの隔離実行

// spookycode.js
export function greeting() {
  return "Hello from the ShadowRealm!";
}

// メイン側
async function shadowGreeter() {
  const shadow = new ShadowRealm();
  const shadowGreet = await shadow.importValue(
    "./spookycode.js",
    "greeting"
  );
  shadowGreet(); // "Hello from the ShadowRealm!"
}
shadowGreeter();
モジュールソース(spookycode.js)
export function greeting() { ... }
ShadowRealmでimportValueして関数を受け取る
const fn = await shadow.importValue("./spookycode.js", "greeting");
外側のレルムで安全に実行
fn(); // "Hello from the ShadowRealm!"
※モジュール内部のグローバルは外側から隔離されており意図しない干渉は起きない。

importValue()はPromiseを返し第二引数で指定したエクスポート名の値だけを取り出す。モジュールの内部実装はShadowRealmの隔離環境で動くため外側のグローバル空間を一切汚さない。CSS-in-JSで使うテーマ計算モジュールなどをこの形で読み込めばグローバル変数を介したスタイルの競合を根本から断てる。

ShadowRealmが変えるCSS設計の安全性

ShadowRealmが変えるCSS設計の安全性
Before(グローバル共有)
テーマ変数: window.currentTheme = 'dark';
※広告スクリプトが window.currentTheme = 'light'; に上書き
After(ShadowRealmでテーマ管理を隔離)
テーマはShadowRealm内のモジュールで完結
※外部スクリプトからはアクセス不能。
※テーマ管理をShadowRealmに隔離することで予期せぬ上書きからCSS変数を守れる。

ShadowRealmはまだブラウザに実装されていないがCSS設計の安全性を高めるうえでいくつかの具体的な使い道が考えられる。

スタイルの衝突防止とテーマ分離

複数の独立したコンポーネントやマイクロフロントエンドがひとつのページに同居するケースではそれぞれが使うCSS変数やテーマオブジェクトが衝突しやすい。ShadowRealmで各コンポーネントのスタイル計算ロジックを包めばそれぞれのグローバル空間は完全に分離され変数の取り合いが起きない。

またユーザーごとにカスタマイズされたテーマを動的に生成するような仕組みでも計算ロジックをShadowRealmに閉じ込めれば他スクリプトによる意図しないテーマ書き換えを防げる。

テスト環境の清浄化

CSS-in-JSの単体テストではグローバルなスタイルシートの状態がテスト結果に影響を与えることがある。ShadowRealm上でテストを実行すればテストごとに完全にクリーンなグローバル環境が得られモックデータの準備も簡素化される。ブラウザテストとNode.jsテストの両方で同じ隔離機構を使える点もメリットだ。

実装状況と将来の展望

実装状況と将来の展望

ShadowRealmの提案はTC39のステージ2.7にある。これは「原則的に承認され検証段階にある」という位置づけでブラウザへの試験実装が始まれば仕様の微調整が入る可能性は残る。現時点では実際のブラウザやNode.jsで使うことはできない。

CSS-Tricksの記事でも指摘されているようにShadowRealmはセキュリティ境界ではなく完全性境界を提供するものだ。つまり悪意あるコードの実行を完全に防ぐサンドボックスではないがグローバル変数や組み込みオブジェクトを意図せず壊してしまうリスクを封じ込めるには十分な隔離性能を持つ。

ステージ3へ進み主要ブラウザが実装を始めればCSS-in-JSライブラリやテストランナー、サードパーティスクリプト管理の分野でいち早く活用が広がるだろう。

この記事のポイント

  • ShadowRealmはコードを独立したグローバル環境で実行するTC39提案のAPI
  • メインスレッド上で動くためスレッド間通信の複雑さがない
  • サードパーティスクリプトやCSS-in-JSのテーマ計算を隔離しスタイル競合を防げる
  • テストコードの実行環境としてもクリーンなグローバル空間を提供する
  • 現時点ではステージ2.7で未実装。ブラウザ対応はこれから
Chromeが同意なく4GBのAIモデルをダウンロードする問題

Chromeが同意なく4GBのAIモデルをダウンロードする問題

Google Chromeがユーザーの明示的な許可なく、約4GBに及ぶGemini Nanoのモデルデータをダウンロードしている事実が明らかになった。このデータは「Prompt API」と呼ばれる新機能のためのものだが、その配信方法と利用規約をめぐって、Web標準の専門家から強い懸念が示されている。

CSS-Tricksの記事によれば、このダウンロードはChromeの標準アップデートの一部として扱われ、ユーザーが削除してもブラウザが自動的に再ダウンロードする仕様だという。2025年5月現在、すでに多くのユーザーのデバイスに配信済みの状態だ。

Chromeが密かにダウンロードするGemini Nanoとは

Chromeが密かにダウンロードするGemini Nanoとは

Gemini NanoはGoogleが開発した軽量AIモデルで、デバイス上で直接テキスト生成や要約などのタスクを実行する。クラウドにデータを送信せず、端末のCPUやGPUのみで推論を行うため、プライバシー保護の観点では優れた設計といえる。

問題はその配信方法だ。CSS-Tricksの著者であるMat Marquis氏が指摘したところによれば、この約4GBのデータはChromeの通常アップデートの一部として、ユーザーに何の通知もなく転送される。U2のアルバムがiTunesライブラリに強制的に追加された過去の事例になぞらえ、同意なき配信の奇妙さを強調している。

従来のブラウザアップデート
セキュリティパッチ
バグ修正
新機能の追加
ユーザーの期待する範囲の更新
今回のChromeアップデート
セキュリティパッチ
バグ修正
新機能の追加
4GBのAIモデルデータ
Gemini Nano(同意なし・通知なし)
ブラウザの更新とは別物の大規模データが混入
問題のダウンロード  通常のアップデート

削除してもChromeが再ダウンロードを実行するため、ユーザーに実質的な拒否権はない。Chromeの内部機能として扱われているが、実際にはブラウザに統合されたわけではなく、独立した製品が同梱されている状態に近い。Mat Marquis氏は、かつてスパイウェアとして批判されたBonzi Buddyがブラウザに同梱されていた事例を引き合いに出し、その類似性を指摘している。

Prompt APIの仕組みとGoogleの利用制限

Prompt APIの仕組みとGoogleの利用制限

Prompt APIは、Web開発者がChromeの組み込みAIモデルに直接アクセスできるJavaScript APIだ。ユーザーのデバイス上でテキストの要約、文章の言い換え、質問応答といった処理を実行できる。Chromeの開発者向けドキュメントでは、すでに1年以上前から公開されている。

このAPIを利用するには、Googleが定める「Generative AI Prohibited Uses Policy」への同意が必須となる。Mozillaが公式に懸念を表明したのは、この利用ポリシーの内容がWeb標準の原則と相容れないからだ。

Web APIに付随する利用ポリシーの問題点

MozillaのGitHub上のコメントによれば、Googleの禁止事項ポリシーは法律の範囲を超えた制限を含んでいる。具体的には、性的に露骨なコンテンツの生成や配布の禁止、誤情報や政府・民主的プロセスに関する誤解を招く主張の促進禁止などが盛り込まれている。

これらの制限はWebプラットフォームのAPIとしては異例だ。通常、ブラウザAPIは技術的な仕様のみを定義し、その使途を特定企業のポリシーで制限することはない。Mozillaは「これはWebプラットフォームにとって悪しき方向性であり、UA(ユーザーエージェント)固有の使用ルールを持つAPIが増える前例となる」と警告している。

従来のWeb API
技術仕様のみを定義
使途は開発者の自由(法律の範囲内)
ブラウザベンダーによる制限なし
Google Prompt API
技術仕様の提供
Googleの利用ポリシーへの同意が必須
企業固有のコンテンツポリシーでAPI利用を制限
Prompt API特有の制限  従来のWeb APIの標準的なあり方

この構造は、Webのオープン性を損なう可能性がある。特定のブラウザベンダーがAPIの利用条件を自由に設定できるなら、Webの相互運用性は徐々に崩れていく。Mozillaの反対表明は、単なる競合他社の立場表明ではなく、Web標準の基本原則を守るための警鐘と受け止めるべきだ。

Web標準プロセスにおけるGoogleの影響力

Web標準プロセスにおけるGoogleの影響力

Mat Marquis氏は、GoogleのWeb標準への関与姿勢を痛烈に批判している。同氏の比喩によれば「GoogleのWeb標準プロセスへの参加は、クマがキャンプに参加するようなものだ」という。つまり、表面上は協調しているように見えても、実質的には自社の都合でプロセスを支配しているという指摘だ。

Googleは「開発者のポジティブな感情」を根拠にPrompt APIの推進を正当化しようとしたが、実際に引用された場所にはポジティブな感情など存在しなかった。この矛盾した説明は、同社がWeb標準を「不可避なもの」として語る際の常套句と重なる。

ブラウザAPIとWeb APIの混同が生むリスク

ここで重要なのは、すべてのブラウザAPIがWeb APIではないという事実だ。Chromeだけが実装するAPIは、事実上の標準として扱われるリスクをはらむ。MicrosoftのIEが独自拡張で市場を支配した過去の過ちを、形を変えて繰り返す可能性がある。

Alex Russell氏が繰り返し指摘しているように、ブラウザの選択肢が限られている現状はすでに問題含みだ。その状況下でGoogleがChrome限定のAI APIを推進することは、Webの多様性をさらに損なう。ブラウザの多様性が生態系に与える影響について、CSS-Tricksでも過去に取り上げられているテーマだ。

ユーザーが取るべき対応と無効化手順

ユーザーが取るべき対応と無効化手順

この問題に対して、現時点でユーザーが取れる対応は限られている。Chromeの設定画面で「オンデバイスAI」の項目をオフにすることは可能だが、すでにダウンロードされた4GBのモデルデータを完全に削除し、再ダウンロードを防ぐ方法は提供されていない。

Chrome設定での確認手順
1. Chromeの設定メニューを開く
2. 「システム」セクションに移動
3. 「オンデバイスAI」の項目を探す
4. トグルをオフにする(デフォルトはオフの場合もあり)
※オフにしても、すでにダウンロードされたモデルデータが削除されるわけではない。Chromeの再起動やアップデートで自動的に再有効化される可能性も指摘されている。

この問題に関する報道は複数のメディアで取り上げられている。Engadgetは「Chromeがユーザーの同意なく4GBのAIファイルをダウンロード」と報じ、Cybernewsは「Chromeが我々のデバイスに静かに4GBのAIモデルをインストールしている」と警告した。Android Authorityでは、このダウンロードがスパイウェアに該当するかどうかの議論まで展開されている。

この記事のポイント

  • Google Chromeがユーザーの同意なく約4GBのGemini Nanoモデルをダウンロードしている
  • 削除しても自動的に再ダウンロードされ、実質的な拒否手段が提供されていない
  • Prompt APIの利用にはGoogle独自のコンテンツポリシーへの同意が必須で、MozillaがWeb標準の観点から反対を表明
  • ブラウザベンダー固有のAPI利用制限は、オープンなWebの原則を損なう前例となる危険性がある
  • Chrome設定の「オンデバイスAI」から機能自体はオフにできるが、データ削除の確実な方法は提供されていない
CSS Olfactive APIでウェブに「匂い」を実装する?次世代の嗅覚体験と実装方法の全容

CSS Olfactive APIでウェブに「匂い」を実装する?次世代の嗅覚体験と実装方法の全容

ウェブデザインの歴史は、視覚と聴覚をいかに豊かにするかの歴史だった。しかし、次世代のウェブ標準として「嗅覚」を制御する「CSS Olfactive API(CSS嗅覚API)」の策定が進められている。この技術により、ブラウザを通じてユーザーに特定の香りを届けることが可能になる。

現在、W3C(World Wide Web Consortium)のCSSワーキンググループでは、香りの定義方法やハードウェアとの連携について活発な議論が行われている。香りをデジタルデータとして扱うための新しいファイル形式や、CSSで香りの強度を指定する新しい単位「whf」も導入される見込みだ。

このAPIは、単なるエンターテインメントの枠を超え、ECサイトでの購買体験や、教育コンテンツの没入感を劇的に変える可能性を秘めている。本記事では、CSS Olfactive APIの仕組みから具体的な実装コード、そしてアクセシビリティへの配慮まで、現時点で判明している仕様を詳しく解説する。

ウェブ体験は「嗅覚」の領域へ:Olfactive APIとは何か

ウェブ体験は「嗅覚」の領域へ:Olfactive APIとは何か

ウェブサイトの没入感を高める試みは、静止画から動画へ、そして空間オーディオへと進化してきた。CSS Olfactive APIは、その次のステップとして「香り」をブラウザの制御下に置くことを目的としている。Olfactive(オルファクティブ)とは「嗅覚の」という意味を持つ言葉だ。

ハードウェアとAPIの連携

このAPIが機能するためには、PCやスマートフォンに接続された「香りの放出デバイス」が必要になる。かつてテーマパークの4Dアトラクションで使われていたような技術が、一般消費者向けの小型デバイスとして開発されている。あるスタートアップ企業によれば、1年以内には手頃な価格の家庭用ディフューザーが市場に投入される予定だという。

APIはこれらのデバイスを抽象化し、ブラウザが直接ハードウェアの仕様を意識することなく、標準化された命令で香りを制御できるようにする。これにより、開発者は特定のメーカーのデバイスに依存することなく、共通のCSSプロパティで嗅覚体験をデザインできる。OSレベルでのドライバ対応が進めば、USB接続やBluetooth経由でシームレスに香りが届けられるようになるだろう。

なぜ今、嗅覚なのか

嗅覚は、人間の脳において感情や記憶を司る「大脳辺縁系」に直接結びついている唯一の感覚だと言われている。視覚や聴覚よりも強く、ユーザーの感情を揺さぶり、特定の記憶を呼び起こす力がある。例えば、コーヒーショップのサイトを開いた瞬間に挽きたての豆の香りが漂えば、ユーザーの購買意欲は視覚情報のみの場合よりも格段に高まるだろう。このように、UX(ユーザーエクスペリエンス)の観点から嗅覚の活用は非常に強力な武器になる。

香りを構成する4つのファミリーと15のサブカテゴリ

香りを構成する4つのファミリーと15のサブカテゴリ

デジタルで香りを表現するために、香水業界で長年使われてきた「フレグランスホイール(香りの輪)」という概念が採用された。CSS Olfactive APIでは、このホイールをベースに香りを分類し、コードで指定できる識別子を割り当てている。

4つのメインファミリー

香りは大きく分けて以下の4つのメインファミリーに分類される。これらは、デザインにおける「プライマリカラー」のような役割を果たす基本的なカテゴリだ。

  • Floral(フローラル):花の香り。最も一般的で親しみやすい。
  • Amber(アンバー):以前はオリエンタルと呼ばれていた、官能的で温かみのある香り。
  • Woody(ウッディ):樹木や森林を思わせる、落ち着いた香り。
  • Fresh(フレッシュ):シトラスや水、草木のような爽やかな香り。

15のサブカテゴリと識別子

メインファミリーはさらに細分化され、合計15のサブカテゴリが定義されている。CSSではこれらを2文字の識別子で指定する。例えば、フローラルの中には「Soft Floral(sf)」や「Floral Amber(fa)」といった細かな違いが存在する。これらを組み合わせることで、複雑な調香を再現する仕組みだ。

「Fresh」ファミリーは特に種類が多く、柑橘系の「Citrus(ct)」、水の香りの「Water(ho)」、果物の「Fruity(fu)」など、ウェブコンテンツと親和性の高い香りが揃っている。これらの識別子は、後述する scent() 関数の引数として使用されることになる。

HTMLとCSSによる実装:<scent>要素とscent-profile

HTMLとCSSによる実装:<scent>要素とscent-profile

実装方法は、既存の <video><audio> 要素と非常によく似ている。HTMLで香りのソースを定義し、CSSで要素に香りのプロファイルを割り当てるという流れだ。

<scent>要素による外部ファイルの読み込み

香りのデータは、専用のファイル形式で提供される。現在、Google、Mozilla、そして香料メーカーの連合がそれぞれ .smll.arma.smly という形式を提案中だ。HTMLでは <scent> 要素を使い、複数のソースを指定できる。

<scent controls autosmell="none">
  <source src="forest.smll" type="scent/smll">
  <source src="forest.arma" type="scent/arma">
  <a href="forest.smll">森林の香りをダウンロード</a>
</scent>

ここで重要なのが autosmell 属性だ。動画の autoplay と同様、ユーザーの意図しないタイミングで香りが放出されるのを防ぐため、デフォルトは none に設定することが推奨されている。アクセシビリティの観点からも、勝手に匂いが出る設定は避けるべきだ。

scent-profileプロパティ

CSSでは、新しく追加される scent-profile プロパティを使用して、特定の要素に香りを紐付ける。背景画像を指定する background-image と似た感覚で利用できる。以下のデモは、CSS Olfactive APIの指定方法を視覚化したものだ。

通常の要素
香りなし
scent-profile適用
森林の香り (wo)

このデモは、要素に scent-profile: url(forest.smll); を適用した際の概念を示している。右側の要素にマウスを乗せたり、フォーカスを合わせたりした際に、デバイスから香りが放出される仕組みだ。

調香のコントロール:whf単位とscent()関数の使い方

調香のコントロール:whf単位とscent()関数の使い方

外部ファイルを読み込むだけでなく、CSS内で直接香りを合成することも可能だ。ここで使われるのが scent() 関数と、新しい単位 whf である。 whf は「Waftage High Frequency(香気拡散強度)」の略で、香りの強さを表す。

香りのブレンドと強度指定

scent() 関数には、最大5つまでのサブカテゴリ識別子を指定できる。それぞれの識別子に whf 単位の数値を添えることで、配合比率を細かく調整できる。最大値は 100whf であり、指定した合計値が100を超えた場合、ブラウザは先頭から順に処理し、100に達した時点で残りの指定を無視する仕様だ。

/* ウッディ20%、水13%、フルーティ67%のブレンド */
.orchard-in-rain {
  scent-profile: scent(wo 20whf, ho 13whf, fu 67whf);
}

/* 全体的にほのかな香りにする場合 */
.subtle-scent {
  scent-profile: scent(wo 5whf, ho 2whf, fu 14whf);
}

この whf 単位の面白い点は、単なる「比率」ではなく、放出デバイスの「出力強度」に直結していることだ。 100whf で指定すれば部屋中に広がるような強い香りに、 10whf 程度であればユーザーが顔を近づけた時にだけ感じる微かな香りになる。デザインの目的に応じて、香りの「距離感」をコントロールできるのが特徴だ。

ネストと兄弟要素の制限

香りが混ざりすぎて不快な体験になるのを防ぐため、APIには強力な制限が設けられている。1つの親要素のツリー内では、1つの scent-profile しか有効にならない。つまり、ある div に香りを設定した場合、その子要素や兄弟要素に別の香りを重ねることはできない。これにより、開発者が誤って「香りのカオス」を作り出してしまうのを防いでいる。複数の香りを切り替えたい場合は、要素同士を十分に離すか、JavaScriptで動的にプロパティを書き換える必要がある。

ユーザーへの配慮とアクセシビリティ:過剰な演出を防ぐ仕組み

ユーザーへの配慮とアクセシビリティ:過剰な演出を防ぐ仕組み

嗅覚は非常にデリケートな感覚であり、特定の人にとっては不快感やアレルギー反応を引き起こす原因にもなり得る。そのため、CSS Olfactive APIではアクセシビリティへの配慮が最優先事項として組み込まれている。

prefers-reduced-pungency メディアクエリ

視覚的な動きを抑える prefers-reduced-motion と同様に、香りの刺激を抑えるための prefers-reduced-pungency メディアクエリが導入される。ユーザーはブラウザやOSの設定で「香りを無効にする」あるいは「弱める」を選択できる。

.product-card {
  scent-profile: scent(fl 50whf, fu 50whf);
}

/* ユーザーが「刺激を抑える」設定にしている場合 */
@media (prefers-reduced-pungency: reduce) {
  .product-card {
    scent-profile: scent(fl 10whf, fu 10whf);
  }
}

/* ユーザーが「香りをオフ」にしている場合 */
@media (prefers-reduced-pungency: remove) {
  .product-card {
    scent-profile: none;
  }
}

開発者はこのメディアクエリを活用し、ユーザーの好みに合わせた最適な強度を提供しなければならない。特に公共の場やオフィスでの利用を想定すると、デフォルトで香りをオフにする設定の普及は必須といえるだろう。

嗅覚インターフェースの倫理

香りは記憶と結びついているため、悪意のあるサイトが不快な匂いを放出してユーザーにトラウマを植え付けるといった攻撃も理論上は可能だ。これを防ぐため、ブラウザベンダーは「信頼できるドメイン」のみに嗅覚APIの権限を与える、あるいはマイクやカメラのように「このサイトが香りの放出を求めています」という許可ダイアログを表示する機能を検討している。技術の進化とともに、嗅覚のプライバシーと安全性を守るガイドラインの策定が急がれている。

今後の展望と課題:ハードウェア普及と実用性の壁

今後の展望と課題:ハードウェア普及と実用性の壁

CSS Olfactive APIは非常に野心的なプロジェクトだが、普及までにはまだ高い壁がある。最大の課題は、やはりハードウェアの普及率だ。どれほど洗練されたCSSを書いても、ユーザーの元に放出デバイスがなければ意味をなさない。

ブラウザの対応状況

驚くべきことに、現在このAPIを試験的に実装しているのは、新興市場向けの「KaiOS」ブラウザのみだ。ChromeやFirefox、Safariといった主要ブラウザは、仕様の推移を慎重に見守っている段階にある。しかし、AppleがVision Proのような空間コンピューティングに注力していることを考えると、没入感を補完する要素として嗅覚が注目される日は遠くないかもしれない。

実用的なユースケースの模索

単なる「お遊び」に終わらせないためには、実用的なメリットを示す必要がある。例えば、火災報知器と連動した「焦げ臭い匂い」のウェブ通知や、アロマセラピーの遠隔体験、あるいは歴史教育における「当時の街の匂い」の再現など、社会に役立つ応用例が期待されている。香りを「情報」として伝達する手段が確立されれば、ウェブの可能性は文字通りもう一つの次元へと広がるだろう。

この記事のポイント

  • CSS Olfactive APIは、ブラウザを通じて香りを制御するための新しいWeb標準である。
  • 香りは「Floral」「Amber」「Woody」「Fresh」の4ファミリーと15のサブカテゴリで定義される。
  • scent-profile プロパティと whf 単位により、香りの種類と強度をCSSで指定できる。
  • prefers-reduced-pungency メディアクエリにより、ユーザーが香りの強度を制御できるアクセシビリティが確保されている。
  • 実用化には専用ハードウェアの普及と、安全に利用するためのセキュリティガイドラインが不可欠だ。
CSSでここまでできる!カスタマイズ可能なselect要素で作る革新的UIデザイン3選

CSSでここまでできる!カスタマイズ可能なselect要素で作る革新的UIデザイン3選

HTMLのselect要素は、長年にわたりWeb制作者にとってスタイリングが最も困難なパーツの一つであった。ブラウザごとに異なるデフォルトの見た目を持ち、ドロップダウン部分に至ってはCSSでの制御がほぼ不可能だったからだ。

しかし現在、Chromium系ブラウザを中心に「カスタマイズ可能なselect要素(Customizable Select)」という新機能の実装が進んでいる。この機能により、開発者はJavaScriptで独自のコンポーネントを自作することなく、標準のselect要素に対して自由なデザインを適用できるようになった。

本記事では、CSS-Tricksで紹介された先進的なデモを基に、この新機能がもたらす可能性と具体的な実装手法を解説する。従来の常識を覆すような、扇形や円形のセレクトメニューがどのように構築されているのか、その核心に迫る。

1. appearance: base-select によるスタイリングの解禁

カスタマイズ可能なselect機能を利用するための第一歩は、新しいCSSプロパティの値を適用することだ。元記事の著者であるPatrick Brosset氏は、この機能を有効化することで、select要素全体(ボタン、ドロップダウン、オプション)のフルカスタマイズが可能になると指摘している。

制御を可能にする「base-select」の宣言

まず、select要素とそのドロップダウン部分(ピッカー)に対して、`appearance: base-select` を指定する。これにより、ブラウザの標準的なスタイルがリセットされ、開発者が自由にスタイルを定義できる土台が整う。ピッカー部分は `::picker(select)` という新しい擬似要素で指定する仕組みだ。

/* カスタマイズ機能を有効化する基本設定 */
select,
::picker(select) {
  appearance: base-select;
}

この宣言がない場合、select要素は従来通りの制限されたスタイルしか適用できない。この「オプトイン(明示的な有効化)」方式により、既存のWebサイトのデザインを壊すことなく新機能を提供できる設計となっている。

擬似要素による構成パーツの個別操作

新機能では、これまでアクセスできなかったパーツも擬似要素として操作できる。例えば、右側に表示される矢印アイコンは `::picker-icon`、選択状態を示すチェックマークは `::checkmark` という擬似要素で制御可能だ。

著者のBrosset氏は、デモにおいて `::picker-icon` を `display: none` で非表示にし、代わりに独自のアイコンや装飾を施している。また、これまではoption要素の中にテキストしか入れられなかったが、新しい仕様ではspanやimgといったHTMLタグを混在させることも可能になった。これはUIデザインの幅を大きく広げる重要な変更だ。

2. フォルダが扇状に広がるUIの構築

最初のデモとして紹介されているのは、フォルダのリストが扇状に展開されるユニークなセレクトメニューだ。このUIを実現するために、最新のCSS関数が効果的に活用されている。

sibling-index() による動的な回転制御

各フォルダ(option要素)を少しずつ異なる角度で回転させるために、`sibling-index()` という関数が使われている。これは、兄弟要素の中での自身のインデックス番号を返す関数だ。例えば、1番目の要素なら1、2番目なら2を返す。

記事によれば、このインデックス番号に一定の角度(例:-4度)を掛け合わせることで、リストの下に行くほど回転角が大きくなる「扇状」のレイアウトを数行のコードで実現している。これは従来、JavaScriptでループ処理を行ってインラインスタイルを付与していた作業だ。

option {
  --rotation-offset: -4deg;
  /* インデックス番号に応じて回転角を計算 */
  rotate: calc(sibling-index() * var(--rotation-offset));
  /* 右側を支点にして回転させる */
  transform-origin: right calc(sibling-index() * -1.5rem);
}
Folder 1 (Index 1)
Folder 2 (Index 2)
Folder 3 (Index 3)

このデモは、sibling-index()を利用して要素ごとに異なる角度を適用するイメージを示している。

@starting-style で実現する登場アニメーション

メニューが開いた瞬間にアニメーションさせる際、課題となるのが「要素が表示された瞬間の初期状態」の定義だ。通常、displayプロパティが none から block に変わる瞬間はトランジションが効かない。

そこで著者は `@starting-style` という新しいアットルールを使用している。これは要素がレンダリングを開始する直前のスタイルを指定できるもので、これにより「閉じた状態(回転0度)」から「開いた状態(扇状)」への滑らかなアニメーションが可能になる。

3. トランプのデッキを再現するカードピッカー

二つ目のデモは、トランプのカードを扇形に並べたカードピッカーだ。ここでは、配置の自由度を極限まで高めるためのテクニックが紹介されている。

アンカーポジショニングによる配置の自由化

デフォルトのselect要素は、ボタンの真下にリストが表示される。しかし、カスタマイズ可能なselectでは「アンカーポジショニング(Anchor Positioning)」という技術が組み込まれており、リストの表示位置を自由に制御できる。

著者の手法では、`position-area: center center` を使用して、ドロップダウンをボタンの中央に重ねて表示させている。さらに `inset: 0` を指定することで、ピッカーが画面全体のスペースを利用できるように設定している。これにより、ボタンの枠に縛られないダイナミックなレイアウトが可能になる。

@property を活用したカスタムプロパティのアニメーション

カードが広がるアニメーションをより精密に制御するため、`@property` を使って独自のCSS変数を定義している。CSS変数は通常、単なる文字列として扱われるため数値的な補間(アニメーション)ができないが、`@property` で型(この場合は “)を指定することで、ブラウザが変数そのものをアニメーションさせることが可能になる。

@property --card-fan-rotation {
  syntax: '<angle>';
  inherits: false;
  initial-value: 7deg;
}

option {
  /* 変数自体をアニメーション対象にする */
  transition: --card-fan-rotation 0.2s ease-out;
}

この手法により、カードの広がり具合を一つの変数で管理し、CSSのみで滑らかな動きを実現している。JavaScriptによる座標計算は一切不要だという点は、パフォーマンスの観点からも非常に優れている。

4. 三角関数を用いた円形絵文字ピッカー

最後のデモは、ボタンを中心に絵文字が円形に並ぶ「ラジアルメニュー(放射状メニュー)」だ。近年のCSSに追加された数学関数の威力が発揮されている。

sin() と cos() による座標計算

要素を円形に配置するには、角度からX座標とY座標を導き出す必要がある。以前はSassの関数やJavaScriptで行っていた計算だが、現在のCSSでは `sin()`(正弦)と `cos()`(余弦)が直接使用できる。

記事によれば、`sibling-index()` で得たインデックス番号を基に角度(–angle)を算出し、それを `translate` プロパティの中で三角関数に渡している。これにより、各option要素が中心から一定の半径(–radius)の位置に自動的に配置される仕組みだ。

option {
  position: absolute;
  --angle: calc((sibling-index() - 2) * (360deg / (sibling-count() - 1)) - 90deg);
  top: 50%;
  left: 50%;
  /* 三角関数で円周上の座標を決定 */
  translate:
    calc(-50% + cos(var(--angle)) * var(--radius)) 
    calc(-50% + sin(var(--angle)) * var(--radius));
}
😊
🚀
🔥
💡

三角関数を利用して、中心のボタンの周囲に要素を円形配置するレイアウトのデモ。

アクセシビリティの継承

これほどまでに見た目が変化しても、ベースは標準のselect要素である。著者は、マウス操作だけでなくキーボードによる選択や、スクリーンリーダーによる読み上げといったブラウザ標準のアクセシビリティ機能がそのまま維持される点を強調している。

独自にUIコンポーネントを自作する場合、これらのアクセシビリティ対応をゼロから実装するのは非常に困難でミスが起きやすい。標準要素を拡張するこのアプローチは、堅牢なWebサイト制作において極めて合理的な選択と言える。

5. 実務での導入とプログレッシブ・エンハンスメント

非常に魅力的な新機能だが、現時点での導入には注意も必要だ。著者のPatrick Brosset氏も述べている通り、この機能はまだChromium系ブラウザの最新バージョンに限定された実装である。

ブラウザ互換性とフォールバック戦略

未対応のブラウザ(SafariやFirefoxなど)では、`appearance: base-select` が無視される。その結果、ユーザーには「標準的な、見慣れたセレクトボックス」が表示されることになる。

これは「プログレッシブ・エンハンスメント(段階的向上)」の考え方に合致する。最新ブラウザを使うユーザーにはリッチな体験を提供し、そうでないユーザーにも基本機能を損なうことなくコンテンツを届けることができる。著者は、デモの中で `@supports` を使って未対応ブラウザ向けのメッセージを表示する工夫も凝らしている。

Web制作における今後の展望

カスタマイズ可能なselect要素が一般化すれば、重い外部ライブラリに頼ることなく、ブランドイメージに合わせたUIが構築可能になる。特に、フォームのデザイン性を重視するキャンペーンサイトや、複雑なオプション選択が必要なECサイトにおいて、その価値は計り知れない。

今後の課題は、他のブラウザエンジン(WebKit, Gecko)での実装状況だ。クロスブラウザでの対応が進めば、Web制作のワークフローにおいて「セレクトボックスのカスタマイズ」はもはや悩みの種ではなく、クリエイティビティを発揮する場へと変わるだろう。

この記事のポイント

  • appearance: base-select を宣言することで、select要素の全パーツをCSSで制御可能になる。
  • sibling-index() や sibling-count() を使うと、要素の順序に基づいた動的なレイアウトがノーコードで実現できる。
  • Anchor Positioning により、ドロップダウンメニューをボタンの周囲の好きな場所に配置できる。
  • 三角関数(sin, cos) を活用すれば、円形などの複雑な配置もCSSのみで完結する。
  • 未対応ブラウザでは標準のselectとして動作するため、プログレッシブ・エンハンスメントとして導入しやすい。

出典

  • CSS-Tricks「Abusing Customizable Selects」(2026年3月11日)