* Update Supabase docs URLs to use env variable Co-authored-by: a <a@alaisteryoung.com> * Refactor: Use DOCS_URL constant for documentation links This change centralizes documentation links using a new DOCS_URL constant, improving maintainability and consistency. Co-authored-by: a <a@alaisteryoung.com> * Refactor: Use DOCS_URL constant for all documentation links This change replaces hardcoded documentation URLs with a centralized constant, improving maintainability and consistency. Co-authored-by: a <a@alaisteryoung.com> * replace more instances * ci: Autofix updates from GitHub workflow * remaining instances * fix duplicate useRouter --------- Co-authored-by: Cursor Agent <cursoragent@cursor.com> Co-authored-by: alaister <10985857+alaister@users.noreply.github.com>
181 lines
6.6 KiB
TypeScript
181 lines
6.6 KiB
TypeScript
import { AlertCircle, HelpCircle } from 'lucide-react'
|
|
|
|
import { useParams } from 'common'
|
|
import AlertError from 'components/ui/AlertError'
|
|
import { GenericSkeletonLoader } from 'components/ui/ShimmeringLoader'
|
|
import { useOrganizationRolesV2Query } from 'data/organization-members/organization-roles-query'
|
|
import { useOrganizationMembersQuery } from 'data/organizations/organization-members-query'
|
|
import { DOCS_URL } from 'lib/constants'
|
|
import { useProfile } from 'lib/profile'
|
|
import { partition } from 'lodash'
|
|
import { useMemo } from 'react'
|
|
import {
|
|
Button,
|
|
Card,
|
|
Loading,
|
|
Table,
|
|
TableBody,
|
|
TableCell,
|
|
TableHead,
|
|
TableHeader,
|
|
TableRow,
|
|
Tooltip,
|
|
TooltipContent,
|
|
TooltipTrigger,
|
|
} from 'ui'
|
|
import { Admonition } from 'ui-patterns'
|
|
import { MemberRow } from './MemberRow'
|
|
|
|
export interface MembersViewProps {
|
|
searchString: string
|
|
}
|
|
|
|
const MembersView = ({ searchString }: MembersViewProps) => {
|
|
const { slug } = useParams()
|
|
const { profile } = useProfile()
|
|
|
|
const {
|
|
data: members = [],
|
|
error: membersError,
|
|
isLoading: isLoadingMembers,
|
|
isError: isErrorMembers,
|
|
isSuccess: isSuccessMembers,
|
|
} = useOrganizationMembersQuery({ slug })
|
|
const {
|
|
data: roles,
|
|
error: rolesError,
|
|
isSuccess: isSuccessRoles,
|
|
isError: isErrorRoles,
|
|
} = useOrganizationRolesV2Query({
|
|
slug,
|
|
})
|
|
|
|
const filteredMembers = useMemo(() => {
|
|
return !searchString
|
|
? members
|
|
: members.filter((member) => {
|
|
if (member.invited_at) {
|
|
return member.primary_email?.includes(searchString)
|
|
}
|
|
if (member.gotrue_id) {
|
|
return (
|
|
member.username.includes(searchString) || member.primary_email?.includes(searchString)
|
|
)
|
|
}
|
|
})
|
|
}, [members, searchString])
|
|
|
|
const [[user], otherMembers] = partition(
|
|
filteredMembers,
|
|
(m) => m.gotrue_id === profile?.gotrue_id
|
|
)
|
|
const sortedMembers = otherMembers.sort((a, b) =>
|
|
(a.primary_email ?? '').localeCompare(b.primary_email ?? '')
|
|
)
|
|
|
|
const userMember = members.find((m) => m.gotrue_id === profile?.gotrue_id)
|
|
const orgScopedRoleIds = (roles?.org_scoped_roles ?? []).map((r) => r.id)
|
|
const isOrgScopedRole = orgScopedRoleIds.includes(userMember?.role_ids?.[0] ?? -1)
|
|
|
|
return (
|
|
<>
|
|
{isLoadingMembers && <GenericSkeletonLoader />}
|
|
|
|
{isErrorMembers && (
|
|
<AlertError error={membersError} subject="Failed to retrieve organization members" />
|
|
)}
|
|
|
|
{isErrorRoles && (
|
|
<AlertError error={rolesError} subject="Failed to retrieve organization roles" />
|
|
)}
|
|
|
|
{isSuccessMembers && (
|
|
<div className="rounded w-full overflow-hidden overflow-x-scroll">
|
|
<Card>
|
|
<Loading active={!filteredMembers}>
|
|
<Table>
|
|
<TableHeader>
|
|
<TableRow>
|
|
<TableHead key="header-user">User</TableHead>
|
|
<TableHead key="header-status" className="w-24" />
|
|
<TableHead key="header-mfa" className="text-center w-32">
|
|
Enabled MFA
|
|
</TableHead>
|
|
<TableHead key="header-role" className="flex items-center space-x-1">
|
|
<span>Role</span>
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<Button asChild type="text" className="px-1">
|
|
<a
|
|
target="_blank"
|
|
rel="noreferrer"
|
|
href={`${DOCS_URL}/guides/platform/access-control`}
|
|
>
|
|
<HelpCircle size={14} className="text-foreground-light" />
|
|
</a>
|
|
</Button>
|
|
</TooltipTrigger>
|
|
<TooltipContent side="bottom">
|
|
How to configure access control?
|
|
</TooltipContent>
|
|
</Tooltip>
|
|
</TableHead>
|
|
<TableHead key="header-action" />
|
|
</TableRow>
|
|
</TableHeader>
|
|
|
|
<TableBody>
|
|
{[
|
|
...(isSuccessRoles && isSuccessMembers && !isOrgScopedRole
|
|
? [
|
|
<TableRow key="project-scope-notice">
|
|
<TableCell colSpan={12} className="!p-0">
|
|
<Admonition
|
|
type="note"
|
|
title="You are currently assigned with project scoped roles in this organization"
|
|
description="All the members within the organization will not be visible to you"
|
|
className="m-0 bg-alternative border-0 rounded-none"
|
|
/>
|
|
</TableCell>
|
|
</TableRow>,
|
|
]
|
|
: []),
|
|
...(!!user ? [<MemberRow key={user.gotrue_id} member={user} />] : []),
|
|
...sortedMembers.map((member) => (
|
|
<MemberRow key={member.gotrue_id} member={member} />
|
|
)),
|
|
...(searchString.length > 0 && filteredMembers.length === 0
|
|
? [
|
|
<TableRow key="no-results" className="bg-panel-secondary-light">
|
|
<TableCell colSpan={12}>
|
|
<div className="flex items-center space-x-3 opacity-75">
|
|
<AlertCircle size={16} strokeWidth={2} />
|
|
<p className="text-foreground-light">
|
|
No users matched the search query "{searchString}"
|
|
</p>
|
|
</div>
|
|
</TableCell>
|
|
</TableRow>,
|
|
]
|
|
: []),
|
|
<TableRow key="footer" className="bg-panel-secondary-light">
|
|
<TableCell colSpan={12}>
|
|
<p className="text-foreground-light">
|
|
{searchString ? `${filteredMembers.length} of ` : ''}
|
|
{members.length || '0'} {members.length == 1 ? 'user' : 'users'}
|
|
</p>
|
|
</TableCell>
|
|
</TableRow>,
|
|
]}
|
|
</TableBody>
|
|
</Table>
|
|
</Loading>
|
|
</Card>
|
|
</div>
|
|
)}
|
|
</>
|
|
)
|
|
}
|
|
|
|
export default MembersView
|