You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
369 lines
12 KiB
369 lines
12 KiB
import React, { useContext, useLayoutEffect, useRef, useState } from 'react';
|
|
import { useSelector } from 'react-redux';
|
|
import PropTypes from 'prop-types';
|
|
|
|
import BigNumber from 'bignumber.js';
|
|
import {
|
|
GAS_RECOMMENDATIONS,
|
|
EDIT_GAS_MODES,
|
|
GAS_ESTIMATE_TYPES,
|
|
CUSTOM_GAS_ESTIMATE,
|
|
} from '../../../../shared/constants/gas';
|
|
import { EVENT } from '../../../../shared/constants/metametrics';
|
|
|
|
import Button from '../../ui/button';
|
|
import Typography from '../../ui/typography/typography';
|
|
import {
|
|
getIsMainnet,
|
|
checkNetworkAndAccountSupports1559,
|
|
getAdvancedInlineGasShown,
|
|
} from '../../../selectors';
|
|
|
|
import {
|
|
COLORS,
|
|
TYPOGRAPHY,
|
|
FONT_WEIGHT,
|
|
} from '../../../helpers/constants/design-system';
|
|
import { areDappSuggestedAndTxParamGasFeesTheSame } from '../../../helpers/utils/confirm-tx.util';
|
|
import { isLegacyTransaction } from '../../../helpers/utils/transactions.util';
|
|
|
|
import InfoTooltip from '../../ui/info-tooltip';
|
|
import ErrorMessage from '../../ui/error-message';
|
|
import TransactionTotalBanner from '../transaction-total-banner/transaction-total-banner.component';
|
|
import RadioGroup from '../../ui/radio-group/radio-group.component';
|
|
import AdvancedGasControls from '../advanced-gas-controls/advanced-gas-controls.component';
|
|
import ActionableMessage from '../../ui/actionable-message/actionable-message';
|
|
|
|
import { I18nContext } from '../../../contexts/i18n';
|
|
import GasTiming from '../gas-timing';
|
|
import { MetaMetricsContext } from '../../../contexts/metametrics';
|
|
|
|
export default function EditGasDisplay({
|
|
mode = EDIT_GAS_MODES.MODIFY_IN_PLACE,
|
|
showEducationButton = false,
|
|
onEducationClick,
|
|
transaction,
|
|
defaultEstimateToUse,
|
|
maxPriorityFeePerGas,
|
|
setMaxPriorityFeePerGas,
|
|
maxPriorityFeePerGasFiat,
|
|
maxFeePerGas,
|
|
setMaxFeePerGas,
|
|
maxFeePerGasFiat,
|
|
estimatedMaximumNative,
|
|
estimatedMinimumNative,
|
|
isGasEstimatesLoading,
|
|
gasEstimateType,
|
|
gasPrice,
|
|
setGasPrice,
|
|
gasLimit,
|
|
setGasLimit,
|
|
properGasLimit,
|
|
estimateToUse,
|
|
setEstimateToUse,
|
|
estimatedMinimumFiat,
|
|
estimatedMaximumFiat,
|
|
dappSuggestedGasFeeAcknowledged,
|
|
setDappSuggestedGasFeeAcknowledged,
|
|
gasErrors,
|
|
gasWarnings,
|
|
onManualChange,
|
|
minimumGasLimit,
|
|
balanceError,
|
|
estimatesUnavailableWarning,
|
|
hasGasErrors,
|
|
txParamsHaveBeenCustomized,
|
|
isNetworkBusy,
|
|
}) {
|
|
const t = useContext(I18nContext);
|
|
const scrollRef = useRef(null);
|
|
|
|
const isMainnet = useSelector(getIsMainnet);
|
|
const supportsEIP1559 =
|
|
useSelector(checkNetworkAndAccountSupports1559) &&
|
|
!isLegacyTransaction(transaction?.txParams);
|
|
const showAdvancedInlineGasIfPossible = useSelector(
|
|
getAdvancedInlineGasShown,
|
|
);
|
|
|
|
const [showAdvancedForm, setShowAdvancedForm] = useState(
|
|
!estimateToUse || estimateToUse === CUSTOM_GAS_ESTIMATE || !supportsEIP1559,
|
|
);
|
|
const [hideRadioButtons, setHideRadioButtons] = useState(
|
|
showAdvancedInlineGasIfPossible,
|
|
);
|
|
|
|
useLayoutEffect(() => {
|
|
if (showAdvancedForm && scrollRef.current) {
|
|
scrollRef.current.scrollIntoView?.();
|
|
}
|
|
}, [showAdvancedForm]);
|
|
|
|
const dappSuggestedAndTxParamGasFeesAreTheSame =
|
|
areDappSuggestedAndTxParamGasFeesTheSame(transaction);
|
|
|
|
const requireDappAcknowledgement = Boolean(
|
|
transaction?.dappSuggestedGasFees &&
|
|
!dappSuggestedGasFeeAcknowledged &&
|
|
dappSuggestedAndTxParamGasFeesAreTheSame,
|
|
);
|
|
|
|
let warningMessage;
|
|
if (
|
|
gasLimit !== undefined &&
|
|
properGasLimit !== undefined &&
|
|
new BigNumber(gasLimit).lessThan(new BigNumber(properGasLimit))
|
|
) {
|
|
warningMessage = t('gasLimitRecommended', [properGasLimit]);
|
|
}
|
|
|
|
const showTopError =
|
|
(balanceError || estimatesUnavailableWarning) &&
|
|
(!isGasEstimatesLoading || txParamsHaveBeenCustomized);
|
|
const radioButtonsEnabled =
|
|
supportsEIP1559 &&
|
|
gasEstimateType === GAS_ESTIMATE_TYPES.FEE_MARKET &&
|
|
!requireDappAcknowledgement;
|
|
|
|
let errorKey;
|
|
if (balanceError) {
|
|
errorKey = 'insufficientFunds';
|
|
} else if (estimatesUnavailableWarning) {
|
|
errorKey = 'gasEstimatesUnavailableWarning';
|
|
}
|
|
|
|
const trackEvent = useContext(MetaMetricsContext);
|
|
return (
|
|
<div className="edit-gas-display">
|
|
<div className="edit-gas-display__content">
|
|
{showTopError && (
|
|
<div className="edit-gas-display__warning">
|
|
<ErrorMessage errorKey={errorKey} />
|
|
</div>
|
|
)}
|
|
{warningMessage && (
|
|
<div className="edit-gas-display__warning">
|
|
<ActionableMessage
|
|
className="actionable-message--warning"
|
|
message={warningMessage}
|
|
iconFillColor="var(--color-warning-default)"
|
|
useIcon
|
|
/>
|
|
</div>
|
|
)}
|
|
{requireDappAcknowledgement && !isGasEstimatesLoading && (
|
|
<div className="edit-gas-display__dapp-acknowledgement-warning">
|
|
<ActionableMessage
|
|
className="actionable-message--warning"
|
|
message={t('gasDisplayDappWarning', [transaction.origin])}
|
|
iconFillColor="var(--color-warning-default)"
|
|
useIcon
|
|
/>
|
|
</div>
|
|
)}
|
|
{isNetworkBusy ? (
|
|
<div className="edit-gas-display__warning">
|
|
<ActionableMessage
|
|
className="actionable-message--warning"
|
|
message={t('networkIsBusy')}
|
|
iconFillColor="var(--color-warning-default)"
|
|
useIcon
|
|
/>
|
|
</div>
|
|
) : null}
|
|
{mode === EDIT_GAS_MODES.SPEED_UP && (
|
|
<div className="edit-gas-display__top-tooltip">
|
|
<Typography
|
|
color={COLORS.TEXT_DEFAULT}
|
|
variant={TYPOGRAPHY.H8}
|
|
fontWeight={FONT_WEIGHT.BOLD}
|
|
>
|
|
{t('speedUpTooltipText')}{' '}
|
|
<InfoTooltip
|
|
position="top"
|
|
contentText={t('speedUpExplanation')}
|
|
/>
|
|
</Typography>
|
|
</div>
|
|
)}
|
|
<TransactionTotalBanner
|
|
total={
|
|
(supportsEIP1559 || isMainnet) && estimatedMinimumFiat
|
|
? `~ ${estimatedMinimumFiat}`
|
|
: estimatedMinimumNative
|
|
}
|
|
detail={
|
|
supportsEIP1559 &&
|
|
estimatedMaximumFiat !== undefined && (
|
|
<>
|
|
<Typography as="span" key="label" fontWeight={FONT_WEIGHT.BOLD}>
|
|
{t('editGasSubTextFeeLabel')}
|
|
</Typography>
|
|
<Typography as="span" key="secondary">
|
|
{estimatedMaximumFiat}
|
|
</Typography>
|
|
<Typography as="span" key="primary">
|
|
{`(${estimatedMaximumNative})`}
|
|
</Typography>
|
|
</>
|
|
)
|
|
}
|
|
timing={
|
|
hasGasErrors === false &&
|
|
supportsEIP1559 && (
|
|
<GasTiming
|
|
maxFeePerGas={maxFeePerGas.toString()}
|
|
maxPriorityFeePerGas={maxPriorityFeePerGas.toString()}
|
|
gasWarnings={gasWarnings}
|
|
/>
|
|
)
|
|
}
|
|
/>
|
|
{requireDappAcknowledgement && (
|
|
<Button
|
|
className="edit-gas-display__dapp-acknowledgement-button"
|
|
onClick={() => setDappSuggestedGasFeeAcknowledged(true)}
|
|
>
|
|
{t('gasDisplayAcknowledgeDappButtonText')}
|
|
</Button>
|
|
)}
|
|
{!requireDappAcknowledgement &&
|
|
radioButtonsEnabled &&
|
|
showAdvancedInlineGasIfPossible && (
|
|
<button
|
|
className="edit-gas-display__advanced-button"
|
|
onClick={() => setHideRadioButtons(!hideRadioButtons)}
|
|
>
|
|
{t('showRecommendations')}{' '}
|
|
{hideRadioButtons ? (
|
|
<i className="fa fa-caret-down"></i>
|
|
) : (
|
|
<i className="fa fa-caret-up"></i>
|
|
)}
|
|
</button>
|
|
)}
|
|
{radioButtonsEnabled && !hideRadioButtons && (
|
|
<RadioGroup
|
|
dataTestId="gas-recommendation"
|
|
name="gas-recommendation"
|
|
options={[
|
|
{
|
|
value: GAS_RECOMMENDATIONS.LOW,
|
|
label: t('editGasLow'),
|
|
recommended: defaultEstimateToUse === GAS_RECOMMENDATIONS.LOW,
|
|
},
|
|
{
|
|
value: GAS_RECOMMENDATIONS.MEDIUM,
|
|
label: t('editGasMedium'),
|
|
recommended:
|
|
defaultEstimateToUse === GAS_RECOMMENDATIONS.MEDIUM,
|
|
},
|
|
{
|
|
value: GAS_RECOMMENDATIONS.HIGH,
|
|
label: t('editGasHigh'),
|
|
recommended: defaultEstimateToUse === GAS_RECOMMENDATIONS.HIGH,
|
|
},
|
|
]}
|
|
selectedValue={estimateToUse}
|
|
onChange={setEstimateToUse}
|
|
/>
|
|
)}
|
|
{!requireDappAcknowledgement &&
|
|
radioButtonsEnabled &&
|
|
!showAdvancedInlineGasIfPossible && (
|
|
<button
|
|
className="edit-gas-display__advanced-button"
|
|
onClick={() => {
|
|
setShowAdvancedForm(!showAdvancedForm);
|
|
trackEvent({
|
|
event: 'Clicked "Advanced options"',
|
|
category: EVENT.CATEGORIES.TRANSACTIONS,
|
|
properties: {
|
|
action: 'Edit Screen',
|
|
legacy_event: true,
|
|
},
|
|
});
|
|
}}
|
|
>
|
|
{t('advancedOptions')}{' '}
|
|
{showAdvancedForm ? (
|
|
<i className="fa fa-caret-up"></i>
|
|
) : (
|
|
<i className="fa fa-caret-down"></i>
|
|
)}
|
|
</button>
|
|
)}
|
|
{!requireDappAcknowledgement &&
|
|
(showAdvancedForm ||
|
|
hasGasErrors ||
|
|
estimatesUnavailableWarning ||
|
|
showAdvancedInlineGasIfPossible) && (
|
|
<AdvancedGasControls
|
|
gasEstimateType={gasEstimateType}
|
|
isGasEstimatesLoading={isGasEstimatesLoading}
|
|
gasLimit={gasLimit}
|
|
setGasLimit={setGasLimit}
|
|
maxPriorityFee={maxPriorityFeePerGas}
|
|
setMaxPriorityFee={setMaxPriorityFeePerGas}
|
|
maxFee={maxFeePerGas}
|
|
setMaxFee={setMaxFeePerGas}
|
|
gasPrice={gasPrice}
|
|
setGasPrice={setGasPrice}
|
|
maxPriorityFeeFiat={maxPriorityFeePerGasFiat}
|
|
maxFeeFiat={maxFeePerGasFiat}
|
|
gasErrors={gasErrors}
|
|
onManualChange={onManualChange}
|
|
minimumGasLimit={minimumGasLimit}
|
|
supportsEIP1559={supportsEIP1559}
|
|
/>
|
|
)}
|
|
</div>
|
|
{supportsEIP1559 && !requireDappAcknowledgement && showEducationButton && (
|
|
<div className="edit-gas-display__education">
|
|
<button onClick={onEducationClick}>
|
|
{t('editGasEducationButtonText')}
|
|
</button>
|
|
</div>
|
|
)}
|
|
<div ref={scrollRef} className="edit-gas-display__scroll-bottom" />
|
|
</div>
|
|
);
|
|
}
|
|
|
|
EditGasDisplay.propTypes = {
|
|
mode: PropTypes.oneOf(Object.values(EDIT_GAS_MODES)),
|
|
showEducationButton: PropTypes.bool,
|
|
onEducationClick: PropTypes.func,
|
|
defaultEstimateToUse: PropTypes.oneOf(Object.values(GAS_RECOMMENDATIONS)),
|
|
maxPriorityFeePerGas: PropTypes.string,
|
|
setMaxPriorityFeePerGas: PropTypes.func,
|
|
maxPriorityFeePerGasFiat: PropTypes.string,
|
|
maxFeePerGas: PropTypes.string,
|
|
setMaxFeePerGas: PropTypes.func,
|
|
maxFeePerGasFiat: PropTypes.string,
|
|
estimatedMaximumNative: PropTypes.string,
|
|
estimatedMinimumNative: PropTypes.string,
|
|
isGasEstimatesLoading: PropTypes.bool,
|
|
gasEstimateType: PropTypes.string,
|
|
gasPrice: PropTypes.string,
|
|
setGasPrice: PropTypes.func,
|
|
gasLimit: PropTypes.number,
|
|
setGasLimit: PropTypes.func,
|
|
properGasLimit: PropTypes.number,
|
|
estimateToUse: PropTypes.string,
|
|
setEstimateToUse: PropTypes.func,
|
|
estimatedMinimumFiat: PropTypes.string,
|
|
estimatedMaximumFiat: PropTypes.string,
|
|
dappSuggestedGasFeeAcknowledged: PropTypes.bool,
|
|
setDappSuggestedGasFeeAcknowledged: PropTypes.func,
|
|
transaction: PropTypes.object,
|
|
gasErrors: PropTypes.object,
|
|
gasWarnings: PropTypes.object,
|
|
onManualChange: PropTypes.func,
|
|
minimumGasLimit: PropTypes.string,
|
|
balanceError: PropTypes.bool,
|
|
estimatesUnavailableWarning: PropTypes.bool,
|
|
hasGasErrors: PropTypes.bool,
|
|
txParamsHaveBeenCustomized: PropTypes.bool,
|
|
isNetworkBusy: PropTypes.bool,
|
|
};
|
|
|