mango-ui-v3/components/JupiterForm.tsx

1111 lines
47 KiB
TypeScript
Raw Normal View History

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'
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'
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'
2021-12-09 09:23:19 -08:00
type UseJupiterProps = Parameters<typeof useJupiter>[0]
const JupiterForm: FunctionComponent = () => {
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)
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 [
tokens.find(
(item) => item?.address === formValue.inputMint?.toBase58() || ''
),
tokens.find(
(item) => item?.address === formValue.outputMint?.toBase58() || ''
),
2021-12-09 09:23:19 -08:00
]
}, [
formValue.inputMint?.toBase58(),
formValue.outputMint?.toBase58(),
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(() => {
// Fetch token list from Jupiter API
fetch(TOKEN_LIST_URL['mainnet-beta'])
.then((response) => response.json())
.then((result) => {
const tokens = allTokenMints.map((mint) =>
result.find((item) => item?.address === mint)
)
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)
}
if (selectedRoute && connected) {
2021-12-09 09:23:19 -08:00
getDepositAndFee()
}
}, [selectedRoute])
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) => {
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]
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-19 12:21:18 -08:00
const outputWalletBalance = () => {
if (walletTokens.length) {
2021-12-19 12:21:18 -08:00
const walletToken = walletTokens.filter((t) => {
return t.account.mint.toString() === outputTokenInfo?.address
})
2021-12-19 12:21:18 -08:00
const largestTokenAccount = sortBy(walletToken, 'uiBalance').reverse()[0]
return largestTokenAccount?.uiBalance || 0.0
}
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,
}))
}
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
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">
Wallet
</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}`
: 'Unavailable'}
</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"
>
Pay
</label>
<div className="space-x-3">
<label htmlFor="amount" className="text-th-fgd-3 text-xs">
Bal: {inputWalletBalance()}
</label>
{connected ? (
<>
<LinkButton
className="text-th-primary text-xs"
onClick={() => {
setFormValue((val) => ({
...val,
amount: inputWalletBalance(),
}))
}}
>
Max
</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">
Receive
</label>
<span className="text-th-fgd-3 text-xs">
Bal: {outputWalletBalance()}
</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">
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-01 04:34:46 -08:00
{routes?.length - 1} other routes
</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">
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">
<span>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-01 04:34:46 -08:00
<span>Price Impact</span>
<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">
<span>Minimum Received</span>
<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-01 04:34:46 -08:00
<span>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>
Fees paid to{' '}
{info.marketMeta?.amm?.label}
</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>
Fees paid to {info.marketMeta?.amm?.label}
</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">
<span>Transaction Fee</span>
<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">
<span>Deposit</span>
<Tooltip
content={
<>
{depositAndFee?.ataDepositLength ? (
<div>
You need to have an Associated Token
Account.
</div>
) : null}
{depositAndFee?.openOrdersDeposits
?.length ? (
<div className="mt-2">
Serum requires an OpenOrders account
for each token. You can close the
account and recover the SOL later.{' '}
<a
href="https://docs.google.com/document/d/1qEWc_Bmc1aAxyCUcilKB4ZYpOu3B0BxIbe__dRYmVns/"
target="_blank"
rel="noopener noreferrer"
>
Here&apos;s how
</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">
{(
depositAndFee?.ataDeposit /
Math.pow(10, 9)
).toFixed(5)}{' '}
SOL for {depositAndFee?.ataDepositLength}{' '}
ATA Account
</div>
) : null}
{depositAndFee?.openOrdersDeposits?.length ? (
<div className="text-right text-th-fgd-1">
{(
sum(depositAndFee?.openOrdersDeposits) /
Math.pow(10, 9)
).toFixed(5)}{' '}
SOL for{' '}
{depositAndFee?.openOrdersDeposits.length}{' '}
Serum OpenOrders{' '}
{depositAndFee?.openOrdersDeposits.length >
1
? 'Accounts'
: 'Account'}
</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" />
Error in Jupiter Try changing your input
</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
? 'Swapping...'
: 'Swap'
: 'Connect Wallet'}
</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">
{routes?.length} routes
</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)}>
<ElementTitle>Before you get started...</ElementTitle>
<div className="flex flex-col justify-center">
<div className="text-center text-th-fgd-3">
Swaps interact directly with your connected wallet, not
your Mango Account.
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}
{connected && !hasSwapped ? (
<Modal isOpen={!hasSwapped} onClose={() => setHasSwapped(true)}>
<ElementTitle>Before you get started...</ElementTitle>
<div className="flex flex-col justify-center">
<div className="text-center text-th-fgd-3">
Swaps interact directly with your connected wallet, not
your Mango Account.
</div>
<Button
className="mt-6 mx-auto"
onClick={() => setHasSwapped(true)}
>
Got It
</Button>
</div>
</Modal>
) : null}
</div>
</div>
<div className="w-full md:w-1/2 xl:w-2/3">
{inputTokenInfo && outputTokenInfo ? (
<SwapTokenInfo
inputTokenId={inputTokenInfo?.extensions?.coingeckoId}
inputTokenSymbol={inputTokenInfo?.symbol}
outputTokenId={outputTokenInfo?.extensions?.coingeckoId}
/>
) : null}
</div>
2021-12-30 03:44:14 -08:00
</div>
</div>
2021-12-09 09:23:19 -08:00
</div>
)
}
export default JupiterForm