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

Александр Мальцев
Александр Мальцев
211K
29
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:

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

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

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

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

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

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

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

<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.

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

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

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

Вид кастомного чекбокса
.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
.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
/* стили при наведении курсора на 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":

<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 разметка:

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

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 разметка:

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

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;
}

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

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

  1. ElenaArapova
    ElenaArapova
    2022-05-09 17:17:45
    Очень крутая статья. Все так понятно описано. Спасибо большое! Очень помогло и много дополнительной информации на будущее.
  1. rigebara1332
    rigebara1332
    2022-05-09 12:12:48
    Вопрос: Это нормально что при нажатии на «поддельный чекбокс» у реального чекбокса в DOM не добавляется атрибут checked? Или я что-то сделал не так? Переключение состояний, касательно стилей, работает отлично, но атрибута нет
    1. Valeria
      Valeria
      2022-04-30 21:05:37
      Спасибо огромное за статью! Очень полезная!!) Присоединяюсь к предыдущим комментаторам, зарегалась, чтобы оставить комментарий!)
      1. Александр Мальцев
        Александр Мальцев
        2022-05-02 14:25:43
        Спасибо за отзыв! Рад что понравилась.
    2. Dmitry
      Dmitry
      2022-03-23 16:55:13
      Это то что я искал!!! супер статья) огромное спасибо и + в карму!
      Зарегался только чтоб отблагодарить автора)
      1. Александр Мальцев
        Александр Мальцев
        2022-03-24 02:59:04
        Благодарю за оценку
    3. fet
      fet
      2021-02-17 22:02:21
      А подскажите как теперь получить кастомный чекбокс с помощью Jquery, но не при смене чекбокса, а просто выбранный сразу при загрузке страницы и так же потом при переключении? спс
      1. Александр Мальцев
        Александр Мальцев
        2021-02-18 16:28:13
        В 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>
        
      2. fet
        fet
        2021-02-19 10:34:09
        спасибо, все работает, но проблема у меня в другом если таких объектов несколько и они без 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>
        
        
      3. rigebara1332
        rigebara1332
        2022-05-09 12:09:56
        Для начала сделай разные ID у элементов.
    4. Andrii Petrov
      Andrii Petrov
      2020-11-04 00:23:59
      В радиобатонах фокус не работает.
      1. Andrii Petrov
        Andrii Petrov
        2020-11-04 00:43:29
        А именно если выбран хотя бы один чекбокс. Фокус на него и всё.
      2. Александр Мальцев
        Александр Мальцев
        2020-11-04 03:43:14
        Радиокнопка в состоянии фокуса:
        радиокнопка в состоянии фокуса
        Это устанавливается с помощью следующих стилей:
        /* стили для радиокнопки, находящейся в фокусе */
        .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;
        }
        
        Если вам нужно как-то по-другому, то измените их соответствующим образом.
      3. Инна
        Инна
        2020-11-07 11:24:13
        Может не по теме, но как сделать так что бы флажки стояли в 2 столбика не в разброс
      4. Александр Мальцев
        Александр Мальцев
        2020-11-07 11:43:06
        Это можно сделать разными способами.
        Например, с использованием 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>
        
      5. Инна
        Инна
        2020-11-07 13:31:12
        Надо вот так (скриншот). Перепробовала все что нашла в интернете, результата такого не добилась. Уходит второй столбик на середину страницы, а надо рядом с первым столбом.
        Спасибо за ответ!
      6. Александр Мальцев
        Александр Мальцев
        2020-11-07 14:24:20
        Тогда можно так:
        <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>
        
    5. Ivan
      Ivan
      2020-10-21 14:07:09
      Зарегистрировался только для того, чтобы поблагодарить автора за написание этой статьи. Спасибо!
      1. Александр Мальцев
        Александр Мальцев
        2020-10-21 15:39:47
        Спасибо за отзыв!
    6. Денис
      Денис
      2020-08-22 22:40:33
      Здравствуйте. При использовании кода

      <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 работает нормально. Подскажите есть ли решение?
      1. Никита Филипчук
        Никита Филипчук
        2020-08-18 14:40:19
        Подскажите как сделать чекбокс справа от текста? от label
        1. Александр Мальцев
          Александр Мальцев
          2020-08-18 15:37:06
          Для этого можно изменить псевдоэлемент ::before на ::after. Как это можно выполнить можно посмотреть в этом примере.
      2. denis
        denis
        2020-07-09 23:02:27
        Спасибо, красивая и простая стилизация. Помогите адаптировать стилизацию с моими 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 скрыт и он не нажимается
        1. Александр Мальцев
          Александр Мальцев
          2020-07-11 15:36:03
          Пожалуйста!
          Для этого ваш скрипт следует переписать следующим образом:
          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;
          });
          
          Пример можно посмотреть здесь.
        2. denis
          denis
          2020-07-14 12:47:29
          Спасибо еще раз. Может у вас есть интересные статьи и про jquery слайдер для числового диапазона?
        3. Александр Мальцев
          Александр Мальцев
          2020-07-15 15:39:25
          Пожалуйста! Пока нет. Имеется только то, что есть на сайте.
      3. Александр
        Александр
        2020-02-18 10:39:06
        Здравствуйте! Круто, просто и стильно. Правда в моём случае пришлось скрыть дефолтный checkbox таким образом:
        .custom-checkbox>input {
         display: none;
        }
        при большом списке примерно около ста элементов в блоке с прокруткой, при клике на любой checkbox который изначально скрыт прокруткой, почему-то перемещало в нижнюю часть страницы, плюс увеличивалась высота страницы по вертикали.
        1. Александр Мальцев
          Александр Мальцев
          2020-02-18 13:37:05
          Добрый день! Спасибо за отзыв.
          В этом случае можно из 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;
          }
          
        2. Александр
          Александр
          2020-02-18 17:10:43
          Это как минимум, Вам огромное спасибо за очень нужные статьи, руководства и помощь. Да, точно фокус не работает, я изначально не заметил этого, спасибо.