import React, { useContext, useMemo } from 'react'; import PropTypes from 'prop-types'; import BigNumber from 'bignumber.js'; import { useSelector } from 'react-redux'; import { getTokenTrackerLink } from '@metamask/etherscan-link'; import { I18nContext } from '../../contexts/i18n'; import ConfirmTransactionBase from '../confirm-transaction-base'; import UserPreferencedCurrencyDisplay from '../../components/app/user-preferenced-currency-display'; import { formatCurrency, convertTokenToFiat, addFiat, roundExponential, } from '../../helpers/utils/confirm-tx.util'; import { PRIMARY } from '../../helpers/constants/common'; import { contractExchangeRateSelector, getCurrentChainId, getCurrentCurrency, getRpcPrefsForCurrentProvider, getSelectedAddress, } from '../../selectors'; import { getConversionRate, getNativeCurrency, getNftContracts, } from '../../ducks/metamask/metamask'; import { TokenStandard } from '../../../shared/constants/transaction'; import { getWeiHexFromDecimalValue, hexWEIToDecETH, } from '../../../shared/modules/conversion.utils'; import { EtherDenomination } from '../../../shared/constants/common'; import { CHAIN_IDS, TEST_CHAINS } from '../../../shared/constants/network'; export default function ConfirmTokenTransactionBase({ image = '', assetName, toAddress, tokenAddress, tokenAmount = '0', tokenSymbol, tokenId, assetStandard, onEdit, ethTransactionTotal, fiatTransactionTotal, hexMaximumTransactionFee, }) { const t = useContext(I18nContext); const contractExchangeRate = useSelector(contractExchangeRateSelector); const nativeCurrency = useSelector(getNativeCurrency); const currentCurrency = useSelector(getCurrentCurrency); const conversionRate = useSelector(getConversionRate); const rpcPrefs = useSelector(getRpcPrefsForCurrentProvider); const chainId = useSelector(getCurrentChainId); const userAddress = useSelector(getSelectedAddress); const nftCollections = useSelector(getNftContracts); const ethTransactionTotalMaxAmount = Number( hexWEIToDecETH(hexMaximumTransactionFee), ); const getTitleTokenDescription = (renderType) => { const useBlockExplorer = rpcPrefs?.blockExplorerUrl || [...TEST_CHAINS, CHAIN_IDS.MAINNET].includes(chainId); const nftCollection = nftCollections.find( (collection) => collection.address.toLowerCase() === tokenAddress.toLowerCase(), ); const titleTokenDescription = tokenSymbol || nftCollection?.name || t('unknownCollection'); if (renderType === 'text') { return titleTokenDescription; } if (useBlockExplorer) { const blockExplorerLink = getTokenTrackerLink( tokenAddress, chainId, null, userAddress, { blockExplorerUrl: rpcPrefs?.blockExplorerUrl ?? null, }, ); const blockExplorerElement = ( <> {titleTokenDescription} ); return blockExplorerElement; } return ( <> {titleTokenDescription} ); }; const assetImage = image; let title, subtitle, subtotalDisplay; if ( assetStandard === TokenStandard.ERC721 || assetStandard === TokenStandard.ERC1155 ) { title = assetName || getTitleTokenDescription(); subtitle = `#${tokenId}`; subtotalDisplay = assetName || `${getTitleTokenDescription('text')} #${tokenId}`; } else if (assetStandard === TokenStandard.ERC20) { title = `${tokenAmount} ${tokenSymbol}`; subtotalDisplay = `${tokenAmount} ${tokenSymbol}`; } const hexWeiValue = useMemo(() => { if (tokenAmount === '0' || !contractExchangeRate) { return '0'; } const decimalEthValue = new BigNumber(tokenAmount) .times( new BigNumber(contractExchangeRate ? String(contractExchangeRate) : 0), ) .toFixed(); return getWeiHexFromDecimalValue({ value: decimalEthValue, fromCurrency: EtherDenomination.ETH, fromDenomination: EtherDenomination.ETH, }); }, [tokenAmount, contractExchangeRate]); const secondaryTotalTextOverride = useMemo(() => { if (typeof contractExchangeRate === 'undefined') { return formatCurrency(fiatTransactionTotal, currentCurrency); } const fiatTransactionAmount = convertTokenToFiat({ value: tokenAmount, toCurrency: currentCurrency, conversionRate, contractExchangeRate, }); const fiatTotal = addFiat(fiatTransactionAmount, fiatTransactionTotal); const roundedFiatTotal = roundExponential(fiatTotal); return formatCurrency(roundedFiatTotal, currentCurrency); }, [ currentCurrency, conversionRate, contractExchangeRate, fiatTransactionTotal, tokenAmount, ]); const subtitleComponent = () => { if (contractExchangeRate === undefined && subtitle === undefined) { return {t('noConversionRateAvailable')}; } if (subtitle) { return {subtitle}; } return ( ); }; return ( ); } ConfirmTokenTransactionBase.propTypes = { image: PropTypes.string, assetName: PropTypes.string, toAddress: PropTypes.string, tokenAddress: PropTypes.string, tokenAmount: PropTypes.string, tokenSymbol: PropTypes.string, tokenId: PropTypes.string, assetStandard: PropTypes.string, onEdit: PropTypes.func, ethTransactionTotal: PropTypes.string, fiatTransactionTotal: PropTypes.string, hexMaximumTransactionFee: PropTypes.string, };