feat: integrate toast notifications for success and error handling

This commit is contained in:
Ardeman 2025-03-12 20:24:54 +08:00
parent c0318fac4c
commit f423f3e1c0
3 changed files with 36 additions and 30 deletions

View File

@ -1,5 +1,6 @@
import { zodResolver } from '@hookform/resolvers/zod' import { zodResolver } from '@hookform/resolvers/zod'
import { useEffect, useState } from 'react' import { useEffect } from 'react'
import toast from 'react-hot-toast'
import { useFetcher, useNavigate } from 'react-router' import { useFetcher, useNavigate } from 'react-router'
import { RemixFormProvider, useRemixForm } from 'remix-hook-form' import { RemixFormProvider, useRemixForm } from 'remix-hook-form'
import { z } from 'zod' import { z } from 'zod'
@ -38,20 +39,22 @@ export const FormAdvertisementsPage = (properties: TProperties) => {
url: adData?.url || '', url: adData?.url || '',
}, },
}) })
const [error, setError] = useState<string>()
const { handleSubmit } = formMethods const { handleSubmit } = formMethods
useEffect(() => { useEffect(() => {
if (!fetcher.data?.success) { if (fetcher.data?.success === false) {
setError(fetcher.data?.message) toast.error(fetcher.data?.message)
return return
} }
navigate('/lg-admin/advertisements') if (fetcher.data?.success === true) {
setError(undefined) toast.success(`Banner iklan berhasil ${adData ? 'diupdate' : 'dibuat'}!`)
navigate('/lg-admin/advertisements')
return
}
// eslint-disable-next-line react-hooks/exhaustive-deps // eslint-disable-next-line react-hooks/exhaustive-deps
}, [fetcher]) }, [fetcher.data])
return ( return (
<div className="relative"> <div className="relative">
@ -64,9 +67,6 @@ export const FormAdvertisementsPage = (properties: TProperties) => {
action={`/actions/admin/advertisements/${adData ? 'update' : 'create'}`} action={`/actions/admin/advertisements/${adData ? 'update' : 'create'}`}
className="space-y-4" className="space-y-4"
> >
{error && (
<div className="text-sm text-red-500 capitalize">{error}</div>
)}
<div className="flex items-end justify-between gap-4"> <div className="flex items-end justify-between gap-4">
<InputFile <InputFile
id="image" id="image"

View File

@ -1,5 +1,6 @@
import { zodResolver } from '@hookform/resolvers/zod' import { zodResolver } from '@hookform/resolvers/zod'
import { useEffect, useState } from 'react' import { useEffect } from 'react'
import toast from 'react-hot-toast'
import { useFetcher, useNavigate } from 'react-router' import { useFetcher, useNavigate } from 'react-router'
import { RemixFormProvider, useRemixForm } from 'remix-hook-form' import { RemixFormProvider, useRemixForm } from 'remix-hook-form'
import { z } from 'zod' import { z } from 'zod'
@ -38,20 +39,25 @@ export const FormCategoryPage = (properties: TProperties) => {
description: categoryData?.description || '', description: categoryData?.description || '',
}, },
}) })
const [error, setError] = useState<string>()
const { handleSubmit, watch, setValue } = formMethods const { handleSubmit, watch, setValue } = formMethods
const watchName = watch('name') const watchName = watch('name')
useEffect(() => { useEffect(() => {
if (!fetcher.data?.success) { if (fetcher.data?.success === false) {
setError(fetcher.data?.message) toast.error(fetcher.data?.message)
return
}
if (fetcher.data?.success === true) {
toast.success(
`Kategori berhasil ${categoryData ? 'diupdate' : 'dibuat'}!`,
)
navigate('/lg-admin/categories')
return return
} }
navigate('/lg-admin/categories')
setError(undefined)
// eslint-disable-next-line react-hooks/exhaustive-deps // eslint-disable-next-line react-hooks/exhaustive-deps
}, [fetcher]) }, [fetcher.data])
useEffect(() => { useEffect(() => {
setValue('code', urlFriendlyCode(watchName)) setValue('code', urlFriendlyCode(watchName))
@ -69,9 +75,6 @@ export const FormCategoryPage = (properties: TProperties) => {
action={`/actions/admin/categories/${categoryData ? 'update' : 'create'}`} action={`/actions/admin/categories/${categoryData ? 'update' : 'create'}`}
className="space-y-4" className="space-y-4"
> >
{error && (
<div className="text-sm text-red-500 capitalize">{error}</div>
)}
<div className="flex items-end justify-between gap-4"> <div className="flex items-end justify-between gap-4">
<Input <Input
id="name" id="name"

View File

@ -1,6 +1,7 @@
import { Field, Label, Select } from '@headlessui/react' import { Field, Label, Select } from '@headlessui/react'
import { zodResolver } from '@hookform/resolvers/zod' import { zodResolver } from '@hookform/resolvers/zod'
import { useEffect, useState } from 'react' import { useEffect } from 'react'
import toast from 'react-hot-toast'
import { useFetcher, useNavigate } from 'react-router' import { useFetcher, useNavigate } from 'react-router'
import { RemixFormProvider, useRemixForm } from 'remix-hook-form' import { RemixFormProvider, useRemixForm } from 'remix-hook-form'
import { z } from 'zod' import { z } from 'zod'
@ -40,20 +41,25 @@ export const FormSubscribePlanPage = (properties: TProperties) => {
status: subscribePlanData?.status || undefined, status: subscribePlanData?.status || undefined,
}, },
}) })
const [error, setError] = useState<string>()
const { handleSubmit, watch, setValue } = formMethods const { handleSubmit, watch, setValue } = formMethods
const watchName = watch('name') const watchName = watch('name')
useEffect(() => { useEffect(() => {
if (!fetcher.data?.success) { if (fetcher.data?.success === false) {
setError(fetcher.data?.message) toast.error(fetcher.data?.message)
return
}
if (fetcher.data?.success === true) {
toast.success(
`Subscribe Plan berhasil ${subscribePlanData ? 'diupdate' : 'dibuat'}!`,
)
navigate('/lg-admin/subscribe-plan')
return return
} }
navigate('/lg-admin/subscribe-plan')
setError(undefined)
// eslint-disable-next-line react-hooks/exhaustive-deps // eslint-disable-next-line react-hooks/exhaustive-deps
}, [fetcher]) }, [fetcher.data])
useEffect(() => { useEffect(() => {
setValue('code', urlFriendlyCode(watchName)) setValue('code', urlFriendlyCode(watchName))
@ -73,9 +79,6 @@ export const FormSubscribePlanPage = (properties: TProperties) => {
action={`/actions/admin/subscribe-plan/${subscribePlanData ? 'update' : 'create'}`} action={`/actions/admin/subscribe-plan/${subscribePlanData ? 'update' : 'create'}`}
className="space-y-4" className="space-y-4"
> >
{error && (
<div className="text-sm text-red-500 capitalize">{error}</div>
)}
<div className="flex items-end justify-between gap-4"> <div className="flex items-end justify-between gap-4">
<Input <Input
id="name" id="name"