mango-ui-v3/components/Notification.tsx

189 lines
5.9 KiB
TypeScript
Raw Normal View History

import { useEffect } from 'react'
2021-04-11 21:17:23 -07:00
import {
CheckCircleIcon,
2021-07-30 05:06:27 -07:00
ExternalLinkIcon,
2021-04-11 21:17:23 -07:00
InformationCircleIcon,
XCircleIcon,
} from '@heroicons/react/outline'
import useMangoStore, { CLUSTER } from '../stores/useMangoStore'
2021-12-09 09:23:19 -08:00
import { Notification, notify } from '../utils/notifications'
import { useTranslation } from 'next-i18next'
2021-12-09 09:23:19 -08:00
import Loading from './Loading'
2021-08-24 12:43:34 -07:00
2021-04-11 21:17:23 -07:00
const NotificationList = () => {
const { t } = useTranslation('common')
2021-04-11 21:17:23 -07:00
const notifications = useMangoStore((s) => s.notifications)
const walletTokens = useMangoStore((s) => s.wallet.tokens)
const notEnoughSoLMessage = t('not-enough-sol')
// if a notification is shown with {"InstructionError":[0,{"Custom":1}]} then
// add a notification letting the user know they may not have enough SOL
useEffect(() => {
if (notifications.length) {
const customErrorNotification = notifications.find(
(n) => n.description && n.description.includes('"Custom":1')
)
const notEnoughSolNotification = notifications.find(
(n) => n.title && n.title.includes(notEnoughSoLMessage)
)
const solBalance = walletTokens.find(
(t) => t.config.symbol === 'SOL'
)?.uiBalance
if (
customErrorNotification &&
solBalance < 0.04 &&
!notEnoughSolNotification
) {
notify({
title: notEnoughSoLMessage,
type: 'info',
})
}
}
}, [notifications, walletTokens])
2021-04-11 21:17:23 -07:00
2021-04-12 20:39:08 -07:00
const reversedNotifications = [...notifications].reverse()
return (
<div
className={`pointer-events-none fixed inset-0 z-50 flex items-end px-4 py-6 text-th-fgd-1 sm:p-6`}
>
<div className={`flex w-full flex-col`}>
{reversedNotifications.map((n) => (
<Notification key={n.id} notification={n} />
))}
</div>
</div>
)
}
2021-12-09 09:23:19 -08:00
const Notification = ({ notification }: { notification: Notification }) => {
const { t } = useTranslation('common')
const setMangoStore = useMangoStore((s) => s.set)
const { type, title, description, txid, show, id } = notification
2021-12-09 09:23:19 -08:00
// overwrite the title if of the error message if it is a time out error
let parsedTitle
2021-12-20 09:05:45 -08:00
if (description) {
if (
description?.includes('Timed out awaiting') ||
description?.includes('was not confirmed')
) {
2022-03-01 11:06:02 -08:00
parsedTitle = 'Transaction status unknown'
2021-12-20 09:05:45 -08:00
}
}
2021-12-09 09:23:19 -08:00
// if the notification is a success, then hide the confirming tx notification with the same txid
useEffect(() => {
if ((type === 'error' || type === 'success') && txid) {
setMangoStore((s) => {
const newNotifications = s.notifications.map((n) =>
n.txid === txid && n.type === 'confirm' ? { ...n, show: false } : n
)
s.notifications = newNotifications
})
}
}, [type, txid])
const hideNotification = () => {
setMangoStore((s) => {
const newNotifications = s.notifications.map((n) =>
n.id === id ? { ...n, show: false } : n
)
s.notifications = newNotifications
})
}
2021-12-09 09:23:19 -08:00
// auto hide a notification after 10 seconds unless it is a confirming or time out notification
useEffect(() => {
const id = setTimeout(
() => {
2021-12-19 10:17:43 -08:00
if (show) {
2021-12-09 09:23:19 -08:00
hideNotification()
}
},
2022-01-21 16:19:54 -08:00
parsedTitle || type === 'confirm' || type === 'error' ? 90000 : 8000
2021-12-09 09:23:19 -08:00
)
return () => {
clearInterval(id)
}
})
if (!show) return null
return (
<div
className={`pointer-events-auto mt-2 w-full max-w-sm overflow-hidden rounded-md border border-th-bkg-4 bg-th-bkg-3 shadow-lg ring-1 ring-black ring-opacity-5`}
>
<div className={`relative flex items-center px-2 py-2.5`}>
2021-07-30 05:06:27 -07:00
<div className={`flex-shrink-0`}>
{type === 'success' ? (
<CheckCircleIcon className={`mr-1 h-7 w-7 text-th-green`} />
2021-07-30 05:06:27 -07:00
) : null}
{type === 'info' && (
<InformationCircleIcon className={`mr-1 h-7 w-7 text-th-primary`} />
2021-07-30 05:06:27 -07:00
)}
{type === 'error' && (
<XCircleIcon className={`mr-1 h-7 w-7 text-th-red`} />
2021-07-30 05:06:27 -07:00
)}
2021-12-09 09:23:19 -08:00
{type === 'confirm' && (
<Loading className="mr-1 h-7 w-7 text-th-fgd-3" />
2021-12-09 09:23:19 -08:00
)}
2021-07-30 05:06:27 -07:00
</div>
2021-12-05 22:48:20 -08:00
<div className={`ml-2 flex-1`}>
<div className={`text-normal font-bold text-th-fgd-1`}>
{parsedTitle || title}
</div>
2021-07-30 05:06:27 -07:00
{description ? (
<p className={`mb-0 mt-0.5 leading-tight text-th-fgd-3`}>
2022-01-10 14:25:05 -08:00
{description}
</p>
2021-07-30 05:06:27 -07:00
) : null}
{txid ? (
<a
href={
'https://explorer.solana.com/tx/' + txid + '?cluster=' + CLUSTER
}
className="mt-1 flex items-center text-sm"
2021-07-30 05:06:27 -07:00
target="_blank"
rel="noreferrer"
>
<div className="flex-1 break-all text-xs">
2021-12-05 22:48:20 -08:00
{type === 'error'
? txid
: `${txid.slice(0, 14)}...${txid.slice(txid.length - 14)}`}
</div>
<ExternalLinkIcon className="mb-0.5 ml-1 h-4 w-4" />
2021-07-30 05:06:27 -07:00
</a>
) : null}
</div>
<div className={`absolute right-2 top-2 flex-shrink-0`}>
2021-07-30 05:06:27 -07:00
<button
onClick={hideNotification}
2021-07-30 05:06:27 -07:00
className={`text-th-fgd-4 hover:text-th-primary focus:outline-none`}
>
<span className={`sr-only`}>{t('close')}</span>
2021-07-30 05:06:27 -07:00
<svg
className={`h-5 w-5`}
xmlns="http://www.w3.org/2000/svg"
viewBox="0 0 20 20"
fill="currentColor"
aria-hidden="true"
>
2021-07-30 05:06:27 -07:00
<path
fillRule="evenodd"
d="M4.293 4.293a1 1 0 011.414 0L10 8.586l4.293-4.293a1 1 0 111.414 1.414L11.414 10l4.293 4.293a1 1 0 01-1.414 1.414L10 11.414l-4.293 4.293a1 1 0 01-1.414-1.414L8.586 10 4.293 5.707a1 1 0 010-1.414z"
clipRule="evenodd"
/>
</svg>
</button>
</div>
</div>
</div>
)
}
export default NotificationList