react-redux-dispatch-async
Version:
hooks & redux middleware to be able to wait async actions with fixed defined suffixes
185 lines (147 loc) • 4.67 kB
Markdown
<p>
<a href="https://www.npmjs.com/package/react-redux-dispatch-async">
<img alt="npm version" src="https://badge.fury.io/js/react-redux-dispatch-async.svg"/></a>
<a href="#hire-an-expert">
<img src="https://img.shields.io/badge/%F0%9F%92%AA-hire%20an%20expert-brightgreen"/>
</a>
<p>
👉 REDUX _middleware_ & **HOOK** 🎉 waiting async _actions_ with **SUFFIXES** 👈
```
+------------------+
| ACTION_REQUESTED |----+
+------------------+ | +------------------+
+----->| ACTION_SUCCEEDED |
| +------------------+
|
| +--------------------+
+----->| ACTION_FAILED |
| +--------------------+
|
| +--------------------+
+----->| ACTION_CANCELED |
+--------------------+
```
`yarn add react-redux-dispatch-async`
[> Dig into it](https://github.com/xcarpentier/react-redux-dispatch-async/blob/master/src/useDispatchAsync.ts#L65)
- ⏳ **`loading`**: action start but not yet completed
- 👏 **`success`**: action completed, you can get the result
- 😱 **`error`**: action failed and you can get the error
- 👎 **`timeout`**: action not completed for tool long (ie. options?.timeoutInMilliseconds)
- 👋 **`canceled`**: action canceled
- 😮 **`unknown`**: should never happen
## Examples
### Usage
```tsx
import React from 'react'
import { useDispatchAsync } from 'react-redux-dispatch-async'
export default function MyUserInterface({ id }: { id: string }) {
// 👉 pass action and arguments into the array
const { status, result, error } = useDispatchAsync(getUserActionRequest, [id])
switch (status) {
case 'loading':
return <AppLoader />
case 'error':
return <Text>{error.message}</Text>
case 'success':
return <User {...result} />
case 'timeout':
return <Text>{'timeout ¯\\_(ツ)_//¯'}</Text>
case 'canceled':
return <Text>{'canceled ¯\\_(ツ)_//¯'}</Text>
default:
return null
}
}
```
If you need more examples you can go to [github](https://github.com/xcarpentier/react-redux-dispatch-async-example) or to [codesandbox](https://codesandbox.io/s/react-redux-dispatch-async-rij31?file=/src/UserContainer.tsx).
```ts
import { createStore, applyMiddleware } from 'redux'
import { createDispatchAsyncMiddleware } from 'react-redux-dispatch-async'
import reducers from 'reducers'
const store = createStore(
reducers,
applyMiddleware(
createDispatchAsyncMiddleware({
request: 'REQUEST', // 👈 define your own async suffixes
success: 'SUCCESS',
failure: 'FAILURE',
cancel: 'CANCEL', // optional
}),
),
)
```
- `[...]_REQUESTED`
- `[...]_SUCCEEDED`
- `[...]_FAILED`
- `[...]_CANCELED`
```ts
dispatchAsyncMiddleware: (c?: {
request: string
success: string
failure: string
cancel?: string
}) => redux.Middleware
```
```ts
// main hook
interface Options {
timeoutInMilliseconds?: number
}
type useDispatchAsync = <R = any>(
actionFunction?: (...args: any[]) => Action<T>,
deps: any[] = [],
options: Options = { timeoutInMilliseconds: 15000 }, // wait 15s
) => UseDispatchAsync<R>
/// return type
interface ResultLoading {
status: 'loading'
}
interface ResultSuccess<R = unknown> {
status: 'success'
result: R
}
interface ResultError {
status: 'error'
error: Error
}
interface ResultCancel {
status: 'canceled'
}
interface ResultTimeout {
status: 'timeout'
}
interface ResultUnknown {
status: 'unknown'
}
export type UseDispatchAsync<R = unknown> =
| ResultLoading
| ResultSuccess<R>
| ResultError
| ResultTimeout
| ResultCancel
| ResultUnknown
// other types for oldest usage
interface DispatchAsyncResultSuccess<T = any> {
success: true
result: T
}
interface DispatchAsyncResultError {
success: false
error: Error
}
export type DispatchAsyncResult<T = any> =
| DispatchAsyncResultSuccess<T>
| DispatchAsyncResultError
```
Looking for a ReactNative freelance expert with more than 14 years experience? Contact me from my [website](https://xaviercarpentier.com)!