forked from github/plane
ee0e3e2e25
* fix: completed cycle list layout validation * fix: completed cycle kanban layout validation * fix: completed cycle spreadsheet layout validation * fix: date dropdown disabled fix * chore: quick action validation added for list, kanban and spreadsheet layout * fix: calendar layout validation added
130 lines
4.6 KiB
TypeScript
130 lines
4.6 KiB
TypeScript
import { FC, useCallback } from "react";
|
|
import { useRouter } from "next/router";
|
|
import { observer } from "mobx-react-lite";
|
|
// hooks
|
|
import { useUser } from "hooks/store";
|
|
// views
|
|
import { SpreadsheetView } from "./spreadsheet-view";
|
|
// types
|
|
import { TIssue, IIssueDisplayFilterOptions, TUnGroupedIssues } from "@plane/types";
|
|
import { EIssueActions } from "../types";
|
|
import { IQuickActionProps } from "../list/list-view-types";
|
|
// constants
|
|
import { EUserProjectRoles } from "constants/project";
|
|
import { ICycleIssuesFilter, ICycleIssues } from "store/issue/cycle";
|
|
import { IModuleIssuesFilter, IModuleIssues } from "store/issue/module";
|
|
import { IProjectIssuesFilter, IProjectIssues } from "store/issue/project";
|
|
import { IProjectViewIssuesFilter, IProjectViewIssues } from "store/issue/project-views";
|
|
import { EIssueFilterType } from "constants/issue";
|
|
|
|
interface IBaseSpreadsheetRoot {
|
|
issueFiltersStore: IProjectIssuesFilter | IModuleIssuesFilter | ICycleIssuesFilter | IProjectViewIssuesFilter;
|
|
issueStore: IProjectIssues | ICycleIssues | IModuleIssues | IProjectViewIssues;
|
|
viewId?: string;
|
|
QuickActions: FC<IQuickActionProps>;
|
|
issueActions: {
|
|
[EIssueActions.DELETE]: (issue: TIssue) => void;
|
|
[EIssueActions.UPDATE]?: (issue: TIssue) => void;
|
|
[EIssueActions.REMOVE]?: (issue: TIssue) => void;
|
|
};
|
|
canEditPropertiesBasedOnProject?: (projectId: string) => boolean;
|
|
isCompletedCycle?: boolean;
|
|
}
|
|
|
|
export const BaseSpreadsheetRoot = observer((props: IBaseSpreadsheetRoot) => {
|
|
const {
|
|
issueFiltersStore,
|
|
issueStore,
|
|
viewId,
|
|
QuickActions,
|
|
issueActions,
|
|
canEditPropertiesBasedOnProject,
|
|
isCompletedCycle = false,
|
|
} = props;
|
|
// router
|
|
const router = useRouter();
|
|
const { workspaceSlug, projectId } = router.query as { workspaceSlug: string; projectId: string };
|
|
// store hooks
|
|
const {
|
|
membership: { currentProjectRole },
|
|
} = useUser();
|
|
// derived values
|
|
const { enableInlineEditing, enableQuickAdd, enableIssueCreation } = issueStore?.viewFlags || {};
|
|
// user role validation
|
|
const isEditingAllowed = !!currentProjectRole && currentProjectRole >= EUserProjectRoles.MEMBER;
|
|
|
|
const canEditProperties = useCallback(
|
|
(projectId: string | undefined) => {
|
|
const isEditingAllowedBasedOnProject =
|
|
canEditPropertiesBasedOnProject && projectId ? canEditPropertiesBasedOnProject(projectId) : isEditingAllowed;
|
|
|
|
return enableInlineEditing && isEditingAllowedBasedOnProject;
|
|
},
|
|
[canEditPropertiesBasedOnProject, enableInlineEditing, isEditingAllowed]
|
|
);
|
|
|
|
const issueIds = (issueStore.groupedIssueIds ?? []) as TUnGroupedIssues;
|
|
|
|
const handleIssues = useCallback(
|
|
async (issue: TIssue, action: EIssueActions) => {
|
|
if (issueActions[action]) {
|
|
issueActions[action]!(issue);
|
|
}
|
|
},
|
|
[issueActions]
|
|
);
|
|
|
|
const handleDisplayFiltersUpdate = useCallback(
|
|
(updatedDisplayFilter: Partial<IIssueDisplayFilterOptions>) => {
|
|
if (!workspaceSlug || !projectId) return;
|
|
|
|
issueFiltersStore.updateFilters(
|
|
workspaceSlug,
|
|
projectId,
|
|
EIssueFilterType.DISPLAY_FILTERS,
|
|
{
|
|
...updatedDisplayFilter,
|
|
},
|
|
viewId
|
|
);
|
|
},
|
|
[issueFiltersStore, projectId, workspaceSlug, viewId]
|
|
);
|
|
|
|
const renderQuickActions = useCallback(
|
|
(issue: TIssue, customActionButton?: React.ReactElement, portalElement?: HTMLDivElement | null) => (
|
|
<QuickActions
|
|
customActionButton={customActionButton}
|
|
issue={issue}
|
|
handleDelete={async () => handleIssues(issue, EIssueActions.DELETE)}
|
|
handleUpdate={
|
|
issueActions[EIssueActions.UPDATE] ? async (data) => handleIssues(data, EIssueActions.UPDATE) : undefined
|
|
}
|
|
handleRemoveFromView={
|
|
issueActions[EIssueActions.REMOVE] ? async () => handleIssues(issue, EIssueActions.REMOVE) : undefined
|
|
}
|
|
portalElement={portalElement}
|
|
readOnly={!isEditingAllowed || isCompletedCycle}
|
|
/>
|
|
),
|
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
[handleIssues]
|
|
);
|
|
|
|
return (
|
|
<SpreadsheetView
|
|
displayProperties={issueFiltersStore.issueFilters?.displayProperties ?? {}}
|
|
displayFilters={issueFiltersStore.issueFilters?.displayFilters ?? {}}
|
|
handleDisplayFilterUpdate={handleDisplayFiltersUpdate}
|
|
issueIds={issueIds}
|
|
quickActions={renderQuickActions}
|
|
handleIssues={handleIssues}
|
|
canEditProperties={canEditProperties}
|
|
quickAddCallback={issueStore.quickAddIssue}
|
|
viewId={viewId}
|
|
enableQuickCreateIssue={enableQuickAdd}
|
|
disableIssueCreation={!enableIssueCreation || !isEditingAllowed || isCompletedCycle}
|
|
/>
|
|
);
|
|
});
|