The Startup

Get smarter at building your thing. Follow to join The Startup’s +8 million monthly readers & +772K followers.

Follow publication

Build a Checkbox Component With Vue 3, Font Awesome, & Tailwind CSS

Kevin Bigelow
The Startup
Published in
5 min readDec 14, 2020

--

Takeaway:

Get your feet wet with some of the new Vue 3 features while building a fully functional checkbox component. We’ll be using Font Awesome to replace the default html checkbox and Tailwind CSS to handle styling.

Code Sandbox:

To make it easy to code-along, I’ve put together a couple code sandboxes.

Lets get started

In the Starting Point sandbox we’ve got a basic checkbox input set up in ./components/checkbox.vue

<template>
<input type="checkbox" :id="fieldId" class="mr-2" /> <label
:for="fieldId"
class="flex flex-row items-center font-bold cursor-pointer"
>
{{ label }}
</label>
</template>
<script>
export default {
props: {
label: String,
fieldId: {
type: String,
required: true,
},
},
};
</script>

We’re importing the checkbox.vue component into App.vue like so:

<check-box fieldId=”volume” label=”Volume On” />

It seems to be working great. Only thing is, this checkbox is not tied to any data 😱

We’ll take the following steps to get this checkbox working:

  1. Add data to App.vue and use v-model to tie the data to <check-box>
  2. In checkbox.vue sync the <input> checked prop to the value passed in by v-model
  3. Replace the default checkbox with Font Awesome icons

Step 1

To add the data to App.vue we could use the Options API, which would be what you’re used to if you’ve been using Vue for a while now.

But for the sake of learning something new, let’s use Vue3’s Composition API.

Everything bold in the code snippet below is what we’ll add to App.vue

<script>
import Checkbox from "./components/checkbox.vue";
import { ref } from "vue";
export default {
name: "App",
setup() {
const volumeOn = ref(true);
return {
volumeOn,
};
},

components: {
"check-box": Checkbox,
},
};
</script>

setup() is the entry point for the composition API.

We declare our variable volumeOn and make it reactive by using ref() Notice that we need to import ref in order to use it. More on refs and reactivity here, it is vital read up on this aspect of Vue3.

We return volumeOn in order for the template to access the variable.

Let’s use v-model to bind the variable to the checkbox component.

<check-box
fieldId="volume"
v-model:checked="volumeOn"
label="Volume On"
/>

With Vue3 we can now pass arguments tov-model! We are passing checked as an argument. If we omitted this argument, volumeOn’s value would be passed to the child component as value — but in our case checked will be passed. Its worth noting that v-model arguments allow for multiple v-models to be used on a component. More on that here.

Step 2

We need to update our checkbox.vue component to utilize the data we are passing in. First we’ll add a prop called checked with the expected value type to be a Boolean This will catch the value passed by v-model:checked

<script>
export default {
props: {
label: String,
fieldId: {
type: String,
required: true,
},
checked: {
type: Boolean,
},

},
};
</script>

Now let’s tie the input’s checked property to our checked prop.

fyi 👉 :checked is shorthand for v-bind:checked

<template>
<input
type="checkbox"
:checked="checked"
:id="fieldId"
class="mr-2"
/>
<label
:for="fieldId"
class="flex flex-row items-center font-bold cursor-pointer"
>
{{ label }}
</label>
</template>

Test it out by changing const volumeOn = ref(true); to false and reloading. The checkbox should now be unchecked by default.

It looks like everything might be working alright, but let’s see if the value of volumenOn is changing at all when we click the box.

The quickest way in codesandbox would be by adding {{ volumeOn }} to the template somewhere.

Spoiler alert, the value does not change 😥 We need to emit the checked value from the checkbox component to sync the value

<input
@input="(event) => $emit('update:checked', event.target.checked)"
type="checkbox"
:checked="checked"
:id="fieldId"
class="mr-2"
/>

update:checked is a custom event synced to v-model:checked event.target.checked is the value we’re passing.

Now everything should be working with our checkbox.

If {{ volumeOn }} is not showing the proper value at this point, reload CodeSandbox, it should work fine.

Step 3

Finally, let’s get rid of those default checkbox styles, and replace them with Font Awesome icons.

We’ll add an icon in the label tag in checkbox.vue and hide the input with a Tailwind CSS class.

<input
@input="(event) => $emit('update:checked',event.target.checked)"
type="checkbox"
:checked="checked"
:id="fieldId"
class="mr-2 hidden"
/>
<label
:for="fieldId"
class="flex flex-row items-center font-bold text-gray-800 cursor-pointer select-none"
>
<i class="fa fa-check-square text-blue-600 mr-2"></i>
{{ label }}
</label>

Here’s how the icon break down:

  1. fa provides all the default font awesome styles
  2. fa-check-square will give us a nice little check in a square, there are thousands of icons to pick from, check them out here
  3. text-blue-600 is a Tailwind CSS class, see all the options they have here
  4. mr-2 will give us a little right margin

But wait; we don’t want a blue checkbox when the checkbox is unchecked. Let’s add some logic to this class list for an unchecked box using Vue.

<i
class="fa mr-2"
:class="{
'fa-check-square text-blue-600': checked,
'fa-square text-gray-500': !checked,
}"
></i>

We’ve added :class for any dynamic classes which will allow us to utilize our component’s props or other data to determine which classes to use. Learn more here.

And there you have it, a fully functional checkbox component.

🔥 Update 🔥

Want to bind multiple custom checkboxes to the same array?

We can achieve this with just a few adjustments to our code for the single checkbox. Check it out here

--

--

The Startup
The Startup

Published in The Startup

Get smarter at building your thing. Follow to join The Startup’s +8 million monthly readers & +772K followers.

Responses (2)

Write a response