mirror of
https://github.com/supabase/supabase.git
synced 2026-05-06 22:18:00 +08:00
## Screenshots ### Extensions search input Before: <img width="955" height="256" alt="image" src="https://github.com/user-attachments/assets/c69e428a-8ab5-4dce-a45a-5d6a6d30472d" /> After: <img width="965" height="212" alt="image" src="https://github.com/user-attachments/assets/a08294cc-14ea-4c8d-af24-a207de3dada9" /> ### Triggers search input Before: <img width="961" height="249" alt="image" src="https://github.com/user-attachments/assets/21df2aeb-cc83-42e2-a35e-23e6451182ad" /> After: <img width="979" height="248" alt="image" src="https://github.com/user-attachments/assets/f365661d-5075-4041-a4f2-8fd1b7fdeb4b" /> ### Hooks search input Before: <img width="974" height="361" alt="image" src="https://github.com/user-attachments/assets/baaad7fb-1ede-46a4-8148-3cc05a53c955" /> After: <img width="976" height="363" alt="image" src="https://github.com/user-attachments/assets/9c3b2467-1e9a-4919-a6df-9e3ff46a30b8" /> ### Backups - restore to new project dialog Before: <img width="544" height="656" alt="image" src="https://github.com/user-attachments/assets/181018ac-cda6-4a57-bfc3-028ac6a1eeed" /> After: <img width="536" height="643" alt="image" src="https://github.com/user-attachments/assets/4c177884-4415-4744-b3d1-67fe83065565" /> <!-- This is an auto-generated comment: release notes by coderabbit.ai --> ## Summary by CodeRabbit * **Refactor** * Standardized search inputs across database interfaces by consolidating into a grouped input pattern for consistent behavior and keyboard focus. * **Style** * Improved layout of the database creation dialog’s password field, including visible reveal control and relocated strength indicator for clearer form presentation. <!-- end of auto-generated comment: release notes by coderabbit.ai --> --------- Co-authored-by: Ali Waseem <waseema393@gmail.com>
200 lines
6.7 KiB
TypeScript
200 lines
6.7 KiB
TypeScript
import { zodResolver } from '@hookform/resolvers/zod'
|
|
import { useState } from 'react'
|
|
import { useForm } from 'react-hook-form'
|
|
import { toast } from 'sonner'
|
|
import {
|
|
Button,
|
|
Dialog,
|
|
DialogContent,
|
|
DialogDescription,
|
|
DialogFooter,
|
|
DialogHeader,
|
|
DialogSection,
|
|
DialogTitle,
|
|
Form,
|
|
FormControl,
|
|
FormField,
|
|
Input_Shadcn_,
|
|
} from 'ui'
|
|
import { Input } from 'ui-patterns/DataInputs/Input'
|
|
import { FormItemLayout } from 'ui-patterns/form/FormItemLayout/FormItemLayout'
|
|
import { z } from 'zod'
|
|
|
|
import { AdditionalMonthlySpend } from './AdditionalMonthlySpend'
|
|
import { NewProjectPrice } from './RestoreToNewProject.utils'
|
|
import { PasswordStrengthBar } from '@/components/ui/PasswordStrengthBar'
|
|
import { useProjectCloneMutation } from '@/data/projects/clone-mutation'
|
|
import { useCloneBackupsQuery } from '@/data/projects/clone-query'
|
|
import { useSelectedProjectQuery } from '@/hooks/misc/useSelectedProject'
|
|
import { passwordStrength, PasswordStrengthScore } from '@/lib/password-strength'
|
|
import { generateStrongPassword } from '@/lib/project'
|
|
|
|
interface CreateNewProjectDialogProps {
|
|
open: boolean
|
|
selectedBackupId: number | null
|
|
recoveryTimeTarget: number | null
|
|
onOpenChange: (value: boolean) => void
|
|
onCloneSuccess: () => void
|
|
additionalMonthlySpend: NewProjectPrice
|
|
hasAccess?: boolean
|
|
}
|
|
|
|
export const CreateNewProjectDialog = ({
|
|
open,
|
|
selectedBackupId,
|
|
recoveryTimeTarget,
|
|
onOpenChange,
|
|
onCloneSuccess,
|
|
additionalMonthlySpend,
|
|
hasAccess,
|
|
}: CreateNewProjectDialogProps) => {
|
|
const { data: project } = useSelectedProjectQuery()
|
|
const [passwordStrengthScore, setPasswordStrengthScore] = useState(0)
|
|
const [passwordStrengthMessage, setPasswordStrengthMessage] = useState('')
|
|
|
|
const FormSchema = z.object({
|
|
name: z.string().min(1),
|
|
password: z.string().min(1),
|
|
})
|
|
|
|
const form = useForm<z.infer<typeof FormSchema>>({
|
|
resolver: zodResolver(FormSchema),
|
|
defaultValues: {
|
|
name: '',
|
|
password: '',
|
|
},
|
|
})
|
|
|
|
const { data: cloneBackups } = useCloneBackupsQuery(
|
|
{ projectRef: project?.ref },
|
|
{ enabled: hasAccess }
|
|
)
|
|
const hasPITREnabled = cloneBackups?.pitr_enabled
|
|
|
|
const { mutate: triggerClone, isPending: cloneMutationLoading } = useProjectCloneMutation({
|
|
onError: (error) => {
|
|
toast.error(`Failed to restore to new project: ${error.message}`)
|
|
},
|
|
onSuccess: () => {
|
|
toast.success('Restoration process started')
|
|
onCloneSuccess()
|
|
},
|
|
})
|
|
|
|
async function checkPasswordStrength(value: string) {
|
|
const { message, strength } = await passwordStrength(value)
|
|
setPasswordStrengthScore(strength)
|
|
setPasswordStrengthMessage(message)
|
|
}
|
|
|
|
const generatePassword = () => {
|
|
const password = generateStrongPassword()
|
|
form.setValue('password', password)
|
|
checkPasswordStrength(password)
|
|
}
|
|
|
|
return (
|
|
<Dialog open={open} onOpenChange={onOpenChange}>
|
|
<DialogContent>
|
|
<DialogHeader className="border-b">
|
|
<DialogTitle>Create new project</DialogTitle>
|
|
<DialogDescription>
|
|
This process will create a new project and restore your database to it.
|
|
</DialogDescription>
|
|
</DialogHeader>
|
|
<Form {...form}>
|
|
<form
|
|
id={'create-new-project-form'}
|
|
onSubmit={form.handleSubmit((data) => {
|
|
if (!project?.ref) {
|
|
toast.error('Project ref is required')
|
|
return
|
|
}
|
|
|
|
if (hasPITREnabled && recoveryTimeTarget) {
|
|
triggerClone({
|
|
projectRef: project?.ref,
|
|
newProjectName: data.name,
|
|
newDbPass: data.password,
|
|
recoveryTimeTarget: recoveryTimeTarget,
|
|
cloneBackupId: undefined,
|
|
})
|
|
} else if (selectedBackupId) {
|
|
triggerClone({
|
|
projectRef: project?.ref,
|
|
cloneBackupId: selectedBackupId,
|
|
newProjectName: data.name,
|
|
newDbPass: data.password,
|
|
recoveryTimeTarget: undefined,
|
|
})
|
|
} else {
|
|
toast.error('No backup or point in time selected')
|
|
return
|
|
}
|
|
})}
|
|
>
|
|
<DialogSection className="pb-6 space-y-4 text-sm">
|
|
<FormField
|
|
control={form.control}
|
|
name="name"
|
|
render={({ field }) => (
|
|
<FormItemLayout label="New Project Name">
|
|
<FormControl>
|
|
<Input_Shadcn_ placeholder="Enter a name" type="text" {...field} />
|
|
</FormControl>
|
|
</FormItemLayout>
|
|
)}
|
|
/>
|
|
<FormField
|
|
control={form.control}
|
|
name="password"
|
|
render={({ field }) => (
|
|
<FormItemLayout
|
|
label="Database password"
|
|
description={
|
|
<PasswordStrengthBar
|
|
passwordStrengthScore={passwordStrengthScore as PasswordStrengthScore}
|
|
password={field.value}
|
|
passwordStrengthMessage={passwordStrengthMessage}
|
|
generateStrongPassword={generatePassword}
|
|
/>
|
|
}
|
|
>
|
|
<FormControl>
|
|
<Input
|
|
id="db-password"
|
|
type="password"
|
|
placeholder="Type in a strong password"
|
|
value={field.value}
|
|
copy={field.value?.length > 0}
|
|
reveal
|
|
onChange={(e) => {
|
|
const value = e.target.value
|
|
field.onChange(value)
|
|
if (value == '') {
|
|
setPasswordStrengthScore(-1)
|
|
setPasswordStrengthMessage('')
|
|
} else checkPasswordStrength(value)
|
|
}}
|
|
/>
|
|
</FormControl>
|
|
</FormItemLayout>
|
|
)}
|
|
/>
|
|
</DialogSection>
|
|
<AdditionalMonthlySpend additionalMonthlySpend={additionalMonthlySpend} />
|
|
<DialogFooter>
|
|
<Button htmlType="reset" type="outline" onClick={() => onOpenChange(false)}>
|
|
Cancel
|
|
</Button>
|
|
<Button htmlType="submit" loading={cloneMutationLoading}>
|
|
Restore to new project
|
|
</Button>
|
|
</DialogFooter>
|
|
</form>
|
|
</Form>
|
|
</DialogContent>
|
|
</Dialog>
|
|
)
|
|
}
|