import React, { forwardRef, useEffect } from "react"; import { observer } from "mobx-react"; import { useRouter } from "next/router"; import { TwitterPicker } from "react-color"; import { Controller, SubmitHandler, useForm } from "react-hook-form"; import { Popover, Transition } from "@headlessui/react"; import { IIssueLabel } from "@plane/types"; // ui import { Button, Input, TOAST_TYPE, setToast } from "@plane/ui"; // constants import { E_LABELS, LABEL_CREATED, LABEL_UPDATED } from "@/constants/event-tracker"; import { getRandomLabelColor, LABEL_COLOR_OPTIONS } from "@/constants/label"; // hooks import { useLabel, useEventTracker } from "@/hooks/store"; // types type Props = { labelForm: boolean; setLabelForm: React.Dispatch>; isUpdating: boolean; labelToUpdate?: IIssueLabel; onClose?: () => void; }; const defaultValues: Partial = { name: "", color: "rgb(var(--color-text-200))", }; export const CreateUpdateLabelInline = observer( forwardRef(function CreateUpdateLabelInline(props, ref) { const { labelForm, setLabelForm, isUpdating, labelToUpdate, onClose } = props; // router const router = useRouter(); const { workspaceSlug, projectId } = router.query; // store hooks const { createLabel, updateLabel } = useLabel(); const { captureEvent } = useEventTracker(); // form info const { handleSubmit, control, reset, formState: { errors, isSubmitting, dirtyFields }, watch, setValue, setFocus, } = useForm({ defaultValues, }); const handleClose = () => { setLabelForm(false); reset(defaultValues); if (onClose) onClose(); }; const handleLabelCreate: SubmitHandler = async (formData) => { if (!workspaceSlug || !projectId || isSubmitting) return; await createLabel(workspaceSlug.toString(), projectId.toString(), formData) .then((res) => { captureEvent(LABEL_CREATED, { label_id: res.id, color: res.color, parent: res.parent, element: E_LABELS, state: "SUCCESS", }); handleClose(); reset(defaultValues); }) .catch((error) => { setToast({ title: "Oops!", type: TOAST_TYPE.ERROR, message: error?.error ?? "Error while adding the label", }); reset(formData); }); }; const handleLabelUpdate: SubmitHandler = async (formData) => { if (!workspaceSlug || !projectId || isSubmitting) return; // eslint-disable-next-line @typescript-eslint/no-non-null-asserted-optional-chain await updateLabel(workspaceSlug.toString(), projectId.toString(), labelToUpdate?.id!, formData) .then((res) => { captureEvent(LABEL_UPDATED, { label_id: res.id, color: res.color, parent: res.parent, change_details: Object.keys(dirtyFields), element: E_LABELS, state: "SUCCESS", }); reset(defaultValues); handleClose(); }) .catch((error) => { setToast({ title: "Oops!", type: TOAST_TYPE.ERROR, message: error?.error ?? "Error while updating the label", }); reset(formData); }); }; /** * For settings focus on name input */ useEffect(() => { setFocus("name"); }, [setFocus, labelForm]); useEffect(() => { if (!labelToUpdate) return; setValue("name", labelToUpdate.name); setValue("color", labelToUpdate.color && labelToUpdate.color !== "" ? labelToUpdate.color : "#000"); }, [labelToUpdate, setValue]); useEffect(() => { if (labelToUpdate) { setValue("color", labelToUpdate.color && labelToUpdate.color !== "" ? labelToUpdate.color : "#000"); return; } setValue("color", getRandomLabelColor()); }, [labelToUpdate, setValue]); return (
{ e.preventDefault(); handleSubmit(isUpdating ? handleLabelUpdate : handleLabelCreate)(); }} className={`flex w-full scroll-m-8 items-center gap-2 bg-custom-background-100 ${labelForm ? "" : "hidden"}`} >
{({ open }) => ( <> ( onChange(value.hex)} /> )} /> )}
( )} />
); }) );