import { memo, useMemo, useEffect, useRef } from 'react' import { Token } from '../../types/jupiter' import mangoStore from '@store/mangoStore' import { IconButton } from '../shared/Button' import { XMarkIcon } from '@heroicons/react/20/solid' import { useTranslation } from 'next-i18next' import Decimal from 'decimal.js' import { getTokenInMax } from './useTokenMax' import useMangoAccount from 'hooks/useMangoAccount' import useJupiterMints from 'hooks/useJupiterMints' import useMangoGroup from 'hooks/useMangoGroup' import { PublicKey } from '@solana/web3.js' import FormatNumericValue from '@components/shared/FormatNumericValue' import { formatTokenSymbol } from 'utils/tokens' // const generateSearchTerm = (item: Token, searchValue: string) => { // const normalizedSearchValue = searchValue.toLowerCase() // const values = `${item.symbol} ${item.name}`.toLowerCase() // const isMatchingWithSymbol = // item.symbol.toLowerCase().indexOf(normalizedSearchValue) >= 0 // const matchingSymbolPercent = isMatchingWithSymbol // ? normalizedSearchValue.length / item.symbol.length // : 0 // return { // token: item, // matchingIdx: values.indexOf(normalizedSearchValue), // matchingSymbolPercent, // } // } // const startSearch = (items: Token[], searchValue: string) => { // return items // .map((item) => generateSearchTerm(item, searchValue)) // .filter((item) => item.matchingIdx >= 0) // .sort((i1, i2) => i1.matchingIdx - i2.matchingIdx) // .sort((i1, i2) => i2.matchingSymbolPercent - i1.matchingSymbolPercent) // .map((item) => item.token) // } const TokenItem = ({ token, onSubmit, useMargin, type, }: { token: TokenInfoWithAmounts onSubmit: (x: string) => void useMargin: boolean type: 'input' | 'output' | undefined }) => { const { t } = useTranslation('trade') const { address, symbol, logoURI, name } = token const bank = useMemo(() => { const group = mangoStore.getState().group if (!group) return return group.getFirstBankByMint(new PublicKey(address)) }, [address]) const isReduceOnly = useMemo(() => { if (!bank) return false const borrowsReduceOnly = bank.areBorrowsReduceOnly() const depositsReduceOnly = bank.areDepositsReduceOnly() return borrowsReduceOnly && depositsReduceOnly }, [bank]) return (
) } // const popularTokenSymbols = ['USDC', 'SOL', 'USDT', 'MNGO', 'BTC'] interface TokenInfoWithAmounts extends Token { amount?: Decimal amountWithBorrow?: Decimal } const SwapFormTokenList = ({ onClose, onTokenSelect, type, useMargin, }: { onClose: () => void onTokenSelect: (x: string) => void type: 'input' | 'output' | undefined useMargin: boolean }) => { const { t } = useTranslation(['common', 'swap']) // const [search, setSearch] = useState('') const { mangoTokens } = useJupiterMints() const inputBank = mangoStore((s) => s.swap.inputBank) const outputBank = mangoStore((s) => s.swap.outputBank) const { group } = useMangoGroup() const { mangoAccount } = useMangoAccount() const focusRef = useRef(null) // const popularTokens = useMemo(() => { // return tokens.filter((token) => { // return !token?.name || !token?.symbol // ? false // : popularTokenSymbols.includes(token.symbol) // }) // }, [tokens]) useEffect(() => { function onEscape(e: KeyboardEvent) { if (e.keyCode === 27) { onClose() } } window.addEventListener('keydown', onEscape) return () => window.removeEventListener('keydown', onEscape) }, [onClose]) const tokenInfos: TokenInfoWithAmounts[] = useMemo(() => { if ( mangoTokens?.length && group && mangoAccount && outputBank && inputBank && type === 'input' ) { const filteredSortedTokens = mangoTokens .map((token) => { const max = getTokenInMax( mangoAccount, new PublicKey(token.address), outputBank.mint, group, useMargin ) return { ...token, ...max } }) .filter((token) => (token.symbol === outputBank?.name ? false : true)) .sort((a, b) => useMargin ? Number(b.amountWithBorrow) - Number(a.amountWithBorrow) : Number(b.amount) - Number(a.amount) ) return filteredSortedTokens } else if (mangoTokens?.length) { const filteredTokens = mangoTokens .map((token) => ({ ...token, amount: new Decimal(0), amountWithBorrow: new Decimal(0), })) .filter((token) => (token.symbol === inputBank?.name ? false : true)) .sort((a, b) => a.symbol.localeCompare(b.symbol)) return filteredTokens } else { return [] } }, [mangoTokens, inputBank, outputBank, mangoAccount, group, useMargin, type]) // const handleUpdateSearch = (e: ChangeEvent) => { // setSearch(e.target.value) // } // const sortedTokens = search ? startSearch(tokenInfos, search) : tokenInfos const sortedTokens = tokenInfos useEffect(() => { if (focusRef?.current) { focusRef.current.focus() } }, [focusRef]) return ( <>

{type === 'input' ? t('swap:pay') : type === 'output' ? t('swap:receive') : ''}

{/* No need for search/popular tokens until we have more tokens */} {/*
} autoFocus value={search} onChange={handleUpdateSearch} />
{popularTokens.length ? (
{popularTokens.map((token) => { let logoURI if (mangoTokens.length) { logoURI = mangoTokens.find( (t) => t.address === token.address )!.logoURI } const disabled = (type === 'input' && token.symbol === outputBank?.name) || (type === 'output' && token.symbol === inputBank?.name) return ( ) })}
) : null} */} {/*
*/}

{t('token')}

{type === 'input' ? (

{t('max')}

) : null}
{sortedTokens.map((token) => ( ))}
) } export default memo(SwapFormTokenList)