import { useCallback, useMemo, useState } from 'react' import FloatingElement from './FloatingElement' import { ElementTitle } from './styles' import useMangoStore, { mangoClient } from '../stores/useMangoStore' import { i80f48ToPercent, formatUsdValue, floorToDecimal } from '../utils/index' import Button, { LinkButton } from './Button' import Tooltip from './Tooltip' import SideBadge from './SideBadge' import { getMarketIndexBySymbol, getTokenBySymbol, nativeI80F48ToUi, PerpAccount, PerpMarket, QUOTE_INDEX, ZERO_BN, ZERO_I80F48, } from '@blockworks-foundation/mango-client' import useTradeHistory from '../hooks/useTradeHistory' import { getAvgEntryPrice, getBreakEvenPrice } from './PerpPositionsTable' import { notify } from '../utils/notifications' import MarketCloseModal from './MarketCloseModal' import Loading from './Loading' const settlePnl = async (perpMarket: PerpMarket, perpAccount: PerpAccount) => { const mangoAccount = useMangoStore.getState().selectedMangoAccount.current const mangoGroup = useMangoStore.getState().selectedMangoGroup.current const mangoCache = useMangoStore.getState().selectedMangoGroup.cache const wallet = useMangoStore.getState().wallet.current const actions = useMangoStore.getState().actions const marketIndex = mangoGroup.getPerpMarketIndex(perpMarket.publicKey) try { const txid = await mangoClient.settlePnl( mangoGroup, mangoCache, mangoAccount, perpMarket, mangoGroup.rootBankAccounts[QUOTE_INDEX], mangoCache.priceCache[marketIndex].price, wallet ) actions.reloadMangoAccount() 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', }) } } export function SettlePnlTooltip() { return (
Settling will update your USDC balance to reflect the unsettled PnL amount.{' '} Learn more
) } export default function MarketPosition() { 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) const connected = useMangoStore((s) => s.wallet.connected) const isLoading = useMangoStore((s) => s.selectedMangoAccount.initialLoad) const setMangoStore = useMangoStore((s) => s.set) const baseSymbol = marketConfig.baseSymbol const marketName = marketConfig.name const tradeHistory = useTradeHistory() const perpTradeHistory = tradeHistory?.filter( (t) => t.marketName === marketName ) const [showMarketCloseModal, setShowMarketCloseModal] = useState(false) const [settling, setSettling] = useState(false) const marketIndex = useMemo(() => { return getMarketIndexBySymbol(mangoGroupConfig, baseSymbol) }, [mangoGroupConfig, baseSymbol]) let perpAccount, perpPnl if (marketName.includes('PERP') && mangoAccount) { perpAccount = mangoAccount.perpAccounts[marketIndex] perpPnl = perpAccount.getPnl( mangoGroup.perpMarkets[marketIndex], mangoGroupCache.perpMarketCache[marketIndex], mangoGroupCache.priceCache[marketIndex].price ) } const handleSizeClick = (size) => { setMangoStore((state) => { state.tradeForm.baseSize = size }) } const handleCloseWarning = useCallback(() => { setShowMarketCloseModal(false) }, []) const handleSettlePnl = (perpMarket, perpAccount) => { setSettling(true) settlePnl(perpMarket, perpAccount).then(() => { setSettling(false) }) } if (!mangoGroup || !selectedMarket) return null return selectedMarket instanceof PerpMarket ? (
Position: {marketConfig.name}
Side
{isLoading ? ( ) : perpAccount && !perpAccount.basePosition.eq(ZERO_BN) ? ( ) : ( '--' )}
Position size
{isLoading ? ( ) : perpAccount && Math.abs( selectedMarket.baseLotsToNumber(perpAccount.basePosition) ) > 0 ? ( handleSizeClick( Math.abs( selectedMarket.baseLotsToNumber(perpAccount.basePosition) ) ) } > {`${Math.abs( selectedMarket.baseLotsToNumber(perpAccount.basePosition) )} ${baseSymbol}`} ) : ( `0 ${baseSymbol}` )}
Notional size
{isLoading ? ( ) : perpAccount ? ( formatUsdValue( Math.abs( selectedMarket.baseLotsToNumber(perpAccount.basePosition) * mangoGroup.getPrice(marketIndex, mangoGroupCache).toNumber() ) ) ) : ( 0 )}
Avg entry price
{isLoading ? ( ) : perpAccount ? ( getAvgEntryPrice( mangoAccount, perpAccount, selectedMarket, perpTradeHistory ) ) : ( 0 )}
Break-even price
{isLoading ? ( ) : perpAccount ? ( getBreakEvenPrice( mangoAccount, perpAccount, selectedMarket, perpTradeHistory ) ) : ( 0 )}
}> Unsettled PnL
{isLoading ? ( ) : perpAccount ? ( formatUsdValue( +nativeI80F48ToUi(perpPnl, marketConfig.quoteDecimals) ) ) : ( '0' )} {settling ? ( ) : ( handleSettlePnl(selectedMarket, perpAccount)} className="ml-2 text-th-primary text-xs disabled:cursor-not-allowed disabled:opacity-60 disabled:hover:underline" disabled={perpAccount ? perpPnl.eq(ZERO_I80F48) : true} > Settle )}
{perpAccount && Math.abs(selectedMarket.baseLotsToNumber(perpAccount.basePosition)) > 0 ? ( ) : null}
{showMarketCloseModal ? ( ) : null}
) : ( <>
Balances {mangoGroup ? (
{mangoGroupConfig.tokens .filter((t) => t.symbol === baseSymbol || t.symbol === 'USDC') .reverse() .map(({ symbol, mintKey }) => { const tokenIndex = mangoGroup.getTokenIndex(mintKey) const deposit = mangoAccount ? mangoAccount.getUiDeposit( mangoGroupCache.rootBankCache[tokenIndex], mangoGroup, tokenIndex ) : null const borrow = mangoAccount ? mangoAccount.getUiBorrow( mangoGroupCache.rootBankCache[tokenIndex], mangoGroup, tokenIndex ) : null return (
{symbol}
Balance
{isLoading ? ( ) : mangoAccount ? ( deposit.gt(borrow) ? ( deposit.toFixed() ) : ( `-${borrow.toFixed()}` ) ) : ( 0 )}
Available Balance
{isLoading ? ( ) : mangoAccount ? ( floorToDecimal( nativeI80F48ToUi( mangoAccount.getAvailableBalance( mangoGroup, mangoGroupCache, tokenIndex ), mangoGroup.tokens[tokenIndex].decimals ).toNumber(), getTokenBySymbol(mangoGroupConfig, symbol) .decimals ) ) : ( 0 )}
Deposit/Borrow Rates
{i80f48ToPercent( mangoGroup.getDepositRate(tokenIndex) ).toFixed(2)} % {' / '} {i80f48ToPercent( mangoGroup.getBorrowRate(tokenIndex) ).toFixed(2)} %
) })}
) : null}
) } export const DataLoader = () => (
)