Eder Díaz blog

Blog

tips & tricks

Making a submit button with loader - WotW

April 11, 2018

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:

a green button that when clicked transforms into a ciruclar loader that fills a circle as it gets loaded

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>

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 0.3s, color 0.3s;
}

.submit-btn:hover {
  background-color: #20bf7e;
  color: white;
}

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,
  },
});

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 0.3s, margin 0.3s, background-color 0.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;
}

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>
  ...
</div>

Then in our js:

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

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;
}

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>

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" ...></div>

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

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

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
}

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>
...

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
}

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
}
...

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;
}

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

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

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

  return ''
}
...

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.

❮ Back to list