CSS стилизация checkbox и radio - 2 варианта

CSS стилизация checkbox и radio - 2 варианта
Содержание:
  1. Как осуществляется создание кастомного чекбокса или переключателя
  2. Создание стильного чекбокса
  3. Разработка кастомного переключателя
  4. Ещё примеры по кастомизации checkbox и label
  5. Комментарии

В этой статье подробно разберём процесс кастомной стилизации чекбоксов и радиокнопок с помощью CSS.

Как осуществляется создание кастомного чекбокса или переключателя

Данный процесс осуществляется посредством скрытия стандартного элемента и создания с помощью CSS другого «поддельного», такого как мы хотим.

Но как же это будет работать, если стандартный input скрыть? Это можно выполнить благодаря тому, что в HTML переключить состояние checked можно не только с помощью самого элемента input, но и посредством связанного с ним label.

В HTML связывание label с input выполняется одним из 2 способов:

1. Посредством помещения элемента input в label:

HTML
<label>
  <input type="checkbox" name="happy" value="yes">Happy
</label>

2. Посредством задания элементу input атрибута id, а labelfor с таким же значением как у id.

HTML
<input type="checkbox" id="happy" name="happy" value="yes">
<label for="happy">Happy</label>

В этой статье мы подробно разберём шаги по кастомизации checkbox и radio, в которых label с input свяжем по 2 варианту. Создание «поддельного» чекбокса выполним с использованием псевдоэлемента ::before, который поместим в label. При этом никакие дополнительные элементы в разметку добавлять не будем.

Создание стильного чекбокса

Процесс замены стандартного вида чекбокса на кастомный осуществим посредством выполнения следующей последовательности шагов.

Шаг 1. Создадим разметку.

HTML
<input type="checkbox" class="custom-checkbox" id="happy" name="happy" value="yes">
<label for="happy">Happy</label>

При создании разметки очень важно соблюдать последовательность расположения элементов. Это необходимо, потому что в зависимости от того, как они расположены мы будем составлять выражения для выбора элементов в CSS и назначать им стили.

В этом примере элемент label расположен после input. Связь label с input осуществляется посредством соответствия значения for элемента label с id элемента input.

В примере к элементу input добавлен класс custom-checkbox. Данный класс мы будем использовать при составлении селекторов и тем самым с помощью него определять элементы к которым следует добавить стилизованный чекбокс вместо обычного. Т.е. его присутствие или отсутствие будет определять с каким чекбоксом (со стандартным или поддельным) будет выводится элемент input с type="checkbox".

Вид чекбокса в браузере по умолчанию

Шаг 2. Напишем стили для скрытия стандартного элемента input.

Вид чекбокса после скрытия
CSS
.custom-checkbox {
  position: absolute;
  z-index: -1;
  opacity: 0;
}

Мы не будем использовать display: none, а установим ему стили, с помощью которых уберём его из потока (position: absolute), поместим его ниже существующих элементов (z-index: -1), а также сделаем его полностью прозрачным (opacity: 0). Зачем это нужно? Это нам необходимо для того, чтобы мы могли получить состояние фокуса, а затем стилизовать «подделный» checkbox или radio, когда он будет находиться в нём.

Шаг 3. Создадим поддельный чекбокс.

Вид кастомного чекбокса
CSS
.custom-checkbox+label {
  display: inline-flex;
  align-items: center;
  user-select: none;
}
.custom-checkbox+label::before {
  content: '';
  display: inline-block;
  width: 1em;
  height: 1em;
  flex-shrink: 0;
  flex-grow: 0;
  border: 1px solid #adb5bd;
  border-radius: 0.25em;
  margin-right: 0.5em;
  background-repeat: no-repeat;
  background-position: center center;
  background-size: 50% 50%;
}

Создание «поддельного» чекбокса выполним с помощью псевдоэлемента ::before. Посредством CSS зададим ему размеры (в данном случае 1emx1em), а затем нарисуем его с помощью border: 1px solid #adb5bd. Свойства начинающие со слова background будут определять положение самого флажка (когда checkbox будет в состоянии checked).

Первое правило необходимо для вертикального центрирования флажка и надписи к нему. Это действие в примере выполнено через CSS Flexbox.

Шаг 4. Создадим стили при нахождении элемента в состоянии checked.

Вид стилизованного чекбокса в состоянии checked
CSS
.custom-checkbox:checked+label::before {
  border-color: #0b76ef;
  background-color: #0b76ef;
  background-image: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 8 8'%3e%3cpath fill='%23fff' d='M6.564.75l-3.59 3.612-1.538-1.55L0 4.26 2.974 7.25 8 2.193z'/%3e%3c/svg%3e");
}

В этом коде при получении элементом состояния checked применим к псевдоэлементу ::before находящемуся в label стили, посредством которых установим цвет границы, цвет фону и фоновую картинку (флажок) в формате svg.

Шаг 5. Добавим код для стилизации чекбокса при нахождении его в состояниях hover, active, focus и disabled.

Вид кастомного чекбокса в состояниях hover, active, focus и disabled
CSS
/* стили при наведении курсора на checkbox */
.custom-checkbox:not(:disabled):not(:checked)+label:hover::before {
  border-color: #b3d7ff;
}
/* стили для активного состояния чекбокса (при нажатии на него) */
.custom-checkbox:not(:disabled):active+label::before {
  background-color: #b3d7ff;
  border-color: #b3d7ff;
}
/* стили для чекбокса, находящегося в фокусе */
.custom-checkbox:focus+label::before {
  box-shadow: 0 0 0 0.2rem rgba(0, 123, 255, 0.25);
}
/* стили для чекбокса, находящегося в фокусе и не находящегося в состоянии checked */
.custom-checkbox:focus:not(:checked)+label::before {
  border-color: #80bdff;
}
/* стили для чекбокса, находящегося в состоянии disabled */
.custom-checkbox:disabled+label::before {
  background-color: #e9ecef;
}

Открыть пример

Разработка кастомного переключателя

Стилизация переключателя (input с type="radio") выполняется аналогично, т.е. посредством тех же шагов которые мы применяли при кастомизации чекбокса.

Вид стилизованного чекбокса в браузере по умолчанию и в состоянии checked

Итоговый набор стилей для кастомного оформления input с type="radio":

HTML
<style>
  /* для элемента input c type="radio" */
  .custom-radio {
    position: absolute;
    z-index: -1;
    opacity: 0;
  }
  /* для элемента label связанного с .custom-radio */
  .custom-radio+label {
    display: inline-flex;
    align-items: center;
    user-select: none;
  }
  /* создание в label псевдоэлемента  before со следующими стилями */
  .custom-radio+label::before {
    content: '';
    display: inline-block;
    width: 1em;
    height: 1em;
    flex-shrink: 0;
    flex-grow: 0;
    border: 1px solid #adb5bd;
    border-radius: 50%;
    margin-right: 0.5em;
    background-repeat: no-repeat;
    background-position: center center;
    background-size: 50% 50%;
  }
  /* стили при наведении курсора на радио */
  .custom-radio:not(:disabled):not(:checked)+label:hover::before {
    border-color: #b3d7ff;
  }
  /* стили для активной радиокнопки (при нажатии на неё) */
  .custom-radio:not(:disabled):active+label::before {
    background-color: #b3d7ff;
    border-color: #b3d7ff;
  }
  /* стили для радиокнопки, находящейся в фокусе */
  .custom-radio:focus+label::before {
    box-shadow: 0 0 0 0.2rem rgba(0, 123, 255, 0.25);
  }
  /* стили для радиокнопки, находящейся в фокусе и не находящейся в состоянии checked */
  .custom-radio:focus:not(:checked)+label::before {
    border-color: #80bdff;
  }
  /* стили для радиокнопки, находящейся в состоянии checked */
  .custom-radio:checked+label::before {
    border-color: #0b76ef;
    background-color: #0b76ef;
    background-image: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='-4 -4 8 8'%3e%3ccircle r='3' fill='%23fff'/%3e%3c/svg%3e");
  }
  /* стили для радиокнопки, находящейся в состоянии disabled */
  .custom-radio:disabled+label::before {
    background-color: #e9ecef;
  }
</style>

<input class="custom-radio" name="color" type="radio" id="color-green" value="green">
<label for="color-green">Green</label>

Открыть пример

Ещё примеры по кастомизации checkbox и label

В этом разделе представлены следующие примеры:

1. Стилизация checkbox, когда input расположен в label.

HTML разметка:

HTML
<label class="custom-checkbox">
  <input type="checkbox" value="value-1">
  <span>Indigo</span>
</label>

CSS код:

CSS
/* для элемента input c type="checkbox" */
.custom-checkbox>input {
  position: absolute;
  z-index: -1;
  opacity: 0;
}

/* для элемента label, связанного с .custom-checkbox */
.custom-checkbox>span {
  display: inline-flex;
  align-items: center;
  user-select: none;
}

/* создание в label псевдоэлемента before со следующими стилями */
.custom-checkbox>span::before {
  content: '';
  display: inline-block;
  width: 1em;
  height: 1em;
  flex-shrink: 0;
  flex-grow: 0;
  border: 1px solid #adb5bd;
  border-radius: 0.25em;
  margin-right: 0.5em;
  background-repeat: no-repeat;
  background-position: center center;
  background-size: 50% 50%;
}

/* стили при наведении курсора на checkbox */
.custom-checkbox>input:not(:disabled):not(:checked)+span:hover::before {
  border-color: #b3d7ff;
}

/* стили для активного чекбокса (при нажатии на него) */
.custom-checkbox>input:not(:disabled):active+span::before {
  background-color: #b3d7ff;
  border-color: #b3d7ff;
}

/* стили для чекбокса, находящегося в фокусе */
.custom-checkbox>input:focus+span::before {
  box-shadow: 0 0 0 0.2rem rgba(0, 123, 255, 0.25);
}

/* стили для чекбокса, находящегося в фокусе и не находящегося в состоянии checked */
.custom-checkbox>input:focus:not(:checked)+span::before {
  border-color: #80bdff;
}

/* стили для чекбокса, находящегося в состоянии checked */
.custom-checkbox>input:checked+span::before {
  border-color: #0b76ef;
  background-color: #0b76ef;
  background-image: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 8 8'%3e%3cpath fill='%23fff' d='M6.564.75l-3.59 3.612-1.538-1.55L0 4.26 2.974 7.25 8 2.193z'/%3e%3c/svg%3e");
}

/* стили для чекбокса, находящегося в состоянии disabled */
.custom-checkbox>input:disabled+span::before {
  background-color: #e9ecef;
}

Открыть пример

2. Стилизация radio, когда input расположен в label.

HTML разметка:

HTML
<label class="custom-radio">
  <input type="radio" name="color" value="indigo">
  <span>Indigo</span>
</label>

CSS код:

CSS
/* для элемента input c type="radio" */
.custom-radio>input {
  position: absolute;
  z-index: -1;
  opacity: 0;
}

/* для элемента label связанного с .custom-radio */
.custom-radio>span {
  display: inline-flex;
  align-items: center;
  user-select: none;
}

/* создание в label псевдоэлемента  before со следующими стилями */
.custom-radio>span::before {
  content: '';
  display: inline-block;
  width: 1em;
  height: 1em;
  flex-shrink: 0;
  flex-grow: 0;
  border: 1px solid #adb5bd;
  border-radius: 50%;
  margin-right: 0.5em;
  background-repeat: no-repeat;
  background-position: center center;
  background-size: 50% 50%;
}

/* стили при наведении курсора на радио */
.custom-radio>input:not(:disabled):not(:checked)+span:hover::before {
  border-color: #b3d7ff;
}

/* стили для активной радиокнопки (при нажатии на неё) */
.custom-radio>input:not(:disabled):active+span::before {
  background-color: #b3d7ff;
  border-color: #b3d7ff;
}

/* стили для радиокнопки, находящейся в фокусе */
.custom-radio>input:focus+span::before {
  box-shadow: 0 0 0 0.2rem rgba(0, 123, 255, 0.25);
}

/* стили для радиокнопки, находящейся в фокусе и не находящейся в состоянии checked */
.custom-radio>input:focus:not(:checked)+span::before {
  border-color: #80bdff;
}

/* стили для радиокнопки, находящейся в состоянии checked */
.custom-radio>input:checked+span::before {
  border-color: #0b76ef;
  background-color: #0b76ef;
  background-image: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='-4 -4 8 8'%3e%3ccircle r='3' fill='%23fff'/%3e%3c/svg%3e");
}

/* стили для радиокнопки, находящейся в состоянии disabled */
.custom-radio>input:disabled+span::before {
  background-color: #e9ecef;
}

Открыть пример

Комментарии: 34

vasika778
vasika778

Подскажите, почему после нажатия кнопка не меняет цвет и не подсвечивается?

спасибо https://jsfiddle.net/2rd9xaz0/1/
Lina
Lina

Здравствуйте. Спасибо вам большое за подробную статью. Но у меня вопрос такой. Может сможете мне помочь. Мне нужно сделать так, что бы при выставлении чекбокса красная обводка исчезала. Добавить я её добавила с помощью класса, но когда ставлю галочку border не исчезает, remove не сработал.

<!DOCTYPE html>
<html lang="en">
<head>
  <meta charset="UTF-8">
  <meta http-equiv="X-UA-Compatible" content="IE=edge">
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
  <title>Согласие на полезную рассылку</title>
  <style>
    .invalid {
      border: dashed red;
    }
    .invalid-block {
      border: none;
    }
  </style>
</head>
<body>
  <form class="form" action="result.html" method="GET" enctype="multipart/form-data">
    <label class="label" for="one">согласен на полезную рассылку от сервиса</label>
    <input class="input" type="checkbox" name="source" id="one" required>
    <button class="submit">Отправить</button>
    <p id="message">Not checked</p>
  </form>

  <script>
    const form = document.querySelector('.form');
    const checkbox = document.getElementById('one');
    const submit = document.querySelector('.submit');
    const label = document.querySelector('label');
    const input = document.querySelector('input');
    submit.onclick = function() {
      if (!checkbox.checked) {
        label.className = 'invalid';
      }
      else if (checkbox.checked) {
        label.className = 'invalid-block';
      }
    }
  </script>
</body>
</html>
Александр Мальцев
Александр Мальцев

Добрый день! В этом случае нужно обрабатывать событие change чекбокса:

checkbox.onchange = () => {
  label.className = checkbox.checked ? 'invalid-block' : 'invalid';
}
Svetlana
Svetlana
Здравствуйте, спасибо за такую полезную статью.

подскажите, пожалуйста, как сделать так, чтобы при входе на страницу галочка в чекбоксе стояла, а пользователь мог снять галочку самостоятельно
Александр Мальцев
Александр Мальцев
Привет! Нужно просто добавить атрибут checked к input.
ElenaArapova
ElenaArapova
Очень крутая статья. Все так понятно описано. Спасибо большое! Очень помогло и много дополнительной информации на будущее.
rigebara1332
rigebara1332
Вопрос: Это нормально что при нажатии на «поддельный чекбокс» у реального чекбокса в DOM не добавляется атрибут checked? Или я что-то сделал не так? Переключение состояний, касательно стилей, работает отлично, но атрибута нет
Valeria
Valeria
Спасибо огромное за статью! Очень полезная!!) Присоединяюсь к предыдущим комментаторам, зарегалась, чтобы оставить комментарий!)
Александр Мальцев
Александр Мальцев
Спасибо за отзыв! Рад что понравилась.
Dmitry
Dmitry
Это то что я искал!!! супер статья) огромное спасибо и + в карму!
Зарегался только чтоб отблагодарить автора)
Александр Мальцев
Александр Мальцев
Благодарю за оценку
fet
fet
А подскажите как теперь получить кастомный чекбокс с помощью Jquery, но не при смене чекбокса, а просто выбранный сразу при загрузке страницы и так же потом при переключении? спс
Александр Мальцев
Александр Мальцев
В JavaScript в данном случае его нужно получать как обычно.
Например:
<div class="checkbox">
  <input class="custom-checkbox" type="checkbox" id="color-1" name="color-1" value="indigo" checked>
  <label for="color-1">Indigo</label>
</div>

<script>
  $(function () {
    // после загрузки страницы
    // получаем чекбокс
    var checkbox = $('#color-1');
    // при переключении выводим его состояние в консоль
    checkbox.on('change', function () {
      console.log($(this).prop('checked'));
    })
  });
</script>
fet
fet
спасибо, все работает, но проблема у меня в другом если таких объектов несколько и они без checked, и надо получить выбранный, спс


<div class="checkbox">
  <input class="custom-checkbox" type="checkbox" id="color-1" name="color-1" value="indigo" checked>
  <label for="color-1">Indigo</label>
</div>

<div class="checkbox">
  <input class="custom-checkbox" type="checkbox" id="color-1" name="color-2" value="indigo">
  <label for="color-1">Indigo</label>
</div>

<div class="checkbox">
  <input class="custom-checkbox" type="checkbox" id="color-1" name="color-3" value="indigo">
  <label for="color-1">Indigo</label>
</div>

rigebara1332
rigebara1332
Для начала сделай разные ID у элементов.
Andrii Petrov
Andrii Petrov
В радиобатонах фокус не работает.
Andrii Petrov
Andrii Petrov
А именно если выбран хотя бы один чекбокс. Фокус на него и всё.
Александр Мальцев
Александр Мальцев
Радиокнопка в состоянии фокуса:
радиокнопка в состоянии фокуса
Это устанавливается с помощью следующих стилей:
/* стили для радиокнопки, находящейся в фокусе */
.custom-radio:focus+label::before {
  box-shadow: 0 0 0 0.2rem rgba(0, 123, 255, 0.25);
}
/* стили для радиокнопки, находящейся в фокусе и не находящейся в состоянии checked */
.custom-radio:focus:not(:checked)+label::before {
  border-color: #80bdff;
}
Если вам нужно как-то по-другому, то измените их соответствующим образом.
Инна
Инна
Может не по теме, но как сделать так что бы флажки стояли в 2 столбика не в разброс
Александр Мальцев
Александр Мальцев
Это можно сделать разными способами.
Например, с использованием CSS Flexbox:
<div style="display: flex;">
  <div style="flex: 1 0 50%;">
    ...
  </div>
  <div style="flex: 1 0 50%;">
    ...
  </div>
</div>
C использованием CSS-свойства column-count:
<div style="column-count: 2;">
  ...
</div>
Инна
Инна
Надо вот так (скриншот). Перепробовала все что нашла в интернете, результата такого не добилась. Уходит второй столбик на середину страницы, а надо рядом с первым столбом.
Спасибо за ответ!
Александр Мальцев
Александр Мальцев
Тогда можно так:
<div>
  <div style="display: inline-block;">
    ...
  </div>
  <div style="display: inline-block;">
    ...
  </div>
</div>
Если нужно с использованием Flexbox, то так:
<div style="display: flex-inline;">
  <div>
    ...
  </div>
  <div>
    ...
  </div>
</div>
Ivan
Ivan
Зарегистрировался только для того, чтобы поблагодарить автора за написание этой статьи. Спасибо!
Александр Мальцев
Александр Мальцев
Спасибо за отзыв!
Денис
Денис
Здравствуйте. При использовании кода

<tmpl_if name="limit_ssl" op="==" value="y"><div class="form-group">
			<div data-uncheck-fields="ssl_letsencrypt">{tmpl_var name='ssl'}</div>
                <label for="col-sm-3 control-label">{tmpl_var name='ssl_txt'}</label>
            </div>
            <tmpl_if name="limit_ssl_letsencrypt" op="==" value="y">
            <div class="form-group">
			 <div data-check-fields="ssl">{tmpl_var name='ssl_letsencrypt'}</div>
                <label for="col-sm-3 control-label">{tmpl_var name='ssl_letsencrypt_txt'}</label>
            </div></tmpl_if>
            </tmpl_if>
не выводятся чекбоксы. В этом коде при нажатии чекбокса letsencrypt должен активироваться чекбокс ssl и наоборот при нажатии ssl отключиться. Чекбоксы не выводятся из-за тега div
<div data-uncheck-fields="ssl_letsencrypt">{tmpl_var name='ssl'}</div>
<div data-check-fields="ssl">{tmpl_var name='ssl_letsencrypt'}</div>
без div работает нормально. Подскажите есть ли решение?
Никита Филипчук
Никита Филипчук
Подскажите как сделать чекбокс справа от текста? от label
Александр Мальцев
Александр Мальцев
Для этого можно изменить псевдоэлемент ::before на ::after. Как это можно выполнить можно посмотреть в этом примере.
denis
denis
Спасибо, красивая и простая стилизация. Помогите адаптировать стилизацию с моими radio, у меня есть код, которые позволяет без стилизации снимать при повторном нажатии на radio checked

$('input[type="radio"]').click(function(){
	var vl = $(this).attr('name');
	var radio = Array.apply(null, document.querySelectorAll("input[name='"+vl+"'"));
	radio.forEach(radioBtn => {
  		var checked = false;
		radioBtn.addEventListener("mousedown", event => {
    			checked = radioBtn.checked;
		});
		radioBtn.addEventListener("click", event => {
			radioBtn.checked = !checked;
		});
	});
});
Но у вас физический radio скрыт и он не нажимается
Александр Мальцев
Александр Мальцев
Пожалуйста!
Для этого ваш скрипт следует переписать следующим образом:
document.addEventListener('mousedown', e => {
  let radioBtn = e.target.previousElementSibling;
  if (!radioBtn) {
    return;
  }
  if (!radioBtn.classList.contains('custom-radio')) {
    return;
  }
  radioBtn.checkedState = radioBtn.checked;
});
document.addEventListener('click', e => {
  let radioBtn = e.target;
  if (!radioBtn.classList.contains('custom-radio')) {
    return;
  }
  radioBtn.checked = !radioBtn.checkedState;
});
Пример можно посмотреть здесь.
denis
denis
Спасибо еще раз. Может у вас есть интересные статьи и про jquery слайдер для числового диапазона?
Александр Мальцев
Александр Мальцев
Пожалуйста! Пока нет. Имеется только то, что есть на сайте.
Александр
Александр
Здравствуйте! Круто, просто и стильно. Правда в моём случае пришлось скрыть дефолтный checkbox таким образом:
.custom-checkbox>input {
 display: none;
}
при большом списке примерно около ста элементов в блоке с прокруткой, при клике на любой checkbox который изначально скрыт прокруткой, почему-то перемещало в нижнюю часть страницы, плюс увеличивалась высота страницы по вертикали.
Александр Мальцев
Александр Мальцев
Добрый день! Спасибо за отзыв.
В этом случае можно из CSS удалить стили для состояния focus. Это позволит уменьшить размер CSS кода, т.к. они в этом случае всё равно работать не будут.
.custom-checkbox:focus+label::before {
  box-shadow: 0 0 0 0.2rem rgba(0, 123, 255, 0.25);
}
.custom-checkbox:focus:not(:checked)+label::before {
  border-color: #80bdff;
}
Александр
Александр
Это как минимум, Вам огромное спасибо за очень нужные статьи, руководства и помощь. Да, точно фокус не работает, я изначально не заметил этого, спасибо.