Files
supabase/apps/studio/components/interfaces/Auth/OAuthApps/LogoPicker.tsx
Ivan Vasilov 35905e70d5 feat: Add a logo picker for OAuth app creation sheet (#44995)
<!-- This is an auto-generated comment: release notes by coderabbit.ai
-->
## Summary by CodeRabbit

* **New Features**
* Logo field now accepts/editable logo URL, plus a new storage-based
Logo Picker to select or remove images from project storage.
* Full storage picker: browse buckets, columns/list views, search,
drag‑and‑drop uploads, file previews (image/audio/video), and
single-file selection with responsive mobile/desktop layouts.

* **Refactor**
* Logo submission streamlined to send the provided URL directly (legacy
file-read/upload flow removed).
<!-- end of auto-generated comment: release notes by coderabbit.ai -->
2026-05-06 16:44:18 +02:00

125 lines
4.5 KiB
TypeScript

import { useBreakpoint } from 'common'
import { ChevronLeft } from 'lucide-react'
import { useEffect, useState } from 'react'
import {
Button,
Dialog,
DialogContent,
DialogHeader,
DialogTitle,
Sheet,
SheetContent,
SheetHeader,
SheetTitle,
} from 'ui'
import { BucketFilePickerExplorer } from '../../Storage/BucketFilePickerDialog/BucketFilePickerExplorer'
import { BucketFilePickerStateContextProvider } from '../../Storage/BucketFilePickerDialog/BucketFilePickerState'
import { BucketsPicker } from '../../Storage/BucketsPickerDialog/BucketsPicker'
import type { Bucket } from '@/data/storage/buckets-query'
export type StorageFilePickerProps = {
open: boolean
onOpenChange: (open: boolean) => void
onSelect: (value: string) => void
}
export function LogoPicker({ open, onOpenChange, onSelect }: StorageFilePickerProps) {
const [selectedBucket, setSelectedBucket] = useState<Bucket | null>(null)
const isMobileLayout = useBreakpoint('lg')
useEffect(() => {
if (!open) {
setSelectedBucket(null)
}
}, [open])
const handleSelect = (value: string) => {
onSelect(value)
onOpenChange(false)
}
return (
<>
{isMobileLayout ? (
<Sheet open={open} onOpenChange={onOpenChange}>
<SheetContent side="bottom" className="flex h-[85vh] flex-col gap-0 p-0">
{selectedBucket ? (
<SheetHeader className="flex flex-row items-center gap-2 border-b border-overlay px-6 py-4">
<Button
type="text"
size="tiny"
icon={<ChevronLeft size={16} />}
className="shrink-0"
onClick={() => setSelectedBucket(null)}
>
Buckets
</Button>
<SheetTitle className="min-w-0 flex-1">Choose a file</SheetTitle>
</SheetHeader>
) : (
<SheetHeader className="flex flex-row items-center gap-2 border-b border-overlay px-6 py-4">
<SheetTitle className="min-w-0 flex-1">Select a bucket</SheetTitle>
</SheetHeader>
)}
<div className="flex h-full min-h-0 min-w-0 flex-1 flex-col px-2 pb-3 pt-2">
{selectedBucket ? (
<BucketFilePickerStateContextProvider
bucket={selectedBucket}
maxFiles={1}
acceptedFileExtensions={['.png', '.jpg', '.jpeg', '.svg']}
>
<BucketFilePickerExplorer onSelect={handleSelect} />
</BucketFilePickerStateContextProvider>
) : (
<BucketsPicker onSelectBucket={setSelectedBucket} allowedBucketType="public" />
)}
</div>
</SheetContent>
</Sheet>
) : (
<Dialog open={open} onOpenChange={onOpenChange}>
<DialogContent
size="xxlarge"
className="flex h-[85vh] max-h-[500px] max-w-5xl flex-col gap-0 overflow-hidden p-0"
>
{selectedBucket ? (
<DialogHeader className="flex flex-row items-center gap-2 border-b border-overlay px-6 py-4 h-[60px]">
<Button
type="text"
size="tiny"
icon={<ChevronLeft size={16} />}
className="shrink-0"
onClick={() => setSelectedBucket(null)}
>
Buckets
</Button>
<DialogTitle className="min-w-0 flex-1">Choose a file</DialogTitle>
</DialogHeader>
) : (
<DialogHeader className="flex flex-row items-center gap-2 border-b border-overlay px-6 py-4 h-[60px]">
<DialogTitle className="min-w-0 flex-1">Select a bucket</DialogTitle>
</DialogHeader>
)}
<div className="flex h-full min-h-0 min-w-0 flex-1 flex-col overflow-hidden p-3">
{selectedBucket ? (
<BucketFilePickerStateContextProvider
bucket={selectedBucket}
maxFiles={1}
acceptedFileExtensions={['.png', '.jpg', '.jpeg', '.svg']}
>
<BucketFilePickerExplorer onSelect={handleSelect} />
</BucketFilePickerStateContextProvider>
) : (
<BucketsPicker onSelectBucket={setSelectedBucket} allowedBucketType="public" />
)}
</div>
</DialogContent>
</Dialog>
)}
</>
)
}