mirror of
https://github.com/kremalicious/metamask-extension.git
synced 2024-12-23 09:52:26 +01:00
228 lines
6.0 KiB
JavaScript
228 lines
6.0 KiB
JavaScript
import PropTypes from 'prop-types';
|
|
import React, { useEffect, useState } from 'react';
|
|
import { useDispatch, useSelector } from 'react-redux';
|
|
import { withRouter } from 'react-router-dom';
|
|
import log from 'loglevel';
|
|
import * as actions from '../../store/actions';
|
|
import txHelper from '../../helpers/utils/tx-helper';
|
|
import SignatureRequest from '../../components/app/signature-request';
|
|
import SignatureRequestSIWE from '../../components/app/signature-request-siwe';
|
|
import SignatureRequestOriginal from '../../components/app/signature-request-original';
|
|
import Loading from '../../components/ui/loading-screen';
|
|
import { useRouting } from '../../hooks/useRouting';
|
|
import { getTotalUnapprovedSignatureRequestCount } from '../../selectors';
|
|
import { MESSAGE_TYPE } from '../../../shared/constants/app';
|
|
import { TransactionStatus } from '../../../shared/constants/transaction';
|
|
import { getSendTo } from '../../ducks/send';
|
|
|
|
const SIGN_MESSAGE_TYPE = {
|
|
MESSAGE: 'message',
|
|
PERSONAL: 'personal',
|
|
TYPED: 'typed',
|
|
};
|
|
|
|
const signatureSelect = (txData) => {
|
|
const {
|
|
type,
|
|
msgParams: { version, siwe },
|
|
} = txData;
|
|
|
|
// Temporarily direct only v3 and v4 requests to new code.
|
|
if (
|
|
type === MESSAGE_TYPE.ETH_SIGN_TYPED_DATA &&
|
|
(version === 'V3' || version === 'V4')
|
|
) {
|
|
return SignatureRequest;
|
|
}
|
|
|
|
if (process.env.SIWE_V1 && siwe?.isSIWEMessage) {
|
|
return SignatureRequestSIWE;
|
|
}
|
|
|
|
return SignatureRequestOriginal;
|
|
};
|
|
|
|
const stopPropagation = (event) => {
|
|
if (event?.stopPropagation) {
|
|
event.stopPropagation();
|
|
}
|
|
};
|
|
|
|
const ConfirmTxScreen = ({ match }) => {
|
|
const dispatch = useDispatch();
|
|
const { navigateToMostRecentOverviewPage } = useRouting();
|
|
const unapprovedMessagesTotal = useSelector(
|
|
getTotalUnapprovedSignatureRequestCount,
|
|
);
|
|
const sendTo = useSelector(getSendTo);
|
|
const {
|
|
unapprovedTxs,
|
|
identities,
|
|
currentNetworkTxList,
|
|
currentCurrency,
|
|
unapprovedMsgs,
|
|
unapprovedPersonalMsgs,
|
|
unapprovedTypedMessages,
|
|
network,
|
|
blockGasLimit,
|
|
provider: { chainId },
|
|
} = useSelector((state) => state.metamask);
|
|
const { txId: index } = useSelector((state) => state.appState);
|
|
const [prevValue, setPrevValues] = useState();
|
|
|
|
useEffect(() => {
|
|
const unconfTxList = txHelper(
|
|
unapprovedTxs || {},
|
|
{},
|
|
{},
|
|
{},
|
|
network,
|
|
chainId,
|
|
);
|
|
if (unconfTxList.length === 0 && !sendTo && unapprovedMessagesTotal === 0) {
|
|
navigateToMostRecentOverviewPage();
|
|
}
|
|
}, []);
|
|
|
|
useEffect(() => {
|
|
if (!prevValue) {
|
|
setPrevValues({ index, unapprovedTxs });
|
|
return;
|
|
}
|
|
|
|
let prevTx;
|
|
const { params: { id: transactionId } = {} } = match;
|
|
if (transactionId) {
|
|
prevTx = currentNetworkTxList.find(({ id }) => `${id}` === transactionId);
|
|
} else {
|
|
const { index: prevIndex, unapprovedTxs: prevUnapprovedTxs } = prevValue;
|
|
const prevUnconfTxList = txHelper(
|
|
prevUnapprovedTxs,
|
|
{},
|
|
{},
|
|
{},
|
|
network,
|
|
chainId,
|
|
);
|
|
const prevTxData = prevUnconfTxList[prevIndex] || {};
|
|
prevTx =
|
|
currentNetworkTxList.find(({ id }) => id === prevTxData.id) || {};
|
|
}
|
|
|
|
const unconfTxList = txHelper(
|
|
unapprovedTxs || {},
|
|
{},
|
|
{},
|
|
{},
|
|
network,
|
|
chainId,
|
|
);
|
|
|
|
if (prevTx && prevTx.status === TransactionStatus.dropped) {
|
|
dispatch(
|
|
actions.showModal({
|
|
name: 'TRANSACTION_CONFIRMED',
|
|
onSubmit: () => navigateToMostRecentOverviewPage(),
|
|
}),
|
|
);
|
|
return;
|
|
}
|
|
|
|
if (unconfTxList.length === 0 && !sendTo && unapprovedMessagesTotal === 0) {
|
|
navigateToMostRecentOverviewPage();
|
|
}
|
|
|
|
setPrevValues({ index, unapprovedTxs });
|
|
}, [
|
|
chainId,
|
|
currentNetworkTxList,
|
|
match,
|
|
network,
|
|
sendTo,
|
|
unapprovedMessagesTotal,
|
|
unapprovedTxs,
|
|
]);
|
|
|
|
const getTxData = () => {
|
|
const { params: { id: transactionId } = {} } = match;
|
|
|
|
const unconfTxList = txHelper(
|
|
unapprovedTxs || {},
|
|
unapprovedMsgs,
|
|
unapprovedPersonalMsgs,
|
|
unapprovedTypedMessages,
|
|
network,
|
|
chainId,
|
|
);
|
|
|
|
log.info(`rendering a combined ${unconfTxList.length} unconf msgs & txs`);
|
|
|
|
return transactionId
|
|
? unconfTxList.find(({ id }) => `${id}` === transactionId)
|
|
: unconfTxList[index];
|
|
};
|
|
|
|
const txData = getTxData() || {};
|
|
|
|
const { msgParams } = txData;
|
|
if (!msgParams) {
|
|
return <Loading />;
|
|
}
|
|
|
|
const signMessage = (type) => (event) => {
|
|
stopPropagation(event);
|
|
const params = txData.msgParams;
|
|
params.metamaskId = txData.id;
|
|
let action;
|
|
if (type === SIGN_MESSAGE_TYPE.MESSAGE) {
|
|
action = actions.signMsg;
|
|
} else if (type === SIGN_MESSAGE_TYPE.PERSONAL) {
|
|
action = actions.signPersonalMsg;
|
|
} else {
|
|
action = actions.signTypedMsg;
|
|
}
|
|
return dispatch(action?.(params));
|
|
};
|
|
|
|
const cancelMessage = (type) => (event) => {
|
|
stopPropagation(event);
|
|
let action;
|
|
if (type === SIGN_MESSAGE_TYPE.MESSAGE) {
|
|
action = actions.cancelMsg;
|
|
} else if (type === SIGN_MESSAGE_TYPE.PERSONAL) {
|
|
action = actions.cancelPersonalMsg;
|
|
} else {
|
|
action = actions.cancelTypedMsg;
|
|
}
|
|
return dispatch(action(txData));
|
|
};
|
|
|
|
const SigComponent = signatureSelect(txData);
|
|
|
|
return (
|
|
<SigComponent
|
|
txData={txData}
|
|
key={txData.id}
|
|
identities={identities}
|
|
currentCurrency={currentCurrency}
|
|
blockGasLimit={blockGasLimit}
|
|
signMessage={signMessage(SIGN_MESSAGE_TYPE.MESSAGE)}
|
|
signPersonalMessage={signMessage(SIGN_MESSAGE_TYPE.PERSONAL)}
|
|
signTypedMessage={signMessage(SIGN_MESSAGE_TYPE.TYPED)}
|
|
cancelMessage={cancelMessage(SIGN_MESSAGE_TYPE.MESSAGE)}
|
|
cancelPersonalMessage={cancelMessage(SIGN_MESSAGE_TYPE.PERSONAL)}
|
|
cancelTypedMessage={cancelMessage(SIGN_MESSAGE_TYPE.TYPED)}
|
|
/>
|
|
);
|
|
};
|
|
|
|
ConfirmTxScreen.propTypes = {
|
|
match: PropTypes.shape({
|
|
params: PropTypes.shape({
|
|
id: PropTypes.string,
|
|
}),
|
|
}),
|
|
};
|
|
|
|
export default withRouter(ConfirmTxScreen);
|