-
Notifications
You must be signed in to change notification settings - Fork 98
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(usePromiseDialog): add usePromiseDialog hook and Confirm component
- Loading branch information
kseniyakuzina
committed
Feb 16, 2024
1 parent
84044f0
commit 780862a
Showing
18 changed files
with
571 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,11 @@ | ||
@use '../../../styles/mixins.scss'; | ||
@use '../../components/variables'; | ||
|
||
$block: '.#{variables.$ns}confirm'; | ||
|
||
#{$block} { | ||
&__message { | ||
margin: 0; | ||
@include mixins.text-body-2; | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,57 @@ | ||
import React from 'react'; | ||
|
||
import type {ButtonProps, DialogProps} from '..'; | ||
import {Dialog} from '..'; | ||
import {block} from '../utils/cn'; | ||
|
||
import './Confirm.scss'; | ||
|
||
const b = block('confirm'); | ||
|
||
export type ConfirmProps = { | ||
title?: string; | ||
message?: string; | ||
content?: React.ReactNode; | ||
confirmButtonText?: string; | ||
cancelButtonText?: string; | ||
onConfirm: () => void; | ||
onCancel: () => void; | ||
confirmButtonProps?: ButtonProps; | ||
cancelButtonProps?: ButtonProps; | ||
onClose?: DialogProps['onClose']; | ||
} & Omit<DialogProps, 'children' | 'onClose'>; | ||
|
||
export const Confirm = ({ | ||
title, | ||
message = '', | ||
content, | ||
confirmButtonText, | ||
cancelButtonText, | ||
onConfirm, | ||
onCancel, | ||
confirmButtonProps, | ||
cancelButtonProps, | ||
onClose, | ||
...dialogProps | ||
}: ConfirmProps) => { | ||
return ( | ||
<Dialog {...dialogProps} onClose={onClose ?? onCancel}> | ||
<Dialog.Header caption={title} /> | ||
<Dialog.Body> | ||
{content} | ||
{message && !content && <p className={b('message')}>{message}</p>} | ||
</Dialog.Body> | ||
<Dialog.Footer | ||
preset="default" | ||
showError={false} | ||
listenKeyEnter | ||
textButtonApply={confirmButtonText} | ||
textButtonCancel={cancelButtonText} | ||
onClickButtonApply={onConfirm} | ||
onClickButtonCancel={onCancel} | ||
propsButtonCancel={cancelButtonProps} | ||
propsButtonApply={confirmButtonProps} | ||
/> | ||
</Dialog> | ||
); | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,58 @@ | ||
<!--GITHUB_BLOCK--> | ||
|
||
# Confirm | ||
|
||
<!--/GITHUB_BLOCK--> | ||
|
||
```tsx | ||
import {Confirm} from '@gravity-ui/uikit'; | ||
``` | ||
|
||
`Confirm` is a utility component, which renders confirmatuion dialogs | ||
|
||
## Properties | ||
|
||
| Name | Description | Type | Required | | ||
| :----------------- | :------------------------------------------------------------- | :-----------: | :------: | | ||
| title | The confirm dialog title | `string` | Yes | | ||
| cancelButtonText | The cancel button text | `string` | Yes | | ||
| confirmButtonText | The ok button text | `string` | Yes | | ||
| message | The confirmation message (used if the content is not provided) | `string` | | | ||
| content | The confirmation custom content | `ReactNode` | | | ||
| confirmButtonProps | The ok button props | `ButtonProps` | | | ||
| cancelButtonProps | The cancel buttonProps | `ButtonProps` | | | ||
|
||
And other Dialog props | ||
|
||
## Returns | ||
|
||
Returns an object with `openConfirm` property. `openConfirm` is a function, which accepts the confirm dialog props object, which can override the default useConfirm arguments (use when you have two almost the same confirmation dialogs in one component) | ||
|
||
## Usage | ||
|
||
```tsx | ||
import {Confirm} from '@gravity-ui/uikit'; | ||
|
||
const [open, setOpen] = React.useState(false); | ||
|
||
return ( | ||
<React.Fragment> | ||
<Button view="normal" onClick={() => setOpen(true)}> | ||
Show confirm | ||
</Button> | ||
<Confirm | ||
{...args} | ||
title="Do you want to confirm?" | ||
onConfirm={() => { | ||
alert('Confirmed'); | ||
setOpen(false); | ||
}} | ||
onCancel={() => setOpen(false)} | ||
cancelButtonText="No" | ||
confirmButtonText="Yes" | ||
open={open} | ||
aria-labelledby="app-confirmation-dialog-title" | ||
/> | ||
</React.Fragment> | ||
); | ||
``` |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,38 @@ | ||
import React from 'react'; | ||
|
||
import type {Meta, StoryFn} from '@storybook/react'; | ||
|
||
import {Button} from '../../Button'; | ||
import {Confirm} from '../Confirm'; | ||
import type {ConfirmProps} from '../Confirm'; | ||
|
||
export default { | ||
title: 'Components/Feedback/Confirm', | ||
component: Confirm, | ||
} as Meta<ConfirmProps>; | ||
|
||
const DefaultTemplate: StoryFn<ConfirmProps> = (args) => { | ||
const [open, setOpen] = React.useState(false); | ||
|
||
return ( | ||
<React.Fragment> | ||
<Button view="normal" onClick={() => setOpen(true)}> | ||
Show confirm | ||
</Button> | ||
<Confirm | ||
{...args} | ||
title="Do you want to confirm?" | ||
onConfirm={() => { | ||
alert('Confirmed'); | ||
setOpen(false); | ||
}} | ||
onCancel={() => setOpen(false)} | ||
cancelButtonText="No" | ||
confirmButtonText="Yes" | ||
open={open} | ||
aria-labelledby="app-confirmation-dialog-title" | ||
/> | ||
</React.Fragment> | ||
); | ||
}; | ||
export const Default = DefaultTemplate.bind({}); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
export * from './Confirm'; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,7 @@ | ||
import React from 'react'; | ||
|
||
import type {PromiseDialogContextType} from './types'; | ||
|
||
export const PromiseDialogContext = React.createContext<PromiseDialogContextType>({ | ||
openDialog: () => Promise.resolve({success: false}), | ||
}); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,91 @@ | ||
import React from 'react'; | ||
|
||
import omit from 'lodash/omit'; | ||
|
||
import {PromiseDialogContext} from './PromiseDialogContext'; | ||
import type {DialogRendererProps, PromiseDialogResult} from './types'; | ||
|
||
type PromiseDialogProviderProps = { | ||
children: React.ReactNode | React.ReactNode[]; | ||
onError: (error: unknown) => void; | ||
}; | ||
|
||
export const PromiseDialogProvider = ({children, onError}: PromiseDialogProviderProps) => { | ||
const [dialogs, setDialogs] = React.useState<Record<number, React.ReactNode>>([]); | ||
const dialogsRef: React.MutableRefObject<Record<number, React.ReactNode>> = | ||
React.useRef(dialogs); | ||
|
||
React.useEffect(() => { | ||
dialogsRef.current = dialogs; | ||
}, [dialogs]); | ||
|
||
const contextValue = React.useMemo( | ||
() => ({ | ||
openDialog: <ResultType extends unknown>( | ||
renderDialog: ({ | ||
onSuccess, | ||
asyncOnSuccess, | ||
onCancel, | ||
// onClose, | ||
}: DialogRendererProps<ResultType>) => React.ReactNode, | ||
) => | ||
new Promise<{success: boolean; value?: ResultType}>((resolve) => { | ||
const currentKeys = Object.keys(dialogsRef.current); | ||
|
||
const key = parseInt(currentKeys[currentKeys.length - 1] || '0', 10) + 1; | ||
|
||
const handleClose = (result: PromiseDialogResult<ResultType>) => { | ||
setTimeout(() => { | ||
setDialogs(omit(dialogsRef.current, key)); | ||
}, 100); | ||
|
||
resolve(result); | ||
}; | ||
|
||
const handleSuccess = (value: ResultType) => { | ||
handleClose({success: true, value}); | ||
}; | ||
|
||
const handleSuccessPromise = (getValue: Promise<ResultType>) => { | ||
getValue | ||
.then((value) => { | ||
handleClose({success: true, value}); | ||
}) | ||
.catch(onError); | ||
}; | ||
|
||
const handleCancel = () => { | ||
handleClose({success: false}); | ||
}; | ||
|
||
// const handleComponentClose = ( | ||
// event: MouseEvent | KeyboardEvent, | ||
// reason: 'outsideClick' | 'escapeKeyDown' | 'closeButtonClick', | ||
// ) => { | ||
// onClose?.(event, reason); | ||
// handleCancel(); | ||
// }; | ||
|
||
const dialog = renderDialog({ | ||
onSuccess: handleSuccess, | ||
asyncOnSuccess: handleSuccessPromise, | ||
onCancel: handleCancel, | ||
// onClose: handleComponentClose, | ||
}); | ||
|
||
setDialogs({ | ||
...dialogsRef.current, | ||
[key]: dialog, | ||
}); | ||
}), | ||
}), | ||
[onError], | ||
); | ||
|
||
return ( | ||
<PromiseDialogContext.Provider value={contextValue}> | ||
{children} | ||
{Object.values(dialogs)} | ||
</PromiseDialogContext.Provider> | ||
); | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,113 @@ | ||
<!--GITHUB_BLOCK--> | ||
|
||
# usePromiseDialog | ||
|
||
<!--/GITHUB_BLOCK--> | ||
|
||
```tsx | ||
import {usePromiseDialog} from '@gravity-ui/uikit'; | ||
``` | ||
|
||
### Provider | ||
|
||
Before usage, you should wrap your components with `PromiseDialogProvider` | ||
|
||
```tsx | ||
import {PromiseDialogProvider} from '@gravity-ui/uikit'; | ||
|
||
<PromiseDialogProvider onError={console.error}>{children}</PromiseDialogProvider>; | ||
``` | ||
|
||
Properties: | ||
|
||
| Name | Description | Type | Default | | ||
| :------ | :------------------------------------------------- | :------------------------: | :-----: | | ||
| onError | The error handler (used when asyncOnSuccess fails) | `(error: unknown) => void` | | | ||
|
||
`usePromiseDialog` is a utility hook, which allows you to open a dialog without adding a Dialog component to your code and controlling it's state | ||
|
||
## Returns | ||
|
||
Returns an object with `openDialog` property. `openDialog` is a generic function, which accepts a dialog content renderer and the dialog props. The content renderer provides an object | ||
`{ onSuccess, asyncOnSuccess, onCancel }`. | ||
|
||
- You should call `onSuccess` with the result of calling the dialog (in this case the promise result will be `{ success: true, value }`); | ||
- `asyncOnSuccess` with the promise which resolves the result (the promise dialog result will be `{ success: true, value }`, but the dialog won't resolve and close in case of an error occurred in the promise, passed with the arguments) | ||
- `onCancel` if you want to cancel the action (in this case the promise result will be `{ success: false }`). | ||
|
||
```ts | ||
function openDialog<ResultRype extends unknown>( | ||
renderContent: ({ | ||
onSuccess, | ||
asyncOnSuccess, | ||
onCancel, | ||
}: { | ||
onSuccess: (value: ResultType) => void; | ||
asyncOnSuccess: (getValue: Promise<ResultRype>) => void; | ||
onCancel: () => void; | ||
}) => React.ReactNode | React.ReactNode[], | ||
dialogProps?: Partial<DialogProps>, | ||
): Promise<{ | ||
success: boolean; | ||
value?: ResultType; | ||
}>; | ||
``` | ||
|
||
## Examples | ||
|
||
#### Base | ||
|
||
```tsx | ||
import {Dialog, usePromiseDialog} from '@gravity-ui/uikit'; | ||
|
||
const {openDialog} = usePromiseDialog(); | ||
|
||
const handleOpenNoteEditor = useCallback(async () => { | ||
const result = await openDialog<string>(({onSuccess, onCancel}) => ( | ||
<Dialog open onClose={onCancel} size="s"> | ||
<NoteEditor onSave={onSuccess} onCancel={onCancel} /> | ||
</Dialog> | ||
)); | ||
|
||
if (result.success) { | ||
const note = result.value; | ||
alert(`Your note is ${note}`); | ||
} else { | ||
alert('You cancelled creating the note'); | ||
} | ||
}, [openDialog]); | ||
``` | ||
|
||
#### Don't close dialog on error | ||
|
||
```tsx | ||
import {Dialog, usePromiseDialog} from '@gravity-ui/uikit'; | ||
|
||
const {openDialog} = usePromiseDialog(); | ||
|
||
const handleOpenNoteEditor = useCallback(async () => { | ||
const handleSave = (note: string) => { | ||
return new Promise<string>((resolve, reject) => { | ||
if (note) { | ||
resolve(note); | ||
} else { | ||
alert('Enter the note'); | ||
reject(); | ||
} | ||
}); | ||
}; | ||
|
||
const result = await openDialog<string>( | ||
({asyncOnSuccess, onCancel}) => <Dialog open onClose={onCancel} size="s"> | ||
<NoteEditor onSave{(note: string) => asyncOnSuccess(handleSave(note))} onCancel={onCancel} /> | ||
</Dialog> | ||
); | ||
|
||
if (result.success) { | ||
const note = result.value; | ||
alert(`Your note is ${note}`); | ||
} else { | ||
alert('You cancelled creating the note'); | ||
} | ||
}, [openDialog]); | ||
``` |
Oops, something went wrong.