import set from "lodash/set"; import { action, computed, makeObservable, observable, runInAction } from "mobx"; import { computedFn } from "mobx-utils"; // types import { THomeDashboardResponse, TWidget, TWidgetFiltersFormData, TWidgetStatsResponse, TWidgetKeys, TWidgetStatsRequestParams, } from "@plane/types"; // services import { DashboardService } from "@/services/dashboard.service"; // plane web store import { CoreRootStore } from "./root.store"; export interface IDashboardStore { // error states widgetStatsError: { [workspaceSlug: string]: Record> }; // observables homeDashboardId: string | null; widgetDetails: { [workspaceSlug: string]: Record }; // { // workspaceSlug: { // dashboardId: TWidget[] // } // } widgetStats: { [workspaceSlug: string]: Record> }; // { // workspaceSlug: { // dashboardId: { // widgetKey: TWidgetStatsResponse; // } // } // } // computed homeDashboardWidgets: TWidget[] | undefined; // computed actions getWidgetDetails: (workspaceSlug: string, dashboardId: string, widgetKey: TWidgetKeys) => TWidget | undefined; getWidgetStats: (workspaceSlug: string, dashboardId: string, widgetKey: TWidgetKeys) => T | undefined; getWidgetStatsError: (workspaceSlug: string, dashboardId: string, widgetKey: TWidgetKeys) => any | null; // actions fetchHomeDashboardWidgets: (workspaceSlug: string) => Promise; fetchWidgetStats: ( workspaceSlug: string, dashboardId: string, params: TWidgetStatsRequestParams ) => Promise; updateDashboardWidget: ( workspaceSlug: string, dashboardId: string, widgetId: string, data: Partial ) => Promise; updateDashboardWidgetFilters: ( workspaceSlug: string, dashboardId: string, widgetId: string, data: TWidgetFiltersFormData ) => Promise; } export class DashboardStore implements IDashboardStore { // error states widgetStatsError: { [workspaceSlug: string]: Record> } = {}; // observables homeDashboardId: string | null = null; widgetDetails: { [workspaceSlug: string]: Record } = {}; widgetStats: { [workspaceSlug: string]: Record> } = {}; // stores routerStore; issueStore; // services dashboardService; constructor(_rootStore: CoreRootStore) { makeObservable(this, { // error states widgetStatsError: observable, // observables homeDashboardId: observable.ref, widgetDetails: observable, widgetStats: observable, // computed homeDashboardWidgets: computed, // fetch actions fetchHomeDashboardWidgets: action, fetchWidgetStats: action, // update actions updateDashboardWidget: action, updateDashboardWidgetFilters: action, }); // router store this.routerStore = _rootStore.router; this.issueStore = _rootStore.issue.issues; // services this.dashboardService = new DashboardService(); } /** * @description get home dashboard widgets * @returns {TWidget[] | undefined} */ get homeDashboardWidgets() { const workspaceSlug = this.routerStore.workspaceSlug; if (!workspaceSlug) return undefined; const { homeDashboardId, widgetDetails } = this; return homeDashboardId ? widgetDetails?.[workspaceSlug]?.[homeDashboardId] : undefined; } /** * @description get widget details * @param {string} workspaceSlug * @param {string} dashboardId * @param {TWidgetKeys} widgetKey * @returns {TWidget | undefined} */ getWidgetDetails = computedFn((workspaceSlug: string, dashboardId: string, widgetKey: TWidgetKeys) => { const widgets = this.widgetDetails?.[workspaceSlug]?.[dashboardId]; if (!widgets) return undefined; return widgets.find((widget) => widget.key === widgetKey); }); /** * @description get widget stats * @param {string} workspaceSlug * @param {string} dashboardId * @param {TWidgetKeys} widgetKey * @returns {T | undefined} */ getWidgetStats = (workspaceSlug: string, dashboardId: string, widgetKey: TWidgetKeys): T | undefined => (this.widgetStats?.[workspaceSlug]?.[dashboardId]?.[widgetKey] as unknown as T) ?? undefined; /** * @description get widget stats error * @param {string} workspaceSlug * @param {string} dashboardId * @param {TWidgetKeys} widgetKey * @returns {any | null} */ getWidgetStatsError = (workspaceSlug: string, dashboardId: string, widgetKey: TWidgetKeys) => this.widgetStatsError?.[workspaceSlug]?.[dashboardId]?.[widgetKey] ?? null; /** * @description fetch home dashboard details and widgets * @param {string} workspaceSlug * @returns {Promise} */ fetchHomeDashboardWidgets = async (workspaceSlug: string): Promise => { try { const response = await this.dashboardService.getHomeDashboardWidgets(workspaceSlug); runInAction(() => { this.homeDashboardId = response.dashboard.id; set(this.widgetDetails, [workspaceSlug, response.dashboard.id], response.widgets); }); return response; } catch (error) { runInAction(() => { this.homeDashboardId = null; }); throw error; } }; /** * @description fetch widget stats * @param {string} workspaceSlug * @param {string} dashboardId * @param {TWidgetStatsRequestParams} widgetKey * @returns widget stats */ fetchWidgetStats = async (workspaceSlug: string, dashboardId: string, params: TWidgetStatsRequestParams) => this.dashboardService .getWidgetStats(workspaceSlug, dashboardId, params) .then((res: any) => { runInAction(() => { if (res.issues) this.issueStore.addIssue(res.issues); set(this.widgetStats, [workspaceSlug, dashboardId, params.widget_key], res); set(this.widgetStatsError, [workspaceSlug, dashboardId, params.widget_key], null); }); return res; }) .catch((error) => { runInAction(() => { set(this.widgetStatsError, [workspaceSlug, dashboardId, params.widget_key], error); }); throw error; }); /** * @description update dashboard widget * @param {string} dashboardId * @param {string} widgetId * @param {Partial} data * @returns updated widget */ updateDashboardWidget = async ( workspaceSlug: string, dashboardId: string, widgetId: string, data: Partial ): Promise => { // find all widgets in dashboard const widgets = this.widgetDetails?.[workspaceSlug]?.[dashboardId]; if (!widgets) throw new Error("Dashboard not found"); // find widget index const widgetIndex = widgets.findIndex((widget) => widget.id === widgetId); // get original widget const originalWidget = { ...widgets[widgetIndex] }; if (widgetIndex === -1) throw new Error("Widget not found"); try { runInAction(() => { this.widgetDetails[workspaceSlug][dashboardId][widgetIndex] = { ...widgets[widgetIndex], ...data, }; }); const response = await this.dashboardService.updateDashboardWidget(dashboardId, widgetId, data); return response; } catch (error) { // revert changes runInAction(() => { this.widgetDetails[workspaceSlug][dashboardId][widgetIndex] = originalWidget; }); throw error; } }; /** * @description update dashboard widget filters * @param {string} dashboardId * @param {string} widgetId * @param {TWidgetFiltersFormData} data * @returns updated widget */ updateDashboardWidgetFilters = async ( workspaceSlug: string, dashboardId: string, widgetId: string, data: TWidgetFiltersFormData ): Promise => { const widgetDetails = this.getWidgetDetails(workspaceSlug, dashboardId, data.widgetKey); if (!widgetDetails) throw new Error("Widget not found"); try { const updatedWidget = { ...widgetDetails, widget_filters: { ...widgetDetails.widget_filters, ...data.filters, }, }; // update widget details optimistically runInAction(() => { set( this.widgetDetails, [workspaceSlug, dashboardId], this.widgetDetails?.[workspaceSlug]?.[dashboardId]?.map((w) => (w.id === widgetId ? updatedWidget : w)) ); }); const response = await this.updateDashboardWidget(workspaceSlug, dashboardId, widgetId, { filters: { ...widgetDetails.widget_filters, ...data.filters, }, }).then((res) => res); return response; } catch (error) { // revert changes runInAction(() => { this.widgetDetails[workspaceSlug][dashboardId] = this.widgetDetails?.[workspaceSlug]?.[dashboardId]?.map((w) => w.id === widgetId ? widgetDetails : w ); }); throw error; } }; }