---
layout: post
title: "Easy Custom Radio Inputs"
date: 2019-01-21
---
Default radio inputs are notoriously horrible looking and are something designers tend to over-think when trying to customize them. Let's walk through how to create custom radio buttons with *pure CSS*, while still preserving performance and accessibility.
## The Final Product
This is what we will be designing:
![Custom radio inputs](/public/images/custom-radio-inputs.png)
[Live CodePen Example](https://codepen.io/bradleytaunt/pen/oNjwMyX)
- - -
## The bones of our radio inputs (HTML)
I know it looks like a lot is going on here, but it's pretty straightforward so let's unpackage line by line:
### Radio inputs
This is the default `radio` input. We give it:
- a `name` (inputs with a shared `name` are grouped together)
- an `id` (so our label can target this input)
- a `class` (so we can style it later)
**Important**: be sure to have a unique `id` for each input so your labels don't end up connected to multiple radios. In this demo we are simply incrementing them by one.
### Labels
Adding the labels is fairly straightforward, we just include the corresponding input's `id` in the label's `for` attribute. The label content is wrapped in a `span` - which I will explain the reasoning for later.
For styling purposes we also add the `radio-label` class.
This is looking pretty terrible - but that's nothing some good ol' CSS can't fix!
## The flesh of our radio inputs (CSS)
First we give some basic styling to our `label` and `input` classes (along with hover states). The `radio` element is actually hidden from view, but by using the `visibility` attribute we still keep it accessible for screen-readers.
.radio-label {
background: white;
border: 1px solid #eee;
border-radius: 5px;
box-shadow: 0 2px 4px rgba(0,0,0,0.05);
cursor: pointer;
display: inline-block;
font-weight: 600;
margin: 0 auto 10px;
/* This 65px padding makes room for the custom input */
padding: 20px 20px 20px 65px;
position: relative;
transition: .3s ease all;
width: 100%;
}
.radio-label:hover {
box-shadow: 0 4px 8px rgba(0,0,0,0.05);
}
.radio-btn {
position: absolute;
visibility: hidden;
}
Remember that `span` element inside the label? We set it's `user-select` property to `none` so we avoid any possible issue with the user selecting the text on-click:
.radio-label span {
-webkit-user-select: none;
-moz-user-select: none;
user-select: none;
}
Next we include the default empty selection element (to mimic the original radio input) via a pseudo element:
.radio-label:before {
background: #eee;
border-radius: 50%;
content:'';
height: 30px;
left: 20px;
position: absolute;
/* Half the height of it's parent minus half of it's own height */
top: calc(50% - 15px);
transition: .3s ease background-color;
width: 30px;
}
## A Few Final Steps
The final step is adding the custom styling for when an `input` item is selected (`:checked`).
You will notice the use of a `base64` element for the custom checkmark - feel free to subsitute this for an actual image or none at all (this is just my personal design preference).
.radio-btn:checked + .radio-label {
background: #ECF5FF;
border-color: #4A90E2;
}
.radio-btn:checked + .radio-label:before {
background-color: #4A90E2;
background-image: url('');
background-repeat: no-repeat;
background-position: center;
background-size: 15px;
}
**And that's it.**
For easier reference the entire CSS file can be found below:
.radio-label {
background: white;
border: 1px solid #eee;
border-radius: 5px;
box-shadow: 0 2px 4px rgba(0,0,0,0.05);
cursor: pointer;
display: inline-block;
font-weight: 600;
margin: 0 auto 10px;
padding: 20px 20px 20px 65px;
position: relative;
transition: .3s ease all;
width: 100%;
}
.radio-label:hover {
box-shadow: 0 4px 8px rgba(0,0,0,0.05);
}
.radio-label:before {
background: #eee;
border-radius: 50%;
content:'';
height: 30px;
left: 20px;
position: absolute;
top: calc(50% - 15px);
transition: .3s ease background-color;
width: 30px;
}
.radio-label span {
-webkit-user-select: none;
-moz-user-select: none;
user-select: none;
}
.radio-btn {
position: absolute;
visibility: hidden;
}
.radio-btn:checked + .radio-label {
background: #ECF5FF;
border-color: #4A90E2;
}
.radio-btn:checked + .radio-label:before {
background-color: #4A90E2;
background-image: url('');
background-repeat: no-repeat;
background-position: center;
background-size: 15px;
}
- - -
## But wait - we can get even fancier!
Since this demo is based off a survey-type questionaire, wouldn't it be interesting to give the different selectable options their own styling based on their context? Take a look at the further customized version below:
We can do so by adding `positive`, `neutral` and `negative` class names to the radio inputs with their own respective properties:
.radio-btn.positive:checked + .radio-label {
background: #EAFFF6;
border-color: #32B67A;
}
.radio-btn.positive:checked + .radio-label:before {
background-color: #32B67A;
}
.radio-btn.neutral:checked + .radio-label:before {
background-image: url('');
}
.radio-btn.negative:checked + .radio-label {
background: #FFF2F2;
border-color: #E75153;
}
.radio-btn.negative:checked + .radio-label:before {
background-color: #E75153;
background-image: url('');
}
I hope this shows new designers that simple custom radio inputs aren't so hard to implement after-all and can actually be pretty fun to design.