Files
supabase/apps/studio/components/ui/PartnerIcon.tsx
Ivan Vasilov 308cd791a2 chore: Prep work for migrating to Tailwind v4 (#45285)
This PR preps the monorepo for a migration to Tailwind v4:
- Bump all Tailwind dependencies and libraries to the latest possible
version, while still compatible with Tailwind 3.
- Cleans up obsolete Tailwind 3 specific options and configs.
- Cleans up unused CSS files and fixes the CSS imports.
- Migrates all `important` uses in `@apply` lines to using the `!`
prefix.
- Move `typography.css` to the `config` package and import it from the
apps.
- Migrated all occurrences of `flex-grow`, `flex-shrink`,
`overflow-clip` and `overflow-ellipsis` since they're deprecated and
will be removed in Tailwind 4.
- Make the default theme object typesafe in the `ui` package.
- Migrate all `bg-opacity`, `border-opacity`, `ring-opacity` and
`divider-opacity` to the new format where they're declared as part of
the property color.
- Bump and unify all imports of `postcss` dependency.
2026-04-28 11:33:53 +02:00

171 lines
7.5 KiB
TypeScript

import { cn, Tooltip, TooltipContent, TooltipTrigger } from 'ui'
import { PARTNER_TO_NAME } from './PartnerManagedResource'
import { MANAGED_BY } from '@/lib/constants/infrastructure'
import type { Organization } from '@/types'
interface PartnerIconProps {
organization: Pick<Organization, 'managed_by'>
showTooltip?: boolean
tooltipText?: string
size?: 'small' | 'medium' | 'large'
}
export function getDefaultPartnerTooltipText(
managedBy: Pick<Organization, 'managed_by'>['managed_by']
) {
switch (managedBy) {
case MANAGED_BY.VERCEL_MARKETPLACE:
return 'Managed via Vercel Marketplace'
case MANAGED_BY.AWS_MARKETPLACE:
return 'Billed via AWS Marketplace'
case MANAGED_BY.STRIPE_PROJECTS:
return 'Connected to Stripe'
default:
return `Managed by ${PARTNER_TO_NAME[managedBy]}`
}
}
export function getPartnerTooltipText({
managedBy,
tooltipText,
}: {
managedBy: Pick<Organization, 'managed_by'>['managed_by']
tooltipText?: string
}) {
return tooltipText ?? getDefaultPartnerTooltipText(managedBy)
}
function getPartnerIcon(
organization: Pick<Organization, 'managed_by'>,
size: 'small' | 'medium' | 'large'
) {
switch (organization.managed_by) {
case MANAGED_BY.VERCEL_MARKETPLACE:
return (
<svg
className={cn(
size === 'small' && 'w-2.5 h-2.5',
size === 'medium' && 'w-3.5 h-3.5',
size === 'large' && 'w-5 h-5'
)}
viewBox="0 0 76 65"
fill="none"
xmlns="http://www.w3.org/2000/svg"
>
<path d="M37.5274 0L75.0548 65H0L37.5274 0Z" fill="hsl(var(--foreground-default) / 1)" />
</svg>
)
case MANAGED_BY.AWS_MARKETPLACE:
return (
<svg
className={cn(
size === 'small' && 'w-3.5 h-3.5',
size === 'medium' && 'w-5 h-5',
size === 'large' && 'w-6 h-6'
)}
viewBox="0 0 20 20"
fill="none"
xmlns="http://www.w3.org/2000/svg"
>
<g fill="hsl(var(--foreground-default) / 1)" transform="scale(1.25)">
<path d="M4.51 7.687c0 .197.02.357.058.475.042.117.096.245.17.384a.233.233 0 01.037.123c0 .053-.032.107-.1.16l-.336.224a.255.255 0 01-.138.048c-.054 0-.107-.026-.16-.074a1.652 1.652 0 01-.192-.251 4.137 4.137 0 01-.164-.315c-.416.491-.937.737-1.565.737-.447 0-.804-.129-1.064-.385-.261-.256-.394-.598-.394-1.025 0-.454.16-.822.484-1.1.325-.278.756-.416 1.304-.416.18 0 .367.016.564.042.197.027.4.07.612.118v-.39c0-.406-.085-.689-.25-.854-.17-.166-.458-.246-.868-.246-.186 0-.377.022-.574.07a4.23 4.23 0 00-.575.181 1.525 1.525 0 01-.186.07.326.326 0 01-.085.016c-.075 0-.112-.054-.112-.166v-.262c0-.085.01-.15.037-.186a.399.399 0 01.15-.113c.185-.096.409-.176.67-.24.26-.07.537-.101.83-.101.633 0 1.096.144 1.394.432.293.288.442.726.442 1.314v1.73h.01zm-2.161.811c.175 0 .356-.032.548-.096.192-.064.362-.182.505-.342a.848.848 0 00.181-.341c.032-.129.054-.283.054-.465V7.03a4.43 4.43 0 00-.49-.09 3.996 3.996 0 00-.5-.033c-.357 0-.617.07-.793.214-.176.144-.26.347-.26.614 0 .25.063.437.196.566.128.133.314.197.559.197zm4.273.577c-.096 0-.16-.016-.202-.054-.043-.032-.08-.106-.112-.208l-1.25-4.127a.938.938 0 01-.048-.214c0-.085.042-.133.127-.133h.522c.1 0 .17.016.207.053.043.032.075.107.107.208l.894 3.535.83-3.535c.026-.106.058-.176.101-.208a.365.365 0 01.213-.053h.426c.1 0 .17.016.212.053.043.032.08.107.102.208l.84 3.578.92-3.578a.459.459 0 01.107-.208.347.347 0 01.208-.053h.495c.085 0 .133.043.133.133 0 .027-.006.054-.01.086a.768.768 0 01-.038.133l-1.283 4.127c-.031.107-.069.177-.111.209a.34.34 0 01-.203.053h-.457c-.101 0-.17-.016-.213-.053-.043-.038-.08-.107-.101-.214L8.213 5.37l-.82 3.439c-.026.107-.058.176-.1.213-.043.038-.118.054-.213.054h-.458zm6.838.144a3.51 3.51 0 01-.82-.096c-.266-.064-.473-.134-.612-.214-.085-.048-.143-.101-.165-.15a.38.38 0 01-.031-.149v-.272c0-.112.042-.166.122-.166a.3.3 0 01.096.016c.032.011.08.032.133.054.18.08.378.144.585.187.213.042.42.064.633.064.336 0 .596-.059.777-.176a.575.575 0 00.277-.508.52.52 0 00-.144-.373c-.095-.102-.276-.193-.537-.278l-.772-.24c-.388-.123-.676-.305-.851-.545a1.275 1.275 0 01-.266-.774c0-.224.048-.422.143-.593.096-.17.224-.32.384-.438.16-.122.34-.213.553-.277.213-.064.436-.091.67-.091.118 0 .24.005.357.021.122.016.234.038.346.06.106.026.208.052.303.085.096.032.17.064.224.096a.461.461 0 01.16.133.289.289 0 01.047.176v.251c0 .112-.042.171-.122.171a.552.552 0 01-.202-.064 2.428 2.428 0 00-1.022-.208c-.303 0-.543.048-.708.15-.165.1-.25.256-.25.475 0 .149.053.277.16.379.106.101.303.202.585.293l.756.24c.383.123.66.294.825.513.165.219.244.47.244.748 0 .23-.047.437-.138.619a1.435 1.435 0 01-.388.47c-.165.133-.362.23-.591.299-.24.075-.49.112-.761.112z" />
<path
fillRule="evenodd"
d="M14.465 11.813c-1.75 1.297-4.294 1.986-6.481 1.986-3.065 0-5.827-1.137-7.913-3.027-.165-.15-.016-.353.18-.235 2.257 1.313 5.04 2.109 7.92 2.109 1.941 0 4.075-.406 6.039-1.239.293-.133.543.192.255.406z"
clipRule="evenodd"
/>
<path
fillRule="evenodd"
d="M15.194 10.98c-.223-.287-1.479-.138-2.048-.069-.17.022-.197-.128-.043-.24 1-.705 2.645-.502 2.836-.267.192.24-.053 1.89-.99 2.68-.143.123-.281.06-.217-.1.212-.53.686-1.72.462-2.003z"
clipRule="evenodd"
/>
</g>
</svg>
)
case MANAGED_BY.STRIPE_PROJECTS:
return (
<svg
className={cn(
size === 'small' && 'h-5 w-5',
size === 'medium' && 'w-7 h-7',
size === 'large' && 'w-10 h-10'
)}
viewBox="0 0 24 24"
fill="none"
xmlns="http://www.w3.org/2000/svg"
>
<rect width="24" height="24" fill="#533AFD" />
<path
fillRule="evenodd"
clipRule="evenodd"
d="M5.625 18.375L18.375 15.6711V5.625L5.625 8.36048V18.375Z"
fill="white"
transform="translate(12 12) scale(0.82) translate(-12 -12)"
/>
</svg>
)
default:
return null
}
}
function PartnerIcon({
organization,
showTooltip = true,
tooltipText,
size = 'small',
}: PartnerIconProps) {
if (
organization.managed_by === MANAGED_BY.VERCEL_MARKETPLACE ||
organization.managed_by === MANAGED_BY.AWS_MARKETPLACE ||
organization.managed_by === MANAGED_BY.STRIPE_PROJECTS
) {
const icon = getPartnerIcon(organization, size)
if (!showTooltip) {
return (
<div
className={cn(
'bg-surface-100 dark:bg-surface-200 border rounded-md overflow-hidden flex items-center justify-center shrink-0',
size === 'small' && 'h-5 w-5',
size === 'medium' && 'w-7 h-7',
size === 'large' && 'w-10 h-10'
)}
>
{icon}
</div>
)
}
const defaultTooltipText = getPartnerTooltipText({
managedBy: organization.managed_by,
tooltipText,
})
return (
<Tooltip>
<TooltipTrigger asChild>
<div
className={cn(
'bg-surface-100 dark:bg-surface-200 border rounded-md overflow-hidden flex items-center justify-center hover:bg-surface-400 dark:hover:bg-surface-400 hover:border-stronger shrink-0',
size === 'small' && 'h-5 w-5',
size === 'medium' && 'w-7 h-7',
size === 'large' && 'w-10 h-10'
)}
>
{icon}
</div>
</TooltipTrigger>
<TooltipContent>{defaultTooltipText}</TooltipContent>
</Tooltip>
)
}
return null
}
export default PartnerIcon