diff --git a/dapp/src/components/TransactionModal/ActiveStakingStep/DepositBTCModal.tsx b/dapp/src/components/TransactionModal/ActiveStakingStep/DepositBTCModal.tsx index 08926094e..6a0bf6e5e 100644 --- a/dapp/src/components/TransactionModal/ActiveStakingStep/DepositBTCModal.tsx +++ b/dapp/src/components/TransactionModal/ActiveStakingStep/DepositBTCModal.tsx @@ -1,8 +1,9 @@ -import React, { useCallback } from "react" +import React, { useCallback, useRef } from "react" import { useActionFlowPause, useActionFlowTokenAmount, useAppDispatch, + useCancelPromise, useDepositBTCTransaction, useInvalidateQueries, useStakeFlowContext, @@ -28,6 +29,12 @@ export default function DepositBTCModal() { queryKey: userKeys.balance(), }) + const sessionId = useRef(Math.random()) + const { cancel, resolve, sessionIdToPromise } = useCancelPromise( + sessionId.current, + "Deposit cancelled", + ) + const onStakeBTCSuccess = useCallback(() => { handleBitcoinBalanceInvalidation() dispatch(setStatus(PROCESS_STATUSES.SUCCEEDED)) @@ -58,13 +65,15 @@ export default function DepositBTCModal() { const onDepositBTCError = useCallback( (error: unknown) => { + if (!sessionIdToPromise[sessionId.current].shouldOpenErrorModal) return + if (eip1193.didUserRejectRequest(error)) { handlePause() } else { onError(error) } }, - [onError, handlePause], + [sessionIdToPromise, handlePause, onError], ) const { mutate: sendBitcoinTransaction, status } = useDepositBTCTransaction({ @@ -79,6 +88,8 @@ export default function DepositBTCModal() { btcAddress, ) + await resolve() + if (verificationStatus === "valid") { sendBitcoinTransaction({ recipient: btcAddress, @@ -92,6 +103,7 @@ export default function DepositBTCModal() { btcAddress, depositReceipt, verifyDepositAddress, + resolve, sendBitcoinTransaction, onError, ]) @@ -105,5 +117,5 @@ export default function DepositBTCModal() { if (status === "pending" || status === "success") return - return + return } diff --git a/dapp/src/components/TransactionModal/ActiveUnstakingStep/SignMessageModal.tsx b/dapp/src/components/TransactionModal/ActiveUnstakingStep/SignMessageModal.tsx index 915a9436a..877f83264 100644 --- a/dapp/src/components/TransactionModal/ActiveUnstakingStep/SignMessageModal.tsx +++ b/dapp/src/components/TransactionModal/ActiveUnstakingStep/SignMessageModal.tsx @@ -1,8 +1,9 @@ -import React, { useCallback, useEffect, useRef, useState } from "react" +import React, { useCallback, useRef, useState } from "react" import { useActionFlowPause, useActionFlowTokenAmount, useAppDispatch, + useCancelPromise, useInvalidateQueries, useModal, useTimeout, @@ -22,15 +23,6 @@ const { userKeys } = queryKeysFactory type WithdrawalStatus = "building-data" | "built-data" | "signature" -const sessionIdToPromise: Record< - number, - { - promise: Promise - cancel: (reason: Error) => void - shouldOpenErrorModal: boolean - } -> = {} - export default function SignMessageModal() { const [status, setWaitingStatus] = useState("building-data") @@ -44,24 +36,15 @@ export default function SignMessageModal() { queryKey: userKeys.position(), }) const sessionId = useRef(Math.random()) + const { cancel, resolve, sessionIdToPromise } = useCancelPromise( + sessionId.current, + "Withdrawal cancelled", + ) const { transactionFee } = useTransactionDetails( amount, ACTION_FLOW_TYPES.UNSTAKE, ) - useEffect(() => { - let cancel = (_: Error) => {} - const promise: Promise = new Promise((_, reject) => { - cancel = reject - }) - - sessionIdToPromise[sessionId.current] = { - cancel, - promise, - shouldOpenErrorModal: true, - } - }, []) - const dataBuiltStepCallback = useCallback(() => { setWaitingStatus("built-data") return Promise.resolve() @@ -69,11 +52,8 @@ export default function SignMessageModal() { const onSignMessageCallback = useCallback(async () => { setWaitingStatus("signature") - return Promise.race([ - sessionIdToPromise[sessionId.current].promise, - Promise.resolve(), - ]) - }, []) + return resolve() + }, [resolve]) const onSignMessageSuccess = useCallback(() => { handleBitcoinPositionInvalidation() @@ -98,7 +78,7 @@ export default function SignMessageModal() { onSignMessageError(error) } }, - [onSignMessageError, handlePause], + [sessionIdToPromise, handlePause, onSignMessageError], ) const { mutate: handleSignMessage } = useMutation({ @@ -151,16 +131,7 @@ export default function SignMessageModal() { }) const onClose = () => { - const currentSessionId = sessionId.current - const sessionData = sessionIdToPromise[currentSessionId] - sessionIdToPromise[currentSessionId] = { - ...sessionData, - shouldOpenErrorModal: false, - } - - sessionIdToPromise[currentSessionId].cancel( - new Error("Withdrawal cancelled"), - ) + cancel() closeModal() } diff --git a/dapp/src/components/TransactionModal/WalletInteractionModal.tsx b/dapp/src/components/TransactionModal/WalletInteractionModal.tsx index 8011648f6..8c3926d1f 100644 --- a/dapp/src/components/TransactionModal/WalletInteractionModal.tsx +++ b/dapp/src/components/TransactionModal/WalletInteractionModal.tsx @@ -50,8 +50,10 @@ const DATA: Record< export default function WalletInteractionModal({ step, + onClose, }: { step: WalletInteractionStep + onClose?: () => void }) { const actionType = useActionFlowType() const connector = useConnector() @@ -60,7 +62,7 @@ export default function WalletInteractionModal({ return ( <> - {step === "opening-wallet" && } + {step === "opening-wallet" && } {header} diff --git a/dapp/src/hooks/index.ts b/dapp/src/hooks/index.ts index c0ebd7792..91b1cfeb6 100644 --- a/dapp/src/hooks/index.ts +++ b/dapp/src/hooks/index.ts @@ -38,3 +38,4 @@ export { default as useScrollbarVisibility } from "./useScrollbarVisibility" export { default as useAccessCode } from "./useAccessCode" export { default as useFormField } from "./useFormField" export { default as useDepositBTCTransaction } from "./useDepositBTCTransaction" +export { default as useCancelPromise } from "./useCancelPromise" diff --git a/dapp/src/hooks/useCancelPromise.ts b/dapp/src/hooks/useCancelPromise.ts new file mode 100644 index 000000000..a6bb1666b --- /dev/null +++ b/dapp/src/hooks/useCancelPromise.ts @@ -0,0 +1,50 @@ +import { useCallback, useEffect } from "react" + +const sessionIdToPromise: Record< + number, + { + promise: Promise + cancel: (reason: Error) => void + shouldOpenErrorModal: boolean + } +> = {} + +export default function useCancelPromise( + sessionId: number, + errorMsgText: string, +) { + useEffect(() => { + let cancel = (_: Error) => {} + const promise: Promise = new Promise((_, reject) => { + cancel = reject + }) + + sessionIdToPromise[sessionId] = { + cancel, + promise, + shouldOpenErrorModal: true, + } + }, [sessionId]) + + const cancel = useCallback(() => { + const sessionData = sessionIdToPromise[sessionId] + sessionIdToPromise[sessionId] = { + ...sessionData, + shouldOpenErrorModal: false, + } + + sessionIdToPromise[sessionId].cancel(new Error(errorMsgText)) + }, [errorMsgText, sessionId]) + + const resolve = useCallback( + () => + Promise.race([sessionIdToPromise[sessionId].promise, Promise.resolve()]), + [sessionId], + ) + + return { + cancel, + resolve, + sessionIdToPromise, + } +} diff --git a/dapp/src/hooks/useTransactionModal.ts b/dapp/src/hooks/useTransactionModal.ts index e052ed131..7e4b9d0be 100644 --- a/dapp/src/hooks/useTransactionModal.ts +++ b/dapp/src/hooks/useTransactionModal.ts @@ -1,4 +1,4 @@ -import { ACTION_FLOW_TYPES, ActionFlowType, MODAL_TYPES } from "#/types" +import { ActionFlowType, MODAL_TYPES } from "#/types" import { useCallback } from "react" import { useModal } from "./useModal" @@ -8,7 +8,7 @@ export function useTransactionModal(type: ActionFlowType) { return useCallback(() => { openModal(MODAL_TYPES[type], { type, - closeOnEsc: type !== ACTION_FLOW_TYPES.UNSTAKE, + closeOnEsc: false, }) }, [openModal, type]) }