Files
supabase/packages/ui-patterns/Dialogs/TextConfirmModal.tsx
Joshen Lim 1127c4ba88 Project Level Permissions (#27347)
* fix: update Permission params

* fix: upgrade check permission hook to support project level role

* fix: usePermissionsLoaded

* fix: Permission params can be undefined

* Scaffold new access management UI

* Add validation

* Update roles view

* Add tooltip

* Add button to apply role to all projects

* Update UI to select projects first instead of roles

* Merge master update UI

* Midway trying to implementation project level perms API

* First pass implementating updating project level permissions

* Add client side validation for assigning/removing roles

* Midway implementing new invites

* Integrate most of the project level permissions functionality

* fix: filter out org-level permissions before checking

* Add relevant UI guards in org level pages for project role POV

* Minor refactors

* Small refactors

* More fixes

* Moar refactors

* More fixes

* More fixes

* Refactor update role logic and smack some test cases on it

* Fixes

* Fix type issue

* Fix type

* more fixes, refactors, adding checks...

* MORE fixes

* Add perms checking for replicas

* Add ButtonTooltip component and use them to prevent repetition of pointer events auto for buttons with tooltips

* Convert all buttons with tooltips to use ButtonTooltip

* refactor

* PRettier

* Small fix

* Remove commented out code in organization-invitation-accept-mutation

* fix: switch to use the platform oauth authorizations routes

* Add perms checking for org audit logs and org oauth apps

* PRettier

* Fix incorrect URL for oauth app flow

* Fix incorrect URL for oauth app flow

* Fix

* Add perms checking for warehouse related UI

* Update roles helper icon

* remove unused lib

* Update package lock... again

* Update package lock... again

* Smalllll update

* Update some checks

* Add gate for project level permissions

* Last fix

* update codegen

* Update warehouse endpoint routes

* Fix

---------

Co-authored-by: phamhieu <phamhieu1998@gmail.com>
Co-authored-by: Alaister Young <a@alaisteryoung.com>
2024-07-01 17:59:54 +08:00

199 lines
5.7 KiB
TypeScript

'use client'
import { zodResolver } from '@hookform/resolvers/zod'
import { ReactNode, forwardRef, useEffect } from 'react'
import { useForm } from 'react-hook-form'
import {
Alert_Shadcn_,
Button,
Dialog,
DialogContent,
DialogSection,
DialogSectionSeparator,
DialogTitle,
FormControl_Shadcn_,
FormDescription_Shadcn_,
FormField_Shadcn_,
FormItem_Shadcn_,
FormLabel_Shadcn_,
FormMessage_Shadcn_,
Form_Shadcn_,
Input_Shadcn_,
cn,
} from 'ui'
import { Admonition } from './../admonition'
import { DialogHeader } from 'ui/src/components/shadcn/ui/dialog'
import { z } from 'zod'
export interface TextConfirmModalProps {
loading: boolean
visible: boolean
title: string
size?: React.ComponentProps<typeof DialogContent>['size']
cancelLabel?: string
confirmLabel?: string
confirmPlaceholder: string
confirmString: string
text?: string | ReactNode
onConfirm: () => void
onCancel: () => void
variant?: React.ComponentProps<typeof Alert_Shadcn_>['variant']
alert?: {
base?: React.ComponentProps<typeof Alert_Shadcn_>
title?: string
description?: string
}
input?: React.ComponentProps<typeof Input_Shadcn_>
label?: React.ComponentProps<typeof FormLabel_Shadcn_>
formMessage?: React.ComponentProps<typeof FormMessage_Shadcn_>
description?: React.ComponentProps<typeof FormDescription_Shadcn_>
blockDeleteButton?: boolean
}
const TextConfirmModal = forwardRef<
React.ElementRef<typeof DialogContent>,
React.ComponentPropsWithoutRef<typeof Dialog> & TextConfirmModalProps
>(
(
{
title,
size = 'small',
onConfirm,
visible,
onCancel,
loading,
cancelLabel = 'Cancel',
confirmLabel = 'Submit',
confirmPlaceholder,
confirmString,
alert,
input,
label,
description,
formMessage,
text,
children,
blockDeleteButton = true,
variant = 'default',
...props
},
ref
) => {
const formSchema = z.object({
confirmValue: z.literal(confirmString, {
required_error: 'Value entered does not match.',
}),
})
// 1. Define your form.
const form = useForm<z.infer<typeof formSchema>>({
resolver: zodResolver(formSchema),
defaultValues: {
confirmValue: '',
},
})
// 2. Define a submit handler.
function onSubmit(values: z.infer<typeof formSchema>) {
// Do something with the form values.
// ✅ This will be type-safe and validated.
onConfirm()
}
useEffect(() => {
if (confirmString) form.reset()
}, [confirmString])
return (
<Dialog
open={visible}
{...props}
onOpenChange={() => {
if (visible) {
onCancel()
}
}}
>
<DialogContent ref={ref} className="p-0 gap-0 pb-5 !block" size={size}>
<DialogHeader className={cn('border-b')} padding={'small'}>
<DialogTitle className="">{title}</DialogTitle>
</DialogHeader>
{alert && (
<Admonition
type={variant as 'default' | 'destructive' | 'warning'}
label={alert.title}
description={alert.description}
className="border-r-0 border-l-0 rounded-none -mt-px [&_svg]:ml-0.5 mb-0"
{...alert?.base}
/>
)}
{children && (
<>
<DialogSection padding={'small'}>{children}</DialogSection>
<DialogSectionSeparator />
</>
)}
{/* // older prop from before refactor */}
{text !== undefined && (
<>
<DialogSection className="p-5" padding={'small'}>
<p className="text-foreground-light text-sm">{text}</p>
</DialogSection>
<DialogSectionSeparator />
</>
)}
<Form_Shadcn_ {...form}>
<form onSubmit={form.handleSubmit(onSubmit)} className="px-5 flex flex-col gap-2 pt-3">
<FormField_Shadcn_
control={form.control}
name="confirmValue"
render={({ field }) => (
<FormItem_Shadcn_ className="flex flex-col gap-y-1">
<FormLabel_Shadcn_ {...label}>
Type <span className="text-foreground break-all">{confirmString}</span> to
confirm.
</FormLabel_Shadcn_>
<FormControl_Shadcn_>
<Input_Shadcn_ placeholder={confirmPlaceholder} {...input} {...field} />
</FormControl_Shadcn_>
<FormDescription_Shadcn_ {...description} />
<FormMessage_Shadcn_ {...formMessage} />
</FormItem_Shadcn_>
)}
/>
<div className="flex gap-2">
{!blockDeleteButton && (
<Button size="medium" block type="default" disabled={loading} onClick={onCancel}>
{cancelLabel}
</Button>
)}
<Button
block
size="medium"
type={
variant === 'destructive'
? 'danger'
: variant === 'warning'
? 'warning'
: 'primary'
}
htmlType="submit"
loading={loading}
disabled={loading}
className="truncate"
>
{confirmLabel}
</Button>
</div>
</form>
</Form_Shadcn_>
</DialogContent>
</Dialog>
)
}
)
TextConfirmModal.displayName = 'TextConfirmModal'
export default TextConfirmModal