diff --git a/src/assets/translations/en/main.json b/src/assets/translations/en/main.json
index 189acd37dc9..10e19262a2b 100644
--- a/src/assets/translations/en/main.json
+++ b/src/assets/translations/en/main.json
@@ -1011,17 +1011,19 @@
},
"hopTitle": {
"bridge": "Bridge via %{swapperName}",
- "swap": "Swap via %{swapperName}"
+ "swap": "Swap via %{swapperName}",
+ "swapEta": "Swap via %{swapperName} in ~%{eta}"
},
"transactionFailed": {
"bridge": "A problem occurred executing the bridge",
"swap": "A problem occurred executing the swap"
},
"transactionTitle": {
- "bridge": "Bridge from %{sellChainName} to %{buyChainName} via %{swapperName}",
+ "bridge": "Swap from %{sellChainName} to %{buyChainName} via %{swapperName}",
"swap": "Swap on %{sellChainName} via %{swapperName}"
},
"approvalTitle": "Token allowance approval",
+ "permit2Title": "Permit token transfer",
"resetTitle": "Token allowance reset",
"fiatAmountOnChain": "%{amountFiatFormatted} on %{chainName}",
"quote": {
diff --git a/src/components/Amount/Amount.tsx b/src/components/Amount/Amount.tsx
index 80c7268ffa5..2fa6d424bf1 100644
--- a/src/components/Amount/Amount.tsx
+++ b/src/components/Amount/Amount.tsx
@@ -14,6 +14,7 @@ export type AmountProps = {
abbreviated?: boolean
truncateLargeNumbers?: boolean
maximumFractionDigits?: number
+ noSpace?: boolean
} & TextProps
export function Amount({
@@ -23,6 +24,7 @@ export function Amount({
maximumFractionDigits,
omitDecimalTrailingZeros = false,
abbreviated = false,
+ noSpace = false,
...props
}: any): React.ReactElement {
const {
@@ -31,9 +33,9 @@ export function Amount({
return (
- {prefix}
+ {prefix && `${prefix}${noSpace ? '' : ' '}`}
{toString(value, { maximumFractionDigits, omitDecimalTrailingZeros, abbreviated })}
- {suffix}
+ {suffix && `${noSpace ? '' : ' '}${suffix}`}
)
}
@@ -62,6 +64,7 @@ const Crypto = ({
omitDecimalTrailingZeros = false,
abbreviated = false,
truncateLargeNumbers = false,
+ noSpace = false,
...props
}: CryptoAmountProps) => {
const {
@@ -77,9 +80,9 @@ const Crypto = ({
return (
- {prefix && `${prefix} `}
+ {prefix && `${prefix}${noSpace ? '' : ' '}`}
{crypto}
- {suffix && ` ${suffix}`}
+ {suffix && `${noSpace ? '' : ' '}${suffix}`}
)
}
@@ -92,6 +95,7 @@ const Fiat = ({
maximumFractionDigits,
omitDecimalTrailingZeros = false,
abbreviated = false,
+ noSpace = false,
...props
}: FiatAmountProps) => {
const {
@@ -107,9 +111,9 @@ const Fiat = ({
return (
- {prefix && `${prefix} `}
+ {prefix && `${prefix}${noSpace ? '' : ' '}`}
{fiat}
- {suffix && ` ${suffix}`}
+ {suffix && `${noSpace ? '' : ' '}${suffix}`}
)
}
diff --git a/src/components/MultiHopTrade/components/TradeConfirm/EtaStep.tsx b/src/components/MultiHopTrade/components/TradeConfirm/EtaStep.tsx
index 3dd0c7dbd65..35528c39ad8 100644
--- a/src/components/MultiHopTrade/components/TradeConfirm/EtaStep.tsx
+++ b/src/components/MultiHopTrade/components/TradeConfirm/EtaStep.tsx
@@ -1,6 +1,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 { useAppSelector } from 'state/store'
@@ -10,6 +11,7 @@ import { StepperStep } from '../MultiHopTradeConfirm/components/StepperStep'
const etaStepProps = { alignItems: 'center', py: 2 }
export const EtaStep = () => {
+ const translate = useTranslate()
const tradeQuoteFirstHop = useAppSelector(selectFirstHop)
const tradeQuoteLastHop = useAppSelector(selectLastHop)
const isMultiHopTrade = useAppSelector(selectIsActiveQuoteMultiHop)
@@ -21,15 +23,19 @@ export const EtaStep = () => {
? tradeQuoteFirstHop.estimatedExecutionTimeMs + tradeQuoteLastHop.estimatedExecutionTimeMs
: tradeQuoteFirstHop.estimatedExecutionTimeMs
}, [isMultiHopTrade, tradeQuoteFirstHop, tradeQuoteLastHop])
+ const swapperName = tradeQuoteFirstHop?.source
const stepIndicator = useMemo(() => {
return
}, [])
const title = useMemo(() => {
return totalEstimatedExecutionTimeMs
- ? `Estimated completion ${prettyMilliseconds(totalEstimatedExecutionTimeMs)}`
- : 'Estimated completion time unknown'
- }, [totalEstimatedExecutionTimeMs])
+ ? translate('trade.hopTitle.swapEta', {
+ swapperName,
+ eta: prettyMilliseconds(totalEstimatedExecutionTimeMs),
+ })
+ : translate('trade.hopTitle.swap', { swapperName })
+ }, [totalEstimatedExecutionTimeMs, swapperName, translate])
return (
{
hopIndex: currentHopIndex ?? 0,
}
}, [activeTradeId, currentHopIndex])
- const swapperName = useAppSelector(selectActiveSwapperName)
+ const swapperName = activeTradeQuote?.steps[0].source
const { state: hopExecutionState } = useSelectorWithArgs(
selectHopExecutionMetadata,
hopExecutionMetadataFilter,
diff --git a/src/components/MultiHopTrade/components/TradeConfirm/ExpandedTradeSteps.tsx b/src/components/MultiHopTrade/components/TradeConfirm/ExpandedTradeSteps.tsx
index cbd467fc7fa..30d1c5df4c7 100644
--- a/src/components/MultiHopTrade/components/TradeConfirm/ExpandedTradeSteps.tsx
+++ b/src/components/MultiHopTrade/components/TradeConfirm/ExpandedTradeSteps.tsx
@@ -1,18 +1,27 @@
import { CheckCircleIcon, WarningIcon } from '@chakra-ui/icons'
-import { Flex, HStack, Stepper, StepStatus, Tag, VStack } from '@chakra-ui/react'
+import {
+ Box,
+ Flex,
+ HStack,
+ Icon,
+ Stepper,
+ StepStatus,
+ Tag,
+ Tooltip,
+ VStack,
+} from '@chakra-ui/react'
import type { TradeQuote, TradeRate } from '@shapeshiftoss/swapper'
import { useMemo } from 'react'
+import { FaInfoCircle } from 'react-icons/fa'
import { useTranslate } from 'react-polyglot'
import { RawText, Text } from 'components/Text'
import { getChainAdapterManager } from 'context/PluginProvider/chainAdapterSingleton'
import {
selectFirstHopSellAccountId,
- selectIsActiveQuoteMultiHop,
selectSecondHopSellAccountId,
} from 'state/slices/tradeInputSlice/selectors'
import {
selectActiveQuoteErrors,
- selectActiveSwapperName,
selectHopExecutionMetadata,
} from 'state/slices/tradeQuoteSlice/selectors'
import { useAppSelector, useSelectorWithArgs } from 'state/store'
@@ -34,14 +43,13 @@ type ExpandedTradeStepsProps = {
export const ExpandedTradeSteps = ({ activeTradeQuote }: ExpandedTradeStepsProps) => {
const translate = useTranslate()
- const swapperName = useAppSelector(selectActiveSwapperName)
// this is the account we're selling from - assume this is the AccountId of the approval Tx
const firstHopSellAccountId = useAppSelector(selectFirstHopSellAccountId)
const lastHopSellAccountId = useAppSelector(selectSecondHopSellAccountId)
- const isMultiHopTrade = useAppSelector(selectIsActiveQuoteMultiHop)
const tradeQuoteFirstHop = activeTradeQuote.steps[0]
const tradeQuoteLastHop = activeTradeQuote.steps[1]
const activeTradeId = activeTradeQuote.id
+ const swapperName = tradeQuoteFirstHop?.source
const firstHopStreamingProgress = useStreamingProgress({
hopIndex: 0,
@@ -160,8 +168,14 @@ export const ExpandedTradeSteps = ({ activeTradeQuote }: ExpandedTradeStepsProps
return (
{firstHopPermit2.isRequired === true ? (
- // TODO: Add permit2 tooltip
-
+ <>
+
+
+
+
+
+
+ >
) : (
<>
@@ -181,6 +195,7 @@ export const ExpandedTradeSteps = ({ activeTradeQuote }: ExpandedTradeStepsProps
firstHopPermit2.isRequired,
firstHopSellAccountId,
tradeQuoteFirstHop,
+ translate,
])
const firstHopActionTitle = useMemo(() => {
@@ -242,8 +257,14 @@ export const ExpandedTradeSteps = ({ activeTradeQuote }: ExpandedTradeStepsProps
return (
{lastHopPermit2.isRequired === true ? (
- // TODO: Add permit2 tooltip
-
+ <>
+
+
+
+
+
+
+ >
) : (
<>
@@ -263,6 +284,7 @@ export const ExpandedTradeSteps = ({ activeTradeQuote }: ExpandedTradeStepsProps
lastHopPermit2.isRequired,
lastHopSellAccountId,
tradeQuoteLastHop,
+ translate,
])
const lastHopActionTitle = useMemo(() => {
@@ -337,38 +359,36 @@ export const ExpandedTradeSteps = ({ activeTradeQuote }: ExpandedTradeStepsProps
isError={activeQuoteError && currentTradeStep === TradeStep.FirstHopSwap}
stepIndicatorVariant='innerSteps'
/>
- {isMultiHopTrade && (
- <>
- {tradeSteps[TradeStep.LastHopReset] ? (
-
- ) : null}
- {tradeSteps[TradeStep.LastHopApproval] ? (
-
- ) : null}
-
- >
- )}
+ {tradeSteps[TradeStep.LastHopReset] ? (
+
+ ) : null}
+ {tradeSteps[TradeStep.LastHopApproval] ? (
+
+ ) : null}
+ {tradeSteps[TradeStep.LastHopSwap] ? (
+
+ ) : null}
)
}
diff --git a/src/components/MultiHopTrade/components/TradeConfirm/TradeConfirmFooter.tsx b/src/components/MultiHopTrade/components/TradeConfirm/TradeConfirmFooter.tsx
index 824e7360328..8676f252176 100644
--- a/src/components/MultiHopTrade/components/TradeConfirm/TradeConfirmFooter.tsx
+++ b/src/components/MultiHopTrade/components/TradeConfirm/TradeConfirmFooter.tsx
@@ -1,4 +1,4 @@
-import { Skeleton, Stack, Switch } from '@chakra-ui/react'
+import { HStack, Skeleton, Stack, Switch } from '@chakra-ui/react'
import type { TradeQuoteStep } from '@shapeshiftoss/swapper'
import type { FC } from 'react'
import { useMemo, useState } from 'react'
@@ -10,10 +10,7 @@ import { bnOrZero } from 'lib/bignumber/bignumber'
import { fromBaseUnit } from 'lib/math'
import { selectFeeAssetById } from 'state/slices/assetsSlice/selectors'
import { selectMarketDataByAssetIdUserCurrency } from 'state/slices/marketDataSlice/selectors'
-import {
- selectHopNetworkFeeUserCurrency,
- selectIsActiveSwapperQuoteLoading,
-} from 'state/slices/tradeQuoteSlice/selectors'
+import { selectIsActiveSwapperQuoteLoading } from 'state/slices/tradeQuoteSlice/selectors'
import { useAppSelector, useSelectorWithArgs } from 'state/store'
import { isPermit2Hop } from '../MultiHopTradeConfirm/hooks/helpers'
@@ -37,19 +34,21 @@ export const TradeConfirmFooter: FC = ({
const [isExactAllowance, toggleIsExactAllowance] = useToggle(true)
const [hasClickedButton, setHasClickedButton] = useState(false)
const currentHopIndex = useCurrentHopIndex()
- const tradeNetworkFeeFiatUserCurrency = useSelectorWithArgs(selectHopNetworkFeeUserCurrency, {
- hopIndex: currentHopIndex,
- })
+ const networkFeeCryptoBaseUnit = tradeQuoteStep.feeData.networkFeeCryptoBaseUnit
+ const feeAsset = useSelectorWithArgs(selectFeeAssetById, tradeQuoteStep.sellAsset.assetId)
+ const networkFeeCryptoPrecision = fromBaseUnit(networkFeeCryptoBaseUnit, feeAsset?.precision ?? 0)
+ const feeAssetUserCurrencyRate = useSelectorWithArgs(
+ selectMarketDataByAssetIdUserCurrency,
+ feeAsset?.assetId ?? '',
+ )
+ const networkFeeUserCurrency = bnOrZero(networkFeeCryptoPrecision)
+ .times(feeAssetUserCurrencyRate.price)
+ .toFixed()
const isActiveSwapperQuoteLoading = useAppSelector(selectIsActiveSwapperQuoteLoading)
const sellChainFeeAsset = useSelectorWithArgs(
selectFeeAssetById,
tradeQuoteStep.sellAsset.assetId,
)
- const buyChainFeeAsset = useSelectorWithArgs(selectFeeAssetById, tradeQuoteStep.buyAsset.assetId)
- const sellChainFeeAssetUserCurrencyRate = useSelectorWithArgs(
- selectMarketDataByAssetIdUserCurrency,
- sellChainFeeAsset?.assetId ?? '',
- )
const {
allowanceResetNetworkFeeCryptoBaseUnit,
@@ -62,22 +61,22 @@ export const TradeConfirmFooter: FC = ({
activeTradeId,
})
- const allowanceResetNetworkFeeCryptoHuman = fromBaseUnit(
+ const allowanceResetNetworkFeeCryptoPrecision = fromBaseUnit(
allowanceResetNetworkFeeCryptoBaseUnit,
sellChainFeeAsset?.precision ?? 0,
)
- const allowanceResetNetworkFeeUserCurrency = bnOrZero(allowanceResetNetworkFeeCryptoHuman)
- .times(sellChainFeeAssetUserCurrencyRate.price)
+ const allowanceResetNetworkFeeUserCurrency = bnOrZero(allowanceResetNetworkFeeCryptoPrecision)
+ .times(feeAssetUserCurrencyRate.price)
.toFixed()
- const approvalNetworkFeeCryptoHuman = fromBaseUnit(
+ const approvalNetworkFeeCryptoPrecision = fromBaseUnit(
approvalNetworkFeeCryptoBaseUnit,
- buyChainFeeAsset?.precision ?? 0,
+ sellChainFeeAsset?.precision ?? 0,
)
- const approvalNetworkFeeUserCurrency = bnOrZero(approvalNetworkFeeCryptoHuman)
- .times(sellChainFeeAssetUserCurrencyRate.price)
+ const approvalNetworkFeeUserCurrency = bnOrZero(approvalNetworkFeeCryptoPrecision)
+ .times(feeAssetUserCurrencyRate.price)
.toFixed()
const { currentTradeStep } = useTradeSteps()
@@ -91,13 +90,30 @@ export const TradeConfirmFooter: FC = ({
-
+
+
+
+
)
- }, [allowanceResetNetworkFeeUserCurrency, isAllowanceResetLoading])
+ }, [
+ allowanceResetNetworkFeeCryptoPrecision,
+ allowanceResetNetworkFeeUserCurrency,
+ isAllowanceResetLoading,
+ sellChainFeeAsset?.symbol,
+ ])
const isPermit2 = useMemo(() => {
return isPermit2Hop(tradeQuoteStep)
@@ -116,7 +132,19 @@ export const TradeConfirmFooter: FC = ({
-
+
+
+
+
@@ -148,6 +176,8 @@ export const TradeConfirmFooter: FC = ({
)
}, [
isAllowanceApprovalLoading,
+ sellChainFeeAsset?.symbol,
+ approvalNetworkFeeCryptoPrecision,
approvalNetworkFeeUserCurrency,
isPermit2,
isExactAllowance,
@@ -164,13 +194,27 @@ export const TradeConfirmFooter: FC = ({
-
+
+
+
+
)
- }, [tradeNetworkFeeFiatUserCurrency, isActiveSwapperQuoteLoading])
+ }, [
+ feeAsset?.symbol,
+ isActiveSwapperQuoteLoading,
+ networkFeeCryptoPrecision,
+ networkFeeUserCurrency,
+ ])
const tradeDetail = useMemo(() => {
switch (currentTradeStep) {
diff --git a/src/components/MultiHopTrade/components/TradeConfirm/TradeConfirmFooterContent/TradeConfirmSummary.tsx b/src/components/MultiHopTrade/components/TradeConfirm/TradeConfirmFooterContent/TradeConfirmSummary.tsx
index a1714be73e1..4b1b6401924 100644
--- a/src/components/MultiHopTrade/components/TradeConfirm/TradeConfirmFooterContent/TradeConfirmSummary.tsx
+++ b/src/components/MultiHopTrade/components/TradeConfirm/TradeConfirmFooterContent/TradeConfirmSummary.tsx
@@ -12,6 +12,7 @@ import {
Tooltip,
useColorModeValue,
} from '@chakra-ui/react'
+import type { AssetId } from '@shapeshiftoss/caip'
import type { AmountDisplayMeta } from '@shapeshiftoss/swapper'
import { bnOrZero, fromBaseUnit, isSome } from '@shapeshiftoss/utils'
import { useCallback, useMemo, useState } from 'react'
@@ -26,6 +27,7 @@ import { useToggle } from 'hooks/useToggle/useToggle'
import { THORSWAP_MAXIMUM_YEAR_TRESHOLD, THORSWAP_UNIT_THRESHOLD } from 'lib/fees/model'
import { middleEllipsis } from 'lib/utils'
import { selectThorVotingPower } from 'state/apis/snapshot/selectors'
+import { selectMarketDataUserCurrency } from 'state/slices/marketDataSlice/selectors'
import {
selectInputBuyAsset,
selectInputSellAmountUsd,
@@ -49,13 +51,21 @@ import { MaxSlippage } from '../../TradeInput/components/MaxSlippage'
import { SwapperIcon } from '../../TradeInput/components/SwapperIcon/SwapperIcon'
import { useTradeReceiveAddress } from '../../TradeInput/hooks/useTradeReceiveAddress'
-const parseAmountDisplayMeta = (items: AmountDisplayMeta[]) => {
+type ProtocolFee = {
+ assetId: AssetId | undefined
+ chainName: string | undefined
+ amountCryptoPrecision: string
+ symbol: string
+}
+
+const parseAmountDisplayMeta = (items: AmountDisplayMeta[]): ProtocolFee[] => {
return items
.filter(({ amountCryptoBaseUnit }) => bnOrZero(amountCryptoBaseUnit).gt(0))
.map(({ amountCryptoBaseUnit, asset }: AmountDisplayMeta) => ({
- symbol: asset.symbol,
+ assetId: asset.assetId,
chainName: getChainAdapterManager().get(asset.chainId)?.getDisplayName(),
amountCryptoPrecision: fromBaseUnit(amountCryptoBaseUnit, asset.precision),
+ symbol: asset.symbol,
}))
}
@@ -104,12 +114,12 @@ export const TradeConfirmSummary = () => {
const affiliateFeeAfterDiscountUserCurrency = useAppSelector(
selectTradeQuoteAffiliateFeeAfterDiscountUserCurrency,
)
+ const marketDataUserCurrency = useAppSelector(selectMarketDataUserCurrency)
const totalNetworkFeeFiatUserCurrency = useAppSelector(selectTotalNetworkFeeUserCurrency)
const tradeQuoteFirstHop = useAppSelector(selectFirstHop)
const translate = useTranslate()
const { priceImpactPercentage } = usePriceImpact(activeQuote)
const { isLoading } = useIsApprovalInitiallyNeeded()
- const redColor = useColorModeValue('red.500', 'red.300')
const greenColor = useColorModeValue('green.600', 'green.200')
const { manualReceiveAddress, walletReceiveAddress } = useTradeReceiveAddress()
const [showFeeModal, setShowFeeModal] = useState(false)
@@ -126,7 +136,24 @@ export const TradeConfirmSummary = () => {
const hasIntermediaryTransactionOutputs =
intermediaryTransactionOutputsParsed && intermediaryTransactionOutputsParsed.length > 0
const protocolFeesParsed = totalProtocolFees
- ? parseAmountDisplayMeta(Object.values(totalProtocolFees).filter(isSome))
+ ? Object.values(
+ parseAmountDisplayMeta(Object.values(totalProtocolFees).filter(isSome)).reduce(
+ (acc, fee) => {
+ const key = `${fee.assetId}-${fee.chainName}`
+ if (acc[key]) {
+ // If we already have this symbol+chain combination, add the amounts
+ acc[key].amountCryptoPrecision = bnOrZero(acc[key].amountCryptoPrecision)
+ .plus(fee.amountCryptoPrecision)
+ .toString()
+ } else {
+ // First time seeing this symbol+chain combination
+ acc[key] = { ...fee }
+ }
+ return acc
+ },
+ {} as Record,
+ ),
+ )
: undefined
const hasProtocolFees = protocolFeesParsed && protocolFeesParsed.length > 0
@@ -191,13 +218,25 @@ export const TradeConfirmSummary = () => {
- {protocolFeesParsed?.map(({ amountCryptoPrecision, symbol }) => (
-
+ {protocolFeesParsed?.map(({ amountCryptoPrecision, assetId, symbol }) => (
+
+
+ {assetId && (
+
+ )}
+
))}
diff --git a/src/components/MultiHopTrade/components/TradeConfirm/hooks/useActiveTradeAllowance.tsx b/src/components/MultiHopTrade/components/TradeConfirm/hooks/useActiveTradeAllowance.tsx
index 7bbc4d92830..ee1dd188934 100644
--- a/src/components/MultiHopTrade/components/TradeConfirm/hooks/useActiveTradeAllowance.tsx
+++ b/src/components/MultiHopTrade/components/TradeConfirm/hooks/useActiveTradeAllowance.tsx
@@ -1,5 +1,5 @@
import type { TradeQuoteStep } from '@shapeshiftoss/swapper'
-import { useCallback, useMemo } from 'react'
+import { useMemo } from 'react'
import { useGetTradeQuotes } from 'components/MultiHopTrade/hooks/useGetTradeQuotes/useGetTradeQuotes'
import { AllowanceType } from 'hooks/queries/useApprovalFees'
import { selectHopExecutionMetadata } from 'state/slices/tradeQuoteSlice/selectors'
@@ -22,7 +22,6 @@ export const useActiveTradeAllowance = ({
isExactAllowance,
activeTradeId,
}: UseSignAllowanceApprovalProps) => {
- // TODO: confirm this is actually needed in the new flow
// DO NOT REMOVE ME. Fetches and upserts permit2 quotes at pre-permit2-signing time
useGetTradeQuotes()
@@ -91,25 +90,9 @@ export const useActiveTradeAllowance = ({
allowanceReset.isInitiallyRequired,
)
- const handleSignAllowanceApproval = useCallback(async () => {
- try {
- await approveMutation.mutateAsync()
- } catch (error) {
- console.error(error)
- }
- }, [approveMutation])
-
- const handleSignAllowanceReset = useCallback(async () => {
- try {
- await allowanceResetMutation.mutateAsync()
- } catch (error) {
- console.error(error)
- }
- }, [allowanceResetMutation])
-
return {
- handleSignAllowanceApproval,
- handleSignAllowanceReset,
+ handleSignAllowanceApproval: approveMutation.mutate,
+ handleSignAllowanceReset: allowanceResetMutation.mutate,
isAllowanceApprovalLoading,
isAllowanceApprovalPending: approveMutation.isPending,
isAllowanceResetLoading,