mergebounce.js 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221
  1. import isObject from 'lodash-es/isObject.js';
  2. import merge from 'lodash-es/merge.js';
  3. import mergeWith from 'lodash-es/mergeWith.js';
  4. import now from 'lodash-es/now.js';
  5. import toNumber from 'lodash-es/toNumber.js';
  6. import { getOpenPromise } from '@converse/openpromise/openpromise.js';
  7. /** Error message constants. */
  8. const FUNC_ERROR_TEXT = 'Expected a function';
  9. /* Built-in method references for those with the same name as other `lodash` methods. */
  10. const nativeMax = Math.max;
  11. const nativeMin = Math.min;
  12. /**
  13. * Creates a debounced function that delays invoking `func` until after `wait`
  14. * milliseconds have elapsed since the last time the debounced function was
  15. * invoked. The debounced function comes with a `cancel` method to cancel
  16. * delayed `func` invocations and a `flush` method to immediately invoke them.
  17. *
  18. * This function differs from lodash's debounce by merging all passed objects
  19. * before passing them to the final invoked function.
  20. *
  21. * Because of this, invoking can only happen on the trailing edge, since
  22. * passed-in data would be discarded if invoking happened on the leading edge.
  23. *
  24. * If `wait` is `0`, `func` invocation is deferred until to the next tick,
  25. * similar to `setTimeout` with a timeout of `0`.
  26. *
  27. * @static
  28. * @category Function
  29. * @param {Function} func The function to mergebounce.
  30. * @param {number} [wait=0] The number of milliseconds to delay.
  31. * @param {Object} [options={}] The options object.
  32. * @param {number} [options.maxWait]
  33. * The maximum time `func` is allowed to be delayed before it's invoked.
  34. * @param {boolean} [options.concatArrays=false]
  35. * By default arrays will be treated as objects when being merged. When
  36. * merging two arrays, the values in the 2nd arrray will replace the
  37. * corresponding values (i.e. those with the same indexes) in the first array.
  38. * When `concatArrays` is set to `true`, arrays will be concatenated instead.
  39. * @param {boolean} [options.dedupeArrays=false]
  40. * This option is similar to `concatArrays`, except that the concatenated
  41. * array will also be deduplicated. Thus any entries that are concatenated to the
  42. * existing array, which are already contained in the existing array, will
  43. * first be removed.
  44. * @param {boolean} [options.promise=false]
  45. * By default, when calling a merge-debounced function that doesn't execute
  46. * immediately, you'll receive the result from its previous execution, or
  47. * `undefined` if it has never executed before. By setting the `promise`
  48. * option to `true`, a promise will be returned instead of the previous
  49. * execution result when the function is debounced. The promise will resolve
  50. * with the result of the next execution, as soon as it happens.
  51. * @returns {Function} Returns the new debounced function.
  52. * @example
  53. *
  54. * // Avoid costly calculations while the window size is in flux.
  55. * window.addEventListener('resize', mergebounce(calculateLayout, 150));
  56. *
  57. * // Invoke `sendMail` when clicked, debouncing subsequent calls.
  58. * element.addEventListner('click', mergebounce(sendMail, 300));
  59. *
  60. * // Ensure `batchLog` is invoked once after 1 second of debounced calls.
  61. * const mergebounced = mergebounce(batchLog, 250, { 'maxWait': 1000 });
  62. * const source = new EventSource('/stream');
  63. * jQuery(source).on('message', mergebounced);
  64. *
  65. * // Cancel the trailing debounced invocation.
  66. * window.addEventListener('popstate', mergebounced.cancel);
  67. */
  68. function mergebounce(func, wait, options={}) {
  69. let lastArgs,
  70. lastThis,
  71. maxWait,
  72. result,
  73. timerId,
  74. lastCallTime,
  75. lastInvokeTime = 0,
  76. maxing = false;
  77. let promise = options.promise ? getOpenPromise() : null;
  78. if (typeof func != 'function') {
  79. throw new TypeError(FUNC_ERROR_TEXT);
  80. }
  81. wait = toNumber(wait) || 0;
  82. if (isObject(options)) {
  83. maxing = 'maxWait' in options;
  84. maxWait = maxing ? nativeMax(toNumber(options.maxWait) || 0, wait) : maxWait;
  85. }
  86. function invokeFunc(time) {
  87. const args = lastArgs;
  88. const thisArg = lastThis;
  89. const existingPromise = promise;
  90. lastArgs = lastThis = undefined;
  91. lastInvokeTime = time;
  92. result = func.apply(thisArg, args);
  93. if (options.promise) {
  94. existingPromise.resolve(result);
  95. promise = getOpenPromise();
  96. }
  97. return options.promise ? existingPromise : result;
  98. }
  99. function leadingEdge(time) {
  100. // Reset any `maxWait` timer.
  101. lastInvokeTime = time;
  102. // Start the timer for the trailing edge.
  103. timerId = setTimeout(timerExpired, wait);
  104. return options.promise ? promise : result;
  105. }
  106. function remainingWait(time) {
  107. const timeSinceLastCall = time - lastCallTime;
  108. const timeSinceLastInvoke = time - lastInvokeTime;
  109. const timeWaiting = wait - timeSinceLastCall;
  110. return maxing
  111. ? nativeMin(timeWaiting, maxWait - timeSinceLastInvoke)
  112. : timeWaiting;
  113. }
  114. function shouldInvoke(time) {
  115. const timeSinceLastCall = time - lastCallTime;
  116. const timeSinceLastInvoke = time - lastInvokeTime;
  117. // Either this is the first call, activity has stopped and we're at the
  118. // trailing edge, the system time has gone backwards and we're treating
  119. // it as the trailing edge, or we've hit the `maxWait` limit.
  120. return (lastCallTime === undefined || (timeSinceLastCall >= wait) ||
  121. (timeSinceLastCall < 0) || (maxing && timeSinceLastInvoke >= maxWait));
  122. }
  123. function timerExpired() {
  124. const time = now();
  125. if (shouldInvoke(time)) {
  126. return trailingEdge(time);
  127. }
  128. // Restart the timer.
  129. timerId = setTimeout(timerExpired, remainingWait(time));
  130. }
  131. function trailingEdge(time) {
  132. timerId = undefined;
  133. // Only invoke if we have `lastArgs` which means `func` has been
  134. // debounced at least once.
  135. if (lastArgs) {
  136. return invokeFunc(time);
  137. }
  138. lastArgs = lastThis = undefined;
  139. return options.promise ? promise : result;
  140. }
  141. function cancel() {
  142. if (timerId !== undefined) {
  143. clearTimeout(timerId);
  144. }
  145. lastInvokeTime = 0;
  146. lastArgs = lastCallTime = lastThis = timerId = undefined;
  147. }
  148. function flush() {
  149. return timerId === undefined ? result : trailingEdge(now());
  150. }
  151. function concatArrays(objValue, srcValue) {
  152. if (Array.isArray(objValue) && Array.isArray(srcValue)) {
  153. if (options?.dedupeArrays) {
  154. return objValue.concat(srcValue.filter(i => objValue.indexOf(i) === -1));
  155. } else {
  156. return objValue.concat(srcValue);
  157. }
  158. }
  159. }
  160. function mergeArguments(args) {
  161. if (lastArgs?.length) {
  162. if (!args.length) {
  163. return lastArgs;
  164. }
  165. if (options?.concatArrays || options?.dedupeArrays) {
  166. return mergeWith(lastArgs, args, concatArrays);
  167. } else {
  168. return merge(lastArgs, args);
  169. }
  170. } else {
  171. return args || [];
  172. }
  173. }
  174. function debounced() {
  175. const time = now();
  176. const isInvoking = shouldInvoke(time);
  177. lastArgs = mergeArguments(Array.from(arguments));
  178. lastThis = this;
  179. lastCallTime = time;
  180. if (isInvoking) {
  181. if (timerId === undefined) {
  182. return leadingEdge(lastCallTime);
  183. }
  184. if (maxing) {
  185. // Handle invocations in a tight loop.
  186. clearTimeout(timerId);
  187. timerId = setTimeout(timerExpired, wait);
  188. return invokeFunc(lastCallTime);
  189. }
  190. }
  191. if (timerId === undefined) {
  192. timerId = setTimeout(timerExpired, wait);
  193. }
  194. return options.promise ? promise : result;
  195. }
  196. debounced.cancel = cancel;
  197. debounced.flush = flush;
  198. return debounced;
  199. }
  200. export default mergebounce;