<div id="app">
<div class="card">
<header class="card-header has-background-grey-darker">
<p class="card-header-title has-text-success">COUNTDOWN TIMER</p>
</header>
<div class="card-image">
<svg xmlns="http://www.w3.org/2000/svg" viewbox="0 0 260 250" width="260" height="250">
<rect x="5" y="5" width="250" height="250" fill="orangered" />
<circle cx="130" cy="125" r="80" stroke="lightsalmon" stroke-width="10" fill="none" />
<circle cx="130" cy="125" r="80" stroke="limegreen" :stroke-dasharray="dasharray" stroke-offset="600"
stroke-width="10" fill="none" transform="rotate(270,130,125)" />
<text x="84" y="140" fill="white" font-size="40">{{ minute | formatTime }}:{{ second | formatTime }}</text>
</svg>
</div>
<div class="card-content">
<div class="field is-horizontal">
<div class="field-label">
<label class="label is-size-7">MINUTES:</label>
</div>
<div class="field-body">
<div class="field">
<div class="control">
<input class="input is-success is-small" :disabled="state==='started' || state==='paused'" @change="updateTime"
v-model="minute" type="number" name="minutes" min="0" max="59" step="1">
</div>
</div>
</div>
</div>
<div class="field is-horizontal">
<div class="field-label">
<label class="label is-size-7">SECONDS:</label>
</div>
<div class="field-body">
<div class="field">
<div class="control">
<input class="input is-success is-small" :disabled="state==='started' || state==='paused'" @change="updateTime"
v-model="second" type="number" name="seconds" min="0" max="59" step="1">
</div>
</div>
</div>
</div>
</div>
<footer class="card-footer">
<div class="buttons has-addons card-footer-item">
<button class="button is-success" :disabled="state==='started' || second==0 && minute==0" @click="start"><span>Start</span></button>
<button class="button is-success" :disabled="state!=='started'" @click="pause">Pause</button>
<button class="button is-success" :disabled="state!=='started' && state !== 'paused'" @click="stop">Stop</button>
</div>
</footer>
</div>
</div>
new Vue({
el: '#app',
circumference: 2 * Math.PI * 80,
data: {
state: 'stopped',
minute: 0,
second: 0,
progress: 0,
timeInSeconds: 0
},
computed: {
dasharray(){
return this.progress + " " + this.$options.circumference
},
},
methods: {
updateTime(){
this.timeInSeconds = Number(this.minute) * 60 + Number(this.second)
},
start() {
this.state = "started";
if (this.progress == 0){
this.progress = this.$options.circumference;
}
this._tick();
this.interval = setInterval(this._tick, 1000);
},
pause() {
this.state = "paused";
clearInterval(this.interval);
},
stop() {
this.state = "stopped";
clearInterval(this.interval);
this.minute = 0;
this.second = 0;
this.progress = 0;
},
_tick: function() {
if (this.minute == 0 && this.second == 0){
this.stop()
}
let delta = (this.$options.circumference / this.timeInSeconds)
if ((this.progress - delta) < (delta / 2)){
this.progress = 0
} else {
this.progress -= delta
}
if (this.second !== 0) {
this.second--;
return;
}
if (this.minute !== 0) {
this.minute--;
this.second = 59;
}
}
},
filters: {
formatTime: function(value) {
if (value >= 10) {
return value;
}
return "0" + value;
}
}
})