mirror of
https://github.com/makeplane/plane
synced 2024-06-14 14:31:34 +00:00
293 lines
12 KiB
TypeScript
293 lines
12 KiB
TypeScript
import { FC, Fragment, useEffect, useRef, useState } from "react";
|
|
import { observer } from "mobx-react";
|
|
import { Check, GripVertical, MoveRight, Pencil, Trash2, X } from "lucide-react";
|
|
import { Select } from "@headlessui/react";
|
|
import { TEstimatePointsObject } from "@plane/types";
|
|
import { Draggable, Spinner } from "@plane/ui";
|
|
// constants
|
|
import { EEstimateUpdateStages } from "@/constants/estimates";
|
|
// helpers
|
|
import { cn } from "@/helpers/common.helper";
|
|
import { useEstimate, useEstimatePoint } from "@/hooks/store";
|
|
|
|
type TEstimatePointItem = {
|
|
workspaceSlug: string;
|
|
projectId: string;
|
|
estimateId: string | undefined;
|
|
mode: EEstimateUpdateStages;
|
|
item: TEstimatePointsObject;
|
|
estimatePoints: TEstimatePointsObject[];
|
|
editItem: (value: string) => void;
|
|
deleteItem: () => void;
|
|
handleEstimatePoints: (value: TEstimatePointsObject[]) => void;
|
|
};
|
|
|
|
export const EstimatePointItem: FC<TEstimatePointItem> = observer((props) => {
|
|
// props
|
|
const {
|
|
workspaceSlug,
|
|
projectId,
|
|
estimateId,
|
|
mode,
|
|
item,
|
|
estimatePoints,
|
|
editItem,
|
|
deleteItem,
|
|
handleEstimatePoints,
|
|
} = props;
|
|
const { id, key, value } = item;
|
|
// hooks
|
|
const { asJson: estimate, creteEstimatePoint, deleteEstimatePoint } = useEstimate(estimateId);
|
|
const { updateEstimatePoint } = useEstimatePoint(estimateId, id);
|
|
// ref
|
|
const inputRef = useRef<HTMLInputElement>(null);
|
|
// states
|
|
const [inputValue, setInputValue] = useState<string | undefined>(undefined);
|
|
// handling editing states
|
|
const [estimateEditLoader, setEstimateEditLoader] = useState(false);
|
|
const [deletedEstimateValue, setDeletedEstimateValue] = useState<string | undefined>(undefined);
|
|
const [isEstimateEditing, setIsEstimateEditing] = useState(false);
|
|
const [isEstimateDeleting, setIsEstimateDeleting] = useState(false);
|
|
|
|
useEffect(() => {
|
|
if (value && inputValue === undefined) setInputValue(value);
|
|
}, [value, inputValue]);
|
|
|
|
const handleCreateEdit = (value: string) => {
|
|
setInputValue(value);
|
|
editItem(value);
|
|
};
|
|
|
|
const handleNewEstimatePoint = async () => {
|
|
if (inputValue) {
|
|
try {
|
|
setEstimateEditLoader(true);
|
|
const estimatePoint = await creteEstimatePoint(workspaceSlug, projectId, { key: key, value: inputValue });
|
|
if (estimatePoint)
|
|
handleEstimatePoints([...estimatePoints, { id: estimatePoint.id, key: key, value: inputValue }]);
|
|
setIsEstimateEditing(false);
|
|
setEstimateEditLoader(false);
|
|
} catch (error) {
|
|
setEstimateEditLoader(false);
|
|
}
|
|
}
|
|
};
|
|
|
|
const handleEdit = async () => {
|
|
if (id) {
|
|
try {
|
|
setEstimateEditLoader(true);
|
|
await updateEstimatePoint(workspaceSlug, projectId, { key: key, value: inputValue });
|
|
setIsEstimateEditing(false);
|
|
setEstimateEditLoader(false);
|
|
} catch (error) {
|
|
setEstimateEditLoader(false);
|
|
}
|
|
} else {
|
|
if (inputValue) editItem(inputValue);
|
|
}
|
|
};
|
|
|
|
const handleDelete = async () => {
|
|
if (id) {
|
|
try {
|
|
setEstimateEditLoader(true);
|
|
await deleteEstimatePoint(workspaceSlug, projectId, id, deletedEstimateValue);
|
|
setIsEstimateDeleting(false);
|
|
setEstimateEditLoader(false);
|
|
} catch (error) {
|
|
setEstimateEditLoader(false);
|
|
}
|
|
} else {
|
|
deleteItem();
|
|
}
|
|
};
|
|
|
|
const selectDropdownOptions = estimate && estimate?.points ? estimate?.points.filter((point) => point.id !== id) : [];
|
|
|
|
return (
|
|
<Draggable data={item}>
|
|
{!id && (
|
|
<>
|
|
{mode === EEstimateUpdateStages.CREATE && (
|
|
<div className="border border-custom-border-200 rounded relative flex items-center px-2.5 gap-2">
|
|
<div className="rounded-sm w-6 h-6 flex-shrink-0 relative flex justify-center items-center hover:bg-custom-background-80 transition-colors cursor-pointer">
|
|
<GripVertical size={14} className="text-custom-text-200" />
|
|
</div>
|
|
<input
|
|
ref={inputRef}
|
|
type="text"
|
|
value={inputValue}
|
|
onChange={(e) => handleCreateEdit(e.target.value)}
|
|
className="flex-grow border-none bg-transparent focus:ring-0 focus:border-0 focus:outline-none py-2.5 w-full"
|
|
/>
|
|
<div
|
|
className="rounded-sm w-6 h-6 flex-shrink-0 relative flex justify-center items-center hover:bg-custom-background-80 transition-colors cursor-pointer"
|
|
onClick={handleDelete}
|
|
>
|
|
<Trash2 size={14} className="text-custom-text-200" />
|
|
</div>
|
|
</div>
|
|
)}
|
|
|
|
{mode === EEstimateUpdateStages.EDIT && (
|
|
<div className="relative flex items-center gap-2">
|
|
<div className="w-full border border-custom-border-200 rounded">
|
|
<input
|
|
ref={inputRef}
|
|
type="text"
|
|
value={inputValue}
|
|
onChange={(e) => setInputValue(e.target.value)}
|
|
className={cn(
|
|
"border-none focus:ring-0 focus:border-0 focus:outline-none p-2.5 w-full",
|
|
isEstimateDeleting ? `bg-custom-background-90` : `bg-transparent`
|
|
)}
|
|
disabled={isEstimateDeleting}
|
|
/>
|
|
</div>
|
|
{estimateEditLoader ? (
|
|
<div className="w-6 h-6 flex-shrink-0 relative flex justify-center items-center rota">
|
|
<Spinner className="w-4 h-4" />
|
|
</div>
|
|
) : (
|
|
<div
|
|
className="rounded-sm w-6 h-6 flex-shrink-0 relative flex justify-center items-center hover:bg-custom-background-80 transition-colors cursor-pointer text-green-500"
|
|
onClick={handleNewEstimatePoint}
|
|
>
|
|
<Check size={14} />
|
|
</div>
|
|
)}
|
|
|
|
<div
|
|
className="rounded-sm w-6 h-6 flex-shrink-0 relative flex justify-center items-center hover:bg-custom-background-80 transition-colors cursor-pointer"
|
|
onClick={handleDelete}
|
|
>
|
|
<X size={14} className="text-custom-text-200" />
|
|
</div>
|
|
</div>
|
|
)}
|
|
</>
|
|
)}
|
|
|
|
{id && (
|
|
<>
|
|
{mode === EEstimateUpdateStages.EDIT && (
|
|
<>
|
|
{!isEstimateEditing && !isEstimateDeleting && (
|
|
<div className="border border-custom-border-200 rounded relative flex items-center px-2.5 gap-2">
|
|
<div className="rounded-sm w-6 h-6 flex-shrink-0 relative flex justify-center items-center hover:bg-custom-background-80 transition-colors cursor-pointer">
|
|
<GripVertical size={14} className="text-custom-text-200" />
|
|
</div>
|
|
<div className="py-2.5 flex-grow" onClick={() => setIsEstimateEditing(true)}>
|
|
{value}
|
|
</div>
|
|
<div
|
|
className="rounded-sm w-6 h-6 flex-shrink-0 relative flex justify-center items-center hover:bg-custom-background-80 transition-colors cursor-pointer"
|
|
onClick={() => setIsEstimateEditing(true)}
|
|
>
|
|
<Pencil size={14} className="text-custom-text-200" />
|
|
</div>
|
|
<div
|
|
className="rounded-sm w-6 h-6 flex-shrink-0 relative flex justify-center items-center hover:bg-custom-background-80 transition-colors cursor-pointer"
|
|
onClick={() => setIsEstimateDeleting(true)}
|
|
>
|
|
<Trash2 size={14} className="text-custom-text-200" />
|
|
</div>
|
|
</div>
|
|
)}
|
|
|
|
{(isEstimateEditing || isEstimateDeleting) && (
|
|
<div className="relative flex items-center gap-2">
|
|
<div className="flex-grow relative flex items-center gap-3">
|
|
<div className="w-full border border-custom-border-200 rounded">
|
|
<input
|
|
ref={inputRef}
|
|
type="text"
|
|
value={inputValue}
|
|
onChange={(e) => setInputValue(e.target.value)}
|
|
className={cn(
|
|
"border-none focus:ring-0 focus:border-0 focus:outline-none p-2.5 w-full",
|
|
isEstimateDeleting ? `bg-custom-background-90` : `bg-transparent`
|
|
)}
|
|
disabled={isEstimateDeleting}
|
|
/>
|
|
</div>
|
|
{isEstimateDeleting && (
|
|
<div className="text-xs relative flex justify-center items-center gap-2 whitespace-nowrap">
|
|
Mark as <MoveRight size={14} />
|
|
</div>
|
|
)}
|
|
{isEstimateDeleting && (
|
|
<div className="relative w-full rounded border border-custom-border-200 flex items-center gap-3 p-2.5">
|
|
<Select
|
|
className="bg-transparent flex-grow focus:ring-0 focus:border-0 focus:outline-none"
|
|
value={deletedEstimateValue}
|
|
onChange={(e) => setDeletedEstimateValue(e.target.value)}
|
|
>
|
|
<option value={undefined}>None</option>
|
|
{selectDropdownOptions.map((option) => (
|
|
<option key={option?.id} value={option?.value}>
|
|
{option?.value}
|
|
</option>
|
|
))}
|
|
</Select>
|
|
</div>
|
|
)}
|
|
</div>
|
|
{estimateEditLoader ? (
|
|
<div className="w-6 h-6 flex-shrink-0 relative flex justify-center items-center rota">
|
|
<Spinner className="w-4 h-4" />
|
|
</div>
|
|
) : (
|
|
<div
|
|
className={cn(
|
|
"rounded-sm w-6 h-6 flex-shrink-0 relative flex justify-center items-center hover:bg-custom-background-80 transition-colors cursor-pointer",
|
|
isEstimateEditing ? `text-green-500` : `text-red-500`
|
|
)}
|
|
onClick={() => (isEstimateEditing ? handleEdit() : handleDelete())}
|
|
>
|
|
{isEstimateEditing ? <Check size={14} /> : <Trash2 size={14} />}
|
|
</div>
|
|
)}
|
|
|
|
<div
|
|
className="rounded-sm w-6 h-6 flex-shrink-0 relative flex justify-center items-center hover:bg-custom-background-80 transition-colors cursor-pointer"
|
|
onClick={() => (isEstimateEditing ? setIsEstimateEditing(false) : setIsEstimateDeleting(false))}
|
|
>
|
|
<X size={14} className="text-custom-text-200" />
|
|
</div>
|
|
</div>
|
|
)}
|
|
</>
|
|
)}
|
|
|
|
{mode === EEstimateUpdateStages.SWITCH && (
|
|
<div className="flex-grow relative flex items-center gap-3">
|
|
<div className="flex-grow border border-custom-border-200 rounded">
|
|
<input
|
|
ref={inputRef}
|
|
type="text"
|
|
value={inputValue}
|
|
onChange={(e) => setInputValue(e.target.value)}
|
|
className="flex-grow border-none focus:ring-0 focus:border-0 focus:outline-none p-2.5 bg-custom-background-90 w-full"
|
|
disabled
|
|
/>
|
|
</div>
|
|
<MoveRight size={14} />
|
|
<div className="flex-grow border border-custom-border-200 rounded">
|
|
<input
|
|
ref={inputRef}
|
|
type="text"
|
|
value={inputValue}
|
|
onChange={(e) => setInputValue(e.target.value)}
|
|
className="flex-grow border-none bg-transparent focus:ring-0 focus:border-0 focus:outline-none p-2.5 w-full"
|
|
/>
|
|
</div>
|
|
</div>
|
|
)}
|
|
</>
|
|
)}
|
|
</Draggable>
|
|
);
|
|
});
|