Hire Me! I'm currently looking for my next role in developer relations and advocacy. If you've got an open role and think I'd be a fit, please reach out. You can also find me on LinkedIn.

A common user interface feature you'll see on sites is a counter by form fields to let you know how much you've typed. Typically this is when a certain minimum or maximum number of characters are required. Instead of simply displaying an error ("You haven't typed enough, dangit!"), this feature will give you a "live" update as you type of how many characters have been typed so far. Here's a quick example of how to do this in Vue.js.

First, let's consider a use case where we require a certain number of characters. You can start with a simple HTML field:

<input type="text" v-model="description" minlength=10>

Even though I'm not going to do a traditional form submission, I wanted to use minlength anyway as it's a well-supported feature of HTML forms. I've bound the field to a Vue data value named description. Now let's show the field with the rest of the layout.

<div id="app" v-cloak>
  Enter a minimum of 10 characters please:
  <input type="text" v-model="description" minlength=10>
  {{ currentLength }} characters
</div>

I've added a bit of descriptive text to clearly tell the user what they need to do but I've also added a character counter after the field itself. Now let's look at the JavaScript:

const app = new Vue({
  el:'#app',
  data:{
    description:''
  },
  computed: {
    currentLength() {
      return this.description.length;
    }
  }
})

As you can see description is just simple data, but currentLength is a computed property based on the field itself.

And that's it. Simple. But let's make it a bit more fancy.

Fancy cat is fancy

Here I've modified the character count to add a span with some classes applied:

<span class="count" :class="{bad:isBad}">
{{ currentLength }} characters
</span>

Notice that the bad class is only applied when isBad is true. Here's the CSS I used:

.count {
  font-style: italic;
}

.bad {
  color: red;
}

And my new computed property:

isBad() { return this.currentLength < 10; }

Now when the user has less than ten characters, the bad class is applied and clearly signifies that the data isn't ready yet. You can play with the completed version below:

See the Pen Char Count (minimum) by Raymond Camden (@cfjedimaster) on CodePen.

Now let's flip it and write count that flags a maximum number of characters. First, the HTML:

<div id="app" v-cloak>
  Enter a maximum of 100 characters please:
  <input type="text" v-model="description" maxlength=100>
  <span class="count" :class="{bad:isBad}">
   characters
  </span>
</div>

Note the help text and the use of maxlength. Unlike minlength which won't have any impact unless the user submits the form, this time the user will immediately be stopped typing when they hit the max. So while most of the code is the same, I modified isBad like so:

isBad() { return this.currentLength > 90; }

Now it returns true when the length is greater than 90. Instead of being a flag of "you have incorrect data", it's more of a warning that you're about to hit your max. Here's a demo of that version:

See the Pen Char Count (maximum) by Raymond Camden (@cfjedimaster) on CodePen.