mirror of
https://github.com/supabase/supabase.git
synced 2026-05-07 17:30:25 -04:00
7f5865872a
## Context Enforce `noUnusedLocals` and `noUnusedParameters` in tsconfig.json + fix all related issues
169 lines
6.3 KiB
TypeScript
169 lines
6.3 KiB
TypeScript
import { ChevronDown, Loader2, RefreshCw, Trash } from 'lucide-react'
|
|
import Link from 'next/link'
|
|
import { useRouter } from 'next/router'
|
|
import { forwardRef, useCallback, useState } from 'react'
|
|
import { toast } from 'sonner'
|
|
import {
|
|
Button,
|
|
DropdownMenu,
|
|
DropdownMenuContent,
|
|
DropdownMenuItem,
|
|
DropdownMenuSeparator,
|
|
DropdownMenuTrigger,
|
|
} from 'ui'
|
|
import ConfirmationModal from 'ui-patterns/Dialogs/ConfirmationModal'
|
|
|
|
import {
|
|
IntegrationConnection,
|
|
IntegrationConnectionProps,
|
|
} from '@/components/interfaces/Integrations/VercelGithub/IntegrationPanels'
|
|
import { ButtonTooltip } from '@/components/ui/ButtonTooltip'
|
|
import { useIntegrationsVercelConnectionSyncEnvsMutation } from '@/data/integrations/integrations-vercel-connection-sync-envs-mutation'
|
|
import type { IntegrationProjectConnection } from '@/data/integrations/integrations.types'
|
|
import { useProjectDetailQuery } from '@/data/projects/project-detail-query'
|
|
import { useSelectedOrganizationQuery } from '@/hooks/misc/useSelectedOrganization'
|
|
|
|
interface IntegrationConnectionItemProps extends IntegrationConnectionProps {
|
|
disabled?: boolean
|
|
onDeleteConnection: (connection: IntegrationProjectConnection) => void | Promise<void>
|
|
}
|
|
|
|
export const IntegrationConnectionItem = forwardRef<HTMLLIElement, IntegrationConnectionItemProps>(
|
|
({ disabled, onDeleteConnection, ...props }, _ref) => {
|
|
const router = useRouter()
|
|
const { data: org } = useSelectedOrganizationQuery()
|
|
|
|
const { type, connection } = props
|
|
const { data: project } = useProjectDetailQuery({ ref: connection.supabase_project_ref })
|
|
const isBranchingEnabled = project?.is_branch_enabled === true
|
|
|
|
const [isOpen, setIsOpen] = useState(false)
|
|
const [isDeleting, setIsDeleting] = useState(false)
|
|
const [dropdownVisible, setDropdownVisible] = useState(false)
|
|
|
|
const onConfirm = useCallback(async () => {
|
|
try {
|
|
setIsDeleting(true)
|
|
await onDeleteConnection(connection)
|
|
} catch (error) {
|
|
// [Joshen] No need for error handler
|
|
} finally {
|
|
setIsDeleting(false)
|
|
setIsOpen(false)
|
|
}
|
|
}, [connection, onDeleteConnection])
|
|
|
|
const onCancel = useCallback(() => {
|
|
setIsOpen(false)
|
|
}, [])
|
|
|
|
const { mutate: syncEnvs, isPending: isSyncEnvLoading } =
|
|
useIntegrationsVercelConnectionSyncEnvsMutation({
|
|
onSuccess: () => {
|
|
toast.success('Successfully synced environment variables')
|
|
setDropdownVisible(false)
|
|
},
|
|
})
|
|
|
|
const onReSyncEnvVars = useCallback(async () => {
|
|
syncEnvs({ connectionId: connection.id })
|
|
}, [connection, syncEnvs])
|
|
|
|
const projectIntegrationUrl = `/project/[ref]/settings/integrations`
|
|
|
|
return (
|
|
<>
|
|
<IntegrationConnection
|
|
showNode={false}
|
|
actions={
|
|
disabled ? (
|
|
<ButtonTooltip
|
|
disabled
|
|
iconRight={<ChevronDown size={14} />}
|
|
type="default"
|
|
tooltip={{
|
|
content: {
|
|
side: 'bottom',
|
|
text: 'You need additional permissions to manage this connection',
|
|
},
|
|
}}
|
|
>
|
|
Manage
|
|
</ButtonTooltip>
|
|
) : (
|
|
<DropdownMenu
|
|
open={dropdownVisible}
|
|
onOpenChange={() => setDropdownVisible(!dropdownVisible)}
|
|
modal={false}
|
|
>
|
|
<DropdownMenuTrigger asChild>
|
|
<Button iconRight={<ChevronDown size={14} />} type="default">
|
|
<span>Manage</span>
|
|
</Button>
|
|
</DropdownMenuTrigger>
|
|
<DropdownMenuContent side="bottom" align="end">
|
|
{router.pathname !== projectIntegrationUrl && (
|
|
<DropdownMenuItem asChild>
|
|
<Link
|
|
href={projectIntegrationUrl.replace(
|
|
'[ref]',
|
|
connection.supabase_project_ref
|
|
)}
|
|
>
|
|
Configure connection
|
|
</Link>
|
|
</DropdownMenuItem>
|
|
)}
|
|
{type === 'Vercel' && org?.managed_by !== 'vercel-marketplace' && (
|
|
<DropdownMenuItem
|
|
className="space-x-2"
|
|
onSelect={(event) => {
|
|
event.preventDefault()
|
|
onReSyncEnvVars()
|
|
}}
|
|
disabled={isSyncEnvLoading}
|
|
>
|
|
{isSyncEnvLoading ? (
|
|
<Loader2 className="animate-spin" size={14} />
|
|
) : (
|
|
<RefreshCw size={14} />
|
|
)}
|
|
<p>Resync environment variables</p>
|
|
</DropdownMenuItem>
|
|
)}
|
|
{((type === 'Vercel' && org?.managed_by !== 'vercel-marketplace') ||
|
|
router.pathname !== projectIntegrationUrl) && <DropdownMenuSeparator />}
|
|
<DropdownMenuItem className="space-x-2" onSelect={() => setIsOpen(true)}>
|
|
<Trash size={14} />
|
|
<p>Delete connection</p>
|
|
</DropdownMenuItem>
|
|
</DropdownMenuContent>
|
|
</DropdownMenu>
|
|
)
|
|
}
|
|
{...props}
|
|
/>
|
|
|
|
<ConfirmationModal
|
|
variant="destructive"
|
|
size={type === 'GitHub' && isBranchingEnabled ? 'medium' : 'small'}
|
|
visible={isOpen}
|
|
title={`Confirm to delete ${type} connection`}
|
|
confirmLabel="Delete connection"
|
|
onCancel={onCancel}
|
|
onConfirm={onConfirm}
|
|
loading={isDeleting}
|
|
>
|
|
<p className="text-sm text-foreground-light">
|
|
{type === 'Vercel'
|
|
? 'Deleting this Vercel connection will stop syncing environment variables to your Vercel project. Existing environment variables will remain unchanged.'
|
|
: 'Deleting this GitHub connection will stop automatic creation and merging of preview branches. Existing preview branches will remain unchanged.'}
|
|
</p>
|
|
</ConfirmationModal>
|
|
</>
|
|
)
|
|
}
|
|
)
|
|
|
|
IntegrationConnectionItem.displayName = 'IntegrationConnectionItem'
|