From 88c59392520064c0ca4aaf1e70663e157cbc974b Mon Sep 17 00:00:00 2001 From: Douglas Daniel Date: Mon, 14 Mar 2022 16:30:30 -0600 Subject: [PATCH] feat(wallet): Multichain Support in Portfolio UI --- .../browser/asset_ratio_response_parser.cc | 2 +- .../browser/asset_ratio_service_unittest.cc | 2 +- .../browser/brave_wallet_constants.h | 38 +-- .../browser/brave_wallet_service.cc | 4 +- .../brave_wallet/common/brave_wallet.mojom | 1 + .../brave_wallet/common/eth_request_helper.cc | 2 +- .../assets/asset-icons/sol-asset-icon.svg | 27 +- .../common/actions/wallet_actions.ts | 12 +- .../brave_wallet_ui/common/async/handlers.ts | 64 +++-- .../brave_wallet_ui/common/async/lib.ts | 249 +++++++++++++----- .../brave_wallet_ui/common/constants/mocks.ts | 12 +- .../common/hooks/assets-management.test.ts | 8 +- .../common/hooks/assets-management.ts | 9 +- .../common/hooks/assets.test.ts | 4 +- .../brave_wallet_ui/common/hooks/assets.ts | 28 +- .../brave_wallet_ui/common/hooks/balance.ts | 16 +- .../common/hooks/select-preset.test.ts | 4 +- .../common/hooks/select-preset.ts | 4 +- .../brave_wallet_ui/common/hooks/swap.test.ts | 37 ++- .../brave_wallet_ui/common/hooks/swap.ts | 3 +- .../common/hooks/transaction-parser.test.ts | 24 +- .../common/hooks/transaction-parser.ts | 2 +- .../common/reducers/wallet_reducer.ts | 60 +++-- .../accounts-assets-networks/index.tsx | 5 +- .../select-account-with-header/index.tsx | 15 +- .../components/buy-send-swap/tabs/buy-tab.tsx | 5 +- .../buy-send-swap/tabs/send-tab.tsx | 5 +- .../buy-send-swap/tabs/swap-tab.tsx | 5 +- .../desktop/asset-watchlist-item/index.tsx | 16 +- .../edit-visible-assets-modal/index.tsx | 12 +- .../desktop/portfolio-asset-item/index.tsx | 24 +- .../desktop/views/accounts/index.tsx | 43 ++- .../components/desktop/views/crypto/index.tsx | 14 +- .../accounts-and-transctions-list/index.tsx | 6 +- .../components/token-lists/index.tsx | 12 +- .../desktop/views/portfolio/index.tsx | 65 +++-- .../desktop/views/portfolio/style.ts | 17 +- .../extension/assets-panel/index.tsx | 8 +- .../extension/connected-panel/index.tsx | 4 +- .../shared/create-network-icon/index.tsx | 14 +- .../shared/select-account/index.tsx | 14 +- .../shared/select-network/index.tsx | 14 +- components/brave_wallet_ui/constants/types.ts | 41 ++- .../brave_wallet_ui/options/asset-options.ts | 56 ++-- .../page/async/wallet_page_async_handler.ts | 5 +- components/brave_wallet_ui/page/container.tsx | 26 +- .../brave_wallet_ui/panel/container.tsx | 21 +- .../stories/screens/buy-send-swap.tsx | 5 +- .../stories/screens/crypto-story-view.tsx | 4 +- .../stories/wallet-concept.tsx | 7 +- .../stories/wallet-extension-panels.tsx | 22 +- .../brave_wallet_ui/utils/api-utils.test.ts | 10 +- components/brave_wallet_ui/utils/api-utils.ts | 14 +- .../utils/network-utils.test.ts | 45 +++- .../brave_wallet_ui/utils/network-utils.ts | 33 ++- 55 files changed, 823 insertions(+), 376 deletions(-) diff --git a/components/brave_wallet/browser/asset_ratio_response_parser.cc b/components/brave_wallet/browser/asset_ratio_response_parser.cc index 59785240574f..401a17ff1fee 100644 --- a/components/brave_wallet/browser/asset_ratio_response_parser.cc +++ b/components/brave_wallet/browser/asset_ratio_response_parser.cc @@ -419,7 +419,7 @@ mojom::BlockchainTokenPtr ParseTokenInfo(const std::string& json) { return mojom::BlockchainToken::New(eth_addr.ToChecksumAddress(), *name, "" /* logo */, is_erc20, is_erc721, - *symbol, decimals, true, "", ""); + *symbol, decimals, true, "", "", ""); } } // namespace brave_wallet diff --git a/components/brave_wallet/browser/asset_ratio_service_unittest.cc b/components/brave_wallet/browser/asset_ratio_service_unittest.cc index b7cb0c1048c2..38bfcfa15255 100644 --- a/components/brave_wallet/browser/asset_ratio_service_unittest.cc +++ b/components/brave_wallet/browser/asset_ratio_service_unittest.cc @@ -418,7 +418,7 @@ TEST_F(AssetRatioServiceUnitTest, GetTokenInfo) { GetTokenInfo("0xdac17f958d2ee523a2206206994597c13d831ec7", mojom::BlockchainToken::New( "0xdAC17F958D2ee523a2206206994597C13D831ec7", "Tether USD", - "", true, false, "USDT", 6, true, "", "")); + "", true, false, "USDT", 6, true, "", "", "")); SetInterceptor("unexpected response"); GetTokenInfo("0xdac17f958d2ee523a2206206994597c13d831ec7", nullptr); diff --git a/components/brave_wallet/browser/brave_wallet_constants.h b/components/brave_wallet/browser/brave_wallet_constants.h index 95e0888d6aff..ffb9b918d6f4 100644 --- a/components/brave_wallet/browser/brave_wallet_constants.h +++ b/components/brave_wallet/browser/brave_wallet_constants.h @@ -29,42 +29,42 @@ constexpr int32_t kAutoLockMinutesMax = 10080; // List of assets from Wyre, available to buy static base::NoDestructor> kBuyTokens( {{"0x0D8775F648430679A709E98d2b0Cb6250d2887EF", "Basic Attention Token", - "bat.png", true, false, "BAT", 18, true, "", ""}, - {"", "Ethereum", "", false, false, "ETH", 18, true, "", ""}, + "bat.png", true, false, "BAT", 18, true, "", "", ""}, + {"", "Ethereum", "", false, false, "ETH", 18, true, "", "", ""}, {"0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48", "USD Coin", "usdc.png", - true, false, "USDC", 6, true, "", ""}, + true, false, "USDC", 6, true, "", "", ""}, {"0x6B175474E89094C44Da98b954EedeAC495271d0F", "DAI", "dai.png", true, - false, "DAI", 18, true, "", ""}, + false, "DAI", 18, true, "", "", ""}, {"0x7Fc66500c84A76Ad7e9c93437bFc5Ac33E2DDaE9", "AAVE", "AAVE.png", true, - false, "AAVE", 18, true, "", ""}, + false, "AAVE", 18, true, "", "", ""}, {"0x4Fabb145d64652a948d72533023f6E7A623C7C53", "Binance USD", "busd.png", - true, false, "BUSD", 18, true, "", ""}, + true, false, "BUSD", 18, true, "", "", ""}, {"0xc00e94Cb662C3520282E6f5717214004A7f26888", "Compound", "comp.png", - true, false, "Comp", 18, true, "", ""}, + true, false, "Comp", 18, true, "", "", ""}, {"0xD533a949740bb3306d119CC777fa900bA034cd52", "Curve", "curve.png", true, - false, "CRV", 18, true, "", ""}, + false, "CRV", 18, true, "", "", ""}, {"0x056Fd409E1d7A124BD7017459dFEa2F387b6d5Cd", "Gemini Dollar", "gusd.png", - true, false, "GUSD", 2, true, "", ""}, + true, false, "GUSD", 2, true, "", "", ""}, {"0x514910771AF9Ca656af840dff83E8264EcF986CA", "Chainlink", - "chainlink.png", true, false, "LINK", 18, true, "", ""}, + "chainlink.png", true, false, "LINK", 18, true, "", "", ""}, {"0x9f8F72aA9304c8B593d555F12eF6589cC3A579A2", "Maker", "mkr.png", true, - false, "MKR", 18, true, "", ""}, + false, "MKR", 18, true, "", "", ""}, {"0xC011a73ee8576Fb46F5E1c5751cA3B9Fe0af2a6F", "Synthetix", - "synthetix.png", true, false, "SNX", 18, true, "", ""}, + "synthetix.png", true, false, "SNX", 18, true, "", "", ""}, {"0x04Fa0d235C4abf4BcF4787aF4CF447DE572eF828", "UMA", "UMA.png", true, - false, "UMA", 18, true, "", ""}, + false, "UMA", 18, true, "", "", ""}, {"0x1f9840a85d5aF5bf1D1762F925BDADdC4201F984", "Uniswap", "uni.png", true, - false, "UNI", 18, true, "", ""}, + false, "UNI", 18, true, "", "", ""}, {"0xA4Bdb11dc0a2bEC88d24A3aa1E6Bb17201112eBe", "Stably Dollar", "usds.png", - true, false, "USDS", 6, true, "", ""}, + true, false, "USDS", 6, true, "", "", ""}, {"0xdAC17F958D2ee523a2206206994597C13D831ec7", "Tether", "usdt.png", true, - false, "USDT", 6, true, "", ""}, + false, "USDT", 6, true, "", "", ""}, {"0x2260FAC5E5542a773Aa44fBCfeDf7C193bc2C599", "Wrapped Bitcoin", - "wbtc.png", true, false, "WBTC", 8, true, "", ""}, + "wbtc.png", true, false, "WBTC", 8, true, "", "", ""}, {"0x0bc529c00C6401aEF6D220BE8C6Ea1667F6Ad93e", "Year.Finance", "yfi.png", - true, false, "YFI", 18, true, "", ""}, + true, false, "YFI", 18, true, "", "", ""}, {"0x9043d4d51C9d2e31e3F169de4551E416970c27Ef", "Palm DAI", "pdai.png", - true, false, "PDAI", 18, true, "", ""}}); + true, false, "PDAI", 18, true, "", "", ""}}); const char kWalletBaseDirectory[] = "BraveWallet"; const char kImageSourceHost[] = "erc-token-images"; diff --git a/components/brave_wallet/browser/brave_wallet_service.cc b/components/brave_wallet/browser/brave_wallet_service.cc index 8174e095d805..04ee7c6b03dc 100644 --- a/components/brave_wallet/browser/brave_wallet_service.cc +++ b/components/brave_wallet/browser/brave_wallet_service.cc @@ -220,8 +220,10 @@ void BraveWalletService::GetUserAssets(const std::string& chain_id, std::vector result; for (const auto& token : tokens->GetList()) { mojom::BlockchainTokenPtr tokenPtr = ValueToBlockchainToken(token); - if (tokenPtr) + if (tokenPtr) { + tokenPtr->chain_id = chain_id; result.push_back(std::move(tokenPtr)); + } } std::move(callback).Run(std::move(result)); diff --git a/components/brave_wallet/common/brave_wallet.mojom b/components/brave_wallet/common/brave_wallet.mojom index c7267fb63311..b6859c5f4578 100644 --- a/components/brave_wallet/common/brave_wallet.mojom +++ b/components/brave_wallet/common/brave_wallet.mojom @@ -320,6 +320,7 @@ struct BlockchainToken { bool visible = true; string token_id = ""; // Non-empty for ERC721 tokens. string coingecko_id = ""; + string chain_id; }; // WebUI-side handler for requests from the browser. diff --git a/components/brave_wallet/common/eth_request_helper.cc b/components/brave_wallet/common/eth_request_helper.cc index c5af5c1c0b79..82d5a4dc0675 100644 --- a/components/brave_wallet/common/eth_request_helper.cc +++ b/components/brave_wallet/common/eth_request_helper.cc @@ -532,7 +532,7 @@ bool ParseWalletWatchAssetParams(const std::string& json, *token = mojom::BlockchainToken::New(eth_addr.ToChecksumAddress(), *symbol /* name */, logo, true, false, - *symbol, decimals, true, "", ""); + *symbol, decimals, true, "", "", ""); return true; } diff --git a/components/brave_wallet_ui/assets/asset-icons/sol-asset-icon.svg b/components/brave_wallet_ui/assets/asset-icons/sol-asset-icon.svg index a7707104fc47..44617bc05aa7 100644 --- a/components/brave_wallet_ui/assets/asset-icons/sol-asset-icon.svg +++ b/components/brave_wallet_ui/assets/asset-icons/sol-asset-icon.svg @@ -1,24 +1,9 @@ - - - - - - + + - - - - - - - - - - - - - - - + + + + diff --git a/components/brave_wallet_ui/common/actions/wallet_actions.ts b/components/brave_wallet_ui/common/actions/wallet_actions.ts index 97dc40ca5878..a1f015238f99 100644 --- a/components/brave_wallet_ui/common/actions/wallet_actions.ts +++ b/components/brave_wallet_ui/common/actions/wallet_actions.ts @@ -28,9 +28,7 @@ import { import { BraveWallet, WalletAccountType, - GetAllNetworksList, - GetAllTokensReturnInfo, - GetNativeAssetBalancesReturnInfo, + GetNativeAssetBalancesPayload, GetBlockchainTokenBalanceReturnInfo, PortfolioTokenHistoryAndInfo, SendTransactionParams, @@ -60,7 +58,7 @@ export const selectAccount = createAction('selectAccount') export const selectNetwork = createAction('selectNetwork') export const setNetwork = createAction('setNetwork') export const getAllNetworks = createAction('getAllNetworks') -export const setAllNetworks = createAction('getAllNetworks') +export const setAllNetworks = createAction('getAllNetworks') export const chainChangedEvent = createAction('chainChangedEvent') export const isEip1559Changed = createAction('isEip1559Changed') export const keyringCreated = createAction('keyringCreated') @@ -71,9 +69,9 @@ export const unlocked = createAction('unlocked') export const backedUp = createAction('backedUp') export const accountsChanged = createAction('accountsChanged') export const selectedAccountChanged = createAction('selectedAccountChanged') -export const setAllTokensList = createAction('setAllTokensList') +export const setAllTokensList = createAction('setAllTokensList') export const getAllTokensList = createAction('getAllTokensList') -export const nativeAssetBalancesUpdated = createAction('nativeAssetBalancesUpdated') +export const nativeAssetBalancesUpdated = createAction('nativeAssetBalancesUpdated') export const tokenBalancesUpdated = createAction('tokenBalancesUpdated') export const pricesUpdated = createAction('tokenBalancesUpdated') export const portfolioPriceHistoryUpdated = createAction('portfolioPriceHistoryUpdated') @@ -115,3 +113,5 @@ export const speedupTransaction = createAction('spe export const defaultCurrenciesUpdated = createAction('defaultCurrenciesUpdated') export const expandWalletNetworks = createAction('expandWalletNetworks') export const refreshBalancesAndPriceHistory = createAction('refreshBalancesAndPriceHistory') +export const setSelectedCoin = createAction('setSelectedCoin') +export const setDefaultNetworks = createAction('setDefaultNetworks') diff --git a/components/brave_wallet_ui/common/async/handlers.ts b/components/brave_wallet_ui/common/async/handlers.ts index d7d6bdaf7833..33fe27b512f2 100644 --- a/components/brave_wallet_ui/common/async/handlers.ts +++ b/components/brave_wallet_ui/common/async/handlers.ts @@ -30,7 +30,8 @@ import { SwapErrorResponse, WalletAccountType, WalletState, - WalletInfo + WalletInfo, + SupportedCoinTypes } from '../../constants/types' // Utils @@ -119,10 +120,6 @@ handler.on(WalletActions.initialize.getType(), async (store) => { await refreshWalletInfo(store) }) -handler.on(WalletActions.chainChangedEvent.getType(), async (store: Store, payload: ChainChangedEventPayloadType) => { - await refreshWalletInfo(store) -}) - handler.on(WalletActions.keyringCreated.getType(), async (store) => { await refreshWalletInfo(store) }) @@ -153,9 +150,12 @@ handler.on(WalletActions.accountsChanged.getType(), async (store) => { await updateAccountInfo(store) }) -handler.on(WalletActions.selectedAccountChanged.getType(), async (store) => { - await refreshWalletInfo(store) -}) +// Will use this observer selectedAccountChanged action again once +// selectedCoin is implemented here https://github.com/brave/brave-browser/issues/21465 + +// handler.on(WalletActions.selectedAccountChanged.getType(), async (store) => { +// await refreshWalletInfo(store) +// }) handler.on(WalletActions.defaultWalletChanged.getType(), async (store) => { await refreshWalletInfo(store) @@ -192,17 +192,31 @@ handler.on(WalletActions.removeFavoriteApp.getType(), async (store: Store, appIt await refreshWalletInfo(store) }) +handler.on(WalletActions.chainChangedEvent.getType(), async (store: Store, payload: ChainChangedEventPayloadType) => { + const keyringService = getAPIProxy().keyringService + const state = getWalletState(store) + const { accounts } = state + const selectedAccountAddress = await keyringService.getSelectedAccount(payload.coin) + const selectedAccount = accounts.find((account) => account.address === selectedAccountAddress.address) ?? accounts[0] + store.dispatch(WalletActions.setSelectedAccount(selectedAccount)) + store.dispatch(WalletActions.setSelectedCoin(payload.coin)) +}) + handler.on(WalletActions.selectNetwork.getType(), async (store: Store, payload: BraveWallet.NetworkInfo) => { const jsonRpcService = getAPIProxy().jsonRpcService - await jsonRpcService.setNetwork(payload.chainId, BraveWallet.CoinType.ETH) - await refreshWalletInfo(store) + await jsonRpcService.setNetwork(payload.chainId, payload.coin) + store.dispatch(WalletActions.setNetwork(payload)) }) handler.on(WalletActions.selectAccount.getType(), async (store: Store, payload: WalletAccountType) => { const { keyringService } = getAPIProxy() - + const state = getWalletState(store) + const { defaultNetworks } = state + const defaultCoinTypesNetwork = defaultNetworks.find((network) => network.coin === payload.coin) ?? defaultNetworks[0] await keyringService.setSelectedAccount(payload.address, payload.coin) + store.dispatch(WalletActions.setNetwork(defaultCoinTypesNetwork)) store.dispatch(WalletActions.setSelectedAccount(payload)) + store.dispatch(WalletActions.setSelectedCoin(payload.coin)) await store.dispatch(refreshTransactionHistory(payload.address)) }) @@ -243,15 +257,31 @@ handler.on(WalletActions.initialized.getType(), async (store: Store, payload: Wa handler.on(WalletActions.getAllNetworks.getType(), async (store) => { const jsonRpcService = getAPIProxy().jsonRpcService - const fullList = await jsonRpcService.getAllNetworks(BraveWallet.CoinType.ETH) - store.dispatch(WalletActions.setAllNetworks(fullList)) + + const getFullNetworkList = await Promise.all(SupportedCoinTypes.map(async (coin: BraveWallet.CoinType) => { + const networkList = await jsonRpcService.getAllNetworks(coin) + return networkList.networks + })) + const networkList = getFullNetworkList.flat(1) + store.dispatch(WalletActions.setAllNetworks(networkList)) }) handler.on(WalletActions.getAllTokensList.getType(), async (store) => { - const { blockchainRegistry, jsonRpcService } = getAPIProxy() - const { chainId } = await jsonRpcService.getChainId(BraveWallet.CoinType.ETH) - const fullList = await blockchainRegistry.getAllTokens(chainId) - store.dispatch(WalletActions.setAllTokensList(fullList)) + const state = getWalletState(store) + const { networkList } = state + const { blockchainRegistry } = getAPIProxy() + const getAllTokensList = await Promise.all(networkList.map(async (network) => { + const list = await blockchainRegistry.getAllTokens(network.chainId) + return list.tokens.map((token) => { + return { + ...token, + chainId: network.chainId, + logo: `chrome://erc-token-images/${token.logo}` + } + }) + })) + const allTokensList = getAllTokensList.flat(1) + store.dispatch(WalletActions.setAllTokensList(allTokensList)) }) handler.on(WalletActions.addUserAsset.getType(), async (store: Store, payload: AddUserAssetPayloadType) => { diff --git a/components/brave_wallet_ui/common/async/lib.ts b/components/brave_wallet_ui/common/async/lib.ts index 6800e5e174fc..77b03b766ec7 100644 --- a/components/brave_wallet_ui/common/async/lib.ts +++ b/components/brave_wallet_ui/common/async/lib.ts @@ -13,12 +13,14 @@ import { WalletAccountType, AccountInfo, BraveKeyrings, - GetBlockchainTokenInfoReturnInfo + GetBlockchainTokenInfoReturnInfo, + SupportedCoinTypes, + SupportedTestNetworks } from '../../constants/types' import * as WalletActions from '../actions/wallet_actions' // Utils -import { GetNetworkInfo } from '../../utils/network-utils' +import { getNetworkInfo, getNetworksByCoinType } from '../../utils/network-utils' import { GetTokenParam, GetFlattenedAccountBalances } from '../../utils/api-utils' import Amount from '../../utils/amount' @@ -163,25 +165,46 @@ export async function getIsSwapSupported (network: BraveWallet.NetworkInfo): Pro } export function refreshVisibleTokenInfo (currentNetwork: BraveWallet.NetworkInfo) { - return async (dispatch: Dispatch) => { + return async (dispatch: Dispatch, getState: () => State) => { const { braveWalletService } = getAPIProxy() + const { wallet: { networkList } } = getState() + + const getVisibleAssets = await Promise.all(networkList.map(async (network) => { + // Creates a network's Native Asset if not returned + const nativeAsset: BraveWallet.BlockchainToken = { + contractAddress: '', + decimals: network.decimals, + isErc20: false, + isErc721: false, + logo: network.iconUrls[0] ?? '', + name: network.symbolName, + symbol: network.symbol, + // MULTICHAIN: Change visible back to false once getUserAssets returns + // SOL and FIL by default. + visible: network.coin === BraveWallet.CoinType.SOL || network.coin === BraveWallet.CoinType.FIL, + tokenId: '', + coingeckoId: '', + chainId: network.chainId + } - // Selected Network's Native Asset - const nativeAsset: BraveWallet.BlockchainToken = { - contractAddress: '', - decimals: currentNetwork.decimals, - isErc20: false, - isErc721: false, - logo: currentNetwork.iconUrls[0] ?? '', - name: currentNetwork.symbolName, - symbol: currentNetwork.symbol, - visible: false, - tokenId: '', - coingeckoId: '' - } - - const visibleTokensInfo = await braveWalletService.getUserAssets(currentNetwork.chainId) - const visibleAssets: BraveWallet.BlockchainToken[] = visibleTokensInfo.tokens.length === 0 ? [nativeAsset] : visibleTokensInfo.tokens + // Get a list of user tokens for each coinType and network. + const getTokenList = network.chainId === BraveWallet.LOCALHOST_CHAIN_ID && + network.coin === BraveWallet.CoinType.SOL || network.coin === BraveWallet.CoinType.FIL + // Since LOCALHOST's chainId is shared between coinType networks, + // this check will make sure we create the correct Native Asset for + // that network. + ? { tokens: [nativeAsset] } // MULTICHAIN: We do not yet support getting userAssets for FIL and SOL + // Will be implemented here https://github.com/brave/brave-browser/issues/21547 + : await braveWalletService.getUserAssets(network.chainId) + + // Adds a logo and chainId to each token object + const tokenList = getTokenList.tokens.map((token) => ({ + ...token, + logo: token.symbol.toLowerCase() === 'sol' ? '' : `chrome://erc-token-images/${token.logo}` + })) as BraveWallet.BlockchainToken[] + return tokenList.length === 0 ? [nativeAsset] : tokenList + })) + const visibleAssets = getVisibleAssets.flat(1) await dispatch(WalletActions.setVisibleTokensInfo(visibleAssets)) } } @@ -189,29 +212,95 @@ export function refreshVisibleTokenInfo (currentNetwork: BraveWallet.NetworkInfo export function refreshBalances () { return async (dispatch: Dispatch, getState: () => State) => { const { jsonRpcService } = getAPIProxy() - const { wallet: { accounts, userVisibleTokensInfo, selectedNetwork } } = getState() + const { wallet: { accounts, userVisibleTokensInfo, networkList, defaultNetworks } } = getState() + + const emptyBalance = { + balance: '0x0', + error: 0, + errorMessage: '' + } + + const getNativeAssetsBalanceReturnInfos = await Promise.all(accounts.map(async (account) => { + const networks = getNetworksByCoinType(networkList, account.coin) + + return Promise.all(networks.map(async (network) => { + // MULTICHAIN: Backend needs to allow chainId for getSolanaBalance method + // Will be implemented here https://github.com/brave/brave-browser/issues/21695 + if (account.coin === BraveWallet.CoinType.SOL || account.coin === BraveWallet.CoinType.FIL) { + if (defaultNetworks.some(n => n.chainId === network.chainId)) { + // Get CoinType SOL balances + if (network.coin === BraveWallet.CoinType.SOL) { + const solBalanceInfo = await jsonRpcService.getSolanaBalance(account.address) + return { + ...solBalanceInfo, + balance: solBalanceInfo.balance.toString(), + chainId: network.chainId + } + } + + // Get CoinType FIL balances + if (network.coin === BraveWallet.CoinType.FIL) { + const balanceInfo = await jsonRpcService.getBalance(account.address, account.coin, network.chainId) + return { + ...balanceInfo, + chainId: network.chainId + } + } + } + + // Return emptyBalance for CoinType FIL or SOL if network + // is not included in defaultNetworks. Will update when this + // is implemented https://github.com/brave/brave-browser/issues/21695 + return { + ...emptyBalance, + chainId: network.chainId + } + } + + // LOCALHOST will return an error until a local instance is + // detected, we now will will return a 0 balance until it's detected. + if (network.chainId === BraveWallet.LOCALHOST_CHAIN_ID) { + const localhostBalanceInfo = await jsonRpcService.getBalance(account.address, account.coin, network.chainId) + const info = localhostBalanceInfo.error === 0 ? localhostBalanceInfo : emptyBalance + return { + ...info, + chainId: network.chainId + } + } - const getBalanceReturnInfos = await Promise.all(accounts.map(async (account) => { - const balanceInfo = await jsonRpcService.getBalance(account.address, account.coin, selectedNetwork.chainId) - return balanceInfo + // Get CoinType ETH balances + const balanceInfo = await jsonRpcService.getBalance(account.address, account.coin, network.chainId) + return { + ...balanceInfo, + chainId: network.chainId + } + })) })) + await dispatch(WalletActions.nativeAssetBalancesUpdated({ - balances: getBalanceReturnInfos + balances: getNativeAssetsBalanceReturnInfos })) const visibleTokens = userVisibleTokensInfo.filter(asset => asset.contractAddress !== '') - const getBlockchainTokenBalanceReturnInfos = await Promise.all(accounts.map(async (account) => { - return Promise.all(visibleTokens.map(async (token) => { - if (token.isErc721) { - return jsonRpcService.getERC721TokenBalance(token.contractAddress, token.tokenId ?? '', account.address, selectedNetwork.chainId) - } - return jsonRpcService.getERC20TokenBalance(token.contractAddress, account.address, selectedNetwork.chainId) - })) + const getBlockchainTokensBalanceReturnInfos = await Promise.all(accounts.map(async (account) => { + if (account.coin === BraveWallet.CoinType.ETH) { + return Promise.all(visibleTokens.map(async (token) => { + if (token.isErc721) { + return jsonRpcService.getERC721TokenBalance(token.contractAddress, token.tokenId ?? '', account.address, token?.chainId ?? '') + } + return jsonRpcService.getERC20TokenBalance(token.contractAddress, account.address, token?.chainId ?? '') + })) + } else { + // MULTICHAIN: We do not yet support getting + // token balances for SOL and FIL + // Will be implemented here https://github.com/brave/brave-browser/issues/21695 + return [] + } })) await dispatch(WalletActions.tokenBalancesUpdated({ - balances: getBlockchainTokenBalanceReturnInfos + balances: getBlockchainTokensBalanceReturnInfos })) } } @@ -219,19 +308,30 @@ export function refreshBalances () { export function refreshPrices () { return async (dispatch: Dispatch, getState: () => State) => { const { assetRatioService } = getAPIProxy() - const { wallet: { accounts, selectedPortfolioTimeline, selectedNetwork, userVisibleTokensInfo, defaultCurrencies } } = getState() - + const { wallet: { accounts, selectedPortfolioTimeline, userVisibleTokensInfo, defaultCurrencies, networkList } } = getState() const defaultFiatCurrency = defaultCurrencies.fiat.toLowerCase() - // Fetch native asset (ETH) price - const getNativeAssetPrice = await assetRatioService.getPrice([selectedNetwork.symbol.toLowerCase()], [defaultFiatCurrency], selectedPortfolioTimeline) - const nativeAssetPrice = getNativeAssetPrice.success ? getNativeAssetPrice.values.find((i) => i.toAsset === defaultFiatCurrency)?.price ?? '' : '' - // Update Token Prices + // Return if userVisibleTokensInfo is empty if (!userVisibleTokensInfo) { return } - const getTokenPrices = await Promise.all(GetFlattenedAccountBalances(accounts, userVisibleTokensInfo).map(async (token) => { + // Get prices for each networks native asset + const mainnetList = networkList.filter((network) => !SupportedTestNetworks.includes(network.chainId)) + const getNativeAssetPrices = await Promise.all(mainnetList.map(async (network) => { + const getNativeAssetPrice = await assetRatioService.getPrice([network.symbol.toLowerCase()], [defaultFiatCurrency], selectedPortfolioTimeline) + const nativeAssetPrice = getNativeAssetPrice.success ? getNativeAssetPrice.values.find((i) => i.toAsset === defaultFiatCurrency)?.price ?? '' : '' + return { + fromAsset: network.symbol.toLowerCase(), + toAsset: defaultFiatCurrency, + price: nativeAssetPrice, + assetTimeframeChange: '' + } + })) + + // Get prices for all other tokens on each network + const blockChainTokenInfo = userVisibleTokensInfo.filter((token) => token.contractAddress !== '') + const getTokenPrices = await Promise.all(GetFlattenedAccountBalances(accounts, blockChainTokenInfo).map(async (token) => { const emptyPrice = { fromAsset: token.token.symbol, toAsset: defaultFiatCurrency, @@ -241,7 +341,7 @@ export function refreshPrices () { // If a tokens balance is 0 we do not make an unnecessary api call for the price of that token const price = token.balance > 0 && token.token.isErc20 - ? await assetRatioService.getPrice([GetTokenParam(selectedNetwork, token.token)], [defaultFiatCurrency], selectedPortfolioTimeline) + ? await assetRatioService.getPrice([GetTokenParam(token.token)], [defaultFiatCurrency], selectedPortfolioTimeline) : { values: [{ ...emptyPrice, price: '0' }], success: true } const tokenPrice = { @@ -254,12 +354,7 @@ export function refreshPrices () { await dispatch(WalletActions.pricesUpdated({ success: true, values: [ - { - fromAsset: selectedNetwork.symbol.toLowerCase(), - toAsset: defaultFiatCurrency, - price: nativeAssetPrice, - assetTimeframeChange: '' - }, + ...getNativeAssetPrices, ...getTokenPrices ] })) @@ -271,34 +366,38 @@ export function refreshTokenPriceHistory (selectedPortfolioTimeline: BraveWallet const apiProxy = getAPIProxy() const { assetRatioService } = apiProxy - const { wallet: { accounts, defaultCurrencies, selectedNetwork, userVisibleTokensInfo } } = getState() + const { wallet: { accounts, defaultCurrencies, userVisibleTokensInfo } } = getState() - // If a tokens balance is 0 we do not make an unnecessary api call for price history of that token + // Get all Price History const priceHistory = await Promise.all(GetFlattenedAccountBalances(accounts, userVisibleTokensInfo) + // If a tokens balance is 0 we do not make an unnecessary api call for price history of that token .filter(({ token, balance }) => !token.isErc721 && balance > 0) .map(async ({ token }) => ({ - contractAddress: token.contractAddress, + // If a visible asset has a contractAddress of '' + // it is a native asset so we use a symbol instead. + contractAddress: token.contractAddress ? token.contractAddress : token.symbol, history: await assetRatioService.getPriceHistory( - GetTokenParam(selectedNetwork, token), defaultCurrencies.fiat.toLowerCase(), selectedPortfolioTimeline + GetTokenParam(token), defaultCurrencies.fiat.toLowerCase(), selectedPortfolioTimeline ) })) ) + // Combine Price History and Balances const priceHistoryWithBalances = accounts.map((account) => { return userVisibleTokensInfo .filter((token) => !token.isErc721) .map((token) => { const balance = token.contractAddress ? account.tokenBalanceRegistry[token.contractAddress.toLowerCase()] - : account.balance + : account.nativeBalanceRegistry[token.chainId || ''] + const contractAddress = token.contractAddress ? token.contractAddress : token.symbol return { token, balance: balance || '0', - history: priceHistory.find((t) => token.contractAddress === t.contractAddress)?.history ?? { success: true, values: [] } + history: priceHistory.find((t) => contractAddress === t.contractAddress)?.history ?? { success: true, values: [] } } }) }) - dispatch(WalletActions.portfolioPriceHistoryUpdated(priceHistoryWithBalances)) } } @@ -307,7 +406,6 @@ export function refreshTransactionHistory (address?: string) { return async (dispatch: Dispatch, getState: () => State) => { const apiProxy = getAPIProxy() const { txService } = apiProxy - const { wallet: { accounts, transactions } } = getState() const accountsToUpdate = address !== undefined @@ -329,21 +427,50 @@ export function refreshTransactionHistory (address?: string) { } export function refreshNetworkInfo () { - return async (dispatch: Dispatch) => { + return async (dispatch: Dispatch, getState: () => State) => { const apiProxy = getAPIProxy() const { jsonRpcService } = apiProxy - - const networkList = await jsonRpcService.getAllNetworks(BraveWallet.CoinType.ETH) + const { wallet: { selectedCoin, isFilecoinEnabled, isSolanaEnabled, isTestNetworksEnabled } } = getState() + + // Get All Networks + const getFullNetworkList = await Promise.all(SupportedCoinTypes.map(async (coin: BraveWallet.CoinType) => { + // MULTICHAIN: While we are still in development for FIL and SOL, + // we will not use their networks unless enabled by brave://flags + if (coin === BraveWallet.CoinType.FIL && !isFilecoinEnabled) { + return [] + } + if (coin === BraveWallet.CoinType.SOL && !isSolanaEnabled) { + return [] + } + const networkList = await jsonRpcService.getAllNetworks(coin) + return networkList.networks + })) + const flattenedNetworkList = getFullNetworkList.flat(1) + const networkList = + isTestNetworksEnabled + ? flattenedNetworkList + : flattenedNetworkList.filter((network) => !SupportedTestNetworks.includes(network.chainId)) dispatch(WalletActions.setAllNetworks(networkList)) - const chainId = await jsonRpcService.getChainId(BraveWallet.CoinType.ETH) - const currentNetwork = GetNetworkInfo(chainId.chainId, networkList.networks) + + // Get default network for each coinType + const defaultNetworks = await Promise.all(SupportedCoinTypes.map(async (coin: BraveWallet.CoinType) => { + const coinsChainId = await jsonRpcService.getChainId(coin) + const network = getNetworkInfo(coinsChainId.chainId, networkList) + return network + })) + dispatch(WalletActions.setDefaultNetworks(defaultNetworks)) + + // Get current selected networks info + const chainId = await jsonRpcService.getChainId(selectedCoin) + const currentNetwork = getNetworkInfo(chainId.chainId, networkList) dispatch(WalletActions.setNetwork(currentNetwork)) return currentNetwork } } export function refreshKeyringInfo () { - return async (dispatch: Dispatch) => { + return async (dispatch: Dispatch, getState: () => State) => { + const { wallet: { selectedCoin } } = getState() const apiProxy = getAPIProxy() const { keyringService, walletHandler } = apiProxy @@ -357,7 +484,7 @@ export function refreshKeyringInfo () { } // Get selectedAccountAddress - const getSelectedAccount = await keyringService.getSelectedAccount(BraveWallet.CoinType.ETH) + const getSelectedAccount = await keyringService.getSelectedAccount(selectedCoin) const selectedAddress = getSelectedAccount.address // Fallback account address if selectedAccount returns null diff --git a/components/brave_wallet_ui/common/constants/mocks.ts b/components/brave_wallet_ui/common/constants/mocks.ts index ee83413ab5d7..8c9329c49e21 100644 --- a/components/brave_wallet_ui/common/constants/mocks.ts +++ b/components/brave_wallet_ui/common/constants/mocks.ts @@ -39,7 +39,7 @@ export const getMockedTransactionInfo = (): BraveWallet.TransactionInfo => { } } -export const mockNetwork: BraveWallet.EthereumChain = { +export const mockNetwork: BraveWallet.NetworkInfo = { chainId: '0x1', chainName: 'Ethereum Main Net', rpcUrls: ['https://mainnet.infura.io/v3/'], @@ -48,7 +48,7 @@ export const mockNetwork: BraveWallet.EthereumChain = { symbolName: 'Ethereum', decimals: 18, iconUrls: [], - isEip1559: true + coin: BraveWallet.CoinType.ETH } export const mockERC20Token: BraveWallet.BlockchainToken = { @@ -61,14 +61,18 @@ export const mockERC20Token: BraveWallet.BlockchainToken = { decimals: 18, visible: true, tokenId: '', - coingeckoId: '' + coingeckoId: '', + chainId: BraveWallet.MAINNET_CHAIN_ID } export const mockAccount: WalletAccountType = { id: 'mockId', name: 'mockAccountName', address: 'mockAddress', - balance: '123456', + nativeBalanceRegistry: { + '0x1': '123456' + }, + coin: BraveWallet.CoinType.ETH, accountType: 'Primary', tokenBalanceRegistry: {} } diff --git a/components/brave_wallet_ui/common/hooks/assets-management.test.ts b/components/brave_wallet_ui/common/hooks/assets-management.test.ts index af1475ec9ac3..dc0468396a4d 100644 --- a/components/brave_wallet_ui/common/hooks/assets-management.test.ts +++ b/components/brave_wallet_ui/common/hooks/assets-management.test.ts @@ -21,7 +21,8 @@ const mockCustomToken = { decimals: 18, visible: true, tokenId: '', - coingeckoId: '' + coingeckoId: '', + chainId: '0x1' } describe('useAssetManagement hook', () => { @@ -41,7 +42,6 @@ describe('useAssetManagement hook', () => { WalletActions.setUserAssetVisible, WalletActions.removeUserAsset, WalletActions.refreshBalancesAndPriceHistory, - mockNetwork, AccountAssetOptions, mockUserVisibleTokensInfo )) @@ -56,7 +56,6 @@ describe('useAssetManagement hook', () => { WalletActions.setUserAssetVisible, WalletActions.removeUserAsset, WalletActions.refreshBalancesAndPriceHistory, - mockNetwork, AccountAssetOptions, mockUserVisibleTokensInfo )) @@ -71,7 +70,6 @@ describe('useAssetManagement hook', () => { WalletActions.setUserAssetVisible, WalletActions.removeUserAsset, WalletActions.refreshBalancesAndPriceHistory, - mockNetwork, AccountAssetOptions, mockUserVisibleTokensInfo )) @@ -87,7 +85,6 @@ describe('useAssetManagement hook', () => { WalletActions.setUserAssetVisible, WalletActions.removeUserAsset, WalletActions.refreshBalancesAndPriceHistory, - mockNetwork, AccountAssetOptions, [...mockUserVisibleTokensInfo, mockCustomToken] )) @@ -102,7 +99,6 @@ describe('useAssetManagement hook', () => { WalletActions.setUserAssetVisible, WalletActions.removeUserAsset, WalletActions.refreshBalancesAndPriceHistory, - mockNetwork, AccountAssetOptions, [...mockUserVisibleTokensInfo, { ...mockCustomToken, visible: false }] )) diff --git a/components/brave_wallet_ui/common/hooks/assets-management.ts b/components/brave_wallet_ui/common/hooks/assets-management.ts index f44b68f7073d..2837ec65c470 100644 --- a/components/brave_wallet_ui/common/hooks/assets-management.ts +++ b/components/brave_wallet_ui/common/hooks/assets-management.ts @@ -22,7 +22,6 @@ export default function useAssetManagement ( setUserAssetVisible: SimpleActionCreator, removeUserAsset: SimpleActionCreator, refreshBalancesPricesAndHistory: EmptyActionCreator, - selectedNetwork: BraveWallet.NetworkInfo, fullTokenList: BraveWallet.BlockchainToken[], userVisibleTokensInfo: BraveWallet.BlockchainToken[] ) { @@ -32,7 +31,7 @@ export default function useAssetManagement ( ...token, logo: stripERC20TokenImageURL(token.logo) || '' }, - chainId: selectedNetwork.chainId + chainId: token?.chainId ?? '' }) } @@ -56,7 +55,7 @@ export default function useAssetManagement ( userVisibleTokensInfo.filter((firstItem) => !updatedTokensList.some((secondItem) => firstItem.contractAddress.toLowerCase() === secondItem.contractAddress.toLowerCase())) - .forEach((token) => removeUserAsset({ token, chainId: selectedNetwork.chainId })) + .forEach((token) => removeUserAsset({ token, chainId: token?.chainId ?? '' })) // Gets a list of custom tokens returned from updatedTokensList payload // then compares customTokens against userVisibleTokensInfo list and updates the custom tokens visibility if it has changed @@ -73,13 +72,13 @@ export default function useAssetManagement ( } // Updates token visibility exluding a networks native token if (foundToken?.visible !== token.visible && token.contractAddress.toLowerCase() !== '') { - setUserAssetVisible({ token, chainId: selectedNetwork.chainId, isVisible: token.visible }) + setUserAssetVisible({ token, chainId: token?.chainId ?? '', isVisible: token.visible }) } }) // Refreshes Balances, Prices and Price History when done. refreshBalancesPricesAndHistory() - }, [userVisibleTokensInfo, selectedNetwork]) + }, [userVisibleTokensInfo]) return { onUpdateVisibleAssets, diff --git a/components/brave_wallet_ui/common/hooks/assets.test.ts b/components/brave_wallet_ui/common/hooks/assets.test.ts index 4a33326f132f..ea235f6d6b7b 100644 --- a/components/brave_wallet_ui/common/hooks/assets.test.ts +++ b/components/brave_wallet_ui/common/hooks/assets.test.ts @@ -37,7 +37,7 @@ const getBuyAssets = async () => { describe('useAssets hook', () => { it('Selected account has balances, should return expectedResult', async () => { - const { result, waitForNextUpdate } = renderHook(() => useAssets(mockAccounts, mockAccounts[0], mockNetwork, mockVisibleList, mockVisibleList, mockAssetPrices, getBuyAssets)) + const { result, waitForNextUpdate } = renderHook(() => useAssets(mockAccounts[0], [mockNetwork], mockNetwork, mockVisibleList, mockVisibleList, mockAssetPrices, getBuyAssets)) await act(async () => { await waitForNextUpdate() }) @@ -45,7 +45,7 @@ describe('useAssets hook', () => { }) it('should return empty array for panelUserAssetList if visible assets is empty', async () => { - const { result, waitForNextUpdate } = renderHook(() => useAssets(mockAccounts, mockAccount, mockNetwork, mockVisibleList, [], mockAssetPrices, getBuyAssets)) + const { result, waitForNextUpdate } = renderHook(() => useAssets(mockAccount, [mockNetwork], mockNetwork, mockVisibleList, [], mockAssetPrices, getBuyAssets)) await act(async () => { await waitForNextUpdate() }) diff --git a/components/brave_wallet_ui/common/hooks/assets.ts b/components/brave_wallet_ui/common/hooks/assets.ts index 617027e2c071..3f781f0274b9 100644 --- a/components/brave_wallet_ui/common/hooks/assets.ts +++ b/components/brave_wallet_ui/common/hooks/assets.ts @@ -19,8 +19,8 @@ import usePricing from './pricing' import useBalance from './balance' export default function useAssets ( - accounts: WalletAccountType[], selectedAccount: WalletAccountType, + networkList: BraveWallet.NetworkInfo[], selectedNetwork: BraveWallet.NetworkInfo, fullTokenList: BraveWallet.BlockchainToken[], userVisibleTokensInfo: BraveWallet.BlockchainToken[], @@ -28,21 +28,29 @@ export default function useAssets ( getBuyAssets: () => Promise ) { const { computeFiatAmount } = usePricing(spotPrices) - const getBalance = useBalance(selectedNetwork) + const getBalance = useBalance(networkList) const nativeAsset = React.useMemo( () => makeNetworkAsset(selectedNetwork), [selectedNetwork] ) + const assetsByNetwork = React.useMemo(() => { + if (!userVisibleTokensInfo) { + return [] + } + + return userVisibleTokensInfo.filter((token) => token.chainId === selectedNetwork.chainId) + }, [userVisibleTokensInfo, selectedNetwork]) + const swapAssetOptions: BraveWallet.BlockchainToken[] = React.useMemo(() => { return [ nativeAsset, ...fullTokenList.filter((asset) => asset.symbol.toUpperCase() === 'BAT'), - ...userVisibleTokensInfo + ...assetsByNetwork .filter(asset => !['BAT', nativeAsset.symbol.toUpperCase()].includes(asset.symbol.toUpperCase())), ...fullTokenList .filter(asset => !['BAT', nativeAsset.symbol.toUpperCase()].includes(asset.symbol.toUpperCase())) - .filter(asset => !userVisibleTokensInfo + .filter(asset => !assetsByNetwork .some(token => token.symbol.toUpperCase() === asset.symbol.toUpperCase())) ] }, [fullTokenList, userVisibleTokensInfo, nativeAsset]) @@ -58,8 +66,8 @@ export default function useAssets ( }).catch(e => console.error(e)) }, []) - const panelUserAssetList = React.useMemo(() => { - if (!userVisibleTokensInfo) { + const assetsByValueAndNetwork = React.useMemo(() => { + if (!assetsByNetwork) { return [] } @@ -67,7 +75,7 @@ export default function useAssets ( return [] } - return userVisibleTokensInfo.sort(function (a, b) { + return assetsByNetwork.sort(function (a, b) { const aBalance = getBalance(selectedAccount, a) const bBalance = getBalance(selectedAccount, b) @@ -76,12 +84,12 @@ export default function useAssets ( return bFiatBalance.toNumber() - aFiatBalance.toNumber() }) - }, [selectedAccount, userVisibleTokensInfo, getBalance, computeFiatAmount]) + }, [selectedAccount, assetsByNetwork, getBalance, computeFiatAmount]) return { swapAssetOptions, - sendAssetOptions: userVisibleTokensInfo, + sendAssetOptions: assetsByNetwork, buyAssetOptions, - panelUserAssetList + panelUserAssetList: assetsByValueAndNetwork } } diff --git a/components/brave_wallet_ui/common/hooks/balance.ts b/components/brave_wallet_ui/common/hooks/balance.ts index 49eede4b3c25..69bc1c7c83b8 100644 --- a/components/brave_wallet_ui/common/hooks/balance.ts +++ b/components/brave_wallet_ui/common/hooks/balance.ts @@ -6,8 +6,9 @@ import * as React from 'react' import { BraveWallet, WalletAccountType } from '../../constants/types' +import { getTokensCoinType } from '../../utils/network-utils' -export default function useBalance (network: BraveWallet.NetworkInfo) { +export default function useBalance (networks: BraveWallet.NetworkInfo[]) { return React.useCallback((account?: WalletAccountType, token?: BraveWallet.BlockchainToken) => { if (!account || !token) { return '' @@ -17,11 +18,18 @@ export default function useBalance (network: BraveWallet.NetworkInfo) { return '' } + const tokensCoinType = getTokensCoinType(networks, token) // Return native asset balance - if (token.symbol.toLowerCase() === network.symbol.toLowerCase()) { - return account.balance + if ( + ( + token.contractAddress === '' || + token.contractAddress === '0xeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee' + ) && + account.coin === tokensCoinType + ) { + return (account.nativeBalanceRegistry || {})[token.chainId || ''] || '' } return (account.tokenBalanceRegistry || {})[token.contractAddress.toLowerCase()] || '' - }, [network]) + }, [networks]) } diff --git a/components/brave_wallet_ui/common/hooks/select-preset.test.ts b/components/brave_wallet_ui/common/hooks/select-preset.test.ts index 20abb1d7e9dd..7f1b03bf4b40 100644 --- a/components/brave_wallet_ui/common/hooks/select-preset.test.ts +++ b/components/brave_wallet_ui/common/hooks/select-preset.test.ts @@ -22,7 +22,7 @@ describe('usePreset hook', () => { [mockERC20Token.contractAddress.toLowerCase()]: balance } }, - mockNetwork, + [mockNetwork], mockFunc, jest.fn(), mockERC20Token, @@ -40,7 +40,7 @@ describe('usePreset hook', () => { const { result: { current: calcPresetAmount } } = renderHook(() => usePreset( mockAccount, - mockNetwork, + [mockNetwork], mockOnSetFromAmount, mockOnSetSendAmount, undefined, diff --git a/components/brave_wallet_ui/common/hooks/select-preset.ts b/components/brave_wallet_ui/common/hooks/select-preset.ts index 2435c5523fbe..6614166ec3cb 100644 --- a/components/brave_wallet_ui/common/hooks/select-preset.ts +++ b/components/brave_wallet_ui/common/hooks/select-preset.ts @@ -13,13 +13,13 @@ import Amount from '../../utils/amount' export default function usePreset ( selectedAccount: WalletAccountType, - selectedNetwork: BraveWallet.NetworkInfo, + networkList: BraveWallet.NetworkInfo[], onSetFromAmount: (value: string) => void, onSetSendAmount: (value: string) => void, swapAsset?: BraveWallet.BlockchainToken, sendAsset?: BraveWallet.BlockchainToken ) { - const getBalance = useBalance(selectedNetwork) + const getBalance = useBalance(networkList) return (sendOrSwap: 'send' | 'swap') => (percent: number) => { const selectedAsset = sendOrSwap === 'send' ? sendAsset : swapAsset diff --git a/components/brave_wallet_ui/common/hooks/swap.test.ts b/components/brave_wallet_ui/common/hooks/swap.test.ts index 4646f480a724..20f3f507d5e2 100644 --- a/components/brave_wallet_ui/common/hooks/swap.test.ts +++ b/components/brave_wallet_ui/common/hooks/swap.test.ts @@ -21,7 +21,7 @@ async function mockGetERC20Allowance (contractAddress: string, ownerAddress: str } const mockIsSwapSupportedFactory = (expected: boolean) => - async (network: BraveWallet.EthereumChain) => + async (network: BraveWallet.NetworkInfo) => expected const mockQuote = { @@ -49,6 +49,7 @@ describe('useSwap hook', () => { const { result, waitForNextUpdate } = renderHook(() => useSwap( mockAccount, mockNetwork, + [mockNetwork], AccountAssetOptions, WalletPageActions.fetchPageSwapQuote, mockGetERC20Allowance, @@ -66,6 +67,7 @@ describe('useSwap hook', () => { const { result, waitFor } = renderHook(() => useSwap( mockAccount, mockNetwork, + [mockNetwork], AccountAssetOptions, WalletPageActions.fetchPageSwapQuote, mockGetERC20Allowance, @@ -85,6 +87,7 @@ describe('useSwap hook', () => { const { waitForNextUpdate } = renderHook(() => useSwap( mockAccount, mockNetwork, + [mockNetwork], AccountAssetOptions, WalletPageActions.fetchPageSwapQuote, mockFn, @@ -107,6 +110,7 @@ describe('useSwap hook', () => { const { waitForNextUpdate } = renderHook(() => useSwap( mockAccount, mockNetwork, + [mockNetwork], swapAssets, WalletPageActions.fetchPageSwapQuote, mockFn, @@ -133,6 +137,7 @@ describe('useSwap hook', () => { const { waitForNextUpdate } = renderHook(() => useSwap( mockAccount, mockNetwork, + [mockNetwork], swapAssets, WalletPageActions.fetchPageSwapQuote, mockFn, @@ -157,6 +162,7 @@ describe('useSwap hook', () => { const { result, waitForValueToChange, waitFor } = renderHook(() => useSwap( mockAccount, mockNetwork, + [mockNetwork], AccountAssetOptions, WalletPageActions.fetchPageSwapQuote, mockGetERC20Allowance, @@ -185,9 +191,12 @@ describe('useSwap hook', () => { const { result, waitForValueToChange, waitFor } = renderHook(() => useSwap( { ...mockAccount, - balance: '1000000000000000000' // 1 ETH + nativeBalanceRegistry: { + '0x1': '1000000000000000000' + } }, mockNetwork, + [mockNetwork], AccountAssetOptions, // From asset is ETH WalletPageActions.fetchPageSwapQuote, mockGetERC20Allowance, @@ -233,6 +242,7 @@ describe('useSwap hook', () => { const { result, waitFor, waitForValueToChange } = renderHook(() => useSwap( mockAccount, mockNetwork, + [mockNetwork], AccountAssetOptions, // To asset is BAT WalletPageActions.fetchPageSwapQuote, mockGetERC20Allowance, @@ -279,6 +289,7 @@ describe('useSwap hook', () => { const { result, waitFor, waitForValueToChange } = renderHook(() => useSwap( mockAccount, // Balance: 123456 Wei mockNetwork, + [mockNetwork], AccountAssetOptions, // From asset is ETH WalletPageActions.fetchPageSwapQuote, mockGetERC20Allowance, @@ -330,6 +341,7 @@ describe('useSwap hook', () => { } }, mockNetwork, + [mockNetwork], AccountAssetOptions.slice(1), // From asset is BAT WalletPageActions.fetchPageSwapQuote, mockGetERC20Allowance, @@ -377,6 +389,7 @@ describe('useSwap hook', () => { const { result, waitFor, waitForValueToChange } = renderHook(() => useSwap( mockAccount, // Balance: 123456 Wei mockNetwork, + [mockNetwork], AccountAssetOptions, // From asset is ETH WalletPageActions.fetchPageSwapQuote, mockGetERC20Allowance, @@ -409,9 +422,12 @@ describe('useSwap hook', () => { const { result, waitFor, waitForValueToChange } = renderHook(() => useSwap( { ...mockAccount, - balance: '1234560' // 1234560 Wei + nativeBalanceRegistry: { + '0x1': '1234560' + } }, mockNetwork, + [mockNetwork], AccountAssetOptions, // From asset is ETH WalletPageActions.fetchPageSwapQuote, mockGetERC20Allowance, @@ -458,12 +474,15 @@ describe('useSwap hook', () => { const { result, waitFor, waitForValueToChange } = renderHook(() => useSwap( { ...mockAccount, - balance: '1000000000000000000', // 1 ETH + nativeBalanceRegistry: { + '0x1': '1000000000000000000' + }, tokenBalanceRegistry: { [AccountAssetOptions[1].contractAddress.toLowerCase()]: '20000000000000000000' // 20 BAT } }, mockNetwork, + [mockNetwork], AccountAssetOptions.slice(1), // From asset is BAT WalletPageActions.fetchPageSwapQuote, mockGetERC20Allowance, @@ -504,9 +523,12 @@ describe('useSwap hook', () => { const { result, waitFor, waitForValueToChange } = renderHook(() => useSwap( { ...mockAccount, - balance: '1000000000000000000' // 1 ETH + nativeBalanceRegistry: { + '0x1': '1000000000000000000' + } }, mockNetwork, + [mockNetwork], AccountAssetOptions, // From asset is ETH WalletPageActions.fetchPageSwapQuote, mockGetERC20Allowance, @@ -558,9 +580,12 @@ describe('useSwap hook', () => { const { result, waitFor, waitForValueToChange } = renderHook(() => useSwap( { ...mockAccount, - balance: '1000000000000000000' // 1 ETH + nativeBalanceRegistry: { + '0x1': '1000000000000000000' + } }, mockNetwork, + [mockNetwork], AccountAssetOptions, // From asset is ETH WalletPageActions.fetchPageSwapQuote, mockGetERC20Allowance, diff --git a/components/brave_wallet_ui/common/hooks/swap.ts b/components/brave_wallet_ui/common/hooks/swap.ts index 398f3ba69f37..caa1064a4a86 100644 --- a/components/brave_wallet_ui/common/hooks/swap.ts +++ b/components/brave_wallet_ui/common/hooks/swap.ts @@ -38,6 +38,7 @@ const SWAP_VALIDATION_ERROR_CODE = 100 export default function useSwap ( selectedAccount: WalletAccountType, selectedNetwork: BraveWallet.NetworkInfo, + networkList: BraveWallet.NetworkInfo[], swapAssetOptions: BraveWallet.BlockchainToken[], fetchSwapQuote: SimpleActionCreator, getERC20Allowance: (contractAddress: string, ownerAddress: string, spenderAddress: string) => Promise, @@ -99,7 +100,7 @@ export default function useSwap ( .catch(e => console.log(e)) }, [fromAsset, quote, selectedAccount]) - const getBalance = useBalance(selectedNetwork) + const getBalance = useBalance(networkList) const fromAssetBalance = getBalance(selectedAccount, fromAsset) const nativeAssetBalance = getBalance(selectedAccount, nativeAsset) diff --git a/components/brave_wallet_ui/common/hooks/transaction-parser.test.ts b/components/brave_wallet_ui/common/hooks/transaction-parser.test.ts index 48743e0e9f70..521ba4623597 100644 --- a/components/brave_wallet_ui/common/hooks/transaction-parser.test.ts +++ b/components/brave_wallet_ui/common/hooks/transaction-parser.test.ts @@ -236,7 +236,9 @@ describe('useTransactionParser hook', () => { [mockERC20Token.contractAddress.toLowerCase()]: '0' }, address: '0xdeadbeef', - balance: '1000000000000000' // 0.001 ETH + nativeBalanceRegistry: { + '0x1': '1000000000000000' // 0.001 ETH + } }], mockAssetPrices, [], [] )) @@ -284,7 +286,9 @@ describe('useTransactionParser hook', () => { [{ ...mockAccount, address: '0xdeadbeef', - balance: '1000000000000000000', // 1 ETH + nativeBalanceRegistry: { + '0x1': '1000000000000000000' // 1 ETH + }, tokenBalanceRegistry: { [mockTxData.baseData.to.toLowerCase()]: '0' } @@ -342,7 +346,9 @@ describe('useTransactionParser hook', () => { [{ ...mockAccount, address: '0xdeadbeef', - balance: '1000000000000000' // 0.001 ETH + nativeBalanceRegistry: { + '0x1': '1000000000000000' // 0.001 ETH + } }], mockAssetPrices, [], [] )) @@ -383,7 +389,9 @@ describe('useTransactionParser hook', () => { [{ ...mockAccount, address: '0xdeadbeef', - balance: '1003150000000000000' // 1.00315 ETH + nativeBalanceRegistry: { + '0x1': '1003150000000000000' // 1.00315 ETH + } }], mockAssetPrices, [], [] )) @@ -430,7 +438,9 @@ describe('useTransactionParser hook', () => { [mockERC20Token.contractAddress.toLowerCase()]: '1000000000000000' // 0.001 DOG }, address: '0xdeadbeef', - balance: '3150000000000000' // 0.00315 ETH + nativeBalanceRegistry: { + '0x1': '3150000000000000' // 0.00315 ETH + } }], mockAssetPrices, [], [] )) @@ -480,7 +490,9 @@ describe('useTransactionParser hook', () => { [mockERC20Token.contractAddress.toLowerCase()]: '1000000000000000000' // 1 DOG }, address: '0xdeadbeef', - balance: '3150000000000000' // 0.00315 ETH + nativeBalanceRegistry: { + '0x1': '3150000000000000' // 0.00315 ETH + } }], mockAssetPrices, [], diff --git a/components/brave_wallet_ui/common/hooks/transaction-parser.ts b/components/brave_wallet_ui/common/hooks/transaction-parser.ts index 52d4dcfc1a9c..25b683f7a5d6 100644 --- a/components/brave_wallet_ui/common/hooks/transaction-parser.ts +++ b/components/brave_wallet_ui/common/hooks/transaction-parser.ts @@ -128,7 +128,7 @@ export function useTransactionParser ( [selectedNetwork] ) const { findAssetPrice, computeFiatAmount } = usePricing(spotPrices) - const getBalance = useBalance(selectedNetwork) + const getBalance = useBalance([selectedNetwork]) const getAddressLabel = useAddressLabels(accounts) const networkSpotPrice = React.useMemo( diff --git a/components/brave_wallet_ui/common/reducers/wallet_reducer.ts b/components/brave_wallet_ui/common/reducers/wallet_reducer.ts index ecffba5a6ba4..9d32bfeee63e 100644 --- a/components/brave_wallet_ui/common/reducers/wallet_reducer.ts +++ b/components/brave_wallet_ui/common/reducers/wallet_reducer.ts @@ -8,10 +8,8 @@ import { AccountInfo, AccountTransactions, BraveWallet, - GetAllNetworksList, - GetAllTokensReturnInfo, GetBlockchainTokenBalanceReturnInfo, - GetNativeAssetBalancesReturnInfo, + GetNativeAssetBalancesPayload, GetPriceHistoryReturnInfo, PortfolioTokenHistoryAndInfo, WalletAccountType, @@ -40,6 +38,7 @@ const defaultState: WalletState = { hasInitialized: false, isFilecoinEnabled: false, isSolanaEnabled: false, + isTestNetworksEnabled: true, isWalletCreated: false, isWalletLocked: true, favoriteApps: [], @@ -57,9 +56,9 @@ const defaultState: WalletState = { decimals: 18, coin: BraveWallet.CoinType.ETH, data: { - ethData: { + ethData: { isEip1559: true - } + } } } as BraveWallet.NetworkInfo, accounts: [], @@ -80,10 +79,12 @@ const defaultState: WalletState = { gasEstimates: undefined, connectedAccounts: [], isMetaMaskInstalled: false, + selectedCoin: BraveWallet.CoinType.ETH, defaultCurrencies: { fiat: '', crypto: '' - } + }, + defaultNetworks: [] as BraveWallet.NetworkInfo[] } const reducer = createReducer({}, defaultState) @@ -101,10 +102,10 @@ reducer.on(WalletActions.initialized, (state: any, payload: WalletInfo) => { id: `${idx + 1}`, name: info.name, address: info.address, - balance: '', accountType: getAccountType(info), deviceId: info.hardware ? info.hardware.deviceId : '', tokenBalanceRegistry: {}, + nativeBalanceRegistry: {}, coin: info.coin } as WalletAccountType }) @@ -142,47 +143,40 @@ reducer.on(WalletActions.setSelectedAccount, (state: any, payload: WalletAccount reducer.on(WalletActions.setNetwork, (state: any, payload: BraveWallet.NetworkInfo) => { return { ...state, - isFetchingPortfolioPriceHistory: true, selectedNetwork: payload } }) reducer.on(WalletActions.setVisibleTokensInfo, (state: WalletState, payload: BraveWallet.BlockchainToken[]) => { - const userVisibleTokensInfo = payload.map((token) => ({ - ...token, - logo: `chrome://erc-token-images/${token.logo}` - })) as BraveWallet.BlockchainToken[] - return { ...state, - userVisibleTokensInfo + userVisibleTokensInfo: payload } }) -reducer.on(WalletActions.setAllNetworks, (state: any, payload: GetAllNetworksList) => { +reducer.on(WalletActions.setAllNetworks, (state: any, payload: BraveWallet.NetworkInfo[]) => { return { ...state, - networkList: payload.networks + networkList: payload } }) -reducer.on(WalletActions.setAllTokensList, (state: WalletState, payload: GetAllTokensReturnInfo) => { +reducer.on(WalletActions.setAllTokensList, (state: WalletState, payload: BraveWallet.BlockchainToken[]) => { return { ...state, - fullTokenList: payload.tokens.map(token => ({ - ...token, - logo: `chrome://erc-token-images/${token.logo}` - })) + fullTokenList: payload } }) -reducer.on(WalletActions.nativeAssetBalancesUpdated, (state: WalletState, payload: GetNativeAssetBalancesReturnInfo) => { +reducer.on(WalletActions.nativeAssetBalancesUpdated, (state: WalletState, payload: GetNativeAssetBalancesPayload) => { let accounts: WalletAccountType[] = [...state.accounts] - accounts.forEach((account, index) => { - if (payload.balances[index].error === BraveWallet.ProviderError.kSuccess) { - accounts[index].balance = Amount.normalize(payload.balances[index].balance) - } + accounts.forEach((account, accountIndex) => { + payload.balances[accountIndex].forEach((info, tokenIndex) => { + if (info.error === BraveWallet.ProviderError.kSuccess) { + accounts[accountIndex].nativeBalanceRegistry[info.chainId] = Amount.normalize(info.balance) + } + }) }) // Refresh selectedAccount object @@ -450,4 +444,18 @@ reducer.on(WalletActions.defaultCurrenciesUpdated, (state: any, payload: Default } }) +reducer.on(WalletActions.setSelectedCoin, (state: any, payload: BraveWallet.CoinType) => { + return { + ...state, + selectedCoin: payload + } +}) + +reducer.on(WalletActions.setDefaultNetworks, (state: any, payload: BraveWallet.NetworkInfo[]) => { + return { + ...state, + defaultNetworks: payload + } +}) + export default reducer diff --git a/components/brave_wallet_ui/components/buy-send-swap/accounts-assets-networks/index.tsx b/components/brave_wallet_ui/components/buy-send-swap/accounts-assets-networks/index.tsx index fdf1dc2234b7..16f023ec7578 100644 --- a/components/brave_wallet_ui/components/buy-send-swap/accounts-assets-networks/index.tsx +++ b/components/brave_wallet_ui/components/buy-send-swap/accounts-assets-networks/index.tsx @@ -2,7 +2,8 @@ import * as React from 'react' import { BraveWallet, BuySendSwapViewTypes, - UserAccountType + UserAccountType, + WalletAccountType } from '../../../constants/types' import { @@ -18,7 +19,7 @@ import { export interface Props { selectedView: BuySendSwapViewTypes - accounts: UserAccountType[] + accounts: WalletAccountType[] networkList: BraveWallet.NetworkInfo[] assetOptions: BraveWallet.BlockchainToken[] selectedNetwork: BraveWallet.NetworkInfo diff --git a/components/brave_wallet_ui/components/buy-send-swap/select-account-with-header/index.tsx b/components/brave_wallet_ui/components/buy-send-swap/select-account-with-header/index.tsx index 94d6616fc4c5..2c7c3436a239 100644 --- a/components/brave_wallet_ui/components/buy-send-swap/select-account-with-header/index.tsx +++ b/components/brave_wallet_ui/components/buy-send-swap/select-account-with-header/index.tsx @@ -1,5 +1,5 @@ import * as React from 'react' -import { UserAccountType } from '../../../constants/types' +import { UserAccountType, WalletAccountType } from '../../../constants/types' import { SearchBar, SelectAccount } from '../../shared' import Header from '../../buy-send-swap/select-header' import { getLocale } from '../../../../common/locale' @@ -10,7 +10,7 @@ import { } from '../shared-styles' export interface Props { - accounts: UserAccountType[] + accounts: WalletAccountType[] selectedAccount: UserAccountType onSelectAccount: (account: UserAccountType) => () => void onAddAccount?: () => void @@ -19,8 +19,15 @@ export interface Props { } function SelectAccountWithHeader (props: Props) { - const { accounts, selectedAccount, onSelectAccount, onBack, onAddAccount, hasAddButton } = props - const [filteredAccountList, setFilteredAccountList] = React.useState(accounts) + const { + accounts, + selectedAccount, + onSelectAccount, + onBack, + onAddAccount, + hasAddButton + } = props + const [filteredAccountList, setFilteredAccountList] = React.useState(accounts) const filterAccountList = (event: any) => { const search = event.target.value diff --git a/components/brave_wallet_ui/components/buy-send-swap/tabs/buy-tab.tsx b/components/brave_wallet_ui/components/buy-send-swap/tabs/buy-tab.tsx index 50906f84530c..b23ef27581ab 100644 --- a/components/brave_wallet_ui/components/buy-send-swap/tabs/buy-tab.tsx +++ b/components/brave_wallet_ui/components/buy-send-swap/tabs/buy-tab.tsx @@ -3,7 +3,8 @@ import { UserAccountType, BuySendSwapViewTypes, BraveWallet, - DefaultCurrencies + DefaultCurrencies, + WalletAccountType } from '../../../constants/types' import { AccountsAssetsNetworks, @@ -12,7 +13,7 @@ import { } from '..' export interface Props { - accounts: UserAccountType[] + accounts: WalletAccountType[] networkList: BraveWallet.NetworkInfo[] selectedNetwork: BraveWallet.NetworkInfo selectedAccount: UserAccountType diff --git a/components/brave_wallet_ui/components/buy-send-swap/tabs/send-tab.tsx b/components/brave_wallet_ui/components/buy-send-swap/tabs/send-tab.tsx index 954f5aa9c238..1c5420366865 100644 --- a/components/brave_wallet_ui/components/buy-send-swap/tabs/send-tab.tsx +++ b/components/brave_wallet_ui/components/buy-send-swap/tabs/send-tab.tsx @@ -4,7 +4,8 @@ import { BuySendSwapViewTypes, ToOrFromType, BraveWallet, - AmountValidationErrorType + AmountValidationErrorType, + WalletAccountType } from '../../../constants/types' import { AccountsAssetsNetworks, @@ -13,7 +14,7 @@ import { } from '..' export interface Props { - accounts: UserAccountType[] + accounts: WalletAccountType[] selectedAsset: BraveWallet.BlockchainToken | undefined selectedNetwork: BraveWallet.NetworkInfo selectedAccount: UserAccountType diff --git a/components/brave_wallet_ui/components/buy-send-swap/tabs/swap-tab.tsx b/components/brave_wallet_ui/components/buy-send-swap/tabs/swap-tab.tsx index a724e56dd761..489da1d0740b 100644 --- a/components/brave_wallet_ui/components/buy-send-swap/tabs/swap-tab.tsx +++ b/components/brave_wallet_ui/components/buy-send-swap/tabs/swap-tab.tsx @@ -7,7 +7,8 @@ import { ExpirationPresetObjectType, ToOrFromType, BraveWallet, - SwapValidationErrorType + SwapValidationErrorType, + WalletAccountType } from '../../../constants/types' import { AccountsAssetsNetworks, @@ -16,7 +17,7 @@ import { } from '..' export interface Props { - accounts: UserAccountType[] + accounts: WalletAccountType[] networkList: BraveWallet.NetworkInfo[] orderType: OrderTypes swapToAsset: BraveWallet.BlockchainToken | undefined diff --git a/components/brave_wallet_ui/components/desktop/asset-watchlist-item/index.tsx b/components/brave_wallet_ui/components/desktop/asset-watchlist-item/index.tsx index a367bb78a02d..f1612e37fdd6 100644 --- a/components/brave_wallet_ui/components/desktop/asset-watchlist-item/index.tsx +++ b/components/brave_wallet_ui/components/desktop/asset-watchlist-item/index.tsx @@ -2,6 +2,7 @@ import * as React from 'react' import { Checkbox } from 'brave-ui' import { BraveWallet } from '../../../constants/types' import { withPlaceholderIcon } from '../../shared' +import { getTokensNetwork } from '../../../utils/network-utils' // Utils import Amount from '../../../utils/amount' @@ -25,7 +26,7 @@ export interface Props { isCustom: boolean isSelected: boolean token: BraveWallet.BlockchainToken - selectedNetwork: BraveWallet.NetworkInfo + networkList: BraveWallet.NetworkInfo[] } const AssetWatchlistItem = (props: Props) => { @@ -35,7 +36,7 @@ const AssetWatchlistItem = (props: Props) => { isCustom, token, isSelected, - selectedNetwork + networkList } = props const onCheck = (key: string, selected: boolean) => { @@ -54,10 +55,17 @@ const AssetWatchlistItem = (props: Props) => { return withPlaceholderIcon(AssetIcon, { size: 'big', marginLeft: 0, marginRight: 8 }) }, []) + const tokensNetwork = React.useMemo(() => { + if (!token) { + return + } + return getTokensNetwork(networkList, token) + }, [token, networkList]) + return ( - + {token.name} { @@ -66,7 +74,7 @@ const AssetWatchlistItem = (props: Props) => { : '' } - {token.symbol} + {token.symbol} on {tokensNetwork?.chainName ?? ''} diff --git a/components/brave_wallet_ui/components/desktop/popup-modals/edit-visible-assets-modal/index.tsx b/components/brave_wallet_ui/components/desktop/popup-modals/edit-visible-assets-modal/index.tsx index ac1f9453836a..2d54100a5a34 100644 --- a/components/brave_wallet_ui/components/desktop/popup-modals/edit-visible-assets-modal/index.tsx +++ b/components/brave_wallet_ui/components/desktop/popup-modals/edit-visible-assets-modal/index.tsx @@ -42,6 +42,7 @@ export interface Props { fullAssetList: BraveWallet.BlockchainToken[] userVisibleTokensInfo: BraveWallet.BlockchainToken[] selectedNetwork: BraveWallet.NetworkInfo + networkList: BraveWallet.NetworkInfo[] onFindTokenInfoByContractAddress: (contractAddress: string) => void foundTokenInfoByContractAddress?: BraveWallet.BlockchainToken } @@ -52,6 +53,7 @@ const EditVisibleAssetsModal = (props: Props) => { userVisibleTokensInfo, addUserAssetError, selectedNetwork, + networkList, onClose, onAddCustomAsset, onFindTokenInfoByContractAddress, @@ -172,7 +174,8 @@ const EditVisibleAssetsModal = (props: Props) => { symbol: selectedNetwork.symbol, visible: true, tokenId: '', - coingeckoId: '' + coingeckoId: '', + chainId: '' } const tokenList = React.useMemo(() => { @@ -251,7 +254,8 @@ const EditVisibleAssetsModal = (props: Props) => { tokenId: tokenID ? new Amount(tokenID).toHex() : '', logo: iconURL, visible: true, - coingeckoId: coingeckoID + coingeckoId: coingeckoID, + chainId: '' } onAddCustomAsset(newToken) } @@ -490,10 +494,10 @@ const EditVisibleAssetsModal = (props: Props) => { <> {filteredTokenList.map((token) => { token, defaultCurrencies, hideBalances, - selectedNetwork, - isPanel + isPanel, + networks } = props const AssetIconWithPlaceholder = React.useMemo(() => { @@ -66,14 +68,18 @@ const PortfolioAssetItem = (props: Props) => { return computeFiatAmount(assetBalance, token.symbol, token.decimals) }, [computeFiatAmount, assetBalance, token]) + const tokensNetwork = React.useMemo(() => { + return getTokensNetwork(networks, token) + }, [token, networks]) + const NetworkDescription = React.useMemo(() => { - if (selectedNetwork && token.contractAddress !== '' && !isPanel) { + if (tokensNetwork && !isPanel) { return getLocale('braveWalletPortfolioAssetNetworkDescription') .replace('$1', token.symbol) - .replace('$2', selectedNetwork?.chainName ?? '') + .replace('$2', tokensNetwork.chainName ?? '') } return token.symbol - }, [selectedNetwork, token]) + }, [tokensNetwork, token]) return ( <> @@ -82,10 +88,10 @@ const PortfolioAssetItem = (props: Props) => { - - {selectedNetwork && token.contractAddress !== '' && !isPanel && + + {tokensNetwork && token.contractAddress !== '' && !isPanel && - + } diff --git a/components/brave_wallet_ui/components/desktop/views/accounts/index.tsx b/components/brave_wallet_ui/components/desktop/views/accounts/index.tsx index fada4ef404b8..43923a3896c2 100644 --- a/components/brave_wallet_ui/components/desktop/views/accounts/index.tsx +++ b/components/brave_wallet_ui/components/desktop/views/accounts/index.tsx @@ -7,7 +7,8 @@ import { AccountTransactions, BraveWallet, DefaultCurrencies, - AddAccountNavTypes + AddAccountNavTypes, + CoinTypesMap } from '../../../../constants/types' import { reduceAddress } from '../../../../utils/reduce-address' import { copyToClipboard } from '../../../../utils/copy-to-clipboard' @@ -61,6 +62,7 @@ export interface Props { transactions: AccountTransactions privateKey: string selectedNetwork: BraveWallet.NetworkInfo + networkList: BraveWallet.NetworkInfo[] userVisibleTokensInfo: BraveWallet.BlockchainToken[] transactionSpotPrices: BraveWallet.AssetPrice[] selectedAccount: WalletAccountType | undefined @@ -89,6 +91,7 @@ function Accounts (props: Props) { userVisibleTokensInfo, selectedAccount, defaultCurrencies, + networkList, goBack, onSelectAccount, onSelectAsset, @@ -103,7 +106,7 @@ function Accounts (props: Props) { onCancelTransaction } = props - const getBalance = useBalance(selectedNetwork) + const getBalance = useBalance(networkList) const groupById = (accounts: WalletAccountType[], key: string) => { return accounts.reduce((result, obj) => { @@ -180,9 +183,29 @@ function Accounts (props: Props) { } }, [selectedAccount, transactions]) + const accountsTokensList = React.useMemo(() => { + if (!selectedAccount) { + return [] + } + // Since LOCALHOST's chainId is shared between coinType's + // this check will make sure we are returning the correct + // LOCALHOST asset for each account. + const coinName = CoinTypesMap[selectedAccount?.coin ?? 0] + const localHostCoins = userVisibleTokensInfo.filter((token) => token.chainId === BraveWallet.LOCALHOST_CHAIN_ID) + const accountsLocalHost = localHostCoins.find((token) => token.symbol.toUpperCase() === coinName) + const chainList = networkList.filter((network) => network.coin === selectedAccount?.coin).map((network) => network.chainId) ?? [] + const list = + userVisibleTokensInfo.filter((token) => chainList.includes(token?.chainId ?? '') && + token.chainId !== BraveWallet.LOCALHOST_CHAIN_ID) ?? [] + if (accountsLocalHost) { + return [...list, accountsLocalHost] + } + return list + }, [userVisibleTokensInfo, selectedAccount, networkList]) + const erc271Tokens = React.useMemo(() => - userVisibleTokensInfo.filter((token) => token.isErc721), - [userVisibleTokensInfo] + accountsTokensList.filter((token) => token.isErc721), + [accountsTokensList] ) return ( @@ -283,13 +306,13 @@ function Accounts (props: Props) { {getLocale('braveWalletAccountsAssets')} - {userVisibleTokensInfo.filter((token) => !token.isErc721).map((item) => + {accountsTokensList.filter((token) => !token.isErc721).map((item, index) => )} @@ -299,12 +322,12 @@ function Accounts (props: Props) { {getLocale('braveWalletTopNavNFTS')} - {erc271Tokens?.map((item) => + {erc271Tokens?.map((item, index) => diff --git a/components/brave_wallet_ui/components/desktop/views/crypto/index.tsx b/components/brave_wallet_ui/components/desktop/views/crypto/index.tsx index b26197d5edfa..9f5a3337ced3 100644 --- a/components/brave_wallet_ui/components/desktop/views/crypto/index.tsx +++ b/components/brave_wallet_ui/components/desktop/views/crypto/index.tsx @@ -40,7 +40,6 @@ export interface Props { onUpdateAccountName: (payload: UpdateAccountNamePayloadType) => { success: boolean } onShowAddModal: () => void onHideAddModal: () => void - onSelectNetwork: (network: BraveWallet.NetworkInfo) => void onRemoveAccount: (address: string, hardware: boolean, coin: BraveWallet.CoinType) => void onViewPrivateKey: (address: string, isDefault: boolean, coin: BraveWallet.CoinType) => void onDoneViewingPrivateKey: () => void @@ -100,7 +99,6 @@ const CryptoView = (props: Props) => { onImportAccount, onImportFilecoinAccount, onUpdateAccountName, - onSelectNetwork, onRemoveAccount, onViewPrivateKey, onDoneViewingPrivateKey, @@ -166,7 +164,9 @@ const CryptoView = (props: Props) => { if (id === 'add-asset') { onShowVisibleAssetsModal(true) } else { - const asset = userVisibleTokensInfo.find((token) => token.symbol.toLowerCase() === id?.toLowerCase()) + const asset = id?.toLowerCase().startsWith('0x') + ? userVisibleTokensInfo.find((token) => token.contractAddress === id) + : userVisibleTokensInfo.find((token) => token.symbol.toLowerCase() === id?.toLowerCase()) onSelectAsset(asset) setHideNav(true) } @@ -224,7 +224,11 @@ const CryptoView = (props: Props) => { const selectAsset = (asset: BraveWallet.BlockchainToken | undefined) => { if (asset) { - history.push(`${WalletRoutes.Portfolio}/${asset.symbol}`) + if (asset.contractAddress === '') { + history.push(`${WalletRoutes.Portfolio}/${asset.symbol}`) + return + } + history.push(`${WalletRoutes.Portfolio}/${asset.contractAddress}`) } else { onSelectAsset(asset) history.push(WalletRoutes.Portfolio) @@ -312,7 +316,6 @@ const CryptoView = (props: Props) => { onSelectAsset={selectAsset} onSelectAccount={onSelectAccount} onClickAddAccount={onClickAddAccount} - onSelectNetwork={onSelectNetwork} onAddCustomAsset={onAddCustomAsset} selectedAsset={selectedAsset} portfolioBalance={portfolioBalance} @@ -360,6 +363,7 @@ const CryptoView = (props: Props) => { onRetryTransaction={onRetryTransaction} onSpeedupTransaction={onSpeedupTransaction} onCancelTransaction={onCancelTransaction} + networkList={networkList} /> diff --git a/components/brave_wallet_ui/components/desktop/views/portfolio/components/accounts-and-transctions-list/index.tsx b/components/brave_wallet_ui/components/desktop/views/portfolio/components/accounts-and-transctions-list/index.tsx index 4cb230cef679..3bdf9eda2e7d 100644 --- a/components/brave_wallet_ui/components/desktop/views/portfolio/components/accounts-and-transctions-list/index.tsx +++ b/components/brave_wallet_ui/components/desktop/views/portfolio/components/accounts-and-transctions-list/index.tsx @@ -41,6 +41,7 @@ export interface Props { selectedAsset: BraveWallet.BlockchainToken | undefined accounts: WalletAccountType[] selectedNetwork: BraveWallet.NetworkInfo + networkList: BraveWallet.NetworkInfo[] fullAssetFiatBalance: Amount formattedFullAssetBalance: string selectedAssetTransactions: BraveWallet.TransactionInfo[] @@ -66,6 +67,7 @@ const AccountsAndTransactionsList = (props: Props) => { selectedAssetTransactions, userVisibleTokensInfo, hideBalances, + networkList, onSelectAccount, onClickAddAccount, onSelectAsset, @@ -74,7 +76,7 @@ const AccountsAndTransactionsList = (props: Props) => { onSpeedupTransaction } = props - const getBalance = useBalance(selectedNetwork) + const getBalance = useBalance(networkList) const findAccount = (address: string): WalletAccountType | undefined => { return accounts.find((account) => address === account.address) @@ -82,7 +84,7 @@ const AccountsAndTransactionsList = (props: Props) => { const accountsList = React.useMemo(() => { if (selectedAsset?.isErc721) { - return accounts.filter((account) => Number(account.balance) !== 0) + return accounts.filter((account) => Number(account.nativeBalanceRegistry[selectedNetwork.chainId]) !== 0) } return accounts }, [selectedAsset, accounts]) diff --git a/components/brave_wallet_ui/components/desktop/views/portfolio/components/token-lists/index.tsx b/components/brave_wallet_ui/components/desktop/views/portfolio/components/token-lists/index.tsx index ee2ecf945903..28126cc36bc0 100644 --- a/components/brave_wallet_ui/components/desktop/views/portfolio/components/token-lists/index.tsx +++ b/components/brave_wallet_ui/components/desktop/views/portfolio/components/token-lists/index.tsx @@ -31,7 +31,7 @@ export interface Props { defaultCurrencies: DefaultCurrencies userAssetList: UserAssetInfoType[] hideBalances: boolean - selectedNetwork: BraveWallet.NetworkInfo + networks: BraveWallet.NetworkInfo[] onSetFilteredAssetList: (filteredList: UserAssetInfoType[]) => void onSelectAsset: (asset: BraveWallet.BlockchainToken | undefined) => () => void onShowAssetModal: () => void @@ -44,7 +44,7 @@ const TokenLists = (props: Props) => { defaultCurrencies, userAssetList, hideBalances, - selectedNetwork, + networks, onSelectAsset, onShowAssetModal, onSetFilteredAssetList @@ -73,16 +73,16 @@ const TokenLists = (props: Props) => { return ( <> - {filteredAssetList.filter((asset) => !asset.asset.isErc721).map((item) => + {filteredAssetList.filter((asset) => !asset.asset.isErc721).map((item, index) => )} {erc721TokenList.length !== 0 && @@ -99,7 +99,7 @@ const TokenLists = (props: Props) => { assetBalance={item.assetBalance} token={item.asset} hideBalances={hideBalances} - selectedNetwork={selectedNetwork} + networks={networks} /> )} diff --git a/components/brave_wallet_ui/components/desktop/views/portfolio/index.tsx b/components/brave_wallet_ui/components/desktop/views/portfolio/index.tsx index 413044556f1f..d802faeb54d6 100644 --- a/components/brave_wallet_ui/components/desktop/views/portfolio/index.tsx +++ b/components/brave_wallet_ui/components/desktop/views/portfolio/index.tsx @@ -9,12 +9,14 @@ import { UserAssetInfoType, DefaultCurrencies, AddAccountNavTypes + // SupportedTestNetworks } from '../../../../constants/types' import { getLocale } from '../../../../../common/locale' import { CurrencySymbols } from '../../../../utils/currency-symbols' // Utils import { sortTransactionByDate } from '../../../../utils/tx-utils' +import { getTokensNetwork, getTokensCoinType } from '../../../../utils/network-utils' import Amount from '../../../../utils/amount' // Options @@ -25,7 +27,6 @@ import { BackButton, withPlaceholderIcon } from '../../../shared' import { ChartControlBar, LineChart, - SelectNetworkDropdown, EditVisibleAssetsModal, WithHideBalancePlaceholder } from '../../' @@ -45,6 +46,7 @@ import { BalanceText, AssetIcon, AssetRow, + AssetColumn, PriceRow, AssetNameText, DetailText, @@ -54,7 +56,8 @@ import { PercentText, ArrowIcon, BalanceRow, - ShowBalanceButton + ShowBalanceButton, + NetworkDescription } from './style' export interface Props { @@ -63,7 +66,6 @@ export interface Props { onSelectAsset: (asset: BraveWallet.BlockchainToken | undefined) => void onSelectAccount: (account: WalletAccountType) => void onClickAddAccount: (tabId: AddAccountNavTypes) => () => void - onSelectNetwork: (network: BraveWallet.NetworkInfo) => void onAddCustomAsset: (token: BraveWallet.BlockchainToken) => void onShowVisibleAssetsModal: (showModal: boolean) => void onUpdateVisibleAssets: (updatedTokensList: BraveWallet.BlockchainToken[]) => void @@ -102,7 +104,6 @@ const Portfolio = (props: Props) => { onSelectAsset, onSelectAccount, onClickAddAccount, - onSelectNetwork, onAddCustomAsset, onShowVisibleAssetsModal, onUpdateVisibleAssets, @@ -142,10 +143,6 @@ const Portfolio = (props: Props) => { const [hideBalances, setHideBalances] = React.useState(false) const parseTransaction = useTransactionParser(selectedNetwork, accounts, transactionSpotPrices, userVisibleTokensInfo) - const toggleShowNetworkDropdown = () => { - setShowNetworkDropdown(!showNetworkDropdown) - } - const onSetFilteredAssetList = (filteredList: UserAssetInfoType[]) => { setfilteredAssetList(filteredList) } @@ -191,11 +188,6 @@ const Portfolio = (props: Props) => { } } - const onClickSelectNetwork = (network: BraveWallet.NetworkInfo) => () => { - onSelectNetwork(network) - toggleShowNetworkDropdown() - } - const onHideNetworkDropdown = () => { if (showNetworkDropdown) { setShowNetworkDropdown(false) @@ -224,7 +216,13 @@ const Portfolio = (props: Props) => { }, [selectedAsset, transactions]) const fullAssetBalances = React.useMemo(() => { - return filteredAssetList.find((asset) => asset.asset.contractAddress.toLowerCase() === selectedAsset?.contractAddress.toLowerCase()) + if (selectedAsset?.contractAddress === '') { + return filteredAssetList.find((asset) => asset.asset.symbol.toLowerCase() === selectedAsset?.symbol.toLowerCase()) + } + return filteredAssetList.find( + (asset) => + asset.asset.contractAddress.toLowerCase() === selectedAsset?.contractAddress.toLowerCase() + ) }, [filteredAssetList, selectedAsset]) const AssetIconWithPlaceholder = React.useMemo(() => { @@ -250,6 +248,21 @@ const Portfolio = (props: Props) => { setHideBalances(!hideBalances) } + const filteredAccountsByCoinType = React.useMemo(() => { + if (!selectedAsset) { + return [] + } + const coinType = getTokensCoinType(networkList, selectedAsset) + return accounts.filter((account) => account.coin === coinType) + }, [networkList, accounts, selectedAsset]) + + const selectedAssetsNetwork = React.useMemo(() => { + if (!selectedAsset) { + return + } + return getTokensNetwork(networkList, selectedAsset) + }, [selectedAsset, networkList]) + return ( @@ -259,15 +272,6 @@ const Portfolio = (props: Props) => { ) : ( )} - {!selectedAsset?.isErc721 && - - } {!selectedAsset?.isErc721 && @@ -295,10 +299,13 @@ const Portfolio = (props: Props) => { {!selectedAsset.isErc721 && - - {selectedAsset.name} + + + {selectedAsset.name} + {selectedAsset.symbol} on {selectedAssetsNetwork?.chainName ?? ''} + - {selectedAsset.name} {getLocale('braveWalletPrice')} ({selectedAsset.symbol}) + {/* {selectedAsset.name} {getLocale('braveWalletPrice')} ({selectedAsset.symbol}) */} {CurrencySymbols[defaultCurrencies.fiat]}{hoverPrice || (selectedAssetFiatPrice ? new Amount(selectedAssetFiatPrice.price).formatAsFiat() : 0.00)} @@ -340,7 +347,7 @@ const Portfolio = (props: Props) => { } */} { onRetryTransaction={onRetryTransaction} onSpeedupTransaction={onSpeedupTransaction} hideBalances={hideBalances} + networkList={networkList} /> {!selectedAsset && { userAssetList={userAssetList} filteredAssetList={filteredAssetList} tokenPrices={transactionSpotPrices} - selectedNetwork={selectedNetwork} + networks={networkList} onSetFilteredAssetList={onSetFilteredAssetList} onSelectAsset={selectAsset} onShowAssetModal={toggleShowVisibleAssetModal} @@ -378,6 +386,7 @@ const Portfolio = (props: Props) => { onClose={toggleShowVisibleAssetModal} onAddCustomAsset={onAddCustomAsset} selectedNetwork={selectedNetwork} + networkList={networkList} onFindTokenInfoByContractAddress={onFindTokenInfoByContractAddress} foundTokenInfoByContractAddress={foundTokenInfoByContractAddress} onUpdateVisibleAssets={onUpdateVisibleAssets} diff --git a/components/brave_wallet_ui/components/desktop/views/portfolio/style.ts b/components/brave_wallet_ui/components/desktop/views/portfolio/style.ts index 39c769de3aab..95f03c89b881 100644 --- a/components/brave_wallet_ui/components/desktop/views/portfolio/style.ts +++ b/components/brave_wallet_ui/components/desktop/views/portfolio/style.ts @@ -81,7 +81,14 @@ export const AssetRow = styled.div` flex-direction: row; align-items: center; justify-content: center; - margin-bottom: 9px; + margin-bottom: 20px; +` + +export const AssetColumn = styled.div` + display: flex; + flex-direction: column; + align-items: flex-start; + justify-content: center; ` export const PriceRow = styled.div` @@ -100,6 +107,14 @@ export const AssetNameText = styled.span` color: ${(p) => p.theme.color.text01}; ` +export const NetworkDescription = styled.span` + font-family: Poppins; + font-size: 14px; + line-height: 16px; + letter-spacing: 0.02em; + color: ${(p) => p.theme.color.text02}; +` + export const DetailText = styled.span` font-family: Poppins; font-size: 13px; diff --git a/components/brave_wallet_ui/components/extension/assets-panel/index.tsx b/components/brave_wallet_ui/components/extension/assets-panel/index.tsx index a39b95120d80..364346d2fd25 100644 --- a/components/brave_wallet_ui/components/extension/assets-panel/index.tsx +++ b/components/brave_wallet_ui/components/extension/assets-panel/index.tsx @@ -21,7 +21,7 @@ export interface Props { spotPrices: BraveWallet.AssetPrice[] userAssetList: BraveWallet.BlockchainToken[] defaultCurrencies: DefaultCurrencies - selectedNetwork: BraveWallet.NetworkInfo + networkList: BraveWallet.NetworkInfo[] selectedAccount: WalletAccountType onAddAsset: () => void } @@ -31,12 +31,12 @@ const AssetsPanel = (props: Props) => { userAssetList, spotPrices, defaultCurrencies, - selectedNetwork, selectedAccount, + networkList, onAddAsset } = props - const getBalance = useBalance(selectedNetwork) + const getBalance = useBalance(networkList) const onClickAsset = (symbol: string) => () => { const url = `brave://wallet${WalletRoutes.Portfolio}/${symbol}` @@ -62,7 +62,7 @@ const AssetsPanel = (props: Props) => { key={asset.contractAddress} assetBalance={getBalance(selectedAccount, asset)} token={asset} - selectedNetwork={selectedNetwork} + networks={networkList} isPanel={true} /> )} diff --git a/components/brave_wallet_ui/components/extension/connected-panel/index.tsx b/components/brave_wallet_ui/components/extension/connected-panel/index.tsx index 02e4b943ac5e..e9f3d92a949b 100644 --- a/components/brave_wallet_ui/components/extension/connected-panel/index.tsx +++ b/components/brave_wallet_ui/components/extension/connected-panel/index.tsx @@ -110,14 +110,14 @@ const ConnectedPanel = (props: Props) => { return !BuySupportedChains.includes(selectedNetwork.chainId) }, [BuySupportedChains, selectedNetwork]) - const formattedAssetBalance = new Amount(selectedAccount.balance) + const formattedAssetBalance = new Amount(selectedAccount.nativeBalanceRegistry[selectedNetwork.chainId]) .divideByDecimals(selectedNetwork.decimals) .formatAsAsset(6, selectedNetwork.symbol) const { computeFiatAmount } = usePricing(spotPrices) const selectedAccountFiatBalance = React.useMemo(() => computeFiatAmount( - selectedAccount.balance, selectedNetwork.symbol, selectedNetwork.decimals + selectedAccount.nativeBalanceRegistry[selectedNetwork.chainId], selectedNetwork.symbol, selectedNetwork.decimals ), [computeFiatAmount, selectedNetwork, selectedAccount]) const onClickViewOnBlockExplorer = useExplorer(selectedNetwork) diff --git a/components/brave_wallet_ui/components/shared/create-network-icon/index.tsx b/components/brave_wallet_ui/components/shared/create-network-icon/index.tsx index 73a38702ded1..a5a069d43cac 100644 --- a/components/brave_wallet_ui/components/shared/create-network-icon/index.tsx +++ b/components/brave_wallet_ui/components/shared/create-network-icon/index.tsx @@ -2,7 +2,7 @@ import * as React from 'react' import { create } from 'ethereum-blockies' // Constants -import { BraveWallet } from '../../../constants/types' +import { BraveWallet, SupportedTestNetworks } from '../../../constants/types' // Utils import { stripERC20TokenImageURL, isRemoteImageURL, isValidIconExtension } from '../../../utils/string-utils' @@ -46,16 +46,6 @@ function CreateNetworkIcon (props: Props) { return false }, [isRemoteURL, isDataURL, networkImageURL]) - // Will remove these hardcoded chainId's. - // We need to return a normal Ethereum Icon URL for Ethereum Mainnet - // and return a grayed out Ethereum Icon Url for Ethereum Test Networks from the backend. - const testNetworkList = [ - BraveWallet.RINKEBY_CHAIN_ID, - BraveWallet.ROPSTEN_CHAIN_ID, - BraveWallet.GOERLI_CHAIN_ID, - BraveWallet.KOVAN_CHAIN_ID, - BraveWallet.LOCALHOST_CHAIN_ID - ] const needsPlaceholder = nativeAsset.logo === '' && (networkImageURL === '' || !isValidIcon) const orb = React.useMemo(() => { @@ -86,7 +76,7 @@ function CreateNetworkIcon (props: Props) { return ( () => void } function SelectAccount (props: Props) { const { accounts, selectedAccount, onSelectAccount } = props + + // MULTICHAIN: Remove me once we support SOL and FIL transaction creation. + const accountsList = React.useMemo(() => { + return accounts.filter((account) => + account.coin !== BraveWallet.CoinType.SOL && + account.coin !== BraveWallet.CoinType.FIL) + }, [accounts]) + return ( <> - {accounts.map((account) => + {accountsList.map((account) => { + return networks.filter( + (network) => + network.coin !== BraveWallet.CoinType.SOL && + network.coin !== BraveWallet.CoinType.FIL + ) + }, [networks]) + return ( <> - {networks.map((network) => + {networkList.map((network) => diff --git a/components/brave_wallet_ui/constants/types.ts b/components/brave_wallet_ui/constants/types.ts index feba21c1ea19..eb684e442a6a 100644 --- a/components/brave_wallet_ui/constants/types.ts +++ b/components/brave_wallet_ui/constants/types.ts @@ -24,7 +24,7 @@ export interface WalletAccountType { name: string address: string tokenBalanceRegistry: TokenBalanceRegistry - balance: string + nativeBalanceRegistry: TokenBalanceRegistry accountType: 'Primary' | 'Secondary' | 'Ledger' | 'Trezor' deviceId?: string coin: BraveWallet.CoinType @@ -189,6 +189,7 @@ export interface WalletState { hasInitialized: boolean isFilecoinEnabled: boolean isSolanaEnabled: boolean + isTestNetworksEnabled: boolean isWalletCreated: boolean isWalletLocked: boolean favoriteApps: BraveWallet.AppItem[] @@ -214,7 +215,9 @@ export interface WalletState { gasEstimates?: BraveWallet.GasEstimation1559 connectedAccounts: WalletAccountType[] isMetaMaskInstalled: boolean + selectedCoin: BraveWallet.CoinType defaultCurrencies: DefaultCurrencies + defaultNetworks: BraveWallet.NetworkInfo[] } export interface PanelState { @@ -334,7 +337,18 @@ export interface GetAllTokensReturnInfo { } export interface GetNativeAssetBalancesReturnInfo { - balances: BraveWallet.JsonRpcService_GetBalance_ResponseParams[] + balances: BraveWallet.JsonRpcService_GetBalance_ResponseParams[][] +} + +export interface BalancePayload { + balance: string + error: number + errorMessage: string + chainId: string +} + +export interface GetNativeAssetBalancesPayload { + balances: BalancePayload[][] } export interface GetBlockchainTokenBalanceReturnInfo { @@ -563,3 +577,26 @@ export interface NFTMetadataReturnType { logo: string } } + +export const SupportedCoinTypes = [ + BraveWallet.CoinType.ETH, + BraveWallet.CoinType.SOL, + BraveWallet.CoinType.FIL +] + +export const SupportedTestNetworks = [ + BraveWallet.RINKEBY_CHAIN_ID, + BraveWallet.ROPSTEN_CHAIN_ID, + BraveWallet.GOERLI_CHAIN_ID, + BraveWallet.KOVAN_CHAIN_ID, + BraveWallet.LOCALHOST_CHAIN_ID, + BraveWallet.SOLANA_DEVNET, + BraveWallet.SOLANA_TESTNET, + BraveWallet.FILECOIN_TESTNET +] + +export enum CoinTypesMap { + ETH = 60, + FIL = 461, + SOL = 501 +} diff --git a/components/brave_wallet_ui/options/asset-options.ts b/components/brave_wallet_ui/options/asset-options.ts index 5b2d0dde4c52..c3ef935f670f 100644 --- a/components/brave_wallet_ui/options/asset-options.ts +++ b/components/brave_wallet_ui/options/asset-options.ts @@ -10,7 +10,9 @@ import { BNBIconUrl, BTCIconUrl, ETHIconUrl, - ZRXIconUrl + ZRXIconUrl, + SOLIconUrl, + FILECOINIconUrl } from '../assets/asset-icons' import { CeloIcon, @@ -26,6 +28,14 @@ export function makeNetworkAsset (network: BraveWallet.NetworkInfo) { logo = OptimismIcon break + case network.symbol.toUpperCase() === 'SOL': + logo = SOLIconUrl + break + + case network.symbol.toUpperCase() === 'FIL': + logo = FILECOINIconUrl + break + case network.chainId === BraveWallet.POLYGON_MAINNET_CHAIN_ID: logo = 'chrome://erc-token-images/matic.png' break @@ -64,7 +74,8 @@ export function makeNetworkAsset (network: BraveWallet.NetworkInfo) { decimals: network.decimals, visible: true, tokenId: '', - coingeckoId: '' + coingeckoId: '', + chainId: network.chainId } as BraveWallet.BlockchainToken } @@ -106,7 +117,8 @@ export const NewAssetOptions: BraveWallet.BlockchainToken[] = [ decimals: 18, visible: true, tokenId: '', - coingeckoId: '' + coingeckoId: '', + chainId: '0x1' }, { contractAddress: '2', @@ -118,7 +130,8 @@ export const NewAssetOptions: BraveWallet.BlockchainToken[] = [ decimals: 18, visible: true, tokenId: '', - coingeckoId: '' + coingeckoId: '', + chainId: '0x1' }, { contractAddress: '3', @@ -130,7 +143,8 @@ export const NewAssetOptions: BraveWallet.BlockchainToken[] = [ decimals: 18, visible: true, tokenId: '', - coingeckoId: '' + coingeckoId: '', + chainId: '0x3' }, { contractAddress: '4', @@ -142,7 +156,8 @@ export const NewAssetOptions: BraveWallet.BlockchainToken[] = [ decimals: 18, visible: true, tokenId: '', - coingeckoId: '' + coingeckoId: '', + chainId: '0x1' }, { contractAddress: '5', @@ -154,7 +169,8 @@ export const NewAssetOptions: BraveWallet.BlockchainToken[] = [ decimals: 18, visible: true, tokenId: '', - coingeckoId: '' + coingeckoId: '', + chainId: '0x1' }, { contractAddress: '6', @@ -166,7 +182,8 @@ export const NewAssetOptions: BraveWallet.BlockchainToken[] = [ decimals: 18, visible: true, tokenId: '', - coingeckoId: '' + coingeckoId: '', + chainId: '0x1' }, { contractAddress: '7', @@ -178,13 +195,17 @@ export const NewAssetOptions: BraveWallet.BlockchainToken[] = [ decimals: 0, visible: true, tokenId: '0x42a5', - coingeckoId: '' + coingeckoId: '', + chainId: '0x1' } ] // Use only with storybook as dummy data. export const AccountAssetOptions: BraveWallet.BlockchainToken[] = [ - ETH, + { + ...ETH, + chainId: '0x1' + }, { contractAddress: '0x0D8775F648430679A709E98d2b0Cb6250d2887EF', name: 'Basic Attention Token', @@ -195,7 +216,8 @@ export const AccountAssetOptions: BraveWallet.BlockchainToken[] = [ decimals: 18, visible: true, tokenId: '', - coingeckoId: '' + coingeckoId: '', + chainId: '0x1' }, { contractAddress: '3', @@ -207,7 +229,8 @@ export const AccountAssetOptions: BraveWallet.BlockchainToken[] = [ decimals: 8, visible: true, tokenId: '', - coingeckoId: '' + coingeckoId: '', + chainId: '0x1' }, { contractAddress: '4', @@ -219,7 +242,8 @@ export const AccountAssetOptions: BraveWallet.BlockchainToken[] = [ decimals: 8, visible: true, tokenId: '', - coingeckoId: '' + coingeckoId: '', + chainId: '0x1' }, { contractAddress: '5', @@ -231,7 +255,8 @@ export const AccountAssetOptions: BraveWallet.BlockchainToken[] = [ decimals: 8, visible: true, tokenId: '', - coingeckoId: '' + coingeckoId: '', + chainId: '0x1' }, { contractAddress: '0xE41d2489571d322189246DaFA5ebDe1F4699F498', @@ -243,6 +268,7 @@ export const AccountAssetOptions: BraveWallet.BlockchainToken[] = [ decimals: 18, visible: true, tokenId: '', - coingeckoId: '' + coingeckoId: '', + chainId: '0x1' } ] diff --git a/components/brave_wallet_ui/page/async/wallet_page_async_handler.ts b/components/brave_wallet_ui/page/async/wallet_page_async_handler.ts index f6f2dca131e9..735098afa131 100644 --- a/components/brave_wallet_ui/page/async/wallet_page_async_handler.ts +++ b/components/brave_wallet_ui/page/async/wallet_page_async_handler.ts @@ -87,11 +87,10 @@ handler.on(WalletPageActions.selectAsset.getType(), async (store: Store, payload const walletState = getWalletState(store) const defaultFiat = walletState.defaultCurrencies.fiat.toLowerCase() const defaultCrypto = walletState.defaultCurrencies.crypto.toLowerCase() - const selectedNetwork = walletState.selectedNetwork if (payload.asset) { const selectedAsset = payload.asset - const defaultPrices = await assetRatioService.getPrice([GetTokenParam(selectedNetwork, selectedAsset)], [defaultFiat, defaultCrypto], payload.timeFrame) - const priceHistory = await assetRatioService.getPriceHistory(GetTokenParam(selectedNetwork, selectedAsset), defaultFiat, payload.timeFrame) + const defaultPrices = await assetRatioService.getPrice([GetTokenParam(selectedAsset)], [defaultFiat, defaultCrypto], payload.timeFrame) + const priceHistory = await assetRatioService.getPriceHistory(GetTokenParam(selectedAsset), defaultFiat, payload.timeFrame) store.dispatch(WalletPageActions.updatePriceInfo({ priceHistory: priceHistory, defaultFiatPrice: defaultPrices.values[0], defaultCryptoPrice: defaultPrices.values[1], timeFrame: payload.timeFrame })) } else { store.dispatch(WalletPageActions.updatePriceInfo({ priceHistory: undefined, defaultFiatPrice: undefined, defaultCryptoPrice: undefined, timeFrame: payload.timeFrame })) diff --git a/components/brave_wallet_ui/page/container.tsx b/components/brave_wallet_ui/page/container.tsx index 71a0165a9a31..39b920cc43b0 100644 --- a/components/brave_wallet_ui/page/container.tsx +++ b/components/brave_wallet_ui/page/container.tsx @@ -38,6 +38,7 @@ import BackupWallet from '../stories/screens/backup-wallet' import Amount from '../utils/amount' import { GetBuyOrFaucetUrl } from '../utils/buy-asset-url' import { mojoTimeDeltaToJSDate } from '../../common/mojomUtils' +import { getTokensCoinType } from '../utils/network-utils' import { findENSAddress, @@ -134,8 +135,8 @@ function Container (props: Props) { sendAssetOptions, buyAssetOptions } = useAssets( - accounts, selectedAccount, + networkList, selectedNetwork, fullTokenList, userVisibleTokensInfo, @@ -176,6 +177,7 @@ function Container (props: Props) { } = useSwap( selectedAccount, selectedNetwork, + networkList, swapAssetOptions, props.walletPageActions.fetchPageSwapQuote, getERC20Allowance, @@ -217,20 +219,19 @@ function Container (props: Props) { props.walletActions.setUserAssetVisible, props.walletActions.removeUserAsset, props.walletActions.refreshBalancesAndPriceHistory, - selectedNetwork, fullTokenList, userVisibleTokensInfo ) const { computeFiatAmount } = usePricing(transactionSpotPrices) - const getAccountBalance = useBalance(selectedNetwork) + const getAccountBalance = useBalance(networkList) const sendAssetBalance = getAccountBalance(selectedAccount, selectedSendAsset) const fromAssetBalance = getAccountBalance(selectedAccount, fromAsset) const toAssetBalance = getAccountBalance(selectedAccount, toAsset) const onSelectPresetAmountFactory = usePreset( selectedAccount, - selectedNetwork, + networkList, onSetFromAmount, onSetSendAmount, fromAsset, @@ -327,15 +328,22 @@ function Container (props: Props) { // This will scrape all the user's accounts and combine the asset balances for a single asset const fullAssetBalance = React.useCallback((asset: BraveWallet.BlockchainToken) => { - const amounts = accounts.map((account) => + const tokensCoinType = getTokensCoinType(networkList, asset) + const amounts = accounts.filter((account) => account.coin === tokensCoinType).map((account) => getAccountBalance(account, asset)) + // If a user has not yet created a FIL or SOL account, + // we return 0 until they create an account + if (amounts.length === 0) { + return '0' + } + return amounts.reduce(function (a, b) { return a !== '' && b !== '' ? new Amount(a).plus(b).format() : '' }) - }, [accounts, getAccountBalance]) + }, [accounts, networkList, getAccountBalance]) // This looks at the users asset list and returns the full balance for each asset const userAssetList = React.useMemo(() => { @@ -503,7 +511,10 @@ function Container (props: Props) { React.useEffect(() => { // Creates a list of Accepted Portfolio Routes const acceptedPortfolioRoutes = userVisibleTokensInfo.map((token) => { - return `${WalletRoutes.Portfolio}/${token.symbol}` + if (token.contractAddress === '') { + return `${WalletRoutes.Portfolio}/${token.symbol}` + } + return `${WalletRoutes.Portfolio}/${token.contractAddress}` }) // Creates a list of Accepted Account Routes const acceptedAccountRoutes = accounts.map((account) => { @@ -661,7 +672,6 @@ function Container (props: Props) { onHideAddModal={onHideAddModal} onUpdateAccountName={onUpdateAccountName} selectedNetwork={selectedNetwork} - onSelectNetwork={onSelectNetwork} isFetchingPortfolioPriceHistory={isFetchingPortfolioPriceHistory} onRemoveAccount={onRemoveAccount} privateKey={privateKey ?? ''} diff --git a/components/brave_wallet_ui/panel/container.tsx b/components/brave_wallet_ui/panel/container.tsx index c94d850e4b4e..fd27218f6860 100644 --- a/components/brave_wallet_ui/panel/container.tsx +++ b/components/brave_wallet_ui/panel/container.tsx @@ -62,7 +62,7 @@ import { import { AppsList } from '../options/apps-list-options' import LockPanel from '../components/extension/lock-panel' import { GetBuyOrFaucetUrl } from '../utils/buy-asset-url' -import { GetNetworkInfo } from '../utils/network-utils' +import { getNetworkInfo } from '../utils/network-utils' import { findENSAddress, findUnstoppableDomainAddress, @@ -146,8 +146,8 @@ function Container (props: Props) { buyAssetOptions, panelUserAssetList } = useAssets( - accounts, selectedAccount, + networkList, selectedNetwork, props.wallet.fullTokenList, userVisibleTokensInfo, @@ -188,6 +188,7 @@ function Container (props: Props) { } = useSwap( selectedAccount, selectedNetwork, + networkList, swapAssetOptions, props.walletPanelActions.fetchPanelSwapQuote, getERC20Allowance, @@ -232,14 +233,14 @@ function Container (props: Props) { } }, [hasIncorrectPassword]) - const getSelectedAccountBalance = useBalance(selectedNetwork) + const getSelectedAccountBalance = useBalance(networkList) const sendAssetBalance = getSelectedAccountBalance(selectedAccount, selectedSendAsset) const fromAssetBalance = getSelectedAccountBalance(selectedAccount, fromAsset) const toAssetBalance = getSelectedAccountBalance(selectedAccount, toAsset) const onSelectPresetAmountFactory = usePreset( selectedAccount, - selectedNetwork, + networkList, onSetFromAmount, onSetSendAmount, fromAsset, @@ -650,7 +651,7 @@ function Container (props: Props) { transactionQueueNumber={pendingTransactions.findIndex(tx => tx.id === selectedPendingTransaction.id) + 1} transactionsQueueLength={pendingTransactions.length} accounts={accounts} - selectedNetwork={GetNetworkInfo(selectedNetwork.chainId, networkList)} + selectedNetwork={getNetworkInfo(selectedNetwork.chainId, networkList)} transactionInfo={selectedPendingTransaction} transactionSpotPrices={transactionSpotPrices} visibleTokens={userVisibleTokensInfo} @@ -710,7 +711,7 @@ function Container (props: Props) { onApproveChangeNetwork={onApproveChangeNetwork} onCancel={onCancelChangeNetwork} onLearnMore={onNetworkLearnMore} - networkPayload={GetNetworkInfo(switchChainRequest.chainId, networkList)} + networkPayload={getNetworkInfo(switchChainRequest.chainId, networkList)} panelType='change' /> @@ -727,7 +728,7 @@ function Container (props: Props) { accounts={accounts} onCancel={onCancelSigning} onSign={onSignData} - selectedNetwork={GetNetworkInfo(selectedNetwork.chainId, networkList)} + selectedNetwork={getNetworkInfo(selectedNetwork.chainId, networkList)} // Pass a boolean here if the signing method is risky showWarning={false} /> @@ -922,7 +923,7 @@ function Container (props: Props) { onSubmit={onSubmitBuy} selectedAsset={selectedWyreAsset} buyAmount={buyAmount} - selectedNetwork={GetNetworkInfo(selectedNetwork.chainId, networkList)} + selectedNetwork={getNetworkInfo(selectedNetwork.chainId, networkList)} networkList={networkList} /> @@ -1037,10 +1038,10 @@ function Container (props: Props) { @@ -1081,7 +1082,7 @@ function Container (props: Props) { defaultCurrencies={defaultCurrencies} spotPrices={transactionSpotPrices} selectedAccount={selectedAccount} - selectedNetwork={GetNetworkInfo(selectedNetwork.chainId, networkList)} + selectedNetwork={getNetworkInfo(selectedNetwork.chainId, networkList)} isConnected={isConnectedToSite} navAction={navigateTo} onLockWallet={onLockWallet} diff --git a/components/brave_wallet_ui/stories/screens/buy-send-swap.tsx b/components/brave_wallet_ui/stories/screens/buy-send-swap.tsx index 835741850918..02362626291c 100644 --- a/components/brave_wallet_ui/stories/screens/buy-send-swap.tsx +++ b/components/brave_wallet_ui/stories/screens/buy-send-swap.tsx @@ -10,7 +10,8 @@ import { BuySupportedChains, SwapValidationErrorType, DefaultCurrencies, - AmountValidationErrorType + AmountValidationErrorType, + WalletAccountType } from '../../constants/types' import Swap from '../../components/buy-send-swap/tabs/swap-tab' import Send from '../../components/buy-send-swap/tabs/send-tab' @@ -20,7 +21,7 @@ import { } from '../../components/buy-send-swap' export interface Props { - accounts: UserAccountType[] + accounts: WalletAccountType[] networkList: BraveWallet.NetworkInfo[] orderType: OrderTypes selectedSendAsset: BraveWallet.BlockchainToken | undefined diff --git a/components/brave_wallet_ui/stories/screens/crypto-story-view.tsx b/components/brave_wallet_ui/stories/screens/crypto-story-view.tsx index 18c15b2b2db6..c30ebb1514b7 100644 --- a/components/brave_wallet_ui/stories/screens/crypto-story-view.tsx +++ b/components/brave_wallet_ui/stories/screens/crypto-story-view.tsx @@ -63,7 +63,6 @@ export interface Props { onDoneViewingPrivateKey: () => void onViewPrivateKey: (address: string, isDefault: boolean, coin: BraveWallet.CoinType) => void onRemoveAccount: (address: string, hardware: boolean, coin: BraveWallet.CoinType) => void - onSelectNetwork: (network: BraveWallet.NetworkInfo) => void onToggleAddModal: () => void onUpdateAccountName: (payload: UpdateAccountNamePayloadType) => { success: boolean } getBalance: (address: string) => Promise @@ -121,7 +120,6 @@ const CryptoStoryView = (props: Props) => { onImportAccount, onImportFilecoinAccount, onUpdateAccountName, - onSelectNetwork, onToggleAddModal, onRemoveAccount, onViewPrivateKey, @@ -297,7 +295,6 @@ const CryptoStoryView = (props: Props) => { onSelectAsset={onSelectAsset} onSelectAccount={onSelectAccount} onClickAddAccount={onClickAddAccount} - onSelectNetwork={onSelectNetwork} addUserAssetError={false} selectedAsset={selectedAsset} portfolioBalance={portfolioBalance} @@ -345,6 +342,7 @@ const CryptoStoryView = (props: Props) => { onRetryTransaction={onClickRetryTransaction} onSpeedupTransaction={onClickSpeedupTransaction} onCancelTransaction={onClickCancelTransaction} + networkList={networkList} /> } {showAddModal && diff --git a/components/brave_wallet_ui/stories/wallet-concept.tsx b/components/brave_wallet_ui/stories/wallet-concept.tsx index d2cfa540c6dd..b6933de2030c 100644 --- a/components/brave_wallet_ui/stories/wallet-concept.tsx +++ b/components/brave_wallet_ui/stories/wallet-concept.tsx @@ -405,7 +405,9 @@ export const _DesktopWalletConcept = (args: { onboarding: boolean, locked: boole id: id, name: name, address: wallet.address, - balance: singleAccountBalance(wallet), + nativeBalanceRegistry: { + ETH: singleAccountBalance(wallet) + }, asset: selectedAsset ? selectedAsset.symbol : '', accountType: 'Primary', tokenBalanceRegistry: {}, @@ -786,7 +788,6 @@ export const _DesktopWalletConcept = (args: { onboarding: boolean, locked: boole onToggleAddModal={onToggleAddModal} onUpdateAccountName={onUpdateAccountName} selectedNetwork={selectedNetwork} - onSelectNetwork={onSelectNetwork} isFetchingPortfolioPriceHistory={false} selectedPortfolioTimeline={selectedTimeline} onRemoveAccount={onRemoveAccount} @@ -831,7 +832,7 @@ export const _DesktopWalletConcept = (args: { onboarding: boolean, locked: boole orderExpiration={orderExpiration} slippageTolerance={slippageTolerance} swapFromAsset={fromAsset} - accounts={mockUserAccounts} + accounts={accounts} selectedNetwork={selectedNetwork} selectedAccount={selectedAccount} selectedTab={selectedWidgetTab} diff --git a/components/brave_wallet_ui/stories/wallet-extension-panels.tsx b/components/brave_wallet_ui/stories/wallet-extension-panels.tsx index 4b784798d229..c0736b5e035d 100644 --- a/components/brave_wallet_ui/stories/wallet-extension-panels.tsx +++ b/components/brave_wallet_ui/stories/wallet-extension-panels.tsx @@ -67,7 +67,9 @@ const accounts: WalletAccountType[] = [ id: '1', name: 'Account 1', address: '0x7d66c9ddAED3115d93Bd1790332f3Cd06Cf52B14', - balance: '311780000000000000', + nativeBalanceRegistry: { + ETH: '311780000000000000' + }, accountType: 'Primary', tokenBalanceRegistry: {}, coin: BraveWallet.CoinType.ETH @@ -76,7 +78,9 @@ const accounts: WalletAccountType[] = [ id: '2', name: 'Account 2', address: '0x73A29A1da97149722eB09c526E4eAd698895bDCf', - balance: '311780000000000000', + nativeBalanceRegistry: { + ETH: '311780000000000000' + }, accountType: 'Primary', tokenBalanceRegistry: {}, coin: BraveWallet.CoinType.ETH @@ -85,7 +89,9 @@ const accounts: WalletAccountType[] = [ id: '3', name: 'Account 3', address: '0x3f29A1da97149722eB09c526E4eAd698895b426', - balance: '311780000000000000', + nativeBalanceRegistry: { + ETH: '311780000000000000' + }, accountType: 'Primary', tokenBalanceRegistry: {}, coin: BraveWallet.CoinType.ETH @@ -396,7 +402,9 @@ export const _ConnectedPanel = (args: { locked: boolean }) => { id: '1', name: 'Account 1', address: '1', - balance: '0.31178', + nativeBalanceRegistry: { + ETH: '0.31178' + }, accountType: 'Primary', tokenBalanceRegistry: {}, coin: BraveWallet.CoinType.ETH @@ -526,7 +534,7 @@ export const _ConnectedPanel = (args: { locked: boolean }) => { } const onSelectPresetAmount = (percent: number) => { - const amount = Number(selectedAccount.balance) * percent + const amount = Number(selectedAccount.nativeBalanceRegistry[0]) * percent setFromAmount(amount.toString()) } @@ -695,7 +703,7 @@ export const _ConnectedPanel = (args: { locked: boolean }) => { addressWarning='' selectedAsset={selectedAsset} selectedAssetAmount={fromAmount} - selectedAssetBalance={selectedAccount.balance.toString()} + selectedAssetBalance={selectedAccount.nativeBalanceRegistry[0].toString()} toAddressOrUrl={toAddress} toAddress={toAddress} amountValidationError={undefined} @@ -742,11 +750,11 @@ export const _ConnectedPanel = (args: { locked: boolean }) => { {selectedPanel === 'assets' && } diff --git a/components/brave_wallet_ui/utils/api-utils.test.ts b/components/brave_wallet_ui/utils/api-utils.test.ts index 4542b375a9f5..291abadb6353 100644 --- a/components/brave_wallet_ui/utils/api-utils.test.ts +++ b/components/brave_wallet_ui/utils/api-utils.test.ts @@ -1,19 +1,21 @@ import { GetTokenParam, GetFlattenedAccountBalances } from './api-utils' -import { mockNetworks } from '../stories/mock-data/mock-networks' import { AccountAssetOptions } from '../options/asset-options' import { mockAccount } from '../common/constants/mocks' describe('Check token param', () => { test('Value should return contract address', () => { - expect(GetTokenParam(mockNetworks[0], AccountAssetOptions[1])).toEqual('0x0D8775F648430679A709E98d2b0Cb6250d2887EF') + expect(GetTokenParam(AccountAssetOptions[1])).toEqual('0x0D8775F648430679A709E98d2b0Cb6250d2887EF') }) test('Value should return symbol', () => { - expect(GetTokenParam(mockNetworks[1], AccountAssetOptions[1])).toEqual('bat') + expect(GetTokenParam({ + ...AccountAssetOptions[1], + contractAddress: '' + })).toEqual('bat') }) test('Value should return coingeckoId', () => { - expect(GetTokenParam(mockNetworks[1], { + expect(GetTokenParam({ ...AccountAssetOptions[1], coingeckoId: 'mockCoingeckoId' })).toEqual('mockCoingeckoId') diff --git a/components/brave_wallet_ui/utils/api-utils.ts b/components/brave_wallet_ui/utils/api-utils.ts index 562ab2dfb979..ec1426aee908 100644 --- a/components/brave_wallet_ui/utils/api-utils.ts +++ b/components/brave_wallet_ui/utils/api-utils.ts @@ -1,19 +1,21 @@ import { BraveWallet, WalletAccountType, GetFlattenedAccountBalancesReturnInfo } from '../constants/types' -export const GetTokenParam = (selectedNetwork: BraveWallet.NetworkInfo, token: BraveWallet.BlockchainToken): string => { +export const GetTokenParam = (token: BraveWallet.BlockchainToken): string => { if (token.coingeckoId) { return token.coingeckoId } - const isEthereumNetwork = selectedNetwork.chainId === BraveWallet.MAINNET_CHAIN_ID + const isEthereumNetwork = token.chainId === BraveWallet.MAINNET_CHAIN_ID if (!isEthereumNetwork) { return token.symbol.toLowerCase() } - return token.symbol.toLowerCase() === selectedNetwork.symbol.toLowerCase() - ? token.symbol.toLowerCase() - : token.contractAddress + if (token.contractAddress === '') { + return token.symbol.toLowerCase() + } + + return token.contractAddress } // This will get the sum balance for each token between all accounts @@ -29,7 +31,7 @@ export const GetFlattenedAccountBalances = (accounts: WalletAccountType[], userV .map(account => { const balance = token.contractAddress ? account.tokenBalanceRegistry[token.contractAddress.toLowerCase()] - : account.balance + : account.nativeBalanceRegistry[token.chainId] return balance || '0' }) diff --git a/components/brave_wallet_ui/utils/network-utils.test.ts b/components/brave_wallet_ui/utils/network-utils.test.ts index dff7e6651e6f..a3aaabb980d2 100644 --- a/components/brave_wallet_ui/utils/network-utils.test.ts +++ b/components/brave_wallet_ui/utils/network-utils.test.ts @@ -1,15 +1,24 @@ -import { GetNetworkInfo, emptyNetwork, reduceNetworkDisplayName } from './network-utils' +import { + getNetworkInfo, + emptyNetwork, + reduceNetworkDisplayName, + getNetworksByCoinType, + getTokensNetwork, + getTokensCoinType +} from './network-utils' import { mockNetworks } from '../stories/mock-data/mock-networks' +import { NewAssetOptions } from '../options/asset-options' +import { BraveWallet } from '../constants/types' describe('getNetworkInfo', () => { it('should return network info', () => { const chainId = mockNetworks[0].chainId - expect(GetNetworkInfo(chainId, mockNetworks)).toEqual(mockNetworks[0]) + expect(getNetworkInfo(chainId, mockNetworks)).toEqual(mockNetworks[0]) }) it('should return network object with default values if network with chainId is not found', () => { const chainId = 'fakeChainId' - expect(GetNetworkInfo(chainId, mockNetworks)).toEqual(emptyNetwork) + expect(getNetworkInfo(chainId, mockNetworks)).toEqual(emptyNetwork) }) }) @@ -29,3 +38,33 @@ describe('reduceNetworkDisplayName', () => { expect(reduceNetworkDisplayName(networkName)).toBe(expected) }) }) + +describe('getNetworksByCoinType', () => { + it('CoinType ETH, should return all ETH networks', () => { + expect(getNetworksByCoinType(mockNetworks, BraveWallet.CoinType.ETH)).toEqual(mockNetworks) + }) + it('CoinType random number, should return an empty array', () => { + expect(getNetworksByCoinType(mockNetworks, 3000)).toEqual([]) + }) +}) + +describe('getTokensNetwork', () => { + it('Ethereum with chainId 0x1, should return ETH Mainnet info', () => { + expect(getTokensNetwork(mockNetworks, NewAssetOptions[0])).toEqual(mockNetworks[0]) + }) + it('Binance Coin with chainId 0x3, should return ETH Ropsten Testnetwork info', () => { + expect(getTokensNetwork(mockNetworks, NewAssetOptions[2])).toEqual(mockNetworks[1]) + }) +}) + +describe('getTokensCoinType', () => { + it('Ethereum with chainId 0x1, should return CoinType ETH', () => { + expect(getTokensCoinType(mockNetworks, NewAssetOptions[0])).toEqual(BraveWallet.CoinType.ETH) + }) + it('Binance Coin with chainId 0x3, should return ETH CoinType ETH', () => { + expect(getTokensCoinType(mockNetworks, NewAssetOptions[2])).toEqual(BraveWallet.CoinType.ETH) + }) + it('Binance Coin with chainId 0x3333333458, should default to CoinType ETH', () => { + expect(getTokensCoinType(mockNetworks, { ...NewAssetOptions[2], chainId: '0x3333333458' })).toEqual(BraveWallet.CoinType.ETH) + }) +}) diff --git a/components/brave_wallet_ui/utils/network-utils.ts b/components/brave_wallet_ui/utils/network-utils.ts index d49876597b77..9c9ec2f8220e 100644 --- a/components/brave_wallet_ui/utils/network-utils.ts +++ b/components/brave_wallet_ui/utils/network-utils.ts @@ -10,13 +10,13 @@ export const emptyNetwork = { decimals: 0, coin: BraveWallet.CoinType.ETH, data: { - ethData: { + ethData: { isEip1559: true - } + } } } -export const GetNetworkInfo = (chainId: string, list: BraveWallet.NetworkInfo[]) => { +export const getNetworkInfo = (chainId: string, list: BraveWallet.NetworkInfo[]) => { for (let it of list) { if (it.chainId === chainId) { return it @@ -39,3 +39,30 @@ export const reduceNetworkDisplayName = (name: string) => { } } } + +export const getNetworksByCoinType = (networks: BraveWallet.NetworkInfo[], coin: BraveWallet.CoinType): BraveWallet.NetworkInfo[] => { + if (!networks) { + return [] + } + return networks.filter((network) => network.coin === coin) +} + +export const getTokensNetwork = (networks: BraveWallet.NetworkInfo[], token: BraveWallet.BlockchainToken): BraveWallet.NetworkInfo => { + if (!networks) { + return emptyNetwork + } + + const network = networks.filter((n) => n.chainId === token.chainId) + if (network.length > 1) { + return network?.find((n) => n.symbol.toLowerCase() === token.symbol.toLowerCase()) ?? emptyNetwork + } + + return network[0] ?? emptyNetwork +} + +export const getTokensCoinType = (networks: BraveWallet.NetworkInfo[], token: BraveWallet.BlockchainToken) => { + if (!networks) { + return '' + } + return getTokensNetwork(networks, token).coin || '' +}