Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/daniel-frank-miller/react-call


https://github.com/daniel-frank-miller/react-call

react typescript

Last synced: about 2 months ago
JSON representation

Awesome Lists containing this project

README

        



⚛️ 📡


react-call


npm i react-call

Call your React components — [demo ↗](https://react-call.desko.dev/)

# Features

Bring your React component, `react-call` gives you the `call()` method.

- 🔰 Easy to use
- ⛑️ Fully Type-Safe
- ⚡️ Causes no renders at all
- 🛜 Setup once, call from anywhere
- ⛓️‍💥 Not limited by a context provider
- 🤯 Call from outside React
- 🌀 Flexible: it's your component
- 🚀 Supports React Native and SSR
- 📦 Extremely lightweight: ~500B
- 🕳️ Zero dependencies

# The pattern

Imperatively call your React component the way `window.confirm()` calls a confirmation dialog.

window.confirm
react-call

```tsx
const message = 'Sure?'
const yes = window.confirm(message)

if (yes) deleteItem()
```

```tsx
const props = { message: 'Sure?' }
const yes = await Confirm.call(props)

if (yes) deleteItem()
```

# Use cases

Present a piece of UI to the user, wait for it to be used and get the response data.

- ✅ Confirmations, dialogs
- ✅ Notifications, toasts
- ✅ Popup forms, modals
- ✅ Or anything! 🦄 [Build your thing](#-build-your-thing)

# Usage

```tsx
// ↙ response props ↘
const accepted = await Confirm.call({ message: 'Continue?' })
```

> [!NOTE]
> A confirmation dialog is used as an example, but any component can become callable. Plus you can create as many as you wish.

# Setup

## 1. ⚛️ Wrap your component

```tsx
import { createCallable } from 'react-call'

interface Props { message: string }
type Response = boolean

export const Confirm = createCallable(({ call, message }) => (


{message}


call.end(true)}>Yes
call.end(false)}>No

))
```

Apart from your props, a special `call` prop is received containing the end() method, which will finish the call returning the given response.

> [!TIP]
> Since it's just a component, state, hooks and any other React features are totally fine. You could have inputs, checkboxes, etc, bind them to a state and return such data via end() method.

## 2. 📡 Place the Root

Place `Root` once, which is what listens to every single call and renders it. Any component that is visible when making your calls will do.

```diff
+
// ^-- it will only render active calls
```

> [!IMPORTANT]
> If more than one call is active, they will render one after the other (newer below, which is one on top of the other if your CSS is position fixed/absolute). It works as a call stack.

> [!WARNING]
> Since it's the source of truth, there can only be one `Root` mounted per createCallable(). Avoid placing it in multiple locations of the React Tree loaded at once, an error will be thrown if so.

# Passing Root props

You can also read props from Root, which are separate from the call props. To do that, just add your RootProps type to createCallable and pass them to your Root.

Root props will be available to your component via `call.root` object.

```diff
+ type RootProps = { userName: string }

export const Confirm = createCallable<
Props,
Response,
+ RootProps
>(({ call, message }) => (
...
+ Hi {call.root.userName}!
...
))
```

```diff

```

You may want to use Root props if you need to:

- Share the same piece of data to every call
- Use something that is availble in Root's parent
- Update your active call components on data changes

# Exit animations

To animate the exit of your component when `call.end()` is run, just pass the duration of your animation in milliseconds to createCallable as a second argument:

```diff
+ const UNMOUNTING_DELAY = 500

export const Confirm = createCallable(
({ call }) => (


),
+ UNMOUNTING_DELAY
)
```

The `call.ended` boolean may be used to apply your animation CSS class.

# 🦄 Build your thing

Again, this is no way limited to confirmation dialogs. You can build anything!

For example, because of the nature of the call stack inside and its ability to display multiple calls at once, a particularly interesting use case is notifications, toasts or similar. You could end up with something like:

```tsx
const userAction = await Toast.call({
message: 'This is a toast',
duration: 5000,
type: 'success',
})
```

But it's just another idea. It all depends on what you're building. The only thing `react-call` does is let you call components imperatively ⚛️ 📡

# TypeScript types

You won't need them most likely, but if you want to split the component declaration and such, you may use the types under the `ReactCall` namespace:

```tsx
import type { ReactCall } from 'react-call'
```

Type | Description
--- | ---
ReactCall.Function | The call() method
ReactCall.Context | The call prop in UserComponent
ReactCall.Props | Your props + the call prop
ReactCall.UserComponent | What is passed to createCallable
ReactCall.Callable | What createCallable returns

# Errors

Error | Solution
--- | ---
No \ found! | You forgot to place the Root, check [Place the Root](#2--place-the-root) section. If it's already in place but not present by the time you call(), you may want to place it higher in your React tree. If you're getting this error on the server see [SSR section](#does-the-setup-work-with-ssr).
Multiple instances of \ found! | You placed more than one Root, check [Place the Root](#2--place-the-root) section as there is a warning about this.

# SSR

✅ The react-call setup supports [Server Side Rendering](https://nextjs.org/docs/pages/building-your-application/rendering/server-side-rendering). This means both createCallable and Root component are fine if run or rendered on the server.

However, bear in mind that because the call() method is meant to be triggered by user interaction, it is designed as a client-only feature.

> [!CAUTION]
> If call() is run on the server a "No \ found!" error will be thrown. As long as you don't run the call() method on the server you'll be fine.