DEV Community

Cover image for Making a submit button with loader - WotW
Eder Díaz
Eder Díaz

Posted on • Edited on

Making a submit button with loader - WotW

Welcome to the second installment of the Widget of the Week series.

This time I'll show you the process to make a submit button that transforms to a loader and then confirms your submission.

The inspiration for this widget is this and looks like this:

submit button

Preparations

For today's widget we will be using vue.js and tweenlite for animations. Both libraries have a cdn link to be imported on any project.

The HTML structure

The markup for this widget is really simple, we just need a container where our Vue instance will mount and inside it, there will be a button and a couple of SVG circles for the loading animation:

<div id="submit-button" class="submit-container">
  <div class="submit-btn">
    <span>Submit</span>    
  </div>

  <!--  grey circle  -->
  <svg class="loader-svg">
    <path stroke="#CCCCCC" fill="none" stroke-width="4" d="M25,2.5A22.5,22.5 0 1 1 2.5,25A22.5,22.5 0 0 1 25,2.5"></path>
  </svg>

  <!--  green circle  -->
  <svg class="loader-svg">
    <path stroke="#20BF7E" fill="none" stroke-width="4" d="M25,2.5A22.5,22.5 0 1 1 2.5,25A22.5,22.5 0 0 1 25,2.5"></path>
  </svg>
</div>
Enter fullscreen mode Exit fullscreen mode

Now let's start matching the style of our button with these CSS rules:

.submit-container {
  position: relative;
  margin: 80px;
}

.submit-btn {
  width: 100px;
  color: #20BF7E;
  font-size: 20px;
  font-family: Arial;
  text-align: center;
  text-decoration: none;
  padding: 10px 20px 10px 20px;
  border: solid #20BF7E 4px;
  text-decoration: none;
  cursor: pointer;
  border-radius: 25px;
  transition: background-color .3s, color .3s;
}

.submit-btn:hover {
  background-color: #20BF7E;
  color: white;
}
Enter fullscreen mode Exit fullscreen mode

We now have a button that has a hover animation that swaps colors between the green background and the white font. Also notice the 25px border radius property, it is going to be really important when we need our button to become a circle.

The interaction logic

Before we initialize the Vue instance, I'd like to check what are the states of the button. We can ignore the hover because that's already solved by our CSS, that leaves us with three states: clicked, loading and loaded. To handle those we can start with something like this:

new Vue({
  el: '#submit-button',
  data: {
    clicked: false,    
    loading: false,
    loaded: false
  }
})
Enter fullscreen mode Exit fullscreen mode

You might ask "Why three booleans and not a single string or number with the 3 values?", and the reason is because they're not mutually exclusive, in other words, the button can be 'clicked' and also 'loading' at the same time.

The click interaction

In preparation for the click animation we need first to create a CSS rule for the button, when it is clicked it transforms into a circle, to be precise a 50px by 50px circle (remember the 25px border radius?). The problem is that it already has a padding declared, and also we need to compensate for the border of the button, so we will need a little bit of math:

.submit-btn {
  ... other rules
  /* more CSS transitions for the props we need to animate */
  transition: width .3s, margin .3s, background-color .3s, color .3s;
}
.submit-btn.round {
  margin-left: 50px;
  border-color: #CCCCCC;
  background: white;

  /*  circle should be 50px width & height */
  /* borderLeft + paddingLeft + paddingRight + borderRight  */
  /* 4 + 20 + 20 + 4 = 48 + 2 = 50 */
  width: 2px; 
  /* borderTop + paddingTop + paddingBottom + borderBottom  */
  /* 4 + 10 + 10 + 4 = 28 + 22 = 50 */
  height: 22px;
}
Enter fullscreen mode Exit fullscreen mode

Now we can start binding the button to Vue, we will first bind the click to a method, the round class to a computed property, and also the submit text needs to disappear when we click the button:

...
<div 
  @click="clickedSubmit" 
  :class="buttonClass"
  class="submit-btn">
    <span v-show="!clicked">Submit</span>  
...
Enter fullscreen mode Exit fullscreen mode

Then in our js:

...
methods: {
  clickedSubmit () {
    this.clicked = true
  }
},
computed: {
  buttonClass () {
    if (this.clicked) {
      return 'round'
    }
    return ''
  }
}
Enter fullscreen mode Exit fullscreen mode

Pretty simple right? Now comes the tricky part.

The loading

Just after our button transforms into a circle we need to put on top our SVG circles, why?, because HTML borders can't be animated the way we need them to, but SVG can!
Now let's match the position of the circles with this CSS:

.loader-svg {
  pointer-events: none;
  position: absolute;
  top: 0px;
  left: 50px;
  width: 50px; 
  height: 50px; 
  transform-origin: 25px 25px 25px;
}
Enter fullscreen mode Exit fullscreen mode

and then, both SVG circles will have this vue binding, to make them appear when the button starts loading:


  <!--  grey circle  -->
  <svg v-if="loading" class="loader-svg">
    ...
  </svg>

  <!--  green circle  -->
  <svg v-if="loading" class="loader-svg">
    ...
  </svg>
Enter fullscreen mode Exit fullscreen mode

We need to know when the button animation ends so we can start the loading animation, according to MDN web docs we can use the 'transitionend' event.
To add a listener to that event in Vue, we need to have a reference to the submit button, let's add this line to our button HTML:

<div class="submit-btn" 
  ref="submit-btn"
  ...
>
Enter fullscreen mode Exit fullscreen mode

Now we can reference it in our clickedSubmit method like this:

...
clickedSubmit () {
  this.clicked = true    
  this.$refs['submit-btn']
    .addEventListener("transitionend", this.animateLoader, false);
}
...
Enter fullscreen mode Exit fullscreen mode

this will trigger the animateLoader method when the animation finishes, so let's create the method:

animateLoader () {
  this.loading = true
  this.$refs['submit-btn']
    .removeEventListener("transitionend", this.animateLoader, false);
  // TODO animate circles
}
Enter fullscreen mode Exit fullscreen mode

That will set the loading flag to true and remove the previously added listener.

Animating the green circle

For the next part we will use a SVG animation trick using the stroke-dasharray and stroke-dashoffset properties.
For the trick to work, the stroke-dasharray must have as a value the circumference of the circle, to calculate it we can go back to our geometry class notebook and see that the formula is pi times the diameter of the circle.
Ours is 50px width, so it will be 3.1416 * 50 = ~157. Also we will bind the stroke-dashoffset to a new Vue data variable:

...
<!--  green circle  -->
<svg v-if="loading" class="loader-svg">
  <path stroke="#20BF7E" fill="none" stroke-width="4" d="M25,2.5A22.5,22.5 0 1 1 2.5,25A22.5,22.5 0 0 1 25,2.5" 
  stroke-dasharray="157" :stroke-dashoffset="loaderOffset">
  </path>
</svg>
...
Enter fullscreen mode Exit fullscreen mode

Now in our Vue instance we will declare, inside the data object, the loaderOffset property and initialize it with the same value 157:

data: {
  clicked: false,    
  loading: false,
  loaded: false,
  loaderOffset: 157
}
Enter fullscreen mode Exit fullscreen mode

After doing that, we can start animating the loader with TweenLite.
We use the TweenLite.to() method to interpolate the loaderOffset property from its initial value to zero in two seconds.
When it finishes animating, the onComplete hook will execute the completeLoading method where we set the loading and loaded properties:

...
animateLoader () {
  this.loading = true
  this.$refs['submit-btn']
    .removeEventListener("transitionend", this.animateLoader, false);

  // animate the loaderOffset property,
  // on production this should be replaced 
  // with the real loading progress
  TweenLite.to(this, 2, {
    loaderOffset: 0, // animate from 157 to 0
    ease: Power4.easeInOut,
    onComplete: this.completeLoading // execute this method when animation ends
  })
},
completeLoading () {
  this.loading = false
  this.loaded = true
}
...
Enter fullscreen mode Exit fullscreen mode

The loaded state

For the last part we just need to create another CSS rule that will be used when our button is loaded:

.submit-btn.loaded {
  color: white;
  background-color: #20BF7E;
}
Enter fullscreen mode Exit fullscreen mode

and then the buttonClass computed property should handle that case too:

...
buttonClass () {
  if (this.loaded) {
    return 'loaded'
  }

  if (this.clicked) {
    return 'round'
  }

  return ''
}
...
Enter fullscreen mode Exit fullscreen mode

We already declared the CSS transitions for those properties so we don't need to do anything else.

And now the final result!

That’s it for the second Widget of the Week.

If you haven't checked the previous one, here it is.

Also if you want to see a specific widget for next week, post it in the comments section.

Top comments (0)