mirror of
https://github.com/makeplane/plane
synced 2024-06-14 14:31:34 +00:00
chore: updated delete dropdown and handled the repeated values while creating and updating the estimate point
This commit is contained in:
parent
18c5b2a0a6
commit
c2e07c6b7c
@ -62,7 +62,6 @@ export const CreateEstimateModal: FC<TCreateEstimateModal> = observer((props) =>
|
|||||||
estimatePoints.map((point) => point.value),
|
estimatePoints.map((point) => point.value),
|
||||||
estimateSystem
|
estimateSystem
|
||||||
);
|
);
|
||||||
console.log("isRepeated", isRepeated);
|
|
||||||
if (!isRepeated) {
|
if (!isRepeated) {
|
||||||
const payload: IEstimateFormData = {
|
const payload: IEstimateFormData = {
|
||||||
estimate: {
|
estimate: {
|
||||||
@ -80,6 +79,11 @@ export const CreateEstimateModal: FC<TCreateEstimateModal> = observer((props) =>
|
|||||||
});
|
});
|
||||||
handleClose();
|
handleClose();
|
||||||
} else {
|
} else {
|
||||||
|
setToast({
|
||||||
|
type: TOAST_TYPE.ERROR,
|
||||||
|
title: "Error!",
|
||||||
|
message: "Estimate point values cannot be repeated",
|
||||||
|
});
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
setToast({
|
setToast({
|
||||||
|
@ -1,11 +1,12 @@
|
|||||||
import { FC, useState } from "react";
|
import { FC, useState } from "react";
|
||||||
import { observer } from "mobx-react";
|
import { observer } from "mobx-react";
|
||||||
import { Check, Info, X } from "lucide-react";
|
import { Check, Info, X } from "lucide-react";
|
||||||
import { Spinner, Tooltip } from "@plane/ui";
|
import { Spinner, TOAST_TYPE, Tooltip, setToast } from "@plane/ui";
|
||||||
// constants
|
// constants
|
||||||
import { EEstimateSystem } from "@/constants/estimates";
|
import { EEstimateSystem } from "@/constants/estimates";
|
||||||
// helpers
|
// helpers
|
||||||
import { cn } from "@/helpers/common.helper";
|
import { cn } from "@/helpers/common.helper";
|
||||||
|
import { isEstimatePointValuesRepeated } from "@/helpers/estimates";
|
||||||
// hooks
|
// hooks
|
||||||
import { useEstimate } from "@/hooks/store";
|
import { useEstimate } from "@/hooks/store";
|
||||||
|
|
||||||
@ -19,7 +20,7 @@ type TEstimatePointCreate = {
|
|||||||
export const EstimatePointCreate: FC<TEstimatePointCreate> = observer((props) => {
|
export const EstimatePointCreate: FC<TEstimatePointCreate> = observer((props) => {
|
||||||
const { workspaceSlug, projectId, estimateId, callback } = props;
|
const { workspaceSlug, projectId, estimateId, callback } = props;
|
||||||
// hooks
|
// hooks
|
||||||
const { asJson: estimate, estimatePointIds, creteEstimatePoint } = useEstimate(estimateId);
|
const { asJson: estimate, estimatePointIds, estimatePointById, creteEstimatePoint } = useEstimate(estimateId);
|
||||||
// states
|
// states
|
||||||
const [loader, setLoader] = useState(false);
|
const [loader, setLoader] = useState(false);
|
||||||
const [estimateInputValue, setEstimateInputValue] = useState("");
|
const [estimateInputValue, setEstimateInputValue] = useState("");
|
||||||
@ -50,18 +51,35 @@ export const EstimatePointCreate: FC<TEstimatePointCreate> = observer((props) =>
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isEstimateValid) {
|
const currentEstimatePointValues = estimatePointIds
|
||||||
const payload = {
|
.map((estimatePointId) => estimatePointById(estimatePointId)?.value || undefined)
|
||||||
key: estimatePointIds?.length + 1,
|
.filter((estimateValue) => estimateValue != undefined) as string[];
|
||||||
value: estimateInputValue,
|
const isRepeated =
|
||||||
};
|
(estimateType &&
|
||||||
await creteEstimatePoint(workspaceSlug, projectId, payload);
|
isEstimatePointValuesRepeated(currentEstimatePointValues, estimateType, estimateInputValue)) ||
|
||||||
setLoader(false);
|
false;
|
||||||
setError(undefined);
|
|
||||||
handleClose();
|
if (!isRepeated) {
|
||||||
|
if (isEstimateValid) {
|
||||||
|
const payload = {
|
||||||
|
key: estimatePointIds?.length + 1,
|
||||||
|
value: estimateInputValue,
|
||||||
|
};
|
||||||
|
await creteEstimatePoint(workspaceSlug, projectId, payload);
|
||||||
|
setLoader(false);
|
||||||
|
setError(undefined);
|
||||||
|
handleClose();
|
||||||
|
} else {
|
||||||
|
setLoader(false);
|
||||||
|
setError("please enter a valid estimate value");
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
setLoader(false);
|
setLoader(false);
|
||||||
setError("please enter a valid estimate value");
|
setToast({
|
||||||
|
type: TOAST_TYPE.ERROR,
|
||||||
|
title: "Error!",
|
||||||
|
message: "Estimate point values cannot be repeated",
|
||||||
|
});
|
||||||
}
|
}
|
||||||
} catch {
|
} catch {
|
||||||
setLoader(false);
|
setLoader(false);
|
||||||
|
@ -1,10 +1,10 @@
|
|||||||
import { FC, useState } from "react";
|
import { FC, useState } from "react";
|
||||||
import { observer } from "mobx-react";
|
import { observer } from "mobx-react";
|
||||||
import { Info, MoveRight, Trash2, X } from "lucide-react";
|
import { MoveRight, Trash2, X } from "lucide-react";
|
||||||
import { Select } from "@headlessui/react";
|
import { TEstimatePointsObject } from "@plane/types";
|
||||||
import { Spinner, Tooltip } from "@plane/ui";
|
import { Spinner } from "@plane/ui";
|
||||||
// helpers
|
// components
|
||||||
import { cn } from "@/helpers/common.helper";
|
import { EstimatePointDropdown } from "@/components/estimates/points";
|
||||||
// hooks
|
// hooks
|
||||||
import { useEstimate, useEstimatePoint } from "@/hooks/store";
|
import { useEstimate, useEstimatePoint } from "@/hooks/store";
|
||||||
|
|
||||||
@ -31,28 +31,37 @@ export const EstimatePointDelete: FC<TEstimatePointDelete> = observer((props) =>
|
|||||||
callback();
|
callback();
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleCreate = async () => {
|
const handleDelete = async () => {
|
||||||
if (!workspaceSlug || !projectId || !projectId) return;
|
if (!workspaceSlug || !projectId || !projectId) return;
|
||||||
try {
|
if (estimateInputValue)
|
||||||
setLoader(true);
|
try {
|
||||||
setError(undefined);
|
setLoader(true);
|
||||||
await deleteEstimatePoint(workspaceSlug, projectId, estimatePointId, estimateInputValue);
|
setError(undefined);
|
||||||
setLoader(false);
|
await deleteEstimatePoint(
|
||||||
setError(undefined);
|
workspaceSlug,
|
||||||
handleClose();
|
projectId,
|
||||||
} catch {
|
estimatePointId,
|
||||||
setLoader(false);
|
estimateInputValue === "none" ? undefined : estimateInputValue
|
||||||
setError("something went wrong. please try again later");
|
);
|
||||||
}
|
setLoader(false);
|
||||||
|
setError(undefined);
|
||||||
|
handleClose();
|
||||||
|
} catch {
|
||||||
|
setLoader(false);
|
||||||
|
setError("something went wrong. please try again later");
|
||||||
|
}
|
||||||
|
else setError("please select option");
|
||||||
};
|
};
|
||||||
|
|
||||||
// derived values
|
// derived values
|
||||||
const selectDropdownOptionIds = estimatePointIds?.filter((pointId) => pointId != estimatePointId) as string[];
|
const selectDropdownOptionIds = estimatePointIds?.filter((pointId) => pointId != estimatePointId) as string[];
|
||||||
const selectDropdownOptions = (selectDropdownOptionIds || [])?.map((pointId) => {
|
const selectDropdownOptions = (selectDropdownOptionIds || [])
|
||||||
const estimatePoint = estimatePointById(pointId);
|
?.map((pointId) => {
|
||||||
if (estimatePoint && estimatePoint?.id)
|
const estimatePoint = estimatePointById(pointId);
|
||||||
return { id: estimatePoint.id, key: estimatePoint.key, value: estimatePoint.value };
|
if (estimatePoint && estimatePoint?.id)
|
||||||
});
|
return { id: estimatePoint.id, key: estimatePoint.key, value: estimatePoint.value };
|
||||||
|
})
|
||||||
|
.filter((estimatePoint) => estimatePoint != undefined) as TEstimatePointsObject[];
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="relative flex items-center gap-2 text-base">
|
<div className="relative flex items-center gap-2 text-base">
|
||||||
@ -60,37 +69,17 @@ export const EstimatePointDelete: FC<TEstimatePointDelete> = observer((props) =>
|
|||||||
<div className="w-full border border-custom-border-200 rounded p-2.5 bg-custom-background-90">
|
<div className="w-full border border-custom-border-200 rounded p-2.5 bg-custom-background-90">
|
||||||
{estimatePoint?.value}
|
{estimatePoint?.value}
|
||||||
</div>
|
</div>
|
||||||
<div className="relative flex justify-center items-center gap-2 whitespace-nowrap">
|
<div className="text-sm first-letter:relative flex justify-center items-center gap-2 whitespace-nowrap">
|
||||||
Mark as <MoveRight size={14} />
|
Mark as <MoveRight size={14} />
|
||||||
</div>
|
</div>
|
||||||
<div
|
<EstimatePointDropdown
|
||||||
className={cn(
|
options={selectDropdownOptions}
|
||||||
"relative w-full rounded border flex items-center gap-3 p-2.5",
|
error={error}
|
||||||
error ? `border-red-500` : `border-custom-border-200`
|
callback={(estimateId: string) => {
|
||||||
)}
|
setEstimateInputValue(estimateId);
|
||||||
>
|
setError(undefined);
|
||||||
<Select
|
}}
|
||||||
className="bg-transparent flex-grow focus:ring-0 focus:border-0 focus:outline-none"
|
/>
|
||||||
value={estimateInputValue}
|
|
||||||
onChange={(e) => setEstimateInputValue(e.target.value)}
|
|
||||||
>
|
|
||||||
<option value={undefined}>None</option>
|
|
||||||
{selectDropdownOptions.map((option) => (
|
|
||||||
<option key={option?.id} value={option?.value}>
|
|
||||||
{option?.value}
|
|
||||||
</option>
|
|
||||||
))}
|
|
||||||
</Select>
|
|
||||||
{error && (
|
|
||||||
<>
|
|
||||||
<Tooltip tooltipContent={error} position="bottom">
|
|
||||||
<div className="flex-shrink-0 w-3.5 h-3.5 overflow-hidden relative flex justify-center items-center text-red-500">
|
|
||||||
<Info size={14} />
|
|
||||||
</div>
|
|
||||||
</Tooltip>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
{loader ? (
|
{loader ? (
|
||||||
<div className="w-6 h-6 flex-shrink-0 relative flex justify-center items-center rota">
|
<div className="w-6 h-6 flex-shrink-0 relative flex justify-center items-center rota">
|
||||||
@ -99,7 +88,7 @@ export const EstimatePointDelete: FC<TEstimatePointDelete> = observer((props) =>
|
|||||||
) : (
|
) : (
|
||||||
<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-red-500"
|
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-red-500"
|
||||||
onClick={handleCreate}
|
onClick={handleDelete}
|
||||||
>
|
>
|
||||||
<Trash2 size={14} />
|
<Trash2 size={14} />
|
||||||
</div>
|
</div>
|
||||||
|
@ -3,3 +3,4 @@ export * from "./preview";
|
|||||||
export * from "./create";
|
export * from "./create";
|
||||||
export * from "./update";
|
export * from "./update";
|
||||||
export * from "./delete";
|
export * from "./delete";
|
||||||
|
export * from "./select-dropdown";
|
||||||
|
119
web/components/estimates/points/edit/select-dropdown.tsx
Normal file
119
web/components/estimates/points/edit/select-dropdown.tsx
Normal file
@ -0,0 +1,119 @@
|
|||||||
|
import { FC, useRef, Fragment, useState } from "react";
|
||||||
|
import { Info, Check, ChevronDown } from "lucide-react";
|
||||||
|
import { Listbox, ListboxButton, ListboxOptions, Transition, ListboxOption } from "@headlessui/react";
|
||||||
|
import { TEstimatePointsObject } from "@plane/types";
|
||||||
|
import { Tooltip } from "@plane/ui";
|
||||||
|
// helpers
|
||||||
|
import { cn } from "@/helpers/common.helper";
|
||||||
|
// hooks
|
||||||
|
import useDynamicDropdownPosition from "@/hooks/use-dynamic-dropdown";
|
||||||
|
import useOutsideClickDetector from "@/hooks/use-outside-click-detector";
|
||||||
|
|
||||||
|
type TEstimatePointDropdown = {
|
||||||
|
options: TEstimatePointsObject[];
|
||||||
|
error: string | undefined;
|
||||||
|
callback: (estimateId: string) => void;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const EstimatePointDropdown: FC<TEstimatePointDropdown> = (props) => {
|
||||||
|
const { options, error, callback } = props;
|
||||||
|
// states
|
||||||
|
const [isDropdownOpen, setIsDropdownOpen] = useState(false);
|
||||||
|
const [selectedOption, setSelectedOption] = useState<string | undefined>(undefined);
|
||||||
|
// ref
|
||||||
|
const dropdownContainerRef = useRef<HTMLDivElement>(null);
|
||||||
|
const buttonRef = useRef<HTMLButtonElement>(null);
|
||||||
|
const dropdownRef = useRef<HTMLDivElement>(null);
|
||||||
|
|
||||||
|
useDynamicDropdownPosition(isDropdownOpen, () => setIsDropdownOpen(false), buttonRef, dropdownRef);
|
||||||
|
useOutsideClickDetector(dropdownContainerRef, () => setIsDropdownOpen(false));
|
||||||
|
|
||||||
|
// derived values
|
||||||
|
const selectedValue = options.find((option) => option?.id === selectedOption) || undefined;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div ref={dropdownContainerRef} className="w-full relative">
|
||||||
|
<Listbox
|
||||||
|
as="div"
|
||||||
|
value={selectedOption}
|
||||||
|
onChange={(selectedOption) => {
|
||||||
|
setSelectedOption(selectedOption);
|
||||||
|
callback(selectedOption);
|
||||||
|
setIsDropdownOpen(false);
|
||||||
|
}}
|
||||||
|
className="w-full flex-shrink-0 text-left"
|
||||||
|
>
|
||||||
|
<ListboxButton
|
||||||
|
type="button"
|
||||||
|
ref={buttonRef}
|
||||||
|
onClick={() => setIsDropdownOpen((prev) => !prev)}
|
||||||
|
className={cn(
|
||||||
|
"relative w-full rounded border flex items-center gap-3 p-2.5",
|
||||||
|
error ? `border-red-500` : `border-custom-border-200`
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className={cn(`w-full text-sm text-left`, !selectedValue ? "text-custom-text-300" : "text-custom-text-100")}
|
||||||
|
>
|
||||||
|
{selectedValue?.value || "Select an estimate point"}
|
||||||
|
</div>
|
||||||
|
<ChevronDown className={`size-3 ${true ? "stroke-onboarding-text-400" : "stroke-onboarding-text-100"}`} />
|
||||||
|
{error && (
|
||||||
|
<>
|
||||||
|
<Tooltip tooltipContent={error} position="bottom">
|
||||||
|
<div className="flex-shrink-0 w-3.5 h-3.5 overflow-hidden relative flex justify-center items-center text-red-500">
|
||||||
|
<Info size={14} />
|
||||||
|
</div>
|
||||||
|
</Tooltip>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</ListboxButton>
|
||||||
|
<Transition
|
||||||
|
show={isDropdownOpen}
|
||||||
|
as={Fragment}
|
||||||
|
enter="transition ease-out duration-100"
|
||||||
|
enterFrom="transform opacity-0 scale-95"
|
||||||
|
enterTo="transform opacity-100 scale-100"
|
||||||
|
leave="transition ease-in duration-75"
|
||||||
|
leaveFrom="transform opacity-100 scale-100"
|
||||||
|
leaveTo="transform opacity-0 scale-95"
|
||||||
|
>
|
||||||
|
<ListboxOptions
|
||||||
|
ref={dropdownRef}
|
||||||
|
className="fixed z-10 mt-1 h-fit w-48 sm:w-60 overflow-y-auto rounded-md border border-custom-border-200 bg-custom-background-100 shadow-sm focus:outline-none"
|
||||||
|
>
|
||||||
|
<div className="p-1.5">
|
||||||
|
<ListboxOption
|
||||||
|
value={"none"}
|
||||||
|
className={cn(
|
||||||
|
`cursor-pointer select-none truncate rounded px-1 py-1.5 hover:bg-custom-background-90`,
|
||||||
|
selectedOption === "none" ? "text-custom-text-100" : "text-custom-text-300"
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
<div className="relative flex items-center text-wrap gap-2 px-1 py-0.5">
|
||||||
|
<div className="text-sm font-medium w-full line-clamp-1">None</div>
|
||||||
|
{selectedOption === "none" && <Check size={12} />}
|
||||||
|
</div>
|
||||||
|
</ListboxOption>
|
||||||
|
{options.map((option) => (
|
||||||
|
<ListboxOption
|
||||||
|
key={option?.key}
|
||||||
|
value={option?.id}
|
||||||
|
className={cn(
|
||||||
|
`cursor-pointer select-none truncate rounded px-1 py-1.5 hover:bg-custom-background-90`,
|
||||||
|
selectedOption === option?.id ? "text-custom-text-100" : "text-custom-text-300"
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
<div className="relative flex items-center text-wrap gap-2 px-1 py-0.5">
|
||||||
|
<div className="text-sm font-medium w-full line-clamp-1">{option.value}</div>
|
||||||
|
{selectedOption === option?.id && <Check size={12} />}
|
||||||
|
</div>
|
||||||
|
</ListboxOption>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</ListboxOptions>
|
||||||
|
</Transition>
|
||||||
|
</Listbox>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
@ -1,11 +1,12 @@
|
|||||||
import { FC, useEffect, useState } from "react";
|
import { FC, useEffect, useState } from "react";
|
||||||
import { observer } from "mobx-react";
|
import { observer } from "mobx-react";
|
||||||
import { Check, Info, X } from "lucide-react";
|
import { Check, Info, X } from "lucide-react";
|
||||||
import { Spinner, Tooltip } from "@plane/ui";
|
import { Spinner, TOAST_TYPE, Tooltip, setToast } from "@plane/ui";
|
||||||
// constants
|
// constants
|
||||||
import { EEstimateSystem } from "@/constants/estimates";
|
import { EEstimateSystem } from "@/constants/estimates";
|
||||||
// helpers
|
// helpers
|
||||||
import { cn } from "@/helpers/common.helper";
|
import { cn } from "@/helpers/common.helper";
|
||||||
|
import { isEstimatePointValuesRepeated } from "@/helpers/estimates";
|
||||||
// hooks
|
// hooks
|
||||||
import { useEstimate, useEstimatePoint } from "@/hooks/store";
|
import { useEstimate, useEstimatePoint } from "@/hooks/store";
|
||||||
|
|
||||||
@ -20,7 +21,7 @@ type TEstimatePointUpdate = {
|
|||||||
export const EstimatePointUpdate: FC<TEstimatePointUpdate> = observer((props) => {
|
export const EstimatePointUpdate: FC<TEstimatePointUpdate> = observer((props) => {
|
||||||
const { workspaceSlug, projectId, estimateId, estimatePointId, callback } = props;
|
const { workspaceSlug, projectId, estimateId, estimatePointId, callback } = props;
|
||||||
// hooks
|
// hooks
|
||||||
const { asJson: estimate, estimatePointIds } = useEstimate(estimateId);
|
const { asJson: estimate, estimatePointIds, estimatePointById } = useEstimate(estimateId);
|
||||||
const { asJson: estimatePoint, updateEstimatePoint } = useEstimatePoint(estimateId, estimatePointId);
|
const { asJson: estimatePoint, updateEstimatePoint } = useEstimatePoint(estimateId, estimatePointId);
|
||||||
// states
|
// states
|
||||||
const [loader, setLoader] = useState(false);
|
const [loader, setLoader] = useState(false);
|
||||||
@ -36,7 +37,7 @@ export const EstimatePointUpdate: FC<TEstimatePointUpdate> = observer((props) =>
|
|||||||
callback();
|
callback();
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleCreate = async () => {
|
const handleUpdate = async () => {
|
||||||
if (!workspaceSlug || !projectId || !projectId || !estimatePointIds) return;
|
if (!workspaceSlug || !projectId || !projectId || !estimatePointIds) return;
|
||||||
if (estimateInputValue)
|
if (estimateInputValue)
|
||||||
try {
|
try {
|
||||||
@ -56,17 +57,34 @@ export const EstimatePointUpdate: FC<TEstimatePointUpdate> = observer((props) =>
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isEstimateValid) {
|
const currentEstimatePointValues = estimatePointIds
|
||||||
const payload = {
|
.map((estimatePointId) => estimatePointById(estimatePointId)?.value || undefined)
|
||||||
value: estimateInputValue,
|
.filter((estimateValue) => estimateValue != undefined) as string[];
|
||||||
};
|
const isRepeated =
|
||||||
await updateEstimatePoint(workspaceSlug, projectId, payload);
|
(estimateType &&
|
||||||
setLoader(false);
|
isEstimatePointValuesRepeated(currentEstimatePointValues, estimateType, estimateInputValue)) ||
|
||||||
setError(undefined);
|
false;
|
||||||
handleClose();
|
|
||||||
|
if (!isRepeated) {
|
||||||
|
if (isEstimateValid) {
|
||||||
|
const payload = {
|
||||||
|
value: estimateInputValue,
|
||||||
|
};
|
||||||
|
await updateEstimatePoint(workspaceSlug, projectId, payload);
|
||||||
|
setLoader(false);
|
||||||
|
setError(undefined);
|
||||||
|
handleClose();
|
||||||
|
} else {
|
||||||
|
setLoader(false);
|
||||||
|
setError("please enter a valid estimate value");
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
setLoader(false);
|
setLoader(false);
|
||||||
setError("please enter a valid estimate value");
|
setToast({
|
||||||
|
type: TOAST_TYPE.ERROR,
|
||||||
|
title: "Error!",
|
||||||
|
message: "Estimate point values cannot be repeated",
|
||||||
|
});
|
||||||
}
|
}
|
||||||
} catch {
|
} catch {
|
||||||
setLoader(false);
|
setLoader(false);
|
||||||
@ -110,7 +128,7 @@ export const EstimatePointUpdate: FC<TEstimatePointUpdate> = observer((props) =>
|
|||||||
) : (
|
) : (
|
||||||
<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"
|
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={handleCreate}
|
onClick={handleUpdate}
|
||||||
>
|
>
|
||||||
<Check size={14} />
|
<Check size={14} />
|
||||||
</div>
|
</div>
|
||||||
|
@ -1,34 +1,33 @@
|
|||||||
import { EEstimateSystem } from "@/constants/estimates";
|
import { EEstimateSystem } from "@/constants/estimates";
|
||||||
|
|
||||||
|
|
||||||
export const isEstimatePointValuesRepeated = (
|
export const isEstimatePointValuesRepeated = (
|
||||||
estimatePoints: string[],
|
estimatePoints: string[],
|
||||||
estimateType: EEstimateSystem,
|
estimateType: EEstimateSystem,
|
||||||
newEstimatePoint?: string | undefined
|
newEstimatePoint?: string | undefined
|
||||||
) => {
|
) => {
|
||||||
const currentEstimatePoints = estimatePoints.map((estimatePoint) => estimatePoint.trim());
|
const currentEstimatePoints = estimatePoints.map((estimatePoint) => estimatePoint.trim());
|
||||||
let isValid = false;
|
let isRepeated = false;
|
||||||
|
|
||||||
if (newEstimatePoint === undefined) {
|
if (newEstimatePoint === undefined) {
|
||||||
if (estimateType === EEstimateSystem.CATEGORIES) {
|
if (estimateType === EEstimateSystem.CATEGORIES) {
|
||||||
const points = new Set(currentEstimatePoints);
|
const points = new Set(currentEstimatePoints);
|
||||||
if (points.size === currentEstimatePoints.length) isValid = true;
|
if (points.size != currentEstimatePoints.length) isRepeated = true;
|
||||||
} else if ([EEstimateSystem.POINTS, EEstimateSystem.TIME].includes(estimateType)) {
|
} else if ([EEstimateSystem.POINTS, EEstimateSystem.TIME].includes(estimateType)) {
|
||||||
currentEstimatePoints.map((point) => {
|
currentEstimatePoints.map((point) => {
|
||||||
if (Number(point) === Number(newEstimatePoint)) isValid = true;
|
if (Number(point) === Number(newEstimatePoint)) isRepeated = true;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if (estimateType === EEstimateSystem.CATEGORIES) {
|
if (estimateType === EEstimateSystem.CATEGORIES) {
|
||||||
currentEstimatePoints.map((point) => {
|
currentEstimatePoints.map((point) => {
|
||||||
if (point === newEstimatePoint.trim()) isValid = true;
|
if (point === newEstimatePoint.trim()) isRepeated = true;
|
||||||
});
|
});
|
||||||
} else if ([EEstimateSystem.POINTS, EEstimateSystem.TIME].includes(estimateType)) {
|
} else if ([EEstimateSystem.POINTS, EEstimateSystem.TIME].includes(estimateType)) {
|
||||||
currentEstimatePoints.map((point) => {
|
currentEstimatePoints.map((point) => {
|
||||||
if (Number(point) === Number(newEstimatePoint.trim())) isValid = true;
|
if (Number(point) === Number(newEstimatePoint.trim())) isRepeated = true;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return isValid;
|
return isRepeated;
|
||||||
};
|
};
|
||||||
|
@ -16,7 +16,7 @@
|
|||||||
"@atlaskit/pragmatic-drag-and-drop-auto-scroll": "^1.3.0",
|
"@atlaskit/pragmatic-drag-and-drop-auto-scroll": "^1.3.0",
|
||||||
"@atlaskit/pragmatic-drag-and-drop-hitbox": "^1.0.3",
|
"@atlaskit/pragmatic-drag-and-drop-hitbox": "^1.0.3",
|
||||||
"@blueprintjs/popover2": "^1.13.3",
|
"@blueprintjs/popover2": "^1.13.3",
|
||||||
"@headlessui/react": "^2.0.3",
|
"@headlessui/react": "^2.0.4",
|
||||||
"@nivo/bar": "0.80.0",
|
"@nivo/bar": "0.80.0",
|
||||||
"@nivo/calendar": "0.80.0",
|
"@nivo/calendar": "0.80.0",
|
||||||
"@nivo/core": "0.80.0",
|
"@nivo/core": "0.80.0",
|
||||||
|
@ -1600,7 +1600,7 @@
|
|||||||
"@tanstack/react-virtual" "^3.0.0-beta.60"
|
"@tanstack/react-virtual" "^3.0.0-beta.60"
|
||||||
client-only "^0.0.1"
|
client-only "^0.0.1"
|
||||||
|
|
||||||
"@headlessui/react@^2.0.3":
|
"@headlessui/react@^2.0.3", "@headlessui/react@^2.0.4":
|
||||||
version "2.0.4"
|
version "2.0.4"
|
||||||
resolved "https://registry.yarnpkg.com/@headlessui/react/-/react-2.0.4.tgz#46cb39ca9dde3c2d15f4706c81dad78405b608f0"
|
resolved "https://registry.yarnpkg.com/@headlessui/react/-/react-2.0.4.tgz#46cb39ca9dde3c2d15f4706c81dad78405b608f0"
|
||||||
integrity sha512-16d/rOLeYsFsmPlRmXGu8DCBzrWD0zV1Ccx3n73wN87yFu8Y9+X04zflv8EJEt9TAYRyLKOmQXUnOnqQl6NgpA==
|
integrity sha512-16d/rOLeYsFsmPlRmXGu8DCBzrWD0zV1Ccx3n73wN87yFu8Y9+X04zflv8EJEt9TAYRyLKOmQXUnOnqQl6NgpA==
|
||||||
|
Loading…
Reference in New Issue
Block a user