From 3576313f0ce30e6f0d8a05fc03da03c590ff3620 Mon Sep 17 00:00:00 2001
From: Fionna Chan <13184582+fionnachan@users.noreply.github.com>
Date: Tue, 3 Sep 2024 18:38:47 +0800
Subject: [PATCH] refactor: move customFeeTokenBalances into a hook (#1872)
---
.../TransferPanel/TransferPanelMain.tsx | 32 +-------
.../DestinationNetworkBox.tsx | 20 ++---
.../TransferPanelMain/SourceNetworkBox.tsx | 9 +--
.../TransferPanelMain/useMaxAmount.ts | 47 ++++-------
.../useNativeCurrencyBalances.ts | 48 ++++++++++++
.../TransferPanel/TransferPanelMainInput.tsx | 78 +++++--------------
6 files changed, 91 insertions(+), 143 deletions(-)
create mode 100644 packages/arb-token-bridge-ui/src/components/TransferPanel/TransferPanelMain/useNativeCurrencyBalances.ts
diff --git a/packages/arb-token-bridge-ui/src/components/TransferPanel/TransferPanelMain.tsx b/packages/arb-token-bridge-ui/src/components/TransferPanel/TransferPanelMain.tsx
index 2d9d797c46..689fca821b 100644
--- a/packages/arb-token-bridge-ui/src/components/TransferPanel/TransferPanelMain.tsx
+++ b/packages/arb-token-bridge-ui/src/components/TransferPanel/TransferPanelMain.tsx
@@ -27,7 +27,6 @@ import { useNetworksRelationship } from '../../hooks/useNetworksRelationship'
import { TransferDisabledDialog } from './TransferDisabledDialog'
import { getBridgeUiConfigForChain } from '../../util/bridgeUiConfig'
import { useUpdateUSDCTokenData } from './TransferPanelMain/hooks'
-import { Balances } from '../../hooks/TransferPanel/useSelectedTokenBalances'
import { useBalances } from '../../hooks/useBalances'
import { DestinationNetworkBox } from './TransferPanelMain/DestinationNetworkBox'
import { SourceNetworkBox } from './TransferPanelMain/SourceNetworkBox'
@@ -268,13 +267,7 @@ export function TransferPanelMain() {
const destinationAddressOrWalletAddress = destinationAddress || walletAddress
- const {
- ethParentBalance,
- erc20ParentBalances,
- updateErc20ParentBalances,
- ethChildBalance,
- updateErc20ChildBalances
- } = useBalances()
+ const { updateErc20ParentBalances, updateErc20ChildBalances } = useBalances()
const { updateUSDCBalances } = useUpdateUSDCBalances({
walletAddress: destinationAddressOrWalletAddress
@@ -319,19 +312,6 @@ export function TransferPanelMain() {
isTeleportMode
])
- // TODO: move into a hook (FS-714)
- // when customFeeTokenBalances is moved to an independent hook file, use `setAmount` directly in useMaxAmount and do not pass `customFeeTokenBalances` as a prop
- const customFeeTokenBalances: Balances = useMemo(() => {
- if (!nativeCurrency.isCustom) {
- return { parentBalance: ethParentBalance, childBalance: ethChildBalance }
- }
-
- return {
- parentBalance: erc20ParentBalances?.[nativeCurrency.address] ?? null,
- childBalance: ethChildBalance
- }
- }, [nativeCurrency, ethParentBalance, ethChildBalance, erc20ParentBalances])
-
const showUSDCSpecificInfo =
!isTeleportMode &&
((isTokenMainnetUSDC(selectedToken?.address) && isArbitrumOne) ||
@@ -348,17 +328,11 @@ export function TransferPanelMain() {
return (
-
+
-
+
diff --git a/packages/arb-token-bridge-ui/src/components/TransferPanel/TransferPanelMain/DestinationNetworkBox.tsx b/packages/arb-token-bridge-ui/src/components/TransferPanel/TransferPanelMain/DestinationNetworkBox.tsx
index 4df5ac0201..d860364eb7 100644
--- a/packages/arb-token-bridge-ui/src/components/TransferPanel/TransferPanelMain/DestinationNetworkBox.tsx
+++ b/packages/arb-token-bridge-ui/src/components/TransferPanel/TransferPanelMain/DestinationNetworkBox.tsx
@@ -18,22 +18,18 @@ import { useBalances } from '../../../hooks/useBalances'
import { CommonAddress } from '../../../util/CommonAddressUtils'
import { isNetwork } from '../../../util/networks'
import { EstimatedGas } from '../EstimatedGas'
-import {
- Balances,
- useSelectedTokenBalances
-} from '../../../hooks/TransferPanel/useSelectedTokenBalances'
+import { useSelectedTokenBalances } from '../../../hooks/TransferPanel/useSelectedTokenBalances'
import { useNativeCurrency } from '../../../hooks/useNativeCurrency'
import { useDialog } from '../../common/Dialog'
import {
NetworkButton,
NetworkSelectionContainer
} from '../../common/NetworkSelectionContainer'
+import { useNativeCurrencyBalances } from './useNativeCurrencyBalances'
function DestinationNetworkBalance({
- customFeeTokenBalances,
showUsdcSpecificInfo
}: {
- customFeeTokenBalances: Balances
showUsdcSpecificInfo: boolean
}) {
const {
@@ -46,6 +42,7 @@ function DestinationNetworkBalance({
const { ethParentBalance, ethChildBalance, erc20ChildBalances } =
useBalances()
+ const nativeCurrencyBalances = useNativeCurrencyBalances()
const selectedTokenBalances = useSelectedTokenBalances()
const nativeCurrency = useNativeCurrency({ provider: childChainProvider })
@@ -97,11 +94,7 @@ function DestinationNetworkBalance({
return (
@@ -110,7 +103,7 @@ function DestinationNetworkBalance({
return (
@@ -118,10 +111,8 @@ function DestinationNetworkBalance({
}
export function DestinationNetworkBox({
- customFeeTokenBalances,
showUsdcSpecificInfo
}: {
- customFeeTokenBalances: Balances
showUsdcSpecificInfo: boolean
}) {
const { address: walletAddress } = useAccount()
@@ -148,7 +139,6 @@ export function DestinationNetworkBox({
{destinationAddressOrWalletAddress &&
utils.isAddress(destinationAddressOrWalletAddress) && (
)}
diff --git a/packages/arb-token-bridge-ui/src/components/TransferPanel/TransferPanelMain/SourceNetworkBox.tsx b/packages/arb-token-bridge-ui/src/components/TransferPanel/TransferPanelMain/SourceNetworkBox.tsx
index b41ba30d1c..714e85102b 100644
--- a/packages/arb-token-bridge-ui/src/components/TransferPanel/TransferPanelMain/SourceNetworkBox.tsx
+++ b/packages/arb-token-bridge-ui/src/components/TransferPanel/TransferPanelMain/SourceNetworkBox.tsx
@@ -11,7 +11,6 @@ import { useAppState } from '../../../state'
import { useNetworks } from '../../../hooks/useNetworks'
import { useNativeCurrency } from '../../../hooks/useNativeCurrency'
import { useNetworksRelationship } from '../../../hooks/useNetworksRelationship'
-import { Balances } from '../../../hooks/TransferPanel/useSelectedTokenBalances'
import {
ETH_BALANCE_ARTICLE_LINK,
USDC_LEARN_MORE_LINK
@@ -32,10 +31,8 @@ import { useSelectedTokenDecimals } from '../../../hooks/TransferPanel/useSelect
import { useBalanceOnSourceChain } from '../../../hooks/useBalanceOnSourceChain'
export function SourceNetworkBox({
- customFeeTokenBalances,
showUsdcSpecificInfo
}: {
- customFeeTokenBalances: Balances
showUsdcSpecificInfo: boolean
}) {
const [networks] = useNetworks()
@@ -47,9 +44,7 @@ export function SourceNetworkBox({
const nativeCurrency = useNativeCurrency({ provider: childChainProvider })
const [{ amount, amount2 }] = useArbQueryParams()
const { setAmount, setAmount2 } = useSetInputAmount()
- const { maxAmount, maxAmount2 } = useMaxAmount({
- customFeeTokenBalances
- })
+ const { maxAmount, maxAmount2 } = useMaxAmount()
const [sourceNetworkSelectionDialogProps, openSourceNetworkSelectionDialog] =
useDialog()
const isBatchTransferSupported = useIsBatchTransferSupported()
@@ -124,7 +119,6 @@ export function SourceNetworkBox({
maxAmount={maxAmount}
isMaxAmount={isMaxAmount}
decimals={decimals}
- customFeeTokenBalances={customFeeTokenBalances}
/>
{isBatchTransferSupported && (
@@ -137,7 +131,6 @@ export function SourceNetworkBox({
maxAmount={maxAmount2}
isMaxAmount={isMaxAmount2}
decimals={nativeCurrency.decimals}
- customFeeTokenBalances={customFeeTokenBalances}
/>
)}
diff --git a/packages/arb-token-bridge-ui/src/components/TransferPanel/TransferPanelMain/useMaxAmount.ts b/packages/arb-token-bridge-ui/src/components/TransferPanel/TransferPanelMain/useMaxAmount.ts
index b9c6d9a14d..f7d6d6b4ce 100644
--- a/packages/arb-token-bridge-ui/src/components/TransferPanel/TransferPanelMain/useMaxAmount.ts
+++ b/packages/arb-token-bridge-ui/src/components/TransferPanel/TransferPanelMain/useMaxAmount.ts
@@ -4,20 +4,13 @@ import { utils } from 'ethers'
import { useNetworksRelationship } from '../../../hooks/useNetworksRelationship'
import { useNetworks } from '../../../hooks/useNetworks'
import { useAppState } from '../../../state'
-import {
- Balances,
- useSelectedTokenBalances
-} from '../../../hooks/TransferPanel/useSelectedTokenBalances'
+import { useSelectedTokenBalances } from '../../../hooks/TransferPanel/useSelectedTokenBalances'
import { defaultErc20Decimals } from '../../../defaults'
import { useGasSummary } from '../../../hooks/TransferPanel/useGasSummary'
import { useNativeCurrency } from '../../../hooks/useNativeCurrency'
-import { useBalances } from '../../../hooks/useBalances'
+import { useNativeCurrencyBalances } from './useNativeCurrencyBalances'
-export function useMaxAmount({
- customFeeTokenBalances
-}: {
- customFeeTokenBalances: Balances
-}) {
+export function useMaxAmount() {
const {
app: { selectedToken }
} = useAppState()
@@ -26,36 +19,30 @@ export function useMaxAmount({
const { childChainProvider, isDepositMode } =
useNetworksRelationship(networks)
const nativeCurrency = useNativeCurrency({ provider: childChainProvider })
- const { ethParentBalance, ethChildBalance } = useBalances()
const { estimatedParentChainGasFees, estimatedChildChainGasFees } =
useGasSummary()
+ const nativeCurrencyBalances = useNativeCurrencyBalances()
+
const nativeCurrencyMaxAmount = useMemo(() => {
- const customFeeTokenParentBalance = customFeeTokenBalances.parentBalance
+ const nativeCurrencySourceBalance = nativeCurrencyBalances.sourceBalance
+
+ if (!nativeCurrencySourceBalance) {
+ return undefined
+ }
+
// For custom fee token deposits, we can set the max amount, as the fees will be paid in ETH
- if (
- nativeCurrency.isCustom &&
- isDepositMode &&
- customFeeTokenParentBalance
- ) {
+ if (nativeCurrency.isCustom && isDepositMode) {
return utils.formatUnits(
- customFeeTokenParentBalance,
+ nativeCurrencySourceBalance,
nativeCurrency.decimals
)
}
// ETH deposits and ETH/custom fee token withdrawals
- const nativeCurrencyBalance = isDepositMode
- ? ethParentBalance
- : ethChildBalance
-
- if (!nativeCurrencyBalance) {
- return undefined
- }
-
const nativeCurrencyBalanceFormatted = utils.formatUnits(
- nativeCurrencyBalance,
+ nativeCurrencySourceBalance,
nativeCurrency.decimals
)
@@ -80,14 +67,12 @@ export function useMaxAmount({
return nativeCurrencyBalanceFormatted
}, [
- customFeeTokenBalances.parentBalance,
estimatedChildChainGasFees,
estimatedParentChainGasFees,
- ethChildBalance,
- ethParentBalance,
isDepositMode,
nativeCurrency.decimals,
- nativeCurrency.isCustom
+ nativeCurrency.isCustom,
+ nativeCurrencyBalances.sourceBalance
])
const maxAmount = useMemo(() => {
diff --git a/packages/arb-token-bridge-ui/src/components/TransferPanel/TransferPanelMain/useNativeCurrencyBalances.ts b/packages/arb-token-bridge-ui/src/components/TransferPanel/TransferPanelMain/useNativeCurrencyBalances.ts
new file mode 100644
index 0000000000..799148769a
--- /dev/null
+++ b/packages/arb-token-bridge-ui/src/components/TransferPanel/TransferPanelMain/useNativeCurrencyBalances.ts
@@ -0,0 +1,48 @@
+import { useMemo } from 'react'
+import { BigNumber } from 'ethers'
+
+import { useNativeCurrency } from '../../../hooks/useNativeCurrency'
+import { useNetworks } from '../../../hooks/useNetworks'
+import { useNetworksRelationship } from '../../../hooks/useNetworksRelationship'
+import { useBalances } from '../../../hooks/useBalances'
+
+export function useNativeCurrencyBalances(): {
+ sourceBalance: BigNumber | null
+ destinationBalance: BigNumber | null
+} {
+ const [networks] = useNetworks()
+ const { childChainProvider, isDepositMode } =
+ useNetworksRelationship(networks)
+ const nativeCurrency = useNativeCurrency({ provider: childChainProvider })
+
+ const { ethParentBalance, erc20ParentBalances, ethChildBalance } =
+ useBalances()
+
+ return useMemo(() => {
+ if (!nativeCurrency.isCustom) {
+ return {
+ sourceBalance: isDepositMode ? ethParentBalance : ethChildBalance,
+ destinationBalance: isDepositMode ? ethChildBalance : ethParentBalance
+ }
+ }
+
+ const customFeeTokenParentBalance =
+ erc20ParentBalances?.[nativeCurrency.address] ?? null
+ const customFeeTokenChildBalance = ethChildBalance
+
+ return {
+ sourceBalance: isDepositMode
+ ? customFeeTokenParentBalance
+ : customFeeTokenChildBalance,
+ destinationBalance: isDepositMode
+ ? customFeeTokenChildBalance
+ : customFeeTokenParentBalance
+ }
+ }, [
+ nativeCurrency,
+ erc20ParentBalances,
+ ethChildBalance,
+ isDepositMode,
+ ethParentBalance
+ ])
+}
diff --git a/packages/arb-token-bridge-ui/src/components/TransferPanel/TransferPanelMainInput.tsx b/packages/arb-token-bridge-ui/src/components/TransferPanel/TransferPanelMainInput.tsx
index c10a404096..a58e79aec7 100644
--- a/packages/arb-token-bridge-ui/src/components/TransferPanel/TransferPanelMainInput.tsx
+++ b/packages/arb-token-bridge-ui/src/components/TransferPanel/TransferPanelMainInput.tsx
@@ -10,12 +10,8 @@ import { useMemo } from 'react'
import { TokenButton, TokenButtonOptions } from './TokenButton'
import { useNetworks } from '../../hooks/useNetworks'
import { useNetworksRelationship } from '../../hooks/useNetworksRelationship'
-import {
- Balances,
- useSelectedTokenBalances
-} from '../../hooks/TransferPanel/useSelectedTokenBalances'
+import { useSelectedTokenBalances } from '../../hooks/TransferPanel/useSelectedTokenBalances'
import { useAppState } from '../../state'
-import { useBalances } from '../../hooks/useBalances'
import { TransferReadinessRichErrorMessage } from './useTransferReadinessUtils'
import { ExternalLink } from '../common/ExternalLink'
import { useTransferDisabledDialogStore } from './TransferDisabledDialog'
@@ -24,59 +20,39 @@ import { useNativeCurrency } from '../../hooks/useNativeCurrency'
import { Loader } from '../common/atoms/Loader'
import { sanitizeAmountQueryParam } from '../../hooks/useArbQueryParams'
import { truncateExtraDecimals } from '../../util/NumberUtils'
+import { useNativeCurrencyBalances } from './TransferPanelMain/useNativeCurrencyBalances'
function MaxButton({
- customFeeTokenBalances,
className = '',
...rest
-}: React.ButtonHTMLAttributes & {
- customFeeTokenBalances: Balances
-}) {
+}: React.ButtonHTMLAttributes) {
const {
app: { selectedToken }
} = useAppState()
const [networks] = useNetworks()
- const { isDepositMode, childChainProvider } =
- useNetworksRelationship(networks)
+ const { isDepositMode } = useNetworksRelationship(networks)
- const { ethParentBalance, ethChildBalance } = useBalances()
const selectedTokenBalances = useSelectedTokenBalances()
- const nativeCurrency = useNativeCurrency({ provider: childChainProvider })
+ const nativeCurrencyBalances = useNativeCurrencyBalances()
const maxButtonVisible = useMemo(() => {
- const ethBalance = isDepositMode ? ethParentBalance : ethChildBalance
- const customFeeTokenBalance = isDepositMode
- ? customFeeTokenBalances.parentBalance
- : customFeeTokenBalances.childBalance
+ const nativeCurrencySourceBalance = nativeCurrencyBalances.sourceBalance
+
const tokenBalance = isDepositMode
? selectedTokenBalances.parentBalance
: selectedTokenBalances.childBalance
if (selectedToken) {
- if (!tokenBalance) {
- return false
- }
-
- return !tokenBalance.isZero()
+ return tokenBalance && !tokenBalance.isZero()
}
- if (nativeCurrency.isCustom) {
- return customFeeTokenBalance && !customFeeTokenBalance.isZero()
- }
-
- if (!ethBalance) {
- return false
- }
-
- return !ethBalance.isZero()
+ return nativeCurrencySourceBalance && !nativeCurrencySourceBalance.isZero()
}, [
+ nativeCurrencyBalances.sourceBalance,
isDepositMode,
- ethParentBalance,
- ethChildBalance,
- customFeeTokenBalances,
- selectedTokenBalances,
- selectedToken,
- nativeCurrency.isCustom
+ selectedTokenBalances.parentBalance,
+ selectedTokenBalances.childBalance,
+ selectedToken
])
if (!maxButtonVisible) {
@@ -98,10 +74,8 @@ function MaxButton({
}
function SourceChainTokenBalance({
- customFeeTokenBalances,
balanceOverride
}: {
- customFeeTokenBalances: Balances
balanceOverride?: AmountInputOptions['balance']
}) {
const {
@@ -111,7 +85,7 @@ function SourceChainTokenBalance({
const { isDepositMode, childChainProvider } =
useNetworksRelationship(networks)
- const { ethParentBalance, ethChildBalance } = useBalances()
+ const nativeCurrencyBalances = useNativeCurrencyBalances()
const selectedTokenBalances = useSelectedTokenBalances()
const nativeCurrency = useNativeCurrency({ provider: childChainProvider })
@@ -120,17 +94,9 @@ function SourceChainTokenBalance({
? selectedTokenBalances.parentBalance
: selectedTokenBalances.childBalance
- const ethBalance = isDepositMode ? ethParentBalance : ethChildBalance
- const customFeeTokenBalance = isDepositMode
- ? customFeeTokenBalances.parentBalance
- : customFeeTokenBalances.childBalance
-
- const nativeCurrencyBalance = nativeCurrency.isCustom
- ? customFeeTokenBalance
- : ethBalance
-
const balance =
- balanceOverride ?? (selectedToken ? tokenBalance : nativeCurrencyBalance)
+ balanceOverride ??
+ (selectedToken ? tokenBalance : nativeCurrencyBalances.sourceBalance)
const formattedBalance = balance
? formatAmount(balance, {
@@ -238,7 +204,6 @@ export type TransferPanelMainInputProps =
maxButtonOnClick: React.ButtonHTMLAttributes['onClick']
value: string
options?: AmountInputOptions
- customFeeTokenBalances: Balances
maxAmount: string | undefined
isMaxAmount: boolean
decimals: number
@@ -254,7 +219,6 @@ export const TransferPanelMainInput = React.memo(
isMaxAmount,
decimals,
options,
- customFeeTokenBalances,
...rest
}: TransferPanelMainInputProps) => {
const [localValue, setLocalValue] = useState(value)
@@ -310,14 +274,8 @@ export const TransferPanelMainInput = React.memo(