/dist/vue-countdown.js

https://github.com/fengyuanchen/vue-countdown · JavaScript · 422 lines · 240 code · 49 blank · 133 comment · 28 complexity · 152d0c58cbc2ac35fac85da9d6560365 MD5 · raw file

  1. /*!
  2. * vue-countdown v1.1.5
  3. * https://fengyuanchen.github.io/vue-countdown
  4. *
  5. * Copyright 2018-present Chen Fengyuan
  6. * Released under the MIT license
  7. *
  8. * Date: 2020-02-25T01:19:32.769Z
  9. */
  10. (function (global, factory) {
  11. typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() :
  12. typeof define === 'function' && define.amd ? define(factory) :
  13. (global = global || self, global.VueCountdown = factory());
  14. }(this, (function () { 'use strict';
  15. var MILLISECONDS_SECOND = 1000;
  16. var MILLISECONDS_MINUTE = 60 * MILLISECONDS_SECOND;
  17. var MILLISECONDS_HOUR = 60 * MILLISECONDS_MINUTE;
  18. var MILLISECONDS_DAY = 24 * MILLISECONDS_HOUR;
  19. var EVENT_VISIBILITY_CHANGE = 'visibilitychange';
  20. var index = {
  21. name: 'countdown',
  22. data: function data() {
  23. return {
  24. /**
  25. * It is counting down.
  26. * @type {boolean}
  27. */
  28. counting: false,
  29. /**
  30. * The absolute end time.
  31. * @type {number}
  32. */
  33. endTime: 0,
  34. /**
  35. * The remaining milliseconds.
  36. * @type {number}
  37. */
  38. totalMilliseconds: 0
  39. };
  40. },
  41. props: {
  42. /**
  43. * Starts the countdown automatically when initialized.
  44. */
  45. autoStart: {
  46. type: Boolean,
  47. default: true
  48. },
  49. /**
  50. * Emits the countdown events.
  51. */
  52. emitEvents: {
  53. type: Boolean,
  54. default: true
  55. },
  56. /**
  57. * The interval time (in milliseconds) of the countdown progress.
  58. */
  59. interval: {
  60. type: Number,
  61. default: 1000,
  62. validator: function validator(value) {
  63. return value >= 0;
  64. }
  65. },
  66. /**
  67. * Generate the current time of a specific time zone.
  68. */
  69. now: {
  70. type: Function,
  71. default: function _default() {
  72. return Date.now();
  73. }
  74. },
  75. /**
  76. * The tag name of the component's root element.
  77. */
  78. tag: {
  79. type: String,
  80. default: 'span'
  81. },
  82. /**
  83. * The time (in milliseconds) to count down from.
  84. */
  85. time: {
  86. type: Number,
  87. default: 0,
  88. validator: function validator(value) {
  89. return value >= 0;
  90. }
  91. },
  92. /**
  93. * Transforms the output props before render.
  94. */
  95. transform: {
  96. type: Function,
  97. default: function _default(props) {
  98. return props;
  99. }
  100. }
  101. },
  102. computed: {
  103. /**
  104. * Remaining days.
  105. * @returns {number} The computed value.
  106. */
  107. days: function days() {
  108. return Math.floor(this.totalMilliseconds / MILLISECONDS_DAY);
  109. },
  110. /**
  111. * Remaining hours.
  112. * @returns {number} The computed value.
  113. */
  114. hours: function hours() {
  115. return Math.floor(this.totalMilliseconds % MILLISECONDS_DAY / MILLISECONDS_HOUR);
  116. },
  117. /**
  118. * Remaining minutes.
  119. * @returns {number} The computed value.
  120. */
  121. minutes: function minutes() {
  122. return Math.floor(this.totalMilliseconds % MILLISECONDS_HOUR / MILLISECONDS_MINUTE);
  123. },
  124. /**
  125. * Remaining seconds.
  126. * @returns {number} The computed value.
  127. */
  128. seconds: function seconds() {
  129. return Math.floor(this.totalMilliseconds % MILLISECONDS_MINUTE / MILLISECONDS_SECOND);
  130. },
  131. /**
  132. * Remaining milliseconds.
  133. * @returns {number} The computed value.
  134. */
  135. milliseconds: function milliseconds() {
  136. return Math.floor(this.totalMilliseconds % MILLISECONDS_SECOND);
  137. },
  138. /**
  139. * Total remaining days.
  140. * @returns {number} The computed value.
  141. */
  142. totalDays: function totalDays() {
  143. return this.days;
  144. },
  145. /**
  146. * Total remaining hours.
  147. * @returns {number} The computed value.
  148. */
  149. totalHours: function totalHours() {
  150. return Math.floor(this.totalMilliseconds / MILLISECONDS_HOUR);
  151. },
  152. /**
  153. * Total remaining minutes.
  154. * @returns {number} The computed value.
  155. */
  156. totalMinutes: function totalMinutes() {
  157. return Math.floor(this.totalMilliseconds / MILLISECONDS_MINUTE);
  158. },
  159. /**
  160. * Total remaining seconds.
  161. * @returns {number} The computed value.
  162. */
  163. totalSeconds: function totalSeconds() {
  164. return Math.floor(this.totalMilliseconds / MILLISECONDS_SECOND);
  165. }
  166. },
  167. render: function render(createElement) {
  168. return createElement(this.tag, this.$scopedSlots.default ? [this.$scopedSlots.default(this.transform({
  169. days: this.days,
  170. hours: this.hours,
  171. minutes: this.minutes,
  172. seconds: this.seconds,
  173. milliseconds: this.milliseconds,
  174. totalDays: this.totalDays,
  175. totalHours: this.totalHours,
  176. totalMinutes: this.totalMinutes,
  177. totalSeconds: this.totalSeconds,
  178. totalMilliseconds: this.totalMilliseconds
  179. }))] : this.$slots.default);
  180. },
  181. watch: {
  182. $props: {
  183. deep: true,
  184. immediate: true,
  185. /**
  186. * Update the countdown when props changed.
  187. */
  188. handler: function handler() {
  189. this.totalMilliseconds = this.time;
  190. this.endTime = this.now() + this.time;
  191. if (this.autoStart) {
  192. this.start();
  193. }
  194. }
  195. }
  196. },
  197. methods: {
  198. /**
  199. * Starts to countdown.
  200. * @public
  201. * @emits Countdown#start
  202. */
  203. start: function start() {
  204. if (this.counting) {
  205. return;
  206. }
  207. this.counting = true;
  208. if (this.emitEvents) {
  209. /**
  210. * Countdown start event.
  211. * @event Countdown#start
  212. */
  213. this.$emit('start');
  214. }
  215. if (document.visibilityState === 'visible') {
  216. this.continue();
  217. }
  218. },
  219. /**
  220. * Continues the countdown.
  221. * @private
  222. */
  223. continue: function _continue() {
  224. var _this = this;
  225. if (!this.counting) {
  226. return;
  227. }
  228. var delay = Math.min(this.totalMilliseconds, this.interval);
  229. if (delay > 0) {
  230. if (window.requestAnimationFrame) {
  231. var init;
  232. var prev;
  233. var step = function step(now) {
  234. if (!init) {
  235. init = now;
  236. }
  237. if (!prev) {
  238. prev = now;
  239. }
  240. var range = now - init;
  241. if (range >= delay // Avoid losing time about one second per minute (now - prev ≈ 16ms) (#43)
  242. || range + (now - prev) / 2 >= delay) {
  243. _this.progress();
  244. } else {
  245. _this.requestId = requestAnimationFrame(step);
  246. }
  247. prev = now;
  248. };
  249. this.requestId = requestAnimationFrame(step);
  250. } else {
  251. this.timeoutId = setTimeout(function () {
  252. _this.progress();
  253. }, delay);
  254. }
  255. } else {
  256. this.end();
  257. }
  258. },
  259. /**
  260. * Pauses the countdown.
  261. * @private
  262. */
  263. pause: function pause() {
  264. if (window.requestAnimationFrame) {
  265. cancelAnimationFrame(this.requestId);
  266. } else {
  267. clearTimeout(this.timeoutId);
  268. }
  269. },
  270. /**
  271. * Progresses to countdown.
  272. * @private
  273. * @emits Countdown#progress
  274. */
  275. progress: function progress() {
  276. if (!this.counting) {
  277. return;
  278. }
  279. this.totalMilliseconds -= this.interval;
  280. if (this.emitEvents && this.totalMilliseconds > 0) {
  281. /**
  282. * Countdown progress event.
  283. * @event Countdown#progress
  284. */
  285. this.$emit('progress', {
  286. days: this.days,
  287. hours: this.hours,
  288. minutes: this.minutes,
  289. seconds: this.seconds,
  290. milliseconds: this.milliseconds,
  291. totalDays: this.totalDays,
  292. totalHours: this.totalHours,
  293. totalMinutes: this.totalMinutes,
  294. totalSeconds: this.totalSeconds,
  295. totalMilliseconds: this.totalMilliseconds
  296. });
  297. }
  298. this.continue();
  299. },
  300. /**
  301. * Aborts the countdown.
  302. * @public
  303. * @emits Countdown#abort
  304. */
  305. abort: function abort() {
  306. if (!this.counting) {
  307. return;
  308. }
  309. this.pause();
  310. this.counting = false;
  311. if (this.emitEvents) {
  312. /**
  313. * Countdown abort event.
  314. * @event Countdown#abort
  315. */
  316. this.$emit('abort');
  317. }
  318. },
  319. /**
  320. * Ends the countdown.
  321. * @public
  322. * @emits Countdown#end
  323. */
  324. end: function end() {
  325. if (!this.counting) {
  326. return;
  327. }
  328. this.pause();
  329. this.totalMilliseconds = 0;
  330. this.counting = false;
  331. if (this.emitEvents) {
  332. /**
  333. * Countdown end event.
  334. * @event Countdown#end
  335. */
  336. this.$emit('end');
  337. }
  338. },
  339. /**
  340. * Updates the count.
  341. * @private
  342. */
  343. update: function update() {
  344. if (this.counting) {
  345. this.totalMilliseconds = Math.max(0, this.endTime - this.now());
  346. }
  347. },
  348. /**
  349. * visibility change event handler.
  350. * @private
  351. */
  352. handleVisibilityChange: function handleVisibilityChange() {
  353. switch (document.visibilityState) {
  354. case 'visible':
  355. this.update();
  356. this.continue();
  357. break;
  358. case 'hidden':
  359. this.pause();
  360. break;
  361. }
  362. }
  363. },
  364. mounted: function mounted() {
  365. document.addEventListener(EVENT_VISIBILITY_CHANGE, this.handleVisibilityChange);
  366. },
  367. beforeDestroy: function beforeDestroy() {
  368. document.removeEventListener(EVENT_VISIBILITY_CHANGE, this.handleVisibilityChange);
  369. this.pause();
  370. }
  371. };
  372. return index;
  373. })));