diff --git a/src/components/MultiHopTrade/components/MultiHopTradeConfirm/components/StepperStep.tsx b/src/components/MultiHopTrade/components/MultiHopTradeConfirm/components/StepperStep.tsx
index 8c9e37d9baa..271834e727d 100644
--- a/src/components/MultiHopTrade/components/MultiHopTradeConfirm/components/StepperStep.tsx
+++ b/src/components/MultiHopTrade/components/MultiHopTradeConfirm/components/StepperStep.tsx
@@ -67,7 +67,11 @@ export const StepperStep = ({
stepIndicatorVariant = 'default',
}: StepperStepProps) => {
const { indicator: indicatorStyles } = useStyleConfig('Stepper', {
- variant: isError ? 'error' : stepIndicatorVariant,
+ variant: isError
+ ? stepIndicatorVariant === 'innerSteps'
+ ? 'innerStepsError'
+ : 'error'
+ : stepIndicatorVariant,
}) as { indicator: SystemStyleObject }
return (
diff --git a/src/components/MultiHopTrade/components/SharedConfirm/SharedConfirmFooter.tsx b/src/components/MultiHopTrade/components/SharedConfirm/SharedConfirmFooter.tsx
index ed2e2e82811..c86e83961d2 100644
--- a/src/components/MultiHopTrade/components/SharedConfirm/SharedConfirmFooter.tsx
+++ b/src/components/MultiHopTrade/components/SharedConfirm/SharedConfirmFooter.tsx
@@ -1,8 +1,8 @@
import { Stack } from '@chakra-ui/react'
type SharedConfirmFooterProps = {
- detail: React.ReactNode | null
- button: React.ReactNode
+ detail: JSX.Element | null
+ button: JSX.Element | null
}
export const SharedConfirmFooter = ({ detail, button }: SharedConfirmFooterProps) => {
diff --git a/src/components/MultiHopTrade/components/TradeConfirm/EtaStep.tsx b/src/components/MultiHopTrade/components/TradeConfirm/EtaStep.tsx
index 35528c39ad8..8ccb35d7e12 100644
--- a/src/components/MultiHopTrade/components/TradeConfirm/EtaStep.tsx
+++ b/src/components/MultiHopTrade/components/TradeConfirm/EtaStep.tsx
@@ -2,8 +2,7 @@ import { ArrowDownIcon } from '@chakra-ui/icons'
import prettyMilliseconds from 'pretty-ms'
import { useMemo } from 'react'
import { useTranslate } from 'react-polyglot'
-import { selectIsActiveQuoteMultiHop } from 'state/slices/tradeInputSlice/selectors'
-import { selectFirstHop, selectLastHop } from 'state/slices/tradeQuoteSlice/selectors'
+import { selectActiveQuote } from 'state/slices/tradeQuoteSlice/selectors'
import { useAppSelector } from 'state/store'
import { StepperStep } from '../MultiHopTradeConfirm/components/StepperStep'
@@ -12,18 +11,15 @@ const etaStepProps = { alignItems: 'center', py: 2 }
export const EtaStep = () => {
const translate = useTranslate()
- const tradeQuoteFirstHop = useAppSelector(selectFirstHop)
- const tradeQuoteLastHop = useAppSelector(selectLastHop)
- const isMultiHopTrade = useAppSelector(selectIsActiveQuoteMultiHop)
- const totalEstimatedExecutionTimeMs = useMemo(() => {
- if (!tradeQuoteFirstHop || !tradeQuoteLastHop) return undefined
- if (!tradeQuoteFirstHop.estimatedExecutionTimeMs || !tradeQuoteLastHop.estimatedExecutionTimeMs)
- return undefined
- return isMultiHopTrade
- ? tradeQuoteFirstHop.estimatedExecutionTimeMs + tradeQuoteLastHop.estimatedExecutionTimeMs
- : tradeQuoteFirstHop.estimatedExecutionTimeMs
- }, [isMultiHopTrade, tradeQuoteFirstHop, tradeQuoteLastHop])
- const swapperName = tradeQuoteFirstHop?.source
+ const activeQuote = useAppSelector(selectActiveQuote)
+ const totalEstimatedExecutionTimeMs = useMemo(
+ () =>
+ activeQuote?.steps.reduce((acc, step) => {
+ return acc + (step.estimatedExecutionTimeMs ?? 0)
+ }, 0),
+ [activeQuote?.steps],
+ )
+ const swapperName = activeQuote?.steps[0].source
const stepIndicator = useMemo(() => {
return
diff --git a/src/components/MultiHopTrade/components/TradeConfirm/ExpandableTradeSteps.tsx b/src/components/MultiHopTrade/components/TradeConfirm/ExpandableStepperSteps.tsx
similarity index 76%
rename from src/components/MultiHopTrade/components/TradeConfirm/ExpandableTradeSteps.tsx
rename to src/components/MultiHopTrade/components/TradeConfirm/ExpandableStepperSteps.tsx
index 0d98b5b78cb..cd9d8b9de57 100644
--- a/src/components/MultiHopTrade/components/TradeConfirm/ExpandableTradeSteps.tsx
+++ b/src/components/MultiHopTrade/components/TradeConfirm/ExpandableStepperSteps.tsx
@@ -9,18 +9,18 @@ import {
selectConfirmedTradeExecutionState,
selectHopExecutionMetadata,
} from 'state/slices/tradeQuoteSlice/selectors'
-import { TradeExecutionState } from 'state/slices/tradeQuoteSlice/types'
+import { TradeExecutionState, TransactionExecutionState } from 'state/slices/tradeQuoteSlice/types'
import { useAppSelector, useSelectorWithArgs } from 'state/store'
import { StepperStep } from '../MultiHopTradeConfirm/components/StepperStep'
-import { ExpandedTradeSteps } from './ExpandedTradeSteps'
+import { ExpandedStepperSteps } from './ExpandedStepperSteps'
import { getHopExecutionStateSummaryStepTranslation } from './helpers'
import { useCurrentHopIndex } from './hooks/useCurrentHopIndex'
-import { useTradeSteps } from './hooks/useTradeSteps'
+import { useStepperSteps } from './hooks/useStepperSteps'
const collapseStyle = { width: '100%' }
-export const ExpandableTradeSteps = () => {
+export const ExpandableStepperSteps = () => {
const [isExpanded, setIsExpanded] = useState(false)
const confirmedTradeExecutionState = useAppSelector(selectConfirmedTradeExecutionState)
const summaryStepProps = useMemo(
@@ -44,22 +44,24 @@ export const ExpandableTradeSteps = () => {
}
}, [activeTradeId, currentHopIndex])
const swapperName = activeTradeQuote?.steps[0].source
- const { state: hopExecutionState } = useSelectorWithArgs(
- selectHopExecutionMetadata,
- hopExecutionMetadataFilter,
- )
+ const {
+ state: hopExecutionState,
+ swap: { state: swapTxState },
+ } = useSelectorWithArgs(selectHopExecutionMetadata, hopExecutionMetadataFilter)
const summaryStepIndicator = useMemo(() => {
- if (confirmedTradeExecutionState === TradeExecutionState.TradeComplete) {
- return
- } else if (activeQuoteError) {
- return
- } else {
- return
+ switch (true) {
+ case confirmedTradeExecutionState === TradeExecutionState.TradeComplete:
+ return
+ case !!activeQuoteError:
+ case swapTxState === TransactionExecutionState.Failed:
+ return
+ default:
+ return
}
- }, [confirmedTradeExecutionState, activeQuoteError])
+ }, [confirmedTradeExecutionState, activeQuoteError, swapTxState])
- const { totalSteps, currentTradeStepIndex: currentStep } = useTradeSteps()
+ const { totalSteps, currentTradeStepIndex: currentStep } = useStepperSteps()
const progressValue = (currentStep / (totalSteps - 1)) * 100
const titleElement = useMemo(() => {
@@ -100,7 +102,7 @@ export const ExpandableTradeSteps = () => {
/>
- {activeTradeQuote && }
+ {activeTradeQuote && }
>
diff --git a/src/components/MultiHopTrade/components/TradeConfirm/ExpandedTradeSteps.tsx b/src/components/MultiHopTrade/components/TradeConfirm/ExpandedStepperSteps.tsx
similarity index 90%
rename from src/components/MultiHopTrade/components/TradeConfirm/ExpandedTradeSteps.tsx
rename to src/components/MultiHopTrade/components/TradeConfirm/ExpandedStepperSteps.tsx
index 30d1c5df4c7..48093fd7b05 100644
--- a/src/components/MultiHopTrade/components/TradeConfirm/ExpandedTradeSteps.tsx
+++ b/src/components/MultiHopTrade/components/TradeConfirm/ExpandedStepperSteps.tsx
@@ -26,10 +26,10 @@ import {
} from 'state/slices/tradeQuoteSlice/selectors'
import { useAppSelector, useSelectorWithArgs } from 'state/store'
-import { StepperStep } from '../MultiHopTradeConfirm/components/StepperStep'
-import { TradeStep } from './helpers'
+import { StepperStep as StepperStepComponent } from '../MultiHopTradeConfirm/components/StepperStep'
+import { StepperStep } from './helpers'
+import { useStepperSteps } from './hooks/useStepperSteps'
import { useStreamingProgress } from './hooks/useStreamingProgress'
-import { useTradeSteps } from './hooks/useTradeSteps'
import { TxLabel } from './TxLabel'
const erroredStepIndicator =
@@ -37,11 +37,11 @@ const completedStepIndicator =
const stepProps = { alignItems: 'center', py: 2, pr: 2, pl: 1.5 }
-type ExpandedTradeStepsProps = {
+type ExpandedStepperStepsProps = {
activeTradeQuote: TradeQuote | TradeRate
}
-export const ExpandedTradeSteps = ({ activeTradeQuote }: ExpandedTradeStepsProps) => {
+export const ExpandedStepperSteps = ({ activeTradeQuote }: ExpandedStepperStepsProps) => {
const translate = useTranslate()
// this is the account we're selling from - assume this is the AccountId of the approval Tx
const firstHopSellAccountId = useAppSelector(selectFirstHopSellAccountId)
@@ -136,7 +136,7 @@ export const ExpandedTradeSteps = ({ activeTradeQuote }: ExpandedTradeStepsProps
swap: lastHopSwap,
} = useSelectorWithArgs(selectHopExecutionMetadata, lastHopExecutionMetadataFilter)
- const { currentTradeStepIndex: currentStep } = useTradeSteps()
+ const { currentTradeStepIndex: currentStep } = useStepperSteps()
const stepIndicator = useMemo(
() => (
@@ -327,65 +327,65 @@ export const ExpandedTradeSteps = ({ activeTradeQuote }: ExpandedTradeStepsProps
tradeQuoteLastHop,
])
- const { tradeSteps, currentTradeStep } = useTradeSteps()
+ const { tradeSteps, currentTradeStep } = useStepperSteps()
return (
- {tradeSteps[TradeStep.FirstHopReset] ? (
-
) : null}
- {tradeSteps[TradeStep.FirstHopApproval] ? (
-
) : null}
-
- {tradeSteps[TradeStep.LastHopReset] ? (
-
) : null}
- {tradeSteps[TradeStep.LastHopApproval] ? (
-
) : null}
- {tradeSteps[TradeStep.LastHopSwap] ? (
-
) : null}
diff --git a/src/components/MultiHopTrade/components/TradeConfirm/TradeConfirmBody.tsx b/src/components/MultiHopTrade/components/TradeConfirm/TradeConfirmBody.tsx
index 3ffac31f2ed..1250b25ca31 100644
--- a/src/components/MultiHopTrade/components/TradeConfirm/TradeConfirmBody.tsx
+++ b/src/components/MultiHopTrade/components/TradeConfirm/TradeConfirmBody.tsx
@@ -8,7 +8,7 @@ import { useAppSelector } from 'state/store'
import { SharedConfirmBody } from '../SharedConfirm/SharedConfirmBody'
import { EtaStep } from './EtaStep'
-import { ExpandableTradeSteps } from './ExpandableTradeSteps'
+import { ExpandableStepperSteps } from './ExpandableStepperSteps'
const InnerSteps = () => {
const confirmedTradeExecutionState = useAppSelector(selectConfirmedTradeExecutionState)
@@ -20,7 +20,7 @@ const InnerSteps = () => {
case TradeExecutionState.FirstHop:
case TradeExecutionState.SecondHop:
case TradeExecutionState.TradeComplete:
- return
+ return
default:
return null
}
diff --git a/src/components/MultiHopTrade/components/TradeConfirm/TradeConfirmFooter.tsx b/src/components/MultiHopTrade/components/TradeConfirm/TradeConfirmFooter.tsx
index 8676f252176..becd09aca0c 100644
--- a/src/components/MultiHopTrade/components/TradeConfirm/TradeConfirmFooter.tsx
+++ b/src/components/MultiHopTrade/components/TradeConfirm/TradeConfirmFooter.tsx
@@ -15,10 +15,10 @@ import { useAppSelector, useSelectorWithArgs } from 'state/store'
import { isPermit2Hop } from '../MultiHopTradeConfirm/hooks/helpers'
import { SharedConfirmFooter } from '../SharedConfirm/SharedConfirmFooter'
-import { TradeStep } from './helpers'
+import { StepperStep } from './helpers'
import { useActiveTradeAllowance } from './hooks/useActiveTradeAllowance'
import { useCurrentHopIndex } from './hooks/useCurrentHopIndex'
-import { useTradeSteps } from './hooks/useTradeSteps'
+import { useStepperSteps } from './hooks/useStepperSteps'
import { TradeConfirmSummary } from './TradeConfirmFooterContent/TradeConfirmSummary'
import { TradeFooterButton } from './TradeFooterButton'
@@ -79,7 +79,7 @@ export const TradeConfirmFooter: FC = ({
.times(feeAssetUserCurrencyRate.price)
.toFixed()
- const { currentTradeStep } = useTradeSteps()
+ const { currentTradeStep } = useStepperSteps()
const tradeResetStepSummary = useMemo(() => {
return (
@@ -221,14 +221,14 @@ export const TradeConfirmFooter: FC = ({
// No trade step is active, quote is still to be confirmed
case undefined:
return
- case TradeStep.FirstHopReset:
- case TradeStep.LastHopReset:
+ case StepperStep.FirstHopReset:
+ case StepperStep.LastHopReset:
return tradeResetStepSummary
- case TradeStep.FirstHopApproval:
- case TradeStep.LastHopApproval:
+ case StepperStep.FirstHopApproval:
+ case StepperStep.LastHopApproval:
return tradeAllowanceStepSummary
- case TradeStep.FirstHopSwap:
- case TradeStep.LastHopSwap:
+ case StepperStep.FirstHopSwap:
+ case StepperStep.LastHopSwap:
return tradeExecutionStepSummary
default:
return null
diff --git a/src/components/MultiHopTrade/components/TradeConfirm/helpers.ts b/src/components/MultiHopTrade/components/TradeConfirm/helpers.ts
index d2128877bfe..b0963b7f98d 100644
--- a/src/components/MultiHopTrade/components/TradeConfirm/helpers.ts
+++ b/src/components/MultiHopTrade/components/TradeConfirm/helpers.ts
@@ -44,7 +44,7 @@ export const getHopExecutionStateSummaryStepTranslation = (
}
}
-type TradeStepParams = {
+type StepperStepParams = {
firstHopAllowanceApproval: ApprovalExecutionMetadata
firstHopPermit2: Omit & {
permit2Signature?: string | undefined
@@ -58,7 +58,7 @@ type TradeStepParams = {
isMultiHopTrade?: boolean
}
-export enum TradeStep {
+export enum StepperStep {
FirstHopReset = 'firstHopReset',
FirstHopApproval = 'firstHopApproval',
FirstHopSwap = 'firstHopSwap',
@@ -68,7 +68,7 @@ export enum TradeStep {
TradeComplete = 'tradeComplete',
}
-export const getTradeSteps = (params: TradeStepParams): Record => {
+export const getStepperSteps = (params: StepperStepParams): Record => {
const {
firstHopAllowanceReset,
firstHopAllowanceApproval,
@@ -79,30 +79,30 @@ export const getTradeSteps = (params: TradeStepParams): Record {
- return Object.values(getTradeSteps(params)).filter(Boolean).length
+export const countStepperSteps = (params: StepperStepParams): number => {
+ return Object.values(getStepperSteps(params)).filter(Boolean).length
}
const isInApprovalState = (state: HopExecutionState): boolean => {
@@ -111,35 +111,35 @@ const isInApprovalState = (state: HopExecutionState): boolean => {
)
}
-export const getCurrentTradeStep = (
+export const getCurrentStepperStep = (
currentHopIndex: number,
hopExecutionState: HopExecutionState,
-): TradeStep | undefined => {
- if (hopExecutionState === HopExecutionState.Complete) return TradeStep.TradeComplete
+): StepperStep | undefined => {
+ if (hopExecutionState === HopExecutionState.Complete) return StepperStep.TradeComplete
if (hopExecutionState === HopExecutionState.Pending) return undefined
if (currentHopIndex === 0) {
if (hopExecutionState === HopExecutionState.AwaitingAllowanceReset)
- return TradeStep.FirstHopReset
- if (isInApprovalState(hopExecutionState)) return TradeStep.FirstHopApproval
- if (hopExecutionState === HopExecutionState.AwaitingSwap) return TradeStep.FirstHopSwap
+ return StepperStep.FirstHopReset
+ if (isInApprovalState(hopExecutionState)) return StepperStep.FirstHopApproval
+ if (hopExecutionState === HopExecutionState.AwaitingSwap) return StepperStep.FirstHopSwap
} else if (currentHopIndex === 1) {
if (hopExecutionState === HopExecutionState.AwaitingAllowanceReset)
- return TradeStep.LastHopReset
- if (isInApprovalState(hopExecutionState)) return TradeStep.LastHopApproval
- if (hopExecutionState === HopExecutionState.AwaitingSwap) return TradeStep.LastHopSwap
+ return StepperStep.LastHopReset
+ if (isInApprovalState(hopExecutionState)) return StepperStep.LastHopApproval
+ if (hopExecutionState === HopExecutionState.AwaitingSwap) return StepperStep.LastHopSwap
}
}
-export const getCurrentTradeStepIndex = (
- params: TradeStepParams & {
+export const getCurrentStepperStepIndex = (
+ params: StepperStepParams & {
currentHopIndex: number
hopExecutionState: HopExecutionState
},
): number => {
- const steps = getTradeSteps(params)
+ const steps = getStepperSteps(params)
const activeSteps = Object.entries(steps).filter(([_, isActive]) => isActive)
- const currentStep = getCurrentTradeStep(params.currentHopIndex, params.hopExecutionState)
+ const currentStep = getCurrentStepperStep(params.currentHopIndex, params.hopExecutionState)
if (!currentStep)
return params.hopExecutionState === HopExecutionState.Pending ? 0 : activeSteps.length - 1
diff --git a/src/components/MultiHopTrade/components/TradeConfirm/hooks/useTradeSteps.tsx b/src/components/MultiHopTrade/components/TradeConfirm/hooks/useStepperSteps.tsx
similarity index 86%
rename from src/components/MultiHopTrade/components/TradeConfirm/hooks/useTradeSteps.tsx
rename to src/components/MultiHopTrade/components/TradeConfirm/hooks/useStepperSteps.tsx
index 7d893ddd0fd..040b974b3a3 100644
--- a/src/components/MultiHopTrade/components/TradeConfirm/hooks/useTradeSteps.tsx
+++ b/src/components/MultiHopTrade/components/TradeConfirm/hooks/useStepperSteps.tsx
@@ -7,14 +7,14 @@ import {
import { useAppSelector } from 'state/store'
import {
- countTradeSteps,
- getCurrentTradeStep,
- getCurrentTradeStepIndex,
- getTradeSteps,
+ countStepperSteps,
+ getCurrentStepperStep,
+ getCurrentStepperStepIndex,
+ getStepperSteps,
} from '../helpers'
import { useCurrentHopIndex } from './useCurrentHopIndex'
-export const useTradeSteps = () => {
+export const useStepperSteps = () => {
const activeTradeId = useAppSelector(selectActiveQuote)?.id
const isMultiHopTrade = useAppSelector(selectIsActiveQuoteMultiHop)
@@ -69,19 +69,19 @@ export const useTradeSteps = () => {
],
)
- const tradeSteps = useMemo(() => getTradeSteps(params), [params])
- const totalSteps = useMemo(() => countTradeSteps(params), [params])
+ const tradeSteps = useMemo(() => getStepperSteps(params), [params])
+ const totalSteps = useMemo(() => countStepperSteps(params), [params])
const currentHopIndex = useCurrentHopIndex()
const currentHopExecutionState = useMemo(() => {
return currentHopIndex === 0 ? firstHopExecutionState : lastHopExecutionState
}, [currentHopIndex, firstHopExecutionState, lastHopExecutionState])
const currentTradeStep = useMemo(
- () => getCurrentTradeStep(currentHopIndex, currentHopExecutionState),
+ () => getCurrentStepperStep(currentHopIndex, currentHopExecutionState),
[currentHopIndex, currentHopExecutionState],
)
const currentTradeStepIndex = useMemo(
() =>
- getCurrentTradeStepIndex({
+ getCurrentStepperStepIndex({
...params,
currentHopIndex,
hopExecutionState: currentHopExecutionState,
diff --git a/src/components/Stepper.theme.ts b/src/components/Stepper.theme.ts
index a3ab8306c21..11e2edce439 100644
--- a/src/components/Stepper.theme.ts
+++ b/src/components/Stepper.theme.ts
@@ -37,6 +37,33 @@ const baseStyle = {
bg: 'border.base',
},
},
+ innerSteps: {
+ step: {
+ '&[data-status=active]': {
+ bg: 'background.surface.raised.base',
+ borderRadius: '8px',
+ },
+ },
+ indicator: {
+ width: '20px',
+ height: '20px',
+ minWidth: '20px',
+ borderWidth: '3px',
+ // Override the throbbing animation
+ '&[data-status=active]:not(.step-pending)': {
+ animation: 'none',
+ },
+ '&[data-status=active]': {
+ borderWidth: '3px',
+ },
+ '&[data-status=incomplete]': {
+ borderWidth: '3px',
+ },
+ '&[data-status=complete]': {
+ borderWidth: '3px',
+ },
+ },
+ },
}
const variants = {
@@ -55,30 +82,22 @@ const variants = {
},
},
},
- innerSteps: {
- step: {
- '&[data-status=active]': {
- bg: 'background.surface.raised.base',
- borderRadius: '8px',
- },
- },
+ innerStepsError: {
+ ...baseStyle.innerSteps,
indicator: {
- width: '20px',
- height: '20px',
- minWidth: '20px',
- borderWidth: '3px',
- // Override the throbbing animation
+ ...baseStyle.innerSteps.indicator,
'&[data-status=active]:not(.step-pending)': {
animation: 'none',
+ borderWidth: '0',
},
'&[data-status=active]': {
- borderWidth: '3px',
+ borderWidth: '0',
},
'&[data-status=incomplete]': {
- borderWidth: '3px',
+ borderWidth: '0',
},
'&[data-status=complete]': {
- borderWidth: '3px',
+ borderWidth: '0',
},
},
},