2021-12-19 12:21:18 -08:00
|
|
|
import {
|
|
|
|
useEffect,
|
|
|
|
useMemo,
|
|
|
|
useState,
|
|
|
|
FunctionComponent,
|
|
|
|
useCallback,
|
|
|
|
} from 'react'
|
2021-12-09 09:23:19 -08:00
|
|
|
import { useJupiter, RouteInfo } from '@jup-ag/react-hook'
|
2021-12-18 13:04:19 -08:00
|
|
|
import { TOKEN_LIST_URL } from '@jup-ag/core'
|
2021-12-09 09:23:19 -08:00
|
|
|
import { PublicKey } from '@solana/web3.js'
|
|
|
|
import useMangoStore from '../stores/useMangoStore'
|
|
|
|
import {
|
|
|
|
connectionSelector,
|
|
|
|
walletConnectedSelector,
|
|
|
|
walletSelector,
|
|
|
|
} from '../stores/selectors'
|
|
|
|
import { sortBy, sum } from 'lodash'
|
|
|
|
import {
|
2022-01-01 04:34:46 -08:00
|
|
|
CogIcon,
|
2021-12-27 01:46:50 -08:00
|
|
|
ExclamationCircleIcon,
|
2021-12-27 03:25:30 -08:00
|
|
|
ExternalLinkIcon,
|
2021-12-28 03:46:41 -08:00
|
|
|
InformationCircleIcon,
|
2021-12-09 09:23:19 -08:00
|
|
|
SwitchVerticalIcon,
|
|
|
|
} from '@heroicons/react/outline'
|
2022-01-01 04:34:46 -08:00
|
|
|
import { ChevronDownIcon } from '@heroicons/react/solid'
|
2021-12-29 03:43:37 -08:00
|
|
|
import { abbreviateAddress } from '../utils'
|
2021-12-09 09:23:19 -08:00
|
|
|
import SwapTokenSelect from './SwapTokenSelect'
|
|
|
|
import { notify } from '../utils/notifications'
|
2021-12-18 13:04:19 -08:00
|
|
|
import { Token } from '../@types/types'
|
2021-12-19 12:21:18 -08:00
|
|
|
import {
|
|
|
|
getTokenAccountsByOwnerWithWrappedSol,
|
|
|
|
nativeToUi,
|
|
|
|
zeroKey,
|
|
|
|
} from '@blockworks-foundation/mango-client'
|
2022-01-01 04:34:46 -08:00
|
|
|
import Button, { IconButton, LinkButton } from './Button'
|
2021-12-27 03:25:30 -08:00
|
|
|
import { useViewport } from '../hooks/useViewport'
|
|
|
|
import { breakpoints } from './TradePageGrid'
|
|
|
|
import useLocalStorageState from '../hooks/useLocalStorageState'
|
|
|
|
import Modal from './Modal'
|
|
|
|
import { ElementTitle } from './styles'
|
2021-12-29 14:12:43 -08:00
|
|
|
import { RefreshClockwiseIcon, WalletIcon } from './icons'
|
2021-12-28 03:46:41 -08:00
|
|
|
import Tooltip from './Tooltip'
|
2021-12-29 14:12:43 -08:00
|
|
|
import SwapSettingsModal from './SwapSettingsModal'
|
2021-12-30 03:44:14 -08:00
|
|
|
import SwapTokenInfo from './SwapTokenInfo'
|
2022-01-02 04:20:42 -08:00
|
|
|
import { numberFormatter } from './SwapTokenInfo'
|
2022-01-02 16:52:03 -08:00
|
|
|
import { useTranslation } from 'next-i18next'
|
2021-12-09 09:23:19 -08:00
|
|
|
|
|
|
|
type UseJupiterProps = Parameters<typeof useJupiter>[0]
|
|
|
|
|
|
|
|
const JupiterForm: FunctionComponent = () => {
|
2022-01-02 16:52:03 -08:00
|
|
|
const { t } = useTranslation(['common', 'swap'])
|
2021-12-09 09:23:19 -08:00
|
|
|
const wallet = useMangoStore(walletSelector)
|
|
|
|
const connection = useMangoStore(connectionSelector)
|
|
|
|
const connected = useMangoStore(walletConnectedSelector)
|
|
|
|
|
2021-12-29 14:12:43 -08:00
|
|
|
const [showSettings, setShowSettings] = useState(false)
|
2021-12-09 09:23:19 -08:00
|
|
|
const [depositAndFee, setDepositAndFee] = useState(null)
|
|
|
|
const [selectedRoute, setSelectedRoute] = useState<RouteInfo>(null)
|
|
|
|
const [showInputTokenSelect, setShowInputTokenSelect] = useState(false)
|
|
|
|
const [showOutputTokenSelect, setShowOutputTokenSelect] = useState(false)
|
2021-12-19 12:21:18 -08:00
|
|
|
const [swapping, setSwapping] = useState(false)
|
2021-12-18 13:04:19 -08:00
|
|
|
const [tokens, setTokens] = useState<Token[]>([])
|
2022-01-01 04:34:46 -08:00
|
|
|
const [outputTokenPrice, setOutputTokenPrice] = useState(null)
|
2021-12-19 10:17:43 -08:00
|
|
|
const [coinGeckoList, setCoinGeckoList] = useState(null)
|
2021-12-19 12:21:18 -08:00
|
|
|
const [walletTokens, setWalletTokens] = useState([])
|
2021-12-29 14:12:43 -08:00
|
|
|
const [slippage, setSlippage] = useState(0.5)
|
2021-12-09 09:23:19 -08:00
|
|
|
const [formValue, setFormValue] = useState<UseJupiterProps>({
|
|
|
|
amount: null,
|
|
|
|
inputMint: new PublicKey('EPjFWdd5AufqSSqeM2qN1xzybapC8G4wEGGkZwyTDt1v'),
|
|
|
|
outputMint: new PublicKey('MangoCzJ36AjZyKwVj3VnYU4GTonjfVEnJmvvWaxLac'),
|
2021-12-29 14:12:43 -08:00
|
|
|
slippage,
|
2021-12-09 09:23:19 -08:00
|
|
|
})
|
2021-12-27 03:25:30 -08:00
|
|
|
const [hasSwapped, setHasSwapped] = useLocalStorageState('hasSwapped', false)
|
2022-01-02 04:20:42 -08:00
|
|
|
const [showWalletDraw, setShowWalletDraw] = useState(false)
|
2021-12-27 03:25:30 -08:00
|
|
|
const [walletTokenPrices, setWalletTokenPrices] = useState(null)
|
|
|
|
const { width } = useViewport()
|
|
|
|
const isMobile = width ? width < breakpoints.sm : false
|
2021-12-28 03:46:41 -08:00
|
|
|
const [feeValue, setFeeValue] = useState(null)
|
2021-12-29 03:43:37 -08:00
|
|
|
const [showRoutesModal, setShowRoutesModal] = useState(false)
|
2021-12-09 09:23:19 -08:00
|
|
|
|
2021-12-19 12:21:18 -08:00
|
|
|
const fetchWalletTokens = useCallback(async () => {
|
|
|
|
const ownedTokens = []
|
|
|
|
const ownedTokenAccounts = await getTokenAccountsByOwnerWithWrappedSol(
|
|
|
|
connection,
|
|
|
|
wallet.publicKey
|
|
|
|
)
|
|
|
|
|
|
|
|
ownedTokenAccounts.forEach((account) => {
|
|
|
|
const decimals = tokens.find(
|
|
|
|
(t) => t?.address === account.mint.toString()
|
|
|
|
)?.decimals
|
2021-12-19 22:23:57 -08:00
|
|
|
|
2021-12-19 12:21:18 -08:00
|
|
|
const uiBalance = nativeToUi(account.amount, decimals || 6)
|
|
|
|
ownedTokens.push({ account, uiBalance })
|
|
|
|
})
|
|
|
|
console.log('ownedToknes', ownedTokens)
|
|
|
|
setWalletTokens(ownedTokens)
|
2021-12-19 22:23:57 -08:00
|
|
|
}, [wallet, connection, tokens])
|
2021-12-19 12:21:18 -08:00
|
|
|
|
2021-12-09 09:23:19 -08:00
|
|
|
// @ts-ignore
|
|
|
|
const [inputTokenInfo, outputTokenInfo] = useMemo(() => {
|
|
|
|
return [
|
2021-12-18 13:04:19 -08:00
|
|
|
tokens.find(
|
2021-12-18 21:53:20 -08:00
|
|
|
(item) => item?.address === formValue.inputMint?.toBase58() || ''
|
2021-12-18 13:04:19 -08:00
|
|
|
),
|
|
|
|
tokens.find(
|
2021-12-18 21:53:20 -08:00
|
|
|
(item) => item?.address === formValue.outputMint?.toBase58() || ''
|
2021-12-18 13:04:19 -08:00
|
|
|
),
|
2021-12-09 09:23:19 -08:00
|
|
|
]
|
|
|
|
}, [
|
|
|
|
formValue.inputMint?.toBase58(),
|
|
|
|
formValue.outputMint?.toBase58(),
|
2021-12-18 13:04:19 -08:00
|
|
|
tokens,
|
2021-12-09 09:23:19 -08:00
|
|
|
])
|
|
|
|
|
2022-01-02 04:20:42 -08:00
|
|
|
useEffect(() => {
|
|
|
|
if (width >= 1680) {
|
|
|
|
setShowWalletDraw(true)
|
|
|
|
}
|
|
|
|
}, [])
|
|
|
|
|
2021-12-19 10:17:43 -08:00
|
|
|
useEffect(() => {
|
|
|
|
const fetchCoinGeckoList = async () => {
|
|
|
|
const response = await fetch(
|
|
|
|
'https://api.coingecko.com/api/v3/coins/list'
|
|
|
|
)
|
|
|
|
const data = await response.json()
|
|
|
|
setCoinGeckoList(data)
|
|
|
|
}
|
|
|
|
|
|
|
|
fetchCoinGeckoList()
|
|
|
|
}, [])
|
|
|
|
|
2021-12-19 12:21:18 -08:00
|
|
|
useEffect(() => {
|
|
|
|
if (connected) {
|
|
|
|
fetchWalletTokens()
|
|
|
|
}
|
|
|
|
}, [connected])
|
|
|
|
|
2021-12-19 10:17:43 -08:00
|
|
|
useEffect(() => {
|
|
|
|
if (!coinGeckoList?.length) return
|
|
|
|
|
2022-01-01 04:34:46 -08:00
|
|
|
const fetchOutputTokenPrice = async () => {
|
2021-12-19 10:17:43 -08:00
|
|
|
const id = coinGeckoList.find(
|
|
|
|
(x) =>
|
|
|
|
x?.symbol?.toLowerCase() === outputTokenInfo?.symbol?.toLowerCase()
|
|
|
|
)?.id
|
2021-12-27 01:46:50 -08:00
|
|
|
if (id) {
|
|
|
|
const results = await fetch(
|
2022-01-01 04:34:46 -08:00
|
|
|
`https://api.coingecko.com/api/v3/simple/price?ids=${id}&vs_currencies=usd`
|
2021-12-27 01:46:50 -08:00
|
|
|
)
|
|
|
|
const json = await results.json()
|
2022-01-01 04:34:46 -08:00
|
|
|
if (json[id].usd) {
|
|
|
|
setOutputTokenPrice(json[id].usd)
|
|
|
|
}
|
2021-12-27 01:46:50 -08:00
|
|
|
}
|
2021-12-19 10:17:43 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (outputTokenInfo) {
|
2022-01-01 04:34:46 -08:00
|
|
|
fetchOutputTokenPrice()
|
2021-12-19 10:17:43 -08:00
|
|
|
}
|
2022-01-01 04:34:46 -08:00
|
|
|
}, [outputTokenInfo, coinGeckoList])
|
2021-12-19 10:17:43 -08:00
|
|
|
|
2021-12-09 09:23:19 -08:00
|
|
|
const amountInDecimal = useMemo(() => {
|
|
|
|
return formValue.amount * 10 ** (inputTokenInfo?.decimals || 1)
|
|
|
|
}, [inputTokenInfo, formValue.amount])
|
|
|
|
|
2021-12-29 14:12:43 -08:00
|
|
|
const { routeMap, allTokenMints, routes, loading, exchange, error, refresh } =
|
2021-12-09 09:23:19 -08:00
|
|
|
useJupiter({
|
|
|
|
...formValue,
|
|
|
|
amount: amountInDecimal,
|
|
|
|
slippage: formValue.slippage,
|
|
|
|
})
|
|
|
|
|
|
|
|
useEffect(() => {
|
2021-12-18 13:04:19 -08:00
|
|
|
// Fetch token list from Jupiter API
|
|
|
|
fetch(TOKEN_LIST_URL['mainnet-beta'])
|
|
|
|
.then((response) => response.json())
|
|
|
|
.then((result) => {
|
|
|
|
const tokens = allTokenMints.map((mint) =>
|
2021-12-18 21:53:20 -08:00
|
|
|
result.find((item) => item?.address === mint)
|
2021-12-18 13:04:19 -08:00
|
|
|
)
|
|
|
|
setTokens(tokens)
|
|
|
|
})
|
|
|
|
}, [allTokenMints])
|
2021-12-09 09:23:19 -08:00
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
if (routes) {
|
|
|
|
setSelectedRoute(routes[0])
|
|
|
|
}
|
|
|
|
}, [routes])
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
const getDepositAndFee = async () => {
|
|
|
|
const fees = await selectedRoute.getDepositAndFee()
|
|
|
|
setDepositAndFee(fees)
|
|
|
|
}
|
2021-12-18 21:53:20 -08:00
|
|
|
if (selectedRoute && connected) {
|
2021-12-09 09:23:19 -08:00
|
|
|
getDepositAndFee()
|
|
|
|
}
|
|
|
|
}, [selectedRoute])
|
|
|
|
|
2021-12-18 21:53:20 -08:00
|
|
|
const outputTokenMints = useMemo(() => {
|
|
|
|
if (routeMap.size && formValue.inputMint) {
|
|
|
|
const routeOptions = routeMap.get(formValue.inputMint.toString())
|
|
|
|
|
|
|
|
const routeOptionTokens = routeOptions.map((address) => {
|
|
|
|
return tokens.find((t) => {
|
|
|
|
return t?.address === address
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
return routeOptionTokens
|
|
|
|
} else {
|
|
|
|
return sortedTokenMints
|
|
|
|
}
|
|
|
|
}, [routeMap, tokens, formValue.inputMint])
|
|
|
|
|
2021-12-19 12:21:18 -08:00
|
|
|
const inputWalletBalance = () => {
|
2021-12-09 09:23:19 -08:00
|
|
|
if (walletTokens.length) {
|
2021-12-19 12:21:18 -08:00
|
|
|
const walletToken = walletTokens.filter((t) => {
|
2021-12-18 21:53:20 -08:00
|
|
|
return t.account.mint.toString() === inputTokenInfo?.address
|
2021-12-09 09:23:19 -08:00
|
|
|
})
|
2021-12-19 12:21:18 -08:00
|
|
|
const largestTokenAccount = sortBy(walletToken, 'uiBalance').reverse()[0]
|
2021-12-19 12:23:35 -08:00
|
|
|
return largestTokenAccount?.uiBalance || 0.0
|
2021-12-09 09:23:19 -08:00
|
|
|
}
|
2021-12-19 12:21:18 -08:00
|
|
|
|
2021-12-09 09:23:19 -08:00
|
|
|
return 0.0
|
2021-12-19 12:21:18 -08:00
|
|
|
}
|
2021-12-18 21:53:20 -08:00
|
|
|
|
2021-12-19 12:21:18 -08:00
|
|
|
const outputWalletBalance = () => {
|
2021-12-18 21:53:20 -08:00
|
|
|
if (walletTokens.length) {
|
2021-12-19 12:21:18 -08:00
|
|
|
const walletToken = walletTokens.filter((t) => {
|
2021-12-18 21:53:20 -08:00
|
|
|
return t.account.mint.toString() === outputTokenInfo?.address
|
|
|
|
})
|
2021-12-19 12:21:18 -08:00
|
|
|
const largestTokenAccount = sortBy(walletToken, 'uiBalance').reverse()[0]
|
2021-12-19 12:23:35 -08:00
|
|
|
return largestTokenAccount?.uiBalance || 0.0
|
2021-12-18 21:53:20 -08:00
|
|
|
}
|
|
|
|
return 0.0
|
2021-12-19 12:21:18 -08:00
|
|
|
}
|
2021-12-09 09:23:19 -08:00
|
|
|
|
2021-12-27 03:25:30 -08:00
|
|
|
const [walletTokensWithInfos] = useMemo(() => {
|
|
|
|
const userTokens = []
|
|
|
|
tokens.map((item) => {
|
|
|
|
const found = walletTokens.find(
|
|
|
|
(token) => token.account.mint.toBase58() === item?.address
|
|
|
|
)
|
|
|
|
if (found) {
|
|
|
|
userTokens.push({ ...found, item })
|
|
|
|
}
|
|
|
|
})
|
|
|
|
return [userTokens]
|
|
|
|
}, [walletTokens, tokens])
|
|
|
|
|
|
|
|
const getWalletTokenPrices = async () => {
|
|
|
|
const ids = walletTokensWithInfos.map(
|
|
|
|
(token) => token.item.extensions.coingeckoId
|
|
|
|
)
|
|
|
|
const response = await fetch(
|
|
|
|
`https://api.coingecko.com/api/v3/simple/price?ids=${ids.toString()}&vs_currencies=usd`
|
|
|
|
)
|
|
|
|
const data = await response.json()
|
|
|
|
setWalletTokenPrices(data)
|
|
|
|
}
|
|
|
|
|
2021-12-28 03:46:41 -08:00
|
|
|
const getSwapFeeTokenValue = async () => {
|
|
|
|
const mints = selectedRoute.marketInfos.map((info) => info.lpFee.mint)
|
|
|
|
const response = await fetch(
|
|
|
|
`https://api.coingecko.com/api/v3/simple/token_price/solana?contract_addresses=${mints.toString()}&vs_currencies=usd`
|
|
|
|
)
|
|
|
|
const data = await response.json()
|
|
|
|
|
|
|
|
const feeValue = selectedRoute.marketInfos.reduce((a, c) => {
|
|
|
|
const feeToken = tokens.find((item) => item?.address === c.lpFee?.mint)
|
|
|
|
const amount = c.lpFee?.amount / Math.pow(10, feeToken?.decimals)
|
|
|
|
if (data[c.lpFee?.mint]) {
|
|
|
|
return a + data[c.lpFee?.mint].usd * amount
|
|
|
|
}
|
|
|
|
if (c.lpFee?.mint === 'EPjFWdd5AufqSSqeM2qN1xzybapC8G4wEGGkZwyTDt1v') {
|
|
|
|
return a + 1 * amount
|
|
|
|
}
|
|
|
|
}, 0)
|
|
|
|
setFeeValue(feeValue)
|
|
|
|
}
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
if (selectedRoute) {
|
|
|
|
getSwapFeeTokenValue()
|
|
|
|
}
|
|
|
|
}, [selectedRoute])
|
|
|
|
|
2021-12-27 03:25:30 -08:00
|
|
|
useEffect(() => {
|
|
|
|
getWalletTokenPrices()
|
|
|
|
}, [walletTokensWithInfos])
|
|
|
|
|
2021-12-09 09:23:19 -08:00
|
|
|
const handleSelectRoute = (route) => {
|
2021-12-29 03:43:37 -08:00
|
|
|
setShowRoutesModal(false)
|
2021-12-09 09:23:19 -08:00
|
|
|
setSelectedRoute(route)
|
|
|
|
}
|
|
|
|
|
|
|
|
const handleSwitchMints = () => {
|
|
|
|
setFormValue((val) => ({
|
|
|
|
...val,
|
|
|
|
inputMint: formValue.outputMint,
|
|
|
|
outputMint: formValue.inputMint,
|
|
|
|
}))
|
|
|
|
}
|
|
|
|
|
2021-12-18 13:04:19 -08:00
|
|
|
const sortedTokenMints = sortBy(tokens, (token) => {
|
|
|
|
return token?.symbol?.toLowerCase()
|
2021-12-09 09:23:19 -08:00
|
|
|
})
|
|
|
|
|
|
|
|
const outAmountUi = selectedRoute
|
|
|
|
? selectedRoute.outAmount / 10 ** (outputTokenInfo?.decimals || 1)
|
|
|
|
: null
|
|
|
|
|
2021-12-22 16:39:57 -08:00
|
|
|
const swapDisabled = loading || !selectedRoute || routes?.length === 0
|
|
|
|
|
2022-01-02 16:52:03 -08:00
|
|
|
const inputTokenInfos = inputTokenInfo ? (inputTokenInfo as any) : null
|
|
|
|
const outputTokenInfos = outputTokenInfo ? (outputTokenInfo as any) : null
|
|
|
|
|
2021-12-09 09:23:19 -08:00
|
|
|
return (
|
2022-01-01 04:34:46 -08:00
|
|
|
<div className="grid grid-cols-12 lg:space-x-4">
|
|
|
|
<div className="col-span-12 lg:col-span-10 lg:col-start-2 ">
|
|
|
|
<div className="flex flex-col md:flex-row md:space-x-6">
|
|
|
|
<div className="w-full md:w-1/2 xl:w-1/3">
|
|
|
|
<div className="relative z-10">
|
|
|
|
{connected &&
|
|
|
|
walletTokensWithInfos.length &&
|
|
|
|
walletTokenPrices &&
|
|
|
|
!isMobile ? (
|
|
|
|
<div
|
2022-01-02 04:20:42 -08:00
|
|
|
className={`flex transform top-22 right-0 w-80 fixed overflow-hidden ease-in-out transition-all duration-700 z-30 ${
|
|
|
|
showWalletDraw ? 'translate-x-0' : 'mr-16 translate-x-full'
|
2022-01-01 04:34:46 -08:00
|
|
|
}`}
|
|
|
|
>
|
2022-01-02 04:20:42 -08:00
|
|
|
<aside
|
|
|
|
className={`bg-th-bkg-3 ml-16 pb-4 pt-6 rounded-l-md w-64`}
|
|
|
|
>
|
|
|
|
<div className="max-h-[480px] overflow-auto thin-scroll">
|
2022-01-01 04:34:46 -08:00
|
|
|
<div className="flex items-center justify-between pb-2 px-4">
|
|
|
|
<div className="font-bold text-base text-th-fgd-1">
|
2022-01-02 16:52:03 -08:00
|
|
|
{t('wallet')}
|
2022-01-01 04:34:46 -08:00
|
|
|
</div>
|
|
|
|
<a
|
|
|
|
className="flex items-center text-th-fgd-4 text-xs hover:text-th-fgd-3"
|
|
|
|
href={`https://explorer.solana.com/address/${wallet?.publicKey}`}
|
|
|
|
target="_blank"
|
|
|
|
rel="noopener noreferrer"
|
2021-12-30 03:44:14 -08:00
|
|
|
>
|
2022-01-01 04:34:46 -08:00
|
|
|
<div className="bg-th-green h-1.5 mr-1.5 rounded-full w-1.5" />
|
|
|
|
{abbreviateAddress(wallet.publicKey)}
|
|
|
|
<ExternalLinkIcon className="h-3.5 ml-0.5 -mt-0.5 w-3.5" />
|
|
|
|
</a>
|
|
|
|
</div>
|
|
|
|
{walletTokensWithInfos
|
|
|
|
.sort((a, b) => {
|
|
|
|
const aId = a.item.extensions.coingeckoId
|
|
|
|
const bId = b.item.extensions.coingeckoId
|
|
|
|
return (
|
|
|
|
b.uiBalance * walletTokenPrices[bId]?.usd -
|
|
|
|
a.uiBalance * walletTokenPrices[aId]?.usd
|
|
|
|
)
|
|
|
|
})
|
|
|
|
.map((token) => {
|
|
|
|
const geckoId = token.item.extensions.coingeckoId
|
|
|
|
return (
|
|
|
|
<div
|
|
|
|
className="cursor-pointer default-transition flex items-center justify-between px-4 py-2 hover:bg-th-bkg-4"
|
|
|
|
key={geckoId}
|
|
|
|
onClick={() =>
|
|
|
|
setFormValue((val) => ({
|
|
|
|
...val,
|
|
|
|
inputMint: new PublicKey(token?.item.address),
|
|
|
|
}))
|
|
|
|
}
|
|
|
|
>
|
|
|
|
<div className="flex items-center">
|
|
|
|
{token.item.logoURI ? (
|
|
|
|
<img
|
|
|
|
src={token.item.logoURI}
|
|
|
|
width="24"
|
|
|
|
height="24"
|
|
|
|
alt={token.item.symbol}
|
|
|
|
/>
|
|
|
|
) : null}
|
|
|
|
<div>
|
|
|
|
<div className="ml-2 text-th-fgd-1">
|
|
|
|
{token.item.symbol}
|
|
|
|
</div>
|
|
|
|
{walletTokenPrices ? (
|
|
|
|
<div className="ml-2 text-th-fgd-4 text-xs">
|
|
|
|
{walletTokenPrices[geckoId]
|
|
|
|
? `$${walletTokenPrices[geckoId].usd}`
|
2022-01-02 16:52:03 -08:00
|
|
|
: t('swap:unavailable')}
|
2022-01-01 04:34:46 -08:00
|
|
|
</div>
|
|
|
|
) : null}
|
|
|
|
</div>
|
2021-12-30 03:44:14 -08:00
|
|
|
</div>
|
2022-01-01 04:34:46 -08:00
|
|
|
<div>
|
|
|
|
<div className="text-right text-th-fgd-1">
|
|
|
|
{token.uiBalance.toLocaleString(undefined, {
|
|
|
|
maximumSignificantDigits: 6,
|
|
|
|
})}
|
|
|
|
</div>
|
|
|
|
<div className="text-th-fgd-4 text-right text-xs">
|
2021-12-30 03:44:14 -08:00
|
|
|
{walletTokenPrices[geckoId]
|
2022-01-01 04:34:46 -08:00
|
|
|
? `$${(
|
|
|
|
token.uiBalance *
|
|
|
|
walletTokenPrices[geckoId].usd
|
|
|
|
).toLocaleString(undefined, {
|
|
|
|
maximumFractionDigits: 2,
|
|
|
|
})}`
|
|
|
|
: '?'}
|
2021-12-30 03:44:14 -08:00
|
|
|
</div>
|
2022-01-01 04:34:46 -08:00
|
|
|
</div>
|
2021-12-30 03:44:14 -08:00
|
|
|
</div>
|
2022-01-01 04:34:46 -08:00
|
|
|
)
|
|
|
|
})}
|
|
|
|
</div>
|
|
|
|
</aside>
|
|
|
|
<button
|
2022-01-02 04:20:42 -08:00
|
|
|
className="absolute bg-th-bkg-4 p-3 right-64 rounded-r-none text-th-fgd-1 hover:text-th-primary top-20"
|
2022-01-01 04:34:46 -08:00
|
|
|
onClick={() => setShowWalletDraw(!showWalletDraw)}
|
|
|
|
>
|
|
|
|
<WalletIcon className="h-5 w-5" />
|
|
|
|
</button>
|
2021-12-30 03:44:14 -08:00
|
|
|
</div>
|
2022-01-01 04:34:46 -08:00
|
|
|
) : null}
|
|
|
|
<div className="bg-th-bkg-2 rounded-lg p-6">
|
|
|
|
<div className="flex justify-between">
|
|
|
|
<label
|
|
|
|
htmlFor="inputMint"
|
|
|
|
className="block text-sm font-semibold"
|
|
|
|
>
|
2022-01-02 16:52:03 -08:00
|
|
|
{t('swap:pay')}
|
2022-01-01 04:34:46 -08:00
|
|
|
</label>
|
|
|
|
<div className="space-x-3">
|
|
|
|
<label htmlFor="amount" className="text-th-fgd-3 text-xs">
|
2022-01-02 16:52:03 -08:00
|
|
|
{t('swap:bal')} {inputWalletBalance()}
|
2022-01-01 04:34:46 -08:00
|
|
|
</label>
|
|
|
|
{connected ? (
|
|
|
|
<>
|
|
|
|
<LinkButton
|
|
|
|
className="text-th-primary text-xs"
|
|
|
|
onClick={() => {
|
|
|
|
setFormValue((val) => ({
|
|
|
|
...val,
|
|
|
|
amount: inputWalletBalance(),
|
|
|
|
}))
|
|
|
|
}}
|
|
|
|
>
|
2022-01-02 16:52:03 -08:00
|
|
|
{t('max')}
|
2022-01-01 04:34:46 -08:00
|
|
|
</LinkButton>
|
|
|
|
</>
|
|
|
|
) : null}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<div className="grid grid-cols-2 mt-2">
|
|
|
|
<div className="col-span-1">
|
|
|
|
<button
|
|
|
|
className="hover:bg-th-bkg-3 -ml-2 p-2"
|
|
|
|
onClick={() => setShowInputTokenSelect(true)}
|
|
|
|
>
|
|
|
|
<div className="flex h-8 items-center">
|
|
|
|
{inputTokenInfo?.logoURI ? (
|
|
|
|
<img
|
|
|
|
src={inputTokenInfo?.logoURI}
|
|
|
|
width="24"
|
|
|
|
height="24"
|
|
|
|
alt={inputTokenInfo?.symbol}
|
|
|
|
/>
|
|
|
|
) : null}
|
|
|
|
<div className="text-base xl:text-lg ml-2">
|
|
|
|
{inputTokenInfo?.symbol}
|
|
|
|
</div>
|
|
|
|
<ChevronDownIcon className="flex-shrink-0 h-5 w-5 ml-1 text-th-fgd-3" />
|
|
|
|
</div>
|
|
|
|
</button>
|
|
|
|
</div>
|
|
|
|
<div className="col-span-1">
|
|
|
|
<input
|
|
|
|
name="amount"
|
|
|
|
id="amount"
|
2022-01-02 04:20:42 -08:00
|
|
|
className="bg-th-bkg-1 border border-th-fgd-4 default-transition font-bold pr-4 h-12 focus:outline-none rounded-md text-base text-right tracking-wide w-full hover:border-th-primary focus:border-th-primary"
|
2022-01-01 04:34:46 -08:00
|
|
|
value={formValue.amount || ''}
|
|
|
|
placeholder="0.00"
|
|
|
|
type="number"
|
|
|
|
pattern="[0-9]*"
|
|
|
|
onInput={(e: any) => {
|
|
|
|
let newValue = e.target?.value || 0
|
|
|
|
newValue = Number.isNaN(newValue) ? 0 : newValue
|
|
|
|
|
2021-12-30 03:44:14 -08:00
|
|
|
setFormValue((val) => ({
|
|
|
|
...val,
|
2022-01-01 04:34:46 -08:00
|
|
|
amount: newValue,
|
2021-12-30 03:44:14 -08:00
|
|
|
}))
|
|
|
|
}}
|
2022-01-01 04:34:46 -08:00
|
|
|
/>
|
2021-12-30 03:44:14 -08:00
|
|
|
</div>
|
2022-01-01 04:34:46 -08:00
|
|
|
</div>
|
2021-12-09 09:23:19 -08:00
|
|
|
|
2022-01-01 04:34:46 -08:00
|
|
|
<div className="flex justify-center my-4">
|
|
|
|
<button onClick={handleSwitchMints}>
|
|
|
|
<SwitchVerticalIcon className="default-transition h-8 w-8 rounded-full p-1.5 bg-th-bkg-4 text-th-fgd-1 hover:text-th-primary" />
|
|
|
|
</button>
|
|
|
|
</div>
|
2021-12-30 03:44:14 -08:00
|
|
|
|
2022-01-01 04:34:46 -08:00
|
|
|
<div className="flex items-center justify-between">
|
|
|
|
<label htmlFor="outputMint" className="font-semibold">
|
2022-01-02 16:52:03 -08:00
|
|
|
{t('swap:receive')}
|
2022-01-01 04:34:46 -08:00
|
|
|
</label>
|
|
|
|
<span className="text-th-fgd-3 text-xs">
|
2022-01-02 16:52:03 -08:00
|
|
|
{t('swap:bal')} {outputWalletBalance()}
|
2022-01-01 04:34:46 -08:00
|
|
|
</span>
|
|
|
|
</div>
|
|
|
|
<div className="grid grid-cols-2 mt-2">
|
|
|
|
<div className="col-span-1">
|
|
|
|
<button
|
|
|
|
className="flex h-12 items-center hover:bg-th-bkg-3 -ml-2 p-2"
|
|
|
|
onClick={() => setShowOutputTokenSelect(true)}
|
|
|
|
>
|
|
|
|
{outputTokenInfo?.logoURI ? (
|
|
|
|
<img
|
|
|
|
src={outputTokenInfo?.logoURI}
|
|
|
|
width="24"
|
|
|
|
height="24"
|
|
|
|
alt={outputTokenInfo?.symbol}
|
|
|
|
/>
|
|
|
|
) : null}
|
|
|
|
<div className="text-base xl:text-lg ml-2">
|
|
|
|
{outputTokenInfo?.symbol}
|
|
|
|
</div>
|
|
|
|
<ChevronDownIcon className="flex-shrink-0 h-5 w-5 ml-1 text-th-fgd-3" />
|
|
|
|
</button>
|
2021-12-29 03:43:37 -08:00
|
|
|
</div>
|
2022-01-01 04:34:46 -08:00
|
|
|
<div className="col-span-1 relative">
|
|
|
|
<input
|
|
|
|
name="amount"
|
|
|
|
id="amount"
|
|
|
|
className="bg-th-bkg-3 border border-th-bkg-4 cursor-not-allowed font-bold pr-4 h-12 focus:outline-none rounded-md text-lg text-right tracking-wide w-full"
|
|
|
|
disabled
|
2022-01-02 04:20:42 -08:00
|
|
|
placeholder="0.00"
|
2022-01-01 04:34:46 -08:00
|
|
|
value={
|
2022-01-02 04:20:42 -08:00
|
|
|
selectedRoute?.outAmount
|
|
|
|
? numberFormatter.format(
|
|
|
|
selectedRoute?.outAmount /
|
|
|
|
10 ** (outputTokenInfo?.decimals || 1)
|
|
|
|
)
|
|
|
|
: ''
|
2022-01-01 04:34:46 -08:00
|
|
|
}
|
|
|
|
/>
|
|
|
|
{selectedRoute?.outAmount && outputTokenPrice ? (
|
|
|
|
<div className="absolute mt-1 right-0 text-th-fgd-3 text-xs">
|
|
|
|
≈ $
|
|
|
|
{(
|
|
|
|
(selectedRoute?.outAmount /
|
|
|
|
10 ** (outputTokenInfo?.decimals || 1)) *
|
|
|
|
outputTokenPrice
|
|
|
|
).toFixed(2)}
|
2021-12-30 03:44:14 -08:00
|
|
|
</div>
|
2022-01-01 04:34:46 -08:00
|
|
|
) : null}
|
2021-12-29 03:43:37 -08:00
|
|
|
</div>
|
|
|
|
</div>
|
2022-01-01 04:34:46 -08:00
|
|
|
|
|
|
|
{routes?.length && selectedRoute ? (
|
|
|
|
<div className="mt-8 text-th-fgd-3 text-xs">
|
|
|
|
<div className="border border-th-bkg-4 mb-4 pb-4 px-3 pt-4 relative rounded-md">
|
|
|
|
{selectedRoute === routes[0] ? (
|
|
|
|
<div className="absolute bg-th-primary font-bold px-1 rounded-sm text-th-bkg-1 text-xs -top-2">
|
2022-01-02 16:52:03 -08:00
|
|
|
{t('swap:best-swap')}
|
2021-12-29 03:43:37 -08:00
|
|
|
</div>
|
2022-01-01 04:34:46 -08:00
|
|
|
) : null}
|
|
|
|
<div className="flex items-center justify-between">
|
|
|
|
<div>
|
|
|
|
<span className="font-bold overflow-ellipsis text-sm text-th-fgd-1 whitespace-nowrap">
|
|
|
|
{selectedRoute?.marketInfos.map((info, index) => {
|
|
|
|
let includeSeparator = false
|
|
|
|
if (
|
|
|
|
selectedRoute?.marketInfos.length > 1 &&
|
|
|
|
index !== selectedRoute?.marketInfos.length - 1
|
|
|
|
) {
|
|
|
|
includeSeparator = true
|
|
|
|
}
|
|
|
|
return (
|
|
|
|
<span key={index}>{`${
|
|
|
|
info.marketMeta.amm.label
|
|
|
|
} ${includeSeparator ? 'x ' : ''}`}</span>
|
|
|
|
)
|
|
|
|
})}
|
|
|
|
</span>
|
|
|
|
<div className="mr-2 mt-0.5 text-th-fgd-3 text-xs font-normal">
|
|
|
|
{inputTokenInfo?.symbol} →{' '}
|
|
|
|
{selectedRoute?.marketInfos.map((r, index) => {
|
|
|
|
const showArrow =
|
|
|
|
index !== selectedRoute?.marketInfos.length - 1
|
|
|
|
? true
|
|
|
|
: false
|
|
|
|
return (
|
|
|
|
<span key={index}>
|
|
|
|
<span>
|
|
|
|
{
|
|
|
|
tokens.find(
|
|
|
|
(item) =>
|
|
|
|
item?.address ===
|
|
|
|
r?.outputMint?.toString()
|
|
|
|
)?.symbol
|
|
|
|
}
|
|
|
|
</span>
|
|
|
|
{showArrow ? ' → ' : ''}
|
|
|
|
</span>
|
|
|
|
)
|
|
|
|
})}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<Button
|
|
|
|
className="bg-transparent border border-th-fgd-4 font-normal pb-1 pt-1 px-2 rounded-md text-th-fgd-3 text-center text-xs"
|
|
|
|
disabled={routes?.length === 1}
|
|
|
|
onClick={() => setShowRoutesModal(true)}
|
2021-12-30 03:44:14 -08:00
|
|
|
>
|
2022-01-02 16:52:03 -08:00
|
|
|
{t('swap:routes-found', {
|
2022-01-02 17:01:04 -08:00
|
|
|
numberOfRoutes: routes?.length,
|
2022-01-02 16:52:03 -08:00
|
|
|
})}
|
2022-01-01 04:34:46 -08:00
|
|
|
</Button>
|
2021-12-29 03:43:37 -08:00
|
|
|
</div>
|
|
|
|
</div>
|
2022-01-01 04:34:46 -08:00
|
|
|
<div className="px-3 space-y-2">
|
|
|
|
<div className="flex items-center justify-between mb-4">
|
|
|
|
<div className="font-bold text-sm text-th-fgd-1">
|
2022-01-02 16:52:03 -08:00
|
|
|
{t('swap:swap-details')}
|
2021-12-29 03:43:37 -08:00
|
|
|
</div>
|
2022-01-01 04:34:46 -08:00
|
|
|
<div className="flex items-center space-x-2">
|
|
|
|
<IconButton onClick={() => refresh()}>
|
|
|
|
<RefreshClockwiseIcon
|
|
|
|
className={`h-4 w-4 ${
|
|
|
|
loading ? 'animate-spin' : ''
|
|
|
|
}`}
|
|
|
|
/>
|
|
|
|
</IconButton>
|
|
|
|
<IconButton onClick={() => setShowSettings(true)}>
|
|
|
|
<CogIcon className="h-4 w-4" />
|
|
|
|
</IconButton>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<div className="flex justify-between">
|
2022-01-02 16:52:03 -08:00
|
|
|
<span>{t('swap:rate')}</span>
|
2022-01-02 04:20:42 -08:00
|
|
|
<div>
|
|
|
|
<div className="text-right text-th-fgd-1">
|
|
|
|
1 {outputTokenInfo?.symbol} ≈{' '}
|
|
|
|
{numberFormatter.format(
|
|
|
|
formValue?.amount / outAmountUi
|
|
|
|
)}{' '}
|
|
|
|
{inputTokenInfo?.symbol}
|
|
|
|
</div>
|
|
|
|
<div
|
|
|
|
className={`text-right ${
|
|
|
|
((formValue?.amount / outAmountUi -
|
|
|
|
outputTokenPrice) /
|
|
|
|
(formValue?.amount / outAmountUi)) *
|
|
|
|
100 <=
|
|
|
|
0
|
|
|
|
? 'text-th-green'
|
|
|
|
: 'text-th-red'
|
|
|
|
}`}
|
|
|
|
>
|
|
|
|
{Math.abs(
|
|
|
|
((formValue?.amount / outAmountUi -
|
|
|
|
outputTokenPrice) /
|
|
|
|
(formValue?.amount / outAmountUi)) *
|
|
|
|
100
|
|
|
|
).toFixed(1)}
|
|
|
|
%{' '}
|
|
|
|
<span className="text-th-fgd-4">{`${
|
|
|
|
((formValue?.amount / outAmountUi -
|
|
|
|
outputTokenPrice) /
|
|
|
|
(formValue?.amount / outAmountUi)) *
|
|
|
|
100 <=
|
|
|
|
0
|
|
|
|
? 'cheaper'
|
|
|
|
: 'more expensive'
|
|
|
|
} than CoinGecko`}</span>
|
|
|
|
</div>
|
2022-01-01 04:34:46 -08:00
|
|
|
</div>
|
|
|
|
</div>
|
2021-12-30 03:44:14 -08:00
|
|
|
<div className="flex justify-between">
|
2022-01-02 16:52:03 -08:00
|
|
|
<span>{t('swap:price-impact')}</span>
|
2022-01-01 04:34:46 -08:00
|
|
|
<div className="text-right text-th-fgd-1">
|
|
|
|
{selectedRoute?.priceImpactPct * 100 < 0.1
|
|
|
|
? '< 0.1%'
|
|
|
|
: `~ ${(
|
|
|
|
selectedRoute?.priceImpactPct * 100
|
|
|
|
).toFixed(4)}%`}
|
2021-12-29 03:43:37 -08:00
|
|
|
</div>
|
|
|
|
</div>
|
2022-01-01 04:34:46 -08:00
|
|
|
<div className="flex justify-between">
|
2022-01-02 16:52:03 -08:00
|
|
|
<span>{t('swap:minimum-received')}</span>
|
2022-01-01 04:34:46 -08:00
|
|
|
<div className="text-right text-th-fgd-1">
|
2022-01-02 04:20:42 -08:00
|
|
|
{numberFormatter.format(
|
|
|
|
selectedRoute?.outAmountWithSlippage /
|
|
|
|
10 ** outputTokenInfo?.decimals || 1
|
|
|
|
)}{' '}
|
2022-01-01 04:34:46 -08:00
|
|
|
{outputTokenInfo?.symbol}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
{!isNaN(feeValue) ? (
|
2021-12-30 03:44:14 -08:00
|
|
|
<div className="flex justify-between">
|
2022-01-02 16:52:03 -08:00
|
|
|
<span>{t('swap:swap-fee')}</span>
|
2021-12-30 03:44:14 -08:00
|
|
|
<div className="flex items-center">
|
2022-01-01 04:34:46 -08:00
|
|
|
<div className="text-right text-th-fgd-1">
|
|
|
|
≈ ${feeValue?.toFixed(2)}
|
|
|
|
</div>
|
2021-12-30 03:44:14 -08:00
|
|
|
<Tooltip
|
|
|
|
content={
|
2022-01-01 04:34:46 -08:00
|
|
|
<div className="space-y-2.5">
|
|
|
|
{selectedRoute?.marketInfos.map(
|
|
|
|
(info, index) => {
|
|
|
|
const feeToken = tokens.find(
|
|
|
|
(item) =>
|
|
|
|
item?.address === info.lpFee?.mint
|
|
|
|
)
|
|
|
|
return (
|
|
|
|
<div key={index}>
|
|
|
|
<span>
|
2022-01-02 16:52:03 -08:00
|
|
|
{t('swap:fees-paid-to', {
|
|
|
|
feeRecipient:
|
|
|
|
info.marketMeta?.amm?.label,
|
|
|
|
})}
|
2022-01-01 04:34:46 -08:00
|
|
|
</span>
|
|
|
|
<div className="text-th-fgd-1">
|
|
|
|
{(
|
|
|
|
info.lpFee?.amount /
|
|
|
|
Math.pow(10, feeToken?.decimals)
|
|
|
|
).toFixed(6)}{' '}
|
|
|
|
{feeToken?.symbol} (
|
|
|
|
{info.lpFee?.pct * 100}
|
|
|
|
%)
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
)}
|
|
|
|
</div>
|
2021-12-30 03:44:14 -08:00
|
|
|
}
|
|
|
|
placement={'left'}
|
|
|
|
>
|
|
|
|
<InformationCircleIcon className="cursor-help h-3.5 ml-1.5 w-3.5 text-th-primary" />
|
|
|
|
</Tooltip>
|
|
|
|
</div>
|
2022-01-01 04:34:46 -08:00
|
|
|
</div>
|
|
|
|
) : (
|
|
|
|
selectedRoute?.marketInfos.map((info, index) => {
|
|
|
|
const feeToken = tokens.find(
|
|
|
|
(item) => item?.address === info.lpFee?.mint
|
|
|
|
)
|
|
|
|
return (
|
|
|
|
<div className="flex justify-between" key={index}>
|
|
|
|
<span>
|
2022-01-02 16:52:03 -08:00
|
|
|
{t('swap:fees-paid-to', {
|
|
|
|
feeRecipient: info.marketMeta?.amm?.label,
|
|
|
|
})}
|
2022-01-01 04:34:46 -08:00
|
|
|
</span>
|
2021-12-30 03:44:14 -08:00
|
|
|
<div className="text-right text-th-fgd-1">
|
|
|
|
{(
|
2022-01-01 04:34:46 -08:00
|
|
|
info.lpFee?.amount /
|
|
|
|
Math.pow(10, feeToken?.decimals)
|
|
|
|
).toFixed(6)}{' '}
|
|
|
|
{feeToken?.symbol} ({info.lpFee?.pct * 100}%)
|
2021-12-30 03:44:14 -08:00
|
|
|
</div>
|
2022-01-01 04:34:46 -08:00
|
|
|
</div>
|
|
|
|
)
|
|
|
|
})
|
|
|
|
)}
|
|
|
|
{connected ? (
|
|
|
|
<>
|
|
|
|
<div className="flex justify-between">
|
2022-01-02 16:52:03 -08:00
|
|
|
<span>{t('swap:transaction-fee')}</span>
|
2022-01-01 04:34:46 -08:00
|
|
|
<div className="text-right text-th-fgd-1">
|
|
|
|
{depositAndFee
|
|
|
|
? depositAndFee?.signatureFee / Math.pow(10, 9)
|
|
|
|
: '-'}{' '}
|
|
|
|
SOL
|
|
|
|
</div>
|
2021-12-30 03:44:14 -08:00
|
|
|
</div>
|
2022-01-01 04:34:46 -08:00
|
|
|
{depositAndFee?.ataDepositLength ||
|
|
|
|
depositAndFee?.openOrdersDeposits?.length ? (
|
|
|
|
<div className="flex justify-between">
|
|
|
|
<div className="flex items-center">
|
2022-01-02 16:52:03 -08:00
|
|
|
<span>{t('deposit')}</span>
|
2022-01-01 04:34:46 -08:00
|
|
|
<Tooltip
|
|
|
|
content={
|
|
|
|
<>
|
|
|
|
{depositAndFee?.ataDepositLength ? (
|
2022-01-02 16:52:03 -08:00
|
|
|
<div>{t('need-ata-account')}</div>
|
2022-01-01 04:34:46 -08:00
|
|
|
) : null}
|
|
|
|
{depositAndFee?.openOrdersDeposits
|
|
|
|
?.length ? (
|
|
|
|
<div className="mt-2">
|
2022-01-02 16:52:03 -08:00
|
|
|
{t('swap:serum-requires-openorders')}{' '}
|
2022-01-01 04:34:46 -08:00
|
|
|
<a
|
|
|
|
href="https://docs.google.com/document/d/1qEWc_Bmc1aAxyCUcilKB4ZYpOu3B0BxIbe__dRYmVns/"
|
|
|
|
target="_blank"
|
|
|
|
rel="noopener noreferrer"
|
|
|
|
>
|
2022-01-02 16:52:03 -08:00
|
|
|
{t('swap:heres-how')}
|
2022-01-01 04:34:46 -08:00
|
|
|
</a>
|
|
|
|
</div>
|
|
|
|
) : null}
|
|
|
|
</>
|
|
|
|
}
|
|
|
|
placement={'left'}
|
|
|
|
>
|
|
|
|
<InformationCircleIcon className="cursor-help h-3.5 ml-1.5 w-3.5 text-th-primary" />
|
|
|
|
</Tooltip>
|
|
|
|
</div>
|
|
|
|
<div>
|
|
|
|
{depositAndFee?.ataDepositLength ? (
|
|
|
|
<div className="text-right text-th-fgd-1">
|
2022-01-02 16:52:03 -08:00
|
|
|
{depositAndFee?.ataDepositLength === 1
|
|
|
|
? t('swap:ata-deposit-details', {
|
|
|
|
cost: (
|
|
|
|
depositAndFee?.ataDeposit /
|
|
|
|
Math.pow(10, 9)
|
|
|
|
).toFixed(5),
|
|
|
|
count:
|
|
|
|
depositAndFee?.ataDepositLength,
|
|
|
|
})
|
|
|
|
: t('swap:ata-deposit-details_plural', {
|
|
|
|
cost: (
|
|
|
|
depositAndFee?.ataDeposit /
|
|
|
|
Math.pow(10, 9)
|
|
|
|
).toFixed(5),
|
|
|
|
count:
|
|
|
|
depositAndFee?.ataDepositLength,
|
|
|
|
})}
|
2022-01-01 04:34:46 -08:00
|
|
|
</div>
|
|
|
|
) : null}
|
|
|
|
{depositAndFee?.openOrdersDeposits?.length ? (
|
|
|
|
<div className="text-right text-th-fgd-1">
|
|
|
|
{depositAndFee?.openOrdersDeposits.length >
|
|
|
|
1
|
2022-01-02 16:52:03 -08:00
|
|
|
? t('swap:serum-details_plural', {
|
|
|
|
cost: (
|
|
|
|
sum(
|
|
|
|
depositAndFee?.openOrdersDeposits
|
|
|
|
) / Math.pow(10, 9)
|
|
|
|
).toFixed(5),
|
|
|
|
count:
|
|
|
|
depositAndFee?.openOrdersDeposits
|
|
|
|
.length,
|
|
|
|
})
|
|
|
|
: t('swap:serum-details', {
|
|
|
|
cost: (
|
|
|
|
sum(
|
|
|
|
depositAndFee?.openOrdersDeposits
|
|
|
|
) / Math.pow(10, 9)
|
|
|
|
).toFixed(5),
|
|
|
|
count:
|
|
|
|
depositAndFee?.openOrdersDeposits
|
|
|
|
.length,
|
|
|
|
})}
|
2022-01-01 04:34:46 -08:00
|
|
|
</div>
|
|
|
|
) : null}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
) : null}
|
|
|
|
</>
|
2021-12-30 03:44:14 -08:00
|
|
|
) : null}
|
2022-01-01 04:34:46 -08:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
) : null}
|
|
|
|
{error && (
|
|
|
|
<div className="flex items-center justify-center mt-2 text-th-red">
|
|
|
|
<ExclamationCircleIcon className="h-5 mr-1.5 w-5" />
|
2022-01-02 16:52:03 -08:00
|
|
|
{t('swap:jupiter-error')}
|
2022-01-01 04:34:46 -08:00
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
<Button
|
|
|
|
disabled={swapDisabled}
|
|
|
|
onClick={async () => {
|
|
|
|
if (!connected && zeroKey !== wallet?.publicKey) {
|
|
|
|
wallet.connect()
|
|
|
|
} else if (!loading && selectedRoute && connected) {
|
|
|
|
setSwapping(true)
|
|
|
|
let txCount = 1
|
|
|
|
let errorTxid
|
|
|
|
const swapResult = await exchange({
|
|
|
|
wallet: wallet,
|
|
|
|
route: selectedRoute,
|
|
|
|
confirmationWaiterFactory: async (txid, totalTxs) => {
|
|
|
|
console.log('txid, totalTxs', txid, totalTxs)
|
|
|
|
if (txCount === totalTxs) {
|
|
|
|
errorTxid = txid
|
|
|
|
notify({
|
|
|
|
type: 'confirm',
|
|
|
|
title: 'Confirming Transaction',
|
|
|
|
txid,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
await connection.confirmTransaction(txid)
|
|
|
|
|
|
|
|
txCount++
|
|
|
|
return await connection.getTransaction(txid, {
|
|
|
|
commitment: 'confirmed',
|
|
|
|
})
|
|
|
|
},
|
|
|
|
})
|
|
|
|
console.log('swapResult', swapResult)
|
|
|
|
|
|
|
|
setSwapping(false)
|
|
|
|
fetchWalletTokens()
|
|
|
|
if ('error' in swapResult) {
|
|
|
|
console.log('Error:', swapResult.error)
|
|
|
|
notify({
|
|
|
|
type: 'error',
|
|
|
|
title: swapResult.error.name,
|
|
|
|
description: swapResult.error.message,
|
|
|
|
txid: errorTxid,
|
|
|
|
})
|
|
|
|
} else if ('txid' in swapResult) {
|
2021-12-30 03:44:14 -08:00
|
|
|
notify({
|
2022-01-01 04:34:46 -08:00
|
|
|
type: 'success',
|
|
|
|
title: 'Swap Successful',
|
|
|
|
description: `Swapped ${
|
|
|
|
swapResult.inputAmount /
|
|
|
|
10 ** (inputTokenInfo?.decimals || 1)
|
|
|
|
} ${inputTokenInfo?.symbol} to ${
|
|
|
|
swapResult.outputAmount /
|
|
|
|
10 ** (outputTokenInfo?.decimals || 1)
|
|
|
|
} ${outputTokenInfo?.symbol}`,
|
|
|
|
txid: swapResult.txid,
|
2021-12-30 03:44:14 -08:00
|
|
|
})
|
2022-01-01 04:34:46 -08:00
|
|
|
setFormValue((val) => ({
|
|
|
|
...val,
|
|
|
|
amount: null,
|
|
|
|
}))
|
2021-12-30 03:44:14 -08:00
|
|
|
}
|
2022-01-01 04:34:46 -08:00
|
|
|
}
|
|
|
|
}}
|
|
|
|
className="h-12 mt-6 text-base w-full"
|
|
|
|
>
|
|
|
|
{connected
|
|
|
|
? swapping
|
2022-01-02 16:52:03 -08:00
|
|
|
? t('swap:swapping')
|
|
|
|
: t('swap')
|
|
|
|
: t('connect-wallet')}
|
2022-01-01 04:34:46 -08:00
|
|
|
</Button>
|
|
|
|
</div>
|
2021-12-29 03:43:37 -08:00
|
|
|
|
2022-01-01 04:34:46 -08:00
|
|
|
{showRoutesModal ? (
|
|
|
|
<Modal
|
|
|
|
isOpen={showRoutesModal}
|
|
|
|
onClose={() => setShowRoutesModal(false)}
|
|
|
|
>
|
|
|
|
<div className="font-bold mb-4 text-th-fgd-1 text-center text-lg">
|
2022-01-02 16:52:03 -08:00
|
|
|
{t('swap:routes-found', {
|
|
|
|
numberOfRoutes: routes?.length,
|
|
|
|
})}
|
2022-01-01 04:34:46 -08:00
|
|
|
</div>
|
|
|
|
<div className="max-h-96 overflow-x-hidden overflow-y-auto thin-scroll pr-1">
|
|
|
|
{routes.map((route, index) => {
|
|
|
|
const selected = selectedRoute === route
|
|
|
|
return (
|
|
|
|
<div
|
|
|
|
key={index}
|
|
|
|
className={`bg-th-bkg-3 border default-transition rounded mb-2 hover:bg-th-bkg-4 ${
|
|
|
|
selected
|
|
|
|
? 'border-th-primary text-th-primary hover:border-th-primary'
|
|
|
|
: 'border-transparent text-th-fgd-1'
|
|
|
|
}`}
|
|
|
|
>
|
|
|
|
<button
|
|
|
|
className="p-4 w-full"
|
|
|
|
onClick={() => handleSelectRoute(route)}
|
|
|
|
>
|
|
|
|
<div className="flex justify-between items-center">
|
|
|
|
<div className="flex flex-col text-left">
|
|
|
|
<div className="whitespace-nowrap overflow-ellipsis">
|
|
|
|
{route.marketInfos.map((info, index) => {
|
|
|
|
let includeSeparator = false
|
|
|
|
if (
|
|
|
|
route.marketInfos.length > 1 &&
|
|
|
|
index !== route.marketInfos.length - 1
|
|
|
|
) {
|
|
|
|
includeSeparator = true
|
|
|
|
}
|
|
|
|
return (
|
|
|
|
<span key={index}>{`${
|
|
|
|
info.marketMeta.amm.label
|
|
|
|
} ${includeSeparator ? 'x ' : ''}`}</span>
|
|
|
|
)
|
|
|
|
})}
|
|
|
|
</div>
|
|
|
|
<div className="text-th-fgd-4 text-xs font-normal">
|
|
|
|
{inputTokenInfo?.symbol} →{' '}
|
|
|
|
{route.marketInfos.map((r, index) => {
|
|
|
|
const showArrow =
|
|
|
|
index !== route.marketInfos.length - 1
|
|
|
|
? true
|
|
|
|
: false
|
|
|
|
return (
|
|
|
|
<span key={index}>
|
|
|
|
<span>
|
|
|
|
{
|
|
|
|
tokens.find(
|
|
|
|
(item) =>
|
|
|
|
item?.address ===
|
|
|
|
r?.outputMint?.toString()
|
|
|
|
)?.symbol
|
|
|
|
}
|
|
|
|
</span>
|
|
|
|
{showArrow ? ' → ' : ''}
|
|
|
|
</span>
|
|
|
|
)
|
|
|
|
})}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<div className="text-lg">
|
2022-01-02 04:20:42 -08:00
|
|
|
{numberFormatter.format(
|
2022-01-01 04:34:46 -08:00
|
|
|
route.outAmount /
|
2022-01-02 04:20:42 -08:00
|
|
|
10 ** (outputTokenInfo?.decimals || 1)
|
|
|
|
)}
|
2022-01-01 04:34:46 -08:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</button>
|
|
|
|
</div>
|
|
|
|
)
|
|
|
|
})}
|
|
|
|
</div>
|
|
|
|
</Modal>
|
|
|
|
) : null}
|
|
|
|
{showInputTokenSelect ? (
|
|
|
|
<SwapTokenSelect
|
|
|
|
isOpen={showInputTokenSelect}
|
|
|
|
onClose={() => setShowInputTokenSelect(false)}
|
|
|
|
sortedTokenMints={sortedTokenMints}
|
|
|
|
onTokenSelect={(token) => {
|
|
|
|
setShowInputTokenSelect(false)
|
2021-12-30 03:44:14 -08:00
|
|
|
setFormValue((val) => ({
|
|
|
|
...val,
|
2022-01-01 04:34:46 -08:00
|
|
|
inputMint: new PublicKey(token?.address),
|
2021-12-30 03:44:14 -08:00
|
|
|
}))
|
2022-01-01 04:34:46 -08:00
|
|
|
}}
|
|
|
|
/>
|
|
|
|
) : null}
|
|
|
|
{showOutputTokenSelect ? (
|
|
|
|
<SwapTokenSelect
|
|
|
|
isOpen={showOutputTokenSelect}
|
|
|
|
onClose={() => setShowOutputTokenSelect(false)}
|
|
|
|
sortedTokenMints={outputTokenMints}
|
|
|
|
onTokenSelect={(token) => {
|
|
|
|
setShowOutputTokenSelect(false)
|
|
|
|
setFormValue((val) => ({
|
|
|
|
...val,
|
|
|
|
outputMint: new PublicKey(token?.address),
|
|
|
|
}))
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
) : null}
|
|
|
|
{showSettings ? (
|
|
|
|
<SwapSettingsModal
|
|
|
|
isOpen={showSettings}
|
|
|
|
onClose={() => setShowSettings(false)}
|
|
|
|
slippage={slippage}
|
|
|
|
setSlippage={setSlippage}
|
|
|
|
/>
|
|
|
|
) : null}
|
|
|
|
{connected && !hasSwapped ? (
|
|
|
|
<Modal isOpen={!hasSwapped} onClose={() => setHasSwapped(true)}>
|
2022-01-02 16:52:03 -08:00
|
|
|
<ElementTitle>{t('swap:get-started')}</ElementTitle>
|
2022-01-01 04:34:46 -08:00
|
|
|
<div className="flex flex-col justify-center">
|
|
|
|
<div className="text-center text-th-fgd-3">
|
2022-01-02 16:52:03 -08:00
|
|
|
{t('swap-in-wallet')}
|
2021-12-30 03:44:14 -08:00
|
|
|
</div>
|
2022-01-01 04:34:46 -08:00
|
|
|
</div>
|
|
|
|
</Modal>
|
|
|
|
) : null}
|
|
|
|
{showInputTokenSelect ? (
|
|
|
|
<SwapTokenSelect
|
|
|
|
isOpen={showInputTokenSelect}
|
|
|
|
onClose={() => setShowInputTokenSelect(false)}
|
|
|
|
sortedTokenMints={sortedTokenMints}
|
|
|
|
onTokenSelect={(token) => {
|
|
|
|
setShowInputTokenSelect(false)
|
|
|
|
setFormValue((val) => ({
|
|
|
|
...val,
|
|
|
|
inputMint: new PublicKey(token?.address),
|
|
|
|
}))
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
) : null}
|
|
|
|
{showOutputTokenSelect ? (
|
|
|
|
<SwapTokenSelect
|
|
|
|
isOpen={showOutputTokenSelect}
|
|
|
|
onClose={() => setShowOutputTokenSelect(false)}
|
|
|
|
sortedTokenMints={outputTokenMints}
|
|
|
|
onTokenSelect={(token) => {
|
|
|
|
setShowOutputTokenSelect(false)
|
|
|
|
setFormValue((val) => ({
|
|
|
|
...val,
|
|
|
|
outputMint: new PublicKey(token?.address),
|
|
|
|
}))
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
) : null}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<div className="w-full md:w-1/2 xl:w-2/3">
|
|
|
|
{inputTokenInfo && outputTokenInfo ? (
|
|
|
|
<SwapTokenInfo
|
2022-01-02 16:52:03 -08:00
|
|
|
inputTokenId={inputTokenInfos?.extensions?.coingeckoId}
|
2022-01-01 04:34:46 -08:00
|
|
|
inputTokenSymbol={inputTokenInfo?.symbol}
|
2022-01-02 16:52:03 -08:00
|
|
|
outputTokenId={outputTokenInfos?.extensions?.coingeckoId}
|
2022-01-01 04:34:46 -08:00
|
|
|
/>
|
|
|
|
) : null}
|
|
|
|
</div>
|
2021-12-30 03:44:14 -08:00
|
|
|
</div>
|
|
|
|
</div>
|
2021-12-09 09:23:19 -08:00
|
|
|
</div>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
export default JupiterForm
|