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

132 lines
4.2 KiB
TypeScript
Raw Normal View History

2023-01-15 19:58:01 -08:00
import { PerpMarket, Serum3Market } from '@blockworks-foundation/mango-v4'
import MaxAmountButton from '@components/shared/MaxAmountButton'
import { FadeInFadeOut } from '@components/shared/Transitions'
import { useWallet } from '@solana/wallet-adapter-react'
import mangoStore from '@store/mangoStore'
import useMangoAccount from 'hooks/useMangoAccount'
import useSelectedMarket from 'hooks/useSelectedMarket'
import useUnownedAccount from 'hooks/useUnownedAccount'
import { useTranslation } from 'next-i18next'
import { useCallback, useMemo } from 'react'
2023-04-01 03:05:53 -07:00
import { floorToDecimal } from 'utils/numbers'
2023-01-15 19:58:01 -08:00
import { useSpotMarketMax } from './SpotSlider'
const MaxSizeButton = ({
minOrderDecimals,
tickDecimals,
2023-01-15 19:58:01 -08:00
useMargin,
large,
}: {
minOrderDecimals: number
tickDecimals: number
2023-01-15 19:58:01 -08:00
useMargin: boolean
large?: boolean
}) => {
const { t } = useTranslation(['common', 'trade'])
const { mangoAccount } = useMangoAccount()
const { selectedMarket, price: oraclePrice } = useSelectedMarket()
2023-01-04 17:09:43 -08:00
const { price, side, tradeType } = mangoStore((s) => s.tradeForm)
2023-01-15 19:58:01 -08:00
const spotMax = useSpotMarketMax(
mangoAccount,
selectedMarket,
side,
2023-07-21 11:47:53 -07:00
useMargin,
2023-01-15 19:58:01 -08:00
)
const { isUnownedAccount } = useUnownedAccount()
const { connected } = useWallet()
2023-01-15 19:58:01 -08:00
const perpMax = useMemo(() => {
const group = mangoStore.getState().group
if (!mangoAccount || !group || !selectedMarket) return 0
2023-01-15 19:58:01 -08:00
if (selectedMarket instanceof PerpMarket) {
try {
2023-01-04 17:09:43 -08:00
if (side === 'buy') {
return mangoAccount.getMaxQuoteForPerpBidUi(
group,
2023-07-21 11:47:53 -07:00
selectedMarket.perpMarketIndex,
)
} else {
return mangoAccount.getMaxBaseForPerpAskUi(
group,
2023-07-21 11:47:53 -07:00
selectedMarket.perpMarketIndex,
)
}
2023-01-15 19:58:01 -08:00
} catch (e) {
console.error('Error calculating max leverage: spot btn group: ', e)
return 0
}
}
2023-01-04 17:09:43 -08:00
}, [mangoAccount, side, selectedMarket])
const handleMax = useCallback(() => {
2023-01-15 19:58:01 -08:00
const max = selectedMarket instanceof Serum3Market ? spotMax : perpMax || 0
const set = mangoStore.getState().set
set((state) => {
2023-01-04 17:09:43 -08:00
if (side === 'buy') {
if (tradeType === 'Market' || !price) {
2023-04-25 17:23:14 -07:00
const baseSize = floorToDecimal(max / oraclePrice, minOrderDecimals)
const quoteSize = floorToDecimal(max, tickDecimals)
state.tradeForm.baseSize = baseSize.toFixed()
state.tradeForm.quoteSize = quoteSize.toFixed()
} else {
2023-04-25 17:23:14 -07:00
const baseSize = floorToDecimal(
2023-01-15 19:58:01 -08:00
max / parseFloat(price),
2023-07-21 11:47:53 -07:00
minOrderDecimals,
2023-04-25 17:23:14 -07:00
)
const quoteSize = floorToDecimal(baseSize.mul(price), tickDecimals)
state.tradeForm.baseSize = baseSize.toFixed()
state.tradeForm.quoteSize = quoteSize.toFixed()
}
} else {
2023-04-25 17:23:14 -07:00
const baseSize = floorToDecimal(max, minOrderDecimals)
2023-01-04 17:09:43 -08:00
if (tradeType === 'Market' || !price) {
2023-04-25 17:23:14 -07:00
const quoteSize = floorToDecimal(
baseSize.mul(oraclePrice),
2023-07-21 11:47:53 -07:00
tickDecimals,
2023-04-25 17:23:14 -07:00
)
state.tradeForm.baseSize = baseSize.toFixed()
state.tradeForm.quoteSize = quoteSize.toFixed()
} else {
2023-04-25 17:23:14 -07:00
const quoteSize = floorToDecimal(baseSize.mul(price), tickDecimals)
state.tradeForm.baseSize = baseSize.toFixed()
state.tradeForm.quoteSize = quoteSize.toFixed()
}
}
})
2023-01-19 20:45:44 -08:00
}, [
minOrderDecimals,
perpMax,
price,
selectedMarket,
side,
spotMax,
tickDecimals,
tradeType,
])
const maxAmount = useMemo(() => {
2023-01-15 19:58:01 -08:00
const max = selectedMarket instanceof Serum3Market ? spotMax : perpMax || 0
2023-01-04 17:09:43 -08:00
const tradePrice = tradeType === 'Market' ? oraclePrice : Number(price)
if (side === 'buy') {
const buyMax = max / tradePrice
return isNaN(buyMax) ? 0 : buyMax
} else {
2023-01-23 19:04:05 -08:00
return max
}
2023-01-23 19:04:05 -08:00
}, [perpMax, spotMax, selectedMarket, price, side, tradeType])
return (
2023-11-04 05:24:17 -07:00
<FadeInFadeOut show={!!price && !isUnownedAccount && connected}>
<MaxAmountButton
className={large ? 'text-sm' : 'text-xs'}
decimals={minOrderDecimals}
label={t('max')}
onClick={handleMax}
value={maxAmount}
/>
</FadeInFadeOut>
)
}
export default MaxSizeButton