mango-ui-v3/components/MarketPosition.tsx

413 lines
14 KiB
TypeScript
Raw Normal View History

2021-08-24 20:15:57 -07:00
import { useCallback, useMemo, useState } from 'react'
import FloatingElement from './FloatingElement'
import { ElementTitle } from './styles'
2021-08-14 13:05:31 -07:00
import useMangoStore, { mangoClient } from '../stores/useMangoStore'
import { i80f48ToPercent, formatUsdValue, floorToDecimal } from '../utils/index'
2021-08-24 20:15:57 -07:00
import Button, { LinkButton } from './Button'
import Tooltip from './Tooltip'
import SideBadge from './SideBadge'
2021-08-14 13:05:31 -07:00
import {
getMarketIndexBySymbol,
getTokenBySymbol,
2021-08-14 13:05:31 -07:00
nativeI80F48ToUi,
PerpAccount,
PerpMarket,
QUOTE_INDEX,
ZERO_BN,
2021-08-16 10:00:43 -07:00
ZERO_I80F48,
2021-08-14 13:05:31 -07:00
} from '@blockworks-foundation/mango-client'
import useTradeHistory from '../hooks/useTradeHistory'
import { getAvgEntryPrice, getBreakEvenPrice } from './PerpPositionsTable'
2021-08-14 13:05:31 -07:00
import { notify } from '../utils/notifications'
2021-08-24 20:15:57 -07:00
import MarketCloseModal from './MarketCloseModal'
2021-09-02 14:35:37 -07:00
import Loading from './Loading'
2021-08-14 13:05:31 -07:00
2021-09-02 14:35:37 -07:00
const settlePnl = async (perpMarket: PerpMarket, perpAccount: PerpAccount) => {
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)
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
)
2021-09-03 09:14:11 -07:00
actions.reloadMangoAccount()
2021-08-14 13:05:31 -07:00
notify({
title: 'Successfully settled PNL',
description: '',
txid,
})
} catch (e) {
console.log('Error settling PNL: ', `${e}`, `${perpAccount}`)
notify({
title: 'Error settling PNL',
description: e.message,
txid: e.txid,
type: 'error',
})
}
}
2021-08-16 14:35:16 -07:00
export function SettlePnlTooltip() {
return (
<div>
Settling will update your USDC balance to reflect the unsettled PnL
amount.{' '}
2021-08-16 14:35:16 -07:00
<a
href="https://docs.mango.markets/mango-v3/overview#settle-pnl"
target="_blank"
rel="noopener noreferrer"
>
Learn more
</a>
</div>
)
}
export default function MarketPosition() {
2021-08-14 13:05:31 -07:00
const mangoGroup = useMangoStore((s) => s.selectedMangoGroup.current)
const mangoGroupConfig = useMangoStore((s) => s.selectedMangoGroup.config)
const mangoGroupCache = useMangoStore((s) => s.selectedMangoGroup.cache)
const mangoAccount = useMangoStore((s) => s.selectedMangoAccount.current)
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 06:17:02 -07:00
const isLoading = useMangoStore((s) => s.selectedMangoAccount.initialLoad)
2021-08-20 04:51:29 -07:00
const setMangoStore = useMangoStore((s) => s.set)
const baseSymbol = marketConfig.baseSymbol
2021-08-14 13:05:31 -07:00
const marketName = marketConfig.name
const tradeHistory = useTradeHistory()
const perpTradeHistory = tradeHistory?.filter(
(t) => t.marketName === marketName
)
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)
2021-08-14 13:05:31 -07:00
const marketIndex = useMemo(() => {
return getMarketIndexBySymbol(mangoGroupConfig, baseSymbol)
}, [mangoGroupConfig, baseSymbol])
2021-09-03 00:00:59 -07:00
let perpAccount, perpPnl
if (marketName.includes('PERP') && mangoAccount) {
perpAccount = mangoAccount.perpAccounts[marketIndex]
2021-09-03 00:00:59 -07:00
perpPnl = perpAccount.getPnl(
mangoGroup.perpMarkets[marketIndex],
mangoGroupCache.perpMarketCache[marketIndex],
mangoGroupCache.priceCache[marketIndex].price
)
}
2021-08-14 13:05:31 -07:00
2021-08-20 04:51:29 -07:00
const handleSizeClick = (size) => {
setMangoStore((state) => {
state.tradeForm.baseSize = size
})
}
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).then(() => {
setSettling(false)
})
}
if (!mangoGroup || !selectedMarket) return null
2021-08-17 08:10:32 -07:00
2021-08-14 13:05:31 -07:00
return selectedMarket instanceof PerpMarket ? (
2021-07-22 04:34:03 -07:00
<FloatingElement showConnect>
2021-08-14 11:16:15 -07:00
<div className={!connected ? 'filter blur-sm' : null}>
2021-09-02 23:41:53 -07:00
<ElementTitle>Position: {marketConfig.name} </ElementTitle>
2021-08-21 05:11:14 -07:00
<div className={`flex items-center justify-between pb-3`}>
<div className="font-normal text-th-fgd-3 leading-4">Side</div>
2021-08-20 06:17:02 -07:00
{isLoading ? (
2021-08-21 06:02:51 -07:00
<DataLoader />
2021-08-20 06:17:02 -07:00
) : perpAccount && !perpAccount.basePosition.eq(ZERO_BN) ? (
2021-08-14 13:05:31 -07:00
<SideBadge
side={perpAccount.basePosition.gt(ZERO_BN) ? 'long' : 'short'}
/>
) : (
2021-08-20 04:51:29 -07:00
'--'
2021-08-14 13:05:31 -07:00
)}
</div>
2021-08-21 05:11:14 -07:00
<div className={`flex justify-between pb-3`}>
2021-08-14 13:05:31 -07:00
<div className="font-normal text-th-fgd-3 leading-4">
Position size
</div>
<div className={`text-th-fgd-1`}>
2021-08-20 06:17:02 -07:00
{isLoading ? (
2021-08-21 06:02:51 -07:00
<DataLoader />
2021-08-20 06:17:02 -07:00
) : perpAccount &&
Math.abs(
selectedMarket.baseLotsToNumber(perpAccount.basePosition)
) > 0 ? (
2021-08-20 04:51:29 -07:00
<span
className="cursor-pointer underline hover:no-underline"
onClick={() =>
handleSizeClick(
Math.abs(
selectedMarket.baseLotsToNumber(perpAccount.basePosition)
)
)
}
>
{`${Math.abs(
2021-08-17 14:01:00 -07:00
selectedMarket.baseLotsToNumber(perpAccount.basePosition)
2021-08-20 04:51:29 -07:00
)} ${baseSymbol}`}
</span>
) : (
`0 ${baseSymbol}`
)}
2021-08-14 13:05:31 -07:00
</div>
</div>
2021-08-21 05:11:14 -07:00
<div className={`flex justify-between pb-3`}>
<div className="font-normal text-th-fgd-3 leading-4">
2021-08-14 13:05:31 -07:00
Notional size
</div>
<div className={`text-th-fgd-1`}>
2021-08-20 06:17:02 -07:00
{isLoading ? (
2021-08-21 06:02:51 -07:00
<DataLoader />
2021-08-20 06:17:02 -07:00
) : perpAccount ? (
formatUsdValue(
2021-09-02 20:39:24 -07:00
Math.abs(
selectedMarket.baseLotsToNumber(perpAccount.basePosition) *
mangoGroup.getPrice(marketIndex, mangoGroupCache).toNumber()
)
2021-08-20 06:17:02 -07:00
)
) : (
0
)}
</div>
</div>
2021-08-21 05:11:14 -07:00
<div className={`flex justify-between pb-3`}>
2021-08-14 13:05:31 -07:00
<div className="font-normal text-th-fgd-3 leading-4">
Avg entry price
</div>
<div className={`text-th-fgd-1`}>
2021-08-20 06:17:02 -07:00
{isLoading ? (
2021-08-21 06:02:51 -07:00
<DataLoader />
2021-08-20 06:17:02 -07:00
) : perpAccount ? (
getAvgEntryPrice(
mangoAccount,
perpAccount,
selectedMarket,
perpTradeHistory
)
) : (
0
)}
2021-08-14 13:05:31 -07:00
</div>
</div>
2021-08-21 05:11:14 -07:00
<div className={`flex justify-between pb-3`}>
<div className="font-normal text-th-fgd-3 leading-4">
2021-08-14 13:05:31 -07:00
Break-even price
</div>
<div className={`text-th-fgd-1`}>
2021-08-20 06:17:02 -07:00
{isLoading ? (
2021-08-21 06:02:51 -07:00
<DataLoader />
2021-08-20 06:17:02 -07:00
) : perpAccount ? (
getBreakEvenPrice(
mangoAccount,
perpAccount,
selectedMarket,
perpTradeHistory
)
) : (
0
)}
</div>
</div>
2021-08-21 05:11:14 -07:00
<div className={`flex justify-between pb-3`}>
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">
Unsettled PnL
</Tooltip.Content>
</Tooltip>
2021-09-03 00:00:59 -07:00
<div
className={`flex items-center ${
perpPnl?.gt(ZERO_I80F48)
? 'text-th-green'
: perpPnl?.lt(ZERO_I80F48)
? 'text-th-red'
: 'text-th-fgd-1'
}`}
>
2021-08-20 06:17:02 -07:00
{isLoading ? (
2021-08-21 06:02:51 -07:00
<DataLoader />
2021-08-20 06:17:02 -07:00
) : perpAccount ? (
formatUsdValue(
2021-09-03 00:00:59 -07:00
+nativeI80F48ToUi(perpPnl, marketConfig.quoteDecimals)
2021-08-20 06:17:02 -07:00
)
) : (
'0'
)}
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"
2021-09-03 00:00:59 -07:00
disabled={perpAccount ? perpPnl.eq(ZERO_I80F48) : true}
2021-09-02 14:35:37 -07:00
>
Settle
</LinkButton>
)}
2021-08-16 06:31:25 -07:00
</div>
</div>
2021-08-24 20:15:57 -07:00
{perpAccount &&
Math.abs(selectedMarket.baseLotsToNumber(perpAccount.basePosition)) >
0 ? (
<Button
onClick={() => setShowMarketCloseModal(true)}
className="mt-3 w-full"
>
<span>Market Close</span>
</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}
</FloatingElement>
) : (
<>
2021-07-22 04:34:03 -07:00
<FloatingElement showConnect>
2021-08-14 11:16:15 -07:00
<div className={!connected ? 'filter blur' : null}>
<ElementTitle>Balances</ElementTitle>
2021-08-14 13:05:31 -07:00
{mangoGroup ? (
2021-07-22 05:25:18 -07:00
<div className="grid grid-cols-2 grid-rows-1 gap-4 pt-2">
2021-08-14 13:05:31 -07:00
{mangoGroupConfig.tokens
2021-07-22 04:34:03 -07:00
.filter((t) => t.symbol === baseSymbol || t.symbol === 'USDC')
.reverse()
.map(({ symbol, mintKey }) => {
2021-08-14 13:05:31 -07:00
const tokenIndex = mangoGroup.getTokenIndex(mintKey)
2021-08-24 03:09:42 -07:00
const deposit = mangoAccount
? mangoAccount.getUiDeposit(
mangoGroupCache.rootBankCache[tokenIndex],
mangoGroup,
tokenIndex
)
2021-08-24 03:09:42 -07:00
: null
const borrow = mangoAccount
? mangoAccount.getUiBorrow(
mangoGroupCache.rootBankCache[tokenIndex],
mangoGroup,
tokenIndex
)
2021-08-24 03:09:42 -07:00
: null
2021-07-22 04:34:03 -07:00
return (
<div
2021-08-16 06:31:25 -07:00
className="border border-th-bkg-4 p-4 rounded-md"
2021-07-22 04:34:03 -07:00
key={mintKey.toString()}
>
2021-08-16 06:31:25 -07:00
<div className="flex items-center justify-between mb-4">
2021-07-22 04:34:03 -07:00
<div className="flex items-center">
<img
alt=""
src={`/assets/icons/${symbol.toLowerCase()}.svg`}
className={`h-5 mr-2.5 w-auto`}
/>
<span className="text-th-fgd-2">{symbol}</span>
</div>
</div>
2021-08-20 18:42:59 -07:00
<div className="pb-3">
<div className="pb-0.5 text-th-fgd-3 text-xs">
Balance
2021-08-20 18:42:59 -07:00
</div>
2021-07-22 05:25:18 -07:00
<div className={`text-th-fgd-1`}>
2021-08-20 06:17:02 -07:00
{isLoading ? (
2021-08-21 06:02:51 -07:00
<DataLoader />
2021-08-20 06:17:02 -07:00
) : mangoAccount ? (
deposit.gt(borrow) ? (
deposit.toFixed()
) : (
2021-09-03 13:25:09 -07:00
`-${borrow.toFixed()}`
2021-08-20 06:17:02 -07:00
)
) : (
2021-08-20 18:42:59 -07:00
0
2021-08-20 06:17:02 -07:00
)}
</div>
2021-07-22 05:25:18 -07:00
</div>
2021-08-20 18:42:59 -07:00
<div className="pb-3">
<Tooltip content="Available to withdraw after accounting for collateral and open orders">
<div className="pb-0.5 text-th-fgd-3 text-xs">
Available Balance
</div>
</Tooltip>
2021-07-22 05:25:18 -07:00
<div className={`text-th-fgd-1`}>
2021-08-20 06:17:02 -07:00
{isLoading ? (
2021-08-21 06:02:51 -07:00
<DataLoader />
2021-08-20 06:17:02 -07:00
) : mangoAccount ? (
floorToDecimal(
nativeI80F48ToUi(
mangoAccount.getAvailableBalance(
mangoGroup,
mangoGroupCache,
tokenIndex
),
mangoGroup.tokens[tokenIndex].decimals
).toNumber(),
getTokenBySymbol(mangoGroupConfig, symbol)
.decimals
)
2021-08-20 06:17:02 -07:00
) : (
2021-08-20 18:42:59 -07:00
0
2021-08-20 06:17:02 -07:00
)}
2021-07-22 04:34:03 -07:00
</div>
</div>
2021-07-22 05:25:18 -07:00
<div>
2021-08-23 07:14:03 -07:00
<Tooltip content="Deposit APY / Borrow APR">
2021-07-22 05:25:18 -07:00
<div
2021-08-20 18:42:59 -07:00
className={`cursor-help font-normal pb-0.5 text-th-fgd-3 default-transition text-xs hover:border-th-bkg-2 hover:text-th-fgd-3`}
2021-07-22 05:25:18 -07:00
>
2021-09-06 04:51:18 -07:00
Deposit/Borrow Rates
</div>
2021-08-16 10:00:43 -07:00
<div className={`text-th-fgd-1`}>
<span className={`text-th-green`}>
{i80f48ToPercent(
mangoGroup.getDepositRate(tokenIndex)
).toFixed(2)}
%
</span>
<span className={`text-th-fgd-4`}>{' / '}</span>
<span className={`text-th-red`}>
{i80f48ToPercent(
mangoGroup.getBorrowRate(tokenIndex)
).toFixed(2)}
%
</span>
</div>
2021-07-22 05:25:18 -07:00
</Tooltip>
</div>
</div>
2021-07-22 04:34:03 -07:00
)
})}
</div>
) : null}
</div>
</FloatingElement>
</>
)
}
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" />
)