import React, { useState, useRef, useContext } from 'react'; import PropTypes from 'prop-types'; import classnames from 'classnames'; import { useSelector } from 'react-redux'; import { useI18nContext } from '../../../hooks/useI18nContext'; import { getRpcPrefsForCurrentProvider } from '../../../selectors'; import { getURLHostName, shortenAddress } from '../../../helpers/utils/util'; import { AccountListItemMenu } from '..'; import Box from '../../ui/box/box'; import { AvatarAccount, Text, AvatarFavicon, Tag, ButtonLink, ButtonIcon, IconName, IconSize, AvatarAccountVariant, } from '../../component-library'; import { Color, TEXT_ALIGN, AlignItems, DISPLAY, TextVariant, FLEX_DIRECTION, BorderRadius, JustifyContent, Size, BorderColor, } from '../../../helpers/constants/design-system'; import { HardwareKeyringNames } from '../../../../shared/constants/hardware-wallets'; import { KeyringType } from '../../../../shared/constants/keyring'; import UserPreferencedCurrencyDisplay from '../../app/user-preferenced-currency-display/user-preferenced-currency-display.component'; import { SECONDARY, PRIMARY } from '../../../helpers/constants/common'; import { findKeyringForAddress } from '../../../ducks/metamask/metamask'; import Tooltip from '../../ui/tooltip/tooltip'; import { MetaMetricsEventCategory, MetaMetricsEventName, } from '../../../../shared/constants/metametrics'; import { MetaMetricsContext } from '../../../contexts/metametrics'; const MAXIMUM_CURRENCY_DECIMALS = 3; const MAXIMUM_CHARACTERS_WITHOUT_TOOLTIP = 17; function getLabel(keyring = {}, t) { const { type } = keyring; switch (type) { case KeyringType.qr: return HardwareKeyringNames.qr; case KeyringType.imported: return t('imported'); case KeyringType.trezor: return HardwareKeyringNames.trezor; case KeyringType.ledger: return HardwareKeyringNames.ledger; case KeyringType.lattice: return HardwareKeyringNames.lattice; default: return null; } } export const AccountListItem = ({ identity, selected = false, onClick, closeMenu, connectedAvatar, connectedAvatarName, }) => { const t = useI18nContext(); const [accountOptionsMenuOpen, setAccountOptionsMenuOpen] = useState(false); const ref = useRef(false); const useBlockie = useSelector((state) => state.metamask.useBlockie); const keyring = useSelector((state) => findKeyringForAddress(state, identity.address), ); const label = getLabel(keyring, t); const rpcPrefs = useSelector(getRpcPrefsForCurrentProvider); const { blockExplorerUrl } = rpcPrefs; const blockExplorerUrlSubTitle = getURLHostName(blockExplorerUrl); const trackEvent = useContext(MetaMetricsContext); return ( { // Without this check, the account will be selected after // the account options menu closes if (!accountOptionsMenuOpen) { onClick(); } }} > {selected && ( )} {identity.name.length > MAXIMUM_CHARACTERS_WITHOUT_TOOLTIP ? ( {identity.name} ) : ( identity.name )} {connectedAvatar ? ( ) : null} {shortenAddress(identity.address)} {label ? ( ) : null}
{ e.stopPropagation(); trackEvent({ event: MetaMetricsEventName.AccountDetailMenuOpened, category: MetaMetricsEventCategory.Navigation, properties: { location: 'Account Options', }, }); setAccountOptionsMenuOpen(true); }} data-testid="account-list-item-menu-button" /> {accountOptionsMenuOpen ? ( setAccountOptionsMenuOpen(false)} isRemovable={keyring?.type !== KeyringType.hdKeyTree} closeMenu={closeMenu} /> ) : null}
); }; AccountListItem.propTypes = { /** * Identity of the account */ identity: PropTypes.shape({ name: PropTypes.string.isRequired, address: PropTypes.string.isRequired, balance: PropTypes.string.isRequired, }).isRequired, /** * Represents if this account is currently selected */ selected: PropTypes.bool, /** * Function to execute when the item is clicked */ onClick: PropTypes.func.isRequired, /** * Function that closes the menu */ closeMenu: PropTypes.func, /** * File location of the avatar icon */ connectedAvatar: PropTypes.string, /** * Text used as the avatar alt text */ connectedAvatarName: PropTypes.string, }; AccountListItem.displayName = 'AccountListItem';