-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #140 from maykinmedia/issue/dialog-autofocus-alter…
…native ✨ - feat: implemented useFormDialog hook
- Loading branch information
Showing
3 changed files
with
153 additions
and
31 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 |
---|---|---|
|
@@ -98,6 +98,8 @@ export const Prompt: Story = { | |
"Annuleren", | ||
args.onConfirm, | ||
args.onCancel, | ||
undefined, | ||
true, | ||
) | ||
} | ||
> | ||
|
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,123 @@ | ||
import React, { useContext, useEffect } from "react"; | ||
|
||
import { Form, FormProps, ModalProps, P } from "../../components"; | ||
import { ModalServiceContext } from "../../contexts"; | ||
import { AttributeData, FormField } from "../../lib"; | ||
import { useDialog } from "./usedialog"; | ||
|
||
/** | ||
* Returns a function which, when called: shows a form dialog with a | ||
* confirmation callback and an optional cancellation callback. | ||
*/ | ||
export const useFormDialog = () => { | ||
const dialog = useDialog(); | ||
|
||
/** | ||
* Shows a prompt dialog with a confirmation callback and an optional | ||
* cancellation callback. | ||
* @param title | ||
* @param message | ||
* @param fields | ||
* @param labelConfirm | ||
* @param labelCancel | ||
* @param onConfirm | ||
* @param onCancel | ||
* @param modalProps | ||
* @param formProps | ||
* @param autofocus | ||
*/ | ||
const fn = ( | ||
title: string, | ||
message: React.ReactNode, | ||
fields: FormField[], | ||
labelConfirm: string, | ||
labelCancel: string, | ||
onConfirm: (data: AttributeData) => void, | ||
onCancel?: () => void, | ||
modalProps?: Partial<ModalProps>, | ||
formProps?: FormProps, | ||
autofocus?: boolean, | ||
) => { | ||
dialog( | ||
title, | ||
<> | ||
{typeof message === "string" ? <P>{message}</P> : message} | ||
<PromptForm | ||
message={message} | ||
fields={fields} | ||
labelConfirm={labelConfirm} | ||
labelCancel={labelCancel} | ||
onConfirm={onConfirm} | ||
onCancel={onCancel} | ||
formProps={formProps} | ||
autofocus={autofocus} | ||
/> | ||
</>, | ||
undefined, | ||
{ allowClose: false, ...modalProps }, | ||
); | ||
}; | ||
|
||
return fn; | ||
}; | ||
|
||
const PromptForm = ({ | ||
fields, | ||
labelConfirm, | ||
labelCancel, | ||
onConfirm, | ||
onCancel, | ||
formProps, | ||
autofocus = false, | ||
}: { | ||
message: React.ReactNode; | ||
fields: FormField[]; | ||
labelConfirm: string; | ||
labelCancel: string; | ||
onConfirm: (data: AttributeData) => void; | ||
onCancel?: () => void; | ||
formProps?: FormProps; | ||
autofocus?: boolean; | ||
}) => { | ||
const { setModalProps } = useContext(ModalServiceContext); | ||
|
||
useEffect(() => { | ||
if (!autofocus) return; | ||
// Delay the focus slightly to ensure modal and form are fully rendered | ||
const timer = setTimeout(() => { | ||
// We focus a form element, and if none are found, we focus the submit button, and otherwise none | ||
const formElement: HTMLFormElement | null = document.querySelector( | ||
"form input , form textarea , form select , form button[type=submit]", | ||
); | ||
if (formElement) { | ||
formElement.focus(); | ||
} | ||
}, 100); | ||
|
||
return () => clearTimeout(timer); | ||
}, []); | ||
|
||
return ( | ||
<Form | ||
fields={fields} | ||
labelSubmit={labelConfirm} | ||
secondaryActions={[ | ||
{ | ||
children: labelCancel, | ||
type: "button", | ||
variant: "secondary", | ||
onClick: () => { | ||
setModalProps({ open: false }); | ||
onCancel?.(); | ||
}, | ||
}, | ||
]} | ||
validateOnChange={true} | ||
onSubmit={(_, data) => { | ||
setModalProps({ open: false }); | ||
onConfirm(data); | ||
}} | ||
{...formProps} | ||
/> | ||
); | ||
}; |
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