import { action, computed, makeObservable, observable, runInAction } from "mobx"; import { computedFn } from "mobx-utils"; import set from "lodash/set"; import sortBy from "lodash/sortBy"; import reverse from "lodash/reverse"; // stores import { RootStore } from "store/root.store"; import { ViewStore } from "./view.store"; // types import { TUserViewService, TViewService } from "services/view/types"; import { TView } from "@plane/types"; // constants import { EViewPageType } from "constants/view"; export type TLoader = "init-loader" | "mutation-loader" | "submitting" | undefined; type TViewRootStore = { // observables loader: TLoader; viewMap: Record; // computed viewIds: string[]; viewById: (viewId: string) => ViewStore | undefined; // actions localViewCreate: (view: TView) => Promise; fetch: (_loader?: TLoader) => Promise; fetchById: (viewId: string) => Promise; create: (view: Partial) => Promise; remove: (viewId: string) => Promise; duplicate: (viewId: string) => Promise; }; export class ViewRootStore implements TViewRootStore { // observables loader: TLoader = "init-loader"; viewMap: Record = {}; constructor( private store: RootStore, private defaultViews: TView[] = [], private service: TViewService, private userService: TUserViewService, private viewPageType: EViewPageType ) { makeObservable(this, { // observables loader: observable.ref, viewMap: observable, // computed viewIds: computed, // actions localViewCreate: action, fetch: action, fetchById: action, create: action, remove: action, duplicate: action, }); } // computed get viewIds() { const views = Object.values(this.viewMap); const localViews = views.filter((view) => view.is_local_view); let apiViews = views.filter((view) => !view.is_local_view && !view.is_create); apiViews = reverse(sortBy(apiViews, "sort_order")); const _viewIds = [...localViews.map((view) => view.id), ...apiViews.map((view) => view.id)]; return _viewIds as string[]; } viewById = computedFn((viewId: string) => this.viewMap?.[viewId] || undefined); // actions localViewCreate = async (view: TView) => { runInAction(() => { if (view.id) set( this.viewMap, [view.id], new ViewStore(this.store, view, this.service, this.userService, this.viewPageType) ); }); }; fetch = async (_loader: TLoader = "init-loader") => { try { const { workspaceSlug, projectId } = this.store.app.router; if (!workspaceSlug) return; this.loader = _loader; if (this.defaultViews && this.defaultViews.length > 0) runInAction(() => { this.defaultViews?.forEach((view) => { if (view.id) set( this.viewMap, [view.id], new ViewStore(this.store, view, this.service, this.userService, this.viewPageType) ); }); }); const views = await this.service.fetch(workspaceSlug, projectId); if (!views) return; runInAction(() => { views.forEach((view) => { if (view.id) set( this.viewMap, [view.id], new ViewStore(this.store, view, this.service, this.userService, this.viewPageType) ); }); this.loader = undefined; }); } catch {} }; fetchById = async (viewId: string) => { try { const { workspaceSlug, projectId } = this.store.app.router; if (!workspaceSlug || !viewId) return; // fetching display properties and display_filters const userView = await this.userService.fetch(workspaceSlug, projectId); if (!userView) return; // fetching kanban display filters from local // fetching display filters from local and from the view if (["all-issues", "assigned", "created", "subscribed"].includes(viewId)) { const view = { ...this.viewById(viewId) }; if (!view) return; runInAction(() => { view.display_filters = userView.display_filters; view.display_properties = userView.display_properties; }); } else { const view = await this.service.fetchById(workspaceSlug, viewId, projectId); if (!view) return; view?.display_filters && (view.display_filters = userView.display_filters); view?.display_properties && (view.display_properties = userView.display_properties); runInAction(() => { if (view.id) set( this.viewMap, [view.id], new ViewStore(this.store, view, this.service, this.userService, this.viewPageType) ); }); } } catch {} }; create = async (data: Partial) => { try { const { workspaceSlug, projectId } = this.store.app.router; if (!workspaceSlug) return; const view = await this.service.create(workspaceSlug, data, projectId); if (!view) return; runInAction(() => { if (view.id) set( this.viewMap, [view.id], new ViewStore(this.store, view, this.service, this.userService, this.viewPageType) ); }); if (data.id) this.remove(data.id); } catch {} }; remove = async (viewId: string) => { try { const { workspaceSlug, projectId } = this.store.app.router; if (!workspaceSlug || !viewId) return; if (this.viewMap?.[viewId] != undefined && !this.viewMap?.[viewId]?.is_create) await this.service.remove?.(workspaceSlug, viewId, projectId); runInAction(() => { delete this.viewMap[viewId]; }); } catch {} }; duplicate = async (viewId: string) => { try { const { workspaceSlug, projectId } = this.store.app.router; if (!workspaceSlug || !this.service.duplicate) return; const view = await this.service.duplicate(workspaceSlug, viewId, projectId); if (!view) return; runInAction(() => { if (view.id) set( this.viewMap, [view.id], new ViewStore(this.store, view, this.service, this.userService, this.viewPageType) ); }); } catch {} }; }