import React, { useEffect, useRef, useState, useCallback } from 'react'; import { useDispatch, useSelector } from 'react-redux'; import { useParams } from 'react-router-dom'; import ConfirmTransactionBase from '../confirm-transaction-base'; import { EDIT_GAS_MODES } from '../../../shared/constants/gas'; import { showModal, updateCustomNonce, getNextNonce, } from '../../store/actions'; import { getTokenData } from '../../helpers/utils/transactions.util'; import { calcTokenAmount, getTokenAddressParam, getTokenValueParam, } from '../../helpers/utils/token-util'; import { readAddressAsContract } from '../../../shared/modules/contract-utils'; import { GasFeeContextProvider } from '../../contexts/gasFee'; import { TransactionModalContextProvider } from '../../contexts/transaction-modal'; import { useTokenTracker } from '../../hooks/useTokenTracker'; import { getTokens, getNativeCurrency, isAddressLedger, } from '../../ducks/metamask/metamask'; import { transactionFeeSelector, txDataSelector, getCurrentCurrency, getSubjectMetadata, getUseNonceField, getCustomNonceValue, getNextSuggestedNonce, getCurrentChainId, getRpcPrefsForCurrentProvider, getIsMultiLayerFeeNetwork, checkNetworkAndAccountSupports1559, } from '../../selectors'; import { useApproveTransaction } from '../../hooks/useApproveTransaction'; import { currentNetworkTxListSelector } from '../../selectors/transactions'; import AdvancedGasFeePopover from '../../components/app/advanced-gas-fee-popover'; import EditGasFeePopover from '../../components/app/edit-gas-fee-popover'; import EditGasPopover from '../../components/app/edit-gas-popover/edit-gas-popover.component'; import Loading from '../../components/ui/loading-screen'; import { isEqualCaseInsensitive } from '../../helpers/utils/util'; import { getCustomTxParamsData } from './confirm-approve.util'; import ConfirmApproveContent from './confirm-approve-content'; const isAddressLedgerByFromAddress = (address) => (state) => { return isAddressLedger(state, address); }; // eslint-disable-next-line prefer-destructuring const EIP_1559_V2_ENABLED = process.env.EIP_1559_V2 === true || process.env.EIP_1559_V2 === 'true'; export default function ConfirmApprove() { const dispatch = useDispatch(); const { id: paramsTransactionId } = useParams(); const { id: transactionId, txParams: { to: tokenAddress, data, from } = {}, } = useSelector(txDataSelector); const currentCurrency = useSelector(getCurrentCurrency); const nativeCurrency = useSelector(getNativeCurrency); const currentNetworkTxList = useSelector(currentNetworkTxListSelector); const subjectMetadata = useSelector(getSubjectMetadata); const tokens = useSelector(getTokens); const useNonceField = useSelector(getUseNonceField); const nextNonce = useSelector(getNextSuggestedNonce); const customNonceValue = useSelector(getCustomNonceValue); const chainId = useSelector(getCurrentChainId); const rpcPrefs = useSelector(getRpcPrefsForCurrentProvider); const isMultiLayerFeeNetwork = useSelector(getIsMultiLayerFeeNetwork); const networkAndAccountSupports1559 = useSelector( checkNetworkAndAccountSupports1559, ); const fromAddressIsLedger = useSelector(isAddressLedgerByFromAddress(from)); const transaction = currentNetworkTxList.find( ({ id }) => id === (Number(paramsTransactionId) || transactionId), ) || {}; const { ethTransactionTotal, fiatTransactionTotal, hexTransactionTotal, } = useSelector((state) => transactionFeeSelector(state, transaction)); const supportsEIP1559V2 = EIP_1559_V2_ENABLED && networkAndAccountSupports1559; const currentToken = (tokens && tokens.find(({ address }) => isEqualCaseInsensitive(tokenAddress, address), )) || { address: tokenAddress, }; const { tokensWithBalances } = useTokenTracker([currentToken]); const tokenTrackerBalance = tokensWithBalances[0]?.balance || ''; const tokenSymbol = currentToken?.symbol; const decimals = Number(currentToken?.decimals); const tokenImage = currentToken?.image; const tokenData = getTokenData(data); const tokenValue = getTokenValueParam(tokenData); const toAddress = getTokenAddressParam(tokenData); const tokenAmount = tokenData && calcTokenAmount(tokenValue, decimals).toString(10); const [customPermissionAmount, setCustomPermissionAmount] = useState(''); const previousTokenAmount = useRef(tokenAmount); const { approveTransaction, showCustomizeGasPopover, closeCustomizeGasPopover, } = useApproveTransaction(); useEffect(() => { if (customPermissionAmount && previousTokenAmount.current !== tokenAmount) { setCustomPermissionAmount(tokenAmount); } previousTokenAmount.current = tokenAmount; }, [customPermissionAmount, tokenAmount]); const [submitWarning, setSubmitWarning] = useState(''); const prevNonce = useRef(nextNonce); const prevCustomNonce = useRef(customNonceValue); useEffect(() => { if ( prevNonce.current !== nextNonce || prevCustomNonce.current !== customNonceValue ) { if (nextNonce !== null && customNonceValue > nextNonce) { setSubmitWarning( `Nonce is higher than suggested nonce of ${nextNonce}`, ); } else { setSubmitWarning(''); } } prevCustomNonce.current = customNonceValue; prevNonce.current = nextNonce; }, [customNonceValue, nextNonce]); const [isContract, setIsContract] = useState(false); const checkIfContract = useCallback(async () => { const { isContractAddress } = await readAddressAsContract( global.eth, toAddress, ); setIsContract(isContractAddress); }, [setIsContract, toAddress]); useEffect(() => { checkIfContract(); }, [checkIfContract]); const { origin } = transaction; const formattedOrigin = origin || ''; const { iconUrl: siteImage = '' } = subjectMetadata[origin] || {}; const tokensText = `${Number(tokenAmount)} ${tokenSymbol}`; const tokenBalance = tokenTrackerBalance ? calcTokenAmount(tokenTrackerBalance, decimals).toString(10) : ''; const customData = customPermissionAmount ? getCustomTxParamsData(data, { customPermissionAmount, decimals }) : null; return tokenSymbol === undefined ? ( ) : ( dispatch( showModal({ name: 'EDIT_APPROVAL_PERMISSION', customTokenAmount, decimals, origin, setCustomAmount, tokenAmount, tokenBalance, tokenSymbol, }), ) } data={customData || data} toAddress={toAddress} currentCurrency={currentCurrency} nativeCurrency={nativeCurrency} ethTransactionTotal={ethTransactionTotal} fiatTransactionTotal={fiatTransactionTotal} hexTransactionTotal={hexTransactionTotal} useNonceField={useNonceField} nextNonce={nextNonce} customNonceValue={customNonceValue} updateCustomNonce={(value) => { dispatch(updateCustomNonce(value)); }} getNextNonce={() => dispatch(getNextNonce())} showCustomizeNonceModal={({ /* eslint-disable no-shadow */ useNonceField, nextNonce, customNonceValue, updateCustomNonce, getNextNonce, /* eslint-disable no-shadow */ }) => dispatch( showModal({ name: 'CUSTOMIZE_NONCE', useNonceField, nextNonce, customNonceValue, updateCustomNonce, getNextNonce, }), ) } warning={submitWarning} txData={transaction} fromAddressIsLedger={fromAddressIsLedger} chainId={chainId} rpcPrefs={rpcPrefs} isContract={isContract} isMultiLayerFeeNetwork={isMultiLayerFeeNetwork} supportsEIP1559V2={supportsEIP1559V2} /> {showCustomizeGasPopover && !supportsEIP1559V2 && ( )} {supportsEIP1559V2 && ( <> )} } hideSenderToRecipient customTxParamsData={customData} /> ); }