Skip to content

Commit

Permalink
✨ - feat: autofocus useprompt
Browse files Browse the repository at this point in the history
  • Loading branch information
Xaohs committed Oct 1, 2024
1 parent c24e39d commit c601399
Showing 1 changed file with 69 additions and 36 deletions.
105 changes: 69 additions & 36 deletions src/hooks/dialog/useprompt.tsx
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import React, { useContext } from "react";
import React, { useContext, useEffect } from "react";

import { Form, ModalProps, P } from "../../components";
import { ModalServiceContext } from "../../contexts";
Expand All @@ -12,18 +12,6 @@ export const usePrompt = () => {
const dialog = useDialog();
const { setModalProps } = useContext(ModalServiceContext);

/**
* Shows a prompt dialog with a confirmation callback and an optional
* cancellation callback.
* @param title
* @param message
* @param label
* @param labelConfirm
* @param labelCancel
* @param onConfirm
* @param onCancel
* @param modalProps
*/
const fn = (
title: string,
message: React.ReactNode,
Expand All @@ -36,33 +24,78 @@ export const usePrompt = () => {
) => {
dialog(
title,
<>
{typeof message === "string" ? <P>{message}</P> : message}
<Form
fields={[{ label, name: "message", required: true }]}
labelSubmit={labelConfirm}
secondaryActions={[
{
children: labelCancel,
variant: "secondary",
type: "button",
onClick: () => {
setModalProps({ open: false });
onCancel?.();
},
},
]}
validateOnChange={true}
onSubmit={(_, { message }) => {
setModalProps({ open: false });
onConfirm(message as string);
}}
/>
</>,
<PromptForm
message={message}
label={label}
labelConfirm={labelConfirm}
labelCancel={labelCancel}
onConfirm={onConfirm}
onCancel={onCancel}
setModalProps={setModalProps}
/>,
undefined,
{ allowClose: false, ...modalProps },
);
};

return fn;
};

const PromptForm = ({
message,
label,
labelConfirm,
labelCancel,
onConfirm,
onCancel,
setModalProps,
}: {
message: React.ReactNode;
label: string;
labelConfirm: string;
labelCancel: string;
onConfirm: (message: string) => void;
onCancel?: () => void;
setModalProps: (props: Partial<ModalProps>) => void;
}) => {
useEffect(() => {
// 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 (
<>
{typeof message === "string" ? <P>{message}</P> : message}
<Form
fields={[{ label, name: "message", required: true }]} // No need to pass ref
labelSubmit={labelConfirm}
secondaryActions={[
{
children: labelCancel,
variant: "secondary",
type: "button",
onClick: () => {
setModalProps({ open: false });
onCancel?.();
},
},
]}
validateOnChange={true}
onSubmit={(_, { message }) => {
setModalProps({ open: false });
onConfirm(message as string);
}}
/>
</>
);
};

0 comments on commit c601399

Please sign in to comment.