mango-ui-v3/components/MarketPosition.tsx

281 lines
8.7 KiB
TypeScript
Raw Normal View History

2021-08-24 20:15:57 -07:00
import { useCallback, useMemo, useState } from 'react'
import { ElementTitle } from './styles'
2021-09-13 14:14:59 -07:00
import useMangoStore from '../stores/useMangoStore'
import { formatUsdValue } from '../utils/index'
2021-08-24 20:15:57 -07:00
import Button, { LinkButton } from './Button'
import Tooltip from './Tooltip'
import PerpSideBadge from './PerpSideBadge'
2021-08-14 13:05:31 -07:00
import {
getMarketIndexBySymbol,
MangoAccount,
2021-08-14 13:05:31 -07:00
PerpAccount,
PerpMarket,
QUOTE_INDEX,
} from '@blockworks-foundation/mango-client'
import { notify } from '../utils/notifications'
2021-08-24 20:15:57 -07:00
import MarketCloseModal from './MarketCloseModal'
import PnlText from './PnlText'
2021-09-02 14:35:37 -07:00
import Loading from './Loading'
import { useViewport } from '../hooks/useViewport'
import { breakpoints } from './TradePageGrid'
import { useTranslation } from 'next-i18next'
import useMangoAccount from '../hooks/useMangoAccount'
2021-08-14 13:05:31 -07:00
export const settlePnl = async (
perpMarket: PerpMarket,
perpAccount: PerpAccount,
t,
mangoAccounts: MangoAccount[] | undefined
) => {
2021-08-14 13:05:31 -07:00
const mangoAccount = useMangoStore.getState().selectedMangoAccount.current
const mangoGroup = useMangoStore.getState().selectedMangoGroup.current
const mangoCache = useMangoStore.getState().selectedMangoGroup.cache
const wallet = useMangoStore.getState().wallet.current
2021-09-03 09:14:11 -07:00
const actions = useMangoStore.getState().actions
2021-08-14 13:05:31 -07:00
const marketIndex = mangoGroup.getPerpMarketIndex(perpMarket.publicKey)
2021-09-13 14:14:59 -07:00
const mangoClient = useMangoStore.getState().connection.client
2021-08-14 13:05:31 -07:00
try {
const txid = await mangoClient.settlePnl(
mangoGroup,
mangoCache,
2021-08-14 13:05:31 -07:00
mangoAccount,
perpMarket,
mangoGroup.rootBankAccounts[QUOTE_INDEX],
mangoCache.priceCache[marketIndex].price,
wallet,
mangoAccounts
2021-08-14 13:05:31 -07:00
)
2021-09-03 09:14:11 -07:00
actions.reloadMangoAccount()
2021-08-14 13:05:31 -07:00
notify({
title: t('pnl-success'),
2021-08-14 13:05:31 -07:00
description: '',
txid,
})
} catch (e) {
console.log('Error settling PNL: ', `${e}`, `${perpAccount}`)
notify({
title: t('pnl-error'),
2021-08-14 13:05:31 -07:00
description: e.message,
txid: e.txid,
type: 'error',
})
}
}
export default function MarketPosition() {
const { t } = useTranslation('common')
2021-08-14 13:05:31 -07:00
const mangoGroup = useMangoStore((s) => s.selectedMangoGroup.current)
const mangoGroupConfig = useMangoStore((s) => s.selectedMangoGroup.config)
const mangoCache = useMangoStore((s) => s.selectedMangoGroup.cache)
const { mangoAccount, initialLoad } = useMangoAccount()
2021-08-14 13:05:31 -07:00
const selectedMarket = useMangoStore((s) => s.selectedMarket.current)
const marketConfig = useMangoStore((s) => s.selectedMarket.config)
2021-08-14 13:05:31 -07:00
const connected = useMangoStore((s) => s.wallet.connected)
2021-08-20 04:51:29 -07:00
const setMangoStore = useMangoStore((s) => s.set)
const price = useMangoStore((s) => s.tradeForm.price)
2022-02-10 09:49:59 -08:00
const perpAccounts = useMangoStore((s) => s.selectedMangoAccount.perpAccounts)
const baseSymbol = marketConfig.baseSymbol
2021-08-14 13:05:31 -07:00
const marketName = marketConfig.name
2021-08-24 20:15:57 -07:00
const [showMarketCloseModal, setShowMarketCloseModal] = useState(false)
2021-09-02 14:35:37 -07:00
const [settling, setSettling] = useState(false)
const { width } = useViewport()
const isMobile = width ? width < breakpoints.sm : false
2021-08-14 13:05:31 -07:00
const marketIndex = useMemo(() => {
return getMarketIndexBySymbol(mangoGroupConfig, baseSymbol)
}, [mangoGroupConfig, baseSymbol])
let perpAccount
if (marketName.includes('PERP') && mangoAccount) {
perpAccount = mangoAccount.perpAccounts[marketIndex]
}
2021-08-14 13:05:31 -07:00
const handleSizeClick = (size, side) => {
const step = selectedMarket.minOrderSize
const priceOrDefault = price
? price
: mangoGroup.getPrice(marketIndex, mangoCache).toNumber()
const roundedSize = Math.round(size / step) * step
const quoteSize = roundedSize * priceOrDefault
2021-08-20 04:51:29 -07:00
setMangoStore((state) => {
state.tradeForm.baseSize = roundedSize
2022-02-10 09:49:59 -08:00
state.tradeForm.quoteSize = quoteSize
state.tradeForm.side = side === 'buy' ? 'sell' : 'buy'
2021-08-20 04:51:29 -07:00
})
}
2021-08-24 20:15:57 -07:00
const handleCloseWarning = useCallback(() => {
setShowMarketCloseModal(false)
}, [])
2021-09-02 14:35:37 -07:00
const handleSettlePnl = (perpMarket, perpAccount) => {
setSettling(true)
settlePnl(perpMarket, perpAccount, t, undefined).then(() => {
2021-09-02 14:35:37 -07:00
setSettling(false)
})
}
if (!mangoGroup || !selectedMarket || !(selectedMarket instanceof PerpMarket))
return null
const {
2022-01-27 19:57:18 -08:00
basePosition = 0,
avgEntryPrice = 0,
breakEvenPrice = 0,
notionalSize = 0,
unsettledPnl = 0,
} = perpAccounts.length
? perpAccounts.find((pa) =>
pa.perpMarket.publicKey.equals(selectedMarket.publicKey)
)
: {}
2021-08-17 08:10:32 -07:00
function SettlePnlTooltip() {
return (
<div>
{t('pnl-help')}{' '}
<a
2021-12-27 15:02:07 -08:00
href="https://docs.mango.markets/mango/settle-pnl"
target="_blank"
rel="noopener noreferrer"
>
{t('learn-more')}
</a>
</div>
)
}
return (
<>
2021-10-29 05:05:55 -07:00
<div
className={!connected && !isMobile ? 'filter blur-sm' : null}
2021-11-04 05:25:11 -07:00
id="perp-positions-tip"
2021-10-29 05:05:55 -07:00
>
{!isMobile ? (
<ElementTitle>
{marketConfig.name} {t('position')}
</ElementTitle>
) : null}
2022-01-31 07:52:28 -08:00
<div className="flex items-center justify-between pb-2">
2022-01-16 01:05:17 -08:00
<div className="font-normal text-th-fgd-3 leading-4">{t('side')}</div>
{initialLoad ? (
2021-08-21 06:02:51 -07:00
<DataLoader />
2021-08-14 13:05:31 -07:00
) : (
<PerpSideBadge perpAccount={perpAccount}></PerpSideBadge>
2021-08-14 13:05:31 -07:00
)}
</div>
2022-01-31 07:52:28 -08:00
<div className="flex justify-between pb-2">
2021-08-14 13:05:31 -07:00
<div className="font-normal text-th-fgd-3 leading-4">
{t('position-size')}
2021-08-14 13:05:31 -07:00
</div>
<div className="text-th-fgd-1">
{initialLoad ? (
2021-08-21 06:02:51 -07:00
<DataLoader />
) : basePosition ? (
2021-08-20 04:51:29 -07:00
<span
className="cursor-pointer underline hover:no-underline"
onClick={() =>
handleSizeClick(
Math.abs(basePosition),
basePosition > 0 ? 'buy' : 'sell'
2021-08-20 04:51:29 -07:00
)
}
>
{`${Math.abs(basePosition)} ${baseSymbol}`}
2021-08-20 04:51:29 -07:00
</span>
) : (
`0 ${baseSymbol}`
)}
2021-08-14 13:05:31 -07:00
</div>
</div>
2022-01-31 07:52:28 -08:00
<div className="flex justify-between pb-2">
<div className="font-normal text-th-fgd-3 leading-4">
{t('notional-size')}
2021-08-14 13:05:31 -07:00
</div>
<div className="text-th-fgd-1">
{initialLoad ? (
2021-08-21 06:02:51 -07:00
<DataLoader />
2021-11-10 03:46:54 -08:00
) : notionalSize ? (
formatUsdValue(Math.abs(notionalSize))
2021-11-10 03:46:54 -08:00
) : (
'$0'
2021-08-20 06:17:02 -07:00
)}
</div>
</div>
2022-01-31 07:52:28 -08:00
<div className="flex justify-between pb-2">
2021-08-14 13:05:31 -07:00
<div className="font-normal text-th-fgd-3 leading-4">
{t('average-entry')}
2021-08-14 13:05:31 -07:00
</div>
<div className="text-th-fgd-1">
{initialLoad ? (
2021-11-10 03:46:54 -08:00
<DataLoader />
) : avgEntryPrice ? (
formatUsdValue(avgEntryPrice)
) : (
'$0'
)}
2021-08-14 13:05:31 -07:00
</div>
</div>
2022-01-31 07:52:28 -08:00
<div className="flex justify-between pb-2">
<div className="font-normal text-th-fgd-3 leading-4">
{t('break-even')}
2021-08-14 13:05:31 -07:00
</div>
<div className="text-th-fgd-1">
{initialLoad ? (
2021-11-10 03:46:54 -08:00
<DataLoader />
) : breakEvenPrice ? (
formatUsdValue(breakEvenPrice)
) : (
'$0'
)}
</div>
</div>
2022-01-31 07:52:28 -08:00
<div className="flex justify-between pb-2">
2021-08-16 14:35:16 -07:00
<Tooltip content={<SettlePnlTooltip />}>
2021-08-16 10:00:43 -07:00
<Tooltip.Content className="font-normal text-th-fgd-3 leading-4">
{t('unsettled-balance')}
2021-08-16 10:00:43 -07:00
</Tooltip.Content>
</Tooltip>
<div className="flex items-center">
{initialLoad ? <DataLoader /> : <PnlText pnl={unsettledPnl} />}
2021-09-02 14:35:37 -07:00
{settling ? (
<Loading className="ml-2" />
) : (
<LinkButton
onClick={() => handleSettlePnl(selectedMarket, perpAccount)}
className="ml-2 text-th-primary text-xs disabled:cursor-not-allowed disabled:opacity-60 disabled:hover:underline"
disabled={unsettledPnl === 0}
2021-09-02 14:35:37 -07:00
>
2022-01-24 15:02:43 -08:00
{t('redeem-pnl')}
2021-09-02 14:35:37 -07:00
</LinkButton>
)}
2021-08-16 06:31:25 -07:00
</div>
</div>
{basePosition ? (
2021-08-24 20:15:57 -07:00
<Button
onClick={() => setShowMarketCloseModal(true)}
2021-09-21 02:50:53 -07:00
className="mt-2.5 w-full"
2021-08-24 20:15:57 -07:00
>
<span>{t('market-close')}</span>
2021-08-24 20:15:57 -07:00
</Button>
) : null}
</div>
2021-08-24 20:15:57 -07:00
{showMarketCloseModal ? (
<MarketCloseModal
isOpen={showMarketCloseModal}
onClose={handleCloseWarning}
market={selectedMarket}
2021-08-25 11:30:16 -07:00
marketIndex={marketIndex}
2021-08-24 20:15:57 -07:00
/>
) : null}
</>
)
}
2021-08-21 06:02:51 -07:00
export const DataLoader = () => (
<div className="animate-pulse bg-th-bkg-3 h-5 w-10 rounded-sm" />
)