mango-v4-ui/components/trade/CloseAllPositionsModal.tsx

129 lines
4.0 KiB
TypeScript

import { FunctionComponent, useCallback, useState } from 'react'
import mangoStore from '@store/mangoStore'
import { useTranslation } from 'next-i18next'
import Modal from '@components/shared/Modal'
import Button, { LinkButton } from '@components/shared/Button'
import { notify } from 'utils/notifications'
import Loading from '@components/shared/Loading'
import { isMangoError } from 'types'
import { ModalProps } from 'types/modal'
import useMangoGroup from 'hooks/useMangoGroup'
import useOpenPerpPositions from 'hooks/useOpenPerpPositions'
import { floorToDecimal, getDecimalCount } from 'utils/numbers'
import MarketLogos from './MarketLogos'
import PerpSideBadge from './PerpSideBadge'
import FormatNumericValue from '@components/shared/FormatNumericValue'
const CloseAllPositionsModal: FunctionComponent<ModalProps> = ({
onClose,
isOpen,
}) => {
const { t } = useTranslation(['common', 'trade'])
const [submitting, setSubmitting] = useState(false)
const openPerpPositions = useOpenPerpPositions()
const { group } = useMangoGroup()
const handleCloseAll = useCallback(async () => {
const client = mangoStore.getState().client
const mangoAccount = mangoStore.getState().mangoAccount.current
const actions = mangoStore.getState().actions
if (!group || !mangoAccount) {
notify({
title: 'Something went wrong. Try again later',
type: 'error',
})
return
}
setSubmitting(true)
try {
const maxSlippage = 0.025
const { signature: tx } = await client.perpCloseAll(
group,
mangoAccount,
maxSlippage,
)
actions.fetchOpenOrders()
notify({
type: 'success',
title: 'Transaction successful',
txid: tx,
})
} catch (e) {
if (isMangoError(e)) {
notify({
title: 'There was an issue.',
description: e.message,
txid: e?.txid,
type: 'error',
})
}
console.error('Place trade error:', e)
} finally {
setSubmitting(false)
onClose()
}
}, [group, onClose])
if (!group) return null
return (
<Modal onClose={onClose} isOpen={isOpen}>
<h3 className="mb-2 text-center">{t('trade:close-all-positions')}</h3>
<div className="pb-6 text-th-fgd-3">{t('trade:price-expect')}</div>
<div className="border-b border-th-bkg-3">
{openPerpPositions.map((position, i) => {
const market = group.getPerpMarketByMarketIndex(position.marketIndex)
const basePosition = position.getBasePositionUi(market)
const floorBasePosition = floorToDecimal(
basePosition,
getDecimalCount(market.minOrderSize),
).toNumber()
if (!basePosition) return null
return (
<div
className="flex items-center justify-between border-t border-th-bkg-3 py-3"
key={market.name + i}
>
<div className="flex items-center">
<MarketLogos market={market} />
<p className="mr-2">{market.name}</p>
<PerpSideBadge basePosition={basePosition} />
</div>
<p className="font-mono text-th-fgd-2">
<FormatNumericValue value={Math.abs(floorBasePosition)} />
<span className="mx-1 text-th-bkg-4">|</span>
<FormatNumericValue
value={Math.abs(floorBasePosition * market.uiPrice)}
isUsd
/>
</p>
</div>
)
})}
</div>
<Button
className="mb-4 mt-6 flex w-full items-center justify-center"
onClick={handleCloseAll}
size="large"
>
{submitting ? (
<Loading />
) : (
<span>{t('trade:close-all-positions')}</span>
)}
</Button>
<LinkButton
className="inline-flex w-full items-center justify-center"
onClick={onClose}
>
{t('cancel')}
</LinkButton>
</Modal>
)
}
export default CloseAllPositionsModal