2023-06-16 05:23:50 -07:00
|
|
|
import {
|
|
|
|
useState,
|
|
|
|
useCallback,
|
|
|
|
useEffect,
|
|
|
|
useMemo,
|
|
|
|
Dispatch,
|
|
|
|
SetStateAction,
|
|
|
|
} from 'react'
|
|
|
|
import { ArrowDownIcon } from '@heroicons/react/20/solid'
|
|
|
|
import NumberFormat, {
|
|
|
|
NumberFormatValues,
|
|
|
|
SourceInfo,
|
|
|
|
} from 'react-number-format'
|
|
|
|
import Decimal from 'decimal.js'
|
|
|
|
import mangoStore from '@store/mangoStore'
|
|
|
|
import { useTranslation } from 'next-i18next'
|
2023-07-25 19:48:13 -07:00
|
|
|
import { SIZE_INPUT_UI_KEY } from '../../utils/constants'
|
2023-06-16 05:23:50 -07:00
|
|
|
import useLocalStorageState from 'hooks/useLocalStorageState'
|
|
|
|
import SwapSlider from './SwapSlider'
|
|
|
|
import PercentageSelectButtons from './PercentageSelectButtons'
|
|
|
|
import Select from '@components/forms/Select'
|
2023-07-25 19:48:13 -07:00
|
|
|
import { floorToDecimal } from 'utils/numbers'
|
|
|
|
import { withValueLimit } from './MarketSwapForm'
|
|
|
|
import SellTokenInput from './SellTokenInput'
|
|
|
|
import BuyTokenInput from './BuyTokenInput'
|
2023-07-30 23:05:26 -07:00
|
|
|
import { notify } from 'utils/notifications'
|
|
|
|
import * as sentry from '@sentry/nextjs'
|
|
|
|
import { isMangoError } from 'types'
|
|
|
|
import Button from '@components/shared/Button'
|
|
|
|
import { useWallet } from '@solana/wallet-adapter-react'
|
|
|
|
import Loading from '@components/shared/Loading'
|
2023-07-31 05:25:46 -07:00
|
|
|
import TokenLogo from '@components/shared/TokenLogo'
|
2023-07-31 17:57:53 -07:00
|
|
|
import InlineNotification from '@components/shared/InlineNotification'
|
2023-06-16 05:23:50 -07:00
|
|
|
|
|
|
|
type LimitSwapFormProps = {
|
|
|
|
setShowTokenSelect: Dispatch<SetStateAction<'input' | 'output' | undefined>>
|
|
|
|
}
|
|
|
|
|
2023-07-31 17:57:53 -07:00
|
|
|
type LimitSwapForm = {
|
|
|
|
limitPrice: string | undefined
|
|
|
|
triggerPrice: string
|
|
|
|
}
|
|
|
|
type FormErrors = Partial<Record<keyof LimitSwapForm, string>>
|
|
|
|
|
2023-07-31 05:25:46 -07:00
|
|
|
const ORDER_TYPES = [
|
|
|
|
// 'trade:limit',
|
|
|
|
'trade:stop-market',
|
|
|
|
'trade:stop-limit',
|
|
|
|
]
|
2023-06-16 05:23:50 -07:00
|
|
|
|
2023-07-25 19:48:13 -07:00
|
|
|
const set = mangoStore.getState().set
|
2023-06-16 05:23:50 -07:00
|
|
|
|
|
|
|
const LimitSwapForm = ({ setShowTokenSelect }: LimitSwapFormProps) => {
|
|
|
|
const { t } = useTranslation(['common', 'swap', 'trade'])
|
2023-07-30 23:05:26 -07:00
|
|
|
const { connected } = useWallet()
|
2023-06-16 05:23:50 -07:00
|
|
|
const [animateSwitchArrow, setAnimateSwitchArrow] = useState(0)
|
|
|
|
const [orderType, setOrderType] = useState(ORDER_TYPES[0])
|
|
|
|
const [triggerPrice, setTriggerPrice] = useState('')
|
2023-07-30 23:05:26 -07:00
|
|
|
const [submitting, setSubmitting] = useState(false)
|
2023-06-16 05:23:50 -07:00
|
|
|
const [swapFormSizeUi] = useLocalStorageState(SIZE_INPUT_UI_KEY, 'slider')
|
2023-07-31 17:57:53 -07:00
|
|
|
const [formErrors, setFormErrors] = useState<FormErrors>({})
|
2023-06-16 05:23:50 -07:00
|
|
|
|
|
|
|
const {
|
|
|
|
margin: useMargin,
|
|
|
|
inputBank,
|
|
|
|
outputBank,
|
|
|
|
amountIn: amountInFormValue,
|
|
|
|
amountOut: amountOutFormValue,
|
|
|
|
limitPrice,
|
|
|
|
} = mangoStore((s) => s.swap)
|
|
|
|
|
|
|
|
const amountInAsDecimal: Decimal | null = useMemo(() => {
|
2023-07-30 23:05:26 -07:00
|
|
|
return Number(amountInFormValue)
|
|
|
|
? new Decimal(amountInFormValue)
|
2023-06-16 05:23:50 -07:00
|
|
|
: new Decimal(0)
|
2023-07-30 23:05:26 -07:00
|
|
|
}, [amountInFormValue])
|
2023-06-16 05:23:50 -07:00
|
|
|
|
2023-07-30 23:05:26 -07:00
|
|
|
const [baseBank, quoteBank] = useMemo(() => {
|
|
|
|
if (inputBank && inputBank.name === 'USDC') {
|
|
|
|
return [outputBank, inputBank]
|
|
|
|
} else if (outputBank && outputBank.name === 'USDC') {
|
|
|
|
return [inputBank, outputBank]
|
|
|
|
} else if (inputBank && inputBank.name === 'SOL') {
|
|
|
|
return [outputBank, inputBank]
|
|
|
|
} else return [inputBank, outputBank]
|
|
|
|
}, [inputBank, outputBank])
|
2023-06-16 05:23:50 -07:00
|
|
|
|
|
|
|
const setAmountInFormValue = useCallback((amountIn: string) => {
|
|
|
|
set((s) => {
|
|
|
|
s.swap.amountIn = amountIn
|
|
|
|
if (!parseFloat(amountIn)) {
|
|
|
|
s.swap.amountOut = ''
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}, [])
|
|
|
|
|
|
|
|
const setAmountOutFormValue = useCallback((amountOut: string) => {
|
|
|
|
set((s) => {
|
|
|
|
s.swap.amountOut = amountOut
|
|
|
|
if (!parseFloat(amountOut)) {
|
|
|
|
s.swap.amountIn = ''
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}, [])
|
|
|
|
|
|
|
|
const setLimitPrice = useCallback((price: string) => {
|
|
|
|
set((s) => {
|
|
|
|
s.swap.limitPrice = price
|
|
|
|
})
|
|
|
|
}, [])
|
|
|
|
|
2023-07-31 17:57:53 -07:00
|
|
|
const initialQuotePrice = useMemo(() => {
|
|
|
|
if (!baseBank || !quoteBank) return
|
2023-07-31 22:13:12 -07:00
|
|
|
return floorToDecimal(
|
|
|
|
baseBank.uiPrice / quoteBank.uiPrice,
|
|
|
|
quoteBank.mintDecimals,
|
|
|
|
)
|
2023-07-31 17:57:53 -07:00
|
|
|
}, [baseBank, quoteBank])
|
|
|
|
|
2023-07-31 05:25:46 -07:00
|
|
|
// set default limit and trigger price
|
2023-07-30 23:05:26 -07:00
|
|
|
useEffect(() => {
|
2023-07-31 17:57:53 -07:00
|
|
|
if (!initialQuotePrice) return
|
2023-07-31 05:25:46 -07:00
|
|
|
if (!triggerPrice) {
|
2023-07-31 22:13:12 -07:00
|
|
|
setTriggerPrice(
|
|
|
|
initialQuotePrice.mul(0.9).toFixed(quoteBank?.mintDecimals),
|
|
|
|
)
|
2023-07-31 05:25:46 -07:00
|
|
|
}
|
2023-07-30 23:05:26 -07:00
|
|
|
if (!limitPrice) {
|
|
|
|
set((s) => {
|
2023-07-31 22:13:12 -07:00
|
|
|
s.swap.limitPrice = initialQuotePrice
|
|
|
|
.mul(0.8)
|
|
|
|
.toFixed(quoteBank?.mintDecimals)
|
2023-07-30 23:05:26 -07:00
|
|
|
})
|
|
|
|
}
|
2023-07-31 22:13:12 -07:00
|
|
|
}, [initialQuotePrice, limitPrice, quoteBank, triggerPrice])
|
2023-07-31 17:57:53 -07:00
|
|
|
|
|
|
|
const [limitPriceDifference, triggerPriceDifference] = useMemo(() => {
|
|
|
|
if (!initialQuotePrice) return [0, 0]
|
2023-07-31 22:13:12 -07:00
|
|
|
const initialPrice = initialQuotePrice.toNumber()
|
2023-07-31 17:57:53 -07:00
|
|
|
const limitDifference = limitPrice
|
2023-07-31 22:13:12 -07:00
|
|
|
? ((parseFloat(limitPrice) - initialPrice) / initialPrice) * 100
|
2023-07-31 17:57:53 -07:00
|
|
|
: 0
|
|
|
|
const triggerDifference = triggerPrice
|
2023-07-31 22:13:12 -07:00
|
|
|
? ((parseFloat(triggerPrice) - initialPrice) / initialPrice) * 100
|
2023-07-31 17:57:53 -07:00
|
|
|
: 0
|
|
|
|
return [limitDifference, triggerDifference]
|
|
|
|
}, [initialQuotePrice, limitPrice, triggerPrice])
|
|
|
|
|
2023-08-01 03:49:01 -07:00
|
|
|
// const isFormValid = useCallback(
|
|
|
|
// (form: LimitSwapForm) => {
|
|
|
|
// const invalidFields: FormErrors = {}
|
|
|
|
// setFormErrors({})
|
|
|
|
// const triggerPriceNumber = parseFloat(form.triggerPrice)
|
|
|
|
// const requiredFields: (keyof LimitSwapForm)[] = [
|
|
|
|
// 'limitPrice',
|
|
|
|
// 'triggerPrice',
|
|
|
|
// ]
|
|
|
|
// for (const key of requiredFields) {
|
|
|
|
// const value = form[key] as string
|
|
|
|
// if (!value) {
|
|
|
|
// if (orderType === 'trade:stop-market') {
|
|
|
|
// if (key !== 'limitPrice') {
|
|
|
|
// invalidFields[key] = t('settings:error-required-field')
|
|
|
|
// }
|
|
|
|
// } else {
|
|
|
|
// invalidFields[key] = t('settings:error-required-field')
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// if (
|
|
|
|
// orderType.includes('stop') &&
|
|
|
|
// initialQuotePrice &&
|
|
|
|
// triggerPriceNumber > initialQuotePrice.toNumber()
|
|
|
|
// ) {
|
|
|
|
// invalidFields.triggerPrice =
|
|
|
|
// 'Trigger price must be less than current price'
|
|
|
|
// }
|
|
|
|
// if (form.limitPrice && form.limitPrice > form.triggerPrice) {
|
|
|
|
// invalidFields.limitPrice = 'Limit price must be less than trigger price'
|
|
|
|
// }
|
|
|
|
// if (Object.keys(invalidFields).length) {
|
|
|
|
// setFormErrors(invalidFields)
|
|
|
|
// }
|
|
|
|
// return invalidFields
|
|
|
|
// },
|
|
|
|
// [initialQuotePrice, orderType],
|
|
|
|
// )
|
2023-07-30 23:05:26 -07:00
|
|
|
|
2023-06-16 05:23:50 -07:00
|
|
|
/*
|
|
|
|
If the use margin setting is toggled, clear the form values
|
|
|
|
*/
|
|
|
|
useEffect(() => {
|
|
|
|
setAmountInFormValue('')
|
|
|
|
setAmountOutFormValue('')
|
|
|
|
}, [useMargin, setAmountInFormValue, setAmountOutFormValue])
|
|
|
|
|
2023-08-01 03:49:01 -07:00
|
|
|
// get the out amount from the in amount and trigger or limit price
|
|
|
|
const getAmountOut = useCallback(
|
|
|
|
(amountIn: string, price: string) => {
|
|
|
|
const amountOut =
|
|
|
|
outputBank?.name === quoteBank?.name
|
|
|
|
? floorToDecimal(
|
|
|
|
parseFloat(amountIn) * parseFloat(price),
|
|
|
|
outputBank?.mintDecimals || 0,
|
|
|
|
)
|
|
|
|
: floorToDecimal(
|
|
|
|
parseFloat(amountIn) / parseFloat(price),
|
|
|
|
outputBank?.mintDecimals || 0,
|
|
|
|
)
|
|
|
|
return amountOut
|
|
|
|
},
|
|
|
|
[outputBank, quoteBank],
|
|
|
|
)
|
|
|
|
|
|
|
|
// get the in amount from the out amount and trigger or limit price
|
|
|
|
const getAmountIn = useCallback(
|
|
|
|
(amountOut: string, price: string) => {
|
|
|
|
const amountIn =
|
|
|
|
outputBank?.name === quoteBank?.name
|
|
|
|
? floorToDecimal(
|
|
|
|
parseFloat(amountOut) / parseFloat(price),
|
|
|
|
inputBank?.mintDecimals || 0,
|
|
|
|
)
|
|
|
|
: floorToDecimal(
|
|
|
|
parseFloat(amountOut) * parseFloat(price),
|
|
|
|
inputBank?.mintDecimals || 0,
|
|
|
|
)
|
|
|
|
return amountIn
|
|
|
|
},
|
|
|
|
[inputBank, outputBank, quoteBank],
|
|
|
|
)
|
|
|
|
|
2023-06-16 05:23:50 -07:00
|
|
|
const handleAmountInChange = useCallback(
|
|
|
|
(e: NumberFormatValues, info: SourceInfo) => {
|
|
|
|
if (info.source !== 'event') return
|
|
|
|
setAmountInFormValue(e.value)
|
2023-07-31 17:57:53 -07:00
|
|
|
const price =
|
|
|
|
orderType === 'trade:stop-market' ? triggerPrice : limitPrice
|
2023-08-01 03:49:01 -07:00
|
|
|
if (parseFloat(e.value) > 0 && price) {
|
|
|
|
const amountOut = getAmountOut(e.value, price)
|
|
|
|
setAmountOutFormValue(amountOut.toString())
|
2023-06-16 05:23:50 -07:00
|
|
|
}
|
|
|
|
},
|
2023-07-30 23:05:26 -07:00
|
|
|
[
|
|
|
|
limitPrice,
|
2023-07-31 17:57:53 -07:00
|
|
|
orderType,
|
2023-07-30 23:05:26 -07:00
|
|
|
setAmountInFormValue,
|
|
|
|
setAmountOutFormValue,
|
2023-07-31 17:57:53 -07:00
|
|
|
triggerPrice,
|
2023-07-30 23:05:26 -07:00
|
|
|
],
|
2023-06-16 05:23:50 -07:00
|
|
|
)
|
|
|
|
|
2023-06-17 05:22:05 -07:00
|
|
|
const handleAmountOutChange = useCallback(
|
|
|
|
(e: NumberFormatValues, info: SourceInfo) => {
|
|
|
|
if (info.source !== 'event') return
|
|
|
|
setAmountOutFormValue(e.value)
|
2023-07-31 17:57:53 -07:00
|
|
|
const price =
|
|
|
|
orderType === 'trade:stop-market' ? triggerPrice : limitPrice
|
2023-08-01 03:49:01 -07:00
|
|
|
if (parseFloat(e.value) > 0 && price) {
|
|
|
|
const amountIn = getAmountIn(e.value, price)
|
|
|
|
setAmountInFormValue(amountIn.toString())
|
2023-06-17 05:22:05 -07:00
|
|
|
}
|
|
|
|
},
|
2023-07-30 23:05:26 -07:00
|
|
|
[
|
2023-07-31 17:57:53 -07:00
|
|
|
orderType,
|
2023-07-30 23:05:26 -07:00
|
|
|
limitPrice,
|
|
|
|
setAmountInFormValue,
|
|
|
|
setAmountOutFormValue,
|
2023-07-31 17:57:53 -07:00
|
|
|
triggerPrice,
|
2023-07-30 23:05:26 -07:00
|
|
|
],
|
2023-06-17 05:22:05 -07:00
|
|
|
)
|
|
|
|
|
|
|
|
const handleAmountInUi = useCallback(
|
2023-06-16 05:23:50 -07:00
|
|
|
(amountIn: string) => {
|
|
|
|
setAmountInFormValue(amountIn)
|
2023-08-01 03:49:01 -07:00
|
|
|
const price =
|
|
|
|
orderType === 'trade:stop-market' ? triggerPrice : limitPrice
|
|
|
|
if (price) {
|
|
|
|
const amountOut = getAmountOut(amountIn, price)
|
|
|
|
setAmountOutFormValue(amountOut.toString())
|
2023-06-16 05:23:50 -07:00
|
|
|
}
|
|
|
|
},
|
2023-08-01 03:49:01 -07:00
|
|
|
[
|
|
|
|
limitPrice,
|
|
|
|
orderType,
|
|
|
|
setAmountInFormValue,
|
|
|
|
setAmountOutFormValue,
|
|
|
|
triggerPrice,
|
|
|
|
],
|
2023-06-16 05:23:50 -07:00
|
|
|
)
|
|
|
|
|
|
|
|
const handleLimitPrice = useCallback(
|
|
|
|
(e: NumberFormatValues, info: SourceInfo) => {
|
|
|
|
if (info.source !== 'event') return
|
2023-07-31 17:57:53 -07:00
|
|
|
setFormErrors({})
|
2023-06-16 05:23:50 -07:00
|
|
|
setLimitPrice(e.value)
|
2023-08-01 03:49:01 -07:00
|
|
|
if (parseFloat(e.value) > 0 && parseFloat(amountInFormValue) > 0) {
|
|
|
|
const amountOut = getAmountOut(amountInFormValue, e.value)
|
|
|
|
setAmountOutFormValue(amountOut.toString())
|
2023-06-17 05:22:05 -07:00
|
|
|
}
|
2023-06-16 05:23:50 -07:00
|
|
|
},
|
2023-08-01 03:49:01 -07:00
|
|
|
[amountInFormValue, setLimitPrice],
|
2023-06-16 05:23:50 -07:00
|
|
|
)
|
|
|
|
|
|
|
|
const handleTriggerPrice = useCallback(
|
|
|
|
(e: NumberFormatValues, info: SourceInfo) => {
|
|
|
|
if (info.source !== 'event') return
|
2023-07-31 17:57:53 -07:00
|
|
|
setFormErrors({})
|
2023-06-16 05:23:50 -07:00
|
|
|
setTriggerPrice(e.value)
|
2023-07-31 17:57:53 -07:00
|
|
|
if (
|
2023-08-01 03:49:01 -07:00
|
|
|
parseFloat(e.value) > 0 &&
|
|
|
|
parseFloat(amountInFormValue) > 0 &&
|
2023-07-31 17:57:53 -07:00
|
|
|
orderType === 'trade:stop-market'
|
|
|
|
) {
|
2023-08-01 03:49:01 -07:00
|
|
|
const amountOut = getAmountOut(amountInFormValue, e.value)
|
|
|
|
setAmountOutFormValue(amountOut.toString())
|
2023-07-31 17:57:53 -07:00
|
|
|
}
|
2023-06-16 05:23:50 -07:00
|
|
|
},
|
2023-08-01 03:49:01 -07:00
|
|
|
[amountInFormValue, orderType, setTriggerPrice],
|
2023-06-16 05:23:50 -07:00
|
|
|
)
|
|
|
|
|
2023-07-31 17:57:53 -07:00
|
|
|
const handleSwitchTokens = useCallback(() => {
|
|
|
|
const price = orderType === 'trade:stop-market' ? triggerPrice : limitPrice
|
|
|
|
if (amountInAsDecimal?.gt(0) && price) {
|
|
|
|
const amountOut =
|
|
|
|
outputBank?.name !== quoteBank?.name
|
|
|
|
? amountInAsDecimal.mul(price)
|
|
|
|
: amountInAsDecimal.div(price)
|
|
|
|
setAmountOutFormValue(amountOut.toString())
|
|
|
|
}
|
|
|
|
set((s) => {
|
|
|
|
s.swap.inputBank = outputBank
|
|
|
|
s.swap.outputBank = inputBank
|
|
|
|
// s.swap.limitPrice = ''
|
|
|
|
})
|
|
|
|
setAnimateSwitchArrow(
|
|
|
|
(prevanimateSwitchArrow) => prevanimateSwitchArrow + 1,
|
|
|
|
)
|
|
|
|
}, [
|
|
|
|
setAmountInFormValue,
|
|
|
|
amountInAsDecimal,
|
|
|
|
limitPrice,
|
|
|
|
inputBank,
|
|
|
|
orderType,
|
|
|
|
outputBank,
|
|
|
|
quoteBank,
|
|
|
|
triggerPrice,
|
|
|
|
])
|
|
|
|
|
|
|
|
const handlePlaceStopLoss = useCallback(async () => {
|
2023-08-01 03:49:01 -07:00
|
|
|
// const invalidFields = isFormValid({ limitPrice, triggerPrice })
|
|
|
|
// if (Object.keys(invalidFields).length) {
|
|
|
|
// return
|
|
|
|
// }
|
2023-07-30 23:05:26 -07:00
|
|
|
try {
|
|
|
|
const client = mangoStore.getState().client
|
|
|
|
const group = mangoStore.getState().group
|
|
|
|
const actions = mangoStore.getState().actions
|
|
|
|
const mangoAccount = mangoStore.getState().mangoAccount.current
|
|
|
|
const inputBank = mangoStore.getState().swap.inputBank
|
|
|
|
const outputBank = mangoStore.getState().swap.outputBank
|
|
|
|
|
|
|
|
if (
|
|
|
|
!mangoAccount ||
|
|
|
|
!group ||
|
|
|
|
!inputBank ||
|
|
|
|
!outputBank ||
|
|
|
|
(!triggerPrice && orderType !== 'trade:limit') ||
|
|
|
|
(!limitPrice && orderType !== 'trade:stop-market')
|
|
|
|
)
|
|
|
|
return
|
|
|
|
setSubmitting(true)
|
|
|
|
|
|
|
|
const orderPrice =
|
|
|
|
orderType === 'trade:limit'
|
|
|
|
? parseFloat(limitPrice!)
|
|
|
|
: parseFloat(triggerPrice)
|
|
|
|
|
|
|
|
const stopLimitPrice =
|
|
|
|
orderType !== 'trade:stop-market' ? parseFloat(limitPrice!) : 0
|
|
|
|
|
|
|
|
try {
|
|
|
|
const tx = await client.tokenConditionalSwapStopLoss(
|
|
|
|
group,
|
|
|
|
mangoAccount,
|
|
|
|
inputBank.mint,
|
|
|
|
orderPrice,
|
|
|
|
outputBank.mint,
|
|
|
|
stopLimitPrice,
|
|
|
|
amountInAsDecimal.toNumber(),
|
|
|
|
null,
|
|
|
|
null,
|
|
|
|
)
|
|
|
|
notify({
|
|
|
|
title: 'Transaction confirmed',
|
|
|
|
type: 'success',
|
|
|
|
txid: tx,
|
|
|
|
noSound: true,
|
|
|
|
})
|
|
|
|
actions.fetchGroup()
|
|
|
|
await actions.reloadMangoAccount()
|
|
|
|
} catch (e) {
|
|
|
|
console.error('onSwap error: ', e)
|
|
|
|
sentry.captureException(e)
|
|
|
|
if (isMangoError(e)) {
|
|
|
|
notify({
|
|
|
|
title: 'Transaction failed',
|
|
|
|
description: e.message,
|
|
|
|
txid: e?.txid,
|
|
|
|
type: 'error',
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
console.error('Swap error:', e)
|
|
|
|
} finally {
|
|
|
|
setSubmitting(false)
|
|
|
|
}
|
|
|
|
}, [orderType, limitPrice, triggerPrice, amountInAsDecimal])
|
|
|
|
|
|
|
|
const limitOrderDisabled =
|
|
|
|
!connected || !amountInFormValue || !amountOutFormValue
|
2023-06-16 05:23:50 -07:00
|
|
|
|
|
|
|
return (
|
|
|
|
<>
|
2023-07-25 19:48:13 -07:00
|
|
|
<SellTokenInput
|
|
|
|
className="rounded-b-none"
|
|
|
|
handleAmountInChange={handleAmountInChange}
|
|
|
|
setShowTokenSelect={setShowTokenSelect}
|
|
|
|
setAmountInFormValue={setAmountInFormValue}
|
|
|
|
/>
|
2023-06-16 05:23:50 -07:00
|
|
|
<div
|
|
|
|
className={`grid ${
|
|
|
|
orderType === 'trade:stop-limit' ? 'grid-cols-3' : 'grid-cols-2'
|
|
|
|
} gap-2 rounded-b-xl bg-th-bkg-2 p-3 pt-1`}
|
|
|
|
id="swap-step-two"
|
|
|
|
>
|
|
|
|
<div className="col-span-1">
|
|
|
|
<p className="mb-2 text-th-fgd-2">{t('trade:order-type')}</p>
|
|
|
|
<Select
|
|
|
|
value={t(orderType)}
|
|
|
|
onChange={(type) => setOrderType(type)}
|
|
|
|
className="w-full"
|
2023-07-31 05:25:46 -07:00
|
|
|
buttonClassName="ring-transparent rounded-t-lg rounded-b-lg focus:outline-none md:hover:bg-th-bkg-1 md:hover:ring-transparent focus-visible:bg-th-bkg-3 whitespace-nowrap"
|
2023-06-16 05:23:50 -07:00
|
|
|
>
|
|
|
|
{ORDER_TYPES.map((type) => (
|
|
|
|
<Select.Option key={type} value={type}>
|
|
|
|
{t(type)}
|
|
|
|
</Select.Option>
|
|
|
|
))}
|
|
|
|
</Select>
|
|
|
|
</div>
|
|
|
|
{orderType !== 'trade:limit' ? (
|
|
|
|
<div className="col-span-1">
|
2023-07-31 17:57:53 -07:00
|
|
|
<p className="mb-2 text-th-fgd-2">
|
|
|
|
{t('trade:trigger-price')}{' '}
|
|
|
|
<span className="text-xs text-th-fgd-3">
|
|
|
|
{triggerPriceDifference
|
|
|
|
? `(${triggerPriceDifference.toFixed(2)}%)`
|
|
|
|
: ''}
|
|
|
|
</span>
|
|
|
|
</p>
|
2023-07-31 05:25:46 -07:00
|
|
|
<div className="relative">
|
|
|
|
<NumberFormat
|
|
|
|
inputMode="decimal"
|
|
|
|
thousandSeparator=","
|
|
|
|
allowNegative={false}
|
|
|
|
isNumericString={true}
|
|
|
|
decimalScale={outputBank?.mintDecimals || 6}
|
|
|
|
name="triggerPrice"
|
|
|
|
id="triggerPrice"
|
|
|
|
className="h-10 w-full rounded-lg bg-th-input-bkg p-3 pl-8 font-mono text-sm text-th-fgd-1 focus:outline-none md:hover:bg-th-bkg-1"
|
|
|
|
placeholder="0.00"
|
|
|
|
value={triggerPrice}
|
|
|
|
onValueChange={handleTriggerPrice}
|
|
|
|
isAllowed={withValueLimit}
|
|
|
|
/>
|
|
|
|
<div className="absolute top-1/2 -translate-y-1/2 left-2">
|
|
|
|
<TokenLogo bank={quoteBank} size={16} />
|
|
|
|
</div>
|
|
|
|
</div>
|
2023-07-31 17:57:53 -07:00
|
|
|
{formErrors.triggerPrice ? (
|
|
|
|
<div className="mt-1">
|
|
|
|
<InlineNotification
|
|
|
|
type="error"
|
|
|
|
desc={formErrors.triggerPrice}
|
|
|
|
hideBorder
|
|
|
|
hidePadding
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
) : null}
|
2023-06-16 05:23:50 -07:00
|
|
|
</div>
|
|
|
|
) : null}
|
|
|
|
{orderType !== 'trade:stop-market' ? (
|
|
|
|
<div className="col-span-1">
|
2023-07-31 17:57:53 -07:00
|
|
|
<p className="mb-2 text-th-fgd-2">
|
|
|
|
{t('trade:limit-price')}{' '}
|
|
|
|
<span className="text-xs text-th-fgd-3">
|
|
|
|
{limitPriceDifference
|
|
|
|
? `(${limitPriceDifference.toFixed(2)}%)`
|
|
|
|
: ''}
|
|
|
|
</span>
|
|
|
|
</p>
|
2023-07-31 05:25:46 -07:00
|
|
|
<div className="relative">
|
|
|
|
<NumberFormat
|
|
|
|
inputMode="decimal"
|
|
|
|
thousandSeparator=","
|
|
|
|
allowNegative={false}
|
|
|
|
isNumericString={true}
|
|
|
|
decimalScale={outputBank?.mintDecimals || 6}
|
|
|
|
name="limitPrice"
|
|
|
|
id="limitPrice"
|
|
|
|
className="h-10 w-full rounded-lg bg-th-input-bkg p-3 pl-8 font-mono text-sm text-th-fgd-1 focus:outline-none md:hover:bg-th-bkg-1"
|
|
|
|
placeholder="0.00"
|
|
|
|
value={limitPrice}
|
|
|
|
onValueChange={handleLimitPrice}
|
|
|
|
isAllowed={withValueLimit}
|
|
|
|
/>
|
|
|
|
<div className="absolute top-1/2 -translate-y-1/2 left-2">
|
|
|
|
<TokenLogo bank={quoteBank} size={16} />
|
|
|
|
</div>
|
|
|
|
</div>
|
2023-07-31 17:57:53 -07:00
|
|
|
{formErrors.limitPrice ? (
|
|
|
|
<div className="mt-1">
|
|
|
|
<InlineNotification
|
|
|
|
type="error"
|
|
|
|
desc={formErrors.limitPrice}
|
|
|
|
hideBorder
|
|
|
|
hidePadding
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
) : null}
|
2023-06-16 05:23:50 -07:00
|
|
|
</div>
|
|
|
|
) : null}
|
|
|
|
</div>
|
|
|
|
<div className="my-2 flex justify-center">
|
|
|
|
<button
|
|
|
|
className="rounded-full border border-th-fgd-4 p-1.5 text-th-fgd-3 focus-visible:border-th-active md:hover:border-th-active md:hover:text-th-active"
|
|
|
|
onClick={handleSwitchTokens}
|
|
|
|
>
|
|
|
|
<ArrowDownIcon
|
|
|
|
className="h-5 w-5"
|
|
|
|
style={
|
|
|
|
animateSwitchArrow % 2 == 0
|
|
|
|
? { transform: 'rotate(0deg)' }
|
|
|
|
: { transform: 'rotate(360deg)' }
|
|
|
|
}
|
|
|
|
/>
|
|
|
|
</button>
|
|
|
|
</div>
|
2023-07-25 19:48:13 -07:00
|
|
|
<BuyTokenInput
|
|
|
|
handleAmountOutChange={handleAmountOutChange}
|
|
|
|
setShowTokenSelect={setShowTokenSelect}
|
|
|
|
setAmountOutFormValue={setAmountOutFormValue}
|
|
|
|
/>
|
2023-06-16 05:23:50 -07:00
|
|
|
{swapFormSizeUi === 'slider' ? (
|
|
|
|
<SwapSlider
|
|
|
|
useMargin={useMargin}
|
|
|
|
amount={amountInAsDecimal.toNumber()}
|
2023-06-17 05:22:05 -07:00
|
|
|
onChange={(v) => handleAmountInUi(v)}
|
2023-06-16 05:23:50 -07:00
|
|
|
step={1 / 10 ** (inputBank?.mintDecimals || 6)}
|
|
|
|
/>
|
|
|
|
) : (
|
|
|
|
<PercentageSelectButtons
|
|
|
|
amountIn={amountInAsDecimal.toString()}
|
2023-06-17 05:22:05 -07:00
|
|
|
setAmountIn={(v) => handleAmountInUi(v)}
|
2023-06-16 05:23:50 -07:00
|
|
|
useMargin={useMargin}
|
|
|
|
/>
|
|
|
|
)}
|
2023-07-30 23:05:26 -07:00
|
|
|
<Button
|
2023-07-31 17:57:53 -07:00
|
|
|
onClick={handlePlaceStopLoss}
|
2023-07-30 23:05:26 -07:00
|
|
|
className="mt-6 mb-4 flex w-full items-center justify-center text-base"
|
|
|
|
disabled={limitOrderDisabled}
|
|
|
|
size="large"
|
|
|
|
>
|
|
|
|
{submitting ? <Loading /> : t('swap:place-limit-order')}
|
|
|
|
</Button>
|
2023-06-16 05:23:50 -07:00
|
|
|
</>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
export default LimitSwapForm
|