import React, { PureComponent } from 'react'; import { memoize } from 'lodash'; import PropTypes from 'prop-types'; import LedgerInstructionField from '../ledger-instruction-field'; import { sanitizeMessage, getURLHostName, getNetworkNameFromProviderType, ///: BEGIN:ONLY_INCLUDE_IN(build-mmi) shortenAddress, ///: END:ONLY_INCLUDE_IN } from '../../../helpers/utils/util'; import { MetaMetricsEventCategory } from '../../../../shared/constants/metametrics'; import SiteOrigin from '../../ui/site-origin'; import Button from '../../ui/button'; import Typography from '../../ui/typography/typography'; import ContractDetailsModal from '../modals/contract-details-modal/contract-details-modal'; import { TypographyVariant, FontWeight, TextAlign, TextColor, ///: BEGIN:ONLY_INCLUDE_IN(build-mmi) IconColor, DISPLAY, BLOCK_SIZES, TextVariant, BackgroundColor, ///: END:ONLY_INCLUDE_IN } from '../../../helpers/constants/design-system'; import NetworkAccountBalanceHeader from '../network-account-balance-header'; import { Numeric } from '../../../../shared/modules/Numeric'; import { EtherDenomination } from '../../../../shared/constants/common'; import ConfirmPageContainerNavigation from '../confirm-page-container/confirm-page-container-navigation'; import SecurityProviderBannerMessage from '../security-provider-banner-message/security-provider-banner-message'; import { SECURITY_PROVIDER_MESSAGE_SEVERITIES } from '../security-provider-banner-message/security-provider-banner-message.constants'; import { formatCurrency } from '../../../helpers/utils/confirm-tx.util'; import { getValueFromWeiHex } from '../../../../shared/modules/conversion.utils'; ///: BEGIN:ONLY_INCLUDE_IN(build-mmi) import { Icon, IconName, Text } from '../../component-library'; import Box from '../../ui/box/box'; ///: END:ONLY_INCLUDE_IN import Footer from './signature-request-footer'; import Message from './signature-request-message'; export default class SignatureRequest extends PureComponent { static propTypes = { /** * The display content of transaction data */ txData: PropTypes.object.isRequired, /** * The display content of sender account */ fromAccount: PropTypes.shape({ address: PropTypes.string.isRequired, balance: PropTypes.string, name: PropTypes.string, }).isRequired, /** * Check if the wallet is ledget wallet or not */ isLedgerWallet: PropTypes.bool, /** * Handler for cancel button */ cancel: PropTypes.func.isRequired, /** * Handler for sign button */ sign: PropTypes.func.isRequired, /** * Whether the hardware wallet requires a connection disables the sign button if true. */ hardwareWalletRequiresConnection: PropTypes.bool.isRequired, /** * Current network chainId */ chainId: PropTypes.string, /** * RPC prefs of the current network */ rpcPrefs: PropTypes.object, nativeCurrency: PropTypes.string, currentCurrency: PropTypes.string.isRequired, conversionRate: PropTypes.number, providerConfig: PropTypes.object, subjectMetadata: PropTypes.object, unapprovedMessagesCount: PropTypes.number, clearConfirmTransaction: PropTypes.func.isRequired, history: PropTypes.object, mostRecentOverviewPage: PropTypes.string, showRejectTransactionsConfirmationModal: PropTypes.func.isRequired, cancelAll: PropTypes.func.isRequired, ///: BEGIN:ONLY_INCLUDE_IN(build-mmi) showCustodianDeepLink: PropTypes.func, isNotification: PropTypes.bool, // Used to show a warning if the signing account is not the selected account // Largely relevant for contract wallet custodians selectedAccount: PropTypes.object, ///: END:ONLY_INCLUDE_IN }; static contextTypes = { t: PropTypes.func, trackEvent: PropTypes.func, }; state = { hasScrolledMessage: false, showContractDetails: false, }; ///: BEGIN:ONLY_INCLUDE_IN(build-mmi) componentDidMount() { if (this.props.txData.custodyId) { this.props.showCustodianDeepLink({ custodyId: this.props.txData.custodyId, fromAddress: this.props.fromAccount.address, closeNotification: this.props.isNotification, onDeepLinkFetched: () => undefined, onDeepLinkShown: () => { this.context.trackEvent({ category: 'MMI', event: 'Show deeplink for signature', }); }, }); } } ///: END:ONLY_INCLUDE_IN setMessageRootRef(ref) { this.messageRootRef = ref; } formatWallet(wallet) { return `${wallet.slice(0, 8)}...${wallet.slice( wallet.length - 8, wallet.length, )}`; } memoizedParseMessage = memoize((data) => { const { message, domain = {}, primaryType, types } = JSON.parse(data); const sanitizedMessage = sanitizeMessage(message, primaryType, types); return { sanitizedMessage, domain, primaryType }; }); handleCancelAll = () => { const { cancelAll, clearConfirmTransaction, history, mostRecentOverviewPage, showRejectTransactionsConfirmationModal, unapprovedMessagesCount, } = this.props; showRejectTransactionsConfirmationModal({ unapprovedTxCount: unapprovedMessagesCount, onSubmit: async () => { await cancelAll(); clearConfirmTransaction(); history.push(mostRecentOverviewPage); }, }); }; render() { const { providerConfig, txData: { msgParams: { data, origin, version }, type, }, fromAccount: { address, balance, name }, cancel, sign, isLedgerWallet, hardwareWalletRequiresConnection, chainId, rpcPrefs, txData, subjectMetadata, nativeCurrency, currentCurrency, conversionRate, unapprovedMessagesCount, } = this.props; const { t, trackEvent } = this.context; const { sanitizedMessage, domain: { verifyingContract }, primaryType, } = this.memoizedParseMessage(data); const rejectNText = t('rejectRequestsN', [unapprovedMessagesCount]); const networkName = getNetworkNameFromProviderType(providerConfig.type); const currentNetwork = networkName === '' ? providerConfig.nickname || t('unknownNetwork') : t(networkName); const balanceInBaseAsset = conversionRate ? formatCurrency( getValueFromWeiHex({ value: balance, fromCurrency: nativeCurrency, toCurrency: currentCurrency, conversionRate, numberOfDecimals: 6, toDenomination: EtherDenomination.ETH, }), currentCurrency, ) : new Numeric(balance, 16, EtherDenomination.WEI) .toDenomination(EtherDenomination.ETH) .round(6) .toBase(10) .toString(); const onSign = (event) => { sign(event); trackEvent({ category: MetaMetricsEventCategory.Transactions, event: 'Confirm', properties: { action: 'Sign Request', legacy_event: true, type, version, }, }); }; const onCancel = (event) => { cancel(event); trackEvent({ category: MetaMetricsEventCategory.Transactions, event: 'Cancel', properties: { action: 'Sign Request', legacy_event: true, type, version, }, }); }; const messageIsScrollable = this.messageRootRef?.scrollHeight > this.messageRootRef?.clientHeight; const targetSubjectMetadata = txData.msgParams.origin ? subjectMetadata?.[txData.msgParams.origin] : null; return (
{(txData?.securityProviderResponse?.flagAsDangerous !== undefined && txData?.securityProviderResponse?.flagAsDangerous !== SECURITY_PROVIDER_MESSAGE_SEVERITIES.NOT_MALICIOUS) || (txData?.securityProviderResponse && Object.keys(txData.securityProviderResponse).length === 0) ? ( ) : null} { ///: BEGIN:ONLY_INCLUDE_IN(build-mmi) this.props.selectedAccount.address === address ? null : ( {this.context.t('mismatchAccount', [ shortenAddress(this.props.selectedAccount.address), shortenAddress(address), ])} ) ///: END:ONLY_INCLUDE_IN }
{this.context.t('sigRequest')} {this.context.t('signatureRequestGuidance')} {verifyingContract ? (
) : null}
{isLedgerWallet ? (
) : null} this.setState({ hasScrolledMessage: true })} setMessageRootRef={this.setMessageRootRef.bind(this)} messageRootRef={this.messageRootRef} messageIsScrollable={messageIsScrollable} primaryType={primaryType} />
); } }