From 7507cb0a0fab706929865ce440c0d7b140b71118 Mon Sep 17 00:00:00 2001 From: Aaryan Khandelwal <65252264+aaryan610@users.noreply.github.com> Date: Mon, 15 Apr 2024 12:50:22 +0530 Subject: [PATCH] [WEB-994] fix: pages list mutation between projects (#4179) * fix: pages list mutation between projects * fix: page tab logic * chore: remove pageType from the project pages store * chore: rename computed helper functions --- web/components/pages/list/root.tsx | 13 +++-- web/components/pages/list/tab-navigation.tsx | 27 ++++----- .../pages/pages-list-main-content.tsx | 5 +- web/components/pages/pages-list-view.tsx | 5 +- .../projects/[projectId]/pages/index.tsx | 6 +- web/store/pages/project-page.store.ts | 55 ++++++++++--------- 6 files changed, 59 insertions(+), 52 deletions(-) diff --git a/web/components/pages/list/root.tsx b/web/components/pages/list/root.tsx index 5833e7851..226bdaf70 100644 --- a/web/components/pages/list/root.tsx +++ b/web/components/pages/list/root.tsx @@ -1,19 +1,24 @@ import { FC } from "react"; import { observer } from "mobx-react"; +// types +import { TPageNavigationTabs } from "@plane/types"; // hooks import { useProjectPages } from "@/hooks/store"; // components import { PageListBlock } from "./"; type TPagesListRoot = { - workspaceSlug: string; + pageType: TPageNavigationTabs; projectId: string; + workspaceSlug: string; }; export const PagesListRoot: FC = observer((props) => { - const { workspaceSlug, projectId } = props; - // hooks - const { filteredPageIds } = useProjectPages(projectId); + const { pageType, projectId, workspaceSlug } = props; + // store hooks + const { getCurrentProjectFilteredPageIds } = useProjectPages(projectId); + // derived values + const filteredPageIds = getCurrentProjectFilteredPageIds(pageType); if (!filteredPageIds) return <>; return ( diff --git a/web/components/pages/list/tab-navigation.tsx b/web/components/pages/list/tab-navigation.tsx index 37a1ef64b..fc28d50a2 100644 --- a/web/components/pages/list/tab-navigation.tsx +++ b/web/components/pages/list/tab-navigation.tsx @@ -42,21 +42,18 @@ export const PageTabNavigation: FC = (props) => { href={`/${workspaceSlug}/projects/${projectId}/pages?type=${tab.key}`} onClick={(e) => handleTabClick(e, tab.key)} > -
-
- {tab.label} -
-
-
+ + {tab.label} + +
))}
diff --git a/web/components/pages/pages-list-main-content.tsx b/web/components/pages/pages-list-main-content.tsx index e961849ec..60e11dd8c 100644 --- a/web/components/pages/pages-list-main-content.tsx +++ b/web/components/pages/pages-list-main-content.tsx @@ -22,7 +22,10 @@ type Props = { export const PagesListMainContent: React.FC = observer((props) => { const { children, pageType, projectId } = props; // store hooks - const { loader, filteredPageIds, pageIds, filters } = useProjectPages(projectId); + const { loader, getCurrentProjectFilteredPageIds, getCurrentProjectPageIds, filters } = useProjectPages(projectId); + // derived values + const pageIds = getCurrentProjectPageIds(pageType); + const filteredPageIds = getCurrentProjectFilteredPageIds(pageType); if (loader === "init-loader") return ; // if no pages exist in the active page type diff --git a/web/components/pages/pages-list-view.tsx b/web/components/pages/pages-list-view.tsx index 2ff4aead9..7f12cfc92 100644 --- a/web/components/pages/pages-list-view.tsx +++ b/web/components/pages/pages-list-view.tsx @@ -18,10 +18,7 @@ export const PagesListView: React.FC = observer((props) => { // store hooks const { getAllPages } = useProjectPages(projectId); // fetching pages list - useSWR( - projectId && pageType ? `PROJECT_PAGES_${projectId}_${pageType}` : null, - projectId && pageType ? () => getAllPages(pageType) : null - ); + useSWR(projectId ? `PROJECT_PAGES_${projectId}` : null, projectId ? () => getAllPages(pageType) : null); // pages loader return ( diff --git a/web/pages/[workspaceSlug]/projects/[projectId]/pages/index.tsx b/web/pages/[workspaceSlug]/projects/[projectId]/pages/index.tsx index 6869c242e..b8e1acfc9 100644 --- a/web/pages/[workspaceSlug]/projects/[projectId]/pages/index.tsx +++ b/web/pages/[workspaceSlug]/projects/[projectId]/pages/index.tsx @@ -36,7 +36,11 @@ const ProjectPagesPage: NextPageWithLayout = observer(() => { projectId={projectId.toString()} pageType={currentPageType()} > - + ); }); diff --git a/web/store/pages/project-page.store.ts b/web/store/pages/project-page.store.ts index ef1560350..136ce61a5 100644 --- a/web/store/pages/project-page.store.ts +++ b/web/store/pages/project-page.store.ts @@ -1,12 +1,14 @@ import set from "lodash/set"; import unset from "lodash/unset"; -import { makeObservable, observable, runInAction, action, computed } from "mobx"; +import { makeObservable, observable, runInAction, action } from "mobx"; import { computedFn } from "mobx-utils"; +// types import { TPage, TPageFilters, TPageNavigationTabs } from "@plane/types"; // helpers import { filterPagesByPageType, orderPages, shouldFilterPage } from "@/helpers/page.helper"; // services import { PageService } from "@/services/page.service"; +// store import { IPageStore, PageStore } from "@/store/pages/page.store"; import { RootStore } from "../root.store"; @@ -17,14 +19,12 @@ type TError = { title: string; description: string }; export interface IProjectPageStore { // observables loader: TLoader; - pageType: TPageNavigationTabs; data: Record; // pageId => PageStore error: TError | undefined; filters: TPageFilters; - // computed - pageIds: string[] | undefined; - filteredPageIds: string[] | undefined; // helper actions + getCurrentProjectPageIds: (pageType: TPageNavigationTabs) => string[] | undefined; + getCurrentProjectFilteredPageIds: (pageType: TPageNavigationTabs) => string[] | undefined; pageById: (pageId: string) => IPageStore | undefined; updateFilters: (filterKey: T, filterValue: TPageFilters[T]) => void; clearAllFilters: () => void; @@ -38,7 +38,6 @@ export interface IProjectPageStore { export class ProjectPageStore implements IProjectPageStore { // observables loader: TLoader = "init-loader"; - pageType: TPageNavigationTabs = "public"; data: Record = {}; // pageId => PageStore error: TError | undefined = undefined; filters: TPageFilters = { @@ -53,13 +52,9 @@ export class ProjectPageStore implements IProjectPageStore { makeObservable(this, { // observables loader: observable.ref, - pageType: observable.ref, data: observable, error: observable, filters: observable, - // computed - pageIds: computed, - filteredPageIds: computed, // helper actions updateFilters: action, clearAllFilters: action, @@ -73,26 +68,35 @@ export class ProjectPageStore implements IProjectPageStore { this.service = new PageService(); } - get pageIds() { + /** + * @description get the current project page ids based on the pageType + * @param {TPageNavigationTabs} pageType + */ + getCurrentProjectPageIds = computedFn((pageType: TPageNavigationTabs) => { const { projectId } = this.store.app.router; if (!projectId) return undefined; - // helps to filter pages based on the pageType - const pagesByType = filterPagesByPageType(this.pageType, Object.values(this?.data || {})); + let pagesByType = filterPagesByPageType(pageType, Object.values(this?.data || {})); + pagesByType = pagesByType.filter((p) => p.project === projectId); const pages = (pagesByType.map((page) => page.id) as string[]) || undefined; return pages ?? undefined; - } + }); - get filteredPageIds() { + /** + * @description get the current project filtered page ids based on the pageType + * @param {TPageNavigationTabs} pageType + */ + getCurrentProjectFilteredPageIds = computedFn((pageType: TPageNavigationTabs) => { const { projectId } = this.store.app.router; if (!projectId) return undefined; // helps to filter pages based on the pageType - const pagesByType = filterPagesByPageType(this.pageType, Object.values(this?.data || {})); + const pagesByType = filterPagesByPageType(pageType, Object.values(this?.data || {})); let filteredPages = pagesByType.filter( (p) => + p.project === projectId && p.name?.toLowerCase().includes(this.filters.searchQuery.toLowerCase()) && shouldFilterPage(p, this.filters.filters) ); @@ -101,15 +105,14 @@ export class ProjectPageStore implements IProjectPageStore { const pages = (filteredPages.map((page) => page.id) as string[]) || undefined; return pages ?? undefined; - } - - pageById = computedFn((pageId: string) => { - const { projectId } = this.store.app.router; - if (!projectId) return undefined; - - return this.data?.[pageId] || undefined; }); + /** + * @description get the page store by id + * @param {string} pageId + */ + pageById = computedFn((pageId: string) => this.data?.[pageId] || undefined); + updateFilters = (filterKey: T, filterValue: TPageFilters[T]) => { runInAction(() => { set(this.filters, [filterKey], filterValue); @@ -125,17 +128,15 @@ export class ProjectPageStore implements IProjectPageStore { }); /** - * @description fetch all the pages based on the navigation tab - * @param {TPageNavigationTabs} pageType + * @description fetch all the pages */ getAllPages = async (pageType: TPageNavigationTabs) => { try { const { workspaceSlug, projectId } = this.store.app.router; if (!workspaceSlug || !projectId) return undefined; - const currentPageIds = this.pageIds; + const currentPageIds = this.getCurrentProjectPageIds(pageType); runInAction(() => { - this.pageType = pageType; this.loader = currentPageIds && currentPageIds.length > 0 ? `mutation-loader` : `init-loader`; this.error = undefined; });