项目中要用到倒计时,用Vue 实现了一个
1 <template>
2 <transition name="bkcd">
3 <div class="bkCountDown" v-show="bkCountDownShow">
4 <div class="kbCountDownTitle">
5 <img src="http://static.crecgec.com/Kaipiao/countDownTitle.png">
6 </div>
7 <div id="kbCountDownContent" class="kbCountDownContent" ref="kbCountDownContent">
8 </div>
9 <!--倒计时结束后提示的信息-->
10 <div class="cdEndCon" v-show="cdEndConShow">{{cdEndContent}}</div>
11 </div>
12 </transition>
13 </template>
14
15 <script>
16 import $ from 'jquery'
17
18 export default {
19 props: {
20 // 控制倒计时页面显示、隐藏
21 bkCountDownShow: {
22 type: Boolean,
23 default: true
24 },
25 // 这个参数:为了实现中途倒计时暂停功能
26 // 控制倒计时暂停/开始
27 cdStartOrEnd: {
28 type: Boolean,
29 default: true
30 },
31 // 倒计时的时间,有父组件传递
32 countDownTime: {
33 type: String,
34 default: '2017/11/9 15:03:01'
35 },
36 // 倒计时结束后显示的内容
37 cdEndContent: {
38 type: String,
39 default: '倒计时已经结束'
40 }
41 },
42 data () {
43 return {
44 // 倒计时结束后显示cdEndCon
45 cdEndConShow: false,
46 timestamp: '', // 倒计时的时间戳
47 cdTimer: '', // setTimeOut值
48 timeInterval: '', // 倒计时结束时间与当前时间的之间的间隔
49 timeIntervalVal: '', // 保存时间间隔的参数
50 d: '',
51 h: '',
52 m: '',
53 s: '',
54 days: 24 * 60 * 60,
55 hours: 60 * 60,
56 minutes: 60
57 }
58 },
59 mounted () {
60 this.countdown()
61 },
62 watch: {
63 // 监控cdStartOrEnd值
64 cdStartOrEnd () {
65 if (this.cdStartOrEnd) {
66 this.tick()
67 } else {
68 clearTimeout(this.cdTimer)
69 }
70 }
71 },
72 methods: {
73 countdown () {
74 this.timestamp = new Date(this.countDownTime).getTime()
75 this.init('kbCountDownContent')
76 },
77 // 初始化
78 init (ele) {
79 $.each(['Hours', 'Minutes', 'Seconds'], function (i) {
80 $('<div class="count' + this + '">').html(
81 `<div class = "countPos">\
82 <span class="digit static">0</span>\
83 </div>\
84 <div class="countPos">\
85 <span class="digit static">0</span>\
86 </div>`
87 ).appendTo($('#' + ele))
88 if (this !== 'Seconds') {
89 $('#' + ele).append('<div class="countDiv countDiv' + i + '"></div>')
90 }
91 })
92 this.tick()
93 },
94 tick () {
95 // 每次进入这个方法,就重新计算和当前时间的间隔,然后赋值给timeInterval
96 this.timeInterval = Math.floor((this.timestamp - (new Date())) / 1000)
97 if (this.timeInterval < 0) {
98 this.timeInterval = 0
99 }
100 this.timeIntervalVal = this.timeInterval
101 // Number of days left
102 // 现在是只有时分秒,可以通过调整下面的代码,来确定显示什么
103 // this.d = Math.floor(this.timeInterval / this.days)
104 // this.updateDuo(0, 1, this.d)
105 // this.timeInterval -= this.d * this.days
106 // Number of hours left
107 this.h = Math.floor(this.timeInterval / this.hours)
108 this.updateDuo(0, 1, this.h)
109 this.timeInterval -= this.h * this.hours
110 // Number of minutes timeInterval
111 this.m = Math.floor(this.timeInterval / this.minutes)
112 this.updateDuo(2, 3, this.m)
113 this.timeInterval -= this.m * this.minutes
114 // Number of seconds timeInterval
115 this.s = this.timeInterval
116 this.updateDuo(4, 5, this.s)
117 // timeIntervalVal大于0,就执行setTimeout方法
118 if (this.timeIntervalVal > 0) {
119 this.cdTimer = setTimeout(this.tick, 1000)
120 } else {
121 // 倒计时结束
122 this.cdEndConShow = true
123 // 这块可以添加emit,给父组件传参
124 // 通过emit给父组件传参数来操作bkCountDownShow
125 // bkCountDownShow = false
126 }
127 },
128 updateDuo (minor, major, value) {
129 this.switchDigit($('#kbCountDownContent').find('.countPos').eq(minor), Math.floor(value / 10) % 10)
130 this.switchDigit($('#kbCountDownContent').find('.countPos').eq(major), value % 10)
131 },
132 switchDigit (position, number) {
133 let digit = position.find('.digit')
134 if (digit.is(':animated')) {
135 return false
136 }
137 if (position.data('digit') === number) {
138 return false
139 }
140 position.data('digit', number)
141 var replacement = $('<span>', {
142 'class': 'digit',
143 css: {
144 top: '-170px',
145 opacity: 0
146 },
147 html: number
148 })
149 digit
150 .before(replacement)
151 .removeClass('static')
152 .animate({top: '170px', opacity: 0}, 'slow', function () {
153 digit.remove()
154 })
155 replacement
156 .delay(100)
157 .animate({top: 0, opacity: 1}, 'slow', function () {
158 replacement.addClass('static')
159 })
160 }
161 }
162 }
163 </script>
164
165 <!-- Add "scoped" attribute to limit CSS to this component only -->
166 <style>
167 *{
168 margin:0;
169 padding:0;
170 font-family: 'Microsoft Yahei',Tahoma,'Simsun','宋体' !important;
171 }
172
173 .bkCountDown{
174 width: 100%;
175 height: 980px;
176 background:url('http://static.crecgec.com/Kaipiao/background.png') #b0b0b0;
177 position: absolute;
178 background-size: cover;
179 overflow: hidden;
180 }
181 .kbCountDownTitle{
182 width: 1070px;
183 height: 120px;
184 line-height: 120px;
185 font-size: 120px;
186 margin: 190px auto 0;
187 text-align: center;
188 color: #fff;
189 }
190 .kbCountDownContent{
191 width:1070px;
192 margin:160px auto 0;
193 text-align:center;
194 letter-spacing:-3px;
195 overflow: hidden;
196 }
197 .countPos{
198 display: inline-block;
199 width: 150px;
200 height: 170px;
201 overflow: hidden;
202 position: relative;
203 margin-left: 15px;
204 }
205
206 .digit{
207 position:absolute;
208 display:block;
209 width:150px;
210 height: 170px;
211 line-height: 170px;
212 text-align:center;
213 color:#fff;
214 font-size: 80px;
215 background: url('http://static.crecgec.com/Kaipiao/countDown.png') 0 0 no-repeat;
216 }
217
218 .digit.static{
219 background: url('http://static.crecgec.com/Kaipiao/countDown.png') 0 0 no-repeat;
220 }
221 .countDays,.countHours,.countMinutes,.countSeconds{
222 float: left;
223 font-size: 0;
224 }
225 .countDiv{
226 display:inline-block;
227 width:10px;
228 height:50px;
229 float: left;
230 margin-top: 60px;
231 margin-left: 15px;
232 background: url('http://static.crecgec.com/Kaipiao/countDown1.png') 0 0 no-repeat;
233 }
234 .cdEndCon{
235 width:1070px;
236 margin:20px auto 0;
237 text-align: center;
238 color: #fff;
239 font-size: 20px;
240 }
241 .bkcd-enter-active, .bkcd-leave-active{
242 transition: opacity .5s
243 }
244 .bkcd-enter, .bkcd-leave-to{
245 opacity: 0
246 }
247 </style>