Skip to content

Commit

Permalink
[use-debounce] Add definition (#4531)
Browse files Browse the repository at this point in the history
* add initial definition

* add all valid use cases

* add some failures
  • Loading branch information
Brianzchen committed Oct 13, 2023
1 parent 5f92108 commit 0524ccf
Show file tree
Hide file tree
Showing 2 changed files with 238 additions and 0 deletions.
Original file line number Diff line number Diff line change
@@ -0,0 +1,61 @@
// @flow
import { describe, test } from 'flow-typed-test';
import * as React from 'react';
import { useDebounce, useDebouncedCallback, useThrottledCallback } from 'use-debounce';

describe('use-debounce', () => {
test('useDebounce', () => {
const [value, debouncedState] = useDebounce('test', 1000, {
maxWait: 1,
leading: true,
trailing: true,
equalityFn: (a, b) => {
return a.toLowerCase() === b.toLowerCase();
},
});
(value: string);

// $FlowExpectedError[incompatible-cast] infers as a string
(value: number);
});

test('useDebouncedCallback', () => {
const [value, setValue] = React.useState('test');
// Debounce callback
const debounced = useDebouncedCallback(
// function
(value: string) => {
setValue(value);
},
// delay in ms
1000
);

debounced.cancel();
debounced.flush();
debounced.isPending();

// $FlowExpectedError[prop-missing]
debounced.foo();
});

test('useThrottledCallback', () => {
const [value, setValue] = React.useState('test');
// Debounce callback
const debounced = useThrottledCallback(
// function
(value: string) => {
setValue(value);
},
// delay in ms
1000
);

debounced.cancel();
debounced.flush();
debounced.isPending();

// $FlowExpectedError[prop-missing]
debounced.foo();
});
});
Original file line number Diff line number Diff line change
@@ -0,0 +1,177 @@
declare module 'use-debounce' {
declare type CallOptions = {|
/**
* Controls if the function should be invoked on the leading edge of the timeout.
*/
leading?: boolean,
/**
* Controls if the function should be invoked on the trailing edge of the timeout.
*/
trailing?: boolean,
|};

declare type Options = {|
...CallOptions,
/**
* The maximum time the given function is allowed to be delayed before it's invoked.
*/
maxWait?: number,
|};

declare type ControlFunctions = {|
/**
* Cancel pending function invocations
*/
cancel: () => void,
/**
* Immediately invoke pending function invocations
*/
flush: () => void,
/**
* Returns `true` if there are any pending function invocations
*/
isPending: () => boolean,
|};

/**
* Subsequent calls to the debounced function `debounced.callback` return the result of the last func invocation.
* Note, that if there are no previous invocations it's mean you will get undefined. You should check it in your code properly.
*/
declare type DebouncedState<R, T: (...args: any) => R> = {|
...ControlFunctions,
(...args: Parameters<T>): R | void,
|};

declare module.exports: {|
useDebounce: <T>(value: T, delay: number, options?: {|
maxWait?: number,
leading?: boolean,
trailing?: boolean,
equalityFn?: (left: T, right: T) => boolean,
|}) => [T, DebouncedState<void, (value: T) => void>],
/**
* Creates a debounced function that delays invoking `func` until after `wait`
* milliseconds have elapsed since the last time the debounced function was
* invoked, or until the next browser frame is drawn.
*
* The debounced function comes with a `cancel` method to cancel delayed `func`
* invocations and a `flush` method to immediately invoke them.
*
* Provide `options` to indicate whether `func` should be invoked on the leading
* and/or trailing edge of the `wait` timeout. The `func` is invoked with the
* last arguments provided to the debounced function.
*
* Subsequent calls to the debounced function return the result of the last
* `func` invocation.
*
* **Note:** If `leading` and `trailing` options are `true`, `func` is
* invoked on the trailing edge of the timeout only if the debounced function
* is invoked more than once during the `wait` timeout.
*
* If `wait` is `0` and `leading` is `false`, `func` invocation is deferred
* until the next tick, similar to `setTimeout` with a timeout of `0`.
*
* If `wait` is omitted in an environment with `requestAnimationFrame`, `func`
* invocation will be deferred until the next frame is drawn (typically about
* 16ms).
*
* See [David Corbacho's article](https://css-tricks.com/debouncing-throttling-explained-examples/)
* for details over the differences between `debounce` and `throttle`.
*
* @category Function
* @param {Function} func The function to debounce.
* @param {number} [wait=0]
* The number of milliseconds to delay; if omitted, `requestAnimationFrame` is
* used (if available, otherwise it will be setTimeout(...,0)).
* @param {Object} [options={}] The options object.
* Controls if `func` should be invoked on the leading edge of the timeout.
* @param {boolean} [options.leading=false]
* The maximum time `func` is allowed to be delayed before it's invoked.
* @param {number} [options.maxWait]
* Controls if `func` should be invoked the trailing edge of the timeout.
* @param {boolean} [options.trailing=true]
* @returns {Function} Returns the new debounced function.
* @example
*
* // Avoid costly calculations while the window size is in flux.
* const resizeHandler = useDebouncedCallback(calculateLayout, 150);
* window.addEventListener('resize', resizeHandler)
*
* // Invoke `sendMail` when clicked, debouncing subsequent calls.
* const clickHandler = useDebouncedCallback(sendMail, 300, {
* leading: true,
* trailing: false,
* })
* <button onClick={clickHandler}>click me</button>
*
* // Ensure `batchLog` is invoked once after 1 second of debounced calls.
* const debounced = useDebouncedCallback(batchLog, 250, { 'maxWait': 1000 })
* const source = new EventSource('/stream')
* source.addEventListener('message', debounced)
*
* // Cancel the trailing debounced invocation.
* window.addEventListener('popstate', debounced.cancel)
*
* // Check for pending invocations.
* const status = debounced.pending() ? "Pending..." : "Ready"
*/
useDebouncedCallback: <R, T: (...args: any) => R>(
func: T, wait?: number, options?: Options,
) => DebouncedState<R, T>,
/**
* Creates a throttled function that only invokes `func` at most once per
* every `wait` milliseconds (or once per browser frame).
*
* The throttled function comes with a `cancel` method to cancel delayed `func`
* invocations and a `flush` method to immediately invoke them.
*
* Provide `options` to indicate whether `func` should be invoked on the leading
* and/or trailing edge of the `wait` timeout. The `func` is invoked with the
* last arguments provided to the throttled function.
*
* Subsequent calls to the throttled function return the result of the last
* `func` invocation.
*
* **Note:** If `leading` and `trailing` options are `true`, `func` is
* invoked on the trailing edge of the timeout only if the throttled function
* is invoked more than once during the `wait` timeout.
*
* If `wait` is `0` and `leading` is `false`, `func` invocation is deferred
* until the next tick, similar to `setTimeout` with a timeout of `0`.
*
* If `wait` is omitted in an environment with `requestAnimationFrame`, `func`
* invocation will be deferred until the next frame is drawn (typically about
* 16ms).
*
* See [David Corbacho's article](https://css-tricks.com/debouncing-throttling-explained-examples/)
* for details over the differences between `throttle` and `debounce`.
*
* @category Function
* @param {Function} func The function to throttle.
* @param {number} [wait=0]
* The number of milliseconds to throttle invocations to; if omitted,
* `requestAnimationFrame` is used (if available, otherwise it will be setTimeout(...,0)).
* @param {Object} [options={}] The options object.
* @param {boolean} [options.leading=true]
* Specify invoking on the leading edge of the timeout.
* @param {boolean} [options.trailing=true]
* Specify invoking on the trailing edge of the timeout.
* @returns {Function} Returns the new throttled function.
* @example
*
* // Avoid excessively updating the position while scrolling.
* const scrollHandler = useThrottledCallback(updatePosition, 100)
* window.addEventListener('scroll', scrollHandler)
*
* // Invoke `renewToken` when the click event is fired, but not more than once every 5 minutes.
* const { callback } = useThrottledCallback(renewToken, 300000, { 'trailing': false })
* <button onClick={callback}>click</button>
*
* // Cancel the trailing throttled invocation.
* window.addEventListener('popstate', throttled.cancel);
*/
useThrottledCallback: <R, T: (...args: any) => R>(
func: T, wait: number, options?: CallOptions,
) => DebouncedState<R, T>,
|};
}

0 comments on commit 0524ccf

Please sign in to comment.