mirror of
https://github.com/makeplane/plane
synced 2024-06-14 14:31:34 +00:00
chore: updated create estimate
This commit is contained in:
parent
3c37bc1943
commit
8dbf1d55ab
@ -2,6 +2,9 @@ import { FC } from "react";
|
||||
import Image from "next/image";
|
||||
import { useTheme } from "next-themes";
|
||||
import { Button } from "@plane/ui";
|
||||
// public images
|
||||
import EstimateEmptyDarkImage from "@/public/empty-state/estimates/dark.svg";
|
||||
import EstimateEmptyLightImage from "@/public/empty-state/estimates/light.svg";
|
||||
|
||||
type TEstimateEmptyScreen = {
|
||||
onButtonClick: () => void;
|
||||
@ -12,9 +15,7 @@ export const EstimateEmptyScreen: FC<TEstimateEmptyScreen> = (props) => {
|
||||
const { onButtonClick } = props;
|
||||
const { resolvedTheme } = useTheme();
|
||||
|
||||
const emptyScreenImage = `/empty-state/project-settings/estimates-${
|
||||
resolvedTheme === "light" ? "light" : "dark"
|
||||
}.webp`;
|
||||
const emptyScreenImage = resolvedTheme === "light" ? EstimateEmptyLightImage : EstimateEmptyDarkImage;
|
||||
|
||||
return (
|
||||
<div className="relative flex flex-col justify-center items-center text-center gap-8 border border-custom-border-300 rounded bg-custom-background-90 py-10">
|
||||
|
44
web/components/estimates/estimate-disable-switch.tsx
Normal file
44
web/components/estimates/estimate-disable-switch.tsx
Normal file
@ -0,0 +1,44 @@
|
||||
import { FC } from "react";
|
||||
import { observer } from "mobx-react";
|
||||
import { TOAST_TYPE, ToggleSwitch, setToast } from "@plane/ui";
|
||||
// hooks
|
||||
import { useProject } from "@/hooks/store";
|
||||
|
||||
type TEstimateDisableSwitch = {
|
||||
workspaceSlug: string;
|
||||
projectId: string;
|
||||
isAdmin: boolean;
|
||||
};
|
||||
|
||||
export const EstimateDisableSwitch: FC<TEstimateDisableSwitch> = observer((props) => {
|
||||
const { workspaceSlug, projectId, isAdmin } = props;
|
||||
// hooks
|
||||
const { updateProject, currentProjectDetails } = useProject();
|
||||
|
||||
const disableEstimate = async () => {
|
||||
if (!workspaceSlug || !projectId) return;
|
||||
try {
|
||||
await updateProject(workspaceSlug, projectId, { estimate: null });
|
||||
setToast({
|
||||
type: TOAST_TYPE.SUCCESS,
|
||||
title: "Success!",
|
||||
message: "Estimates have been disabled",
|
||||
});
|
||||
} catch (err) {
|
||||
setToast({
|
||||
type: TOAST_TYPE.ERROR,
|
||||
title: "Error!",
|
||||
message: "Estimate could not be disabled. Please try again",
|
||||
});
|
||||
}
|
||||
};
|
||||
|
||||
return (
|
||||
<ToggleSwitch
|
||||
value={Boolean(currentProjectDetails?.estimate)}
|
||||
onChange={disableEstimate}
|
||||
disabled={!isAdmin}
|
||||
size="sm"
|
||||
/>
|
||||
);
|
||||
});
|
@ -1,18 +0,0 @@
|
||||
import { FC } from "react";
|
||||
import { observer } from "mobx-react";
|
||||
|
||||
type TEstimateDisable = {
|
||||
workspaceSlug: string;
|
||||
projectId: string;
|
||||
};
|
||||
|
||||
export const EstimateDisable: FC<TEstimateDisable> = observer((props) => {
|
||||
const { workspaceSlug, projectId } = props;
|
||||
// hooks
|
||||
|
||||
return (
|
||||
<div>
|
||||
Estimate Disable {workspaceSlug} {projectId}
|
||||
</div>
|
||||
);
|
||||
});
|
46
web/components/estimates/estimate-list-item.tsx
Normal file
46
web/components/estimates/estimate-list-item.tsx
Normal file
@ -0,0 +1,46 @@
|
||||
import { FC } from "react";
|
||||
import { observer } from "mobx-react";
|
||||
import { Pen } from "lucide-react";
|
||||
// helpers
|
||||
import { cn } from "@/helpers/common.helper";
|
||||
// hooks
|
||||
import { useProjectEstimates } from "@/hooks/store";
|
||||
|
||||
type TEstimateListItem = {
|
||||
estimateId: string;
|
||||
isAdmin: boolean;
|
||||
isEditable: boolean;
|
||||
onEditClick?: (estimateId: string) => void;
|
||||
};
|
||||
|
||||
export const EstimateListItem: FC<TEstimateListItem> = observer((props) => {
|
||||
const { estimateId, isAdmin, isEditable, onEditClick } = props;
|
||||
// hooks
|
||||
const { estimateById } = useProjectEstimates();
|
||||
|
||||
const currentEstimate = estimateById(estimateId);
|
||||
|
||||
if (!currentEstimate) return <></>;
|
||||
|
||||
return (
|
||||
<div
|
||||
className={cn(
|
||||
"relative border-b border-custom-border-200 flex justify-between items-center gap-3 py-3.5",
|
||||
isAdmin && isEditable ? `text-custom-text-100` : `text-custom-text-200`
|
||||
)}
|
||||
>
|
||||
<div className="space-y-1">
|
||||
<h3 className="font-medium text-base">{currentEstimate?.name}</h3>
|
||||
<p className="text-xs">{currentEstimate?.points?.map((estimatePoint) => estimatePoint?.value).join(", ")}</p>
|
||||
</div>
|
||||
{isAdmin && isEditable && (
|
||||
<div
|
||||
className="relative flex-shrink-0 w-6 h-6 flex justify-center items-center rounded cursor-pointer transition-colors overflow-hidden hover:bg-custom-background-80"
|
||||
onClick={() => onEditClick && onEditClick(estimateId)}
|
||||
>
|
||||
<Pen size={12} />
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
);
|
||||
});
|
31
web/components/estimates/estimate-list.tsx
Normal file
31
web/components/estimates/estimate-list.tsx
Normal file
@ -0,0 +1,31 @@
|
||||
import { FC } from "react";
|
||||
import { observer } from "mobx-react";
|
||||
// components
|
||||
import { EstimateListItem } from "@/components/estimates";
|
||||
|
||||
type TEstimateList = {
|
||||
estimateIds: string[] | undefined;
|
||||
isAdmin: boolean;
|
||||
isEditable?: boolean;
|
||||
onEditClick?: (estimateId: string) => void;
|
||||
};
|
||||
|
||||
export const EstimateList: FC<TEstimateList> = observer((props) => {
|
||||
const { estimateIds, isAdmin, isEditable = false, onEditClick } = props;
|
||||
|
||||
if (!estimateIds || estimateIds?.length <= 0) return <></>;
|
||||
return (
|
||||
<div>
|
||||
{estimateIds &&
|
||||
estimateIds.map((estimateId) => (
|
||||
<EstimateListItem
|
||||
key={estimateId}
|
||||
estimateId={estimateId}
|
||||
isAdmin={isAdmin}
|
||||
isEditable={isEditable}
|
||||
onEditClick={onEditClick}
|
||||
/>
|
||||
))}
|
||||
</div>
|
||||
);
|
||||
});
|
@ -1,16 +1,18 @@
|
||||
export * from "./root";
|
||||
|
||||
export * from "./empty-screen";
|
||||
export * from "./loader-screen";
|
||||
|
||||
export * from "./estimate-search";
|
||||
export * from "./estimate-disable";
|
||||
export * from "./inline-editable";
|
||||
export * from "./estimate-disable-switch";
|
||||
|
||||
export * from "./root";
|
||||
export * from "./estimate-item";
|
||||
// estimates
|
||||
export * from "./estimate-list";
|
||||
export * from "./estimate-list-item";
|
||||
|
||||
// estimate create
|
||||
// create
|
||||
export * from "./create";
|
||||
|
||||
// estimate update
|
||||
|
||||
// estimate delete
|
||||
// estimate points
|
||||
export * from "./points/estimate-point-item";
|
||||
export * from "./points/inline-editable";
|
||||
|
@ -1,9 +1,11 @@
|
||||
import { FC } from "react";
|
||||
import { Loader } from "@plane/ui";
|
||||
|
||||
export const EstimateLoaderScreen: FC = () => (
|
||||
<div>
|
||||
{/* header */}
|
||||
{/* estimate disable */}
|
||||
<div>Loading screen</div>
|
||||
</div>
|
||||
<Loader className="mt-5 space-y-5">
|
||||
<Loader.Item height="40px" />
|
||||
<Loader.Item height="40px" />
|
||||
<Loader.Item height="40px" />
|
||||
<Loader.Item height="40px" />
|
||||
</Loader>
|
||||
);
|
||||
|
@ -3,7 +3,7 @@ import { Check, GripVertical, MoveRight, Pencil, Trash2, X } from "lucide-react"
|
||||
import { Select } from "@headlessui/react";
|
||||
import { Draggable } from "@plane/ui";
|
||||
import { InlineEdit } from "./inline-editable";
|
||||
import { TEstimatePointsObject } from "./types";
|
||||
import { TEstimatePointsObject } from "../types";
|
||||
|
||||
type Props = {
|
||||
item: TEstimatePointsObject;
|
@ -1,26 +1,29 @@
|
||||
import { FC, useState } from "react";
|
||||
import { observer } from "mobx-react";
|
||||
import useSWR from "swr";
|
||||
import { IEstimate } from "@plane/types";
|
||||
import { Button, Loader, TOAST_TYPE, setToast } from "@plane/ui";
|
||||
// components
|
||||
import { EmptyState } from "@/components/empty-state";
|
||||
import { EstimateEmptyScreen, EstimateLoaderScreen, CreateEstimateModal } from "@/components/estimates";
|
||||
// constants
|
||||
import { EmptyStateType } from "@/constants/empty-state";
|
||||
// ee components
|
||||
import {
|
||||
EstimateLoaderScreen,
|
||||
EstimateEmptyScreen,
|
||||
EstimateDisableSwitch,
|
||||
CreateEstimateModal,
|
||||
EstimateList,
|
||||
} from "@/components/estimates";
|
||||
// hooks
|
||||
import { useProject, useProjectEstimates } from "@/hooks/store";
|
||||
import { useProjectEstimates } from "@/hooks/store";
|
||||
|
||||
type TEstimateRoot = {
|
||||
workspaceSlug: string;
|
||||
projectId: string;
|
||||
isAdmin: boolean;
|
||||
};
|
||||
|
||||
export const EstimateRoot: FC<TEstimateRoot> = (props) => {
|
||||
const { workspaceSlug, projectId } = props;
|
||||
export const EstimateRoot: FC<TEstimateRoot> = observer((props) => {
|
||||
const { workspaceSlug, projectId, isAdmin } = props;
|
||||
// hooks
|
||||
const { updateProject, currentProjectDetails } = useProject();
|
||||
const { loader, projectEstimateIds, estimateById, getProjectEstimates } = useProjectEstimates();
|
||||
const { loader, currentActiveEstimateId, estimateById, archivedEstimateIds, getProjectEstimates } =
|
||||
useProjectEstimates();
|
||||
// states
|
||||
const [isEstimateCreateModalOpen, setIsEstimateCreateModalOpen] = useState(false);
|
||||
// const [isEstimateDeleteModalOpen, setIsEstimateDeleteModalOpen] = useState<string | null>(null);
|
||||
@ -31,96 +34,70 @@ export const EstimateRoot: FC<TEstimateRoot> = (props) => {
|
||||
async () => workspaceSlug && projectId && getProjectEstimates(workspaceSlug, projectId)
|
||||
);
|
||||
|
||||
// const editEstimate = (estimate: IEstimate) => {
|
||||
// setIsEstimateCreateModalOpen(true);
|
||||
// console.log("estimate", estimate);
|
||||
// // Order the points array by key before updating the estimate to update state
|
||||
// // setEstimateToUpdate({
|
||||
// // ...estimate,
|
||||
// // points: orderArrayBy(estimate.points, "key"),
|
||||
// // });
|
||||
// };
|
||||
|
||||
const disableEstimates = () => {
|
||||
if (!workspaceSlug || !projectId) return;
|
||||
|
||||
updateProject(workspaceSlug.toString(), projectId.toString(), { estimate: null }).catch((err) => {
|
||||
const error = err?.error;
|
||||
const errorString = Array.isArray(error) ? error[0] : error;
|
||||
|
||||
setToast({
|
||||
type: TOAST_TYPE.ERROR,
|
||||
title: "Error!",
|
||||
message: errorString ?? "Estimate could not be disabled. Please try again",
|
||||
});
|
||||
});
|
||||
const onEditClick = (estimateId: string) => {
|
||||
const currentEstimate = estimateById(estimateId);
|
||||
setEstimateToUpdate(currentEstimate);
|
||||
setIsEstimateCreateModalOpen(true);
|
||||
};
|
||||
|
||||
return (
|
||||
<div className="container mx-auto">
|
||||
<EstimateLoaderScreen />
|
||||
<EstimateEmptyScreen onButtonClick={() => {}} />
|
||||
{/* <EstimateLoaderScreen />
|
||||
<EstimateEmptyScreen onButtonClick={() => {}} /> */}
|
||||
|
||||
{loader === "init-loader" || isSWRLoading ? (
|
||||
<Loader className="mt-5 space-y-5">
|
||||
<Loader.Item height="40px" />
|
||||
<Loader.Item height="40px" />
|
||||
<Loader.Item height="40px" />
|
||||
<Loader.Item height="40px" />
|
||||
</Loader>
|
||||
<EstimateLoaderScreen />
|
||||
) : (
|
||||
<>
|
||||
{/* header section */}
|
||||
<section className="flex items-center justify-between border-b border-custom-border-100 py-3.5">
|
||||
<h3 className="text-xl font-medium">Estimates</h3>
|
||||
<div className="col-span-12 space-y-5 sm:col-span-7">
|
||||
<div className="flex items-center gap-2">
|
||||
<Button
|
||||
variant="primary"
|
||||
onClick={() => {
|
||||
setIsEstimateCreateModalOpen(true);
|
||||
setEstimateToUpdate(undefined);
|
||||
}}
|
||||
size="sm"
|
||||
>
|
||||
Add Estimate
|
||||
</Button>
|
||||
{currentProjectDetails?.estimate && (
|
||||
<Button variant="neutral-primary" onClick={disableEstimates} size="sm">
|
||||
Disable Estimates
|
||||
</Button>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
</section>
|
||||
<div className="space-y-10">
|
||||
{/* header */}
|
||||
<div className="text-xl font-medium text-custom-text-100 border-b border-custom-border-200 py-3.5">
|
||||
Estimates
|
||||
</div>
|
||||
|
||||
{/* listing of estimates */}
|
||||
{!projectEstimateIds || (projectEstimateIds && projectEstimateIds.length <= 0) ? (
|
||||
<div className="h-full w-full py-8">
|
||||
<EmptyState type={EmptyStateType.PROJECT_SETTINGS_ESTIMATE} />
|
||||
{currentActiveEstimateId ? (
|
||||
<div className="space-y-4">
|
||||
{/* estimates activated deactivated section */}
|
||||
<div className="relative border-b border-custom-border-200 pb-4 flex justify-between items-center gap-3">
|
||||
<div className="space-y-1">
|
||||
<h3 className="text-lg font-medium text-custom-text-100">Enable estimates for my project</h3>
|
||||
<p className="text-sm text-custom-text-200">
|
||||
They help you in communicating complexity and workload of the team.
|
||||
</p>
|
||||
</div>
|
||||
<EstimateDisableSwitch workspaceSlug={workspaceSlug} projectId={projectId} isAdmin={isAdmin} />
|
||||
</div>
|
||||
{/* active estimates section */}
|
||||
<EstimateList
|
||||
estimateIds={[currentActiveEstimateId]}
|
||||
isAdmin={isAdmin}
|
||||
isEditable
|
||||
onEditClick={onEditClick}
|
||||
/>
|
||||
</div>
|
||||
) : (
|
||||
<section className="h-full overflow-y-auto bg-custom-background-100">
|
||||
{projectEstimateIds &&
|
||||
projectEstimateIds.map((estimateId: string) => {
|
||||
const estimate = estimateById(estimateId);
|
||||
if (!estimate) return <></>;
|
||||
return (
|
||||
<></>
|
||||
// <EstimateListItem
|
||||
// key={estimateId}
|
||||
// estimate={estimate}
|
||||
// editEstimate={(estimate) => editEstimate(estimate)}
|
||||
// deleteEstimate={(estimateId) => setIsEstimateDeleteModalOpen(estimateId)}
|
||||
// />
|
||||
);
|
||||
})}
|
||||
</section>
|
||||
<EstimateEmptyScreen onButtonClick={() => setIsEstimateCreateModalOpen(true)} />
|
||||
)}
|
||||
</>
|
||||
|
||||
{/* archived estimates section */}
|
||||
{archivedEstimateIds && archivedEstimateIds.length > 0 && (
|
||||
<div>
|
||||
<div className="border-b border-custom-border-200 space-y-1 pb-4">
|
||||
<h3 className="text-lg font-medium text-custom-text-100">Archived estimates</h3>
|
||||
<p className="text-sm text-custom-text-200">
|
||||
Estimates have gone through a change, these are the estimates you had in your older versions which
|
||||
were not in use. Read more about them
|
||||
<a href={"#"} target="_blank" className="text-custom-primary-100/80 hover:text-custom-primary-100">
|
||||
here.
|
||||
</a>
|
||||
</p>
|
||||
</div>
|
||||
<EstimateList estimateIds={archivedEstimateIds} isAdmin={isAdmin} />
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
)}
|
||||
|
||||
{/* */}
|
||||
{/* modals for create and update */}
|
||||
<CreateEstimateModal
|
||||
workspaceSlug={workspaceSlug}
|
||||
projectId={projectId}
|
||||
@ -131,7 +108,6 @@ export const EstimateRoot: FC<TEstimateRoot> = (props) => {
|
||||
setEstimateToUpdate(undefined);
|
||||
}}
|
||||
/>
|
||||
|
||||
{/* <DeleteEstimateModal
|
||||
isOpen={!!isEstimateDeleteModalOpen}
|
||||
handleClose={() => setIsEstimateDeleteModalOpen(null)}
|
||||
@ -139,4 +115,4 @@ export const EstimateRoot: FC<TEstimateRoot> = (props) => {
|
||||
/> */}
|
||||
</div>
|
||||
);
|
||||
};
|
||||
});
|
||||
|
@ -31,8 +31,8 @@ const EstimatesSettingsPage: NextPageWithLayout = observer(() => {
|
||||
return (
|
||||
<>
|
||||
<PageHead title={pageTitle} />
|
||||
<div className={`w-full overflow-y-auto py-8 pr-9 ${isAdmin ? "" : "pointer-events-none opacity-60"}`}>
|
||||
<EstimateRoot workspaceSlug={workspaceSlug?.toString()} projectId={projectId?.toString()} />
|
||||
<div className={`w-full overflow-y-auto py-8 pr-9 ${isAdmin ? "" : "pointer-events-none opacity-60"}`}>
|
||||
<EstimateRoot workspaceSlug={workspaceSlug?.toString()} projectId={projectId?.toString()} isAdmin={isAdmin} />
|
||||
</div>
|
||||
</>
|
||||
);
|
||||
|
19
web/public/empty-state/estimates/dark.svg
Normal file
19
web/public/empty-state/estimates/dark.svg
Normal file
@ -0,0 +1,19 @@
|
||||
<svg width="98" height="121" viewBox="0 0 98 121" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||
<path d="M14.8707 74.5375C16.3076 72.9545 18.6808 73.4171 20.2598 74.5344C22.2919 75.9722 22.873 78.598 23.1728 80.9255C23.6803 84.8657 23.8304 90.4355 28.2986 92.0828C30.363 92.8439 32.7261 92.2154 34.4697 90.9796C36.3742 89.6297 37.5622 87.5534 38.3579 85.4014C39.3213 82.796 39.879 80.0089 40.3517 77.2785C40.8351 74.4863 41.1519 71.664 41.2941 68.8338C41.4333 66.0649 41.3427 63.3142 41.2327 60.5471C41.1467 58.3866 41.1139 56.1179 41.8437 54.052C42.5808 51.9655 44.2945 50.3785 46.5478 50.1692C48.7349 49.966 50.9875 50.8883 52.57 52.3768C56.3668 55.9484 56.5454 61.7791 57.3941 66.5679C57.8399 69.084 58.4917 71.63 59.8579 73.819C60.9816 75.6193 62.5432 77.1286 64.3419 78.2484C67.9699 80.507 72.5929 81.1661 76.6187 79.6144C80.5491 78.0995 83.713 74.7878 85.082 70.8073C85.4201 69.8243 85.6289 68.8088 85.7348 67.7756C85.7841 67.2952 85.0326 67.2986 84.9838 67.7756C84.5336 72.1666 81.715 76.1433 77.8526 78.2259C73.8783 80.3689 69.0654 80.097 65.1824 77.8754C63.2735 76.7832 61.6129 75.2427 60.4591 73.364C59.0598 71.0857 58.4606 68.4301 58.0247 65.8227C57.2373 61.1135 56.8802 55.7066 53.3508 52.0908C50.6136 49.2866 45.5767 48.1613 42.635 51.2848C39.6059 54.501 40.5139 59.682 40.6106 63.6745C40.7523 69.5217 40.1494 75.3825 38.8131 81.0769C38.2314 83.5555 37.5623 86.1115 36.1024 88.2369C34.7681 90.1793 32.5875 91.6848 30.1594 91.6609C27.7378 91.637 26.0364 89.8572 25.1768 87.7354C24.2698 85.4969 24.2104 83.0793 23.8965 80.7163C23.6084 78.5472 23.0835 76.1467 21.4568 74.5592C20.1918 73.3248 18.3152 72.5301 16.5348 72.814C15.6794 72.9505 14.9208 73.3661 14.3396 74.0064C14.0153 74.3636 14.5449 74.8963 14.8707 74.5375Z" fill="#3F76FF"/>
|
||||
<path d="M55.4847 43.7299H38.3499C37.9179 43.7299 37.5664 43.3784 37.5664 42.9464C37.5664 42.5144 37.9179 42.1631 38.3499 42.1631H55.4847C55.9167 42.1631 56.268 42.5144 56.268 42.9464C56.268 43.3784 55.9167 43.7299 55.4847 43.7299Z" fill="#363A3F"/>
|
||||
<path d="M81.9847 89.2953H64.8499C64.4179 89.2953 64.0664 88.9438 64.0664 88.5118C64.0664 88.0798 64.4179 87.7285 64.8499 87.7285H81.9847C82.4167 87.7285 82.768 88.0798 82.768 88.5118C82.768 88.9438 82.4167 89.2953 81.9847 89.2953Z" fill="#363A3F"/>
|
||||
<path d="M96.0179 120.34H78.8831C78.4511 120.34 78.0996 119.989 78.0996 119.557C78.0996 119.125 78.4511 118.773 78.8831 118.773H96.0179C96.4499 118.773 96.8012 119.125 96.8012 119.557C96.8012 119.989 96.4499 120.34 96.0179 120.34Z" fill="#363A3F"/>
|
||||
<path d="M17.9183 26.7064H0.78347C0.351487 26.7064 0 26.3549 0 25.923C0 25.491 0.351492 25.1396 0.78347 25.1396H17.9183C18.3503 25.1396 18.7016 25.491 18.7016 25.923C18.7016 26.3549 18.3503 26.7064 17.9183 26.7064Z" fill="#363A3F"/>
|
||||
<path d="M87.2283 111.247H9.56712C9.29059 111.247 9.06641 111.023 9.06641 110.746V36.6404C9.06641 36.3636 9.2906 36.1396 9.56712 36.1396C9.84364 36.1396 10.0678 36.3636 10.0678 36.6404V66.8777C10.0678 90.8292 29.4843 110.246 53.4358 110.246H87.2283C87.5048 110.246 87.729 110.47 87.729 110.746C87.729 111.023 87.5048 111.247 87.2283 111.247Z" fill="#363A3F"/>
|
||||
<path d="M25.9613 81.4652H8.82644C8.39446 81.4652 8.04297 81.1137 8.04297 80.6817C8.04297 80.2498 8.39446 79.8984 8.82644 79.8984H25.9613C26.3932 79.8984 26.7446 80.2498 26.7446 80.6817C26.7446 81.1137 26.3932 81.4652 25.9613 81.4652Z" fill="#363A3F"/>
|
||||
<path d="M14.8118 76.609C16.0562 76.609 17.065 75.6002 17.065 74.3558C17.065 73.1113 16.0562 72.1025 14.8118 72.1025C13.5674 72.1025 12.5586 73.1113 12.5586 74.3558C12.5586 75.6002 13.5674 76.609 14.8118 76.609Z" fill="#2E2E2E"/>
|
||||
<path d="M29.8431 94.1344C31.0875 94.1344 32.0963 93.1256 32.0963 91.8812C32.0963 90.6367 31.0875 89.6279 29.8431 89.6279C28.5986 89.6279 27.5898 90.6367 27.5898 91.8812C27.5898 93.1256 28.5986 94.1344 29.8431 94.1344Z" fill="#2E2E2E"/>
|
||||
<path d="M46.8587 52.0738C48.1031 52.0738 49.1119 51.065 49.1119 49.8206C49.1119 48.5762 48.1031 47.5674 46.8587 47.5674C45.6143 47.5674 44.6055 48.5762 44.6055 49.8206C44.6055 51.065 45.6143 52.0738 46.8587 52.0738Z" fill="#2E2E2E"/>
|
||||
<path d="M71.8997 82.1178C73.1441 82.1178 74.1529 81.109 74.1529 79.8646C74.1529 78.6201 73.1441 77.6113 71.8997 77.6113C70.6553 77.6113 69.6465 78.6201 69.6465 79.8646C69.6465 81.109 70.6553 82.1178 71.8997 82.1178Z" fill="#2E2E2E"/>
|
||||
<path d="M86.4368 70.1012C87.6812 70.1012 88.69 69.0924 88.69 67.8479C88.69 66.6035 87.6812 65.5947 86.4368 65.5947C85.1924 65.5947 84.1836 66.6035 84.1836 67.8479C84.1836 69.0924 85.1924 70.1012 86.4368 70.1012Z" fill="#2E2E2E"/>
|
||||
<path d="M82.1267 20.0406L86.2476 21.9926C84.5574 19.6339 83.101 15.95 82.3213 13.0186C81.0011 15.7492 78.873 19.0903 76.7664 21.0861L81.1217 19.9656C78.438 33.1184 68.3439 42.5608 56.7847 42.5608L56.6211 43.0359C68.6949 43.0359 79.3865 33.7244 82.1267 20.0406Z" fill="#2E2E2E"/>
|
||||
<path d="M69.6669 0H95.7042C96.8085 0 97.7071 0.8985 97.7071 2.00286C97.7071 3.10723 96.8085 4.00573 95.7042 4.00573H69.6669C68.5626 4.00573 67.6641 3.10723 67.6641 2.00286C67.6641 0.8985 68.5626 0 69.6669 0Z" fill="#3F76FF"/>
|
||||
<path d="M87.2735 115.346C86.999 115.346 86.722 115.285 86.4582 115.162C85.8076 114.856 85.399 114.245 85.3658 113.527L85.0959 107.706C85.0612 106.958 85.4514 106.278 86.1144 105.931C86.7772 105.584 87.5586 105.651 88.153 106.105L92.1177 109.137C92.5944 109.502 92.8668 110.055 92.8648 110.655C92.8631 111.255 92.5873 111.806 92.1084 112.168L88.4136 114.958C88.0726 115.215 87.6757 115.346 87.2735 115.346Z" fill="#363A3F"/>
|
||||
<path d="M14.2019 37.1274C14.2019 37.402 14.1407 37.679 14.017 37.9428C13.7114 38.5934 13.1002 39.0019 12.3824 39.0352L6.56105 39.3051C5.8134 39.3398 5.13372 38.9496 4.78654 38.2865C4.43937 37.6237 4.50684 36.8423 4.96062 36.248L7.99279 32.2833C8.35708 31.8066 8.9106 31.5342 9.51009 31.5361C10.1106 31.5379 10.6616 31.8136 11.0235 32.2926L13.8131 35.9873C14.0703 36.3284 14.2019 36.7252 14.2019 37.1274Z" fill="#363A3F"/>
|
||||
<path d="M46.6797 31.9697C43.7154 31.2596 40.5482 31.9023 38.0143 33.5762C35.3571 35.3315 33.5978 38.1422 33.0023 41.2561C31.7737 47.6804 35.5272 54.0982 41.4594 56.6551C47.6955 59.343 55.114 57.5394 59.6426 52.5507C61.8873 50.078 63.2901 46.9499 63.7125 43.6419C64.1286 40.383 63.654 36.6231 61.6807 33.9032C59.7234 31.2053 56.464 30.3 53.3163 29.8579C50.262 29.4289 46.2102 29.2424 44.5587 32.4752C44.2438 33.0916 44.0958 33.7558 44.1307 34.4471C44.155 34.9281 44.9062 34.931 44.8818 34.4471C44.8055 32.9356 45.8132 31.6726 47.1049 30.9933C48.6018 30.206 50.4074 30.2726 52.04 30.4481C54.9873 30.7649 58.3335 31.3301 60.459 33.5937C62.5762 35.8486 63.2109 39.2849 63.0795 42.2781C62.9423 45.4023 61.8658 48.4773 59.9796 50.976C56.2574 55.9067 49.7451 58.2525 43.754 56.6597C37.9165 55.1076 37.8402 51.6433 37.9365 45.5083C37.9879 42.2354 34.8963 37.0837 37.4205 34.9508C39.8919 32.8626 43.3172 31.9363 46.4801 32.6939C46.9499 32.8065 47.1503 32.0824 46.6797 31.9697Z" fill="#3F76FF"/>
|
||||
</svg>
|
After Width: | Height: | Size: 6.7 KiB |
19
web/public/empty-state/estimates/light.svg
Normal file
19
web/public/empty-state/estimates/light.svg
Normal file
@ -0,0 +1,19 @@
|
||||
<svg width="98" height="121" viewBox="0 0 98 121" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||
<path d="M14.8707 74.5375C16.3076 72.9545 18.6808 73.4171 20.2598 74.5344C22.2919 75.9722 22.873 78.598 23.1728 80.9255C23.6803 84.8657 23.8304 90.4355 28.2986 92.0828C30.363 92.8439 32.7261 92.2154 34.4697 90.9796C36.3742 89.6297 37.5622 87.5534 38.3579 85.4014C39.3213 82.796 39.879 80.0089 40.3517 77.2785C40.8351 74.4863 41.1519 71.664 41.2941 68.8338C41.4333 66.0649 41.3427 63.3142 41.2327 60.5471C41.1467 58.3866 41.1139 56.1179 41.8437 54.052C42.5808 51.9655 44.2945 50.3785 46.5478 50.1692C48.7349 49.966 50.9875 50.8883 52.57 52.3768C56.3668 55.9484 56.5454 61.7791 57.3941 66.5679C57.8399 69.084 58.4917 71.63 59.8579 73.819C60.9816 75.6193 62.5432 77.1286 64.3419 78.2484C67.9699 80.507 72.5929 81.1661 76.6187 79.6144C80.5491 78.0995 83.713 74.7878 85.082 70.8073C85.4201 69.8243 85.6289 68.8088 85.7348 67.7756C85.7841 67.2952 85.0326 67.2986 84.9838 67.7756C84.5336 72.1666 81.715 76.1433 77.8526 78.2259C73.8783 80.3689 69.0654 80.097 65.1824 77.8754C63.2735 76.7832 61.6129 75.2427 60.4591 73.364C59.0598 71.0857 58.4606 68.4301 58.0247 65.8227C57.2373 61.1135 56.8802 55.7066 53.3508 52.0908C50.6136 49.2866 45.5767 48.1613 42.635 51.2848C39.6059 54.501 40.5139 59.682 40.6106 63.6745C40.7523 69.5217 40.1494 75.3825 38.8131 81.0769C38.2314 83.5555 37.5623 86.1115 36.1024 88.2369C34.7681 90.1793 32.5875 91.6848 30.1594 91.6609C27.7378 91.637 26.0364 89.8572 25.1768 87.7354C24.2698 85.4969 24.2104 83.0793 23.8965 80.7163C23.6084 78.5472 23.0835 76.1467 21.4568 74.5592C20.1918 73.3248 18.3152 72.5301 16.5348 72.814C15.6794 72.9505 14.9208 73.3661 14.3396 74.0064C14.0153 74.3636 14.5449 74.8963 14.8707 74.5375Z" fill="#3F76FF"/>
|
||||
<path d="M55.4847 43.7299H38.3499C37.9179 43.7299 37.5664 43.3784 37.5664 42.9464C37.5664 42.5144 37.9179 42.1631 38.3499 42.1631H55.4847C55.9167 42.1631 56.268 42.5144 56.268 42.9464C56.268 43.3784 55.9167 43.7299 55.4847 43.7299Z" fill="#D9D9E0"/>
|
||||
<path d="M81.9847 89.2953H64.8499C64.4179 89.2953 64.0664 88.9438 64.0664 88.5118C64.0664 88.0798 64.4179 87.7285 64.8499 87.7285H81.9847C82.4167 87.7285 82.768 88.0798 82.768 88.5118C82.768 88.9438 82.4167 89.2953 81.9847 89.2953Z" fill="#D9D9E0"/>
|
||||
<path d="M96.0179 120.34H78.8831C78.4511 120.34 78.0996 119.989 78.0996 119.557C78.0996 119.125 78.4511 118.773 78.8831 118.773H96.0179C96.4499 118.773 96.8012 119.125 96.8012 119.557C96.8012 119.989 96.4499 120.34 96.0179 120.34Z" fill="#D9D9E0"/>
|
||||
<path d="M17.9183 26.7064H0.78347C0.351487 26.7064 0 26.3549 0 25.923C0 25.491 0.351492 25.1396 0.78347 25.1396H17.9183C18.3503 25.1396 18.7016 25.491 18.7016 25.923C18.7016 26.3549 18.3503 26.7064 17.9183 26.7064Z" fill="#D9D9E0"/>
|
||||
<path d="M87.2283 111.247H9.56712C9.29059 111.247 9.06641 111.023 9.06641 110.746V36.6404C9.06641 36.3636 9.2906 36.1396 9.56712 36.1396C9.84364 36.1396 10.0678 36.3636 10.0678 36.6404V66.8777C10.0678 90.8292 29.4843 110.246 53.4358 110.246H87.2283C87.5048 110.246 87.729 110.47 87.729 110.746C87.729 111.023 87.5048 111.247 87.2283 111.247Z" fill="#D9D9E0"/>
|
||||
<path d="M25.9613 81.4652H8.82644C8.39446 81.4652 8.04297 81.1137 8.04297 80.6817C8.04297 80.2498 8.39446 79.8984 8.82644 79.8984H25.9613C26.3932 79.8984 26.7446 80.2498 26.7446 80.6817C26.7446 81.1137 26.3932 81.4652 25.9613 81.4652Z" fill="#D9D9E0"/>
|
||||
<path d="M14.8118 76.609C16.0562 76.609 17.065 75.6002 17.065 74.3558C17.065 73.1113 16.0562 72.1025 14.8118 72.1025C13.5674 72.1025 12.5586 73.1113 12.5586 74.3558C12.5586 75.6002 13.5674 76.609 14.8118 76.609Z" fill="#2E2E2E"/>
|
||||
<path d="M29.8431 94.1344C31.0875 94.1344 32.0963 93.1256 32.0963 91.8812C32.0963 90.6367 31.0875 89.6279 29.8431 89.6279C28.5986 89.6279 27.5898 90.6367 27.5898 91.8812C27.5898 93.1256 28.5986 94.1344 29.8431 94.1344Z" fill="#2E2E2E"/>
|
||||
<path d="M46.8587 52.0738C48.1031 52.0738 49.1119 51.065 49.1119 49.8206C49.1119 48.5762 48.1031 47.5674 46.8587 47.5674C45.6143 47.5674 44.6055 48.5762 44.6055 49.8206C44.6055 51.065 45.6143 52.0738 46.8587 52.0738Z" fill="#2E2E2E"/>
|
||||
<path d="M71.8997 82.1178C73.1441 82.1178 74.1529 81.109 74.1529 79.8646C74.1529 78.6201 73.1441 77.6113 71.8997 77.6113C70.6553 77.6113 69.6465 78.6201 69.6465 79.8646C69.6465 81.109 70.6553 82.1178 71.8997 82.1178Z" fill="#2E2E2E"/>
|
||||
<path d="M86.4368 70.1012C87.6812 70.1012 88.69 69.0924 88.69 67.8479C88.69 66.6035 87.6812 65.5947 86.4368 65.5947C85.1924 65.5947 84.1836 66.6035 84.1836 67.8479C84.1836 69.0924 85.1924 70.1012 86.4368 70.1012Z" fill="#2E2E2E"/>
|
||||
<path d="M82.1267 20.0406L86.2476 21.9926C84.5574 19.6339 83.101 15.95 82.3213 13.0186C81.0011 15.7492 78.873 19.0903 76.7664 21.0861L81.1217 19.9656C78.438 33.1184 68.3439 42.5608 56.7847 42.5608L56.6211 43.0359C68.6949 43.0359 79.3865 33.7244 82.1267 20.0406Z" fill="#2E2E2E"/>
|
||||
<path d="M69.6669 0H95.7042C96.8085 0 97.7071 0.8985 97.7071 2.00286C97.7071 3.10723 96.8085 4.00573 95.7042 4.00573H69.6669C68.5626 4.00573 67.6641 3.10723 67.6641 2.00286C67.6641 0.8985 68.5626 0 69.6669 0Z" fill="#3F76FF"/>
|
||||
<path d="M87.2735 115.346C86.999 115.346 86.722 115.285 86.4582 115.162C85.8076 114.856 85.399 114.245 85.3658 113.527L85.0959 107.706C85.0612 106.958 85.4514 106.278 86.1144 105.931C86.7772 105.584 87.5586 105.651 88.153 106.105L92.1177 109.137C92.5944 109.502 92.8668 110.055 92.8648 110.655C92.8631 111.255 92.5873 111.806 92.1084 112.168L88.4136 114.958C88.0726 115.215 87.6757 115.346 87.2735 115.346Z" fill="#D9D9E0"/>
|
||||
<path d="M14.2019 37.1274C14.2019 37.402 14.1407 37.679 14.017 37.9428C13.7114 38.5934 13.1002 39.0019 12.3824 39.0352L6.56105 39.3051C5.8134 39.3398 5.13372 38.9496 4.78654 38.2865C4.43937 37.6237 4.50684 36.8423 4.96062 36.248L7.99279 32.2833C8.35708 31.8066 8.9106 31.5342 9.51009 31.5361C10.1106 31.5379 10.6616 31.8136 11.0235 32.2926L13.8131 35.9873C14.0703 36.3284 14.2019 36.7252 14.2019 37.1274Z" fill="#D9D9E0"/>
|
||||
<path d="M46.6797 31.9697C43.7154 31.2596 40.5482 31.9023 38.0143 33.5762C35.3571 35.3315 33.5978 38.1422 33.0023 41.2561C31.7737 47.6804 35.5272 54.0982 41.4594 56.6551C47.6955 59.343 55.114 57.5394 59.6426 52.5507C61.8873 50.078 63.2901 46.9499 63.7125 43.6419C64.1286 40.383 63.654 36.6231 61.6807 33.9032C59.7234 31.2053 56.464 30.3 53.3163 29.8579C50.262 29.4289 46.2102 29.2424 44.5587 32.4752C44.2438 33.0916 44.0958 33.7558 44.1307 34.4471C44.155 34.9281 44.9062 34.931 44.8818 34.4471C44.8055 32.9356 45.8132 31.6726 47.1049 30.9933C48.6018 30.206 50.4074 30.2726 52.04 30.4481C54.9873 30.7649 58.3335 31.3301 60.459 33.5937C62.5762 35.8486 63.2109 39.2849 63.0795 42.2781C62.9423 45.4023 61.8658 48.4773 59.9796 50.976C56.2574 55.9067 49.7451 58.2525 43.754 56.6597C37.9165 55.1076 37.8402 51.6433 37.9365 45.5083C37.9879 42.2354 34.8963 37.0837 37.4205 34.9508C39.8919 32.8626 43.3172 31.9363 46.4801 32.6939C46.9499 32.8065 47.1503 32.0824 46.6797 31.9697Z" fill="#3F76FF"/>
|
||||
</svg>
|
After Width: | Height: | Size: 6.7 KiB |
@ -21,8 +21,9 @@ export interface IProjectEstimateStore {
|
||||
estimates: Record<string, IEstimate>;
|
||||
error: TErrorCodes | undefined;
|
||||
// computed
|
||||
currentActiveEstimateId: string | undefined;
|
||||
archivedEstimateIds: string[] | undefined;
|
||||
areEstimateEnabledByProjectId: (projectId: string) => boolean;
|
||||
projectEstimateIds: string[] | undefined;
|
||||
estimateIdsByProjectId: (projectId: string) => string[] | undefined;
|
||||
estimateById: (estimateId: string) => IEstimate | undefined;
|
||||
// actions
|
||||
@ -55,6 +56,8 @@ export class ProjectEstimateStore implements IProjectEstimateStore {
|
||||
estimates: observable,
|
||||
error: observable,
|
||||
// computed
|
||||
currentActiveEstimateId: computed,
|
||||
archivedEstimateIds: computed,
|
||||
projectEstimateIds: computed,
|
||||
// actions
|
||||
getWorkspaceEstimates: action,
|
||||
@ -67,6 +70,45 @@ export class ProjectEstimateStore implements IProjectEstimateStore {
|
||||
}
|
||||
|
||||
// computed
|
||||
|
||||
/**
|
||||
* @description get current active estimate id for a project
|
||||
* @returns { string | undefined }
|
||||
*/
|
||||
get currentActiveEstimateId(): string | undefined {
|
||||
const { projectId } = this.store.router;
|
||||
if (!projectId) return undefined;
|
||||
const projectDetails = this.store.projectRoot.project.getProjectById(projectId);
|
||||
if (!projectDetails) return undefined;
|
||||
return projectDetails.estimate ?? undefined;
|
||||
}
|
||||
|
||||
/**
|
||||
* @description get all archived estimate ids for a project
|
||||
* @returns { string[] | undefined }
|
||||
*/
|
||||
get archivedEstimateIds(): string[] | undefined {
|
||||
const { projectId } = this.store.router;
|
||||
if (!projectId) return undefined;
|
||||
const archivedEstimateIds = Object.values(this.estimates || {})
|
||||
.filter((p) => p.project === projectId && p.id !== this.currentActiveEstimateId)
|
||||
.map((p) => p.id) as string[];
|
||||
return archivedEstimateIds ?? undefined;
|
||||
}
|
||||
|
||||
/**
|
||||
* @description get all estimate ids for a project
|
||||
* @returns { string[] | undefined }
|
||||
*/
|
||||
get projectEstimateIds(): string[] | undefined {
|
||||
const { projectId } = this.store.router;
|
||||
if (!projectId) return undefined;
|
||||
const projectEstimatesIds = Object.values(this.estimates || {})
|
||||
.filter((p) => p.project === projectId)
|
||||
.map((p) => p.id) as string[];
|
||||
return projectEstimatesIds ?? undefined;
|
||||
}
|
||||
|
||||
/**
|
||||
* @description get estimates are enabled in the project or not
|
||||
* @returns { boolean }
|
||||
@ -78,19 +120,6 @@ export class ProjectEstimateStore implements IProjectEstimateStore {
|
||||
return Boolean(projectDetails.estimate) || false;
|
||||
});
|
||||
|
||||
/**
|
||||
* @description get all estimate ids for a project
|
||||
* @returns { string[] | undefined }
|
||||
*/
|
||||
get projectEstimateIds(): string[] | undefined {
|
||||
const { projectId } = this.store.router;
|
||||
if (!projectId) return undefined;
|
||||
const projectEstimatesIds = Object.values(this.estimates || {})
|
||||
.filter((p) => p.project === projectId)
|
||||
.map((p) => p.id && p != undefined) as string[];
|
||||
return projectEstimatesIds ?? undefined;
|
||||
}
|
||||
|
||||
/**
|
||||
* @description get all estimate ids for a project
|
||||
* @returns { string[] | undefined }
|
||||
@ -99,7 +128,7 @@ export class ProjectEstimateStore implements IProjectEstimateStore {
|
||||
if (!projectId) return undefined;
|
||||
const projectEstimatesIds = Object.values(this.estimates || {})
|
||||
.filter((p) => p.project === projectId)
|
||||
.map((p) => p.id && p != undefined) as string[];
|
||||
.map((p) => p.id) as string[];
|
||||
return projectEstimatesIds ?? undefined;
|
||||
});
|
||||
|
||||
@ -223,11 +252,20 @@ export class ProjectEstimateStore implements IProjectEstimateStore {
|
||||
this.error = undefined;
|
||||
|
||||
const estimate = await this.service.createEstimate(workspaceSlug, projectId, payload);
|
||||
if (estimate) {
|
||||
runInAction(() => {
|
||||
if (estimate.id) set(this.estimates, [estimate.id], new Estimate(this.store, estimate));
|
||||
// FIXME: i am getting different response from the server and once backend changes remove the get request and uncomment the commented code
|
||||
const estimates = await this.getProjectEstimates(workspaceSlug, projectId, "mutation-loader");
|
||||
if (estimates && estimates.length > 0)
|
||||
await this.store.projectRoot.project.updateProject(workspaceSlug, projectId, {
|
||||
estimate: estimates[estimates.length - 1].id,
|
||||
});
|
||||
}
|
||||
// if (estimate) {
|
||||
// await this.store.projectRoot.project.updateProject(workspaceSlug, projectId, {
|
||||
// estimate: estimate.id,
|
||||
// });
|
||||
// runInAction(() => {
|
||||
// if (estimate.id) set(this.estimates, [estimate.id], new Estimate(this.store, estimate));
|
||||
// });
|
||||
// }
|
||||
|
||||
return estimate;
|
||||
} catch (error) {
|
||||
|
Loading…
Reference in New Issue
Block a user