2021-12-09 13:21:53 -08:00
|
|
|
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 from '../stores/useMangoStore'
|
2021-12-09 09:23:19 -08:00
|
|
|
import { Notification, notify } from '../utils/notifications'
|
2021-10-20 05:42:40 -07:00
|
|
|
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 = () => {
|
2021-10-20 05:42:40 -07:00
|
|
|
const { t } = useTranslation('common')
|
2021-04-11 21:17:23 -07:00
|
|
|
const notifications = useMangoStore((s) => s.notifications)
|
2021-08-24 09:28:25 -07:00
|
|
|
const walletTokens = useMangoStore((s) => s.wallet.tokens)
|
2021-10-20 05:42:40 -07:00
|
|
|
const notEnoughSoLMessage = t('not-enough-sol')
|
2021-08-24 09:28:25 -07:00
|
|
|
|
|
|
|
// 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()
|
|
|
|
|
2021-03-30 15:47:08 -07:00
|
|
|
return (
|
|
|
|
<div
|
2021-05-01 07:02:21 -07:00
|
|
|
className={`fixed inset-0 flex items-end px-4 py-6 pointer-events-none sm:p-6 text-th-fgd-1 z-50`}
|
2021-03-30 15:47:08 -07:00
|
|
|
>
|
2021-04-12 09:49:02 -07:00
|
|
|
<div className={`flex flex-col w-full`}>
|
2021-12-09 13:21:53 -08:00
|
|
|
{reversedNotifications.map((n) => (
|
|
|
|
<Notification key={n.id} notification={n} />
|
2021-03-30 15:47:08 -07:00
|
|
|
))}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2021-12-09 09:23:19 -08:00
|
|
|
const Notification = ({ notification }: { notification: Notification }) => {
|
2021-10-20 05:42:40 -07:00
|
|
|
const { t } = useTranslation('common')
|
2021-12-09 13:21:53 -08:00
|
|
|
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
|
2021-12-17 08:20:48 -08:00
|
|
|
let parsedTitle
|
2021-12-20 09:05:45 -08:00
|
|
|
if (description) {
|
|
|
|
if (
|
|
|
|
description?.includes('Timed out awaiting') ||
|
|
|
|
description?.includes('was not confirmed')
|
|
|
|
) {
|
|
|
|
parsedTitle = 'Unable to confirm transaction'
|
|
|
|
}
|
2021-12-17 08:20:48 -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])
|
|
|
|
|
2021-12-09 13:21:53 -08:00
|
|
|
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()
|
|
|
|
}
|
|
|
|
},
|
2021-12-29 10:39:17 -08:00
|
|
|
parsedTitle || type === 'confirm' || type === 'error' ? 30000 : 8000
|
2021-12-09 09:23:19 -08:00
|
|
|
)
|
|
|
|
|
|
|
|
return () => {
|
|
|
|
clearInterval(id)
|
|
|
|
}
|
|
|
|
})
|
2021-03-30 15:47:08 -07:00
|
|
|
|
2021-12-09 13:21:53 -08:00
|
|
|
if (!show) return null
|
2021-03-30 15:47:08 -07:00
|
|
|
|
|
|
|
return (
|
|
|
|
<div
|
2021-10-11 11:02:20 -07:00
|
|
|
className={`max-w-sm w-full bg-th-bkg-3 border border-th-bkg-4 shadow-lg rounded-md mt-2 pointer-events-auto ring-1 ring-black ring-opacity-5 overflow-hidden`}
|
2021-03-30 15:47:08 -07:00
|
|
|
>
|
2022-01-10 14:25:05 -08:00
|
|
|
<div className={`flex items-center px-2 py-2.5 relative`}>
|
2021-07-30 05:06:27 -07:00
|
|
|
<div className={`flex-shrink-0`}>
|
|
|
|
{type === 'success' ? (
|
|
|
|
<CheckCircleIcon className={`text-th-green h-7 w-7 mr-1`} />
|
|
|
|
) : null}
|
|
|
|
{type === 'info' && (
|
2021-08-24 09:28:25 -07:00
|
|
|
<InformationCircleIcon className={`text-th-primary h-7 w-7 mr-1`} />
|
2021-07-30 05:06:27 -07:00
|
|
|
)}
|
|
|
|
{type === 'error' && (
|
2021-08-24 09:28:25 -07:00
|
|
|
<XCircleIcon className={`text-th-red h-7 w-7 mr-1`} />
|
2021-07-30 05:06:27 -07:00
|
|
|
)}
|
2021-12-09 09:23:19 -08:00
|
|
|
{type === 'confirm' && (
|
2021-12-29 10:39:17 -08:00
|
|
|
<Loading className="text-th-fgd-3 h-7 w-7 mr-1" />
|
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`}>
|
2022-01-10 14:25:05 -08:00
|
|
|
<div className={`font-bold text-normal text-th-fgd-1`}>
|
2021-12-17 08:20:48 -08:00
|
|
|
{parsedTitle || title}
|
|
|
|
</div>
|
2021-07-30 05:06:27 -07:00
|
|
|
{description ? (
|
2022-01-10 14:25:05 -08:00
|
|
|
<p className={`mb-0 mt-0.5 text-th-fgd-3 leading-tight`}>
|
|
|
|
{description}
|
|
|
|
</p>
|
2021-07-30 05:06:27 -07:00
|
|
|
) : null}
|
|
|
|
{txid ? (
|
|
|
|
<a
|
|
|
|
href={'https://explorer.solana.com/tx/' + txid}
|
2022-01-10 14:25:05 -08:00
|
|
|
className="flex items-center mt-1 text-sm"
|
2021-07-30 05:06:27 -07:00
|
|
|
target="_blank"
|
|
|
|
rel="noreferrer"
|
|
|
|
>
|
2022-01-10 14:25:05 -08:00
|
|
|
<div className="break-all flex-1 text-xs">
|
2021-12-05 22:48:20 -08:00
|
|
|
{type === 'error'
|
|
|
|
? txid
|
|
|
|
: `${txid.slice(0, 14)}...${txid.slice(txid.length - 14)}`}
|
|
|
|
</div>
|
2021-07-30 05:06:27 -07:00
|
|
|
<ExternalLinkIcon className="h-4 mb-0.5 ml-1 w-4" />
|
|
|
|
</a>
|
|
|
|
) : null}
|
|
|
|
</div>
|
|
|
|
<div className={`absolute flex-shrink-0 right-2 top-2`}>
|
|
|
|
<button
|
2021-12-09 13:21:53 -08:00
|
|
|
onClick={hideNotification}
|
2021-07-30 05:06:27 -07:00
|
|
|
className={`text-th-fgd-4 hover:text-th-primary focus:outline-none`}
|
|
|
|
>
|
2021-10-20 05:42:40 -07:00
|
|
|
<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-03-30 15:47:08 -07:00
|
|
|
>
|
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>
|
2021-03-30 15:47:08 -07:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
export default NotificationList
|