2023-01-25 20:25:44 +00:00
|
|
|
import type { ReactNode } from 'react';
|
2023-01-19 12:10:52 +00:00
|
|
|
import { useAssetsDataProvider } from '@vegaprotocol/assets';
|
|
|
|
import { ETHERSCAN_TX, useEtherscanLink } from '@vegaprotocol/environment';
|
2023-02-28 18:56:29 +00:00
|
|
|
import { formatNumber, toBigNum } from '@vegaprotocol/utils';
|
|
|
|
import { t } from '@vegaprotocol/i18n';
|
2023-01-19 12:10:52 +00:00
|
|
|
import type { Toast, ToastContent } from '@vegaprotocol/ui-toolkit';
|
2023-02-06 20:09:56 +00:00
|
|
|
import { ToastHeading } from '@vegaprotocol/ui-toolkit';
|
|
|
|
import { Panel } from '@vegaprotocol/ui-toolkit';
|
|
|
|
import { CLOSE_AFTER } from '@vegaprotocol/ui-toolkit';
|
|
|
|
import { useToasts } from '@vegaprotocol/ui-toolkit';
|
2023-01-19 12:10:52 +00:00
|
|
|
import { ExternalLink, Intent, ProgressBar } from '@vegaprotocol/ui-toolkit';
|
2023-02-06 20:09:56 +00:00
|
|
|
import { useCallback } from 'react';
|
2023-01-19 12:10:52 +00:00
|
|
|
import compact from 'lodash/compact';
|
|
|
|
import type { EthStoredTxState } from '@vegaprotocol/web3';
|
|
|
|
import {
|
|
|
|
EthTxStatus,
|
|
|
|
isEthereumError,
|
|
|
|
TransactionContent,
|
|
|
|
useEthTransactionStore,
|
|
|
|
} from '@vegaprotocol/web3';
|
|
|
|
|
|
|
|
const intentMap: { [s in EthTxStatus]: Intent } = {
|
|
|
|
Default: Intent.Primary,
|
|
|
|
Requested: Intent.Warning,
|
|
|
|
Pending: Intent.Warning,
|
|
|
|
Error: Intent.Danger,
|
2023-01-25 20:25:44 +00:00
|
|
|
Complete: Intent.Warning,
|
2023-01-19 12:10:52 +00:00
|
|
|
Confirmed: Intent.Success,
|
|
|
|
};
|
|
|
|
|
2023-01-25 20:25:44 +00:00
|
|
|
const isWithdrawTransaction = (tx: EthStoredTxState) =>
|
|
|
|
tx.methodName === 'withdraw_asset';
|
2023-01-19 12:10:52 +00:00
|
|
|
|
2023-01-25 20:25:44 +00:00
|
|
|
const isDepositTransaction = (tx: EthStoredTxState) =>
|
2023-01-26 12:00:00 +00:00
|
|
|
tx.methodName === 'deposit_asset';
|
2023-01-19 12:10:52 +00:00
|
|
|
|
2023-01-25 20:25:44 +00:00
|
|
|
const EthTransactionDetails = ({ tx }: { tx: EthStoredTxState }) => {
|
|
|
|
const { data: assets } = useAssetsDataProvider();
|
|
|
|
if (!assets) return null;
|
2023-01-19 12:10:52 +00:00
|
|
|
|
2023-01-25 20:25:44 +00:00
|
|
|
const isWithdraw = isWithdrawTransaction(tx);
|
|
|
|
const isDeposit = isDepositTransaction(tx);
|
2023-01-19 12:10:52 +00:00
|
|
|
|
2023-01-25 20:25:44 +00:00
|
|
|
let assetInfo: ReactNode;
|
|
|
|
if ((isWithdraw || isDeposit) && tx.args.length > 2 && tx.assetId) {
|
|
|
|
const asset = assets.find((a) => a.id === tx.assetId);
|
2023-01-19 12:10:52 +00:00
|
|
|
if (asset) {
|
2023-01-25 20:25:44 +00:00
|
|
|
let label = '';
|
|
|
|
if (isWithdraw) label = t('Withdraw');
|
|
|
|
if (isDeposit) label = t('Deposit');
|
|
|
|
assetInfo = (
|
2023-02-06 20:09:56 +00:00
|
|
|
<strong>
|
|
|
|
{label}{' '}
|
|
|
|
{formatNumber(toBigNum(tx.args[1], asset.decimals), asset.decimals)}{' '}
|
|
|
|
{asset.symbol}
|
|
|
|
</strong>
|
2023-01-19 12:10:52 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-06 20:09:56 +00:00
|
|
|
if (assetInfo || tx.requiresConfirmation) {
|
|
|
|
return (
|
|
|
|
<Panel>
|
|
|
|
{assetInfo}
|
|
|
|
{tx.status === EthTxStatus.Pending && (
|
|
|
|
<>
|
|
|
|
<p className="mt-[2px]">
|
|
|
|
{t('Awaiting confirmations')}{' '}
|
|
|
|
{`(${tx.confirmations}/${tx.requiredConfirmations})`}
|
|
|
|
</p>
|
|
|
|
<ProgressBar
|
|
|
|
value={(tx.confirmations / tx.requiredConfirmations) * 100}
|
|
|
|
/>
|
|
|
|
</>
|
|
|
|
)}
|
|
|
|
</Panel>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
2023-01-19 12:10:52 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
type EthTxToastContentProps = {
|
|
|
|
tx: EthStoredTxState;
|
|
|
|
};
|
|
|
|
|
|
|
|
const EthTxRequestedToastContent = ({ tx }: EthTxToastContentProps) => {
|
|
|
|
return (
|
2023-02-06 20:09:56 +00:00
|
|
|
<>
|
|
|
|
<ToastHeading>{t('Action required')}</ToastHeading>
|
2023-01-19 12:10:52 +00:00
|
|
|
<p>
|
|
|
|
{t(
|
|
|
|
'Please go to your wallet application and approve or reject the transaction.'
|
|
|
|
)}
|
|
|
|
</p>
|
|
|
|
<EthTransactionDetails tx={tx} />
|
2023-02-06 20:09:56 +00:00
|
|
|
</>
|
2023-01-19 12:10:52 +00:00
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
const EthTxPendingToastContent = ({ tx }: EthTxToastContentProps) => {
|
|
|
|
return (
|
2023-02-06 20:09:56 +00:00
|
|
|
<>
|
|
|
|
<ToastHeading>{t('Awaiting confirmation')}</ToastHeading>
|
2023-01-27 08:39:39 +00:00
|
|
|
<p>{t('Please wait for your transaction to be confirmed.')}</p>
|
2023-01-25 20:25:44 +00:00
|
|
|
<EtherscanLink tx={tx} />
|
2023-01-19 12:10:52 +00:00
|
|
|
<EthTransactionDetails tx={tx} />
|
2023-02-06 20:09:56 +00:00
|
|
|
</>
|
2023-01-19 12:10:52 +00:00
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
const EthTxErrorToastContent = ({ tx }: EthTxToastContentProps) => {
|
|
|
|
let errorMessage = '';
|
|
|
|
|
|
|
|
if (isEthereumError(tx.error)) {
|
|
|
|
errorMessage = tx.error.reason;
|
|
|
|
} else if (tx.error instanceof Error) {
|
|
|
|
errorMessage = tx.error.message;
|
|
|
|
}
|
|
|
|
return (
|
2023-02-06 20:09:56 +00:00
|
|
|
<>
|
|
|
|
<ToastHeading>{t('Error occurred')}</ToastHeading>
|
|
|
|
<p className="first-letter:uppercase">{errorMessage}</p>
|
2023-01-19 12:10:52 +00:00
|
|
|
<EthTransactionDetails tx={tx} />
|
2023-02-06 20:09:56 +00:00
|
|
|
</>
|
2023-01-19 12:10:52 +00:00
|
|
|
);
|
|
|
|
};
|
|
|
|
|
2023-01-25 20:25:44 +00:00
|
|
|
const EtherscanLink = ({ tx }: EthTxToastContentProps) => {
|
2023-01-19 12:10:52 +00:00
|
|
|
const etherscanLink = useEtherscanLink();
|
2023-01-25 20:25:44 +00:00
|
|
|
return tx.txHash ? (
|
|
|
|
<p className="break-all">
|
|
|
|
<ExternalLink
|
|
|
|
href={etherscanLink(ETHERSCAN_TX.replace(':hash', tx.txHash))}
|
|
|
|
rel="noreferrer"
|
|
|
|
>
|
|
|
|
{t('View on Etherscan')}
|
|
|
|
</ExternalLink>
|
|
|
|
</p>
|
|
|
|
) : null;
|
|
|
|
};
|
|
|
|
|
|
|
|
const EthTxConfirmedToastContent = ({ tx }: EthTxToastContentProps) => {
|
2023-01-19 12:10:52 +00:00
|
|
|
return (
|
2023-02-06 20:09:56 +00:00
|
|
|
<>
|
|
|
|
<ToastHeading>{t('Transaction confirmed')}</ToastHeading>
|
2023-01-27 08:39:39 +00:00
|
|
|
<p>{t('Your transaction has been confirmed.')}</p>
|
2023-01-25 20:25:44 +00:00
|
|
|
<EtherscanLink tx={tx} />
|
|
|
|
<EthTransactionDetails tx={tx} />
|
2023-02-06 20:09:56 +00:00
|
|
|
</>
|
2023-01-25 20:25:44 +00:00
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
const EthTxCompletedToastContent = ({ tx }: EthTxToastContentProps) => {
|
|
|
|
const isDeposit = isDepositTransaction(tx);
|
|
|
|
return (
|
2023-02-06 20:09:56 +00:00
|
|
|
<>
|
|
|
|
<ToastHeading>
|
2023-01-26 12:00:00 +00:00
|
|
|
{t('Processing')} {isDeposit && t('deposit')}
|
2023-02-06 20:09:56 +00:00
|
|
|
</ToastHeading>
|
2023-01-25 20:25:44 +00:00
|
|
|
<p>
|
2023-01-27 08:39:39 +00:00
|
|
|
{t('Your transaction has been completed.')}{' '}
|
2023-01-26 12:00:00 +00:00
|
|
|
{isDeposit && t('Waiting for deposit confirmation.')}
|
2023-01-25 20:25:44 +00:00
|
|
|
</p>
|
|
|
|
<EtherscanLink tx={tx} />
|
2023-01-19 12:10:52 +00:00
|
|
|
<EthTransactionDetails tx={tx} />
|
2023-02-06 20:09:56 +00:00
|
|
|
</>
|
2023-01-19 12:10:52 +00:00
|
|
|
);
|
|
|
|
};
|
|
|
|
|
2023-02-06 20:09:56 +00:00
|
|
|
const isFinal = (tx: EthStoredTxState) =>
|
|
|
|
[EthTxStatus.Confirmed, EthTxStatus.Error].includes(tx.status);
|
|
|
|
|
2023-01-19 12:10:52 +00:00
|
|
|
export const useEthereumTransactionToasts = () => {
|
2023-02-06 20:09:56 +00:00
|
|
|
const [setToast, removeToast] = useToasts((store) => [
|
|
|
|
store.setToast,
|
|
|
|
store.remove,
|
|
|
|
]);
|
|
|
|
|
2023-03-07 08:17:02 +00:00
|
|
|
const dismissTx = useEthTransactionStore((state) => state.dismiss);
|
2023-02-06 20:09:56 +00:00
|
|
|
|
|
|
|
const onClose = useCallback(
|
|
|
|
(tx: EthStoredTxState) => () => {
|
2023-03-07 08:17:02 +00:00
|
|
|
dismissTx(tx.id);
|
2023-02-06 20:09:56 +00:00
|
|
|
removeToast(`eth-${tx.id}`);
|
|
|
|
},
|
2023-03-07 08:17:02 +00:00
|
|
|
[dismissTx, removeToast]
|
2023-01-19 12:10:52 +00:00
|
|
|
);
|
2023-02-06 20:09:56 +00:00
|
|
|
|
2023-01-19 12:10:52 +00:00
|
|
|
const fromEthTransaction = useCallback(
|
|
|
|
(tx: EthStoredTxState): Toast => {
|
|
|
|
let content: ToastContent = <TransactionContent {...tx} />;
|
2023-02-06 20:09:56 +00:00
|
|
|
const closeAfter = isFinal(tx) ? CLOSE_AFTER : undefined;
|
2023-01-19 12:10:52 +00:00
|
|
|
if (tx.status === EthTxStatus.Requested) {
|
|
|
|
content = <EthTxRequestedToastContent tx={tx} />;
|
|
|
|
}
|
|
|
|
if (tx.status === EthTxStatus.Pending) {
|
|
|
|
content = <EthTxPendingToastContent tx={tx} />;
|
|
|
|
}
|
2023-01-25 20:25:44 +00:00
|
|
|
if (tx.status === EthTxStatus.Complete) {
|
|
|
|
content = <EthTxCompletedToastContent tx={tx} />;
|
|
|
|
}
|
|
|
|
if (tx.status === EthTxStatus.Confirmed) {
|
2023-01-19 12:10:52 +00:00
|
|
|
content = <EthTxConfirmedToastContent tx={tx} />;
|
|
|
|
}
|
|
|
|
if (tx.status === EthTxStatus.Error) {
|
|
|
|
content = <EthTxErrorToastContent tx={tx} />;
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
id: `eth-${tx.id}`,
|
|
|
|
intent: intentMap[tx.status],
|
2023-02-06 20:09:56 +00:00
|
|
|
onClose: onClose(tx),
|
2023-01-25 20:25:44 +00:00
|
|
|
loader: [EthTxStatus.Pending, EthTxStatus.Complete].includes(tx.status),
|
2023-01-19 12:10:52 +00:00
|
|
|
content,
|
2023-02-06 20:09:56 +00:00
|
|
|
closeAfter,
|
2023-01-19 12:10:52 +00:00
|
|
|
};
|
|
|
|
},
|
2023-02-06 20:09:56 +00:00
|
|
|
[onClose]
|
2023-01-19 12:10:52 +00:00
|
|
|
);
|
|
|
|
|
2023-02-06 20:09:56 +00:00
|
|
|
useEthTransactionStore.subscribe(
|
|
|
|
(state) => compact(state.transactions.filter((tx) => tx?.dialogOpen)),
|
|
|
|
(txs) => {
|
|
|
|
txs.forEach((tx) => {
|
|
|
|
setToast(fromEthTransaction(tx));
|
|
|
|
});
|
|
|
|
}
|
|
|
|
);
|
2023-01-19 12:10:52 +00:00
|
|
|
};
|