-
-
Notifications
You must be signed in to change notification settings - Fork 67
/
Copy patheventloop.js
128 lines (111 loc) · 3.08 KB
/
eventloop.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
/* global requestIdleCallback, requestAnimationFrame, cancelIdleCallback, cancelAnimationFrame */
import * as time from './time.js'
/**
* Utility module to work with EcmaScript's event loop.
*
* @module eventloop
*/
/**
* @type {Array<function>}
*/
let queue = []
const _runQueue = () => {
for (let i = 0; i < queue.length; i++) {
queue[i]()
}
queue = []
}
/**
* @param {function():void} f
*/
export const enqueue = f => {
queue.push(f)
if (queue.length === 1) {
setTimeout(_runQueue, 0)
}
}
/**
* @typedef {Object} TimeoutObject
* @property {function} TimeoutObject.destroy
*/
/**
* @param {function(number):void} clearFunction
*/
const createTimeoutClass = clearFunction => class TT {
/**
* @param {number} timeoutId
*/
constructor (timeoutId) {
this._ = timeoutId
}
destroy () {
clearFunction(this._)
}
}
const Timeout = createTimeoutClass(clearTimeout)
/**
* @param {number} timeout
* @param {function} callback
* @return {TimeoutObject}
*/
export const timeout = (timeout, callback) => new Timeout(setTimeout(callback, timeout))
const Interval = createTimeoutClass(clearInterval)
/**
* @param {number} timeout
* @param {function} callback
* @return {TimeoutObject}
*/
export const interval = (timeout, callback) => new Interval(setInterval(callback, timeout))
/* c8 ignore next */
export const Animation = createTimeoutClass(arg => typeof requestAnimationFrame !== 'undefined' && cancelAnimationFrame(arg))
/**
* @param {function(number):void} cb
* @return {TimeoutObject}
*/
/* c8 ignore next */
export const animationFrame = cb => typeof requestAnimationFrame === 'undefined' ? timeout(0, cb) : new Animation(requestAnimationFrame(cb))
/* c8 ignore next */
// @ts-ignore
const Idle = createTimeoutClass(arg => typeof cancelIdleCallback !== 'undefined' && cancelIdleCallback(arg))
/**
* Note: this is experimental and is probably only useful in browsers.
*
* @param {function} cb
* @return {TimeoutObject}
*/
/* c8 ignore next 2 */
// @ts-ignore
export const idleCallback = cb => typeof requestIdleCallback !== 'undefined' ? new Idle(requestIdleCallback(cb)) : timeout(1000, cb)
/**
* @param {number} timeout Timeout of the debounce action
* @param {number} triggerAfter Optional. Trigger callback after a certain amount of time
* without waiting for debounce.
*/
export const createDebouncer = (timeout, triggerAfter = -1) => {
let timer = -1
/**
* @type {number?}
*/
let lastCall = null
/**
* @param {((...args: any)=>void)?} cb function to trigger after debounce. If null, it will reset the
* debounce.
*/
return cb => {
clearTimeout(timer)
if (cb) {
if (triggerAfter >= 0) {
const now = time.getUnixTime()
if (lastCall === null) lastCall = now
if (now - lastCall > triggerAfter) {
lastCall = null
timer = /** @type {any} */ (setTimeout(cb, 0))
return
}
}
timer = /** @type {any} */ (setTimeout(() => { lastCall = null; cb() }, timeout))
} else {
lastCall = null
}
}
}