From 38989dba59e9a612636dd3dc2ac1dbe02d409f9c Mon Sep 17 00:00:00 2001
From: Chloe <44501120+chloezxyy@users.noreply.github.com>
Date: Mon, 9 Oct 2023 13:44:15 +0800
Subject: [PATCH] feature(ui-ux): address book ui for transfer domain (#4042)
* feature(ui-ux): address book UI for transfer domain
* feature(ui-ux): address book bottom sheet in portfolio page
* update comment
* remove eslint-disable-next-line react-hooks/exhaustive-deps
* feat(ui-ux): standardize evm tag UI across screens
* feat(ui-ux): display address label
* fix(ui-ux): remove caret if not from send screen
* chore: update package
* fix: lint for address row
---
mobile-app/app/components/AddressEvmTag.tsx | 27 ++
mobile-app/app/components/EvmTag.tsx | 31 ++
mobile-app/app/components/SummaryTitle.tsx | 96 +++--
mobile-app/app/hooks/useWalletAddress.ts | 4 +-
.../Portfolio/components/AddressRow.tsx | 93 +++--
.../components/AddressSelectionButtonV2.tsx | 31 +-
.../components/BottomSheetAddressDetailV2.tsx | 84 ++--
.../CreateOrEditAddressLabelForm.tsx | 1 +
.../Portfolio/screens/AddressBookScreen.tsx | 364 ++++++++++++------
.../screens/Portfolio/screens/SendScreen.tsx | 4 +-
10 files changed, 483 insertions(+), 252 deletions(-)
create mode 100644 mobile-app/app/components/AddressEvmTag.tsx
create mode 100644 mobile-app/app/components/EvmTag.tsx
diff --git a/mobile-app/app/components/AddressEvmTag.tsx b/mobile-app/app/components/AddressEvmTag.tsx
new file mode 100644
index 0000000000..cc7a0af821
--- /dev/null
+++ b/mobile-app/app/components/AddressEvmTag.tsx
@@ -0,0 +1,27 @@
+import { tailwind } from "@tailwind";
+import { LinearGradient } from "expo-linear-gradient";
+
+export function AddressEvmTag({
+ children,
+ customStyle,
+ testID,
+}: {
+ children: React.ReactElement;
+ customStyle?: string;
+ testID: string;
+}): JSX.Element {
+ return (
+
+ {children}
+
+ );
+}
diff --git a/mobile-app/app/components/EvmTag.tsx b/mobile-app/app/components/EvmTag.tsx
new file mode 100644
index 0000000000..091ccfb555
--- /dev/null
+++ b/mobile-app/app/components/EvmTag.tsx
@@ -0,0 +1,31 @@
+import { tailwind } from "@tailwind";
+import { Text } from "@components/Text";
+import { LinearGradient } from "expo-linear-gradient";
+
+export function EvmTag({
+ text = "EVM",
+ index,
+ testIDSuffix,
+}: {
+ text?: string;
+ index: number;
+ testIDSuffix: string;
+}): JSX.Element {
+ return (
+
+
+ {text}
+
+
+ );
+}
diff --git a/mobile-app/app/components/SummaryTitle.tsx b/mobile-app/app/components/SummaryTitle.tsx
index c21ad98cd2..527f7cf021 100644
--- a/mobile-app/app/components/SummaryTitle.tsx
+++ b/mobile-app/app/components/SummaryTitle.tsx
@@ -7,10 +7,16 @@ import { RandomAvatar } from "@screens/AppNavigator/screens/Portfolio/components
import { AddressType } from "@waveshq/walletkit-ui/dist/store";
import { LocalAddress, WhitelistedAddress } from "@store/userPreferences";
import { DomainType } from "@contexts/DomainContext";
-import { LinearGradient } from "expo-linear-gradient";
+
+import {
+ AddressType as JellyfishAddressType,
+ getAddressType,
+} from "@waveshq/walletkit-core";
+import { useNetworkContext } from "@waveshq/walletkit-ui";
import { View } from ".";
import { ThemedTextV2, ThemedViewV2 } from "./themed";
import { EVMLinearGradient } from "./EVMLinearGradient";
+import { AddressEvmTag } from "./AddressEvmTag";
interface ISummaryTitleProps {
title: string;
@@ -33,6 +39,7 @@ export function SummaryTitle(props: ISummaryTitleProps): JSX.Element {
const IconA = getNativeIcon(props.iconA);
const IconB =
props.iconB !== undefined ? getNativeIcon(props.iconB) : undefined;
+ const { networkName } = useNetworkContext();
return (
<>
@@ -127,60 +134,47 @@ export function SummaryTitle(props: ISummaryTitleProps): JSX.Element {
>
{props.customToAddressTitle ?? translate("screens/common", "To")}
- {/* TODO @chloe cater for selection of evm addr from addr pair */}
+
+ {/* Checks if selected address is a Whitelisted EVM address */}
{(props.matchedAddress as WhitelistedAddress)?.addressDomainType ===
- DomainType.EVM ? (
-
- {props.addressType === AddressType.WalletAddress && (
-
-
-
- )}
-
- {props.toAddressLabel != null &&
- props.toAddressLabel.length > 0
- ? props.toAddressLabel
- : props.toAddress}
-
-
+ <>
+ {props.addressType === AddressType.WalletAddress && (
+
+
+
+ )}
+
+ {props.toAddressLabel}
+
+ >
+
) : (
+ // Whitelisted address - DVM
{props.addressType === AddressType.WalletAddress && (
-
-
+
+
)}
- {props.toAddressLabel != null &&
- props.toAddressLabel.length > 0
- ? props.toAddressLabel
- : props.toAddress}
+ {props.toAddressLabel}
)}
diff --git a/mobile-app/app/hooks/useWalletAddress.ts b/mobile-app/app/hooks/useWalletAddress.ts
index 8d6a9f2095..18bd7b9c7b 100644
--- a/mobile-app/app/hooks/useWalletAddress.ts
+++ b/mobile-app/app/hooks/useWalletAddress.ts
@@ -4,6 +4,7 @@ import { useCallback } from "react";
export interface WalletAddressI {
dvm: string;
evm: string;
+ generatedLabel: string;
}
export function useWalletAddress(): {
@@ -19,7 +20,8 @@ export function useWalletAddress(): {
const account = wallet.get(i);
const dvm = await account.getAddress();
const evm = await account.getEvmAddress();
- addresses.push({ dvm, evm });
+ const generatedLabel = `Address ${i + 1}`;
+ addresses.push({ dvm, evm, generatedLabel });
}
return addresses;
}, [wallet, addressLength]);
diff --git a/mobile-app/app/screens/AppNavigator/screens/Portfolio/components/AddressRow.tsx b/mobile-app/app/screens/AppNavigator/screens/Portfolio/components/AddressRow.tsx
index 02802c8ae3..2167c8d30c 100644
--- a/mobile-app/app/screens/AppNavigator/screens/Portfolio/components/AddressRow.tsx
+++ b/mobile-app/app/screens/AppNavigator/screens/Portfolio/components/AddressRow.tsx
@@ -8,28 +8,27 @@ import {
ThemedTouchableOpacity,
ThemedViewV2,
} from "@components/themed";
-import { RandomAvatar } from "@screens/AppNavigator/screens/Portfolio/components/RandomAvatar";
import { WalletTextInputV2 } from "@components/WalletTextInputV2";
import { Control, Controller } from "react-hook-form";
import { NetworkName } from "@defichain/jellyfish-network";
import { fromAddress } from "@defichain/jellyfish-address";
import {
LocalAddress,
- WhitelistedAddress,
selectAllLabeledWalletAddress,
+ WhitelistedAddress,
} from "@store/userPreferences";
import { debounce } from "lodash";
import { useEffect, useMemo, useState } from "react";
import { useSelector } from "react-redux";
import { RootState } from "@store";
-import { WalletAddressI, useWalletAddress } from "@hooks/useWalletAddress";
+import { useWalletAddress, WalletAddressI } from "@hooks/useWalletAddress";
import { DomainType, useDomainContext } from "@contexts/DomainContext";
-import { LinearGradient } from "expo-linear-gradient";
import {
- getAddressType,
AddressType as JellyfishAddressType,
+ getAddressType,
} from "@waveshq/walletkit-core";
-import { Text } from "@components";
+import { RandomAvatar } from "@screens/AppNavigator/screens/Portfolio/components/RandomAvatar";
+import { AddressEvmTag } from "@components/AddressEvmTag";
export function AddressRow({
control,
@@ -47,6 +46,7 @@ export function AddressRow({
onlyLocalAddress,
matchedAddress,
setMatchedAddress,
+ setAddressLabel,
}: {
control: Control;
networkName: NetworkName;
@@ -65,6 +65,7 @@ export function AddressRow({
onlyLocalAddress?: boolean;
matchedAddress?: LocalAddress | WhitelistedAddress | undefined;
setMatchedAddress?: (address?: LocalAddress | WhitelistedAddress) => void;
+ setAddressLabel?: React.Dispatch>;
}): JSX.Element {
const { fetchWalletAddresses } = useWalletAddress();
const { domain } = useDomainContext();
@@ -94,6 +95,15 @@ export function AddressRow({
return true;
}, [onlyLocalAddress, addressType, address]);
+ const addressObj = jellyfishWalletAddress.find(
+ (e: WalletAddressI) => e.dvm === address || e.evm === address,
+ );
+
+ const displayAddressLabel =
+ matchedAddress?.label !== ""
+ ? matchedAddress?.label
+ : addressObj?.generatedLabel;
+
const debounceMatchAddress = debounce(() => {
// Check if address input field is not empty
if (address !== undefined && setMatchedAddress !== undefined) {
@@ -111,10 +121,6 @@ export function AddressRow({
return;
}
- const addressObj = jellyfishWalletAddress.find(
- (e: WalletAddressI) => e.dvm === address || e.evm === address,
- );
-
if (addressObj) {
// Your addresses - Unlabelled
setMatchedAddress({
@@ -145,6 +151,12 @@ export function AddressRow({
}
}, 200);
+ useEffect(() => {
+ if (setAddressLabel !== undefined) {
+ setAddressLabel(displayAddressLabel);
+ }
+ }, [displayAddressLabel]);
+
useEffect(() => {
debounceMatchAddress();
}, [address, addressBook]);
@@ -270,7 +282,7 @@ export function AddressRow({
{addressType !== undefined && (
<>
- {/* Verified tag for DVM/EVM address */}
+ {/* Verified tag for unsaved but verified DVM/EVM address */}
{addressType === AddressType.OthersButValid && (
<>
>
)}
+
+ {/* Whitelisted and Yours Addresses */}
{addressType !== AddressType.OthersButValid &&
validLocalAddress && (
<>
- {/* TODO @chloe cater for selection of evm addr from addr pair */}
+ {/* Checks if selected address is a Whitelisted EVM address */}
{(matchedAddress as WhitelistedAddress)?.addressDomainType ===
- DomainType.EVM ? (
- // || (matchedAddress as LocalAddress)?.evmAddress ?
-
- {/* TODO add avatar for after updating address book design */}
-
+ <>
+ {addressType === AddressType.WalletAddress && (
+
+
+
)}
- >
- {matchedAddress?.label || matchedAddress?.address}
-
-
+
+ {displayAddressLabel}
+
+ >
+
) : (
+ // Whitelisted address - DVM
- {matchedAddress?.label || matchedAddress?.address}
+ {displayAddressLabel}
)}
diff --git a/mobile-app/app/screens/AppNavigator/screens/Portfolio/components/AddressSelectionButtonV2.tsx b/mobile-app/app/screens/AppNavigator/screens/Portfolio/components/AddressSelectionButtonV2.tsx
index 6f046c97fa..2205c7272b 100644
--- a/mobile-app/app/screens/AppNavigator/screens/Portfolio/components/AddressSelectionButtonV2.tsx
+++ b/mobile-app/app/screens/AppNavigator/screens/Portfolio/components/AddressSelectionButtonV2.tsx
@@ -5,6 +5,9 @@ import {
} from "@components/themed";
import { tailwind } from "@tailwind";
import { useAddressLabel } from "@hooks/useAddressLabel";
+import { useWalletAddress, WalletAddressI } from "@hooks/useWalletAddress";
+import { useEffect, useState } from "react";
+import { useLogger } from "@shared-contexts/NativeLoggingProvider";
import { RandomAvatar } from "./RandomAvatar";
interface AddressSelectionButtonProps {
@@ -15,9 +18,33 @@ interface AddressSelectionButtonProps {
}
export function AddressSelectionButtonV2(
- props: AddressSelectionButtonProps
+ props: AddressSelectionButtonProps,
): JSX.Element {
const addressLabel = useAddressLabel(props.address);
+ const logger = useLogger();
+ const { fetchWalletAddresses } = useWalletAddress();
+ const [availableAddresses, setAvailableAddresses] = useState<
+ WalletAddressI[]
+ >([]);
+ const generatedAddressLabel = availableAddresses.find(
+ (address) => address.dvm === props.address,
+ )?.generatedLabel;
+
+ const displayAddressLabel =
+ addressLabel === "" || addressLabel === null
+ ? generatedAddressLabel
+ : addressLabel;
+
+ // Getting addresses
+ const fetchAddresses = async (): Promise => {
+ const addresses = await fetchWalletAddresses();
+ setAvailableAddresses(addresses);
+ };
+
+ useEffect(() => {
+ fetchAddresses().catch(logger.error);
+ }, [props.address]);
+
return (
- {addressLabel != null ? addressLabel : props.address}
+ {displayAddressLabel}
);
diff --git a/mobile-app/app/screens/AppNavigator/screens/Portfolio/components/BottomSheetAddressDetailV2.tsx b/mobile-app/app/screens/AppNavigator/screens/Portfolio/components/BottomSheetAddressDetailV2.tsx
index bc172161f1..135b8d5b21 100644
--- a/mobile-app/app/screens/AppNavigator/screens/Portfolio/components/BottomSheetAddressDetailV2.tsx
+++ b/mobile-app/app/screens/AppNavigator/screens/Portfolio/components/BottomSheetAddressDetailV2.tsx
@@ -20,11 +20,11 @@ import {
} from "@shared-contexts/WalletContext";
import { useLogger } from "@shared-contexts/NativeLoggingProvider";
import { BottomSheetFlatList } from "@gorhom/bottom-sheet";
-import { useThemeContext, useNetworkContext } from "@waveshq/walletkit-ui";
+import { useNetworkContext, useThemeContext } from "@waveshq/walletkit-ui";
import {
- wallet as walletReducer,
- hasTxQueued,
hasOceanTXQueued,
+ hasTxQueued,
+ wallet as walletReducer,
} from "@waveshq/walletkit-ui/dist/store";
import { useSelector } from "react-redux";
import { loans } from "@store/loans";
@@ -36,12 +36,12 @@ import {
setAddresses,
setUserPreferences,
} from "@store/userPreferences";
-import { useAddressLabel } from "@hooks/useAddressLabel";
import { useAppDispatch } from "@hooks/useAppDispatch";
import { openURL } from "@api/linking";
import { ThemedFlatListV2 } from "@components/themed/ThemedFlatListV2";
-import { WalletAddressI, useWalletAddress } from "@hooks/useWalletAddress";
+import { useWalletAddress, WalletAddressI } from "@hooks/useWalletAddress";
import { DomainType, useDomainContext } from "@contexts/DomainContext";
+import { useAddressLabel } from "@hooks/useAddressLabel";
import { RandomAvatar } from "./RandomAvatar";
interface BottomSheetAddressDetailProps {
@@ -183,7 +183,6 @@ export const BottomSheetAddressDetailV2 = (
) {
return;
}
-
dispatch(walletReducer.actions.setHasFetchedToken(false));
dispatch(loans.actions.setHasFetchedVaultsData(false));
await setIndex(index);
@@ -192,19 +191,22 @@ export const BottomSheetAddressDetailV2 = (
};
const AddressListItem = useCallback(
- // eslint-disable-next-line react/no-unused-prop-types
({
item,
index,
}: {
+ // eslint-disable-next-line react/no-unused-prop-types
item: WalletAddressI;
+ // eslint-disable-next-line react/no-unused-prop-types
index: number;
}): JSX.Element => {
const isSelected = item.dvm === props.address;
- const hasLabel =
- labeledAddresses?.[item.dvm]?.label != null &&
- labeledAddresses?.[item.dvm]?.label !== "";
const displayAddress = domain === DomainType.EVM ? item.evm : item.dvm;
+
+ // if no existing address label, then display label from generatedAddress key
+ const displayAddressLabel =
+ labeledAddresses?.[item.dvm]?.label ?? item.generatedLabel;
+
return (
- {hasLabel && (
-
-
- {labeledAddresses[item.dvm]?.label}
-
- {isSelected && (
-
- )}
-
- )}
- {isSelected && !hasLabel && (
+
+ {displayAddressLabel}
+
+ {isSelected && (
)}
+
+
await openURL(getAddressUrl(displayAddress))
@@ -349,6 +338,11 @@ export const BottomSheetAddressDetailV2 = (
const activeAddress = availableAddresses.find(
({ dvm }) => dvm === props.address,
);
+ const displayAddressLabel =
+ activeLabel === null
+ ? activeAddress?.generatedLabel
+ : labeledAddresses?.[props.address]?.label;
+
const activeDomainAddress =
domain === DomainType.DVM ? activeAddress?.dvm : activeAddress?.evm;
return (
@@ -356,18 +350,16 @@ export const BottomSheetAddressDetailV2 = (
style={tailwind("flex flex-col w-full px-5 py-2 items-center")}
>
- {activeLabel != null && (
-
-
- {activeLabel}
-
-
- )}
+
+
+ {displayAddressLabel}
+
+
onActiveAddressPress(activeDomainAddress ?? "")}
diff --git a/mobile-app/app/screens/AppNavigator/screens/Portfolio/components/CreateOrEditAddressLabelForm.tsx b/mobile-app/app/screens/AppNavigator/screens/Portfolio/components/CreateOrEditAddressLabelForm.tsx
index f0bf08b989..b359e48d28 100644
--- a/mobile-app/app/screens/AppNavigator/screens/Portfolio/components/CreateOrEditAddressLabelForm.tsx
+++ b/mobile-app/app/screens/AppNavigator/screens/Portfolio/components/CreateOrEditAddressLabelForm.tsx
@@ -106,6 +106,7 @@ export const CreateOrEditAddressLabelForm = memo(
const isSaveDisabled = (): boolean => {
if (
labelInput === undefined ||
+ labelInput === "" || // disable if label is empty or no change in label
labelInput === addressLabel?.label ||
labelInputErrorMessage !== ""
) {
diff --git a/mobile-app/app/screens/AppNavigator/screens/Portfolio/screens/AddressBookScreen.tsx b/mobile-app/app/screens/AppNavigator/screens/Portfolio/screens/AddressBookScreen.tsx
index 3f1aaf21af..a466304281 100644
--- a/mobile-app/app/screens/AppNavigator/screens/Portfolio/screens/AddressBookScreen.tsx
+++ b/mobile-app/app/screens/AppNavigator/screens/Portfolio/screens/AddressBookScreen.tsx
@@ -1,4 +1,4 @@
-import { View, Text } from "@components";
+import { View } from "@components";
import {
ThemedIcon,
ThemedSectionTitleV2,
@@ -13,11 +13,11 @@ import {
import { RootState } from "@store";
import {
LocalAddress,
- WhitelistedAddress,
selectAddressBookArray,
selectLocalWalletAddressArray,
setUserPreferences,
userPreferences,
+ WhitelistedAddress,
} from "@store/userPreferences";
import { getColor, tailwind } from "@tailwind";
import { translate } from "@translations";
@@ -38,7 +38,7 @@ import { debounce } from "lodash";
import { openURL } from "@api/linking";
import { Logging } from "@api";
import { useWalletContext } from "@shared-contexts/WalletContext";
-import { WalletAddressI, useWalletAddress } from "@hooks/useWalletAddress";
+import { useWalletAddress, WalletAddressI } from "@hooks/useWalletAddress";
import { useAppDispatch } from "@hooks/useAppDispatch";
import LightEmptyAddress from "@assets/images/empty-address-light.png";
import DarkEmptyAddress from "@assets/images/empty-address-dark.png";
@@ -47,14 +47,15 @@ import { useNavigatorScreenOptions } from "@hooks/useNavigatorScreenOptions";
import { SearchInput } from "@components/SearchInput";
import { RefreshIcon } from "@screens/WalletNavigator/assets/RefreshIcon";
import { DomainType } from "@contexts/DomainContext";
-import { LinearGradient } from "expo-linear-gradient";
+import { RandomAvatar } from "@screens/AppNavigator/screens/Portfolio/components/RandomAvatar";
+import { EvmTag } from "@components/EvmTag";
+import { useLogger } from "@shared-contexts/NativeLoggingProvider";
import { ButtonGroup } from "../../Dex/components/ButtonGroup";
import {
FavoriteCheckIcon,
FavoriteUnCheckIcon,
} from "../../Settings/assets/FavoriteIcon";
import { SettingsParamList } from "../../Settings/SettingsNavigator";
-import { RandomAvatar } from "../components/RandomAvatar";
type Props = StackScreenProps;
@@ -69,7 +70,8 @@ export function AddressBookScreen({ route, navigation }: Props): JSX.Element {
const { isLight } = useThemeContext();
const { network } = useNetworkContext();
const dispatch = useAppDispatch();
- // condition to hide icon from send page
+ const logger = useLogger();
+ // condition to hide icon if not from send page
const enableAddressSelect =
selectedAddress !== undefined && onAddressSelect !== undefined;
const userPreferencesFromStore = useSelector(
@@ -115,6 +117,16 @@ export function AddressBookScreen({ route, navigation }: Props): JSX.Element {
: ButtonGroupTabKey.Whitelisted,
);
+ const [availableAddresses, setAvailableAddresses] = useState<
+ WalletAddressI[]
+ >([]);
+
+ // Getting addresses
+ const fetchAddresses = async (): Promise => {
+ const addresses = await fetchWalletAddresses();
+ setAvailableAddresses(addresses);
+ };
+
useEffect(() => {
// combine redux store and jellyfish wallet
let isSubscribed = true;
@@ -225,6 +237,10 @@ export function AddressBookScreen({ route, navigation }: Props): JSX.Element {
});
};
+ useEffect(() => {
+ fetchAddresses().catch(logger.error);
+ }, []);
+
useEffect(() => {
// sync all store changes to local storage
const updateLocalStorage = async (): Promise => {
@@ -267,46 +283,28 @@ export function AddressBookScreen({ route, navigation }: Props): JSX.Element {
});
}, []);
- const AddressListItem = useCallback(
+ const WhitelistedAddressItem = useCallback(
({
selectedAddress,
onAddressSelect,
...props
}: {
- item: LocalAddress | WhitelistedAddress;
+ item: WhitelistedAddress;
index: number;
testIDSuffix: string;
selectedAddress?: string;
onAddressSelect?: (address: string) => void;
}): JSX.Element => {
const { item, index, testIDSuffix } = props;
- const isDisabledToSelect = !!(
- (
- enableAddressSelect &&
- activeButtonGroup === ButtonGroupTabKey.Whitelisted &&
- (item as WhitelistedAddress).addressDomainType ===
- addressDomainType &&
- addressDomainType === DomainType.EVM
- ) // disable address selection if its from the same EVM domain
- );
+ const isDisabledToSelect =
+ enableAddressSelect &&
+ activeButtonGroup === ButtonGroupTabKey.Whitelisted &&
+ (item as WhitelistedAddress).addressDomainType === addressDomainType &&
+ addressDomainType === DomainType.EVM; // disable address selection if its from the same EVM domain
- const onChangeAddress = (
- addressDetail: LocalAddress | WhitelistedAddress,
- ): void => {
+ const onChangeAddress = (addressDetail: WhitelistedAddress): void => {
if (onAddressSelect) {
- // for whitelisted address
- if (activeButtonGroup === ButtonGroupTabKey.Whitelisted) {
- onAddressSelect(addressDetail.address);
- } else {
- // for wallet address
- onAddressSelect(
- addressDomainType === DomainType.EVM
- ? (addressDetail as LocalAddress).evmAddress
- : addressDetail.address,
- );
- }
- } else {
- onDFIAddressClick();
+ onAddressSelect(addressDetail.address);
}
};
@@ -349,33 +347,27 @@ export function AddressBookScreen({ route, navigation }: Props): JSX.Element {
"flex-auto": Platform.OS === "web",
})}
>
- {activeButtonGroup === ButtonGroupTabKey.YourAddress ? (
-
-
-
- ) : (
-
- await onFavouriteAddress(item as WhitelistedAddress)
- }
- testID={`address_row_star_${index}_${testIDSuffix}`}
- disabled={enableAddressSelect}
- >
- {(item as WhitelistedAddress).isFavourite === true ? (
-
- ) : (
-
- )}
-
- )}
+
+ await onFavouriteAddress(item as WhitelistedAddress)
+ }
+ testID={`address_row_star_${index}_${testIDSuffix}`}
+ disabled={enableAddressSelect}
+ >
+ {item.isFavourite ? (
+
+ ) : (
+
+ )}
+
@@ -387,29 +379,14 @@ export function AddressBookScreen({ route, navigation }: Props): JSX.Element {
>
{item.label}
- {activeButtonGroup === ButtonGroupTabKey.Whitelisted &&
- (item as WhitelistedAddress).addressDomainType ===
- DomainType.EVM && (
-
-
- EVM
-
-
- )}
+ {(item as WhitelistedAddress).addressDomainType ===
+ DomainType.EVM && (
+
+ )}
)}
{/* for DFI address */}
- onChangeAddress(item)}
/>
{/* for EVM address */}
- {activeButtonGroup === ButtonGroupTabKey.YourAddress && (
- {
- await openURL(
- getAddressUrl((item as LocalAddress).evmAddress),
- );
- }}
- />
- )}
- {!enableAddressSelect &&
- activeButtonGroup === ButtonGroupTabKey.Whitelisted && (
+ {enableAddressSelect && (
+
- )}
+
+ )}
@@ -451,6 +420,109 @@ export function AddressBookScreen({ route, navigation }: Props): JSX.Element {
[filteredAddressBook, filteredWalletAddress, activeButtonGroup],
);
+ const YourAddressListItem = useCallback(
+ ({
+ selectedAddress,
+ onAddressSelect,
+ ...props
+ }: {
+ item: LocalAddress;
+ index: number;
+ testIDSuffix: string;
+ selectedAddress?: string;
+ onAddressSelect?: (address: string) => void;
+ }): JSX.Element => {
+ const { item, index, testIDSuffix } = props;
+
+ const generatedAddressLabel = availableAddresses.find(
+ (address) =>
+ address.dvm === item.address || address.evm === item.evmAddress,
+ )?.generatedLabel;
+ const displayAddressLabel =
+ item.label === "" ? generatedAddressLabel : item.label;
+
+ const onChangeAddress = (addressDetail: string): void => {
+ if (onAddressSelect) {
+ onAddressSelect(addressDetail);
+ }
+ };
+
+ const onDFIAddressClick = async () => {
+ await openURL(getAddressUrl(item.address));
+ };
+
+ return (
+ // Your Address card
+
+
+
+
+
+
+ {item.label !== "" ? (
+
+ {item.label}
+
+ ) : (
+
+ {displayAddressLabel}
+
+ )}
+
+
+
+
+ {/* DVM address card */}
+ {
+ onChangeAddress(item.address);
+ }}
+ enableAddressSelect={enableAddressSelect}
+ />
+ {/* EVM address card */}
+ {
+ onChangeAddress(item.evmAddress);
+ }}
+ enableAddressSelect={enableAddressSelect}
+ />
+
+
+
+
+ );
+ },
+ [filteredAddressBook, filteredWalletAddress, activeButtonGroup],
+ );
+
const goToAddAddressForm = (): void => {
navigation.navigate({
name: "AddOrEditAddressBookScreen",
@@ -587,29 +659,41 @@ export function AddressBookScreen({ route, navigation }: Props): JSX.Element {
) : (
<>
+ {/* Search address */}
{!isSearchFocus && searchString?.trim().length === 0 && (
)}
- {(activeButtonGroup === ButtonGroupTabKey.Whitelisted
- ? filteredAddressBook
- : filteredWalletAddress
- ).map((item: LocalAddress | WhitelistedAddress, index: number) => (
-
- ))}
+
+ {/* wWhitelisted address tab */}
+ {activeButtonGroup === ButtonGroupTabKey.Whitelisted &&
+ filteredAddressBook.map(
+ (item: WhitelistedAddress, index: number) => (
+
+ ),
+ )}
+
+ {/* Your address tab */}
+ {activeButtonGroup === ButtonGroupTabKey.YourAddress &&
+ filteredWalletAddress.map((item: LocalAddress, index: number) => (
+
+ ))}
>
)}
@@ -684,7 +768,7 @@ export function DiscoverWalletAddressV2({
);
}
-function YourAddressLink({
+function WhitelistedAddressLink({
disabled,
onClick,
address,
@@ -729,3 +813,55 @@ function YourAddressLink({
);
}
+
+function YourAddressLink({
+ disabled,
+ onClick,
+ address,
+ isEvmAddress,
+ testIDSuffix,
+ enableAddressSelect,
+}: {
+ disabled?: boolean;
+ onClick: () => Promise;
+ address: string;
+ isEvmAddress?: boolean;
+ testIDSuffix: string;
+ enableAddressSelect: boolean;
+}) {
+ return (
+
+
+
+ {address}
+
+ {isEvmAddress && }
+
+ {enableAddressSelect && (
+
+ )}
+
+ );
+}
diff --git a/mobile-app/app/screens/AppNavigator/screens/Portfolio/screens/SendScreen.tsx b/mobile-app/app/screens/AppNavigator/screens/Portfolio/screens/SendScreen.tsx
index 3621325055..d78d3b90b4 100644
--- a/mobile-app/app/screens/AppNavigator/screens/Portfolio/screens/SendScreen.tsx
+++ b/mobile-app/app/screens/AppNavigator/screens/Portfolio/screens/SendScreen.tsx
@@ -102,6 +102,7 @@ export function SendScreen({ route, navigation }: Props): JSX.Element {
const [transactionCardStatus, setTransactionCardStatus] = useState(
TransactionCardStatus.Default,
);
+ const [addressLabel, setAddressLabel] = useState("");
// form
const { control, setValue, formState, getValues, trigger, watch } = useForm({
@@ -263,7 +264,7 @@ export function SendScreen({ route, navigation }: Props): JSX.Element {
amount: new BigNumber(values.amount),
amountInUsd: amountInUSDValue,
fee,
- toAddressLabel: matchedAddress?.label,
+ toAddressLabel: addressLabel,
addressType,
matchedAddress,
};
@@ -448,6 +449,7 @@ export function SendScreen({ route, navigation }: Props): JSX.Element {
onAddressType={setAddressType}
matchedAddress={matchedAddress}
setMatchedAddress={setMatchedAddress}
+ setAddressLabel={setAddressLabel}
/>
)}