2023-08-25 19:21:51 -04:00
|
|
|
import { useRouter } from 'next/router'
|
|
|
|
import React, { createContext, useCallback, useContext, useEffect, useMemo, useRef, useState } from 'react'
|
|
|
|
import Button from 'react-bootstrap/Button'
|
|
|
|
import Toast from 'react-bootstrap/Toast'
|
|
|
|
import ToastBody from 'react-bootstrap/ToastBody'
|
|
|
|
import ToastContainer from 'react-bootstrap/ToastContainer'
|
|
|
|
import styles from './toast.module.css'
|
|
|
|
|
|
|
|
const ToastContext = createContext(() => {})
|
|
|
|
|
|
|
|
export const ToastProvider = ({ children }) => {
|
|
|
|
const router = useRouter()
|
|
|
|
const [toasts, setToasts] = useState([])
|
|
|
|
const toastId = useRef(0)
|
|
|
|
const dispatchToast = useCallback((toastConfig) => {
|
|
|
|
toastConfig = {
|
|
|
|
...toastConfig,
|
|
|
|
id: toastId.current++
|
|
|
|
}
|
|
|
|
setToasts(toasts => [...toasts, toastConfig])
|
|
|
|
}, [])
|
2023-10-04 13:47:09 -05:00
|
|
|
|
|
|
|
const removeToast = useCallback(id => {
|
|
|
|
setToasts(toasts => toasts.filter(toast => toast.id !== id))
|
|
|
|
}, [])
|
|
|
|
|
2023-08-25 19:21:51 -04:00
|
|
|
const toaster = useMemo(() => ({
|
|
|
|
success: body => {
|
|
|
|
dispatchToast({
|
|
|
|
body,
|
|
|
|
variant: 'success',
|
|
|
|
autohide: true,
|
|
|
|
delay: 5000
|
|
|
|
})
|
|
|
|
},
|
2023-10-04 13:47:09 -05:00
|
|
|
danger: (body, onCloseCallback) => {
|
|
|
|
const id = toastId.current
|
2023-08-25 19:21:51 -04:00
|
|
|
dispatchToast({
|
2023-10-04 13:47:09 -05:00
|
|
|
id,
|
2023-08-25 19:21:51 -04:00
|
|
|
body,
|
|
|
|
variant: 'danger',
|
2023-10-04 13:47:09 -05:00
|
|
|
autohide: false,
|
|
|
|
onCloseCallback
|
2023-08-25 19:21:51 -04:00
|
|
|
})
|
2023-10-04 13:47:09 -05:00
|
|
|
return {
|
|
|
|
removeToast: () => removeToast(id)
|
|
|
|
}
|
2023-08-25 19:21:51 -04:00
|
|
|
}
|
|
|
|
}), [dispatchToast])
|
|
|
|
|
|
|
|
// Clear all toasts on page navigation
|
|
|
|
useEffect(() => {
|
|
|
|
const handleRouteChangeStart = () => setToasts([])
|
|
|
|
router.events.on('routeChangeStart', handleRouteChangeStart)
|
|
|
|
|
|
|
|
return () => {
|
|
|
|
router.events.off('routeChangeStart', handleRouteChangeStart)
|
|
|
|
}
|
|
|
|
}, [router])
|
|
|
|
|
|
|
|
return (
|
|
|
|
<ToastContext.Provider value={toaster}>
|
2023-08-30 22:16:47 -04:00
|
|
|
<ToastContainer className={`pb-3 pe-3 ${styles.toastContainer}`} position='bottom-end' containerPosition='fixed'>
|
2023-08-25 19:21:51 -04:00
|
|
|
{toasts.map(toast => (
|
|
|
|
<Toast
|
|
|
|
key={toast.id} bg={toast.variant} show autohide={toast.autohide}
|
2023-09-11 15:57:41 -04:00
|
|
|
delay={toast.delay} className={`${styles.toast} ${styles[toast.variant]}`} onClose={() => removeToast(toast.id)}
|
2023-08-25 19:21:51 -04:00
|
|
|
>
|
|
|
|
<ToastBody>
|
|
|
|
<div className='d-flex align-items-center'>
|
|
|
|
<div className='flex-grow-1'>{toast.body}</div>
|
|
|
|
<Button
|
|
|
|
variant={null}
|
|
|
|
className='p-0 ps-2'
|
|
|
|
aria-label='close'
|
2023-10-04 13:47:09 -05:00
|
|
|
onClick={() => {
|
|
|
|
if (toast.onCloseCallback) toast.onCloseCallback()
|
|
|
|
removeToast(toast.id)
|
|
|
|
}}
|
2023-08-25 19:21:51 -04:00
|
|
|
><div className={styles.toastClose}>X</div>
|
|
|
|
</Button>
|
|
|
|
</div>
|
|
|
|
</ToastBody>
|
|
|
|
</Toast>
|
|
|
|
))}
|
|
|
|
</ToastContainer>
|
|
|
|
{children}
|
|
|
|
</ToastContext.Provider>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
export const useToast = () => useContext(ToastContext)
|