timer.vue 2.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100
  1. <template>
  2. <div class="countdown">
  3. <div v-show="days > 0" class="countdown-item days">{{ days }} 天</div>
  4. <div class="countdown-item hours">{{ doubleDigit(hours) }}</div>
  5. <div class="paddingB2">:</div>
  6. <div class="countdown-item minutes">{{ doubleDigit(minutes) }}</div>
  7. <div class="paddingB2">:</div>
  8. <div class="countdown-item seconds">{{ doubleDigit(seconds) }}</div>
  9. </div>
  10. </template>
  11. <script>
  12. export default {
  13. props: {
  14. targetTime: {
  15. type: String,
  16. required: true
  17. }
  18. },
  19. data() {
  20. return {
  21. days: 0,
  22. hours: 0,
  23. minutes: 0,
  24. seconds: 0
  25. }
  26. },
  27. computed: {
  28. doubleDigit() {
  29. return n => (n < 10 ? `0${n}` : n)
  30. }
  31. },
  32. mounted() {
  33. this.starCountTime()
  34. },
  35. watch: {
  36. targetTime: {
  37. deep: true,
  38. handler(newVal, oldVal) {
  39. this.starCountTime()
  40. }
  41. }
  42. },
  43. methods: {
  44. starCountTime() {
  45. this.countDown()
  46. this.intervalId = setInterval(() => this.countDown(), 1000)
  47. },
  48. countDown() {
  49. const targetDate = new Date(this.targetTime)
  50. const nowDate = new Date()
  51. const diffSeconds = Math.round((targetDate - nowDate) / 1000)
  52. if (diffSeconds < 0) {
  53. clearInterval(this.intervalId)
  54. this.$emit('timeDone')
  55. return
  56. }
  57. this.days = Math.floor(diffSeconds / (3600 * 24))
  58. this.hours = Math.floor((diffSeconds % (3600 * 24)) / 3600)
  59. this.minutes = Math.floor((diffSeconds % 3600) / 60)
  60. this.seconds = diffSeconds % 60
  61. }
  62. },
  63. beforeDestroy() {
  64. clearInterval(this.intervalId)
  65. }
  66. }
  67. </script>
  68. <style lang="scss">
  69. .countdown {
  70. display: flex;
  71. align-items: center;
  72. justify-content: center;
  73. }
  74. .countdown-item {
  75. font-size: 22rpx;
  76. text-align: center;
  77. margin: 0 6rpx;
  78. }
  79. .countdown-item.days {
  80. font-size: 22rpx;
  81. margin-right: 8rpx;
  82. }
  83. .countdown-item.hours,
  84. .countdown-item.minutes,
  85. .countdown-item.seconds {
  86. color: white;
  87. width: 35rpx;
  88. height: 42rpx;
  89. line-height: 42rpx;
  90. background: $color-theme;
  91. border-radius: 8rpx;
  92. }
  93. </style>