How to Make a Simple Vue Custom Select Component

Creating a custom select tag with its own styling is notoriously difficult. Sometimes it’s impossible to build from scratch without a combination of styled divs and custom JavaScript. In this article, you’ll learn how to create a custom select component in Vue that can be easily styled with your own CSS. In fact, it’s the same component we use in production on Qvault, and you can see it in action on our JavaScript playground.

Vue Custom Select Example

Code Sandbox Demo

Sorry for the interruption! I just wanted to mention that you should check out our new free coding course. You’ll learn the absolute basics of writing code in one of the most useful languages in the world, JavaScript.


<template> <div class="custom-select" :tabindex="tabindex" @blur="open = false"> <div class="selected" :class="{ open: open }" @click="open = !open"> {{ selected }} </div> <div class="items" :class="{ selectHide: !open }"> <div v-for="(option, i) of options" :key="i" @click=" selected = option; open = false; $emit('input', option); " > {{ option }} </div> </div> </div> </template>
Code language: HTML, XML (xml)

The following is important to note:

  • The tabindex property allows our component to be focused, which in turn allows it to be blurred. The blur event closes our component when a user clicks outside of the component.
  • By emitting the selected option using the ‘input’ parameter, the parent component can react to changes.

The JavaScript

<script> export default { props: { options: { type: Array, required: true, }, default: { type: String, required: false, default: null, }, tabindex: { type: Number, required: false, default: 0, }, }, data() { return { selected: this.default ? this.default : this.options.length > 0 ? this.options[0] : null, open: false, }; }, mounted() { this.$emit("input", this.selected); }, }; </script>
Code language: JavaScript (javascript)

Important things to note about the JavaScript:

  • We also emit the selected value on mount so that the parent doesn’t need to set the default value explicitly.
  • If our select component is a small part of a larger form, then we want to be able to set the correct tabindex.


<style scoped> .custom-select { position: relative; width: 100%; text-align: left; outline: none; height: 47px; line-height: 47px; } .custom-select .selected { background-color: #0a0a0a; border-radius: 6px; border: 1px solid #666666; color: #fff; padding-left: 1em; cursor: pointer; user-select: none; } .custom-select { border: 1px solid #ad8225; border-radius: 6px 6px 0px 0px; } .custom-select .selected:after { position: absolute; content: ""; top: 22px; right: 1em; width: 0; height: 0; border: 5px solid transparent; border-color: #fff transparent transparent transparent; } .custom-select .items { color: #fff; border-radius: 0px 0px 6px 6px; overflow: hidden; border-right: 1px solid #ad8225; border-left: 1px solid #ad8225; border-bottom: 1px solid #ad8225; position: absolute; background-color: #0a0a0a; left: 0; right: 0; z-index: 1; } .custom-select .items div { color: #fff; padding-left: 1em; cursor: pointer; user-select: none; } .custom-select .items div:hover { background-color: #ad8225; } .selectHide { display: none; } </style>
Code language: CSS (css)

This CSS is just an example, it’s what we use in the Qvault app. Feel free to change the styling to whatever your needs are.

If you put the three sections together in the same file, you will be left with a well-encapsulated custom select single file component! Copying and pasting code gets a bad rep, it’s usually much better than importing libraries, especially for visual components. When you do “steal” code from somewhere, always be sure you understand it!

Try to practice good engineering and computer science principles by encapsulating your components and building well-thought-out APIs. The jobs you will qualify for will be that much better, and you’ll write better code as a result!

How to make other custom Vue components

Before we move on, if you are looking for our other custom Vue.js components tutorials you can find them here:

Have questions or feedback?

Follow and hit me up on Twitter @q_vault if you have any questions or comments. If I’ve made a mistake in the article, please let me know so I can get it corrected!