2024-05-23 08:11:30 +00:00
|
|
|
import set from "lodash/set";
|
2024-05-27 04:17:31 +00:00
|
|
|
import sortBy from "lodash/sortBy";
|
2024-05-23 08:11:30 +00:00
|
|
|
import update from "lodash/update";
|
2024-05-24 09:37:44 +00:00
|
|
|
import { action, computed, makeObservable, observable, runInAction } from "mobx";
|
2024-05-23 08:11:30 +00:00
|
|
|
import { computedFn } from "mobx-utils";
|
2024-05-24 09:37:44 +00:00
|
|
|
import { IEstimate as IEstimateType, IEstimateFormData } from "@plane/types";
|
2024-05-23 08:11:30 +00:00
|
|
|
// services
|
|
|
|
import { EstimateService } from "@/services/project/estimate.service";
|
|
|
|
// store
|
|
|
|
import { IEstimate, Estimate } from "@/store/estimates/estimate";
|
|
|
|
import { RootStore } from "@/store/root.store";
|
|
|
|
|
|
|
|
type TEstimateLoader = "init-loader" | "mutation-loader" | undefined;
|
|
|
|
type TErrorCodes = {
|
|
|
|
status: string;
|
|
|
|
message?: string;
|
|
|
|
};
|
|
|
|
|
|
|
|
export interface IProjectEstimateStore {
|
|
|
|
// observables
|
|
|
|
loader: TEstimateLoader;
|
|
|
|
estimates: Record<string, IEstimate>;
|
|
|
|
error: TErrorCodes | undefined;
|
|
|
|
// computed
|
2024-05-24 11:54:50 +00:00
|
|
|
currentActiveEstimateId: string | undefined;
|
|
|
|
archivedEstimateIds: string[] | undefined;
|
2024-05-24 09:37:44 +00:00
|
|
|
areEstimateEnabledByProjectId: (projectId: string) => boolean;
|
|
|
|
estimateIdsByProjectId: (projectId: string) => string[] | undefined;
|
2024-05-23 08:11:30 +00:00
|
|
|
estimateById: (estimateId: string) => IEstimate | undefined;
|
|
|
|
// actions
|
2024-05-24 09:37:44 +00:00
|
|
|
getWorkspaceEstimates: (workspaceSlug: string, loader?: TEstimateLoader) => Promise<IEstimateType[] | undefined>;
|
|
|
|
getProjectEstimates: (
|
|
|
|
workspaceSlug: string,
|
|
|
|
projectId: string,
|
|
|
|
loader?: TEstimateLoader
|
|
|
|
) => Promise<IEstimateType[] | undefined>;
|
|
|
|
getEstimateById: (workspaceSlug: string, projectId: string, estimateId: string) => Promise<IEstimateType | undefined>;
|
|
|
|
createEstimate: (
|
|
|
|
workspaceSlug: string,
|
|
|
|
projectId: string,
|
|
|
|
data: IEstimateFormData
|
|
|
|
) => Promise<IEstimateType | undefined>;
|
2024-05-23 08:11:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export class ProjectEstimateStore implements IProjectEstimateStore {
|
|
|
|
// observables
|
|
|
|
loader: TEstimateLoader = undefined;
|
|
|
|
estimates: Record<string, IEstimate> = {}; // estimate_id -> estimate
|
|
|
|
error: TErrorCodes | undefined = undefined;
|
|
|
|
// service
|
|
|
|
service: EstimateService;
|
|
|
|
|
|
|
|
constructor(private store: RootStore) {
|
|
|
|
makeObservable(this, {
|
|
|
|
// observables
|
|
|
|
loader: observable.ref,
|
|
|
|
estimates: observable,
|
|
|
|
error: observable,
|
|
|
|
// computed
|
2024-05-24 11:54:50 +00:00
|
|
|
currentActiveEstimateId: computed,
|
|
|
|
archivedEstimateIds: computed,
|
2024-05-23 08:11:30 +00:00
|
|
|
projectEstimateIds: computed,
|
|
|
|
// actions
|
2024-05-24 09:37:44 +00:00
|
|
|
getWorkspaceEstimates: action,
|
|
|
|
getProjectEstimates: action,
|
2024-05-23 08:11:30 +00:00
|
|
|
getEstimateById: action,
|
|
|
|
createEstimate: action,
|
|
|
|
});
|
|
|
|
// service
|
|
|
|
this.service = new EstimateService();
|
|
|
|
}
|
|
|
|
|
|
|
|
// computed
|
2024-05-24 11:54:50 +00:00
|
|
|
|
2024-05-24 09:37:44 +00:00
|
|
|
/**
|
2024-05-24 11:54:50 +00:00
|
|
|
* @description get current active estimate id for a project
|
|
|
|
* @returns { string | undefined }
|
2024-05-24 09:37:44 +00:00
|
|
|
*/
|
2024-05-24 11:54:50 +00:00
|
|
|
get currentActiveEstimateId(): string | undefined {
|
|
|
|
const { projectId } = this.store.router;
|
|
|
|
if (!projectId) return undefined;
|
2024-05-24 09:37:44 +00:00
|
|
|
const projectDetails = this.store.projectRoot.project.getProjectById(projectId);
|
2024-05-24 11:54:50 +00:00
|
|
|
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;
|
|
|
|
}
|
2024-05-24 09:37:44 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @description get all estimate ids for a project
|
|
|
|
* @returns { string[] | undefined }
|
|
|
|
*/
|
|
|
|
get projectEstimateIds(): string[] | undefined {
|
2024-05-23 08:11:30 +00:00
|
|
|
const { projectId } = this.store.router;
|
|
|
|
if (!projectId) return undefined;
|
|
|
|
const projectEstimatesIds = Object.values(this.estimates || {})
|
|
|
|
.filter((p) => p.project === projectId)
|
2024-05-24 11:54:50 +00:00
|
|
|
.map((p) => p.id) as string[];
|
2024-05-23 08:11:30 +00:00
|
|
|
return projectEstimatesIds ?? undefined;
|
|
|
|
}
|
|
|
|
|
2024-05-24 11:54:50 +00:00
|
|
|
/**
|
|
|
|
* @description get estimates are enabled in the project or not
|
|
|
|
* @returns { boolean }
|
|
|
|
*/
|
|
|
|
areEstimateEnabledByProjectId = computedFn((projectId: string) => {
|
|
|
|
if (!projectId) return false;
|
|
|
|
const projectDetails = this.store.projectRoot.project.getProjectById(projectId);
|
|
|
|
if (!projectDetails) return false;
|
|
|
|
return Boolean(projectDetails.estimate) || false;
|
|
|
|
});
|
|
|
|
|
2024-05-24 09:37:44 +00:00
|
|
|
/**
|
|
|
|
* @description get all estimate ids for a project
|
|
|
|
* @returns { string[] | undefined }
|
|
|
|
*/
|
|
|
|
estimateIdsByProjectId = computedFn((projectId: string) => {
|
|
|
|
if (!projectId) return undefined;
|
|
|
|
const projectEstimatesIds = Object.values(this.estimates || {})
|
|
|
|
.filter((p) => p.project === projectId)
|
2024-05-24 11:54:50 +00:00
|
|
|
.map((p) => p.id) as string[];
|
2024-05-24 09:37:44 +00:00
|
|
|
return projectEstimatesIds ?? undefined;
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @description get estimate by id
|
|
|
|
* @returns { IEstimate | undefined }
|
|
|
|
*/
|
2024-05-23 08:11:30 +00:00
|
|
|
estimateById = computedFn((estimateId: string) => {
|
|
|
|
if (!estimateId) return undefined;
|
|
|
|
return this.estimates[estimateId] ?? undefined;
|
|
|
|
});
|
|
|
|
|
|
|
|
// actions
|
|
|
|
/**
|
2024-05-24 09:37:44 +00:00
|
|
|
* @description fetch all estimates for a workspace
|
2024-05-23 08:11:30 +00:00
|
|
|
* @returns { IEstimateType[] | undefined }
|
|
|
|
*/
|
2024-05-24 09:37:44 +00:00
|
|
|
getWorkspaceEstimates = async (
|
|
|
|
workspaceSlug: string,
|
|
|
|
loader: TEstimateLoader = "mutation-loader"
|
|
|
|
): Promise<IEstimateType[] | undefined> => {
|
2024-05-23 08:11:30 +00:00
|
|
|
try {
|
|
|
|
this.error = undefined;
|
2024-05-24 09:37:44 +00:00
|
|
|
if (!this.projectEstimateIds) this.loader = loader ? loader : "init-loader";
|
|
|
|
|
|
|
|
const estimates = await this.service.fetchWorkspaceEstimates(workspaceSlug);
|
|
|
|
if (estimates && estimates.length > 0) {
|
|
|
|
runInAction(() => {
|
|
|
|
estimates.forEach((estimate) => {
|
|
|
|
if (estimate.id) set(this.estimates, [estimate.id], new Estimate(this.store, estimate));
|
|
|
|
});
|
2024-05-23 08:11:30 +00:00
|
|
|
});
|
2024-05-24 09:37:44 +00:00
|
|
|
}
|
2024-05-23 08:11:30 +00:00
|
|
|
|
|
|
|
return estimates;
|
|
|
|
} catch (error) {
|
2024-05-24 09:37:44 +00:00
|
|
|
this.loader = undefined;
|
2024-05-23 08:11:30 +00:00
|
|
|
this.error = {
|
|
|
|
status: "error",
|
|
|
|
message: "Error fetching estimates",
|
|
|
|
};
|
|
|
|
}
|
2024-05-23 10:07:25 +00:00
|
|
|
};
|
2024-05-23 08:11:30 +00:00
|
|
|
|
2024-05-24 09:37:44 +00:00
|
|
|
/**
|
|
|
|
* @description fetch all estimates for a project
|
|
|
|
* @returns { IEstimateType[] | undefined }
|
|
|
|
*/
|
|
|
|
getProjectEstimates = async (
|
|
|
|
workspaceSlug: string,
|
|
|
|
projectId: string,
|
|
|
|
loader: TEstimateLoader = "mutation-loader"
|
|
|
|
): Promise<IEstimateType[] | undefined> => {
|
2024-05-23 08:11:30 +00:00
|
|
|
try {
|
|
|
|
this.error = undefined;
|
2024-05-24 09:37:44 +00:00
|
|
|
if (!this.projectEstimateIds) this.loader = loader ? loader : "init-loader";
|
|
|
|
|
|
|
|
const estimates = await this.service.fetchProjectEstimates(workspaceSlug, projectId);
|
|
|
|
if (estimates && estimates.length > 0) {
|
|
|
|
runInAction(() => {
|
|
|
|
estimates.forEach((estimate) => {
|
|
|
|
if (estimate.id) set(this.estimates, [estimate.id], new Estimate(this.store, estimate));
|
|
|
|
});
|
2024-05-23 08:11:30 +00:00
|
|
|
});
|
2024-05-24 09:37:44 +00:00
|
|
|
}
|
2024-05-23 08:11:30 +00:00
|
|
|
|
|
|
|
return estimates;
|
|
|
|
} catch (error) {
|
2024-05-24 09:37:44 +00:00
|
|
|
this.loader = undefined;
|
2024-05-23 08:11:30 +00:00
|
|
|
this.error = {
|
|
|
|
status: "error",
|
|
|
|
message: "Error fetching estimates",
|
|
|
|
};
|
|
|
|
}
|
2024-05-23 10:07:25 +00:00
|
|
|
};
|
2024-05-23 08:11:30 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @description update an estimate for a project
|
|
|
|
* @param { string } estimateId
|
|
|
|
* @returns IEstimateType | undefined
|
|
|
|
*/
|
2024-05-24 09:37:44 +00:00
|
|
|
getEstimateById = async (
|
|
|
|
workspaceSlug: string,
|
|
|
|
projectId: string,
|
|
|
|
estimateId: string
|
|
|
|
): Promise<IEstimateType | undefined> => {
|
2024-05-23 08:11:30 +00:00
|
|
|
try {
|
|
|
|
this.error = undefined;
|
|
|
|
|
2024-05-24 09:37:44 +00:00
|
|
|
const estimate = await this.service.fetchEstimateById(workspaceSlug, projectId, estimateId);
|
|
|
|
if (estimate) {
|
|
|
|
runInAction(() => {
|
|
|
|
if (estimate.id)
|
|
|
|
update(this.estimates, [estimate.id], (estimateStore) => {
|
|
|
|
if (estimateStore) estimateStore.updateEstimate(estimate);
|
|
|
|
else return new Estimate(this.store, estimate);
|
|
|
|
});
|
2024-05-23 08:11:30 +00:00
|
|
|
});
|
2024-05-24 09:37:44 +00:00
|
|
|
}
|
2024-05-23 08:11:30 +00:00
|
|
|
|
|
|
|
return estimate;
|
|
|
|
} catch (error) {
|
|
|
|
this.error = {
|
|
|
|
status: "error",
|
|
|
|
message: "Error fetching estimate by id",
|
|
|
|
};
|
|
|
|
}
|
2024-05-23 10:07:25 +00:00
|
|
|
};
|
2024-05-23 08:11:30 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @description create an estimate for a project
|
2024-05-24 09:37:44 +00:00
|
|
|
* @param { Partial<IEstimateType> } payload
|
2024-05-23 08:11:30 +00:00
|
|
|
* @returns
|
|
|
|
*/
|
2024-05-24 09:37:44 +00:00
|
|
|
createEstimate = async (
|
|
|
|
workspaceSlug: string,
|
|
|
|
projectId: string,
|
|
|
|
payload: IEstimateFormData
|
|
|
|
): Promise<IEstimateType | undefined> => {
|
2024-05-23 08:11:30 +00:00
|
|
|
try {
|
|
|
|
this.error = undefined;
|
|
|
|
|
2024-05-24 09:37:44 +00:00
|
|
|
const estimate = await this.service.createEstimate(workspaceSlug, projectId, payload);
|
2024-05-24 11:54:50 +00:00
|
|
|
// FIXME: i am getting different response from the server and once backend changes remove the get request and uncomment the commented code
|
2024-05-27 04:17:31 +00:00
|
|
|
let estimates = await this.getProjectEstimates(workspaceSlug, projectId, "mutation-loader");
|
|
|
|
estimates = sortBy(estimates, "created_at");
|
2024-05-24 11:54:50 +00:00
|
|
|
if (estimates && estimates.length > 0)
|
|
|
|
await this.store.projectRoot.project.updateProject(workspaceSlug, projectId, {
|
|
|
|
estimate: estimates[estimates.length - 1].id,
|
2024-05-24 09:37:44 +00:00
|
|
|
});
|
2024-05-24 11:54:50 +00:00
|
|
|
// 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));
|
|
|
|
// });
|
|
|
|
// }
|
2024-05-23 08:11:30 +00:00
|
|
|
|
|
|
|
return estimate;
|
|
|
|
} catch (error) {
|
|
|
|
this.error = {
|
|
|
|
status: "error",
|
|
|
|
message: "Error creating estimate",
|
|
|
|
};
|
|
|
|
}
|
2024-05-23 10:07:25 +00:00
|
|
|
};
|
2024-05-23 08:11:30 +00:00
|
|
|
}
|