[[!meta title="Easy Custom Radio Inputs"]] [[!meta 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.