Handle remove member operation in frontend (#32)
* Add remove member gql client method * Handle remove member UI operation * Refactor fetching of project members * Rename type MemberPermission to ProjectMember * Add types to gql client response * Remove circular dependency in gql client types --------- Co-authored-by: neeraj <neeraj.rtly@gmail.com>
This commit is contained in:
parent
02f7ebb9bd
commit
cbc394f9f8
@ -12,12 +12,12 @@ import {
|
||||
Checkbox,
|
||||
} from '@material-tailwind/react';
|
||||
|
||||
import { Member, Permission } from '../../../../types/project';
|
||||
import { ProjectMember, Permission } from '../../../../types/project';
|
||||
|
||||
interface AddMemberDialogProp {
|
||||
open: boolean;
|
||||
handleOpen: () => void;
|
||||
handleAddMember: (member: Member) => void;
|
||||
handleAddMember: (projectMember: ProjectMember) => void;
|
||||
}
|
||||
|
||||
interface formData {
|
||||
@ -52,13 +52,17 @@ const AddMemberDialog = ({
|
||||
reset();
|
||||
handleOpen();
|
||||
|
||||
const member: Member = {
|
||||
email: data.emailAddress,
|
||||
id: Math.random(),
|
||||
const projectMember: ProjectMember = {
|
||||
id: Math.random().toString(),
|
||||
permissions: [],
|
||||
member: {
|
||||
name: '',
|
||||
email: data.emailAddress,
|
||||
id: Math.random().toString(),
|
||||
},
|
||||
};
|
||||
|
||||
handleAddMember(member);
|
||||
handleAddMember(projectMember);
|
||||
}, []);
|
||||
|
||||
return (
|
||||
|
@ -6,13 +6,13 @@ import { Button, Typography } from '@material-tailwind/react';
|
||||
import DomainCard from './DomainCard';
|
||||
import {
|
||||
DomainDetails,
|
||||
ProjectsOutletContext,
|
||||
ProjectSearchOutletContext,
|
||||
} from '../../../../types/project';
|
||||
|
||||
const Domains = () => {
|
||||
const { id } = useParams();
|
||||
|
||||
const { projects } = useOutletContext<ProjectsOutletContext>();
|
||||
const { projects } = useOutletContext<ProjectSearchOutletContext>();
|
||||
|
||||
const currentProject = useMemo(() => {
|
||||
return projects.find((project) => {
|
||||
|
@ -17,7 +17,7 @@ import DisplayEnvironmentVariables from './DisplayEnvironmentVariables';
|
||||
import {
|
||||
EnvironmentVariable,
|
||||
Environments,
|
||||
ProjectsOutletContext,
|
||||
ProjectSearchOutletContext,
|
||||
} from '../../../../types/project';
|
||||
import HorizontalLine from '../../../HorizontalLine';
|
||||
|
||||
@ -36,7 +36,7 @@ export type EnvironmentVariablesFormValues = {
|
||||
export const EnvironmentVariablesTabPanel = () => {
|
||||
const { id } = useParams();
|
||||
|
||||
const { projects } = useOutletContext<ProjectsOutletContext>();
|
||||
const { projects } = useOutletContext<ProjectSearchOutletContext>();
|
||||
|
||||
const currentProject = useMemo(() => {
|
||||
return projects.find((project) => {
|
||||
|
@ -13,7 +13,7 @@ import {
|
||||
|
||||
import DeleteProjectDialog from './DeleteProjectDialog';
|
||||
import ConfirmDialog from '../../../shared/ConfirmDialog';
|
||||
import { ProjectsOutletContext } from '../../../../types/project';
|
||||
import { ProjectSearchOutletContext } from '../../../../types/project';
|
||||
|
||||
const TEAMS = ['Airfoil'];
|
||||
const DEFAULT_SELECT_TEAM = undefined;
|
||||
@ -34,7 +34,7 @@ const CopyIcon = ({ value }: { value: string }) => {
|
||||
|
||||
const GeneralTabPanel = () => {
|
||||
const { id } = useParams();
|
||||
const { projects } = useOutletContext<ProjectsOutletContext>();
|
||||
const { projects } = useOutletContext<ProjectSearchOutletContext>();
|
||||
|
||||
const currentProject = useMemo(() => {
|
||||
return projects.find((project: any) => project.id === id);
|
||||
|
@ -1,5 +1,4 @@
|
||||
import React, { useCallback, useState } from 'react';
|
||||
import toast from 'react-hot-toast';
|
||||
|
||||
import {
|
||||
Select,
|
||||
@ -34,7 +33,8 @@ interface MemberCardProps {
|
||||
isOwner: boolean;
|
||||
isPending: boolean;
|
||||
permissions: string[];
|
||||
handleDeletePendingMember: (id: number) => void;
|
||||
handleDeletePendingMember: (id: string) => void;
|
||||
removeMemberHandler: () => Promise<void>;
|
||||
}
|
||||
|
||||
const MemberCard = ({
|
||||
@ -44,6 +44,7 @@ const MemberCard = ({
|
||||
isPending,
|
||||
permissions,
|
||||
handleDeletePendingMember,
|
||||
removeMemberHandler,
|
||||
}: MemberCardProps) => {
|
||||
const [selectedPermission, setSelectedPermission] = useState(
|
||||
permissions.join('+'),
|
||||
@ -127,7 +128,7 @@ const MemberCard = ({
|
||||
confirmButtonTitle="Yes, Remove member"
|
||||
handleConfirm={() => {
|
||||
setRemoveMemberDialogOpen((preVal) => !preVal);
|
||||
toast.success('Member removed from project');
|
||||
removeMemberHandler();
|
||||
}}
|
||||
color="red"
|
||||
>
|
||||
|
@ -1,4 +1,4 @@
|
||||
import React, { useCallback, useMemo, useState } from 'react';
|
||||
import React, { useCallback, useEffect, useMemo, useState } from 'react';
|
||||
import { useOutletContext, useParams } from 'react-router-dom';
|
||||
import toast, { Toaster } from 'react-hot-toast';
|
||||
|
||||
@ -6,31 +6,60 @@ import { Chip, Button, Typography } from '@material-tailwind/react';
|
||||
|
||||
import MemberCard from './MemberCard';
|
||||
|
||||
import { Member } from '../../../../types/project';
|
||||
import {
|
||||
ProjectMember,
|
||||
ProjectSearchOutletContext,
|
||||
} from '../../../../types/project';
|
||||
import AddMemberDialog from './AddMemberDialog';
|
||||
import { useGQLClient } from '../../../../context/GQLClientContext';
|
||||
|
||||
const FIRST_MEMBER_CARD = 0;
|
||||
|
||||
const MembersTabPanel = () => {
|
||||
const { id } = useParams();
|
||||
const client = useGQLClient();
|
||||
|
||||
const [addmemberDialogOpen, setAddMemberDialogOpen] = useState(false);
|
||||
|
||||
// @ts-expect-error create context type for projects
|
||||
const { projects } = useOutletContext();
|
||||
const { projects } = useOutletContext<ProjectSearchOutletContext>();
|
||||
|
||||
const currentProject = useMemo(() => {
|
||||
return projects.find((project: any) => project.id === id);
|
||||
return projects.find((project) => project.id === id);
|
||||
}, [id]);
|
||||
|
||||
const [updatedMembers, setUpdatedMembers] = useState([
|
||||
...currentProject?.members,
|
||||
]);
|
||||
const [projectMembers, setProjectMembers] = useState<ProjectMember[]>([]);
|
||||
|
||||
const addMemberHandler = useCallback((member: Member) => {
|
||||
setUpdatedMembers((val) => [...val, member]);
|
||||
const addMemberHandler = useCallback((projectMember: ProjectMember) => {
|
||||
setProjectMembers((val) => [...val, projectMember]);
|
||||
toast.success('Invitation sent');
|
||||
}, []);
|
||||
|
||||
const fetchProjectMembers = useCallback(async () => {
|
||||
if (currentProject) {
|
||||
const { projectMembers } = await client.getProjectMembers(
|
||||
currentProject.id,
|
||||
);
|
||||
|
||||
setProjectMembers(projectMembers);
|
||||
}
|
||||
}, [currentProject]);
|
||||
|
||||
const removeMemberHandler = async (projectMemberId: string) => {
|
||||
const { removeMember: isMemberRemoved } =
|
||||
await client.removeMember(projectMemberId);
|
||||
|
||||
if (isMemberRemoved) {
|
||||
toast.success('Member removed from project');
|
||||
await fetchProjectMembers();
|
||||
} else {
|
||||
toast.error('Not able to remove member');
|
||||
}
|
||||
};
|
||||
|
||||
useEffect(() => {
|
||||
fetchProjectMembers();
|
||||
}, []);
|
||||
|
||||
return (
|
||||
<div className="p-2 mb-20">
|
||||
<div className="flex justify-between mb-2">
|
||||
@ -40,7 +69,7 @@ const MembersTabPanel = () => {
|
||||
<Chip
|
||||
className="normal-case ml-3 font-normal"
|
||||
size="sm"
|
||||
value={updatedMembers.length}
|
||||
value={projectMembers.length}
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
@ -53,20 +82,23 @@ const MembersTabPanel = () => {
|
||||
</Button>
|
||||
</div>
|
||||
</div>
|
||||
{updatedMembers.map((member, index) => {
|
||||
{projectMembers.map((projectMember, index) => {
|
||||
return (
|
||||
<MemberCard
|
||||
member={member.member}
|
||||
key={member.id}
|
||||
member={projectMember.member}
|
||||
key={projectMember.id}
|
||||
isFirstCard={index === FIRST_MEMBER_CARD}
|
||||
isOwner={member.member.id === currentProject?.owner.id}
|
||||
isPending={member.name === ''}
|
||||
permissions={member.permissions}
|
||||
handleDeletePendingMember={(id: number) => {
|
||||
setUpdatedMembers(
|
||||
updatedMembers.filter((member) => member.id !== id),
|
||||
isOwner={projectMember.member.id === currentProject?.owner.id}
|
||||
isPending={projectMember.member.name === ''}
|
||||
permissions={projectMember.permissions}
|
||||
handleDeletePendingMember={(id: string) => {
|
||||
setProjectMembers(
|
||||
projectMembers.filter(
|
||||
(projectMember) => projectMember.member.id !== id,
|
||||
),
|
||||
);
|
||||
}}
|
||||
removeMemberHandler={() => removeMemberHandler(projectMember.id)}
|
||||
/>
|
||||
);
|
||||
})}
|
||||
|
@ -13,11 +13,11 @@ const ProjectSearch = () => {
|
||||
|
||||
useEffect(() => {
|
||||
const fetch = async () => {
|
||||
const res = await client.getOrganizations();
|
||||
const { organizations } = await client.getOrganizations();
|
||||
|
||||
// Note: select first organization as organization switching not yet implemented
|
||||
const projects = res.organizations[0]?.projects || [];
|
||||
const orgName = res.organizations[0]?.name || '';
|
||||
const projects = organizations[0].projects || [];
|
||||
const orgName = organizations[0].name || '';
|
||||
|
||||
const updatedProjectsPromises = projects.map(async (project: any) => {
|
||||
const { deployments } = await client.getDeployments(String(project.id));
|
||||
|
@ -5,10 +5,10 @@ import { Link } from 'react-router-dom';
|
||||
import { Button, Typography, Chip } from '@material-tailwind/react';
|
||||
|
||||
import ProjectCard from '../components/projects/ProjectCard';
|
||||
import { ProjectsOutletContext } from '../types/project';
|
||||
import { ProjectSearchOutletContext } from '../types/project';
|
||||
|
||||
const Projects = () => {
|
||||
const { projects } = useOutletContext<ProjectsOutletContext>();
|
||||
const { projects } = useOutletContext<ProjectSearchOutletContext>();
|
||||
|
||||
return (
|
||||
<div>
|
||||
|
@ -5,13 +5,13 @@ import { Button, Typography } from '@material-tailwind/react';
|
||||
|
||||
import HorizontalLine from '../../components/HorizontalLine';
|
||||
import ProjectTabs from '../../components/projects/project/ProjectTabs';
|
||||
import { ProjectsOutletContext } from '../../types/project';
|
||||
import { ProjectSearchOutletContext } from '../../types/project';
|
||||
|
||||
const Project = () => {
|
||||
const { id } = useParams();
|
||||
const navigate = useNavigate();
|
||||
|
||||
const { projects } = useOutletContext<ProjectsOutletContext>();
|
||||
const { projects } = useOutletContext<ProjectSearchOutletContext>();
|
||||
|
||||
const project = useMemo(() => {
|
||||
return projects.find((project) => {
|
||||
|
@ -4,10 +4,10 @@ import { Link, useOutletContext } from 'react-router-dom';
|
||||
import { Button, Typography, Chip } from '@material-tailwind/react';
|
||||
|
||||
import ProjectCard from '../../components/projects/ProjectCard';
|
||||
import { ProjectsOutletContext } from '../../types/project';
|
||||
import { ProjectSearchOutletContext } from '../../types/project';
|
||||
|
||||
const Projects = () => {
|
||||
const { projects } = useOutletContext<ProjectsOutletContext>();
|
||||
const { projects } = useOutletContext<ProjectSearchOutletContext>();
|
||||
|
||||
return (
|
||||
<div>
|
||||
|
@ -2,6 +2,7 @@ export interface ProjectDetails {
|
||||
icon: string;
|
||||
name: string;
|
||||
title: string;
|
||||
owner: Member;
|
||||
organization: string;
|
||||
description: string;
|
||||
url: string;
|
||||
@ -18,13 +19,14 @@ export interface ProjectDetails {
|
||||
};
|
||||
repositoryId: number;
|
||||
repositories: RepositoryDetails[];
|
||||
members: MemberPermission[];
|
||||
members: ProjectMember[];
|
||||
ownerId: number;
|
||||
environmentVariables: EnvironmentVariable[];
|
||||
}
|
||||
|
||||
export interface MemberPermission {
|
||||
id: number;
|
||||
export interface ProjectMember {
|
||||
id: string;
|
||||
member: Member;
|
||||
permissions: string[];
|
||||
}
|
||||
|
||||
@ -103,9 +105,9 @@ export enum Permission {
|
||||
export interface Member {
|
||||
name: string;
|
||||
email: string;
|
||||
id: number;
|
||||
id: string;
|
||||
}
|
||||
|
||||
export interface ProjectsOutletContext {
|
||||
export interface ProjectSearchOutletContext {
|
||||
projects: ProjectDetails[];
|
||||
}
|
||||
|
@ -1,22 +1,37 @@
|
||||
import { ApolloClient, InMemoryCache, NormalizedCacheObject } from '@apollo/client';
|
||||
import { ApolloClient, DefaultOptions, InMemoryCache, NormalizedCacheObject } from '@apollo/client';
|
||||
|
||||
import { getUser, getOrganizations, getDeployments } from './queries';
|
||||
import { getUser, getOrganizations, getDeployments, getProjectMembers } from './queries';
|
||||
import { GetDeploymentsResponse, GetOrganizationsResponse, GetProjectMembersResponse, GetUserResponse, RemoveMemberResponse } from './types';
|
||||
import { removeMember } from './mutations';
|
||||
|
||||
export interface GraphQLConfig {
|
||||
gqlEndpoint: string;
|
||||
}
|
||||
|
||||
// TODO: check options
|
||||
const defaultOptions: DefaultOptions = {
|
||||
watchQuery: {
|
||||
fetchPolicy: 'no-cache',
|
||||
errorPolicy: 'ignore'
|
||||
},
|
||||
query: {
|
||||
fetchPolicy: 'no-cache',
|
||||
errorPolicy: 'all'
|
||||
}
|
||||
};
|
||||
|
||||
export class GQLClient {
|
||||
private client: ApolloClient<NormalizedCacheObject>;
|
||||
|
||||
constructor (config: GraphQLConfig) {
|
||||
this.client = new ApolloClient({
|
||||
uri: config.gqlEndpoint,
|
||||
cache: new InMemoryCache()
|
||||
cache: new InMemoryCache(),
|
||||
defaultOptions
|
||||
});
|
||||
}
|
||||
|
||||
async getUser () : Promise<any> {
|
||||
async getUser () : Promise<GetUserResponse> {
|
||||
const { data } = await this.client.query({
|
||||
query: getUser
|
||||
});
|
||||
@ -24,7 +39,7 @@ export class GQLClient {
|
||||
return data;
|
||||
}
|
||||
|
||||
async getOrganizations () : Promise<any> {
|
||||
async getOrganizations () : Promise<GetOrganizationsResponse> {
|
||||
const { data } = await this.client.query({
|
||||
query: getOrganizations
|
||||
});
|
||||
@ -32,7 +47,7 @@ export class GQLClient {
|
||||
return data;
|
||||
}
|
||||
|
||||
async getDeployments (projectId: string) : Promise<any> {
|
||||
async getDeployments (projectId: string) : Promise<GetDeploymentsResponse> {
|
||||
const { data } = await this.client.query({
|
||||
query: getDeployments,
|
||||
variables: {
|
||||
@ -42,4 +57,26 @@ export class GQLClient {
|
||||
|
||||
return data;
|
||||
}
|
||||
|
||||
async removeMember (memberId: string): Promise<RemoveMemberResponse> {
|
||||
const { data } = await this.client.mutate({
|
||||
mutation: removeMember,
|
||||
variables: {
|
||||
memberId
|
||||
}
|
||||
});
|
||||
|
||||
return data;
|
||||
}
|
||||
|
||||
async getProjectMembers (projectId: string) : Promise<GetProjectMembersResponse> {
|
||||
const { data } = await this.client.query({
|
||||
query: getProjectMembers,
|
||||
variables: {
|
||||
projectId
|
||||
}
|
||||
});
|
||||
|
||||
return data;
|
||||
}
|
||||
}
|
||||
|
7
packages/gql-client/src/mutations.ts
Normal file
7
packages/gql-client/src/mutations.ts
Normal file
@ -0,0 +1,7 @@
|
||||
import { gql } from '@apollo/client';
|
||||
|
||||
export const removeMember = gql`
|
||||
mutation ($memberId: String!) {
|
||||
removeMember(memberId: $memberId)
|
||||
}
|
||||
`;
|
@ -81,3 +81,19 @@ query ($projectId: String!) {
|
||||
}
|
||||
}
|
||||
`;
|
||||
|
||||
export const getProjectMembers = gql`
|
||||
query ($projectId: String!) {
|
||||
projectMembers(projectId: $projectId) {
|
||||
id
|
||||
member {
|
||||
id
|
||||
name
|
||||
email
|
||||
}
|
||||
createdAt
|
||||
updatedAt
|
||||
permissions
|
||||
}
|
||||
}
|
||||
`;
|
||||
|
131
packages/gql-client/src/types.ts
Normal file
131
packages/gql-client/src/types.ts
Normal file
@ -0,0 +1,131 @@
|
||||
// Note: equivalent to types present in GQL schema
|
||||
|
||||
export enum Role {
|
||||
Owner = 'Owner',
|
||||
Maintainer = 'Maintainer',
|
||||
Reader = 'Reader',
|
||||
}
|
||||
|
||||
export enum Permission {
|
||||
View = 'View',
|
||||
Edit = 'Edit',
|
||||
}
|
||||
|
||||
export enum Environment {
|
||||
Production = 'Production',
|
||||
Preview = 'Preview',
|
||||
Development = 'Development',
|
||||
}
|
||||
|
||||
export enum DeploymentStatus {
|
||||
Building = 'Building',
|
||||
Ready = 'Ready',
|
||||
Error = 'Error',
|
||||
}
|
||||
|
||||
export enum DomainStatus {
|
||||
Live = 'Live',
|
||||
Pending = 'Pending',
|
||||
}
|
||||
|
||||
export type EnvironmentVariable = {
|
||||
id: string
|
||||
environments: Environment[]
|
||||
key: string
|
||||
value: string
|
||||
createdAt: string
|
||||
updatedAt: string
|
||||
}
|
||||
|
||||
export type Domain = {
|
||||
id: string
|
||||
branch: string
|
||||
name: string
|
||||
isRedirected: boolean
|
||||
status: DomainStatus
|
||||
createdAt: string
|
||||
updatedAt: string
|
||||
}
|
||||
|
||||
export type Deployment = {
|
||||
id: string
|
||||
domain: Domain
|
||||
branch: string
|
||||
commitHash: string
|
||||
title: string
|
||||
environment: Environment
|
||||
isCurrent: boolean
|
||||
status: DeploymentStatus
|
||||
createdAt: string
|
||||
updatedAt: string
|
||||
}
|
||||
|
||||
export type User = {
|
||||
id: string
|
||||
name: string
|
||||
email: string
|
||||
createdAt: string
|
||||
updatedAt: string
|
||||
}
|
||||
|
||||
export type OrganizationMember = {
|
||||
id: string
|
||||
member: User
|
||||
role: Role
|
||||
createdAt: string
|
||||
updatedAt: string
|
||||
}
|
||||
|
||||
export type ProjectMember = {
|
||||
id: string
|
||||
member: User
|
||||
permissions: Permission[]
|
||||
createdAt: string
|
||||
updatedAt: string
|
||||
}
|
||||
|
||||
export type Project = {
|
||||
id: string
|
||||
owner: User
|
||||
deployments: Deployment[]
|
||||
name: string
|
||||
repository: string
|
||||
prodBranch: string
|
||||
description: string
|
||||
template: string
|
||||
framework: string
|
||||
webhooks: string[]
|
||||
members: ProjectMember[]
|
||||
environmentVariables: EnvironmentVariable[]
|
||||
createdAt: string
|
||||
updatedAt: string
|
||||
}
|
||||
|
||||
export type Organization = {
|
||||
id: string
|
||||
name: string
|
||||
projects: Project[]
|
||||
createdAt: string
|
||||
updatedAt: string
|
||||
members: OrganizationMember[]
|
||||
}
|
||||
|
||||
export type GetProjectMembersResponse = {
|
||||
projectMembers: ProjectMember[]
|
||||
}
|
||||
|
||||
export type RemoveMemberResponse = {
|
||||
removeMember: boolean;
|
||||
}
|
||||
|
||||
export type GetDeploymentsResponse = {
|
||||
deployments: Deployment[]
|
||||
}
|
||||
|
||||
export type GetOrganizationsResponse = {
|
||||
organizations: Organization[]
|
||||
}
|
||||
|
||||
export type GetUserResponse = {
|
||||
user: User
|
||||
}
|
Loading…
Reference in New Issue
Block a user