Files
plane/web/core/components/project/project-settings-member-defaults.tsx
Prateek Shourya 67cbe94d4a [WEB-3964] refactor: permission layer (#7094)
* refactor: permission layer

* refactor: add original_role to project member serializer

* chore: minor fixes related to permission layer

* fix: strict type checking while checking user permissions
2025-05-30 19:57:07 +05:30

191 lines
6.1 KiB
TypeScript

"use client";
import { useEffect, ReactNode } from "react";
import { observer } from "mobx-react";
import { Controller, useForm } from "react-hook-form";
import useSWR from "swr";
// plane imports
import { EUserPermissions, EUserPermissionsLevel } from "@plane/constants";
import { useTranslation } from "@plane/i18n";
import { IProject, IUserLite, IWorkspace } from "@plane/types";
import { Loader, TOAST_TYPE, ToggleSwitch, setToast } from "@plane/ui";
// components
import { MemberSelect } from "@/components/project";
// constants
import { PROJECT_MEMBERS } from "@/constants/fetch-keys";
// hooks
import { useProject, useUserPermissions } from "@/hooks/store";
const defaultValues: Partial<IProject> = {
project_lead: null,
default_assignee: null,
};
type TDefaultSettingItemProps = {
title: string;
description: string;
children: ReactNode;
};
const DefaultSettingItem: React.FC<TDefaultSettingItemProps> = ({ title, description, children }) => (
<div className="flex items-center justify-between gap-x-2">
<div className="flex flex-col gap-0.5">
<h4 className="text-sm font-medium">{title}</h4>
<p className="text-xs text-custom-text-300">{description}</p>
</div>
<div className="w-full max-w-48 sm:max-w-64">{children}</div>
</div>
);
type TProjectSettingsMemberDefaultsProps = {
workspaceSlug: string;
projectId: string;
};
export const ProjectSettingsMemberDefaults: React.FC<TProjectSettingsMemberDefaultsProps> = observer((props) => {
const { workspaceSlug, projectId } = props;
// plane hooks
const { t } = useTranslation();
// store hooks
const { allowPermissions } = useUserPermissions();
const { currentProjectDetails, fetchProjectDetails, updateProject } = useProject();
// derived values
const isAdmin = allowPermissions(
[EUserPermissions.ADMIN],
EUserPermissionsLevel.PROJECT,
workspaceSlug,
currentProjectDetails?.id
);
// form info
const { reset, control } = useForm<IProject>({ defaultValues });
// fetching user members
useSWR(
workspaceSlug && projectId ? PROJECT_MEMBERS(projectId) : null,
workspaceSlug && projectId ? () => fetchProjectDetails(workspaceSlug, projectId) : null
);
useEffect(() => {
if (!currentProjectDetails) return;
reset({
...currentProjectDetails,
default_assignee:
(currentProjectDetails.default_assignee as IUserLite)?.id ?? currentProjectDetails.default_assignee,
project_lead: (currentProjectDetails.project_lead as IUserLite)?.id ?? currentProjectDetails.project_lead,
workspace: (currentProjectDetails.workspace as IWorkspace).id,
});
}, [currentProjectDetails, reset]);
const submitChanges = async (formData: Partial<IProject>) => {
if (!workspaceSlug || !projectId) return;
reset({
...currentProjectDetails,
default_assignee:
(currentProjectDetails?.default_assignee as IUserLite)?.id ?? currentProjectDetails?.default_assignee,
project_lead: (currentProjectDetails?.project_lead as IUserLite)?.id ?? currentProjectDetails?.project_lead,
...formData,
});
await updateProject(workspaceSlug, projectId, {
default_assignee:
formData.default_assignee === "none"
? null
: (formData.default_assignee ?? currentProjectDetails?.default_assignee),
project_lead:
formData.project_lead === "none" ? null : (formData.project_lead ?? currentProjectDetails?.project_lead),
})
.then(() => {
setToast({
title: `${t("success")}!`,
type: TOAST_TYPE.SUCCESS,
message: t("project_settings.general.toast.success"),
});
})
.catch((err) => {
console.error(err);
});
};
const toggleGuestViewAllIssues = async (value: boolean) => {
if (!workspaceSlug || !projectId) return;
updateProject(workspaceSlug, projectId, {
guest_view_all_features: value,
})
.then(() => {
setToast({
title: `${t("success")}!`,
type: TOAST_TYPE.SUCCESS,
message: t("project_settings.general.toast.success"),
});
})
.catch((err) => {
console.error(err);
});
};
return (
<div className="flex flex-col gap-y-6 my-6">
<DefaultSettingItem title="Project Lead" description="Select the project lead for the project.">
{currentProjectDetails ? (
<Controller
control={control}
name="project_lead"
render={({ field: { value } }) => (
<MemberSelect
value={value}
onChange={(val: string) => {
submitChanges({ project_lead: val });
}}
isDisabled={!isAdmin}
/>
)}
/>
) : (
<Loader className="h-9 w-full">
<Loader.Item width="100%" height="100%" />
</Loader>
)}
</DefaultSettingItem>
<DefaultSettingItem title="Default Assignee" description="Select the default assignee for the project.">
{currentProjectDetails ? (
<Controller
control={control}
name="default_assignee"
render={({ field: { value } }) => (
<MemberSelect
value={value}
onChange={(val: string) => {
submitChanges({ default_assignee: val });
}}
isDisabled={!isAdmin}
/>
)}
/>
) : (
<Loader className="h-9 w-full">
<Loader.Item width="100%" height="100%" />
</Loader>
)}
</DefaultSettingItem>
{currentProjectDetails && (
<DefaultSettingItem
title="Guest access"
description="This will allow guests to have view access to all the project work items."
>
<div className="flex items-center justify-end">
<ToggleSwitch
value={!!currentProjectDetails?.guest_view_all_features}
onChange={() => toggleGuestViewAllIssues(!currentProjectDetails?.guest_view_all_features)}
disabled={!isAdmin}
size="sm"
/>
</div>
</DefaultSettingItem>
)}
</div>
);
});