mirror of
https://github.com/makeplane/plane
synced 2024-06-14 14:31:34 +00:00
d470adf262
* added image-resizing support * link form removed * updated image upload logic and 35% default width on upload * removed shadow, added alert if not saved and ts errors * prevent enter key on Link selector to trigger modal submit * added workspace slug to all tiptap instances * added delete plugin with loading indicator * added better syncing of "Saved" state of editor and Image uploads * removed redundant description_html check
230 lines
6.0 KiB
TypeScript
230 lines
6.0 KiB
TypeScript
import React, { useEffect, useState, forwardRef, useRef } from "react";
|
|
|
|
import { useRouter } from "next/router";
|
|
|
|
// react-hook-form
|
|
import { useForm } from "react-hook-form";
|
|
// services
|
|
import aiService from "services/ai.service";
|
|
import trackEventServices from "services/track-event.service";
|
|
// hooks
|
|
import useToast from "hooks/use-toast";
|
|
import useUserAuth from "hooks/use-user-auth";
|
|
// ui
|
|
import { Input, PrimaryButton, SecondaryButton } from "components/ui";
|
|
|
|
import { IIssue, IPageBlock } from "types";
|
|
import Tiptap, { ITiptapRichTextEditor } from "components/tiptap";
|
|
type Props = {
|
|
isOpen: boolean;
|
|
handleClose: () => void;
|
|
inset?: string;
|
|
content: string;
|
|
htmlContent?: string;
|
|
onResponse: (response: string) => void;
|
|
projectId: string;
|
|
block?: IPageBlock;
|
|
issue?: IIssue;
|
|
};
|
|
|
|
type FormData = {
|
|
prompt: string;
|
|
task: string;
|
|
};
|
|
|
|
const TiptapEditor = React.forwardRef<ITiptapRichTextEditor, ITiptapRichTextEditor>(
|
|
(props, ref) => <Tiptap {...props} forwardedRef={ref} />
|
|
);
|
|
|
|
TiptapEditor.displayName = "TiptapEditor";
|
|
|
|
export const GptAssistantModal: React.FC<Props> = ({
|
|
isOpen,
|
|
handleClose,
|
|
inset = "top-0 left-0",
|
|
content,
|
|
htmlContent,
|
|
onResponse,
|
|
projectId,
|
|
block,
|
|
issue,
|
|
}) => {
|
|
const [response, setResponse] = useState("");
|
|
const [invalidResponse, setInvalidResponse] = useState(false);
|
|
|
|
const router = useRouter();
|
|
const { workspaceSlug } = router.query;
|
|
|
|
const { user } = useUserAuth();
|
|
|
|
const editorRef = useRef<any>(null);
|
|
|
|
const { setToastAlert } = useToast();
|
|
|
|
const {
|
|
handleSubmit,
|
|
register,
|
|
reset,
|
|
setFocus,
|
|
formState: { isSubmitting },
|
|
} = useForm({
|
|
defaultValues: {
|
|
prompt: content,
|
|
task: "",
|
|
},
|
|
});
|
|
|
|
const onClose = () => {
|
|
handleClose();
|
|
setResponse("");
|
|
setInvalidResponse(false);
|
|
reset();
|
|
};
|
|
|
|
const handleResponse = async (formData: FormData) => {
|
|
if (!workspaceSlug || !projectId) return;
|
|
|
|
if (formData.task === "") {
|
|
setToastAlert({
|
|
type: "error",
|
|
title: "Error!",
|
|
message: "Please enter some task to get AI assistance.",
|
|
});
|
|
return;
|
|
}
|
|
|
|
await aiService
|
|
.createGptTask(
|
|
workspaceSlug as string,
|
|
projectId as string,
|
|
{
|
|
prompt: content && content !== "" ? content : htmlContent ?? "",
|
|
task: formData.task,
|
|
},
|
|
user
|
|
)
|
|
.then((res) => {
|
|
setResponse(res.response_html);
|
|
setFocus("task");
|
|
|
|
if (res.response === "") setInvalidResponse(true);
|
|
else setInvalidResponse(false);
|
|
})
|
|
.catch((err) => {
|
|
const error = err?.data?.error;
|
|
|
|
if (err.status === 429)
|
|
setToastAlert({
|
|
type: "error",
|
|
title: "Error!",
|
|
message:
|
|
error ||
|
|
"You have reached the maximum number of requests of 50 requests per month per user.",
|
|
});
|
|
else
|
|
setToastAlert({
|
|
type: "error",
|
|
title: "Error!",
|
|
message: error || "Some error occurred. Please try again.",
|
|
});
|
|
});
|
|
};
|
|
|
|
useEffect(() => {
|
|
if (isOpen) setFocus("task");
|
|
}, [isOpen, setFocus]);
|
|
|
|
useEffect(() => {
|
|
editorRef.current?.setEditorValue(htmlContent ?? `<p>${content}</p>`);
|
|
}, [htmlContent, editorRef, content]);
|
|
|
|
return (
|
|
<div
|
|
className={`absolute ${inset} z-20 w-full space-y-4 rounded-[10px] border border-custom-border-200 bg-custom-background-100 p-4 shadow ${isOpen ? "block" : "hidden"
|
|
}`}
|
|
>
|
|
{((content && content !== "") || (htmlContent && htmlContent !== "<p></p>")) && (
|
|
<div className="text-sm">
|
|
Content:
|
|
<TiptapEditor
|
|
workspaceSlug={workspaceSlug as string}
|
|
value={htmlContent ?? `<p>${content}</p>`}
|
|
customClassName="-m-3"
|
|
noBorder
|
|
borderOnFocus={false}
|
|
editable={false}
|
|
ref={editorRef}
|
|
/>
|
|
</div>
|
|
)}
|
|
{response !== "" && (
|
|
<div className="page-block-section text-sm">
|
|
Response:
|
|
<Tiptap
|
|
workspaceSlug={workspaceSlug as string}
|
|
value={`<p>${response}</p>`}
|
|
customClassName="-mx-3 -my-3"
|
|
noBorder
|
|
borderOnFocus={false}
|
|
editable={false}
|
|
/>
|
|
</div>
|
|
)}
|
|
{invalidResponse && (
|
|
<div className="text-sm text-red-500">
|
|
No response could be generated. This may be due to insufficient content or task
|
|
information. Please try again.
|
|
</div>
|
|
)}
|
|
<Input
|
|
type="text"
|
|
name="task"
|
|
register={register}
|
|
placeholder={`${content && content !== ""
|
|
? "Tell AI what action to perform on this content..."
|
|
: "Ask AI anything..."
|
|
}`}
|
|
autoComplete="off"
|
|
/>
|
|
<div className={`flex gap-2 ${response === "" ? "justify-end" : "justify-between"}`}>
|
|
{response !== "" && (
|
|
<PrimaryButton
|
|
onClick={() => {
|
|
onResponse(response);
|
|
onClose();
|
|
if (block)
|
|
trackEventServices.trackUseGPTResponseEvent(
|
|
block,
|
|
"USE_GPT_RESPONSE_IN_PAGE_BLOCK",
|
|
user
|
|
);
|
|
else if (issue)
|
|
trackEventServices.trackUseGPTResponseEvent(
|
|
issue,
|
|
"USE_GPT_RESPONSE_IN_ISSUE",
|
|
user
|
|
);
|
|
}}
|
|
>
|
|
Use this response
|
|
</PrimaryButton>
|
|
)}
|
|
<div className="flex items-center gap-2">
|
|
<SecondaryButton onClick={onClose}>Close</SecondaryButton>
|
|
<PrimaryButton
|
|
type="button"
|
|
onClick={handleSubmit(handleResponse)}
|
|
loading={isSubmitting}
|
|
>
|
|
{isSubmitting
|
|
? "Generating response..."
|
|
: response === ""
|
|
? "Generate response"
|
|
: "Generate again"}
|
|
</PrimaryButton>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
};
|