302 lines
9.8 KiB
TypeScript
Vendored
302 lines
9.8 KiB
TypeScript
Vendored
import { useEffect, useState } from 'react'
|
|
import { useForm } from 'react-hook-form'
|
|
import { zodResolver } from '@hookform/resolvers/zod'
|
|
import { useTranslation } from 'react-i18next'
|
|
import { toast } from 'sonner'
|
|
import { getCurrencyDisplay, getCurrencyLabel } from '@/lib/currency'
|
|
import { addTimeToDate } from '@/lib/time'
|
|
import { Button } from '@/components/ui/button'
|
|
import {
|
|
Form,
|
|
FormControl,
|
|
FormDescription,
|
|
FormField,
|
|
FormItem,
|
|
FormLabel,
|
|
FormMessage,
|
|
} from '@/components/ui/form'
|
|
import { Input } from '@/components/ui/input'
|
|
import {
|
|
Sheet,
|
|
SheetClose,
|
|
SheetContent,
|
|
SheetDescription,
|
|
SheetFooter,
|
|
SheetHeader,
|
|
SheetTitle,
|
|
} from '@/components/ui/sheet'
|
|
import { DateTimePicker } from '@/components/datetime-picker'
|
|
import { createRedemption, updateRedemption, getRedemption } from '../api'
|
|
import { SUCCESS_MESSAGES } from '../constants'
|
|
import {
|
|
getRedemptionFormSchema,
|
|
type RedemptionFormValues,
|
|
REDEMPTION_FORM_DEFAULT_VALUES,
|
|
transformFormDataToPayload,
|
|
transformRedemptionToFormDefaults,
|
|
} from '../lib'
|
|
import { type Redemption } from '../types'
|
|
import { useRedemptions } from './redemptions-provider'
|
|
|
|
type RedemptionsMutateDrawerProps = {
|
|
open: boolean
|
|
onOpenChange: (open: boolean) => void
|
|
currentRow?: Redemption
|
|
}
|
|
|
|
export function RedemptionsMutateDrawer({
|
|
open,
|
|
onOpenChange,
|
|
currentRow,
|
|
}: RedemptionsMutateDrawerProps) {
|
|
const { t } = useTranslation()
|
|
const isUpdate = !!currentRow
|
|
const { triggerRefresh } = useRedemptions()
|
|
const [isSubmitting, setIsSubmitting] = useState(false)
|
|
|
|
const form = useForm<RedemptionFormValues>({
|
|
resolver: zodResolver(getRedemptionFormSchema(t)),
|
|
defaultValues: REDEMPTION_FORM_DEFAULT_VALUES,
|
|
})
|
|
|
|
// Load existing data when updating
|
|
useEffect(() => {
|
|
if (open && isUpdate && currentRow) {
|
|
// For update, fetch fresh data
|
|
getRedemption(currentRow.id).then((result) => {
|
|
if (result.success && result.data) {
|
|
form.reset(transformRedemptionToFormDefaults(result.data))
|
|
}
|
|
})
|
|
} else if (open && !isUpdate) {
|
|
// For create, reset to defaults
|
|
form.reset(REDEMPTION_FORM_DEFAULT_VALUES)
|
|
}
|
|
}, [open, isUpdate, currentRow, form])
|
|
|
|
const onSubmit = async (data: RedemptionFormValues) => {
|
|
setIsSubmitting(true)
|
|
try {
|
|
const basePayload = transformFormDataToPayload(data)
|
|
|
|
if (isUpdate && currentRow) {
|
|
const result = await updateRedemption({
|
|
...basePayload,
|
|
id: currentRow.id,
|
|
})
|
|
if (result.success) {
|
|
toast.success(t(SUCCESS_MESSAGES.REDEMPTION_UPDATED))
|
|
onOpenChange(false)
|
|
triggerRefresh()
|
|
}
|
|
} else {
|
|
// Create mode
|
|
const result = await createRedemption(basePayload)
|
|
if (result.success) {
|
|
const count = result.data?.length || 0
|
|
toast.success(
|
|
count > 1
|
|
? t('Successfully created {{count}} redemption codes', {
|
|
count,
|
|
})
|
|
: t(SUCCESS_MESSAGES.REDEMPTION_CREATED)
|
|
)
|
|
onOpenChange(false)
|
|
triggerRefresh()
|
|
}
|
|
}
|
|
} finally {
|
|
setIsSubmitting(false)
|
|
}
|
|
}
|
|
|
|
const handleSetExpiry = (months: number, days: number, hours: number) => {
|
|
const newDate = addTimeToDate(months, days, hours)
|
|
form.setValue('expired_time', newDate)
|
|
}
|
|
|
|
const { meta: currencyMeta } = getCurrencyDisplay()
|
|
const currencyLabel = getCurrencyLabel()
|
|
const tokensOnly = currencyMeta.kind === 'tokens'
|
|
const quotaLabel = t('Quota ({{currency}})', { currency: currencyLabel })
|
|
const quotaPlaceholder = tokensOnly
|
|
? t('Enter quota in tokens')
|
|
: t('Enter quota in {{currency}}', { currency: currencyLabel })
|
|
|
|
return (
|
|
<Sheet
|
|
open={open}
|
|
onOpenChange={(v) => {
|
|
onOpenChange(v)
|
|
if (!v) {
|
|
form.reset()
|
|
}
|
|
}}
|
|
>
|
|
<SheetContent className='flex h-dvh w-full flex-col gap-0 overflow-hidden p-0 sm:max-w-[600px]'>
|
|
<SheetHeader className='border-b px-4 py-3 text-start sm:px-6 sm:py-4'>
|
|
<SheetTitle>
|
|
{isUpdate
|
|
? t('Update Redemption Code')
|
|
: t('Create Redemption Code')}
|
|
</SheetTitle>
|
|
<SheetDescription>
|
|
{isUpdate
|
|
? t('Update the redemption code by providing necessary info.')
|
|
: t(
|
|
'Add new redemption code(s) by providing necessary info.'
|
|
)}{' '}
|
|
{t('Click save when you're done.')}
|
|
</SheetDescription>
|
|
</SheetHeader>
|
|
<Form {...form}>
|
|
<form
|
|
id='redemption-form'
|
|
onSubmit={form.handleSubmit(onSubmit)}
|
|
className='flex-1 space-y-4 overflow-y-auto px-3 py-3 pb-4 sm:space-y-6 sm:px-4'
|
|
>
|
|
<FormField
|
|
control={form.control}
|
|
name='name'
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>{t('Name')}</FormLabel>
|
|
<FormControl>
|
|
<Input {...field} placeholder={t('Enter a name')} />
|
|
</FormControl>
|
|
<FormDescription>
|
|
{t('Name for this redemption code (1-20 characters)')}
|
|
</FormDescription>
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
|
|
<FormField
|
|
control={form.control}
|
|
name='quota_dollars'
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>{quotaLabel}</FormLabel>
|
|
<FormControl>
|
|
<Input
|
|
{...field}
|
|
type='number'
|
|
step={tokensOnly ? 1 : 0.01}
|
|
placeholder={quotaPlaceholder}
|
|
onChange={(e) =>
|
|
field.onChange(parseFloat(e.target.value) || 0)
|
|
}
|
|
/>
|
|
</FormControl>
|
|
<FormDescription>
|
|
{tokensOnly
|
|
? t('Enter the quota amount in tokens')
|
|
: t('Enter the quota amount in {{currency}}', {
|
|
currency: currencyLabel,
|
|
})}
|
|
</FormDescription>
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
|
|
<FormField
|
|
control={form.control}
|
|
name='expired_time'
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>{t('Expiration Time')}</FormLabel>
|
|
<div className='space-y-2'>
|
|
<FormControl>
|
|
<DateTimePicker
|
|
value={field.value}
|
|
onChange={field.onChange}
|
|
placeholder={t('Never expires')}
|
|
/>
|
|
</FormControl>
|
|
<div className='grid grid-cols-4 gap-1.5 sm:flex sm:gap-2'>
|
|
<Button
|
|
type='button'
|
|
variant='outline'
|
|
size='sm'
|
|
onClick={() => handleSetExpiry(0, 0, 0)}
|
|
>
|
|
{t('Never')}
|
|
</Button>
|
|
<Button
|
|
type='button'
|
|
variant='outline'
|
|
size='sm'
|
|
onClick={() => handleSetExpiry(1, 0, 0)}
|
|
>
|
|
{t('1M')}
|
|
</Button>
|
|
<Button
|
|
type='button'
|
|
variant='outline'
|
|
size='sm'
|
|
onClick={() => handleSetExpiry(0, 7, 0)}
|
|
>
|
|
{t('1W')}
|
|
</Button>
|
|
<Button
|
|
type='button'
|
|
variant='outline'
|
|
size='sm'
|
|
onClick={() => handleSetExpiry(0, 1, 0)}
|
|
>
|
|
{t('1 Day')}
|
|
</Button>
|
|
</div>
|
|
</div>
|
|
<FormDescription>
|
|
{t('Leave empty for never expires')}
|
|
</FormDescription>
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
|
|
{!isUpdate && (
|
|
<FormField
|
|
control={form.control}
|
|
name='count'
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>{t('Quantity')}</FormLabel>
|
|
<FormControl>
|
|
<Input
|
|
{...field}
|
|
type='number'
|
|
min='1'
|
|
max='100'
|
|
placeholder={t('Number of codes to create')}
|
|
onChange={(e) =>
|
|
field.onChange(parseInt(e.target.value, 10) || 1)
|
|
}
|
|
/>
|
|
</FormControl>
|
|
<FormDescription>
|
|
{t('Create multiple redemption codes at once (1-100)')}
|
|
</FormDescription>
|
|
<FormMessage />
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
)}
|
|
</form>
|
|
</Form>
|
|
<SheetFooter className='grid grid-cols-2 gap-2 border-t px-4 py-3 sm:flex sm:px-6 sm:py-4'>
|
|
<SheetClose asChild>
|
|
<Button variant='outline'>{t('Close')}</Button>
|
|
</SheetClose>
|
|
<Button form='redemption-form' type='submit' disabled={isSubmitting}>
|
|
{isSubmitting ? t('Saving...') : t('Save changes')}
|
|
</Button>
|
|
</SheetFooter>
|
|
</SheetContent>
|
|
</Sheet>
|
|
)
|
|
}
|