2020-10-12 18:51:01 +02:00
|
|
|
import { useMemo } from 'react'
|
2020-10-06 20:28:38 +02:00
|
|
|
import { useSelector } from 'react-redux'
|
|
|
|
import contractMap from 'eth-contract-metadata'
|
|
|
|
import BigNumber from 'bignumber.js'
|
|
|
|
import { isEqual, shuffle } from 'lodash'
|
|
|
|
import { checksumAddress } from '../helpers/utils/util'
|
|
|
|
import { getTokenFiatAmount } from '../helpers/utils/token-util'
|
2020-11-03 00:41:28 +01:00
|
|
|
import {
|
|
|
|
getTokenExchangeRates,
|
|
|
|
getConversionRate,
|
|
|
|
getCurrentCurrency,
|
|
|
|
} from '../selectors'
|
2020-10-06 20:28:38 +02:00
|
|
|
import { getSwapsTokens } from '../ducks/swaps/swaps'
|
2020-10-09 21:17:56 +02:00
|
|
|
import { useSwapsEthToken } from './useSwapsEthToken'
|
2020-10-06 20:28:38 +02:00
|
|
|
import { useEqualityCheck } from './useEqualityCheck'
|
|
|
|
|
2020-11-03 00:41:28 +01:00
|
|
|
const tokenList = shuffle(
|
|
|
|
Object.entries(contractMap)
|
|
|
|
.map(([address, tokenData]) => ({
|
|
|
|
...tokenData,
|
|
|
|
address: address.toLowerCase(),
|
|
|
|
}))
|
|
|
|
.filter((tokenData) => Boolean(tokenData.erc20)),
|
|
|
|
)
|
2020-10-06 20:28:38 +02:00
|
|
|
|
2020-11-03 00:41:28 +01:00
|
|
|
export function getRenderableTokenData(
|
|
|
|
token,
|
|
|
|
contractExchangeRates,
|
|
|
|
conversionRate,
|
|
|
|
currentCurrency,
|
|
|
|
) {
|
2020-10-06 20:28:38 +02:00
|
|
|
const { symbol, name, address, iconUrl, string, balance, decimals } = token
|
|
|
|
|
2020-11-03 00:41:28 +01:00
|
|
|
const formattedFiat =
|
|
|
|
getTokenFiatAmount(
|
|
|
|
symbol === 'ETH' ? 1 : contractExchangeRates[address],
|
|
|
|
conversionRate,
|
|
|
|
currentCurrency,
|
|
|
|
string,
|
|
|
|
symbol,
|
|
|
|
true,
|
|
|
|
) || ''
|
|
|
|
const rawFiat =
|
|
|
|
getTokenFiatAmount(
|
|
|
|
symbol === 'ETH' ? 1 : contractExchangeRates[address],
|
|
|
|
conversionRate,
|
|
|
|
currentCurrency,
|
|
|
|
string,
|
|
|
|
symbol,
|
|
|
|
false,
|
|
|
|
) || ''
|
|
|
|
const usedIconUrl =
|
|
|
|
iconUrl ||
|
|
|
|
(contractMap[checksumAddress(address)] &&
|
|
|
|
`images/contract/${contractMap[checksumAddress(address)].logo}`)
|
2020-10-06 20:28:38 +02:00
|
|
|
return {
|
|
|
|
...token,
|
|
|
|
primaryLabel: symbol,
|
|
|
|
secondaryLabel: name || contractMap[checksumAddress(address)]?.name,
|
2020-11-03 00:41:28 +01:00
|
|
|
rightPrimaryLabel:
|
|
|
|
string && `${new BigNumber(string).round(6).toString()} ${symbol}`,
|
2020-10-06 20:28:38 +02:00
|
|
|
rightSecondaryLabel: formattedFiat,
|
|
|
|
iconUrl: usedIconUrl,
|
|
|
|
identiconAddress: usedIconUrl ? null : address,
|
|
|
|
balance,
|
|
|
|
decimals,
|
|
|
|
name: name || contractMap[checksumAddress(address)]?.name,
|
|
|
|
rawFiat,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-03 00:41:28 +01:00
|
|
|
export function useTokensToSearch({
|
2020-10-09 21:17:56 +02:00
|
|
|
providedTokens,
|
|
|
|
usersTokens = [],
|
|
|
|
topTokens = {},
|
|
|
|
onlyEth,
|
|
|
|
singleToken,
|
|
|
|
}) {
|
2020-10-06 20:28:38 +02:00
|
|
|
const tokenConversionRates = useSelector(getTokenExchangeRates, isEqual)
|
|
|
|
const conversionRate = useSelector(getConversionRate)
|
|
|
|
const currentCurrency = useSelector(getCurrentCurrency)
|
|
|
|
|
|
|
|
const memoizedTopTokens = useEqualityCheck(topTokens)
|
|
|
|
const memoizedUsersToken = useEqualityCheck(usersTokens)
|
|
|
|
|
2020-10-09 21:17:56 +02:00
|
|
|
const swapsEthToken = useSwapsEthToken()
|
2020-10-12 18:51:01 +02:00
|
|
|
const ethToken = getRenderableTokenData(
|
2020-10-09 21:17:56 +02:00
|
|
|
swapsEthToken,
|
2020-10-06 20:28:38 +02:00
|
|
|
tokenConversionRates,
|
|
|
|
conversionRate,
|
|
|
|
currentCurrency,
|
2020-10-12 18:51:01 +02:00
|
|
|
)
|
|
|
|
const memoizedEthToken = useEqualityCheck(ethToken)
|
2020-10-06 20:28:38 +02:00
|
|
|
|
|
|
|
const swapsTokens = useSelector(getSwapsTokens) || []
|
|
|
|
let tokensToSearch
|
|
|
|
if (onlyEth) {
|
2020-10-12 18:51:01 +02:00
|
|
|
tokensToSearch = [memoizedEthToken]
|
2020-10-06 20:28:38 +02:00
|
|
|
} else if (singleToken) {
|
|
|
|
tokensToSearch = providedTokens
|
|
|
|
} else if (providedTokens) {
|
2020-10-12 18:51:01 +02:00
|
|
|
tokensToSearch = [memoizedEthToken, ...providedTokens]
|
2020-10-06 20:28:38 +02:00
|
|
|
} else if (swapsTokens.length) {
|
2020-10-12 18:51:01 +02:00
|
|
|
tokensToSearch = [memoizedEthToken, ...swapsTokens]
|
2020-10-06 20:28:38 +02:00
|
|
|
} else {
|
2020-10-12 18:51:01 +02:00
|
|
|
tokensToSearch = [memoizedEthToken, ...tokenList]
|
2020-10-06 20:28:38 +02:00
|
|
|
}
|
|
|
|
const memoizedTokensToSearch = useEqualityCheck(tokensToSearch)
|
|
|
|
return useMemo(() => {
|
2020-11-03 00:41:28 +01:00
|
|
|
const usersTokensAddressMap = memoizedUsersToken.reduce(
|
|
|
|
(acc, token) => ({ ...acc, [token.address]: token }),
|
|
|
|
{},
|
|
|
|
)
|
2020-10-06 20:28:38 +02:00
|
|
|
|
|
|
|
const tokensToSearchBuckets = {
|
2020-10-12 18:51:01 +02:00
|
|
|
owned: singleToken ? [] : [memoizedEthToken],
|
2020-10-06 20:28:38 +02:00
|
|
|
top: [],
|
|
|
|
others: [],
|
|
|
|
}
|
|
|
|
|
|
|
|
memoizedTokensToSearch.forEach((token) => {
|
2020-11-03 00:41:28 +01:00
|
|
|
const renderableDataToken = getRenderableTokenData(
|
|
|
|
{ ...usersTokensAddressMap[token.address], ...token },
|
|
|
|
tokenConversionRates,
|
|
|
|
conversionRate,
|
|
|
|
currentCurrency,
|
|
|
|
)
|
|
|
|
if (
|
|
|
|
usersTokensAddressMap[token.address] &&
|
|
|
|
(renderableDataToken.symbol === 'ETH' ||
|
|
|
|
Number(renderableDataToken.balance ?? 0) !== 0)
|
|
|
|
) {
|
2020-10-06 20:28:38 +02:00
|
|
|
tokensToSearchBuckets.owned.push(renderableDataToken)
|
|
|
|
} else if (memoizedTopTokens[token.address]) {
|
2020-11-03 00:41:28 +01:00
|
|
|
tokensToSearchBuckets.top[
|
|
|
|
memoizedTopTokens[token.address].index
|
|
|
|
] = renderableDataToken
|
2020-10-06 20:28:38 +02:00
|
|
|
} else {
|
|
|
|
tokensToSearchBuckets.others.push(renderableDataToken)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
2020-11-03 00:41:28 +01:00
|
|
|
tokensToSearchBuckets.owned = tokensToSearchBuckets.owned.sort(
|
|
|
|
({ rawFiat }, { rawFiat: secondRawFiat }) => {
|
|
|
|
return new BigNumber(rawFiat || 0).gt(secondRawFiat || 0) ? -1 : 1
|
|
|
|
},
|
|
|
|
)
|
|
|
|
tokensToSearchBuckets.top = tokensToSearchBuckets.top.filter(
|
|
|
|
(token) => token,
|
|
|
|
)
|
2020-10-06 20:28:38 +02:00
|
|
|
return [
|
|
|
|
...tokensToSearchBuckets.owned,
|
|
|
|
...tokensToSearchBuckets.top,
|
|
|
|
...tokensToSearchBuckets.others,
|
|
|
|
]
|
2020-11-03 00:41:28 +01:00
|
|
|
}, [
|
|
|
|
memoizedTokensToSearch,
|
|
|
|
memoizedUsersToken,
|
|
|
|
tokenConversionRates,
|
|
|
|
conversionRate,
|
|
|
|
currentCurrency,
|
|
|
|
memoizedTopTokens,
|
|
|
|
memoizedEthToken,
|
|
|
|
singleToken,
|
|
|
|
])
|
2020-10-06 20:28:38 +02:00
|
|
|
}
|