import { FC, FormEvent, useCallback, useRef, useState } from "react"; import { observer } from "mobx-react"; import { usePathname, useRouter } from "next/navigation"; // editor import { EditorRefApi } from "@plane/rich-text-editor"; // types import { TIssue } from "@plane/types"; import { Button, ToggleSwitch, TOAST_TYPE, setToast } from "@plane/ui"; // components import { InboxIssueTitle, InboxIssueDescription, InboxIssueProperties, } from "@/components/inbox/modals/create-edit-modal"; // constants import { ISSUE_CREATED } from "@/constants/event-tracker"; // helpers import { renderFormattedPayloadDate } from "@/helpers/date-time.helper"; // hooks import { useEventTracker, useProjectInbox, useWorkspace } from "@/hooks/store"; import useKeypress from "@/hooks/use-keypress"; type TInboxIssueCreateRoot = { workspaceSlug: string; projectId: string; handleModalClose: () => void; }; export const defaultIssueData: Partial = { id: undefined, name: "", description_html: "", priority: "none", state_id: "", label_ids: [], assignee_ids: [], start_date: renderFormattedPayloadDate(new Date()), target_date: "", }; export const InboxIssueCreateRoot: FC = observer((props) => { const { workspaceSlug, projectId, handleModalClose } = props; const router = useRouter(); const pathname = usePathname(); // refs const descriptionEditorRef = useRef(null); const submitBtnRef = useRef(null); // hooks const { captureIssueEvent } = useEventTracker(); const { createInboxIssue } = useProjectInbox(); const { getWorkspaceBySlug } = useWorkspace(); const workspaceId = getWorkspaceBySlug(workspaceSlug)?.id; // states const [createMore, setCreateMore] = useState(false); const [formSubmitting, setFormSubmitting] = useState(false); const [formData, setFormData] = useState>(defaultIssueData); const handleFormData = useCallback( >(issueKey: T, issueValue: Partial[T]) => { setFormData({ ...formData, [issueKey]: issueValue, }); }, [formData] ); const handleEscKeyDown = (event: KeyboardEvent) => { if (descriptionEditorRef.current?.isEditorReadyToDiscard()) { handleModalClose(); } else { setToast({ type: TOAST_TYPE.ERROR, title: "Error!", message: "Editor is still processing changes. Please wait before proceeding.", }); event.preventDefault(); // Prevent default action if editor is not ready to discard } }; useKeypress("Escape", handleEscKeyDown); const handleFormSubmit = async (event: FormEvent) => { event.preventDefault(); if (!descriptionEditorRef.current?.isEditorReadyToDiscard()) { setToast({ type: TOAST_TYPE.ERROR, title: "Error!", message: "Editor is still processing changes. Please wait before proceeding.", }); return; } const payload: Partial = { name: formData.name || "", description_html: formData.description_html || "

", priority: formData.priority || "none", state_id: formData.state_id || "", label_ids: formData.label_ids || [], assignee_ids: formData.assignee_ids || [], target_date: formData.target_date || null, }; setFormSubmitting(true); await createInboxIssue(workspaceSlug, projectId, payload) .then((res) => { if (!createMore) { router.push(`/${workspaceSlug}/projects/${projectId}/inbox/?currentTab=open&inboxIssueId=${res?.issue?.id}`); handleModalClose(); } else { descriptionEditorRef?.current?.clearEditor(); setFormData(defaultIssueData); } captureIssueEvent({ eventName: ISSUE_CREATED, payload: { ...formData, state: "SUCCESS", element: "Inbox page", }, path: pathname, }); setToast({ type: TOAST_TYPE.SUCCESS, title: `Success!`, message: "Issue created successfully.", }); }) .catch((error) => { console.error(error); captureIssueEvent({ eventName: ISSUE_CREATED, payload: { ...formData, state: "FAILED", element: "Inbox page", }, path: pathname, }); setToast({ type: TOAST_TYPE.ERROR, title: `Error!`, message: "Some error occurred. Please try again.", }); }); setFormSubmitting(false); }; const isTitleLengthMoreThan255Character = formData?.name ? formData.name.length > 255 : false; if (!workspaceSlug || !projectId || !workspaceId) return <>; return (

Create Inbox Issue

submitBtnRef?.current?.click()} />
setCreateMore((prevData) => !prevData)} role="button" > {}} size="sm" /> Create more
); });