plane/web/store/view/view-root.store.ts

214 lines
6.3 KiB
TypeScript
Raw Normal View History

2024-01-30 11:15:14 +00:00
import { action, computed, makeObservable, observable, runInAction } from "mobx";
import { computedFn } from "mobx-utils";
2024-02-02 08:22:38 +00:00
import set from "lodash/set";
import sortBy from "lodash/sortBy";
import reverse from "lodash/reverse";
2024-01-30 08:56:59 +00:00
// stores
import { RootStore } from "store/root.store";
2024-02-02 08:22:38 +00:00
import { ViewStore } from "./view.store";
2024-01-30 08:56:59 +00:00
// types
import { TUserViewService, TViewService } from "services/view/types";
2024-01-30 08:56:59 +00:00
import { TView } from "@plane/types";
// constants
import { EViewPageType } from "constants/view";
2024-01-30 08:56:59 +00:00
2024-02-05 14:39:17 +00:00
export type TLoader = "init-loader" | "mutation-loader" | "submitting" | undefined;
2024-01-30 08:56:59 +00:00
2024-02-02 08:22:38 +00:00
type TViewRootStore = {
2024-01-30 08:56:59 +00:00
// observables
2024-02-05 14:39:17 +00:00
loader: TLoader;
2024-02-02 08:22:38 +00:00
viewMap: Record<string, ViewStore>;
2024-01-30 08:56:59 +00:00
// computed
viewIds: string[];
2024-02-02 08:22:38 +00:00
viewById: (viewId: string) => ViewStore | undefined;
2024-01-30 08:56:59 +00:00
// actions
2024-02-05 14:39:17 +00:00
localViewCreate: (view: TView) => Promise<void>;
fetch: (_loader?: TLoader) => Promise<void>;
fetchById: (viewId: string) => Promise<void>;
2024-01-30 08:56:59 +00:00
create: (view: Partial<TView>) => Promise<void>;
remove: (viewId: string) => Promise<void>;
2024-01-30 08:56:59 +00:00
duplicate: (viewId: string) => Promise<void>;
};
2024-02-02 08:22:38 +00:00
export class ViewRootStore implements TViewRootStore {
2024-02-05 14:39:17 +00:00
// observables
loader: TLoader = "init-loader";
2024-02-02 08:22:38 +00:00
viewMap: Record<string, ViewStore> = {};
2024-01-30 08:56:59 +00:00
constructor(
private store: RootStore,
private defaultViews: TView[] = [],
private service: TViewService,
private userService: TUserViewService,
private viewPageType: EViewPageType
) {
2024-01-30 08:56:59 +00:00
makeObservable(this, {
// observables
2024-02-05 14:39:17 +00:00
loader: observable.ref,
viewMap: observable,
2024-01-30 08:56:59 +00:00
// computed
viewIds: computed,
// actions
2024-02-05 14:39:17 +00:00
localViewCreate: action,
fetch: action,
fetchById: action,
2024-01-30 08:56:59 +00:00
create: action,
remove: action,
2024-01-30 08:56:59 +00:00
duplicate: action,
});
}
// computed
get viewIds() {
2024-02-05 14:39:17 +00:00
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[];
2024-01-30 08:56:59 +00:00
}
viewById = computedFn((viewId: string) => this.viewMap?.[viewId] || undefined);
2024-01-30 08:56:59 +00:00
// 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)
);
});
};
2024-02-05 14:39:17 +00:00
fetch = async (_loader: TLoader = "init-loader") => {
try {
const { workspaceSlug, projectId } = this.store.app.router;
if (!workspaceSlug) return;
2024-01-30 08:56:59 +00:00
this.loader = _loader;
2024-02-05 14:39:17 +00:00
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)
);
});
2024-02-05 14:39:17 +00:00
});
const views = await this.service.fetch(workspaceSlug, projectId);
if (!views) return;
2024-01-30 08:56:59 +00:00
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;
2024-01-30 08:56:59 +00:00
});
} catch {}
2024-01-30 08:56:59 +00:00
};
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 {}
};
2024-01-30 11:15:14 +00:00
create = async (data: Partial<TView>) => {
try {
const { workspaceSlug, projectId } = this.store.app.router;
if (!workspaceSlug) return;
2024-01-30 08:56:59 +00:00
const view = await this.service.create(workspaceSlug, data, projectId);
if (!view) return;
2024-01-30 08:56:59 +00:00
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 {}
2024-01-30 08:56:59 +00:00
};
remove = async (viewId: string) => {
try {
const { workspaceSlug, projectId } = this.store.app.router;
if (!workspaceSlug || !viewId) return;
2024-01-30 08:56:59 +00:00
if (this.viewMap?.[viewId] != undefined && !this.viewMap?.[viewId]?.is_create)
await this.service.remove?.(workspaceSlug, viewId, projectId);
2024-01-30 08:56:59 +00:00
runInAction(() => {
delete this.viewMap[viewId];
});
} catch {}
2024-01-30 08:56:59 +00:00
};
duplicate = async (viewId: string) => {
try {
const { workspaceSlug, projectId } = this.store.app.router;
if (!workspaceSlug || !this.service.duplicate) return;
2024-01-30 08:56:59 +00:00
const view = await this.service.duplicate(workspaceSlug, viewId, projectId);
if (!view) return;
2024-01-30 08:56:59 +00:00
runInAction(() => {
if (view.id)
set(
this.viewMap,
[view.id],
new ViewStore(this.store, view, this.service, this.userService, this.viewPageType)
);
});
} catch {}
2024-01-30 08:56:59 +00:00
};
}