神戸学院大学 経営学部 林坂ゼミ

HTML 入門トップページ

« 戻る 次へ »

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」を開きます.

目次に戻る