import { HealthType, MangoAccount, PerpMarket, Serum3Market, } from '@blockworks-foundation/mango-v4' import FormatNumericValue from '@components/shared/FormatNumericValue' import HealthImpact from '@components/shared/HealthImpact' import Tooltip from '@components/shared/Tooltip' import mangoStore from '@store/mangoStore' import Decimal from 'decimal.js' import useMangoGroup from 'hooks/useMangoGroup' import useSelectedMarket from 'hooks/useSelectedMarket' import { useTranslation } from 'next-i18next' import { useMemo } from 'react' import Slippage from './Slippage' import { floorToDecimal, formatNumericValue, getDecimalCount, } from 'utils/numbers' import { formatTokenSymbol } from 'utils/tokens' import useOpenPerpPositions from 'hooks/useOpenPerpPositions' import { calculateEstPriceForBaseSize } from 'utils/tradeForm' const TradeSummary = ({ mangoAccount, useMargin, }: { mangoAccount: MangoAccount | undefined useMargin: boolean }) => { const { t } = useTranslation(['common', 'trade']) const { group } = useMangoGroup() const tradeForm = mangoStore((s) => s.tradeForm) const orderbook = mangoStore((s) => s.selectedMarket.orderbook) const { selectedMarket, quoteBank } = useSelectedMarket() const openPerpPositions = useOpenPerpPositions() // calc new avg price if an open position exists const avgEntryPrice = useMemo(() => { if ( !openPerpPositions?.length || !selectedMarket || !orderbook || selectedMarket instanceof Serum3Market ) return const openPosition = openPerpPositions.find( (pos) => pos.marketIndex === selectedMarket.perpMarketIndex, ) const { baseSize, price, reduceOnly, side, tradeType } = tradeForm if (!openPosition || !price || !tradeForm.baseSize) return let orderPrice = parseFloat(price) if (tradeType === 'Market') { orderPrice = calculateEstPriceForBaseSize( orderbook, parseFloat(tradeForm.baseSize), tradeForm.side, ) } const currentSize = openPosition.getBasePositionUi(selectedMarket) const tradeSize = side === 'buy' ? parseFloat(baseSize) : parseFloat(baseSize) * -1 const newTotalSize = currentSize + tradeSize const currentAvgPrice = openPosition.getAverageEntryPriceUi(selectedMarket) // don't calc when closing position if (newTotalSize === 0) { return } // don't calc when reducing position if ( (currentSize < 0 !== tradeSize < 0 && newTotalSize < 0 === currentSize < 0) || reduceOnly ) { return } // if trade side changes with new trade return new trade price if (currentSize < 0 !== newTotalSize < 0) { return price } const newTotalCost = currentAvgPrice * currentSize + orderPrice * tradeSize const newAvgEntryPrice = newTotalCost / newTotalSize return newAvgEntryPrice }, [openPerpPositions, selectedMarket, tradeForm, orderbook]) const maintProjectedHealth = useMemo(() => { if (!mangoAccount || !group) return 100 let simulatedHealthRatio = 0 try { if (selectedMarket instanceof Serum3Market) { simulatedHealthRatio = tradeForm.side === 'sell' ? mangoAccount.simHealthRatioWithSerum3AskUiChanges( group, Number(tradeForm.baseSize), selectedMarket.serumMarketExternal, HealthType.maint, ) : mangoAccount.simHealthRatioWithSerum3BidUiChanges( group, Number(tradeForm.quoteSize), selectedMarket.serumMarketExternal, HealthType.maint, ) } else if (selectedMarket instanceof PerpMarket) { simulatedHealthRatio = tradeForm.side === 'sell' ? mangoAccount.simHealthRatioWithPerpAskUiChanges( group, selectedMarket.perpMarketIndex, parseFloat(tradeForm.baseSize) || 0, HealthType.maint, ) : mangoAccount.simHealthRatioWithPerpBidUiChanges( group, selectedMarket.perpMarketIndex, parseFloat(tradeForm.baseSize) || 0, HealthType.maint, ) } } catch (e) { console.warn('Error calculating projected health: ', e) } return simulatedHealthRatio > 100 ? 100 : simulatedHealthRatio < 0 ? 0 : Math.trunc(simulatedHealthRatio) }, [group, mangoAccount, selectedMarket, tradeForm]) const balanceBank = useMemo(() => { if ( !group || !selectedMarket || selectedMarket instanceof PerpMarket || !useMargin ) return if (tradeForm.side === 'buy') { return group.getFirstBankByTokenIndex(selectedMarket.quoteTokenIndex) } else { return group.getFirstBankByTokenIndex(selectedMarket.baseTokenIndex) } }, [group, selectedMarket, tradeForm.side]) const [balance, borrowAmount] = useMemo(() => { if (!balanceBank || !mangoAccount) return [0, 0] let borrowAmount const balance = mangoAccount.getTokenDepositsUi(balanceBank) if (tradeForm.side === 'buy') { const remainingBalance = balance - parseFloat(tradeForm.quoteSize) borrowAmount = remainingBalance < 0 ? Math.abs(remainingBalance) : 0 } else { const remainingBalance = balance - parseFloat(tradeForm.baseSize) borrowAmount = remainingBalance < 0 ? Math.abs(remainingBalance) : 0 } return [balance, borrowAmount] }, [balanceBank, mangoAccount, tradeForm]) const orderValue = useMemo(() => { if ( !quoteBank || !tradeForm.price || !tradeForm.baseSize || isNaN(parseFloat(tradeForm.price)) || isNaN(parseFloat(tradeForm.baseSize)) ) return 0 const basePriceDecimal = new Decimal(tradeForm.price) const quotePriceDecimal = new Decimal(quoteBank.uiPrice) const sizeDecimal = new Decimal(tradeForm.baseSize) return floorToDecimal( basePriceDecimal.mul(quotePriceDecimal).mul(sizeDecimal), 2, ) }, [quoteBank, tradeForm]) return (
{t('trade:order-value')}
{orderValue ?
{t('borrow-amount')}
{t('loan-origination-fee')}
{t('free-collateral')}
{group && mangoAccount ? (
{t('trade:avg-entry-price')}
{t('common:route')}
Openbook