Skip to content

Instantly share code, notes, and snippets.

@c-kick
Last active October 27, 2023 13:24
Show Gist options
  • Star 1 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save c-kick/8d9c5a533dff11408e400c5abd6924a4 to your computer and use it in GitHub Desktop.
Save c-kick/8d9c5a533dff11408e400c5abd6924a4 to your computer and use it in GitHub Desktop.
Rate-limiter/debouncer module for JavaScript. See https://code.hnldesign.nl/demo/hnl.debounce.html
/**
*
* debounceThis ES6 module v1.4 (10-2023)
* Debounces/rate-limits the provided function (callback)
*
* Provides a way to debounce or rate-limit a function, which can be useful in scenarios where events may be
* triggered frequently and rapidly, such as scrolling or resizing the window.
*
* Example usage:
*
* import {debounceThis} from "debouncer.mjs";
*
* window.addEventListener('scroll', debounceThis((e)=> {
* //function that will be debounced/rate-limited
* updateScrollPos('after-scroll');
* }, {
* //optional parameters. Defaults:
* threshold: 150, //The amount of time (in milliseconds) to wait before executing the callback function.
* execStart: false, //Whether to execute the callback function immediately on the first event.
* execWhile: false, //Whether to execute the callback function at each interval while the debouncing function is being called.
* execDone: true, //Whether to execute the callback function once the debouncing function stops being called and the threshold has passed.
* }))
*
* New (1.4): you can check the 'debounceType' (start, while or done) in the event supplied to the provided function
* useful if you need to evaluate which debounce-stage triggered the function
*
* See demo at https://code.hnldesign.nl/demo/hnl.debounce.html
*/
const _defaults = {
threshold: 100,
execStart: false,
execWhile: false,
execDone: true,
}
export function debounceThis(callback, opts) {
// get/set options
let options = {
timer: 0,
whileTimer: 0,
busy: false,
..._defaults,
...opts,
};
return function (...args) {
clearTimeout(options.timer);
if (!options.busy && options.execStart) {
args[0].debounceType = 'start';
callback.apply(this, args);
options.busy = true;
}
if (options.execWhile && !options.whileTimer) {
options.whileTimer = setTimeout(() => {
args[0].debounceType = 'while';
callback.apply(this, args);
options.whileTimer = false;
}, options.threshold);
}
options.timer = setTimeout(() => {
args[0].debounceType = 'done';
options.busy = false;
if (options.execDone) callback.apply(this, args);
clearInterval(options.whileTimer);
}, options.threshold);
}
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment