mango-ui-v3/components/LeverageSlider.tsx

187 lines
5.4 KiB
TypeScript

import React, { useEffect, useMemo, useState } from 'react'
import Slider from 'rc-slider'
import 'rc-slider/assets/index.css'
import useMangoStore from '../stores/useMangoStore'
import {
getMarketIndexBySymbol,
getWeights,
I80F48,
PerpMarket,
} from '@blockworks-foundation/mango-client'
import tw from 'twin.macro'
import styled from '@emotion/styled'
import 'rc-slider/assets/index.css'
type StyledSliderProps = {
enableTransition?: boolean
disabled?: boolean
}
const StyledSlider = styled(Slider)<StyledSliderProps>`
.rc-slider-rail {
${tw`bg-th-primary h-2 rounded-full`}
opacity: 0.6;
}
.rc-slider-track {
${tw`bg-th-primary h-2 rounded-full ring-1 ring-th-primary ring-inset`}
${({ enableTransition }) =>
enableTransition && tw`transition-all duration-500`}
}
.rc-slider-step {
${tw`hidden`}
}
.rc-slider-handle {
${tw`border-4 border-th-primary h-4 w-4 ring-white light:ring-gray-400 hover:ring-4 hover:ring-opacity-50 active:ring-8 active:ring-opacity-50`}
background: #fff;
margin-top: -4px;
${({ enableTransition }) =>
enableTransition && tw`transition-all duration-500`}
${({ disabled }) => disabled && tw`bg-th-fgd-3 border-th-fgd-4`}
}
${({ disabled }) => disabled && 'background-color: transparent'}
`
type SliderProps = {
onChange: (x) => void
onAfterChange?: (x) => void
step: number
value: number
side: 'buy' | 'sell'
price: number
disabled?: boolean
max?: number
maxButtonTransition?: boolean
decimalCount: number
}
const percentToClose = (size, total) => {
return (size / total) * 100
}
export default function LeverageSlider({
onChange,
onAfterChange,
step,
value,
disabled,
maxButtonTransition,
side,
price,
decimalCount,
}: SliderProps) {
const [enableTransition, setEnableTransition] = useState(false)
const mangoAccount = useMangoStore((s) => s.selectedMangoAccount.current)
const mangoGroup = useMangoStore((s) => s.selectedMangoGroup.current)
const mangoGroupConfig = useMangoStore((s) => s.selectedMangoGroup.config)
const mangoCache = useMangoStore((s) => s.selectedMangoGroup.cache)
const marketConfig = useMangoStore((s) => s.selectedMarket.config)
const market = useMangoStore((s) => s.selectedMarket.current)
const marketIndex = getMarketIndexBySymbol(
mangoGroupConfig,
marketConfig.baseSymbol
)
const initLeverage = useMemo(() => {
if (!mangoGroup || !marketConfig) return 1
const ws = getWeights(mangoGroup, marketConfig.marketIndex, 'Init')
const w =
marketConfig.kind === 'perp' ? ws.perpAssetWeight : ws.spotAssetWeight
return Math.round((100 * -1) / (w.toNumber() - 1)) / 100
}, [mangoGroup, marketConfig])
const { max, deposits, borrows } = useMemo(() => {
if (!mangoAccount) return { max: 0 }
const priceOrDefault = price
? I80F48.fromNumber(price)
: mangoGroup.getPrice(marketIndex, mangoCache)
const {
max: maxQuote,
deposits,
borrows,
} = mangoAccount.getMaxLeverageForMarket(
mangoGroup,
mangoCache,
marketIndex,
market,
side,
priceOrDefault
)
if (maxQuote.toNumber() <= 0) return { max: 0 }
// multiply the maxQuote by a scaler value to account for
// srm fees or rounding issues in getMaxLeverageForMarket
const maxScaler = market instanceof PerpMarket ? 0.99 : 0.95
const scaledMax =
(maxQuote.toNumber() * maxScaler) / priceOrDefault.toNumber()
return { max: scaledMax, deposits, borrows }
}, [mangoAccount, mangoGroup, mangoCache, marketIndex, market, side, price])
useEffect(() => {
if (maxButtonTransition) {
setEnableTransition(true)
}
}, [maxButtonTransition])
useEffect(() => {
if (enableTransition) {
const transitionTimer = setTimeout(() => {
setEnableTransition(false)
}, 500)
return () => clearTimeout(transitionTimer)
}
}, [enableTransition])
// if (!mangoAccount) return null
const roundedDeposits = parseFloat(deposits?.toFixed(decimalCount))
const roundedBorrows = parseFloat(borrows?.toFixed(decimalCount))
const closeDepositString =
percentToClose(value, roundedDeposits) > 100
? '100% Close Position + Open Short'
: `${percentToClose(value, roundedDeposits).toFixed(1)}% Close Position`
const closeBorrowString =
percentToClose(value, roundedBorrows) > 100
? '100% Close Position + Open Long'
: `${percentToClose(value, roundedBorrows).toFixed(1)}% Close Position`
const setMaxLeverage = function () {
onChange(Math.round(max / step) * step)
}
return (
<>
<div className="flex mt-2 items-center pl-1 pr-1">
<StyledSlider
min={0}
max={max}
value={value || 0}
onChange={onChange}
onAfterChange={onAfterChange}
step={step}
disabled={disabled}
/>
<button
className="bg-th-bkg-4 hover:brightness-[1.15] font-normal rounded text-th-fgd-1 text-xs p-2 ml-2"
onClick={setMaxLeverage}
>
{initLeverage}x
</button>
</div>
{side === 'sell' ? (
<div className="text-th-fgd-4 text-xs tracking-normal mt-1">
<span>{roundedDeposits > 0 ? closeDepositString : null}</span>
</div>
) : (
<div className="text-th-fgd-4 text-xs tracking-normal mt-1">
<span>{roundedBorrows > 0 ? closeBorrowString : null}</span>
</div>
)}
</>
)
}