spike-timer.vue 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142
  1. <template>
  2. <div v-if="timerFlag" class="countdown_none">
  3. {{ formattedDatetime }}
  4. </div>
  5. <div v-else class="countdown marginT4">
  6. <div class="countdown-digit">
  7. <div class="countdown-item hours">{{ doubleDigit(hours)[0] }}</div>
  8. <div class="countdown-item hours">{{ doubleDigit(hours)[1] }}</div>
  9. </div>
  10. <div class="paddingB2 marginX4" style="font-size: 64rpx">:</div>
  11. <div class="countdown-digit">
  12. <div class="countdown-item minutes">{{ doubleDigit(minutes)[0] }}</div>
  13. <div class="countdown-item minutes">{{ doubleDigit(minutes)[1] }}</div>
  14. </div>
  15. <div class="paddingB2 marginX4" style="font-size: 64rpx">:</div>
  16. <div class="countdown-digit">
  17. <div class="countdown-item seconds">{{ doubleDigit(seconds)[0] }}</div>
  18. <div class="countdown-item seconds">{{ doubleDigit(seconds)[1] }}</div>
  19. </div>
  20. </div>
  21. </template>
  22. <script>
  23. export default {
  24. props: {
  25. targetTime: {
  26. type: String,
  27. required: true
  28. },
  29. timerFlag: {
  30. type: Boolean,
  31. default: false
  32. }
  33. },
  34. data() {
  35. return {
  36. hours: 0,
  37. minutes: 0,
  38. seconds: 0,
  39. formattedDatetime: ''
  40. }
  41. },
  42. computed: {
  43. doubleDigit() {
  44. return (n) => (n < 10 ? `0${n}` : `${n}`)
  45. }
  46. },
  47. created() {
  48. this.starCountTime()
  49. },
  50. watch: {
  51. targetTime: {
  52. deep: true,
  53. handler(newVal, oldVal) {
  54. this.starCountTime()
  55. }
  56. }
  57. },
  58. methods: {
  59. starCountTime() {
  60. if (this.timerFlag) {
  61. this.formattedDatetime = this.greaterThanAday()
  62. } else {
  63. this.intervalId = setInterval(() => this.countDown(), 1000)
  64. }
  65. },
  66. countDown() {
  67. const targetDate = new Date(this.targetTime)
  68. const nowDate = new Date()
  69. const diffSeconds = Math.round((targetDate - nowDate) / 1000)
  70. if (diffSeconds < 0) {
  71. clearInterval(this.intervalId)
  72. this.$emit('timeDone')
  73. return
  74. }
  75. this.hours = Math.floor((diffSeconds % (3600 * 24)) / 3600)
  76. this.minutes = Math.floor((diffSeconds % 3600) / 60)
  77. this.seconds = diffSeconds % 60
  78. },
  79. greaterThanAday() {
  80. const date = new Date(this.targetTime)
  81. const month = date.getMonth() + 1
  82. const day = date.getDate()
  83. const hour = date.getHours()
  84. const minute = date.getMinutes()
  85. return `${month.toString().padStart(2, '0')}月${day
  86. .toString()
  87. .padStart(2, '0')}日 ${hour.toString().padStart(2, '0')}:${minute
  88. .toString()
  89. .padStart(2, '0')}`
  90. }
  91. },
  92. beforeDestroy() {
  93. clearInterval(this.intervalId)
  94. }
  95. }
  96. </script>
  97. <style lang="scss">
  98. @font-face {
  99. font-family: DS_DIGIB;
  100. src: url(https://supermart1.oss-cn-hangzhou.aliyuncs.com/resource/magic/DS-DIGIB-2.ttf);
  101. }
  102. .countdown {
  103. display: flex;
  104. align-items: center;
  105. justify-content: center;
  106. }
  107. .countdown-digit {
  108. display: flex;
  109. flex-direction: row;
  110. }
  111. .countdown-item {
  112. font-family: DS_DIGIB;
  113. font-size: 64rpx;
  114. text-align: center;
  115. margin: 0 8rpx;
  116. color: white;
  117. width: 56rpx;
  118. height: 92rpx;
  119. line-height: 92rpx;
  120. background: #000;
  121. border-radius: 8rpx;
  122. text-shadow: none;
  123. }
  124. .countdown_none {
  125. width: 472rpx !important;
  126. height: 90rpx !important;
  127. font-size: 64rpx;
  128. text-align: center;
  129. margin: 0 8rpx;
  130. color: white;
  131. width: 56rpx;
  132. height: 92rpx;
  133. line-height: 92rpx;
  134. text-shadow: none;
  135. }
  136. </style>