Files
supabase/apps/studio/components/interfaces/Auth/Users/InviteUserModal.tsx
Gildas Garcia a6bfa0478c chore: migrate auth Modal to Dialog (#46392)
## Problem

We still uses the deprecated `Modal` for:
- Banning a user
- Inviting a user
- Adding a redirection URL
- Removing a redirection URL

## Test

Hard to test the JWT. I had to force its display by settings its `open`
prop to `true` in
`apps/studio/components/interfaces/JwtSecrets/jwt-settings.tsx`

## Solution

- use `Dialog` instead

<!-- This is an auto-generated comment: release notes by coderabbit.ai
-->

## Summary by CodeRabbit

* **Refactor**
* Modernized authentication UI components (user management, redirect URL
configuration) with updated dialog structures for improved consistency
and state handling
* Enhanced async operation handling in redirect URL deletion workflows
to ensure proper synchronization and state management

<!-- review_stack_entry_start -->

[![Review Change
Stack](https://storage.googleapis.com/coderabbit_public_assets/review-stack-in-coderabbit-ui.svg)](https://app.coderabbit.ai/change-stack/supabase/supabase/pull/46392?utm_source=github_walkthrough&utm_medium=github&utm_campaign=change_stack)

<!-- review_stack_entry_end -->

<!-- end of auto-generated comment: release notes by coderabbit.ai -->
2026-05-27 14:08:51 +02:00

113 lines
3.2 KiB
TypeScript

import { zodResolver } from '@hookform/resolvers/zod'
import { PermissionAction } from '@supabase/shared-types/out/constants'
import { useParams } from 'common'
import { SubmitHandler, useForm } from 'react-hook-form'
import { toast } from 'sonner'
import {
Button,
Dialog,
DialogContent,
DialogFooter,
DialogHeader,
DialogSection,
DialogSectionSeparator,
DialogTitle,
Form,
FormControl,
FormField,
Input,
} from 'ui'
import { FormItemLayout } from 'ui-patterns/form/FormItemLayout/FormItemLayout'
import * as z from 'zod'
import { useUserInviteMutation } from '@/data/auth/user-invite-mutation'
import { useAsyncCheckPermissions } from '@/hooks/misc/useCheckPermissions'
export type InviteUserModalProps = {
visible: boolean
setVisible: (visible: boolean) => void
}
const formSchema = z.object({
email: z.string().min(1, 'Please enter a valid email').email('Please enter a valid email'),
})
const formId = 'invite-user-form'
const InviteUserModal = ({ visible, setVisible }: InviteUserModalProps) => {
const { ref: projectRef } = useParams()
const form = useForm<z.infer<typeof formSchema>>({
resolver: zodResolver(formSchema),
defaultValues: {
email: '',
},
})
const handleToggle = () => setVisible(!visible)
const { mutate: inviteUser, isPending: isInviting } = useUserInviteMutation({
onSuccess: (_, variables) => {
toast.success(`Sent invite email to ${variables.email}`)
setVisible(false)
},
})
const { can: canInviteUsers } = useAsyncCheckPermissions(
PermissionAction.AUTH_EXECUTE,
'invite_user'
)
const onInviteUser: SubmitHandler<z.infer<typeof formSchema>> = async (values) => {
if (!projectRef) return console.error('Project ref is required')
inviteUser(
{ projectRef, email: values.email },
{
onSuccess: () => {
form.reset()
},
}
)
}
return (
<Dialog key="invite-user-modal" open={visible} onOpenChange={handleToggle}>
<DialogContent size="small">
<DialogHeader>
<DialogTitle>Invite a new user</DialogTitle>
</DialogHeader>
<DialogSectionSeparator />
<Form {...form}>
<DialogSection>
<form id={formId} onSubmit={form.handleSubmit(onInviteUser)} noValidate>
<FormField
control={form.control}
name="email"
render={({ field }) => (
<FormItemLayout layout="vertical" label="User email">
<FormControl className="relative col-span-6">
<Input {...field} />
</FormControl>
</FormItemLayout>
)}
/>
</form>
</DialogSection>
<DialogFooter>
<Button type="default" onClick={handleToggle}>
Cancel
</Button>
<Button
form={formId}
htmlType="submit"
loading={isInviting}
disabled={!canInviteUsers || isInviting}
>
Invite user
</Button>
</DialogFooter>
</Form>
</DialogContent>
</Dialog>
)
}
export default InviteUserModal