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

Александр Мальцев
Александр Мальцев
122K
22
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;
}

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

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

  1. fet
    fet
    17.02.2021, 22:02
    А подскажите как теперь получить кастомный чекбокс с помощью Jquery, но не при смене чекбокса, а просто выбранный сразу при загрузке страницы и так же потом при переключении? спс
    1. Александр Мальцев
      Александр Мальцев
      18.02.2021, 16:28
      В 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>
      
      1. fet
        fet
        19.02.2021, 10:34
        спасибо, все работает, но проблема у меня в другом если таких объектов несколько и они без 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>
        
        
    2. Andrii Petrov
      Andrii Petrov
      04.11.2020, 00:23
      В радиобатонах фокус не работает.
      1. Andrii Petrov
        Andrii Petrov
        04.11.2020, 00:43
        А именно если выбран хотя бы один чекбокс. Фокус на него и всё.
        1. Александр Мальцев
          Александр Мальцев
          04.11.2020, 03:43
          Радиокнопка в состоянии фокуса:
          радиокнопка в состоянии фокуса
          Это устанавливается с помощью следующих стилей:
          /* стили для радиокнопки, находящейся в фокусе */
          .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;
          }
          
          Если вам нужно как-то по-другому, то измените их соответствующим образом.
          1. Инна
            Инна
            07.11.2020, 11:24
            Может не по теме, но как сделать так что бы флажки стояли в 2 столбика не в разброс
            1. Александр Мальцев
              Александр Мальцев
              07.11.2020, 11:43
              Это можно сделать разными способами.
              Например, с использованием 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>
              
              1. Инна
                Инна
                07.11.2020, 13:31
                Надо вот так (скриншот). Перепробовала все что нашла в интернете, результата такого не добилась. Уходит второй столбик на середину страницы, а надо рядом с первым столбом.
                Спасибо за ответ!
                1. Александр Мальцев
                  Александр Мальцев
                  07.11.2020, 14:24
                  Тогда можно так:
                  <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>
                  
      2. Ivan
        Ivan
        21.10.2020, 14:07
        Зарегистрировался только для того, чтобы поблагодарить автора за написание этой статьи. Спасибо!
        1. Александр Мальцев
          Александр Мальцев
          21.10.2020, 15:39
          Спасибо за отзыв!
        2. Денис
          Денис
          22.08.2020, 22:40
          Здравствуйте. При использовании кода

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