Files
supabase/apps/studio/components/layouts/AppLayout/OrganizationDropdown.tsx
Terry Sutton 7c678b9db8 Update new project flow to add a warning before redirecting (#36349)
* Update new project flow to add a warning before redirecting

* Simplify the logic for picking an org.

* Small UI tweaks

* Readd the alert about a not found org.

* Link to all orgs if slug is undefined

* Nit

---------

Co-authored-by: Ivan Vasilov <vasilov.ivan@gmail.com>
Co-authored-by: Joshen Lim <joshenlimek@gmail.com>
2025-06-17 09:55:05 +02:00

149 lines
5.5 KiB
TypeScript

import { Boxes, Check, ChevronsUpDown, Plus } from 'lucide-react'
import Link from 'next/link'
import { useRouter } from 'next/router'
import { useState } from 'react'
import { useParams } from 'common'
import PartnerIcon from 'components/ui/PartnerIcon'
import ShimmeringLoader from 'components/ui/ShimmeringLoader'
import { useOrganizationsQuery } from 'data/organizations/organizations-query'
import { useIsFeatureEnabled } from 'hooks/misc/useIsFeatureEnabled'
import { useSelectedOrganization } from 'hooks/misc/useSelectedOrganization'
import {
Badge,
Button,
CommandEmpty_Shadcn_,
CommandGroup_Shadcn_,
CommandInput_Shadcn_,
CommandItem_Shadcn_,
CommandList_Shadcn_,
CommandSeparator_Shadcn_,
Command_Shadcn_,
PopoverContent_Shadcn_,
PopoverTrigger_Shadcn_,
Popover_Shadcn_,
ScrollArea,
cn,
} from 'ui'
export const OrganizationDropdown = () => {
const router = useRouter()
const { slug: routeSlug } = useParams()
const selectedOrganization = useSelectedOrganization()
const { data: organizations, isLoading: isLoadingOrganizations } = useOrganizationsQuery()
const organizationCreationEnabled = useIsFeatureEnabled('organizations:create')
const slug = selectedOrganization?.slug
const orgName = selectedOrganization?.name
const [open, setOpen] = useState(false)
if (isLoadingOrganizations) {
return <ShimmeringLoader className="w-[90px]" />
}
return (
<>
<Link
href={slug ? `/org/${slug}` : '/organizations'}
className="flex items-center gap-2 flex-shrink-0 text-sm"
>
<Boxes size={14} strokeWidth={1.5} className="text-foreground-lighter" />
<span
className={cn(
'max-w-32 lg:max-w-none truncate hidden md:block',
!!selectedOrganization ? 'text-foreground' : 'text-foreground-lighter'
)}
>
{orgName ?? 'Select an organization'}
</span>
{!!selectedOrganization && (
<Badge variant="default">{selectedOrganization?.plan.name}</Badge>
)}
</Link>
<Popover_Shadcn_ open={open} onOpenChange={setOpen} modal={false}>
<PopoverTrigger_Shadcn_ asChild>
<Button
type="text"
className={cn('px-1.5 py-4 [&_svg]:w-5 [&_svg]:h-5 ml-1')}
iconRight={<ChevronsUpDown strokeWidth={1.5} />}
/>
</PopoverTrigger_Shadcn_>
<PopoverContent_Shadcn_ className="p-0" side="bottom" align="start">
<Command_Shadcn_>
<CommandInput_Shadcn_ placeholder="Find organization..." />
<CommandList_Shadcn_>
<CommandEmpty_Shadcn_>No organizations found</CommandEmpty_Shadcn_>
<CommandGroup_Shadcn_>
<ScrollArea className={(organizations || []).length > 7 ? 'h-[210px]' : ''}>
{organizations?.map((org) => {
const href = !!routeSlug
? router.pathname.replace('[slug]', org.slug)
: `/org/${org.slug}`
return (
<CommandItem_Shadcn_
key={org.slug}
value={`${org.name.replaceAll('"', '')} - ${org.slug}`}
className="cursor-pointer w-full"
onSelect={() => {
setOpen(false)
router.push(href)
}}
onClick={() => setOpen(false)}
>
<Link href={href} className="w-full flex items-center justify-between">
<div className="flex items-center gap-2">
<span>{org.name}</span>
<PartnerIcon organization={org} />
</div>
{org.slug === slug && <Check size={16} />}
</Link>
</CommandItem_Shadcn_>
)
})}
</ScrollArea>
</CommandGroup_Shadcn_>
<CommandSeparator_Shadcn_ />
<CommandGroup_Shadcn_>
<CommandItem_Shadcn_
className="cursor-pointer w-full"
onSelect={(e) => {
setOpen(false)
router.push(`/organizations`)
}}
onClick={() => setOpen(false)}
>
<Link href="/organizations" className="flex items-center gap-2 w-full">
<p>All Organizations</p>
</Link>
</CommandItem_Shadcn_>
</CommandGroup_Shadcn_>
{organizationCreationEnabled && (
<>
<CommandSeparator_Shadcn_ />
<CommandGroup_Shadcn_>
<CommandItem_Shadcn_
className="cursor-pointer w-full"
onSelect={(e) => {
setOpen(false)
router.push(`/new`)
}}
onClick={() => setOpen(false)}
>
<Link href="/new" className="flex items-center gap-2 w-full">
<Plus size={14} strokeWidth={1.5} />
<p>New organization</p>
</Link>
</CommandItem_Shadcn_>
</CommandGroup_Shadcn_>
</>
)}
</CommandList_Shadcn_>
</Command_Shadcn_>
</PopoverContent_Shadcn_>
</Popover_Shadcn_>
</>
)
}