function ignoreMultiClick(func, manual = false) {
let lock = false
return function (...args) {
if (lock) return
lock = true
let done = () => (lock = false)
if (manual) return func.call(this, ...args, done)
let promise = func.call(this, ...args)
Promise.resolve(promise).finally(done)
return promise
}
}
将想监听点击回调函数func作为传递给ignoreMultiClick进行装饰,会返回一个新的函数,使用该函数作为点击的回调事件即可。 - j7 R( m _- ]$ t. q) w 这里同样用了一个标记lock来上锁,有两种方法解锁: : G$ _ ~ q- f手动解锁:可以给ignoreMultiClick传递一个参数manual,意思是主动调用解锁。若该参数为truthy,则点击事件触发时会给原始的点击回调func传递一个参数done,done是一个函数,调用它可以解锁。 0 n. f* R3 O: W1 m. ^+ c: I4 S自动解锁:可以使原监听函数func返回一个promise,在该promise决议后自动执行解锁操作。因为Promise管理回调函数非常方便,并且像axios这样非常常用的请求库返回值本身也是一个promise,所以默认情况使用这种方式。4 G3 ?: X7 B' ]: r5 @
当然返回promise并不是必须的,有时候我们在发请求前会进行一些验证,验证没通过则直接return,此时装饰器函数也能正常处理,因为使用Promise.resolve包裹了一下promise:Promise.resolve(promise).finally(done)。 ( U% Y; x" ~) c7 v2 a) _- Z三、使用实例 / y9 i/ J/ k, s2 e4 n, v 自动解锁使用例子:! Z% l h. k) E3 S' A$ u