2021-02-04 19:15:23 +01:00
|
|
|
import EventEmitter from 'events';
|
|
|
|
import React, { useContext, useRef, useState } from 'react';
|
|
|
|
import { useDispatch, useSelector } from 'react-redux';
|
|
|
|
import PropTypes from 'prop-types';
|
|
|
|
import { useHistory } from 'react-router-dom';
|
|
|
|
import { I18nContext } from '../../../contexts/i18n';
|
|
|
|
import { useNewMetricEvent } from '../../../hooks/useMetricEvent';
|
|
|
|
import { MetaMetricsContext } from '../../../contexts/metametrics.new';
|
|
|
|
import { getCurrentCurrency, getUSDConversionRate } from '../../../selectors';
|
2020-10-06 20:28:38 +02:00
|
|
|
import {
|
2020-11-03 00:41:28 +01:00
|
|
|
getUsedQuote,
|
|
|
|
getFetchParams,
|
|
|
|
getApproveTxParams,
|
2020-11-04 17:14:08 +01:00
|
|
|
getUsedSwapsGasPrice,
|
2020-10-06 20:28:38 +02:00
|
|
|
fetchQuotesAndSetQuoteState,
|
|
|
|
navigateBackToBuildQuote,
|
|
|
|
prepareForRetryGetQuotes,
|
|
|
|
prepareToLeaveSwaps,
|
2021-02-04 19:15:23 +01:00
|
|
|
} from '../../../ducks/swaps/swaps';
|
|
|
|
import Mascot from '../../../components/ui/mascot';
|
|
|
|
import PulseLoader from '../../../components/ui/pulse-loader';
|
|
|
|
import { getBlockExplorerUrlForTx } from '../../../helpers/utils/transactions.util';
|
2020-10-06 20:28:38 +02:00
|
|
|
import {
|
|
|
|
QUOTES_EXPIRED_ERROR,
|
|
|
|
SWAP_FAILED_ERROR,
|
|
|
|
ERROR_FETCHING_QUOTES,
|
|
|
|
QUOTES_NOT_AVAILABLE_ERROR,
|
|
|
|
OFFLINE_FOR_MAINTENANCE,
|
2021-02-04 19:15:23 +01:00
|
|
|
} from '../../../helpers/constants/swaps';
|
|
|
|
import { ASSET_ROUTE, DEFAULT_ROUTE } from '../../../helpers/constants/routes';
|
2020-10-06 20:28:38 +02:00
|
|
|
|
2021-02-04 19:15:23 +01:00
|
|
|
import { getRenderableNetworkFeesForQuote } from '../swaps.util';
|
|
|
|
import SwapsFooter from '../swaps-footer';
|
|
|
|
import SwapFailureIcon from './swap-failure-icon';
|
|
|
|
import SwapSuccessIcon from './swap-success-icon';
|
|
|
|
import QuotesTimeoutIcon from './quotes-timeout-icon';
|
|
|
|
import ViewOnEtherScanLink from './view-on-ether-scan-link';
|
2020-10-06 20:28:38 +02:00
|
|
|
|
2020-11-03 00:41:28 +01:00
|
|
|
export default function AwaitingSwap({
|
2020-10-06 20:28:38 +02:00
|
|
|
swapComplete,
|
|
|
|
errorKey,
|
|
|
|
txHash,
|
|
|
|
networkId,
|
|
|
|
tokensReceived,
|
|
|
|
rpcPrefs,
|
|
|
|
submittingSwap,
|
|
|
|
inputValue,
|
|
|
|
maxSlippage,
|
|
|
|
}) {
|
2021-02-04 19:15:23 +01:00
|
|
|
const t = useContext(I18nContext);
|
|
|
|
const metaMetricsEvent = useContext(MetaMetricsContext);
|
|
|
|
const history = useHistory();
|
|
|
|
const dispatch = useDispatch();
|
|
|
|
const animationEventEmitter = useRef(new EventEmitter());
|
2020-10-06 20:28:38 +02:00
|
|
|
|
2021-02-04 19:15:23 +01:00
|
|
|
const fetchParams = useSelector(getFetchParams);
|
|
|
|
const { destinationTokenInfo, sourceTokenInfo } = fetchParams?.metaData || {};
|
|
|
|
const usedQuote = useSelector(getUsedQuote);
|
|
|
|
const approveTxParams = useSelector(getApproveTxParams);
|
|
|
|
const swapsGasPrice = useSelector(getUsedSwapsGasPrice);
|
|
|
|
const currentCurrency = useSelector(getCurrentCurrency);
|
|
|
|
const usdConversionRate = useSelector(getUSDConversionRate);
|
2020-10-06 20:28:38 +02:00
|
|
|
|
2020-11-03 00:41:28 +01:00
|
|
|
const [trackedQuotesExpiredEvent, setTrackedQuotesExpiredEvent] = useState(
|
|
|
|
false,
|
2021-02-04 19:15:23 +01:00
|
|
|
);
|
2020-10-06 20:28:38 +02:00
|
|
|
|
2021-02-04 19:15:23 +01:00
|
|
|
let feeinUnformattedFiat;
|
2020-11-04 17:14:08 +01:00
|
|
|
|
|
|
|
if (usedQuote && swapsGasPrice) {
|
2020-10-27 20:02:21 +01:00
|
|
|
const renderableNetworkFees = getRenderableNetworkFeesForQuote(
|
|
|
|
usedQuote.gasEstimateWithRefund || usedQuote.averageGas,
|
|
|
|
approveTxParams?.gas || '0x0',
|
2020-11-04 17:14:08 +01:00
|
|
|
swapsGasPrice,
|
2020-10-27 20:02:21 +01:00
|
|
|
currentCurrency,
|
2020-11-10 18:39:45 +01:00
|
|
|
usdConversionRate,
|
2020-11-04 17:14:08 +01:00
|
|
|
usedQuote?.trade?.value,
|
2020-10-27 20:02:21 +01:00
|
|
|
sourceTokenInfo?.symbol,
|
|
|
|
usedQuote.sourceAmount,
|
2021-02-04 19:15:23 +01:00
|
|
|
);
|
|
|
|
feeinUnformattedFiat = renderableNetworkFees.rawNetworkFees;
|
2020-10-06 20:28:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
const quotesExpiredEvent = useNewMetricEvent({
|
|
|
|
event: 'Quotes Timed Out',
|
2020-11-10 16:49:01 +01:00
|
|
|
sensitiveProperties: {
|
2020-10-06 20:28:38 +02:00
|
|
|
token_from: sourceTokenInfo?.symbol,
|
|
|
|
token_from_amount: fetchParams?.value,
|
|
|
|
token_to: destinationTokenInfo?.symbol,
|
|
|
|
request_type: fetchParams?.balanceError ? 'Quote' : 'Order',
|
|
|
|
slippage: fetchParams?.slippage,
|
|
|
|
custom_slippage: fetchParams?.slippage === 2,
|
2020-11-10 18:39:45 +01:00
|
|
|
gas_fees: feeinUnformattedFiat,
|
2020-10-06 20:28:38 +02:00
|
|
|
},
|
|
|
|
category: 'swaps',
|
2021-02-04 19:15:23 +01:00
|
|
|
});
|
2020-10-06 20:28:38 +02:00
|
|
|
|
2020-11-03 00:41:28 +01:00
|
|
|
const blockExplorerUrl =
|
2021-02-04 19:15:23 +01:00
|
|
|
txHash && getBlockExplorerUrlForTx(networkId, txHash, rpcPrefs);
|
2020-10-06 20:28:38 +02:00
|
|
|
|
2021-02-04 19:15:23 +01:00
|
|
|
let headerText;
|
|
|
|
let statusImage;
|
|
|
|
let descriptionText;
|
|
|
|
let submitText;
|
|
|
|
let content;
|
2020-10-06 20:28:38 +02:00
|
|
|
|
|
|
|
if (errorKey === OFFLINE_FOR_MAINTENANCE) {
|
2021-02-04 19:15:23 +01:00
|
|
|
headerText = t('offlineForMaintenance');
|
|
|
|
descriptionText = t('metamaskSwapsOfflineDescription');
|
|
|
|
submitText = t('close');
|
|
|
|
statusImage = <SwapFailureIcon />;
|
2020-10-06 20:28:38 +02:00
|
|
|
} else if (errorKey === SWAP_FAILED_ERROR) {
|
2021-02-04 19:15:23 +01:00
|
|
|
headerText = t('swapFailedErrorTitle');
|
2021-03-01 17:32:02 +01:00
|
|
|
descriptionText = t('swapFailedErrorDescriptionWithSupportLink', [
|
|
|
|
<a
|
|
|
|
className="awaiting-swap__support-link"
|
|
|
|
key="awaiting-swap-support-link"
|
|
|
|
href="https://support.metamask.io"
|
|
|
|
target="_blank"
|
|
|
|
rel="noopener noreferrer"
|
|
|
|
>
|
|
|
|
support.metamask.io
|
|
|
|
</a>,
|
|
|
|
]);
|
2021-02-04 19:15:23 +01:00
|
|
|
submitText = t('tryAgain');
|
|
|
|
statusImage = <SwapFailureIcon />;
|
2020-10-06 20:28:38 +02:00
|
|
|
content = blockExplorerUrl && (
|
|
|
|
<ViewOnEtherScanLink
|
|
|
|
txHash={txHash}
|
|
|
|
blockExplorerUrl={blockExplorerUrl}
|
|
|
|
isCustomBlockExplorerUrl={Boolean(rpcPrefs.blockExplorerUrl)}
|
|
|
|
/>
|
2021-02-04 19:15:23 +01:00
|
|
|
);
|
2020-10-06 20:28:38 +02:00
|
|
|
} else if (errorKey === QUOTES_EXPIRED_ERROR) {
|
2021-02-04 19:15:23 +01:00
|
|
|
headerText = t('swapQuotesExpiredErrorTitle');
|
|
|
|
descriptionText = t('swapQuotesExpiredErrorDescription');
|
|
|
|
submitText = t('tryAgain');
|
|
|
|
statusImage = <QuotesTimeoutIcon />;
|
2020-10-06 20:28:38 +02:00
|
|
|
|
|
|
|
if (!trackedQuotesExpiredEvent) {
|
2021-02-04 19:15:23 +01:00
|
|
|
setTrackedQuotesExpiredEvent(true);
|
|
|
|
quotesExpiredEvent();
|
2020-10-06 20:28:38 +02:00
|
|
|
}
|
|
|
|
} else if (errorKey === ERROR_FETCHING_QUOTES) {
|
2021-02-04 19:15:23 +01:00
|
|
|
headerText = t('swapFetchingQuotesErrorTitle');
|
|
|
|
descriptionText = t('swapFetchingQuotesErrorDescription');
|
|
|
|
submitText = t('back');
|
|
|
|
statusImage = <SwapFailureIcon />;
|
2020-10-06 20:28:38 +02:00
|
|
|
} else if (errorKey === QUOTES_NOT_AVAILABLE_ERROR) {
|
2021-02-04 19:15:23 +01:00
|
|
|
headerText = t('swapQuotesNotAvailableErrorTitle');
|
|
|
|
descriptionText = t('swapQuotesNotAvailableErrorDescription');
|
|
|
|
submitText = t('tryAgain');
|
|
|
|
statusImage = <SwapFailureIcon />;
|
2020-10-06 20:28:38 +02:00
|
|
|
} else if (!errorKey && !swapComplete) {
|
2021-02-04 19:15:23 +01:00
|
|
|
headerText = t('swapProcessing');
|
|
|
|
statusImage = <PulseLoader />;
|
|
|
|
submitText = t('swapsViewInActivity');
|
2020-11-03 00:41:28 +01:00
|
|
|
descriptionText = t('swapOnceTransactionHasProcess', [
|
|
|
|
<span
|
|
|
|
key="swapOnceTransactionHasProcess-1"
|
|
|
|
className="awaiting-swap__amount-and-symbol"
|
|
|
|
>
|
|
|
|
{destinationTokenInfo.symbol}
|
|
|
|
</span>,
|
2021-02-04 19:15:23 +01:00
|
|
|
]);
|
2020-12-03 00:25:19 +01:00
|
|
|
content = blockExplorerUrl && (
|
|
|
|
<ViewOnEtherScanLink
|
|
|
|
txHash={txHash}
|
|
|
|
blockExplorerUrl={blockExplorerUrl}
|
|
|
|
isCustomBlockExplorerUrl={Boolean(rpcPrefs.blockExplorerUrl)}
|
|
|
|
/>
|
2021-02-04 19:15:23 +01:00
|
|
|
);
|
2020-10-06 20:28:38 +02:00
|
|
|
} else if (!errorKey && swapComplete) {
|
2021-02-04 19:15:23 +01:00
|
|
|
headerText = t('swapTransactionComplete');
|
|
|
|
statusImage = <SwapSuccessIcon />;
|
|
|
|
submitText = t('swapViewToken', [destinationTokenInfo.symbol]);
|
2020-11-03 00:41:28 +01:00
|
|
|
descriptionText = t('swapTokenAvailable', [
|
|
|
|
<span
|
|
|
|
key="swapTokenAvailable-2"
|
|
|
|
className="awaiting-swap__amount-and-symbol"
|
|
|
|
>
|
|
|
|
{`${tokensReceived || ''} ${destinationTokenInfo.symbol}`}
|
|
|
|
</span>,
|
2021-02-04 19:15:23 +01:00
|
|
|
]);
|
2020-10-06 20:28:38 +02:00
|
|
|
content = blockExplorerUrl && (
|
|
|
|
<ViewOnEtherScanLink
|
|
|
|
txHash={txHash}
|
|
|
|
blockExplorerUrl={blockExplorerUrl}
|
|
|
|
isCustomBlockExplorerUrl={Boolean(rpcPrefs.blockExplorerUrl)}
|
|
|
|
/>
|
2021-02-04 19:15:23 +01:00
|
|
|
);
|
2020-10-06 20:28:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div className="awaiting-swap">
|
|
|
|
<div className="awaiting-swap__content">
|
|
|
|
{!(swapComplete || errorKey) && (
|
|
|
|
<Mascot
|
|
|
|
animationEventEmitter={animationEventEmitter.current}
|
|
|
|
width="90"
|
|
|
|
height="90"
|
|
|
|
/>
|
|
|
|
)}
|
2020-11-03 00:41:28 +01:00
|
|
|
<div className="awaiting-swap__status-image">{statusImage}</div>
|
|
|
|
<div className="awaiting-swap__header">{headerText}</div>
|
|
|
|
<div className="awaiting-swap__main-descrption">{descriptionText}</div>
|
2020-10-06 20:28:38 +02:00
|
|
|
{content}
|
|
|
|
</div>
|
|
|
|
<SwapsFooter
|
|
|
|
onSubmit={async () => {
|
|
|
|
if (errorKey === OFFLINE_FOR_MAINTENANCE) {
|
2021-02-04 19:15:23 +01:00
|
|
|
await dispatch(prepareToLeaveSwaps());
|
|
|
|
history.push(DEFAULT_ROUTE);
|
2020-10-06 20:28:38 +02:00
|
|
|
} else if (errorKey === QUOTES_EXPIRED_ERROR) {
|
2021-02-04 19:15:23 +01:00
|
|
|
dispatch(prepareForRetryGetQuotes());
|
2020-11-03 00:41:28 +01:00
|
|
|
await dispatch(
|
|
|
|
fetchQuotesAndSetQuoteState(
|
|
|
|
history,
|
|
|
|
inputValue,
|
|
|
|
maxSlippage,
|
|
|
|
metaMetricsEvent,
|
|
|
|
),
|
2021-02-04 19:15:23 +01:00
|
|
|
);
|
2020-10-06 20:28:38 +02:00
|
|
|
} else if (errorKey) {
|
2021-02-04 19:15:23 +01:00
|
|
|
await dispatch(navigateBackToBuildQuote(history));
|
2020-10-06 20:28:38 +02:00
|
|
|
} else if (destinationTokenInfo?.symbol === 'ETH') {
|
2021-02-04 19:15:23 +01:00
|
|
|
history.push(DEFAULT_ROUTE);
|
2020-10-06 20:28:38 +02:00
|
|
|
} else {
|
2021-02-04 19:15:23 +01:00
|
|
|
history.push(`${ASSET_ROUTE}/${destinationTokenInfo?.address}`);
|
2020-10-06 20:28:38 +02:00
|
|
|
}
|
|
|
|
}}
|
|
|
|
onCancel={async () => await dispatch(navigateBackToBuildQuote(history))}
|
|
|
|
submitText={submitText}
|
|
|
|
disabled={submittingSwap}
|
|
|
|
hideCancel={errorKey !== QUOTES_EXPIRED_ERROR}
|
|
|
|
/>
|
|
|
|
</div>
|
2021-02-04 19:15:23 +01:00
|
|
|
);
|
2020-10-06 20:28:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
AwaitingSwap.propTypes = {
|
|
|
|
swapComplete: PropTypes.bool,
|
|
|
|
networkId: PropTypes.string.isRequired,
|
|
|
|
txHash: PropTypes.string,
|
|
|
|
tokensReceived: PropTypes.string,
|
|
|
|
rpcPrefs: PropTypes.object.isRequired,
|
|
|
|
errorKey: PropTypes.oneOf([
|
|
|
|
QUOTES_EXPIRED_ERROR,
|
|
|
|
SWAP_FAILED_ERROR,
|
|
|
|
ERROR_FETCHING_QUOTES,
|
|
|
|
QUOTES_NOT_AVAILABLE_ERROR,
|
|
|
|
OFFLINE_FOR_MAINTENANCE,
|
|
|
|
]),
|
|
|
|
submittingSwap: PropTypes.bool,
|
|
|
|
inputValue: PropTypes.oneOfType([PropTypes.number, PropTypes.string]),
|
|
|
|
maxSlippage: PropTypes.number,
|
2021-02-04 19:15:23 +01:00
|
|
|
};
|