aboutsummaryrefslogtreecommitdiff
path: root/_posts/2019-01-21-easy-custom-radio-inputs.md
diff options
context:
space:
mode:
authorBradley Taunt <bt@btxx.org>2024-05-25 16:14:03 -0400
committerBradley Taunt <bt@btxx.org>2024-05-25 16:16:54 -0400
commite417a818e207a6cca6e2f3c471611673ab836a62 (patch)
tree664686a365c3d1e73349b5a667fa892f46445fef /_posts/2019-01-21-easy-custom-radio-inputs.md
Initial commit for Jekyll testing and conversion, updated
Diffstat (limited to '_posts/2019-01-21-easy-custom-radio-inputs.md')
-rw-r--r--_posts/2019-01-21-easy-custom-radio-inputs.md227
1 files changed, 227 insertions, 0 deletions
diff --git a/_posts/2019-01-21-easy-custom-radio-inputs.md b/_posts/2019-01-21-easy-custom-radio-inputs.md
new file mode 100644
index 0000000..b22d90c
--- /dev/null
+++ b/_posts/2019-01-21-easy-custom-radio-inputs.md
@@ -0,0 +1,227 @@
+---
+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)
+
+
+ <input class="radio-btn" name="radio-collection" id="radio-1" type="radio">
+ <label class="radio-label" for="radio-1"><span>I am very satisfied</span></label>
+
+ <input class="radio-btn" name="radio-collection" id="radio-2" type="radio">
+ <label class="radio-label" for="radio-2"><span>I am satisfied</span></label>
+
+ <input class="radio-btn" name="radio-collection" id="radio-3" type="radio">
+ <label class="radio-label" for="radio-3"><span>I am indifferent</span></label>
+
+ <input class="radio-btn" name="radio-collection" id="radio-4" type="radio">
+ <label class="radio-label" for="radio-4"><span>I am unsatisfied</span></label>
+
+ <input class="radio-btn" name="radio-collection" id="radio-5" type="radio">
+ <label class="radio-label" for="radio-5"><span>I am very unsatisfied</span></label>
+
+
+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
+
+
+ <input class="radio-btn" name="radio-collection" id="radio-1" type="radio">
+
+
+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.
+
+
+ <label class="radio-label" for="radio-1"><span>I am very satisfied</span></label>
+
+
+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.