mirror of
https://github.com/supabase/supabase.git
synced 2026-05-07 17:30:25 -04:00
0cb71a2497
This PR integrates with the new marketplace db to allow Grafana (and other partners) OAuth apps to install from the integrations page. A demo of this working locally is available here: https://supabase.slack.com/archives/C01GN60J0BS/p1775551752479709. End to end flow is documented here: https://www.notion.so/supabase/Grafana-Integration-Flow-33a5004b775f80eeaf91c098beb8071f. TODO: - [ ] Make sure `NEXT_PUBLIC_MARKETPLACE_API_URL` variable is set to the new marketplace db. - [x] Test with the `marketplaceIntegrations` enabled and disabled in staging once https://github.com/supabase/platform/pull/31298 is merged and available in staging. <!-- This is an auto-generated comment: release notes by coderabbit.ai --> ## Summary by CodeRabbit * **New Features** * Add OAuth "Install integration" button that detects installed integrations and supports GET/POST install flows * Marketplace listings now include install links, installation method, partner info, and listing assets/logos * **Infrastructure** * Allow marketplace API origin for images and content in security and image config * Centralize marketplace types and switch marketplace data source for more reliable listings <!-- end of auto-generated comment: release notes by coderabbit.ai --> --------- Co-authored-by: Joshen Lim <joshenlimek@gmail.com>
44 lines
1.3 KiB
TypeScript
44 lines
1.3 KiB
TypeScript
import { useState } from 'react'
|
|
import { cn, Dialog, DialogContent } from 'ui'
|
|
|
|
export const FilesViewer = ({ files }: { files: string[] }) => {
|
|
const [selected, setSelected] = useState(files[0])
|
|
const [showDialog, setShowDialog] = useState(false)
|
|
|
|
return (
|
|
<>
|
|
<div className="flex flex-col gap-y-4">
|
|
<button onClick={() => setShowDialog(true)}>
|
|
<img
|
|
alt={selected}
|
|
src={selected}
|
|
className="rounded-md border object-cover aspect-video"
|
|
/>
|
|
</button>
|
|
|
|
{files.length > 1 && (
|
|
<div className="grid grid-cols-10 gap-x-2">
|
|
{files.map((x) => (
|
|
<button key={x} onClick={() => setSelected(x)}>
|
|
<img
|
|
alt={x}
|
|
src={x}
|
|
className={cn(
|
|
'col-span-1 bg-surface-100 rounded-md object-cover aspect-square border transition',
|
|
selected === x ? 'border-button-hover' : 'border-secondary'
|
|
)}
|
|
/>
|
|
</button>
|
|
))}
|
|
</div>
|
|
)}
|
|
</div>
|
|
<Dialog open={showDialog} onOpenChange={setShowDialog}>
|
|
<DialogContent size="xxlarge">
|
|
<img alt={selected} src={selected} className="rounded-md border" />
|
|
</DialogContent>
|
|
</Dialog>
|
|
</>
|
|
)
|
|
}
|