mirror of
https://github.com/supabase/supabase.git
synced 2026-06-17 21:23:59 +08:00
## Problem Our `<Button>` component breaks the default `button` contract by redefining the `type` prop to set its variant (`primary`, `default`, etc) instead of the button type (`submit`, `button`, etc). This is confusing and forces to write more code when using it with shadcn components that expect/inject the standard button props. ## Solution - rename the `type` prop to `variant` - rename the `htmlType` prop to `type` - propagate the changes where necessary - format code ## How to test As this is just prop renaming, if it builds it's ok --------- Co-authored-by: Ivan Vasilov <vasilov.ivan@gmail.com>
130 lines
3.8 KiB
TypeScript
130 lines
3.8 KiB
TypeScript
import { Upload } from 'lucide-react'
|
|
import { useEffect, useRef, useState } from 'react'
|
|
import { useForm } from 'react-hook-form'
|
|
import {
|
|
Button,
|
|
FormControl,
|
|
FormField,
|
|
FormItem,
|
|
FormMessage,
|
|
Input,
|
|
Tabs_Shadcn_,
|
|
TabsContent_Shadcn_,
|
|
TabsList_Shadcn_,
|
|
TabsTrigger_Shadcn_,
|
|
} from 'ui'
|
|
import { FormItemLayout } from 'ui-patterns/form/FormItemLayout/FormItemLayout'
|
|
|
|
import { SSOConfigFormSchema } from './SSOConfig'
|
|
|
|
export const SSOMetadata = ({
|
|
form,
|
|
}: {
|
|
form: ReturnType<typeof useForm<SSOConfigFormSchema>>
|
|
}) => {
|
|
const fileInputRef = useRef<HTMLInputElement>(null)
|
|
const [fileName, setFileName] = useState<string | null>(null)
|
|
const [tab, setTab] = useState<'url' | 'file'>('url')
|
|
|
|
useEffect(() => {
|
|
if (form.getValues('metadataXmlFile')) {
|
|
setTab('file')
|
|
} else if (form.getValues('metadataXmlUrl')) {
|
|
setTab('url')
|
|
}
|
|
}, [form])
|
|
|
|
const handleFileChange = async (e: React.ChangeEvent<HTMLInputElement>) => {
|
|
form.clearErrors('metadataXmlFile')
|
|
const file = e.target.files?.[0]
|
|
if (!file) return
|
|
if (!file.name.endsWith('.xml')) {
|
|
form.setError('metadataXmlFile', {
|
|
type: 'manual',
|
|
message: 'Please upload a valid .xml file',
|
|
})
|
|
return
|
|
}
|
|
try {
|
|
const text = await file.text()
|
|
form.setValue('metadataXmlFile', text, { shouldDirty: true })
|
|
setFileName(file.name)
|
|
} catch (err) {
|
|
form.setError('metadataXmlFile', {
|
|
type: 'manual',
|
|
message: 'Failed to read file',
|
|
})
|
|
}
|
|
}
|
|
|
|
return (
|
|
<FormItemLayout
|
|
label="Metadata"
|
|
layout="flex-row-reverse"
|
|
description="Provide a link to your metadata .xml file or upload one."
|
|
className="gap-1"
|
|
>
|
|
<div className=" w-96">
|
|
<Tabs_Shadcn_
|
|
value={tab}
|
|
onValueChange={(value: string) => setTab(value as 'url' | 'file')}
|
|
className="max-w-2xl"
|
|
>
|
|
<TabsList_Shadcn_ className="mx-auto gap-5 w-auto">
|
|
<TabsTrigger_Shadcn_ className=" " value="url">
|
|
URL
|
|
</TabsTrigger_Shadcn_>
|
|
<TabsTrigger_Shadcn_ className=" " value="file">
|
|
Upload file
|
|
</TabsTrigger_Shadcn_>
|
|
</TabsList_Shadcn_>
|
|
<TabsContent_Shadcn_ value="url">
|
|
<FormField
|
|
name="metadataXmlUrl"
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormControl>
|
|
<Input
|
|
{...field}
|
|
placeholder="https://example.com/metadata.xml"
|
|
autoComplete="off"
|
|
/>
|
|
</FormControl>
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
</TabsContent_Shadcn_>
|
|
<TabsContent_Shadcn_ value="file">
|
|
<FormField
|
|
name="metadataXmlUrl"
|
|
render={() => (
|
|
<div className="flex flex-col gap-2 max-w-md">
|
|
<div className="flex items-center gap-2">
|
|
<input
|
|
ref={fileInputRef}
|
|
type="file"
|
|
accept=".xml"
|
|
className="hidden"
|
|
onChange={handleFileChange}
|
|
/>
|
|
<Button
|
|
variant="default"
|
|
icon={<Upload className="w-4 h-4" />}
|
|
onClick={() => fileInputRef.current?.click()}
|
|
>
|
|
Upload XML
|
|
</Button>
|
|
{fileName && <span className="text-xs text-foreground-light">{fileName}</span>}
|
|
</div>
|
|
<FormMessage />
|
|
</div>
|
|
)}
|
|
/>
|
|
</TabsContent_Shadcn_>
|
|
</Tabs_Shadcn_>
|
|
</div>
|
|
</FormItemLayout>
|
|
)
|
|
}
|