import { FC, useCallback, useEffect } from "react"; import { observer } from "mobx-react-lite"; // types import { GroupByColumnTypes, TGroupedIssues } from "@plane/types"; // constants import { EIssueLayoutTypes, EIssuesStoreType } from "@/constants/issue"; import { EUserProjectRoles } from "@/constants/project"; // hooks import { useIssues, useUser } from "@/hooks/store"; // hooks import { useGroupIssuesDragNDrop } from "@/hooks/use-group-dragndrop"; import { useIssueStoreType } from "@/hooks/use-issue-layout-store"; import { useIssuesActions } from "@/hooks/use-issues-actions"; // components import { IssueLayoutHOC } from "../issue-layout-HOC"; import { List } from "./default"; // types import { IQuickActionProps, TRenderQuickActions } from "./list-view-types"; type ListStoreType = | EIssuesStoreType.PROJECT | EIssuesStoreType.MODULE | EIssuesStoreType.CYCLE | EIssuesStoreType.PROJECT_VIEW | EIssuesStoreType.DRAFT | EIssuesStoreType.PROFILE | EIssuesStoreType.ARCHIVED; interface IBaseListRoot { QuickActions: FC; addIssuesToView?: (issueIds: string[]) => Promise; canEditPropertiesBasedOnProject?: (projectId: string) => boolean; viewId?: string | undefined; isCompletedCycle?: boolean; } export const BaseListRoot = observer((props: IBaseListRoot) => { const { QuickActions, viewId, addIssuesToView, canEditPropertiesBasedOnProject, isCompletedCycle = false, } = props; // router const storeType = useIssueStoreType() as ListStoreType; //stores const { issuesFilter, issues } = useIssues(storeType); const { fetchIssues, fetchNextIssues, quickAddIssue, updateIssue, removeIssue, removeIssueFromView, archiveIssue, restoreIssue, } = useIssuesActions(storeType); // mobx store const { membership: { currentProjectRole }, } = useUser(); const { issueMap } = useIssues(); const displayFilters = issuesFilter?.issueFilters?.displayFilters; const displayProperties = issuesFilter?.issueFilters?.displayProperties; const orderBy = displayFilters?.order_by || undefined; const group_by = (displayFilters?.group_by || null) as GroupByColumnTypes | null; const showEmptyGroup = displayFilters?.show_empty_groups ?? false; useEffect(() => { fetchIssues("init-loader", { canGroup: true, perPageCount: group_by ? 50 : 100 }, viewId); }, [fetchIssues, storeType, group_by, viewId]); const groupedIssueIds = issues?.groupedIssueIds as TGroupedIssues | undefined; // auth const isEditingAllowed = !!currentProjectRole && currentProjectRole >= EUserProjectRoles.MEMBER; const { enableInlineEditing, enableQuickAdd, enableIssueCreation } = issues?.viewFlags || {}; const canEditProperties = useCallback( (projectId: string | undefined) => { const isEditingAllowedBasedOnProject = canEditPropertiesBasedOnProject && projectId ? canEditPropertiesBasedOnProject(projectId) : isEditingAllowed; return !!enableInlineEditing && isEditingAllowedBasedOnProject; }, [canEditPropertiesBasedOnProject, enableInlineEditing, isEditingAllowed] ); const handleOnDrop = useGroupIssuesDragNDrop(storeType, orderBy, group_by); const renderQuickActions: TRenderQuickActions = useCallback( ({ issue, parentRef }) => ( removeIssue(issue.project_id, issue.id)} handleUpdate={async (data) => updateIssue && updateIssue(issue.project_id, issue.id, data)} handleRemoveFromView={async () => removeIssueFromView && removeIssueFromView(issue.project_id, issue.id)} handleArchive={async () => archiveIssue && archiveIssue(issue.project_id, issue.id)} handleRestore={async () => restoreIssue && restoreIssue(issue.project_id, issue.id)} readOnly={!isEditingAllowed || isCompletedCycle} /> ), // eslint-disable-next-line react-hooks/exhaustive-deps [isEditingAllowed, isCompletedCycle, removeIssue, updateIssue, removeIssueFromView, archiveIssue, restoreIssue] ); const loadMoreIssues = useCallback( (groupId?: string) => { fetchNextIssues(groupId); }, [fetchNextIssues] ); return (
); });