import { useSession } from "next-auth/react"; import { useRouter, useSearchParams } from "next/navigation"; import { useState } from "react"; import { useLocale } from "@calcom/lib/hooks/useLocale"; import { useParamsWithFallback } from "@calcom/lib/hooks/useParamsWithFallback"; import { MembershipRole } from "@calcom/prisma/enums"; import type { RouterOutputs } from "@calcom/trpc/react"; import { trpc } from "@calcom/trpc/react"; import { Button, Meta, showToast, TextField } from "@calcom/ui"; import { Plus } from "@calcom/ui/components/icon"; import { getLayout } from "../../../settings/layouts/SettingsLayout"; import DisableTeamImpersonation from "../components/DisableTeamImpersonation"; import InviteLinkSettingsModal from "../components/InviteLinkSettingsModal"; import MakeTeamPrivateSwitch from "../components/MakeTeamPrivateSwitch"; import MemberInvitationModal from "../components/MemberInvitationModal"; import MemberListItem from "../components/MemberListItem"; import TeamInviteList from "../components/TeamInviteList"; type Team = RouterOutputs["viewer"]["teams"]["get"]; interface MembersListProps { team: Team | undefined; } const checkIfExist = (comp: string, query: string) => comp.toLowerCase().replace(/\s+/g, "").includes(query.toLowerCase().replace(/\s+/g, "")); function MembersList(props: MembersListProps) { const { team } = props; const { t } = useLocale(); const [query, setQuery] = useState(""); const members = team?.members; const membersList = members ? members && query === "" ? members : members.filter((member) => { const email = member.email ? checkIfExist(member.email, query) : false; const username = member.username ? checkIfExist(member.username, query) : false; const name = member.name ? checkIfExist(member.name, query) : false; return email || username || name; }) : undefined; return (
setQuery(e.target.value)} value={query} placeholder={`${t("search")}...`} /> {membersList?.length && team ? (
    {membersList.map((member) => { return ; })}
) : null}
); } const MembersView = () => { const searchParams = useSearchParams(); const { t, i18n } = useLocale(); const router = useRouter(); const session = useSession(); const utils = trpc.useContext(); const params = useParamsWithFallback(); const teamId = Number(params.id); const showDialog = searchParams?.get("inviteModal") === "true"; const [showMemberInvitationModal, setShowMemberInvitationModal] = useState(showDialog); const [showInviteLinkSettingsModal, setInviteLinkSettingsModal] = useState(false); const { data: orgMembersNotInThisTeam, isLoading: isOrgListLoading } = trpc.viewer.organizations.getMembers.useQuery( { teamIdToExclude: teamId, distinctUser: true, }, { enabled: searchParams !== null, } ); const { data: team, isLoading: isTeamsLoading } = trpc.viewer.teams.get.useQuery( { teamId }, { onError: () => { router.push("/settings"); }, } ); const isLoading = isOrgListLoading || isTeamsLoading; const inviteMemberMutation = trpc.viewer.teams.inviteMember.useMutation(); const isInviteOpen = !team?.membership.accepted; const isAdmin = team && (team.membership.role === MembershipRole.OWNER || team.membership.role === MembershipRole.ADMIN); return ( <> setShowMemberInvitationModal(true)} data-testid="new-member-button"> {t("add")} ) : ( <> ) } /> {!isLoading && ( <>
{team && ( <> {isInviteOpen && ( )} )} {((team?.isPrivate && isAdmin) || !team?.isPrivate) && ( <>
)} {team && session.data && ( )} {team && isAdmin && ( <>
)}
{showMemberInvitationModal && team && ( setShowMemberInvitationModal(false)} onSubmit={(values, resetFields) => { inviteMemberMutation.mutate( { teamId, language: i18n.language, role: values.role, usernameOrEmail: values.emailOrUsername, sendEmailInvitation: values.sendInviteEmail, }, { onSuccess: async (data) => { await utils.viewer.teams.get.invalidate(); setShowMemberInvitationModal(false); if (data.sendEmailInvitation) { if (Array.isArray(data.usernameOrEmail)) { showToast( t("email_invite_team_bulk", { userCount: data.usernameOrEmail.length, }), "success" ); resetFields(); } else { showToast( t("email_invite_team", { email: data.usernameOrEmail, }), "success" ); } } }, onError: (error) => { showToast(error.message, "error"); }, } ); }} onSettingsOpen={() => { setShowMemberInvitationModal(false); setInviteLinkSettingsModal(true); }} /> )} {showInviteLinkSettingsModal && team?.inviteToken && ( { setInviteLinkSettingsModal(false); setShowMemberInvitationModal(true); }} /> )} )} ); }; MembersView.getLayout = getLayout; export default MembersView;