CountDown.vue 2.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103
  1. <template>
  2. <span>
  3. {{ lastTime | format }}
  4. </span>
  5. </template>
  6. <script>
  7. function fixedZero(val) {
  8. return val * 1 < 10 ? `0${val}` : val
  9. }
  10. export default {
  11. name: 'CountDown',
  12. props: {
  13. format: {
  14. type: Function,
  15. default: undefined
  16. },
  17. target: {
  18. type: [Date, Number],
  19. required: true,
  20. },
  21. onEnd: {
  22. type: Function,
  23. default: () => {
  24. }
  25. }
  26. },
  27. data() {
  28. return {
  29. dateTime: '0',
  30. originTargetTime: 0,
  31. lastTime: 0,
  32. timer: 0,
  33. interval: 1000
  34. }
  35. },
  36. filters: {
  37. format(time) {
  38. const hours = 60 * 60 * 1000
  39. const minutes = 60 * 1000
  40. const h = Math.floor(time / hours)
  41. const m = Math.floor((time - h * hours) / minutes)
  42. const s = Math.floor((time - h * hours - m * minutes) / 1000)
  43. return `${fixedZero(h)}:${fixedZero(m)}:${fixedZero(s)}`
  44. }
  45. },
  46. created() {
  47. this.initTime()
  48. this.tick()
  49. },
  50. methods: {
  51. initTime() {
  52. let lastTime = 0
  53. let targetTime = 0
  54. this.originTargetTime = this.target
  55. try {
  56. if (Object.prototype.toString.call(this.target) === '[object Date]') {
  57. targetTime = this.target
  58. } else {
  59. targetTime = new Date(this.target).getTime()
  60. }
  61. } catch (e) {
  62. throw new Error('invalid target prop')
  63. }
  64. lastTime = targetTime - new Date().getTime()
  65. this.lastTime = lastTime < 0 ? 0 : lastTime
  66. },
  67. tick() {
  68. const {onEnd} = this
  69. this.timer = setTimeout(() => {
  70. if (this.lastTime < this.interval) {
  71. clearTimeout(this.timer)
  72. this.lastTime = 0
  73. if (typeof onEnd === 'function') {
  74. onEnd()
  75. }
  76. } else {
  77. this.lastTime -= this.interval
  78. this.tick()
  79. }
  80. }, this.interval)
  81. }
  82. },
  83. beforeUpdate () {
  84. if (this.originTargetTime !== this.target) {
  85. this.initTime()
  86. }
  87. },
  88. beforeDestroy() {
  89. clearTimeout(this.timer)
  90. }
  91. }
  92. </script>
  93. <style scoped>
  94. </style>