mirror of
https://github.com/supabase/supabase.git
synced 2026-05-12 13:19:37 +08:00
This PR shows installed marketplace apps in the left pane on the integrations page and shows the `Installed` badge on the card. `Official` badge is not shown on marketplace listings. <img width="1652" height="892" alt="image" src="https://github.com/user-attachments/assets/16c0a218-1658-426e-9b5b-d28e44c5c3b6" /> <!-- This is an auto-generated comment: release notes by coderabbit.ai --> ## Summary by CodeRabbit * **Bug Fixes** * Fixed excessive spacing on the integrations page * Enhanced integration installation detection for improved accuracy * **Style** * Updated official badge display on integration cards <!-- end of auto-generated comment: release notes by coderabbit.ai -->
116 lines
4.1 KiB
TypeScript
116 lines
4.1 KiB
TypeScript
import { BadgeCheck } from 'lucide-react'
|
|
import Image from 'next/image'
|
|
import Link from 'next/link'
|
|
import { Badge, Card, CardContent, cn } from 'ui'
|
|
import { ShimmeringLoader } from 'ui-patterns/ShimmeringLoader'
|
|
|
|
import { IntegrationDefinition } from './Integrations.constants'
|
|
import { useSelectedProjectQuery } from '@/hooks/misc/useSelectedProject'
|
|
import { BASE_PATH } from '@/lib/constants'
|
|
|
|
type IntegrationCardProps = IntegrationDefinition & {
|
|
isInstalled?: boolean
|
|
featured?: boolean
|
|
image?: string
|
|
}
|
|
|
|
const INTEGRATION_CARD_STYLE = cn(
|
|
'w-full h-full bg-surface-100 hover:bg-surface-200 hover:border-strong',
|
|
'border border-border rounded-md ease-out duration-200 transition-all'
|
|
)
|
|
|
|
export const IntegrationLoadingCard = () => {
|
|
return (
|
|
<div className={cn(INTEGRATION_CARD_STYLE, 'pl-5 pr-6 py-3 gap-3 inline-flex h-[110px]')}>
|
|
<div className="w-10 h-10 relative">
|
|
<ShimmeringLoader className="w-full h-full bg-white border rounded-md" />
|
|
</div>
|
|
<div className="grow basis-0 w-full flex flex-col justify-between items-start gap-y-2">
|
|
<div className="w-full flex-col justify-start items-start gap-y-1 flex">
|
|
<ShimmeringLoader className="w-3/4 py-2.5" />
|
|
<ShimmeringLoader className="w-full py-2.5" />
|
|
</div>
|
|
</div>
|
|
</div>
|
|
)
|
|
}
|
|
|
|
export const IntegrationCard = ({
|
|
id,
|
|
listingId,
|
|
status,
|
|
name,
|
|
icon,
|
|
description,
|
|
isInstalled,
|
|
featured = false,
|
|
image,
|
|
}: IntegrationCardProps) => {
|
|
const { data: project } = useSelectedProjectQuery()
|
|
const shouldShowOfficialBadge = !listingId
|
|
|
|
if (featured) {
|
|
return (
|
|
<Link href={`/project/${project?.ref}/integrations/${id}/overview`} className="h-full">
|
|
<Card className="h-full">
|
|
{/* Full-width image/icon at the top */}
|
|
<div className="w-full h-24 bg-surface-400 rounded-t-md flex items-center justify-center overflow-hidden relative">
|
|
{image ? (
|
|
<Image
|
|
fill
|
|
src={`${BASE_PATH}/${image}`}
|
|
alt={`${name} integration`}
|
|
className="w-full h-full object-cover invert dark:invert-0"
|
|
objectFit="cover"
|
|
/>
|
|
) : (
|
|
<div className="w-12 h-12 text-foreground relative">
|
|
{icon({ className: 'w-full h-full text-foreground' })}
|
|
</div>
|
|
)}
|
|
</div>
|
|
<CardContent className="p-6 px-4">
|
|
<div className="flex-col justify-start items-center text-center gap-y-0.5 flex">
|
|
<h3>{name}</h3>
|
|
<p className="text-foreground-light text-sm">{description}</p>
|
|
<div className="flex items-center gap-x-1 mt-4">
|
|
{status && <Badge variant="warning">{status}</Badge>}
|
|
{shouldShowOfficialBadge && <Badge>Official</Badge>}
|
|
</div>
|
|
</div>
|
|
</CardContent>
|
|
</Card>
|
|
</Link>
|
|
)
|
|
}
|
|
|
|
return (
|
|
<Link href={`/project/${project?.ref}/integrations/${id}/overview`} className="h-full">
|
|
<Card className="h-full">
|
|
<CardContent className="flex flex-col p-4 @2xl:p-6 h-full">
|
|
<div className="flex items-start justify-between mb-4">
|
|
<div className="shrink-0 w-10 h-10 relative bg-white border rounded-md flex items-center justify-center">
|
|
{icon()}
|
|
</div>
|
|
{isInstalled && (
|
|
<div className="flex items-center gap-x-1">
|
|
<BadgeCheck size={14} className="text-brand-link" />
|
|
<span className="text-brand-link text-xs">Installed</span>
|
|
</div>
|
|
)}
|
|
</div>
|
|
<div className="flex-col justify-start items-start gap-y-0.5 flex flex-1">
|
|
<h3 className="text-foreground text-sm">{name}</h3>
|
|
|
|
<p className="text-foreground-light text-xs flex-1">{description}</p>
|
|
<div className="flex items-center gap-x-1 mt-4">
|
|
{status && <Badge variant="warning">{status}</Badge>}
|
|
{shouldShowOfficialBadge && <Badge>Official</Badge>}
|
|
</div>
|
|
</div>
|
|
</CardContent>
|
|
</Card>
|
|
</Link>
|
|
)
|
|
}
|