import React, { useState } from "react"; import isEmpty from "lodash/isEmpty"; import { observer } from "mobx-react-lite"; import { useRouter } from "next/router"; import type { TIssue } from "@plane/types"; // hooks import { TOAST_TYPE, setToast } from "@plane/ui"; import { ConfirmIssueDiscard } from "@/components/issues"; import { IssueFormRoot } from "@/components/issues/issue-modal/form"; import { isEmptyHtmlString } from "@/helpers/string.helper"; import { useEventTracker } from "@/hooks/store"; // services import { IssueDraftService } from "@/services/issue"; export interface DraftIssueProps { changesMade: Partial<TIssue> | null; data?: Partial<TIssue>; issueTitleRef: React.MutableRefObject<HTMLInputElement | null>; isCreateMoreToggleEnabled: boolean; onCreateMoreToggleChange: (value: boolean) => void; onChange: (formData: Partial<TIssue> | null) => void; onClose: (saveDraftIssueInLocalStorage?: boolean) => void; onSubmit: (formData: Partial<TIssue>) => Promise<void>; projectId: string; isDraft: boolean; } const issueDraftService = new IssueDraftService(); export const DraftIssueLayout: React.FC<DraftIssueProps> = observer((props) => { const { changesMade, data, issueTitleRef, onChange, onClose, onSubmit, projectId, isCreateMoreToggleEnabled, onCreateMoreToggleChange, isDraft, } = props; // states const [issueDiscardModal, setIssueDiscardModal] = useState(false); // router const router = useRouter(); const { workspaceSlug } = router.query; // store hooks const { captureIssueEvent } = useEventTracker(); const handleClose = () => { if (data?.id) { onClose(false); setIssueDiscardModal(false); } else { if (changesMade) { Object.entries(changesMade).forEach(([key, value]) => { const issueKey = key as keyof TIssue; if (value === null || value === undefined || value === "") delete changesMade[issueKey]; if (typeof value === "object" && isEmpty(value)) delete changesMade[issueKey]; if (Array.isArray(value) && value.length === 0) delete changesMade[issueKey]; if (issueKey === "project_id") delete changesMade.project_id; if (issueKey === "priority" && value && value === "none") delete changesMade.priority; if ( issueKey === "description_html" && changesMade.description_html && isEmptyHtmlString(changesMade.description_html) ) delete changesMade.description_html; }); if (isEmpty(changesMade)) { onClose(false); setIssueDiscardModal(false); } else setIssueDiscardModal(true); } else { onClose(false); setIssueDiscardModal(false); } } }; const handleCreateDraftIssue = async () => { if (!changesMade || !workspaceSlug || !projectId) return; const payload = { ...changesMade, name: changesMade?.name && changesMade?.name?.trim() !== "" ? changesMade.name?.trim() : "Untitled", }; await issueDraftService .createDraftIssue(workspaceSlug.toString(), projectId.toString(), payload) .then((res) => { setToast({ type: TOAST_TYPE.SUCCESS, title: "Success!", message: "Draft Issue created successfully.", }); captureIssueEvent({ eventName: "Draft issue created", payload: { ...res, state: "SUCCESS" }, path: router.asPath, }); onChange(null); setIssueDiscardModal(false); onClose(false); }) .catch(() => { setToast({ type: TOAST_TYPE.ERROR, title: "Error!", message: "Issue could not be created. Please try again.", }); captureIssueEvent({ eventName: "Draft issue created", payload: { ...payload, state: "FAILED" }, path: router.asPath, }); }); }; return ( <> <ConfirmIssueDiscard isOpen={issueDiscardModal} handleClose={() => setIssueDiscardModal(false)} onConfirm={handleCreateDraftIssue} onDiscard={() => { onChange(null); setIssueDiscardModal(false); onClose(false); }} /> <IssueFormRoot isCreateMoreToggleEnabled={isCreateMoreToggleEnabled} onCreateMoreToggleChange={onCreateMoreToggleChange} data={data} issueTitleRef={issueTitleRef} onChange={onChange} onClose={handleClose} onSubmit={onSubmit} projectId={projectId} isDraft={isDraft} /> </> ); });