Переключение темы

Типичный сценарий: у сайта уже есть светлая тема и вы хотите создать её тёмный аналог. Или, даже если вы начинаете с нуля, у вас уже есть обе темы: светлая и тёмная. Одна тема должна быть определена как тема по умолчанию, которую пользователи получают при первом посещении — в большинстве случаев это светлая тема (хотя мы можем позволить браузеру пользователя сделать этот выбор за нас, как вы увидите дальше). Также должен быть способ переключиться на другую тему (что можно сделать автоматически, как вы тоже увидите) — например, пользователь щёлкает по кнопке и цветовая тема меняется. Для этого существует несколько подходов.

Использование класса для <body>

Хитрость заключается в замене класса, который будет указателем для изменения стиля на всей странице.

<body class="dark-theme или light-theme">

Это пример скрипта для кнопки, переключающей данный класс.

// Выбираем кнопку
const btn = document.querySelector('.btn-toggle');
// Отслеживаем щелчок по кнопке
btn.addEventListener('click', function() {
  // Затем переключаем (добавляем/удаляем) класс .dark-theme для body
  document.body.classList.toggle('dark-theme'); 
})

Вот как мы можем использовать эту идею.

<body>
  <button class="btn-toggle">Переключатель тёмной темы</button>
  <h1>Привет! Это просто заголовок</h1>
  <p>Я просто скучный текст, существующий исключительно ради данной демонстрации.</p>
  <p>А я ещё один текст, похожий на тот что выше, потому что два лучше, чем один.</p>
  <a href="#">Я ссылка, не нажимайте на меня!</a>
</body>

Основная идея такого подхода состоит в том, что мы стилизуем элементы как обычно, назовём это нашим режимом «по умолчанию». Затем создаём полный набор цветовых стилей с помощью класса, заданного в элементе <body>, который мы можем использовать в качестве «тёмного» режима. Допустим, по умолчанию применяется светлая цветовая схема. Все эти «светлые» стили написаны точно так же, как вы обычно пишете CSS. Учитывая наш HTML, применим глобальный стиль к <body> и к <a>.

body {
  color: #222;
  background: #fff;
}
a {
  color: #0033cc;
}

Довольно неплохо. Теперь у нас есть тёмный текст (#222) и тёмные ссылки (#0033cc) на светлом фоне (#fff). Начало нашей темы «по умолчанию» положено.

Теперь переопределим значения этих свойств, но в этот раз для другого класса.

body {
  color: #222;
  background: #fff;
}
a {
  color: #0033cc;
}

/* Стили Тёмной темы */
body.dark-theme {
  color: #eee;
  background: #121212;
}
body.dark-theme a {
  color: #809fff;
}

Стили тёмной темы будут потомками того же родительского класса — в нашем примере это .dark-theme, который мы добавили к тегу <body>.

Как «переключать» классы у <body> для доступа к тёмным стилям? Мы можем использовать JavaScript! Выберем класс кнопки (.btn-toggle), добавим отслеживание щелчка, а затем вставим класс тёмной темы (.dark-theme) в список классов элемента <body>. Это эффективно отменит все установленные нами «светлые» цвета, благодаря каскаду и специфичности.

Вот полный рабочий код в действии. Пощёлкайте по кнопке для переключения на тёмный режим и выхода из него.

Использование разных таблиц стилей

Вместо хранения всех стилей в одном месте, мы можем переключаться между таблицами стилей для каждой темы. Предполагается, что у вас уже есть полностью готовые таблицы стилей.

К примеру, светлая тема по умолчанию, вроде light-theme.css.

/* light-theme.css */
body {
  color: #222;
  background: #fff;
}
a {
  color: #0033cc;
}

Затем создаём стили для тёмной темы и сохраняем их в отдельном файле, который назовём dark-theme.css.

/* dark-theme.css */
body {
  color: #eee;
  background: #121212;
}
body a {
  color: #809fff;
}

Это даёт нам две отдельные таблицы стилей, по одной для каждой темы, на которые можно сослаться в разделе <head>. Давайте сперва сделаем ссылку на светлые стили, поскольку мы называем их стилями по умолчанию.

<!DOCTYPE html>
<html lang="ru">
<head>
  <!-- Таблица стилей светлой темы -->
  <link href="light-theme.css" rel="stylesheet" id="theme-link">
</head>
  <!-- И т.д. -->
</html>

Мы используем идентификатор #theme-link, который можно выбрать через JavaScript, чтобы снова переключаться между светлым и тёмным режимами. Только на этот раз мы переключаем файлы вместо классов.

// Выбираем кнопку
const btn = document.querySelector(".btn-toggle");
// Выбираем таблицу стилей
const theme = document.querySelector("#theme-link");
// Отслеживаем щелчок по кнопке
btn.addEventListener("click", function() {
  // Если текущий адрес содержит "light-theme.css"
  if (theme.getAttribute("href") == "light-theme.css") {
    // …то переключаемся на "dark-theme.css"
    theme.href = "dark-theme.css";
    // В противном случае… 
  } else {
    // …переключаемся на "light-theme.css"
    theme.href = "light-theme.css";
  }
});

Посмотреть результат в CodePen

Использование пользовательских свойств

Мы также можем воспользоваться мощью пользовательских свойств CSS для создания тёмной темы! В итоге нам не придётся писать отдельные наборы правил стилей для каждой темы, при этом значительно ускоряется написание стилей и упрощается внесение изменений в тему, если это понадобится.

Мы по-прежнему можем менять класс у <body> и использовать этот класс для повторной установки пользовательских свойств.

// Выбираем кнопку
const btn = document.querySelector('.btn-toggle');
// Отслеживаем щелчок по кнопке
btn.addEventListener('click', function() {
  // Затем переключаем (добавляем/удаляем) класс .dark-theme для body
  document.body.classList.toggle('dark-theme'); 
})

Сперва определим значения светлых цветов по умолчанию в виде пользовательских свойств для элемента <body>.

body {
  --text-color: #222;
  --bkg-color: #fff;
  --anchor-color: #0033cc;
}

Теперь мы можем переопределить эти значения для класса .dark-theme, как мы это уже делали в первом методе.

body.dark-theme {
  --text-color: #eee;
  --bkg-color: #121212;
  --anchor-color: #809fff;
}

Вот наши наборы правил для элементов <body> и <a>, использующих пользовательские свойства.

body {
  color: var(--text-color);
  background: var(--bkg-color);
}
a {
  color: var(--anchor-color);
}

С тем же успехом мы могли бы определить наши пользовательские свойства внутри :root. Это вполне легальная и даже обычная практика. В таком случае все определения стилей темы по умолчанию будут помещены внутрь :root {}, а все свойства темной темы войдут внутрь :root.dark-mode {}.

Использование скриптов на стороне сервера

Если вы уже работали с серверным языком, скажем, PHP, то можете использовать его вместо JavaScript. Это отличный подход, если предпочитаете работать непосредственно с разметкой.

<?php
$themeClass = '';
if (isset($_GET['theme']) && $_GET['theme'] == 'dark') {
  $themeClass = 'dark-theme';
}
$themeToggle = ($themeClass == 'dark-theme') ? 'light' : 'dark';
?>
<!DOCTYPE html>
<html lang="ru">
  <!-- и т.д. -->
<body class="<?php echo $themeClass; ?>">
  <a href="?theme=<?php echo $themeToggle; ?>">Переключатель тёмной темы</a>
  <!-- и т.д. -->
</body>
</html>

Мы можем попросить пользователя отправить запрос GET или POST. Затем позволим нашему коду (в данном случае PHP) применить соответствующий класс к <body> при перезагрузке страницы. В данной демонстрации я использую запрос GET (параметры URL).

И да, мы можем поменять местами таблицы стилей как во втором методе.

<?php
$themeStyleSheet = 'light-theme.css';
if (isset($_GET['theme']) && $_GET['theme'] == 'dark') {
  $themeStyleSheet = 'dark-theme.css';
}
$themeToggle = ($themeStyleSheet == 'dark-theme.css') ? 'light' : 'dark';
?>
<!DOCTYPE html>
<html lang="ru">
<head>
  <!-- и т.д. -->
  <link href="<?php echo $themeStyleSheet; ?>" rel="stylesheet">
</head>
<body>
  <a href="?theme=<?php echo $themeToggle; ?>">Переключатель тёмной темы</a>
  <!-- и т.д. -->
</body>
</html>

У этого метода есть очевидный недостаток: необходимо обновить страницу, чтобы переключение сработало. Но подобное серверное решение полезно для сохранения выбранной пользователем темы при перезагрузке страницы, как мы увидим позже.

Какой метод выбрать?

«Правильный» метод зависит от требований вашего проекта. Например, если вы работаете над большим проектом, то можете использовать свойства CSS, которые помогут справиться с большой базой кода. С другой стороны, если ваш проект должен поддерживать устаревшие браузеры, тогда потребуется другой подход.

Кроме того, никто не говорит, что мы можем использовать только один метод. Иногда их сочетание оказывается наиболее эффективным вариантом. Могут быть и другие возможные методы, помимо упомянутых.

Автор: Мохамед Адхухам
Последнее изменение: 27.02.2024