Bootstrap によるレスポンシブ Web デザイン
ダークモードを利用する
ダークモードの指定
Bootstrap 5.3 以降は data-bs-theme 属性でテーマ (light / dark) を切り替えられるようになりました.具体的には,<html> タグの中に次のとおり data-bs-theme 属性でテーマを指定するだけです.(サンプル(1:lightモード)を見る)(サンプル(2:darkモード)を見る)
index.html(抜粋)
<!DOCTYPE html>
<html lang="ja" data-bs-theme="light">
<head>
index.html(抜粋)
<!DOCTYPE html>
<html lang="ja" data-bs-theme="dark">
<head>
Navbarの設置
次に,ダークモード/ライトモードの切り替えができるようにするために,Navbar を設置します.(サンプル(3)を見る)
index.html(抜粋)
<!DOCTYPE html>
<html lang="ja" data-bs-theme="dark">
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<title>Bootstrap demo</title>
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.2/dist/css/bootstrap.min.css" rel="stylesheet" integrity="sha384-T3c6CoIi6uLrA9TneNEoa7RxnatzjcDSCmG1MXxSR1GAsXEV/Dwwykc2MPK8M2HN" crossorigin="anonymous">
</head>
<body class="d-flex flex-column min-vh-100">
<nav class="navbar navbar-expand-lg bg-body-tertiary">
<div class="container-fluid">
<a class="navbar-brand" href="#">Navbar</a>
<button class="navbar-toggler" type="button" data-bs-toggle="collapse" data-bs-target="#navbarSupportedContent" aria-controls="navbarSupportedContent" aria-expanded="false" aria-label="Toggle navigation">
<span class="navbar-toggler-icon"></span>
</button>
<div class="collapse navbar-collapse" id="navbarSupportedContent">
<ul class="navbar-nav me-auto mb-2 mb-lg-0">
<li class="nav-item">
<a class="nav-link active" aria-current="page" href="#">Home</a>
</li>
<li class="nav-item">
<a class="nav-link" href="#">Link</a>
</li>
<li class="nav-item dropdown">
<a class="nav-link dropdown-toggle" href="#" role="button" data-bs-toggle="dropdown" aria-expanded="false">
Dropdown
</a>
<ul class="dropdown-menu">
<li><a class="dropdown-item" href="#">Action</a></li>
<li><a class="dropdown-item" href="#">Another action</a></li>
<li><hr class="dropdown-divider"></li>
<li><a class="dropdown-item" href="#">Something else here</a></li>
</ul>
</li>
<li class="nav-item">
<a class="nav-link disabled" aria-disabled="true">Disabled</a>
</li>
</ul>
<form class="d-flex" role="search">
<input class="form-control me-2" type="search" placeholder="Search" aria-label="Search">
<button class="btn btn-outline-success" type="submit">Search</button>
</form>
</div>
</div>
</nav>
<main class="container-md">
<h1>Bootstrapのサンプル</h1>
<p>
<code>.container-md</code>
</p>
<p>
ここは本文の領域です.レイアウトを確認するための文章です...
</p>
</main>
<footer class="container-fluid mt-auto">
<p>
ここはフッター領域です.ここはフッター領域です.ここはフッター領域です.ここはフッター領域です.ここはフッター領域です.ここはフッター領域です.ここはフッター領域です.ここはフッター領域です.ここはフッター領域です...
</p>
</footer>
<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.3.2/dist/js/bootstrap.bundle.min.js" integrity="sha384-C6RzsynM9kWDrMNeT87bh95OGNyZPhcTNXj1NW7RuBCsyN/o0jlpcV8Qyq46cDfL" crossorigin="anonymous"></script>
</body>
</html>
Navbar から不要なメニューを一旦削除しておきましょう.(サンプル(4)を見る)
index.html(抜粋)
<!DOCTYPE html>
<html lang="ja" data-bs-theme="dark">
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<title>Bootstrap demo</title>
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.2/dist/css/bootstrap.min.css" rel="stylesheet" integrity="sha384-T3c6CoIi6uLrA9TneNEoa7RxnatzjcDSCmG1MXxSR1GAsXEV/Dwwykc2MPK8M2HN" crossorigin="anonymous">
</head>
<body class="d-flex flex-column min-vh-100">
<nav class="navbar navbar-expand-lg bg-body-tertiary">
<div class="container-fluid">
<a class="navbar-brand" href="#">Navbar</a>
<button class="navbar-toggler" type="button" data-bs-toggle="collapse" data-bs-target="#navbarSupportedContent" aria-controls="navbarSupportedContent" aria-expanded="false" aria-label="Toggle navigation">
<span class="navbar-toggler-icon"></span>
</button>
<div class="collapse navbar-collapse" id="navbarSupportedContent">
<ul class="navbar-nav me-auto mb-2 mb-lg-0">
<li class="nav-item">
<a class="nav-link active" aria-current="page" href="#">Home</a>
</li>
</ul>
</div>
</div>
</nav>
<main class="container-md">
<h1>Bootstrapのサンプル</h1>
<p>
<code>.container-md</code>
</p>
<p>
ここは本文の領域です.レイアウトを確認するための文章です...
</p>
</main>
<footer class="container-fluid mt-auto">
<p>
ここはフッター領域です.ここはフッター領域です.ここはフッター領域です.ここはフッター領域です.ここはフッター領域です.ここはフッター領域です.ここはフッター領域です.ここはフッター領域です.ここはフッター領域です...
</p>
</footer>
<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.3.2/dist/js/bootstrap.bundle.min.js" integrity="sha384-C6RzsynM9kWDrMNeT87bh95OGNyZPhcTNXj1NW7RuBCsyN/o0jlpcV8Qyq46cDfL" crossorigin="anonymous"></script>
</body>
</html>
さらに,モードの切り替えのためのドロップダウンメニューを設置します.ただし,まだ切り替え動作はできないことにも注意してください.(サンプル(5)を見る)
index.html(抜粋)
<!DOCTYPE html>
<html lang="ja" data-bs-theme="dark">
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<title>Bootstrap demo</title>
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.2/dist/css/bootstrap.min.css" rel="stylesheet" integrity="sha384-T3c6CoIi6uLrA9TneNEoa7RxnatzjcDSCmG1MXxSR1GAsXEV/Dwwykc2MPK8M2HN" crossorigin="anonymous">
</head>
<body class="d-flex flex-column min-vh-100">
<nav class="navbar navbar-expand-lg bg-body-tertiary">
<div class="container-fluid">
<a class="navbar-brand" href="#">Navbar</a>
<button class="navbar-toggler" type="button" data-bs-toggle="collapse" data-bs-target="#navbarSupportedContent" aria-controls="navbarSupportedContent" aria-expanded="false" aria-label="Toggle navigation">
<span class="navbar-toggler-icon"></span>
</button>
<div class="collapse navbar-collapse" id="navbarSupportedContent">
<ul class="navbar-nav me-auto mb-2 mb-lg-0">
<li class="nav-item">
<a class="nav-link active" aria-current="page" href="#">Home</a>
</li>
</ul>
<ul class="navbar-nav mb-2 mb-lg-0">
<!-- テーマ切替ドロップダウン(右寄せの左側) -->
<li class="nav-item dropdown me-lg-2">
<a
class="nav-link dropdown-toggle"
href="#"
role="button"
data-bs-toggle="dropdown"
aria-expanded="false"
id="themeDropdown"
>
テーマ
</a>
<ul class="dropdown-menu dropdown-menu-end" aria-labelledby="themeDropdown">
<li>
<button type="button" class="dropdown-item d-flex align-items-center gap-2" data-theme-value="light">
☀ Light
<span class="ms-auto checkmark" aria-hidden="true" hidden>✔</span>
</button>
</li>
<li>
<button type="button" class="dropdown-item d-flex align-items-center gap-2" data-theme-value="dark">
🌙 Dark
<span class="ms-auto checkmark" aria-hidden="true" hidden>✔</span>
</button>
</li>
<li><hr class="dropdown-divider"></li>
<li>
<button type="button" class="dropdown-item d-flex align-items-center gap-2" data-theme-value="auto">
🖥 Auto(OSに追従)
<span class="ms-auto checkmark" aria-hidden="true" hidden>✔</span>
</button>
</li>
</ul>
</li>
</ul>
</div>
</div>
</nav>
<main class="container-md">
<h1>Bootstrapのサンプル</h1>
<p>
<code>.container-md</code>
</p>
<p>
ここは本文の領域です.レイアウトを確認するための文章です...
</p>
</main>
<footer class="container-fluid mt-auto">
<p>
ここはフッター領域です.ここはフッター領域です.ここはフッター領域です.ここはフッター領域です.ここはフッター領域です.ここはフッター領域です.ここはフッター領域です.ここはフッター領域です.ここはフッター領域です...
</p>
</footer>
<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.3.2/dist/js/bootstrap.bundle.min.js" integrity="sha384-C6RzsynM9kWDrMNeT87bh95OGNyZPhcTNXj1NW7RuBCsyN/o0jlpcV8Qyq46cDfL" crossorigin="anonymous"></script>
</body>
</html>
JavaScript でテーマを切り替える
Navbar のメニューを選択することでテーマが切り替わるように JavaScript で処理を実装します(サンプル(6)を見る).具体的には,次のコードの9〜17行目では,ブラウザのWebStorage(の Local Storage)に記録された値を取得し,初期テーマを設定します.この処理はページ画面の描画が始まるよりも前に行いたいことから,<head> の中に記述しています.一方で,テーマを切り替える処理は89行目以降で行っています.テーマを切り替えたときにブラウザの Local Storage に値を格納することから,それ以降同じ Web サイトを閲覧するとそのテーマで表示されるようになります.
index.html(抜粋)
<!DOCTYPE html>
<html lang="ja" data-bs-theme="dark">
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<title>Bootstrap demo</title>
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.2/dist/css/bootstrap.min.css" rel="stylesheet" integrity="sha384-T3c6CoIi6uLrA9TneNEoa7RxnatzjcDSCmG1MXxSR1GAsXEV/Dwwykc2MPK8M2HN" crossorigin="anonymous">
<!-- 初期テーマ適用(最速反映のため head 内で実行) -->
<script>
(function () {
const STORAGE_KEY = "bs-theme";
const stored = localStorage.getItem(STORAGE_KEY);
const prefersDark = window.matchMedia("(prefers-color-scheme: dark)").matches;
const initialTheme = stored || (prefersDark ? "dark" : "light");
document.documentElement.setAttribute("data-bs-theme", initialTheme);
})();
</script>
</head>
<body class="d-flex flex-column min-vh-100">
<nav class="navbar navbar-expand-lg bg-body-tertiary">
<div class="container-fluid">
<a class="navbar-brand" href="#">Navbar</a>
<button class="navbar-toggler" type="button" data-bs-toggle="collapse" data-bs-target="#navbarSupportedContent" aria-controls="navbarSupportedContent" aria-expanded="false" aria-label="Toggle navigation">
<span class="navbar-toggler-icon"></span>
</button>
<div class="collapse navbar-collapse" id="navbarSupportedContent">
<ul class="navbar-nav me-auto mb-2 mb-lg-0">
<li class="nav-item">
<a class="nav-link active" aria-current="page" href="#">Home</a>
</li>
</ul>
<ul class="navbar-nav mb-2 mb-lg-0">
<!-- テーマ切替ドロップダウン(右寄せの左側) -->
<li class="nav-item dropdown me-lg-2">
<a
class="nav-link dropdown-toggle"
href="#"
role="button"
data-bs-toggle="dropdown"
aria-expanded="false"
id="themeDropdown"
>
テーマ
</a>
<ul class="dropdown-menu dropdown-menu-end" aria-labelledby="themeDropdown">
<li>
<button type="button" class="dropdown-item d-flex align-items-center gap-2" data-theme-value="light">
☀ Light
<span class="ms-auto checkmark" aria-hidden="true" hidden>✔</span>
</button>
</li>
<li>
<button type="button" class="dropdown-item d-flex align-items-center gap-2" data-theme-value="dark">
🌙 Dark
<span class="ms-auto checkmark" aria-hidden="true" hidden>✔</span>
</button>
</li>
<li><hr class="dropdown-divider"></li>
<li>
<button type="button" class="dropdown-item d-flex align-items-center gap-2" data-theme-value="auto">
🖥 Auto(OSに追従)
<span class="ms-auto checkmark" aria-hidden="true" hidden>✔</span>
</button>
</li>
</ul>
</li>
</ul>
</div>
</div>
</nav>
<main class="container-md">
<h1>Bootstrapのサンプル</h1>
<p>
<code>.container-md</code>
</p>
<p>
ここは本文の領域です.レイアウトを確認するための文章です...
</p>
</main>
<footer class="container-fluid mt-auto">
<p>
ここはフッター領域です.ここはフッター領域です.ここはフッター領域です.ここはフッター領域です.ここはフッター領域です.ここはフッター領域です.ここはフッター領域です.ここはフッター領域です.ここはフッター領域です...
</p>
</footer>
<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.3.2/dist/js/bootstrap.bundle.min.js" integrity="sha384-C6RzsynM9kWDrMNeT87bh95OGNyZPhcTNXj1NW7RuBCsyN/o0jlpcV8Qyq46cDfL" crossorigin="anonymous"></script>
<!-- テーマ切替ロジック -->
<script>
(function () {
const STORAGE_KEY = "bs-theme";
const media = window.matchMedia("(prefers-color-scheme: dark)");
const dropdownItems = document.querySelectorAll('[data-theme-value]');
const themeDropdown = document.getElementById('themeDropdown');
function applyTheme(theme) {
document.documentElement.setAttribute("data-bs-theme", theme);
updateDropdownUI(theme);
}
function setThemeAndPersist(value) {
if (value === "auto") {
// Auto: 保存を消して OS 設定に追従
localStorage.removeItem(STORAGE_KEY);
applyTheme(media.matches ? "dark" : "light");
} else {
// Light/Dark: 明示設定して保存
localStorage.setItem(STORAGE_KEY, value);
applyTheme(value);
}
}
function updateDropdownUI(theme) {
// チェックマーク&aria-current更新
dropdownItems.forEach((el) => {
const val = el.getAttribute("data-theme-value");
const selected =
(val === "auto" && !localStorage.getItem(STORAGE_KEY)) ||
(val === theme && !!localStorage.getItem(STORAGE_KEY));
el.setAttribute("aria-current", selected ? "true" : "false");
const mark = el.querySelector(".checkmark");
if (mark) mark.hidden = !selected;
});
// トリガーのテキストも更新
const hasStored = !!localStorage.getItem(STORAGE_KEY);
const label = hasStored ? (theme === "dark" ? "Dark" : "Light") : "Auto";
if (themeDropdown) themeDropdown.textContent = `テーマ(${label})`;
}
// 初期描画
(function init() {
const stored = localStorage.getItem(STORAGE_KEY);
const theme = stored || (media.matches ? "dark" : "light");
applyTheme(theme);
})();
// クリックイベント
dropdownItems.forEach((el) => {
el.addEventListener("click", () => {
const value = el.getAttribute("data-theme-value");
setThemeAndPersist(value);
});
});
// OS設定の変更に追従(Auto のときのみ)
media.addEventListener?.("change", (e) => {
if (!localStorage.getItem(STORAGE_KEY)) {
applyTheme(e.matches ? "dark" : "light");
}
});
})();
</script>
</body>
</html>
なお,ブラウザに格納された Local Storage の値を確認するためには,Google Chrome の場合,「表示」→「開発/管理」→「デベロッパー ツール」メニュー(または,右クリックして「検証」)から「Application」を開きます.