mango-ui-v3/components/LeverageSlider.tsx

168 lines
4.8 KiB
TypeScript
Raw Normal View History

2021-08-18 11:23:12 -07:00
import React, { useEffect, useMemo, useState } from 'react'
import Slider from 'rc-slider'
import 'rc-slider/assets/index.css'
import useMangoStore from '../stores/useMangoStore'
2021-08-18 13:15:17 -07:00
import {
getMarketIndexBySymbol,
I80F48,
PerpMarket,
} from '@blockworks-foundation/mango-client'
2021-08-18 11:23:12 -07:00
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 {
2021-08-18 13:29:39 -07:00
${tw`bg-th-primary h-2 rounded-full`}
opacity: 0.6;
2021-08-18 11:23:12 -07:00
}
.rc-slider-track {
2021-08-18 13:29:39 -07:00
${tw`bg-th-primary h-2 rounded-full ring-1 ring-th-primary ring-inset`}
2021-08-18 11:23:12 -07:00
${({ 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;
2021-08-18 13:29:39 -07:00
margin-top: -4px;
2021-08-18 11:23:12 -07:00
${({ 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: string
2021-08-18 13:15:17 -07:00
price: number
2021-08-18 11:23:12 -07:00
disabled?: boolean
max?: number
maxButtonTransition?: boolean
decimalCount: number
2021-08-18 11:23:12 -07:00
}
const percentToClose = (size, total) => {
return (size / total) * 100
}
2021-08-18 11:23:12 -07:00
export default function LeverageSlider({
onChange,
onAfterChange,
step,
value,
disabled,
maxButtonTransition,
side,
2021-08-18 13:15:17 -07:00
price,
decimalCount,
2021-08-18 11:23:12 -07:00
}: 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 { max, deposits, borrows } = useMemo(() => {
2021-08-18 13:15:17 -07:00
if (!mangoAccount) return 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 0
2021-08-18 13:29:39 -07:00
// multiply the maxQuote by a scaler value to account for
// srm fees or rounding issues in getMaxLeverageForMarket
2021-08-18 13:15:17 -07:00
const maxScaler = market instanceof PerpMarket ? 0.99 : 0.95
const scaledMax =
(maxQuote.toNumber() * maxScaler) /
2021-08-18 13:15:17 -07:00
mangoGroup.getPrice(marketIndex, mangoCache).toNumber()
return { max: scaledMax, deposits, borrows }
2021-08-18 13:29:39 -07:00
}, [mangoAccount, mangoGroup, mangoCache, marketIndex, market, side, price])
2021-08-18 11:23:12 -07:00
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 + Leverage'
: `${percentToClose(value, roundedDeposits).toFixed(2)}% Close Position`
const closeBorrowString =
percentToClose(value, roundedBorrows) > 100
? '100% Close Position + Leverage'
: `${percentToClose(value, roundedDeposits).toFixed(2)}% Close Position`
2021-08-18 11:23:12 -07:00
return (
<>
<div className="relative mt-4 pl-1 pr-0">
<StyledSlider
min={0}
max={max}
value={value || 0}
onChange={onChange}
onAfterChange={onAfterChange}
step={step}
disabled={disabled}
/>
</div>
{side === 'sell' ? (
<div className="text-th-fgd-4 text-xs tracking-normal mt-2.5">
<span>{roundedDeposits > 0 ? closeDepositString : null}</span>
</div>
) : (
<div className="text-th-fgd-4 text-xs tracking-normal mt-2.5">
<span>{roundedBorrows > 0 ? closeBorrowString : null}</span>
</div>
)}
</>
2021-08-18 11:23:12 -07:00
)
}