stacker.news/components/toast.js

212 lines
6.3 KiB
JavaScript
Raw Normal View History

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)
2024-02-01 13:18:42 +00:00
const dispatchToast = useCallback((toast) => {
toast = {
...toast,
id: toastId.current++
}
const { flowId } = toast
setToasts(toasts => {
if (flowId) {
// replace previous toast with same flow id
const idx = toasts.findIndex(toast => toast.flowId === flowId)
if (idx === -1) return [...toasts, toast]
return [
...toasts.slice(0, idx),
toast,
...toasts.slice(idx + 1)
]
}
return [...toasts, toast]
})
return () => removeToast(toast)
}, [])
2024-02-01 13:18:42 +00:00
const removeToast = useCallback(({ id, onCancel, tag }) => {
setToasts(toasts => toasts.filter(toast => {
2024-02-18 19:33:32 +00:00
if (toast.id === id) {
// remove the toast with the passed id with no exceptions
return false
2024-02-01 13:18:42 +00:00
}
2024-02-18 19:33:32 +00:00
const sameTag = tag && tag === toast.tag
if (!sameTag) {
// don't touch toasts with different tags
return true
}
const toRemoveHasCancel = !!toast.onCancel
if (toRemoveHasCancel) {
// don't remove this toast so the user can decide to cancel this toast now
return true
}
// remove toasts with same tag if they are not cancelable
return false
2024-02-01 13:18:42 +00:00
}))
}, [])
const toaster = useMemo(() => ({
2024-02-01 13:18:42 +00:00
success: (body, options) => {
const toast = {
body,
variant: 'success',
autohide: true,
2024-02-01 13:18:42 +00:00
delay: 5000,
2024-02-01 13:50:51 +00:00
tag: options?.tag || body,
2024-02-01 13:18:42 +00:00
...options
}
return dispatchToast(toast)
},
2024-02-01 13:18:42 +00:00
warning: (body, options) => {
const toast = {
body,
variant: 'warning',
autohide: true,
2024-02-01 13:18:42 +00:00
delay: 5000,
2024-02-01 13:50:51 +00:00
tag: options?.tag || body,
2024-02-01 13:18:42 +00:00
...options
}
return dispatchToast(toast)
},
2024-02-01 13:18:42 +00:00
danger: (body, options) => {
const toast = {
body,
variant: 'danger',
autohide: false,
2024-02-01 13:50:51 +00:00
tag: options?.tag || body,
2024-02-01 13:18:42 +00:00
...options
}
2024-02-01 13:18:42 +00:00
return dispatchToast(toast)
}
}), [dispatchToast, removeToast])
2024-02-01 13:18:42 +00:00
// Only clear toasts with no cancel function on page navigation
// since navigation should not interfere with being able to cancel an action.
useEffect(() => {
2024-02-01 13:18:42 +00:00
const handleRouteChangeStart = () => setToasts(toasts => toasts.filter(({ onCancel }) => onCancel), [])
router.events.on('routeChangeStart', handleRouteChangeStart)
return () => {
router.events.off('routeChangeStart', handleRouteChangeStart)
}
}, [router])
2024-02-01 13:18:42 +00:00
// this function merges toasts with the same tag into one toast.
// for example: 3x 'zap pending' -> '(3) zap pending'
const tagReducer = (toasts, toast) => {
const { tag } = toast
// has tag?
if (!tag) return [...toasts, toast]
// existing tag?
const idx = toasts.findIndex(toast => toast.tag === tag)
if (idx === -1) return [...toasts, toast]
// merge toasts with same tag
const prevToast = toasts[idx]
let { rawBody, body, amount } = prevToast
rawBody ??= body
amount = amount ? amount + 1 : 2
body = `(${amount}) ${rawBody}`
return [
...toasts.slice(0, idx),
{ ...toast, rawBody, amount, body },
...toasts.slice(idx + 1)
]
}
// only show toast with highest ID of each tag
const visibleToasts = toasts.reduce(tagReducer, [])
return (
<ToastContext.Provider value={toaster}>
<ToastContainer className={`pb-3 pe-3 ${styles.toastContainer}`} position='bottom-end' containerPosition='fixed'>
2024-02-01 13:18:42 +00:00
{visibleToasts.map(toast => {
const textStyle = toast.variant === 'warning' ? 'text-dark' : ''
return (
<Toast
key={toast.id} bg={toast.variant} show autohide={toast.autohide}
2024-02-15 22:49:54 +00:00
delay={toast.delay} className={`${styles.toast} ${styles[toast.variant]} ${textStyle}`} onClose={() => removeToast(toast)}
2024-02-01 13:18:42 +00: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'
onClick={() => {
toast.onCancel?.()
toast.onClose?.()
removeToast(toast)
}}
>{toast.onCancel ? <div className={`${styles.toastCancel} ${textStyle}`}>cancel</div> : <div className={`${styles.toastClose} ${textStyle}`}>X</div>}
</Button>
</div>
</ToastBody>
</Toast>
)
})}
</ToastContainer>
{children}
</ToastContext.Provider>
)
}
export const useToast = () => useContext(ToastContext)
export const withToastFlow = (toaster) => flowFn => {
const wrapper = async (...args) => {
const {
flowId,
type: t,
onPending,
onSuccess,
onCancel,
onError
} = flowFn(...args)
let canceled
toaster.warning(`${t} pending`, {
autohide: false,
onCancel: async () => {
try {
await onCancel?.()
canceled = true
toaster.warning(`${t} canceled`, { flowId })
} catch (err) {
toaster.danger(`failed to cancel ${t}`, { flowId })
}
},
flowId
})
try {
const ret = await onPending()
if (!canceled) {
toaster.success(`${t} successful`, { flowId })
await onSuccess?.()
}
return ret
} catch (err) {
// ignore errors if canceled since they might be caused by cancellation
if (canceled) return
const reason = err?.message?.toString().toLowerCase() || 'unknown reason'
toaster.danger(`${t} failed: ${reason}`, { flowId })
await onError?.()
throw err
}
}
return wrapper
}