diff --git a/.changeset/curvy-ladybugs-collect.md b/.changeset/curvy-ladybugs-collect.md new file mode 100644 index 000000000000..b8292e2361e8 --- /dev/null +++ b/.changeset/curvy-ladybugs-collect.md @@ -0,0 +1,6 @@ +--- +"live-mobile": minor +"@ledgerhq/native-ui": minor +--- + +Create a new tabSelector component inside native ui. Rename the existing one in DrawerTabSelector since it's used only in a drawer. Integration of the new assets/accounts lists in wallet screen diff --git a/apps/ledger-live-mobile/__tests__/jest-setup.js b/apps/ledger-live-mobile/__tests__/jest-setup.js index 295e2492c6f5..9810fa08797c 100644 --- a/apps/ledger-live-mobile/__tests__/jest-setup.js +++ b/apps/ledger-live-mobile/__tests__/jest-setup.js @@ -117,6 +117,11 @@ jest.mock("react-native-reanimated", () => { // Silence the warning: Animated: `useNativeDriver` is not supported because the native animated module is missing jest.mock("react-native/Libraries/Animated/NativeAnimatedHelper"); +jest.mock("~/analytics", () => ({ + ...jest.requireActual("~/analytics"), + track: jest.fn(), +})); + jest.mock("@react-native-firebase/messaging", () => ({ messaging: jest.fn(() => ({ hasPermission: jest.fn(() => Promise.resolve(true)), diff --git a/apps/ledger-live-mobile/src/components/RootNavigator/types/BaseNavigator.ts b/apps/ledger-live-mobile/src/components/RootNavigator/types/BaseNavigator.ts index 01afa72c8f12..3f737052b21e 100644 --- a/apps/ledger-live-mobile/src/components/RootNavigator/types/BaseNavigator.ts +++ b/apps/ledger-live-mobile/src/components/RootNavigator/types/BaseNavigator.ts @@ -304,6 +304,7 @@ export type BaseNavigatorStackParamList = { [NavigatorName.AnalyticsOptInPrompt]: NavigatorScreenParams; [ScreenName.MockedAddAssetButton]: undefined; + [ScreenName.MockedWalletScreen]: undefined; // WALLET SYNC [NavigatorName.WalletSync]: NavigatorScreenParams; diff --git a/apps/ledger-live-mobile/src/const/navigation.ts b/apps/ledger-live-mobile/src/const/navigation.ts index 2ac7d07a081d..8006569c73a5 100644 --- a/apps/ledger-live-mobile/src/const/navigation.ts +++ b/apps/ledger-live-mobile/src/const/navigation.ts @@ -529,6 +529,7 @@ export enum ScreenName { LedgerSyncDeepLinkHandler = "LedgerSyncDeepLinkHandler", MockedAddAssetButton = "MockedAddAssetButton", + MockedWalletScreen = "MockedWalletScreen", GenericLandingPage = "GenericLandingPage", // Web3Hub diff --git a/apps/ledger-live-mobile/src/locales/en/common.json b/apps/ledger-live-mobile/src/locales/en/common.json index 9e9109e52f0e..30f0df5aeb49 100644 --- a/apps/ledger-live-mobile/src/locales/en/common.json +++ b/apps/ledger-live-mobile/src/locales/en/common.json @@ -2342,7 +2342,8 @@ "seeMarket": "See all" }, "walletBalance": "Wallet balance", - "seelAllAssets": "See All Assets", + "seeAllAssets": "See all assets", + "seeAllAccounts": "See all accounts", "marketPriceSection": { "title": "{{currencyTicker}} market price", "currencyPrice": "1 {{currencyTicker}} price", diff --git a/apps/ledger-live-mobile/src/newArch/features/Accounts/components/AccountsListView/index.tsx b/apps/ledger-live-mobile/src/newArch/features/Accounts/components/AccountsListView/index.tsx index a2fd54cfe027..fa4a9bc7334b 100644 --- a/apps/ledger-live-mobile/src/newArch/features/Accounts/components/AccountsListView/index.tsx +++ b/apps/ledger-live-mobile/src/newArch/features/Accounts/components/AccountsListView/index.tsx @@ -6,6 +6,8 @@ import { Flex } from "@ledgerhq/native-ui"; import { Pressable } from "react-native"; import AccountItem from "./components/AccountItem"; import globalSyncRefreshControl from "~/components/globalSyncRefreshControl"; +import { withDiscreetMode } from "~/context/DiscreetModeContext"; +import isEqual from "lodash/isEqual"; const ESTIMED_ITEM_SIZE = 150; @@ -50,4 +52,4 @@ const AccountsListView: React.FC = props => { return ; }; -export default AccountsListView; +export default React.memo(withDiscreetMode(AccountsListView), isEqual); diff --git a/apps/ledger-live-mobile/src/newArch/features/Accounts/components/AddAccountButton/index.tsx b/apps/ledger-live-mobile/src/newArch/features/Accounts/components/AddAccountButton/index.tsx index 75dbf5011aed..ea85222ab161 100644 --- a/apps/ledger-live-mobile/src/newArch/features/Accounts/components/AddAccountButton/index.tsx +++ b/apps/ledger-live-mobile/src/newArch/features/Accounts/components/AddAccountButton/index.tsx @@ -20,12 +20,16 @@ const StyledPressable = styled(Pressable)` column-gap: 12px; `; -const AddAccountButton: React.FC = () => { +type Props = { + sourceScreenName: string; +}; + +const AddAccountButton: React.FC = ({ sourceScreenName }) => { const { t } = useTranslation(); const [isAddModalOpened, setAddModalOpened] = useState(false); const openAddModal = () => { - track("button_clicked", { button: "add a new account", page: "Accounts" }); + track("button_clicked", { button: "Add a new account", page: sourceScreenName }); setAddModalOpened(true); }; const closeAddModal = () => setAddModalOpened(false); diff --git a/apps/ledger-live-mobile/src/newArch/features/Accounts/screens/AccountsList/index.tsx b/apps/ledger-live-mobile/src/newArch/features/Accounts/screens/AccountsList/index.tsx index 3c9640564561..011fa126506a 100644 --- a/apps/ledger-live-mobile/src/newArch/features/Accounts/screens/AccountsList/index.tsx +++ b/apps/ledger-live-mobile/src/newArch/features/Accounts/screens/AccountsList/index.tsx @@ -33,7 +33,7 @@ export default function AccountsList({ route }: Props) { <> - + {showHeader && ( {t("accounts.title")} @@ -49,7 +49,7 @@ export default function AccountsList({ route }: Props) { )} - {canAddAccount && } + {canAddAccount && } diff --git a/apps/ledger-live-mobile/src/newArch/features/Assets/Navigator.tsx b/apps/ledger-live-mobile/src/newArch/features/Assets/Navigator.tsx index d1217258ca10..6d0f120ad376 100644 --- a/apps/ledger-live-mobile/src/newArch/features/Assets/Navigator.tsx +++ b/apps/ledger-live-mobile/src/newArch/features/Assets/Navigator.tsx @@ -2,31 +2,33 @@ import React, { useCallback, useMemo } from "react"; import { Platform } from "react-native"; import { createStackNavigator } from "@react-navigation/stack"; import { useTheme } from "styled-components/native"; -import { useRoute } from "@react-navigation/native"; +import { useNavigation, useRoute } from "@react-navigation/native"; import { ScreenName } from "~/const"; import { getStackNavigatorConfig } from "~/navigation/navigatorConfig"; import { track } from "~/analytics"; -import { NavigationHeaderCloseButtonAdvanced } from "~/components/NavigationHeaderCloseButton"; import { AssetsNavigatorParamsList } from "./types"; import AssetsList from "./screens/AssetsList"; +import { NavigationHeaderBackButton } from "~/components/NavigationHeaderBackButton"; export default function Navigator() { const { colors } = useTheme(); + const navigation = useNavigation(); const route = useRoute(); - const onClose = useCallback(() => { + const goBack = useCallback(() => { track("button_clicked", { - button: "Close", - screen: route.name, + button: "Back", + page: route.name, }); - }, [route]); + navigation.goBack(); + }, [navigation, route]); const stackNavigationConfig = useMemo( () => ({ ...getStackNavigatorConfig(colors, true), - headerRight: () => , + headerLeft: () => , }), - [colors, onClose], + [colors, goBack], ); return ( diff --git a/apps/ledger-live-mobile/src/newArch/features/Assets/components/AssetsListView/index.tsx b/apps/ledger-live-mobile/src/newArch/features/Assets/components/AssetsListView/index.tsx index 8721578a7254..b3464be474ed 100644 --- a/apps/ledger-live-mobile/src/newArch/features/Assets/components/AssetsListView/index.tsx +++ b/apps/ledger-live-mobile/src/newArch/features/Assets/components/AssetsListView/index.tsx @@ -7,6 +7,8 @@ import AssetItem from "./components/AssetItem"; import { Asset } from "~/types/asset"; import BigNumber from "bignumber.js"; import globalSyncRefreshControl from "~/components/globalSyncRefreshControl"; +import { withDiscreetMode } from "~/context/DiscreetModeContext"; +import isEqual from "lodash/isEqual"; const ESTIMED_ITEM_SIZE = 150; @@ -53,4 +55,4 @@ const AssetsListView: React.FC = props => { return ; }; -export default AssetsListView; +export default React.memo(withDiscreetMode(AssetsListView), isEqual); diff --git a/apps/ledger-live-mobile/src/newArch/features/Assets/screens/AssetsList/index.tsx b/apps/ledger-live-mobile/src/newArch/features/Assets/screens/AssetsList/index.tsx index 0e9a8e8044b4..55c90f259b5f 100644 --- a/apps/ledger-live-mobile/src/newArch/features/Assets/screens/AssetsList/index.tsx +++ b/apps/ledger-live-mobile/src/newArch/features/Assets/screens/AssetsList/index.tsx @@ -31,7 +31,7 @@ export default function AssetsList({ route }: Props) { <> - + {showHeader && ( {t("assets.title")} diff --git a/apps/ledger-live-mobile/src/newArch/features/WalletSync/screens/Synchronize/QrCodeMethod.tsx b/apps/ledger-live-mobile/src/newArch/features/WalletSync/screens/Synchronize/QrCodeMethod.tsx index ca15131a7a62..c8f63998f6a5 100644 --- a/apps/ledger-live-mobile/src/newArch/features/WalletSync/screens/Synchronize/QrCodeMethod.tsx +++ b/apps/ledger-live-mobile/src/newArch/features/WalletSync/screens/Synchronize/QrCodeMethod.tsx @@ -1,5 +1,5 @@ import React from "react"; -import { Flex, TabSelector } from "@ledgerhq/native-ui"; +import { DrawerTabSelector, Flex } from "@ledgerhq/native-ui"; import QrCode from "LLM/features/WalletSync/components/Synchronize/QrCode"; import ScanQrCode from "../../components/Synchronize/ScanQrCode"; import { Options, OptionsType } from "LLM/features/WalletSync/types/Activation"; @@ -59,7 +59,7 @@ const QrCodeMethod = ({ return ( - void; }; -const maxAssetsToDisplay = 5; +const maxItemsToDysplay = 5; const PortfolioAssets = ({ hideEmptyTokenAccount, openAddModal }: Props) => { const { t } = useTranslation(); @@ -45,13 +47,35 @@ const PortfolioAssets = ({ hideEmptyTokenAccount, openAddModal }: Props) => { !blacklistedTokenIdsSet.has(asset.currency.id) ); }) - .slice(0, maxAssetsToDisplay), + .slice(0, maxItemsToDysplay), [distribution, blacklistedTokenIdsSet], ); - const goToAssets = useCallback( + const { selectedTab, handleToggle, handleLayout, assetsAnimatedStyle, accountsAnimatedStyle } = + useListsAnimation(TAB_OPTIONS.Assets); + + const showAssets = selectedTab === TAB_OPTIONS.Assets; + const showAccounts = selectedTab === TAB_OPTIONS.Accounts; + + const onPressButton = useCallback( (uiEvent: GestureResponderEvent) => { startNavigationTTITimer({ source: ScreenName.Portfolio, uiEvent }); + track("button_clicked", { + button: showAssets ? "See all assets" : "See all accounts", + page: "Wallet", + }); + if (!showAssets && isAccountListUIEnabled) { + navigation.navigate(NavigatorName.Accounts, { + screen: ScreenName.AccountsList, + params: { + sourceScreenName: ScreenName.Portfolio, + showHeader: true, + canAddAccount: true, + isSyncEnabled: true, + }, + }); + return; + } if (isAccountListUIEnabled) { navigation.navigate(NavigatorName.Assets, { screen: ScreenName.AssetsList, @@ -67,9 +91,12 @@ const PortfolioAssets = ({ hideEmptyTokenAccount, openAddModal }: Props) => { }); } }, - [startNavigationTTITimer, isAccountListUIEnabled, navigation], + [startNavigationTTITimer, showAssets, isAccountListUIEnabled, navigation], ); + const showAddAccountButton = + isAccountListUIEnabled && showAccounts && distribution.list.length >= maxItemsToDysplay; + return ( <> { accountsLength={distribution.list && distribution.list.length} discreet={discreetMode} /> - + - - {distribution.list.length < maxAssetsToDisplay ? ( + {isAccountListUIEnabled ? ( + + ) : ( + + )} + {showAddAccountButton ? : null} + {distribution.list.length < maxItemsToDysplay ? ( ) : ( - )} diff --git a/apps/ledger-live-mobile/src/screens/Portfolio/ReadOnly/index.tsx b/apps/ledger-live-mobile/src/screens/Portfolio/ReadOnly/index.tsx index 2f514a0a4b5a..748b21ce9f92 100644 --- a/apps/ledger-live-mobile/src/screens/Portfolio/ReadOnly/index.tsx +++ b/apps/ledger-live-mobile/src/screens/Portfolio/ReadOnly/index.tsx @@ -118,7 +118,7 @@ function ReadOnlyPortfolio({ navigation }: NavigationProps) { , ...(!hasOrderedNano diff --git a/apps/ledger-live-mobile/src/screens/Portfolio/TabSection.tsx b/apps/ledger-live-mobile/src/screens/Portfolio/TabSection.tsx new file mode 100644 index 000000000000..21122834f259 --- /dev/null +++ b/apps/ledger-live-mobile/src/screens/Portfolio/TabSection.tsx @@ -0,0 +1,66 @@ +import React, { memo } from "react"; +import { Box, Flex } from "@ledgerhq/native-ui"; +import Animated from "react-native-reanimated"; +import { TabSelector } from "@ledgerhq/native-ui"; +import AssetsListView from "LLM/features/Assets/components/AssetsListView"; +import AccountsListView from "LLM/features/Accounts/components/AccountsListView"; +import { ScreenName } from "~/const"; +import { LayoutChangeEvent } from "react-native"; + +export const TAB_OPTIONS = { + Assets: "Assets", + Accounts: "Accounts", +} as const; + +type BaseAnimationStyle = { + transform: { + translateX: number; + }[]; + opacity: number; +}; + +type TabSectionProps = { + t: (key: string) => string; + handleToggle: (value: string) => void; + handleLayout: (event: LayoutChangeEvent) => void; + assetsAnimatedStyle: BaseAnimationStyle; + accountsAnimatedStyle: BaseAnimationStyle; + maxItemsToDysplay: number; +}; + +const TabSection: React.FC = ({ + t, + handleToggle, + handleLayout, + assetsAnimatedStyle, + accountsAnimatedStyle, + maxItemsToDysplay, +}) => ( + <> + + + + + + + + + + + + +); + +export default memo(TabSection); diff --git a/apps/ledger-live-mobile/src/screens/Portfolio/__integrations__/mockedAccount.ts b/apps/ledger-live-mobile/src/screens/Portfolio/__integrations__/mockedAccount.ts new file mode 100644 index 000000000000..08eeae598b1c --- /dev/null +++ b/apps/ledger-live-mobile/src/screens/Portfolio/__integrations__/mockedAccount.ts @@ -0,0 +1,698 @@ +import BigNumber from "bignumber.js"; +import { AccountsState } from "~/reducers/types"; + +export const MockedAccounts: AccountsState = { + active: [ + { + type: "Account", + id: "mock:1:cronos:0.7681622993167663:", + used: true, + seedIdentifier: "mock", + derivationMode: "", + index: 1, + freshAddress: "0x71DE878364D9B9DC510E5BD13E0AC3372D00D5AF", + freshAddressPath: "44'/60'/0'/0/0", + blockHeight: 157668, + creationDate: new Date("2024-11-13T12:20:06.444Z"), + balance: BigNumber("83315786").multipliedBy(10), + spendableBalance: BigNumber("83315786").multipliedBy(10), + operations: [], + operationsCount: 117, + pendingOperations: [], + currency: { + type: "CryptoCurrency", + id: "cronos", + coinType: 60, + name: "Cronos", + managerAppName: "Ethereum", + ticker: "CRO", + scheme: "cro", + color: "#002D74", + family: "evm", + ethereumLikeInfo: { chainId: 25 }, + units: [{ name: "CRO", code: "CRO", magnitude: 18 }], + explorerViews: [ + { + tx: "https://cronoscan.com/tx/$hash", + address: "https://cronoscan.com/address/$address", + token: "https://cronoscan.com/token/$contractAddress?a=$address", + }, + ], + }, + lastSyncDate: new Date("2024-12-11T09:54:35.938Z"), + swapHistory: [], + balanceHistoryCache: { + HOUR: { + balances: [ + 27291355, 27291355, 27291355, 27291355, 27291355, 27291355, 27291355, 27291355, + 28254686, 34522086, 34522086, 34522086, 34522086, 34522086, 34522086, 34522086, + 34522086, 34522086, 34522086, 34522086, 34522086, 34522086, 34522086, 34522086, + 34522086, 34522086, 35413283, 35413283, 35413283, 35413283, 35423548, 35423548, + 35423548, 35423548, 35423548, 35423548, 31419934, 31419934, 27592754, 27592754, + 27592754, 27592754, 27592754, 27592754, 27592754, 27929605, 27929605, 27929605, + 29280430, 29280430, 29280430, 29280430, 30745273, 30745273, 30745273, 30745273, + 30745273, 32553674, 32553674, 32553674, 32553674, 32553674, 32319070, 34499362, + 34499362, 34499362, 34499362, 34499362, 34499362, 34499362, 34499362, 34499362, + 34499362, 31726261, 31474137, 31474137, 31474137, 33954460, 33954460, 34432840, + 34432840, 34432840, 34432840, 34475408, 34475408, 42435001, 42435001, 42414060, + 43673041, 43673041, 44256130, 44619945, 44370969, 44370969, 44370969, 44370969, + 44370969, 44370969, 44370969, 44370969, 44370969, 45470901, 45470901, 45470901, + 45470901, 45470901, 45470901, 45470901, 45470901, 45470901, 45470901, 45470901, + 45470901, 45470901, 45470901, 43917362, 43917362, 41970580, 46074386, 46074386, + 46399465, 46399465, 46399465, 46399465, 46399465, 46399465, 46399465, 46399465, + 46996242, 46996242, 45872904, 45872904, 45872904, 45872904, 45872904, 46185938, + 46185938, 46533875, 47081103, 47081103, 47081103, 47081103, 47081103, 47081103, + 47081103, 47081103, 47081103, 47081103, 47081103, 47081103, 47081103, 44585722, + 44585722, 44763386, 44763386, 44763386, 44763386, 44763386, 44763386, 44141697, + 47492549, 48701434, 48701434, 48701434, 48701434, 48701434, 48701434, 48701434, + 48701434, 49543493, 49543493, 49543493, 49543493, 49543493, 49543493, 49543493, + 49543493, 49543493, 49543493, 49543493, 49594821, 49594821, 47035383, 47035383, + 47035383, 47035383, 47035383, 48118480, 46118727, 49257559, 49418387, 50444133, + 50444133, + ], + latestDate: 1733907600000, + }, + DAY: { + balances: [ + 0, 2089543, 7816282, 16835018, 23178383, 19582248, 24381793, 31133457, 35132963, + 24959952, 34522086, 27592754, 32319070, 42435001, 45470901, 45872904, 44763386, + 47035383, + ], + latestDate: 1733871600000, + }, + WEEK: { + balances: [0, 31133457, 45470901], + latestDate: 1733612400000, + }, + }, + xpub: "B3FFC394FA819AD7F12ED1742C331BE0DBB94F723BE97ECAB6B894C283B9D6D9", + }, + { + type: "Account", + id: "mock:1:dash:0.6363282668745929:", + used: true, + seedIdentifier: "mock", + derivationMode: "", + index: 1, + freshAddress: "13jgNfUbhAxPXVChS4xfU8VQhvead", + freshAddressPath: "44'/5'/0'/0/0", + blockHeight: 109869, + creationDate: new Date("2024-11-13T12:20:06.444Z"), + balance: BigNumber("83315786").multipliedBy(10), + spendableBalance: BigNumber("83315786").multipliedBy(10), + operations: [], + operationsCount: 74, + pendingOperations: [], + currency: { + type: "CryptoCurrency", + id: "dash", + coinType: 5, + name: "Dash", + managerAppName: "Dash", + ticker: "DASH", + scheme: "dash", + color: "#0e76aa", + family: "bitcoin", + blockAvgTime: 150, + bitcoinLikeInfo: { + P2PKH: 76, + P2SH: 16, + XPUBVersion: 50221816, + }, + units: [ + { name: "dash", code: "DASH", magnitude: 8 }, + { name: "satoshi", code: "sat", magnitude: 0 }, + ], + explorerViews: [ + { + tx: "https://explorer.dash.org/insight/tx/$hash", + address: "https://explorer.dash.org/insight/address/$address", + }, + ], + explorerId: "dash", + }, + lastSyncDate: new Date("2024-12-11T09:54:36.544Z"), + swapHistory: [], + balanceHistoryCache: { + HOUR: { + balances: [ + 439997026, 439997026, 439997026, 439997026, 439997026, 439997026, 695643000, 695643000, + 695643000, 708158592, 708158592, 708158592, 708158592, 708158592, 698865453, 698865453, + 698865453, 683451138, 683451138, 683451138, 733858028, 733858028, 733858028, 733858028, + 733858028, 733858028, 733858028, 733130378, 733130378, 757202252, 757202252, 757202252, + 757202252, 757202252, 673866939, 673866939, 673866939, 673866939, 673866939, 673866939, + 681250366, 681250366, 681250366, 696709231, 696709231, 696709231, 696709231, 705224229, + 705224229, 709940593, 709940593, 709940593, 709940593, 734152057, 594196608, 594196608, + 594196608, 594196608, 594196608, 594196608, 594196608, 594196608, 594196608, 602978906, + 602978906, 602978906, 602978906, 602978906, 616905249, 616905249, 616905249, 637053749, + 637053749, 678503110, 678503110, 678503110, 678503110, 678503110, 678503110, 678503110, + 678503110, 736795357, 737528947, 737528947, 769548549, 769548549, 769548549, 769548549, + 769548549, 769548549, 769548549, 769548549, 769548549, 769548549, 769548549, 769548549, + 769548549, 769548549, 769548549, 769548549, 769548549, 769548549, 769548549, 848716948, + 848716948, 848716948, 848716948, 848716948, 848716948, 848716948, 973757043, 973757043, + 973757043, 973757043, 973757043, 973757043, 973757043, 973757043, 973757043, 973757043, + 973757043, 973757043, 973757043, 973757043, 1058909998, 1058909998, 1058909998, + 1210635569, 1239310947, 1239310947, 1239310947, 1268075425, 1268075425, 1304463901, + 1304463901, 1231995829, 1231995829, 1270950387, 1270950387, 1270950387, 1270950387, + 1270950387, 1270950387, 1270950387, 1270950387, 1300531616, 1295642992, 1295642992, + 1295642992, 1295642992, 1295642992, 1295642992, 1295642992, 1295642992, 1295642992, + 1295642992, 1295642992, 1295642992, 1352711599, 1352711599, 1352711599, 1352711599, + 1349031766, 1349031766, 1349031766, 1349031766, 1349031766, 1349031766, 1349031766, + 1349031766, 1349031766, 1367811094, 1367811094, 1367811094, 1367811094, 1367811094, + 1367811094, 1367811094, 1367811094, 1385096511, 1385096511, 1385096511, 1403982759, + 1403982759, 1403982759, 1403982759, 1403982759, 1403982759, 1403982759, 1403982759, + 1403982759, 1403982759, 1348027904, + ], + latestDate: 1733907600000, + }, + DAY: { + balances: [ + 0, 123498661, 155262842, 458223934, 516857732, 247086424, 698865453, 673866939, + 594196608, 769548549, 973757043, 1304463901, 1352711599, 1403982759, + ], + latestDate: 1733871600000, + }, + WEEK: { + balances: [0, 458223934, 973757043], + latestDate: 1733612400000, + }, + }, + xpub: "503A8385F5BD71DDF0B0C0115DAE4B0DC0F08EF89F52C3B0CFD498706645889F", + }, + { + type: "Account", + id: "mock:1:dogecoin:0.807651428430218:", + used: true, + seedIdentifier: "mock", + derivationMode: "", + index: 1, + freshAddress: "1HcgFNGTEtMrQ2op4ZgjtifdsvpTJ4", + freshAddressPath: "44'/3'/0'/0/0", + blockHeight: 128411, + creationDate: new Date("2024-11-13T12:20:06.444Z"), + balance: BigNumber("83315786").multipliedBy(10), + spendableBalance: BigNumber("83315786").multipliedBy(10), + operations: [], + operationsCount: 62, + pendingOperations: [], + currency: { + type: "CryptoCurrency", + id: "dogecoin", + coinType: 3, + name: "Dogecoin", + managerAppName: "Dogecoin", + ticker: "DOGE", + scheme: "dogecoin", + color: "#65d196", + family: "bitcoin", + blockAvgTime: 60, + bitcoinLikeInfo: { + P2PKH: 30, + P2SH: 22, + XPUBVersion: 49990397, + }, + symbol: "Ð", + units: [ + { name: "dogecoin", code: "DOGE", magnitude: 8 }, + { name: "satoshi", code: "sat", magnitude: 0 }, + ], + explorerViews: [ + { + tx: "https://dogechain.info/tx/$hash", + address: "https://dogechain.info/address/$address", + }, + ], + keywords: ["doge", "dogecoin"], + explorerId: "doge", + }, + lastSyncDate: new Date("2024-12-11T09:54:35.262Z"), + swapHistory: [], + balanceHistoryCache: { + HOUR: { + balances: [ + 9071428571427, 9071428571427, 8845306122448, 8845306122448, 8845306122448, + 8845306122448, 8845306122448, 8845306122448, 8845306122448, 8845306122448, + 8845306122448, 8845306122448, 8845306122448, 8845306122448, 8845306122448, + 8845306122448, 8845306122448, 6931428571428, 6931428571428, 6931428571428, + 6931428571428, 4454489795918, 4454489795918, 4454489795918, 4454489795918, + 4454489795918, 4454489795918, 6823469387754, 7044897959182, 7044897959182, + 7044897959182, 7044897959182, 7044897959182, 9663265306120, 9663265306120, + 16830612244895, 16830612244895, 16830612244895, 16830612244895, 16830612244895, + 18663061224486, 18663061224486, 19344081632649, 19344081632649, 21240408163261, + 21240408163261, 21240408163261, 24449795918363, 27900612244893, 27900612244893, + 27349795918363, 27349795918363, 27349795918363, 27349795918363, 25480204081629, + 25480204081629, 25480204081629, 25480204081629, 25480204081629, 28224897959180, + 30746326530608, 30606326530608, 30606326530608, 30207346938772, 30207346938772, + 30207346938772, 30207346938772, 30207346938772, 30207346938772, 30207346938772, + 30207346938772, 30207346938772, 22946530612242, 22946530612242, 22946530612242, + 22946530612242, 22946530612242, 22946530612242, 22946530612242, 22946530612242, + 22946530612242, 22946530612242, 22946530612242, 30633673469384, 30633673469384, + 30862857142853, 30862857142853, 30862857142853, 30862857142853, 30862857142853, + 30862857142853, 30875918367342, 30875918367342, 30875918367342, 30875918367342, + 30875918367342, 30875918367342, 30875918367342, 30875918367342, 30875918367342, + 30875918367342, 30875918367342, 29710816326525, 29710816326525, 25899183673464, + 25899183673464, 25899183673464, 23711836734688, 23711836734688, 23711836734688, + 23711836734688, 23711836734688, 23985714285708, 23985714285708, 23985714285708, + 23985714285708, 23985714285708, 23985714285708, 23807346938770, 23807346938770, + 23807346938770, 23807346938770, 25598979591831, 25598979591831, 25598979591831, + 25598979591831, 25598979591831, 25598979591831, 25598979591831, 25598979591831, + 25598979591831, 25473877551015, 25473877551015, 25473877551015, 25473877551015, + 25473877551015, 25503877551015, 25503877551015, 26764897959178, 26764897959178, + 26764897959178, 26905306122443, 26905306122443, 26905306122443, 26905306122443, + 26905306122443, 29148367346932, 29148367346932, 40093673469380, 37618163265299, + 37618163265299, 39562040816319, 39562040816319, 39562040816319, 39562040816319, + 39402653061217, 39402653061217, 39402653061217, 39402653061217, 39402653061217, + 39402653061217, 39402653061217, 39402653061217, 39402653061217, 39402653061217, + 51104489795910, 51104489795910, 51104489795910, 51104489795910, 51104489795910, + 51104489795910, 51104489795910, 51104489795910, 51104489795910, 51418367346930, + 51418367346930, 60991836734685, 60991836734685, 60991836734685, 60991836734685, + 67725714285705, 67725714285705, 67725714285705, 67725714285705, 69243265306113, + 69243265306113, 69243265306113, 69243265306113, 69243265306113, 70634489795908, + 70634489795908, 70634489795908, 70634489795908, + ], + latestDate: 1733907600000, + }, + DAY: { + balances: [ + 0, 3341224489795, 1280408163264, 8845306122448, 16830612244895, 30606326530608, + 30862857142853, 23711836734688, 25473877551015, 39402653061217, 67725714285705, + ], + latestDate: 1733871600000, + }, + WEEK: { + balances: [0, 23711836734688], + latestDate: 1733612400000, + }, + }, + xpub: "6024CBE36F9E9250A3EF639F86A5B641F7CF324AEA4F9B205D0663CE68C84775", + }, + { + type: "Account", + id: "mock:1:energy_web:0.37512881170687795:", + used: true, + seedIdentifier: "mock", + derivationMode: "", + index: 1, + freshAddress: "0xDF900A77A89ED58E513EBB053C0350935DC34D9C", + freshAddressPath: "44'/60'/0'/0/0", + blockHeight: 120593, + creationDate: new Date("2024-11-13T12:20:06.444Z"), + balance: BigNumber("83315786").multipliedBy(10), + spendableBalance: BigNumber("83315786").multipliedBy(10), + operations: [], + operationsCount: 95, + pendingOperations: [], + currency: { + type: "CryptoCurrency", + id: "energy_web", + coinType: 60, + name: "Energy Web", + managerAppName: "Ethereum", + ticker: "EWT", + scheme: "energy_web", + color: "#A566FF", + family: "evm", + units: [ + { name: "EWT", code: "EWT", magnitude: 18 }, + { name: "Gwei", code: "Gwei", magnitude: 9 }, + { name: "Mwei", code: "Mwei", magnitude: 6 }, + { name: "Kwei", code: "Kwei", magnitude: 3 }, + { name: "wei", code: "wei", magnitude: 0 }, + ], + ethereumLikeInfo: { chainId: 246 }, + explorerViews: [ + { + tx: "https://explorer.energyweb.org/tx/$hash", + address: "https://explorer.energyweb.org/address/$address", + token: "https://explorer.energyweb.org/token/$contractAddress?a=$address", + }, + ], + }, + lastSyncDate: new Date("2024-12-11T09:54:36.175Z"), + swapHistory: [], + balanceHistoryCache: { + HOUR: { + balances: [ + 35424513, 35424513, 35424513, 36122031, 36122031, 36122031, 36122031, 36122031, + 36122031, 36122031, 36338157, 36338157, 36338157, 36338157, 36258770, 38126301, + 38126301, 38602217, 38602217, 38602217, 38602217, 38602217, 38602217, 38602217, + 38602217, 38602217, 38602217, 38602217, 38602217, 38602217, 38602217, 38602217, + 38602217, 44134183, 44240672, 45199075, 45199075, 45199075, 45199075, 45199075, + 45199075, 45199075, 45860595, 45689227, 45689227, 45689227, 45796537, 45796537, + 45796537, 45796537, 45796537, 45796537, 45796537, 45796537, 45796537, 45796537, + 45855941, 45855941, 45855941, 45855941, 45640225, 45640225, 46493097, 46493097, + 46493097, 46493097, 46493097, 46493097, 46493097, 46493097, 46493097, 46314201, + 50564737, 48829428, 44601475, 44601475, 44601475, 44601475, 44601475, 44891377, + 44415187, 44415187, 44774759, 44774759, 44774759, 44774759, 43763385, 43763385, + 43763385, 48322850, 48679548, 50486717, 50486717, 50486717, 50486717, 50486717, + 52679327, 52683022, 52683022, 52683022, 52683022, 52683022, 52683022, 52683022, + 52683022, 52683022, 52683022, 52683022, 53819637, 53819637, 53819637, 53819637, + 53819637, 52256791, 52256791, 52256791, 52256791, 52256791, 53383688, 53383688, + 53383688, 53383688, 51530392, 51530392, 51530392, 51530392, 51530392, 51530392, + 51530392, 52003434, 52003434, 52003434, 52003434, 52003434, 52003434, 52003434, + 52003434, 52003434, 52003434, 52003434, 52003434, 52003434, 52003434, 52003434, + 52840018, 52840018, 52840018, 52840018, 52840018, 52840018, 52840018, 50626876, + 50626876, 50626876, 50626876, 50320959, 52382717, 52382717, 52382717, 52382717, + 52382717, 52382717, 52382717, 52382717, 52382717, 52382717, 52382717, 52382717, + 54304315, 53269397, 53269397, 54692767, 54692767, 54692767, 54692767, 55541669, + 55541669, 55146509, 55146509, 55690726, 55690726, 55690726, 56302834, 50327253, + 50327253, 51084722, 51084722, 51084722, 51084722, 51084722, 51084722, 51084722, + 53587219, + ], + latestDate: 1733907600000, + }, + DAY: { + balances: [ + 0, 1097195, 20359569, 21945273, 27191988, 36075629, 36258770, 45199075, 46493097, + 43763385, 53819637, 52003434, 52382717, 56302834, + ], + latestDate: 1733871600000, + }, + WEEK: { + balances: [0, 21945273, 53819637], + latestDate: 1733612400000, + }, + }, + xpub: "602FFC9EBDC4FBE203D2D9CB0FAE59C016D0636B0A96130DCDC60C7B368BE8AC", + }, + { + type: "Account", + id: "mock:1:ethereum_classic:0.3802128410576043:", + used: true, + seedIdentifier: "mock", + derivationMode: "", + index: 1, + freshAddress: "0x79D7D3119D60CC666E0DE76AB3A1BFEE46001EAF", + freshAddressPath: "44'/61'/0'/0/0", + blockHeight: 132056, + creationDate: new Date("2024-11-13T12:20:06.444Z"), + balance: BigNumber("83315786").multipliedBy(10), + spendableBalance: BigNumber("83315786").multipliedBy(10), + operations: [], + operationsCount: 184, + pendingOperations: [], + currency: { + type: "CryptoCurrency", + id: "ethereum_classic", + coinType: 61, + name: "Ethereum Classic", + managerAppName: "Ethereum Classic", + ticker: "ETC", + scheme: "ethereumclassic", + color: "#3ca569", + units: [ + { name: "ETC", code: "ETC", magnitude: 18 }, + { name: "Gwei", code: "Gwei", magnitude: 9 }, + { name: "Mwei", code: "Mwei", magnitude: 6 }, + { name: "Kwei", code: "Kwei", magnitude: 3 }, + { name: "wei", code: "wei", magnitude: 0 }, + ], + family: "evm", + blockAvgTime: 15, + ethereumLikeInfo: { chainId: 61 }, + explorerViews: [ + { + tx: "https://blockscout.com/etc/mainnet/tx/$hash/internal-transactions", + address: "https://blockscout.com/etc/mainnet/address/$address/transactions", + }, + ], + keywords: ["etc", "ethereum classic"], + explorerId: "etc", + }, + lastSyncDate: new Date("2024-12-11T09:54:36.743Z"), + swapHistory: [], + balanceHistoryCache: { + HOUR: { + balances: [ + 543975903614457800000, 548857104395234600000, 547624688698929800000, + 547624688698929800000, 548731910883758440000, 548731910883758440000, + 550494043211953900000, 577989499899037500000, 577989499899037500000, + 579767113145318700000, 579767113145318700000, 579767113145318700000, + 579767113145318700000, 579767113145318700000, 612192232617621300000, + 612192232617621300000, 612192232617621300000, 612192232617621300000, + 612192232617621300000, 612192232617621300000, 612192232617621300000, + 612192232617621300000, 612192232617621300000, 612192232617621300000, + 612192232617621300000, 625611496264387100000, 625611496264387100000, + 612320791546072500000, 612320791546072500000, 612320791546072500000, + 612320791546072500000, 612320791546072500000, 612320791546072500000, + 612320791546072500000, 612320791546072500000, 612320791546072500000, + 612320791546072500000, 612320791546072500000, 612320791546072500000, + 612320791546072500000, 623154068789123000000, 623154068789123000000, + 623154068789123000000, 623154068789123000000, 623154068789123000000, + 623154068789123000000, 623154068789123000000, 623154068789123000000, + 623154068789123000000, 623168203540418700000, 623168203540418700000, + 623168203540418700000, 623168203540418700000, 623168203540418700000, + 623168203540418700000, 623168203540418700000, 623168203540418700000, + 623168203540418700000, 623168203540418700000, 623168203540418700000, + 623168203540418700000, 623168203540418700000, 620924143501379800000, + 620924143501379800000, 620924143501379800000, 620924143501379800000, + 620924143501379800000, 620924143501379800000, 620924143501379800000, + 620924143501379800000, 620924143501379800000, 620924143501379800000, + 620924143501379800000, 620924143501379800000, 620924143501379800000, + 620924143501379800000, 620924143501379800000, 598105270242983000000, + 598105270242983000000, 598105270242983000000, 599209800094231700000, + 599209800094231700000, 599209800094231700000, 576554486100827900000, + 576554486100827900000, 576554486100827900000, 576554486100827900000, + 576554486100827900000, 608722487716228000000, 610216059769805500000, + 610216059769805500000, 610216059769805500000, 610216059769805500000, + 610216059769805500000, 614488120078077700000, 614488120078077700000, + 614488120078077700000, 616833815709766400000, 622637140741738000000, + 622637140741738000000, 622637140741738000000, 622637140741738000000, + 622637140741738000000, 618973547822575300000, 620446254290906600000, + 620446254290906600000, 620446254290906600000, 620446254290906600000, + 620446254290906600000, 620446254290906600000, 620446254290906600000, + 611590496062462100000, 611590496062462100000, 611590496062462100000, + 611590496062462100000, 611590496062462100000, 611590496062462100000, + 611590496062462100000, 611590496062462100000, 611590496062462100000, + 611590496062462100000, 611590496062462100000, 611590496062462100000, + 611590496062462100000, 615915056875546900000, 615915056875546900000, + 615915056875546900000, 615915056875546900000, 618478831527226200000, + 618714410715487700000, 618714410715487700000, 618714410715487700000, + 618714410715487700000, 618714410715487700000, 618714410715487700000, + 618714410715487700000, 618714410715487700000, 618714410715487700000, + 619450763949653400000, 618780372888200800000, 618780372888200800000, + 618780372888200800000, 618780372888200800000, 618780372888200800000, + 618780372888200800000, 618780372888200800000, 618780372888200800000, + 618780372888200800000, 618780372888200800000, 618780372888200800000, + 618780372888200800000, 618780372888200800000, 618780372888200800000, + 618780372888200800000, 618780372888200800000, 618780372888200800000, + 618780372888200800000, 621366359291916200000, 633420609813555800000, + 633420609813555800000, 633420609813555800000, 645895537457090900000, + 645895537457090900000, 645895537457090900000, 645895537457090900000, + 640096250925489600000, 660594332637813800000, 660594332637813800000, + 660594332637813800000, 660594332637813800000, 660594332637813800000, + 660594332637813800000, 660594332637813800000, 664733122433869500000, + 665034663794844100000, 665034663794844100000, 656816315541495600000, + 656816315541495600000, 656816315541495600000, 656816315541495600000, + 656816315541495600000, 656816315541495600000, 656816315541495600000, + 656816315541495600000, 656816315541495600000, 656816315541495600000, + 656816315541495600000, 656816315541495600000, 656816315541495600000, + 658904893316281900000, 660133270512216400000, 660133270512216400000, + 660133270512216400000, + ], + latestDate: 1733907600000, + }, + DAY: { + balances: [ + 0, 23390321060779430000, 68410177020932890000, 26644006192367230000, + 60639429225281000000, 121539341724439640000, 165781786363330400000, + 203335128222386750000, 265976980547889860000, 289407686612371240000, + 329510668371811240000, 413592919162684240000, 464141482129635800000, + 477792959547687960000, 458445177357474600000, 465155818805950000000, + 562590024904085600000, 552687621996365350000, 612192232617621300000, + 612320791546072500000, 620924143501379800000, 576554486100827900000, + 620446254290906600000, 618714410715487700000, 633420609813555800000, + 656816315541495600000, + ], + latestDate: 1733871600000, + }, + WEEK: { + balances: [ + 0, 23390321060779430000, 265976980547889860000, 465155818805950000000, + 620446254290906600000, + ], + latestDate: 1733612400000, + }, + }, + xpub: "5CBDD12CE86122F06B9931625A5D18666643D2230AB62FE0D45E8BC08259559D", + }, + { + type: "Account", + id: "mock:1:linea:0.03503010215576742:", + used: true, + seedIdentifier: "mock", + derivationMode: "", + index: 1, + freshAddress: "0xB0E1D3F5F8AC8E17109C286AC15C96FAED86C094", + freshAddressPath: "44'/60'/0'/0/0", + blockHeight: 175413, + creationDate: new Date("2024-11-13T12:20:06.444Z"), + balance: BigNumber("83315786").multipliedBy(10), + spendableBalance: BigNumber("83315786").multipliedBy(10), + operations: [], + operationsCount: 180, + pendingOperations: [], + currency: { + type: "CryptoCurrency", + id: "linea", + coinType: 60, + name: "Linea", + managerAppName: "Ethereum", + ticker: "ETH", + scheme: "linea", + color: "#000000", + family: "evm", + units: [ + { name: "ETH", code: "ETH", magnitude: 18 }, + { name: "Gwei", code: "Gwei", magnitude: 9 }, + { name: "Mwei", code: "Mwei", magnitude: 6 }, + { name: "Kwei", code: "Kwei", magnitude: 3 }, + { name: "wei", code: "wei", magnitude: 0 }, + ], + disableCountervalue: false, + ethereumLikeInfo: { chainId: 59144 }, + explorerViews: [ + { + tx: "https://lineascan.build/tx/$hash", + address: "https://lineascan.build/address/$address", + token: "https://lineascan.build/token/$address", + }, + ], + }, + lastSyncDate: new Date("2024-12-11T09:54:36.635Z"), + swapHistory: [], + balanceHistoryCache: { + HOUR: { + balances: [ + 51579808, 51579808, 51579808, 51579808, 51579808, 51579808, 53942965, 53942965, + 53942965, 50091147, 50091147, 50091147, 50091147, 50091147, 54178802, 49701052, + 45209887, 45493767, 45493767, 45493767, 50819599, 50819599, 50819599, 51466473, + 51466473, 51466473, 51466473, 51466473, 55992540, 55992540, 55992540, 55992540, + 55992540, 55992540, 55992540, 55992540, 55992540, 55992540, 55992540, 55992540, + 55992540, 55992540, 55992540, 55992540, 55992540, 55992540, 55992540, 55740005, + 55740005, 61429242, 61429242, 61429242, 61429242, 61429242, 61429242, 61429242, + 61429242, 61429242, 61429242, 61608275, 61608275, 55130360, 56421097, 56421097, + 56421097, 53576410, 53576410, 53576410, 53576410, 53576410, 53576410, 53576410, + 53576410, 53576410, 53576410, 53576410, 50422247, 51211198, 51211198, 53889991, + 47720868, 47720868, 47720868, 47720868, 47720868, 47720868, 47720868, 47720868, + 48879794, 49262498, 49262498, 49262498, 49262498, 49262498, 49262498, 49262498, + 49262498, 49262498, 49262498, 49262498, 49262498, 48300126, 48248798, 48248798, + 48248798, 48248798, 48248798, 48248798, 48248798, 48248798, 48248798, 48248798, + 48248798, 48248798, 48248798, 48248798, 48248798, 48919625, 48919625, 48919625, + 48919625, 48919625, 48919625, 48919625, 48919625, 48919625, 48919625, 52113345, + 52113345, 52113345, 52113345, 52113345, 52113345, 52113345, 53916955, 54248331, + 54248331, 54248331, 54248331, 54248331, 54248331, 54248331, 54248331, 54248331, + 54248331, 54248331, 54248331, 54248331, 58075238, 58075238, 65594777, 65594777, + 65682377, 65682377, 65682377, 65849091, 65849091, 65849091, 70860383, 70860383, + 70860383, 70860383, 70860383, 70860383, 70860383, 70860383, 71002049, 71002049, + 63886430, 63886430, 63886430, 63886430, 63008921, 63008921, 63008921, 63008921, + 63008921, 63008921, 63008921, 63008921, 63008921, 63008921, 63008921, 63008921, + 63008921, 62736128, 62736128, 62736128, 62736128, 62736128, 64237927, 64237927, + 64237927, + ], + latestDate: 1733907600000, + }, + DAY: { + balances: [ + 0, 3170451, 13226569, 13214111, 22720671, 21945408, 26887578, 30858477, 40549547, + 46441224, 45387692, 46033608, 40708322, 44409712, 31692595, 40116881, 41839598, + 45287084, 46027170, 41091981, 51090201, 37865960, 53026583, 54178802, 55992540, + 56421097, 47720868, 48248798, 53916955, 70860383, 63008921, + ], + latestDate: 1733871600000, + }, + WEEK: { + balances: [0, 26887578, 44409712, 51090201, 48248798], + latestDate: 1733612400000, + }, + }, + xpub: "F57C96A272FAB0E47F8FD7E5A3F29C58F088A139927828A7AF8731E315099918", + }, + { + type: "Account", + id: "mock:1:near:0.6709116415216639:", + used: true, + seedIdentifier: "mock", + derivationMode: "", + index: 1, + freshAddress: "1aYQttd1RvLjJ3NXTNXgNMwhvfmRYMRt6z", + freshAddressPath: "44'/397'/0'/0/0", + blockHeight: 112715, + creationDate: new Date("2024-11-13T12:20:06.444Z"), + balance: BigNumber("83315786").multipliedBy(10), + spendableBalance: BigNumber("83315786").multipliedBy(10), + operations: [], + operationsCount: 155, + pendingOperations: [], + currency: { + type: "CryptoCurrency", + id: "near", + coinType: 397, + name: "NEAR", + managerAppName: "NEAR", + ticker: "NEAR", + scheme: "near", + color: "#000", + family: "near", + units: [ + { name: "NEAR", code: "NEAR", magnitude: 24 }, + { name: "yoctoNEAR", code: "yoctoNEAR", magnitude: 0 }, + ], + explorerViews: [ + { + address: "https://nearblocks.io/address/$address", + tx: "https://nearblocks.io/txns/$hash", + }, + ], + keywords: ["near"], + }, + lastSyncDate: new Date("2024-12-11T09:54:36.635Z"), + swapHistory: [], + balanceHistoryCache: { + HOUR: { + balances: [ + 66108332, 66108332, 66125304, 66125304, 66125304, 66125304, 66125304, 66125304, + 66125304, 66125304, 66125304, 66125304, 68454654, 68454654, 63429127, 63429127, + 58645596, 58399905, 58399905, 58399905, 58399905, 58467384, 58467384, 58467384, + 58467384, 58467384, 58467384, 62923508, 63569834, 63569834, 63569834, 63569834, + 63569834, 63569834, 63569834, 64161547, 65865375, 65865375, 66980774, 66980774, + 66980774, 66980774, 66980774, 66980774, 66980774, 66980774, 66980774, 66980774, + 66029215, 66029215, 65438597, 65438597, 65438597, 65438597, 65893708, 65681141, + 65681141, 72627170, 72587340, 72794432, 72794432, 72794432, 72794432, 72794432, + 72794432, 72794432, 72794432, 72794432, 72794432, 72794432, 72794432, 72794432, + 72794432, 72794432, 72794432, 72338226, 72338226, 72338226, 72338226, 72338226, + 72338226, 72338226, 72338226, 72338226, 72338226, 72338226, 72338226, 72338226, + 72338226, 72338226, 72338226, 72338226, 72338226, 72794843, 72794843, 69533507, + 69533507, 69328605, 69328605, 69328605, 69328605, 69328605, 69328605, 69328605, + 65070401, 65070401, 65070401, 65070401, 64801577, 66193602, 66193602, 66193602, + 66193602, 66193602, 66193602, 66193602, 66193602, 66193602, 66193602, 66193602, + 63635122, 63635122, 63635122, 63635122, 65806106, 66781071, 66781071, 66781071, + 66781071, 66781071, 66781071, 66781071, 74492100, 74492100, 74492100, 74492100, + 74492100, 74492100, 74492100, 74492100, 74492100, 74492100, 74492100, 74492100, + 74492100, 74492100, 74492100, 74492100, 74492100, 75952289, 76125710, 76125710, + 76125710, 76125710, 76125710, 76125710, 76125710, 76125710, 77349104, 77349104, + 77349104, 77349104, 77349104, 77349104, 77349104, 82814548, 82814548, 82814548, + 82821939, 82821939, 82821939, 82821939, 82821939, 82821939, 82643043, 82858895, + 82858895, 82858895, 82858895, 83315786, 83315786, 83315786, 83315786, 83315786, + 83315786, 83315786, 83315786, 83315786, 83315786, 83315786, 83315786, 83315786, + 83315786, + ], + latestDate: 1733907600000, + }, + DAY: { + balances: [ + 0, 2294036, 23635414, 27645186, 34078342, 34660063, 44882483, 51299076, 53804171, + 49607425, 39470410, 38031574, 38838593, 45738359, 60004361, 63429127, 66980774, + 72794432, 72338226, 66193602, 74492100, 77349104, 83315786, + ], + latestDate: 1733871600000, + }, + WEEK: { + balances: [0, 34660063, 38838593, 66193602], + latestDate: 1733612400000, + }, + }, + xpub: "C2AC267866DDDC2D8D45040716FD3BD972DD182AA2F8D82E882FA42B10723165", + }, + ], +}; diff --git a/apps/ledger-live-mobile/src/screens/Portfolio/__integrations__/portfolioAssets.integration.test.tsx b/apps/ledger-live-mobile/src/screens/Portfolio/__integrations__/portfolioAssets.integration.test.tsx new file mode 100644 index 000000000000..bb61ed20d050 --- /dev/null +++ b/apps/ledger-live-mobile/src/screens/Portfolio/__integrations__/portfolioAssets.integration.test.tsx @@ -0,0 +1,164 @@ +import React from "react"; +import { fireEvent, render, screen } from "@tests/test-renderer"; +import { State } from "~/reducers/types"; +import PortfolioAssets from "../PortfolioAssets"; +import TestNavigator, { INITIAL_STATE, SlicedMockedAccounts } from "./shared"; +import { track } from "~/analytics"; + +const mockLayoutEvent = (width: number) => ({ + nativeEvent: { + layout: { + width, + }, + }, +}); + +describe("portfolioAssets", () => { + it("should render empty portfolio", async () => { + render( + + null} /> + , + { + overrideInitialState: (state: State) => ({ + ...INITIAL_STATE.overrideInitialState(state), + accounts: { ...state.accounts }, + }), + }, + ); + + expect(await screen.findByText(/add account/i)).toBeVisible(); + expect(screen.queryByText(/bitcoin/i)).toBeNull(); + }); + + it("should render portfolio with assets and accounts list", async () => { + const { user } = render( + + null} /> + , + { ...INITIAL_STATE }, + ); + + fireEvent(screen.getByTestId("portfolio-assets-layout"), "layout", mockLayoutEvent(722)); + + expect(screen.getByTestId("AssetsList")).toBeVisible(); + expect(screen.getByText(/accounts/i)).toBeVisible(); + expect(screen.getByText(/see all assets/i)).toBeVisible(); + expect(screen.getByTestId("assetItem-Cronos")).toBeVisible(); + expect(screen.getByText(/cronos 2/i)).not.toBeVisible(); + + await user.press(screen.getByText(/accounts/i)); + + expect(screen.getByText(/see all accounts/i)).toBeVisible(); + expect(screen.getByText(/add new or existing account/i)).toBeVisible(); + expect(screen.getByText(/cronos 2/i)).toBeDefined(); + //expect(screen.getByText("Cronos 2")).toBeVisible(); + // FIXME this is not visible in the test after the animation. It seems that the useSharedValue are not updated in the test environment so cronos 2 is always not visible even if it should be visible. + }); + + it("should hide see all button and display add account button because there is less than 5 assets", async () => { + const { user } = render( + + null} /> + , + { + overrideInitialState: (state: State) => ({ + ...INITIAL_STATE.overrideInitialState(state), + accounts: SlicedMockedAccounts, + }), + }, + ); + + expect(await screen.getByTestId("AssetsList")).toBeVisible(); + expect(screen.queryByText(/see all assets/i)).toBeNull(); + expect(screen.getByText(/add account/i)).toBeVisible(); + + await user.press(screen.getByText(/accounts/i)); + + expect(track).toHaveBeenCalledWith("button_clicked", { + button: "Accounts", + page: "Wallet", + }); + + expect(screen.queryByText(/see all accounts/i)).toBeNull(); + + await user.press(screen.getByText(/assets/i)); + + expect(track).toHaveBeenCalledWith("button_clicked", { + button: "Assets", + page: "Wallet", + }); + }); + + it("should render assets list screen", async () => { + const { user } = render( + + null} /> + , + { ...INITIAL_STATE }, + ); + + expect(await screen.getByTestId("AssetsList")).toBeVisible(); + expect(screen.getByText(/see all assets/i)).toBeVisible(); + await user.press(screen.getByText(/see all assets/i)); + + expect(track).toHaveBeenCalledWith("button_clicked", { + button: "See all assets", + page: "Wallet", + }); + + const lineaAsset = screen.getByText(/linea/i); + const ethClassicAsset = screen.getByText(/ethereum classic/i); + const energyWebAsset = screen.getByText(/energy web/i); + const dogecoinAsset = screen.getByText(/dogecoin/i); + const dashAsset = screen.getAllByText(/dash/i)[0]; + const cronosAsset = screen.getByText(/cronos/i); + + [lineaAsset, ethClassicAsset, energyWebAsset, dogecoinAsset, dashAsset, cronosAsset].forEach( + asset => { + expect(asset).toBeVisible(); + }, + ); + }); + + it("should render accounts list screen", async () => { + const { user } = render( + + null} /> + , + { ...INITIAL_STATE }, + ); + + expect(await screen.getByTestId("AssetsList")).toBeVisible(); + + await user.press(screen.getByText(/accounts/i)); + + expect(screen.getByText(/see all accounts/i)).toBeVisible(); + await user.press(screen.getByText(/see all accounts/i)); + + expect(track).toHaveBeenCalledWith("button_clicked", { + button: "See all accounts", + page: "Wallet", + }); + + expect(screen.getByText(/add new or existing account/i)).toBeVisible(); + + const lineaAccount = screen.getByText(/linea 2/i); + const ethClassicAccount = screen.getByText(/ethereum classic 2/i); + const energyWebAccount = screen.getByText(/energy web 2/i); + const dogecoinAccount = screen.getByText(/dogecoin 2/i); + const dashAccount = screen.getByText(/dash 2/i); + const cronosAccount = screen.getByText(/cronos 2/i); + + [ + lineaAccount, + ethClassicAccount, + energyWebAccount, + dogecoinAccount, + dashAccount, + cronosAccount, + ].forEach(account => { + expect(account).toBeVisible(); + }); + }); +}); diff --git a/apps/ledger-live-mobile/src/screens/Portfolio/__integrations__/shared.tsx b/apps/ledger-live-mobile/src/screens/Portfolio/__integrations__/shared.tsx new file mode 100644 index 000000000000..d48f3dda5223 --- /dev/null +++ b/apps/ledger-live-mobile/src/screens/Portfolio/__integrations__/shared.tsx @@ -0,0 +1,52 @@ +import React from "react"; +import { createStackNavigator } from "@react-navigation/stack"; +import { QueryClientProvider, QueryClient } from "@tanstack/react-query"; +import { NavigatorName, ScreenName } from "~/const"; +import { MockedAccounts } from "./mockedAccount"; +import { State } from "~/reducers/types"; +import AccountsNavigator from "LLM/features/Accounts/Navigator"; +import { BaseNavigatorStackParamList } from "~/components/RootNavigator/types/BaseNavigator"; +import AssetsNavigator from "LLM/features/Assets/Navigator"; + +const Stack = createStackNavigator(); + +const TestNavigator = ({ children }: { children: React.ReactNode }) => ( + + + {() => children} + + + + +); + +export const SlicedMockedAccounts = { + ...MockedAccounts, + active: MockedAccounts.active.slice(0, 3), +}; + +export const INITIAL_STATE = { + overrideInitialState: (state: State) => ({ + ...state, + accounts: MockedAccounts, + settings: { + ...state.settings, + readOnlyModeEnabled: false, + overriddenFeatureFlags: { + llmAccountListUI: { + enabled: true, + }, + }, + }, + }), +}; + +export default TestNavigator; diff --git a/apps/ledger-live-mobile/src/screens/Portfolio/index.tsx b/apps/ledger-live-mobile/src/screens/Portfolio/index.tsx index ffdacc790d7d..356d2a1421ab 100644 --- a/apps/ledger-live-mobile/src/screens/Portfolio/index.tsx +++ b/apps/ledger-live-mobile/src/screens/Portfolio/index.tsx @@ -106,7 +106,7 @@ function PortfolioScreen({ navigation }: NavigationProps) { ) : null} , showAssets ? ( - + { + const [selectedTab, setSelectedTab] = useState(initialTab); + const [containerWidth, setContainerWidth] = useState(0); + + const assetsTranslateX = useSharedValue(0); + const assetsOpacity = useSharedValue(1); + + const accountsTranslateX = useSharedValue(containerWidth); + const accountsOpacity = useSharedValue(0); + + const handleToggle = (value: string) => { + setSelectedTab(value as TabListType); + track("button_clicked", { + button: value, + page: "Wallet", + }); + }; + + const handleLayout = (event: LayoutChangeEvent) => { + // Get the width of the container and set it to the state so we can use the width in the animation + const { width } = event.nativeEvent.layout; + setContainerWidth(width); + }; + + const assetsAnimatedStyle = useAnimatedStyle( + () => ({ + transform: [{ translateX: assetsTranslateX.value }], + opacity: assetsOpacity.value, + }), + [assetsTranslateX, assetsOpacity], + ); + + const accountsAnimatedStyle = useAnimatedStyle( + () => ({ + transform: [{ translateX: accountsTranslateX.value }], + opacity: accountsOpacity.value, + }), + [accountsTranslateX, accountsOpacity], + ); + + useEffect(() => { + if (selectedTab === TAB_OPTIONS.Assets) { + // Assets tab is selected so here is the default position + assetsTranslateX.value = withTiming(0, { duration: 250 }); + assetsOpacity.value = withTiming(1, { duration: 250 }); + // Accounts tab is not selected so here is the end position + accountsTranslateX.value = withTiming(containerWidth / 3, { duration: 250 }); + accountsOpacity.value = withTiming(0, { duration: 250 }); + } else { + // Assets tab is not selected so here is the end position + assetsTranslateX.value = withTiming(-containerWidth / 3, { duration: 250 }); + assetsOpacity.value = withTiming(0, { duration: 250 }); + // Accounts tab is selected so here is the default position + accountsTranslateX.value = withTiming(-containerWidth / 2, { duration: 250 }); + accountsOpacity.value = withTiming(1, { duration: 250 }); + } + }, [ + selectedTab, + containerWidth, + assetsTranslateX, + accountsTranslateX, + assetsOpacity, + accountsOpacity, + ]); + + return { + handleToggle, + handleLayout, + selectedTab, + assetsAnimatedStyle, + accountsAnimatedStyle, + }; +}; + +export default useListsAnimation; diff --git a/libs/ui/packages/native/src/components/Form/DrawerTabSelector/index.tsx b/libs/ui/packages/native/src/components/Form/DrawerTabSelector/index.tsx new file mode 100644 index 000000000000..84b2b8754d1c --- /dev/null +++ b/libs/ui/packages/native/src/components/Form/DrawerTabSelector/index.tsx @@ -0,0 +1,129 @@ +import React, { useEffect } from "react"; +import Text from "../../Text"; +import Flex from "../../Layout/Flex"; +import styled, { useTheme } from "styled-components/native"; +import { TouchableOpacity } from "react-native"; +import Animated, { useSharedValue, useAnimatedStyle, withSpring } from "react-native-reanimated"; + +const StyledTouchableOpacity = styled(TouchableOpacity)<{ width: number }>` + width: ${(p) => p.width}px; + flex: 1; + height: 100%; +`; + +const StyledFlex = styled(Flex)<{ isSelected: boolean }>` + height: 100%; + justify-content: center; + align-items: center; +`; + +const StyledText = styled(Text)<{ isSelected: boolean }>` + line-height: 14.52px; + overflow: visible; + text-align: center; + font-size: 12px; + color: ${(p) => + p.isSelected ? p.theme.colors.constant.black : p.theme.colors.opacityDefault.c50}; +`; + +interface OptionButtonProps { + option: T; + selectedOption: T; + handleSelectOption: (option: T) => void; + label: string; + width: number; +} + +const OptionButton = ({ + option, + selectedOption, + handleSelectOption, + label, + width, +}: OptionButtonProps) => { + const isSelected = selectedOption === option; + + return ( + handleSelectOption(option)}> + + + {label} + + + + ); +}; + +interface TabSelectorProps { + options: T[]; + selectedOption: T; + handleSelectOption: (option: T) => void; + labels: { [key in T]: string }; +} + +export default function DrawerTabSelector({ + options, + selectedOption, + handleSelectOption, + labels, +}: TabSelectorProps): JSX.Element { + const { colors } = useTheme(); + + const longuestLabel = + labels[options[0]].length > labels[options[1]].length ? options[0] : options[1]; + + const widthFactor = 8; + const margin = 20; + const width = labels[longuestLabel].length * widthFactor + margin; + const semiWidth = width / 2; + const translateX = useSharedValue(-semiWidth); + + useEffect(() => { + translateX.value = withSpring(selectedOption === options[0] ? -semiWidth : semiWidth, { + damping: 30, + stiffness: 80, + }); + }, [selectedOption, translateX, options]); + + const animatedStyle = useAnimatedStyle(() => { + return { + transform: [{ translateX: translateX.value }], + }; + }); + + return ( + + + {options.map((option) => ( + + ))} + + ); +} diff --git a/libs/ui/packages/native/src/components/Form/TabSelector/index.tsx b/libs/ui/packages/native/src/components/Form/TabSelector/index.tsx index 8e7604400271..f55fd6c3dcfc 100644 --- a/libs/ui/packages/native/src/components/Form/TabSelector/index.tsx +++ b/libs/ui/packages/native/src/components/Form/TabSelector/index.tsx @@ -1,129 +1,89 @@ -import React, { useEffect } from "react"; -import Text from "../../Text"; +import React, { useState } from "react"; +import { Pressable, LayoutChangeEvent } from "react-native"; +import Animated, { useSharedValue, useAnimatedStyle, withTiming } from "react-native-reanimated"; +import styled from "styled-components/native"; import Flex from "../../Layout/Flex"; -import styled, { useTheme } from "styled-components/native"; -import { TouchableOpacity } from "react-native"; -import Animated, { useSharedValue, useAnimatedStyle, withSpring } from "react-native-reanimated"; +import Text from "../../Text"; +import Box from "../../Layout/Box"; -const StyledTouchableOpacity = styled(TouchableOpacity)<{ width: number }>` - width: ${(p) => p.width}px; - flex: 1; +const Container = styled(Flex)` height: 100%; + justify-content: space-between; + flex-direction: row; + position: relative; `; -const StyledFlex = styled(Flex)<{ isSelected: boolean }>` +const AnimatedBackground = styled(Animated.View)` + position: absolute; height: 100%; - justify-content: center; - align-items: center; + border-radius: 8px; + background-color: ${({ theme }) => theme.colors.opacityDefault.c05}; `; -const StyledText = styled(Text)<{ isSelected: boolean }>` - line-height: 14.52px; - overflow: visible; - text-align: center; - font-size: 12px; - color: ${(p) => - p.isSelected ? p.theme.colors.constant.black : p.theme.colors.opacityDefault.c50}; +const Tab = styled(Flex)` + flex: 1; + padding: 4px; + border-radius: 8px; + align-items: center; + justify-content: center; `; -interface OptionButtonProps { - option: T; - selectedOption: T; - handleSelectOption: (option: T) => void; - label: string; - width: number; -} - -const OptionButton = ({ - option, - selectedOption, - handleSelectOption, - label, - width, -}: OptionButtonProps) => { - const isSelected = selectedOption === option; - - return ( - handleSelectOption(option)}> - - - {label} - - - - ); +type TabSelectorProps = { + labels: string[]; + onToggle: (value: string) => void; }; -interface TabSelectorProps { - options: T[]; - selectedOption: T; - handleSelectOption: (option: T) => void; - labels: { [key in T]: string }; -} - -export default function TabSelector({ - options, - selectedOption, - handleSelectOption, - labels, -}: TabSelectorProps): JSX.Element { - const { colors } = useTheme(); - - const longuestLabel = - labels[options[0]].length > labels[options[1]].length ? options[0] : options[1]; +export default function TabSelector({ labels, onToggle }: TabSelectorProps): JSX.Element { + const translateX = useSharedValue(0); + const [containerWidth, setContainerWidth] = useState(0); + const [selectedIndex, setSelectedIndex] = useState(0); - const widthFactor = 8; - const margin = 20; - const width = labels[longuestLabel].length * widthFactor + margin; - const semiWidth = width / 2; - const translateX = useSharedValue(-semiWidth); + const handlePress = (value: string, index: number) => { + setSelectedIndex(index); + translateX.value = (containerWidth / labels.length) * index; + if (selectedIndex !== index) onToggle(value); + }; - useEffect(() => { - translateX.value = withSpring(selectedOption === options[0] ? -semiWidth : semiWidth, { - damping: 30, - stiffness: 80, - }); - }, [selectedOption, translateX, options]); + const handleLayout = (event: LayoutChangeEvent) => { + const { width } = event.nativeEvent.layout; + setContainerWidth(width); + }; const animatedStyle = useAnimatedStyle(() => { return { - transform: [{ translateX: translateX.value }], + transform: [{ translateX: withTiming(translateX.value, { duration: 250 }) }], + width: containerWidth / labels.length, }; }); return ( - - - {options.map((option) => ( - - ))} - + + + {labels.map((label, index) => ( + handlePress(label, index)} + style={({ pressed }: { pressed: boolean }) => [ + { opacity: pressed && selectedIndex !== index ? 0.5 : 1, flex: 1 }, + ]} + > + + + {label} + + + + ))} + + ); } diff --git a/libs/ui/packages/native/src/components/Form/index.ts b/libs/ui/packages/native/src/components/Form/index.ts index 3f14bbd68d27..1f890f18d928 100644 --- a/libs/ui/packages/native/src/components/Form/index.ts +++ b/libs/ui/packages/native/src/components/Form/index.ts @@ -4,4 +4,5 @@ export { default as Slider } from "./Slider"; export { default as Switch } from "./Switch"; export { default as Toggle } from "./Toggle"; export { default as SelectableList } from "./SelectableList"; +export { default as DrawerTabSelector } from "./DrawerTabSelector"; export { default as TabSelector } from "./TabSelector"; diff --git a/libs/ui/packages/native/storybook/stories/Form/DrawerTabSelector.stories.tsx b/libs/ui/packages/native/storybook/stories/Form/DrawerTabSelector.stories.tsx new file mode 100644 index 000000000000..a0f8da4be3d6 --- /dev/null +++ b/libs/ui/packages/native/storybook/stories/Form/DrawerTabSelector.stories.tsx @@ -0,0 +1,41 @@ +import React, { useState } from "react"; +import { View } from "react-native"; +import DrawerTabSelector from "../../../src/components/Form/DrawerTabSelector"; + +export default { + title: "Form/DrawerTabSelector", + component: DrawerTabSelector, + argTypes: { + options: { control: "array" }, + selectedOption: { control: "text" }, + labels: { control: "object" }, + }, +}; + +export const TabSelectorStory = (args: typeof TabSelectorStoryArgs) => { + const [selectedOption, setSelectedOption] = useState(args.selectedOption); + + return ( + + + + ); +}; + +TabSelectorStory.storyName = "TabSelectorStory"; + +const TabSelectorStoryArgs = { + options: ["option1", "option2"], + selectedOption: "option1", + labels: { + option1: "Option 1", + option2: "Option 2", + }, +}; + +TabSelectorStory.args = TabSelectorStoryArgs; diff --git a/libs/ui/packages/native/storybook/stories/Form/TabSelector.stories.tsx b/libs/ui/packages/native/storybook/stories/Form/TabSelector.stories.tsx index 581ec2535adb..9affaeed5ee1 100644 --- a/libs/ui/packages/native/storybook/stories/Form/TabSelector.stories.tsx +++ b/libs/ui/packages/native/storybook/stories/Form/TabSelector.stories.tsx @@ -1,4 +1,4 @@ -import React, { useState } from "react"; +import React from "react"; import { View } from "react-native"; import TabSelector from "../../../src/components/Form/TabSelector"; @@ -6,36 +6,54 @@ export default { title: "Form/TabSelector", component: TabSelector, argTypes: { - options: { control: "array" }, - selectedOption: { control: "text" }, - labels: { control: "object" }, + labels: { control: "array" }, + onToggle: { action: "toggled" }, }, }; -export const TabSelectorStory = (args: typeof TabSelectorStoryArgs) => { - const [selectedOption, setSelectedOption] = useState(args.selectedOption); +type TabSelectorStoryArgs = { + labels: string[]; + labels2: string[]; + labels3: string[]; + labels4: string[]; + onToggle: (value: string) => void; +}; + +export const TabSelectorStory = (args: TabSelectorStoryArgs) => { + const handleToggle = (value: string) => { + args.onToggle(value); + }; return ( - - - + <> + + + + + + + + + + + + + ); }; TabSelectorStory.storyName = "TabSelectorStory"; const TabSelectorStoryArgs = { - options: ["option1", "option2"], - selectedOption: "option1", - labels: { - option1: "Option 1", - option2: "Option 2", - }, + labels: ["First tab"], + labels2: ["First tab", "Second Tab"], + labels3: ["First tab", "Second Tab", "Third Tab"], + labels4: [ + "First tab First tab First tab", + "Second Tab Second Tab Second Tab", + "Third Tab Third Tab Third Tab", + "Fourth Tab Fourth Tab Fourth Tab", + ], }; TabSelectorStory.args = TabSelectorStoryArgs; diff --git a/libs/ui/packages/native/storybook/stories/index.ts b/libs/ui/packages/native/storybook/stories/index.ts index f5edc8322ffa..83b4af30a087 100644 --- a/libs/ui/packages/native/storybook/stories/index.ts +++ b/libs/ui/packages/native/storybook/stories/index.ts @@ -31,7 +31,8 @@ export * as NumberInputStory from "./Form/Input/NumberInput.stories"; export * as LintStory from "./Link/Link.stories"; export * as SwitchStory from "./Form/Switch.stories"; export * as ToggleStory from "./Form/Toggle.stories"; -export * as TabSelector from "./Form/TabSelector.stories"; +export * as DrawerTabSelector from "./Form/DrawerTabSelector.stories"; +export * as TabSelectorStory from "./Form/TabSelector.stories"; export * as ChipTabsStory from "./Tabs/ChipTabs/ChipTabs.stories"; export * as GraphTabsStory from "./Tabs/GraphTabs/GraphTabs.stories"; export * as BadgeStory from "./Tag/Badge.stories";