forked from github/plane
137 lines
4.3 KiB
TypeScript
137 lines
4.3 KiB
TypeScript
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 (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" && !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);
|
|
else setIssueDiscardModal(true);
|
|
}
|
|
};
|
|
|
|
const handleCreateDraftIssue = async () => {
|
|
if (!changesMade || !workspaceSlug || !projectId) return;
|
|
|
|
const payload = {
|
|
...changesMade,
|
|
name: changesMade.name?.trim() === "" ? "Untitled" : changesMade.name?.trim(),
|
|
};
|
|
|
|
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}
|
|
/>
|
|
</>
|
|
);
|
|
});
|