@react-hookz/web
Version:
React hooks done right, for browser and SSR.
90 lines (89 loc) • 3.48 kB
JavaScript
;
var __assign = (this && this.__assign) || function () {
__assign = Object.assign || function(t) {
for (var s, i = 1, n = arguments.length; i < n; i++) {
s = arguments[i];
for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p))
t[p] = s[p];
}
return t;
};
return __assign.apply(this, arguments);
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.useAsync = void 0;
var react_1 = require("react");
var __1 = require("..");
/**
* Tracks result and error of provided async function and provides handles to execute and reset it.
*
* @param asyncFn Function that returns a promise.
* @param initialValue Value that will be set on initialisation, before the async function is
* executed.
*/
function useAsync(asyncFn, initialValue) {
var _a = (0, __1.useSafeState)({
status: 'not-executed',
error: undefined,
result: initialValue,
}), state = _a[0], setState = _a[1];
var promiseRef = (0, react_1.useRef)();
var argsRef = (0, react_1.useRef)();
var methods = (0, __1.useSyncedRef)({
execute: function () {
var params = [];
for (var _i = 0; _i < arguments.length; _i++) {
params[_i] = arguments[_i];
}
argsRef.current = params;
var promise = asyncFn.apply(void 0, params);
promiseRef.current = promise;
setState(function (s) { return (__assign(__assign({}, s), { status: 'loading' })); });
// eslint-disable-next-line promise/catch-or-return
promise.then(function (result) {
// we dont want to handle result/error of non-latest function
// this approach helps to avoid race conditions
// eslint-disable-next-line promise/always-return
if (promise === promiseRef.current) {
setState(function (s) { return (__assign(__assign({}, s), { status: 'success', error: undefined, result: result })); });
}
}, function (error) {
// we dont want to handle result/error of non-latest function
// this approach helps to avoid race conditions
if (promise === promiseRef.current) {
setState(function (s) { return (__assign(__assign({}, s), { status: 'error', error: error })); });
}
});
return promise;
},
reset: function () {
setState({
status: 'not-executed',
error: undefined,
result: initialValue,
});
promiseRef.current = undefined;
argsRef.current = undefined;
},
});
return [
state,
(0, react_1.useMemo)(function () { return ({
reset: function () {
methods.current.reset();
},
execute: function () {
var _a;
var params = [];
for (var _i = 0; _i < arguments.length; _i++) {
params[_i] = arguments[_i];
}
return (_a = methods.current).execute.apply(_a, params);
},
}); },
// eslint-disable-next-line react-hooks/exhaustive-deps
[]),
{ promise: promiseRef.current, lastArgs: argsRef.current },
];
}
exports.useAsync = useAsync;