2022-01-02 20:42:59 -08:00
|
|
|
import React, { FunctionComponent, useEffect, useState } from 'react'
|
|
|
|
import { PlusCircleIcon, TrashIcon } from '@heroicons/react/outline'
|
|
|
|
import Modal from './Modal'
|
|
|
|
import Input from './Input'
|
|
|
|
import { ElementTitle } from './styles'
|
|
|
|
import useMangoStore from '../stores/useMangoStore'
|
|
|
|
import Button, { LinkButton } from './Button'
|
|
|
|
import { notify } from '../utils/notifications'
|
|
|
|
import { useTranslation } from 'next-i18next'
|
|
|
|
import ButtonGroup from './ButtonGroup'
|
|
|
|
import InlineNotification from './InlineNotification'
|
|
|
|
|
|
|
|
interface CreateAlertModalProps {
|
|
|
|
onClose: () => void
|
|
|
|
isOpen: boolean
|
|
|
|
repayAmount?: string
|
|
|
|
tokenSymbol?: string
|
|
|
|
}
|
|
|
|
|
|
|
|
const CreateAlertModal: FunctionComponent<CreateAlertModalProps> = ({
|
|
|
|
isOpen,
|
|
|
|
onClose,
|
|
|
|
}) => {
|
|
|
|
const { t } = useTranslation('common')
|
|
|
|
const actions = useMangoStore((s) => s.actions)
|
|
|
|
const mangoGroup = useMangoStore((s) => s.selectedMangoGroup.current)
|
|
|
|
const mangoAccount = useMangoStore((s) => s.selectedMangoAccount.current)
|
|
|
|
const activeAlerts = useMangoStore((s) => s.alerts.activeAlerts)
|
|
|
|
const loading = useMangoStore((s) => s.alerts.loading)
|
|
|
|
const submitting = useMangoStore((s) => s.alerts.submitting)
|
|
|
|
const error = useMangoStore((s) => s.alerts.error)
|
|
|
|
const [email, setEmail] = useState<string>('')
|
|
|
|
const [invalidAmountMessage, setInvalidAmountMessage] = useState('')
|
|
|
|
const [health, setHealth] = useState('')
|
|
|
|
const [showCustomHealthForm, setShowCustomHealthForm] = useState(false)
|
|
|
|
const [showAlertForm, setShowAlertForm] = useState(false)
|
|
|
|
|
|
|
|
const healthPresets = ['5', '10', '15', '25', '30']
|
|
|
|
|
|
|
|
const validateEmailInput = (amount) => {
|
|
|
|
if (Number(amount) <= 0) {
|
|
|
|
setInvalidAmountMessage(t('enter-amount'))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const onChangeEmailInput = (amount) => {
|
|
|
|
setEmail(amount)
|
|
|
|
setInvalidAmountMessage('')
|
|
|
|
}
|
|
|
|
|
|
|
|
async function onCreateAlert() {
|
|
|
|
if (!email) {
|
|
|
|
notify({
|
|
|
|
title: 'An email address is required',
|
|
|
|
type: 'error',
|
|
|
|
})
|
|
|
|
return
|
|
|
|
} else if (!health) {
|
|
|
|
notify({
|
|
|
|
title: 'Alert health is required',
|
|
|
|
type: 'error',
|
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
const body = {
|
|
|
|
mangoGroupPk: mangoGroup.publicKey.toString(),
|
|
|
|
mangoAccountPk: mangoAccount.publicKey.toString(),
|
|
|
|
health,
|
|
|
|
alertProvider: 'mail',
|
|
|
|
email,
|
|
|
|
}
|
|
|
|
const success: any = await actions.createAlert(body)
|
|
|
|
if (success) {
|
|
|
|
setShowAlertForm(false)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const handleCancelCreateAlert = () => {
|
|
|
|
if (activeAlerts.length > 0) {
|
|
|
|
setShowAlertForm(false)
|
|
|
|
} else {
|
|
|
|
onClose()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
actions.loadAlerts(mangoAccount.publicKey)
|
|
|
|
}, [])
|
|
|
|
|
|
|
|
return (
|
|
|
|
<Modal isOpen={isOpen} onClose={onClose}>
|
|
|
|
{!loading && !submitting ? (
|
|
|
|
<>
|
|
|
|
{activeAlerts.length > 0 && !showAlertForm ? (
|
|
|
|
<>
|
|
|
|
<Modal.Header>
|
|
|
|
<div className="flex items-center justify-between w-full">
|
|
|
|
<div className="w-20" />
|
2022-01-31 07:52:28 -08:00
|
|
|
<ElementTitle noMarginBottom>
|
2022-01-02 20:42:59 -08:00
|
|
|
{t('active-alerts')}
|
|
|
|
</ElementTitle>
|
|
|
|
<Button
|
|
|
|
className="col-span-1 flex items-center justify-center pt-0 pb-0 h-8 text-xs w-20"
|
|
|
|
disabled={activeAlerts.length >= 3}
|
|
|
|
onClick={() => setShowAlertForm(true)}
|
|
|
|
>
|
|
|
|
<div className="flex items-center">
|
|
|
|
<PlusCircleIcon className="h-4 w-4 mr-1.5" />
|
|
|
|
{t('new')}
|
|
|
|
</div>
|
|
|
|
</Button>
|
|
|
|
</div>
|
|
|
|
</Modal.Header>
|
|
|
|
<div className="border-b border-th-fgd-4">
|
|
|
|
{activeAlerts.map((alert, index) => (
|
|
|
|
<div
|
|
|
|
className="border-t border-th-fgd-4 flex items-center justify-between p-4"
|
|
|
|
key={`${alert._id}${index}`}
|
|
|
|
>
|
|
|
|
<div className="text-th-fgd-1">
|
|
|
|
{t('alert-info', { health: alert.health })}
|
|
|
|
</div>
|
|
|
|
<TrashIcon
|
|
|
|
className="cursor-pointer default-transition h-5 text-th-fgd-3 w-5 hover:text-th-primary"
|
|
|
|
onClick={() => actions.deleteAlert(alert._id)}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
))}
|
|
|
|
</div>
|
|
|
|
{activeAlerts.length >= 3 ? (
|
|
|
|
<div className="mt-1 text-center text-xxs text-th-fgd-3">
|
|
|
|
{t('alerts-max')}
|
|
|
|
</div>
|
|
|
|
) : null}
|
|
|
|
</>
|
|
|
|
) : showAlertForm ? (
|
|
|
|
<>
|
|
|
|
<div>
|
2022-01-31 07:52:28 -08:00
|
|
|
<ElementTitle noMarginBottom>{t('create-alert')}</ElementTitle>
|
2022-01-02 20:42:59 -08:00
|
|
|
<p className="mt-1 text-center text-th-fgd-4">
|
|
|
|
{t('alerts-disclaimer')}
|
|
|
|
</p>
|
|
|
|
</div>
|
|
|
|
{error ? (
|
|
|
|
<div className="my-4">
|
|
|
|
<InlineNotification title={error} type="error" />
|
|
|
|
</div>
|
|
|
|
) : null}
|
|
|
|
<div className="mb-1.5 text-th-fgd-1">{t('email-address')}</div>
|
|
|
|
<Input
|
|
|
|
type="email"
|
|
|
|
className={`border border-th-fgd-4 flex-grow pr-11`}
|
|
|
|
error={!!invalidAmountMessage}
|
|
|
|
onBlur={(e) => validateEmailInput(e.target.value)}
|
|
|
|
value={email || ''}
|
|
|
|
onChange={(e) => onChangeEmailInput(e.target.value)}
|
|
|
|
/>
|
|
|
|
<div className="flex items-end mt-4">
|
|
|
|
<div className="w-full">
|
|
|
|
<div className="flex justify-between mb-1.5">
|
|
|
|
<div className="text-th-fgd-1">{t('alert-health')}</div>
|
|
|
|
<LinkButton
|
|
|
|
className="font-normal text-th-fgd-3 text-xs"
|
|
|
|
onClick={() =>
|
|
|
|
setShowCustomHealthForm(!showCustomHealthForm)
|
|
|
|
}
|
|
|
|
>
|
|
|
|
{showCustomHealthForm ? t('presets') : t('custom')}
|
|
|
|
</LinkButton>
|
|
|
|
</div>
|
|
|
|
{showCustomHealthForm ? (
|
|
|
|
<Input
|
|
|
|
type="number"
|
|
|
|
min="0"
|
|
|
|
max="100"
|
|
|
|
onChange={(e) => setHealth(e.target.value)}
|
|
|
|
suffix={
|
|
|
|
<div className="font-bold text-base text-th-fgd-3">
|
|
|
|
%
|
|
|
|
</div>
|
|
|
|
}
|
|
|
|
value={health}
|
|
|
|
/>
|
|
|
|
) : (
|
|
|
|
<ButtonGroup
|
|
|
|
activeValue={health.toString()}
|
|
|
|
onChange={(p) => setHealth(p)}
|
|
|
|
unit="%"
|
|
|
|
values={healthPresets}
|
|
|
|
/>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<div className="flex items-center mt-6">
|
|
|
|
<Button onClick={() => onCreateAlert()}>
|
|
|
|
{t('create-alert')}
|
|
|
|
</Button>
|
|
|
|
<LinkButton
|
|
|
|
className="ml-4 text-th-fgd-3 hover:text-th-fgd-1"
|
|
|
|
onClick={handleCancelCreateAlert}
|
|
|
|
>
|
|
|
|
{t('cancel')}
|
|
|
|
</LinkButton>
|
|
|
|
</div>
|
|
|
|
</>
|
|
|
|
) : error ? (
|
|
|
|
<div>
|
|
|
|
<InlineNotification title={error} type="error" />
|
|
|
|
<Button
|
|
|
|
className="flex justify-center mt-6 mx-auto"
|
|
|
|
onClick={() => actions.loadAlerts()}
|
|
|
|
>
|
|
|
|
{t('try-again')}
|
|
|
|
</Button>
|
|
|
|
</div>
|
|
|
|
) : (
|
|
|
|
<div>
|
|
|
|
<Modal.Header>
|
2022-01-31 07:52:28 -08:00
|
|
|
<ElementTitle noMarginBottom>{t('no-alerts')}</ElementTitle>
|
2022-01-02 20:42:59 -08:00
|
|
|
</Modal.Header>
|
|
|
|
<p className="mb-4 text-center">{t('no-alerts-desc')}</p>
|
|
|
|
<Button
|
|
|
|
className="flex justify-center m-auto"
|
|
|
|
onClick={() => setShowAlertForm(true)}
|
|
|
|
>
|
|
|
|
{t('new-alert')}
|
|
|
|
</Button>
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
</>
|
|
|
|
) : (
|
|
|
|
<div className="space-y-3">
|
|
|
|
<div className="animate-pulse bg-th-bkg-3 h-12 rounded-md w-full" />
|
|
|
|
<div className="animate-pulse bg-th-bkg-3 h-12 rounded-md w-full" />
|
|
|
|
<div className="animate-pulse bg-th-bkg-3 h-12 rounded-md w-full" />
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
</Modal>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
export default React.memo(CreateAlertModal)
|