diff --git a/src/app/components/account/account-list-item-layout.tsx b/src/app/components/account/account-list-item-layout.tsx
index 5fde1bb6215..82cc25d2a59 100644
--- a/src/app/components/account/account-list-item-layout.tsx
+++ b/src/app/components/account/account-list-item-layout.tsx
@@ -1,58 +1,82 @@
+import { ComponentProps, ReactNode } from 'react';
+
import { SettingsSelectors } from '@tests/selectors/settings.selectors';
-import { Flex, HStack, Stack, StackProps, styled } from 'leather-styles/jsx';
+import { HStack } from 'leather-styles/jsx';
import { useViewportMinWidth } from '@app/common/hooks/use-media-query';
import { BulletSeparator } from '@app/ui/components/bullet-separator/bullet-separator';
import { CheckmarkIcon } from '@app/ui/components/icons/checkmark-icon';
+import { Item } from '@app/ui/components/items/item';
+import { ItemDefaultLayout } from '@app/ui/components/items/item-default.layout';
import { Spinner } from '@app/ui/components/spinner';
import { truncateMiddle } from '@app/ui/utils/truncate-middle';
-import { Flag } from '../layout/flag';
import { StacksAccountLoader } from '../loaders/stacks-account-loader';
import { BitcoinNativeSegwitAccountLoader } from './bitcoin-account-loader';
-interface AccountListItemLayoutProps extends StackProps {
- isLoading: boolean;
- isActive: boolean;
+interface AccountListItemLayoutProps extends ComponentProps<'div'> {
+ accountName: ReactNode;
+ avatar: ReactNode;
+ balanceLabel: ReactNode;
index: number;
- accountName: React.ReactNode;
- avatar: React.JSX.Element;
- balanceLabel: React.ReactNode;
+ isLoading: boolean;
+ isPressable?: boolean;
+ isSelected: boolean;
onSelectAccount(): void;
}
export function AccountListItemLayout(props: AccountListItemLayoutProps) {
const {
- index,
- isLoading,
- isActive,
accountName,
avatar,
balanceLabel,
+ index,
+ isLoading,
+ isPressable,
+ isSelected,
onSelectAccount,
- children = null,
- ...rest
} = props;
const isBreakpointSm = useViewportMinWidth('sm');
return (
-
-
-
-
-
- {accountName}
- {isActive && }
+
+
+ {accountName}
+ {isSelected && }
- {isLoading ? (
+ }
+ contentLeftBottom={
+
+
+
+ {account => (
+
+ {truncateMiddle(account.address, isBreakpointSm ? 4 : 3)}
+
+ )}
+
+
+ {signer => (
+
+ {truncateMiddle(signer.address, 5)}
+
+ )}
+
+
+
+ }
+ contentRightTop={
+ isLoading ? (
) : (
- balanceLabel
- )}
-
-
-
-
- {account => (
-
- {truncateMiddle(account.address, isBreakpointSm ? 4 : 3)}
-
- )}
-
-
-
- {signer => (
-
- {truncateMiddle(signer.address, 5)}
-
- )}
-
-
-
-
-
- {children}
-
+ {balanceLabel}
+ )
+ }
+ />
+
+
);
}
diff --git a/src/app/components/bitcoin-transaction-item/bitcoin-transaction-item.tsx b/src/app/components/bitcoin-transaction-item/bitcoin-transaction-item.tsx
index 227990cc93f..3838cd0a656 100644
--- a/src/app/components/bitcoin-transaction-item/bitcoin-transaction-item.tsx
+++ b/src/app/components/bitcoin-transaction-item/bitcoin-transaction-item.tsx
@@ -1,8 +1,6 @@
import { useMemo } from 'react';
import { useLocation, useNavigate } from 'react-router-dom';
-import { BoxProps } from 'leather-styles/jsx';
-
import { BitcoinTx } from '@shared/models/transactions/bitcoin-transaction.model';
import { RouteUrls } from '@shared/route-urls';
@@ -15,7 +13,6 @@ import {
isBitcoinTxInbound,
} from '@app/common/transactions/bitcoin/utils';
import { openInNewTab } from '@app/common/utils/open-in-new-tab';
-import { usePressable } from '@app/components/item-hover';
import { IncreaseFeeButton } from '@app/components/stacks-transaction-item/increase-fee-button';
import { TransactionTitle } from '@app/components/transaction/transaction-title';
import {
@@ -34,11 +31,10 @@ import { InscriptionIcon } from './bitcoin-transaction-inscription-icon';
import { BitcoinTransactionStatus } from './bitcoin-transaction-status';
import { BitcoinTransactionValue } from './bitcoin-transaction-value';
-interface BitcoinTransactionItemProps extends BoxProps {
+interface BitcoinTransactionItemProps {
transaction: BitcoinTx;
}
-export function BitcoinTransactionItem({ transaction, ...rest }: BitcoinTransactionItemProps) {
- const [component, bind, { isHovered }] = usePressable(true);
+export function BitcoinTransactionItem({ transaction }: BitcoinTransactionItemProps) {
const { pathname } = useLocation();
const navigate = useNavigate();
@@ -92,7 +88,6 @@ export function BitcoinTransactionItem({ transaction, ...rest }: BitcoinTransact
const increaseFeeButton = (
@@ -101,6 +96,7 @@ export function BitcoinTransactionItem({ transaction, ...rest }: BitcoinTransact
return (
}
txTitle={}
txValue={txValue}
- belowCaptionEl={increaseFeeButton}
- {...bind}
- {...rest}
- >
- {component}
-
+ />
);
}
diff --git a/src/app/components/crypto-assets/bitcoin/brc20-token-asset-list/brc20-token-asset-list.tsx b/src/app/components/crypto-assets/bitcoin/brc20-token-asset-list/brc20-token-asset-list.tsx
index 5bd77800863..8652feb9670 100644
--- a/src/app/components/crypto-assets/bitcoin/brc20-token-asset-list/brc20-token-asset-list.tsx
+++ b/src/app/components/crypto-assets/bitcoin/brc20-token-asset-list/brc20-token-asset-list.tsx
@@ -3,11 +3,13 @@ import { useNavigate } from 'react-router-dom';
import { RouteUrls } from '@shared/route-urls';
import { noop } from '@shared/utils';
-import { Brc20TokenAssetItem } from '@app/components/crypto-assets/bitcoin/brc20-token-asset-list/components/brc20-token-asset-item';
import { useNativeSegwitBalance } from '@app/query/bitcoin/balance/btc-native-segwit-balance.hooks';
import { Brc20Token } from '@app/query/bitcoin/ordinals/brc20/brc20-tokens.query';
import { useCurrentAccountNativeSegwitAddressIndexZero } from '@app/store/accounts/blockchain/bitcoin/native-segwit-account.hooks';
+import { Brc20TokenAssetItemLayout } from './components/brc20-token-asset-item.layout';
+import { Brc20AssetListLayout } from './components/brc20-token-asset-list.layout';
+
export function Brc20TokenAssetList(props: { brc20Tokens?: Brc20Token[] }) {
const navigate = useNavigate();
const currentAccountBtcAddress = useCurrentAccountNativeSegwitAddressIndexZero();
@@ -24,13 +26,18 @@ export function Brc20TokenAssetList(props: { brc20Tokens?: Brc20Token[] }) {
}
if (!props.brc20Tokens?.length) return null;
- return props.brc20Tokens.map(token => (
- navigateToBrc20SendForm(token) : noop}
- displayNotEnoughBalance={!hasPositiveBtcBalanceForFees}
- key={token.tick}
- />
- ));
+
+ return (
+
+ {props.brc20Tokens?.map(token => (
+ navigateToBrc20SendForm(token) : noop}
+ />
+ ))}
+
+ );
}
diff --git a/src/app/components/crypto-assets/bitcoin/brc20-token-asset-list/components/brc20-token-asset-item.layout.tsx b/src/app/components/crypto-assets/bitcoin/brc20-token-asset-list/components/brc20-token-asset-item.layout.tsx
index 0e9223ec493..750b729abc4 100644
--- a/src/app/components/crypto-assets/bitcoin/brc20-token-asset-list/components/brc20-token-asset-item.layout.tsx
+++ b/src/app/components/crypto-assets/bitcoin/brc20-token-asset-list/components/brc20-token-asset-item.layout.tsx
@@ -1,69 +1,52 @@
-import { BoxProps, Flex, HStack, styled } from 'leather-styles/jsx';
-
-import type { Money } from '@shared/models/money.model';
+import { createMoney } from '@shared/models/money.model';
import { formatBalance } from '@app/common/format-balance';
-import { AssetCaption } from '@app/components/crypto-assets/components/asset-caption';
-import { usePressable } from '@app/components/item-hover';
-import { Flag } from '@app/components/layout/flag';
+import { Brc20Token } from '@app/query/bitcoin/ordinals/brc20/brc20-tokens.query';
import { Brc20TokenIcon } from '@app/ui/components/icons/brc20-token-icon';
+import { Item } from '@app/ui/components/items/item';
+import { ItemDefaultLayout } from '@app/ui/components/items/item-default.layout';
import { BasicTooltip } from '@app/ui/components/tooltip/basic-tooltip';
-interface Brc20TokenAssetItemLayoutProps extends BoxProps {
- balance: Money;
- caption: string;
+interface Brc20TokenAssetItemLayoutProps {
+ token: Brc20Token;
isPressable?: boolean;
onClick?(): void;
- title: string;
displayNotEnoughBalance?: boolean;
}
export function Brc20TokenAssetItemLayout({
- balance,
- caption,
isPressable,
onClick,
- title,
displayNotEnoughBalance,
+ token,
}: Brc20TokenAssetItemLayoutProps) {
- const [component, bind] = usePressable(isPressable);
-
+ const balance = createMoney(Number(token.overall_balance), token.tick, 0);
const formattedBalance = formatBalance(balance.amount.toString());
return (
-
- } spacing="space.04" width="100%">
-
-
- {title}
-
-
-
- {formattedBalance.value}
-
-
-
-
-
-
- {component}
-
-
+
+
+ }
+ contentLeftTop={{token.tick}}
+ contentLeftBottom={{'BRC-20'}}
+ contentRightTop={
+
+ {formattedBalance.value}
+
+ }
+ />
+
+
);
}
diff --git a/src/app/components/crypto-assets/bitcoin/brc20-token-asset-list/components/brc20-token-asset-item.tsx b/src/app/components/crypto-assets/bitcoin/brc20-token-asset-list/components/brc20-token-asset-item.tsx
deleted file mode 100644
index 29820b11099..00000000000
--- a/src/app/components/crypto-assets/bitcoin/brc20-token-asset-list/components/brc20-token-asset-item.tsx
+++ /dev/null
@@ -1,29 +0,0 @@
-import { createMoney } from '@shared/models/money.model';
-
-import { Brc20Token } from '@app/query/bitcoin/ordinals/brc20/brc20-tokens.query';
-
-import { Brc20TokenAssetItemLayout } from './brc20-token-asset-item.layout';
-
-interface Brc20TokenAssetItemProps {
- token: Brc20Token;
- isPressable?: boolean;
- onClick?(): void;
- displayNotEnoughBalance?: boolean;
-}
-export function Brc20TokenAssetItem({
- token,
- isPressable,
- onClick,
- displayNotEnoughBalance,
-}: Brc20TokenAssetItemProps) {
- return (
-
- );
-}
diff --git a/src/app/components/crypto-assets/bitcoin/brc20-token-asset-list/components/brc20-token-asset-list.layout.tsx b/src/app/components/crypto-assets/bitcoin/brc20-token-asset-list/components/brc20-token-asset-list.layout.tsx
new file mode 100644
index 00000000000..41ea4db9c38
--- /dev/null
+++ b/src/app/components/crypto-assets/bitcoin/brc20-token-asset-list/components/brc20-token-asset-list.layout.tsx
@@ -0,0 +1,10 @@
+import { CryptoAssetSelectors } from '@tests/selectors/crypto-asset.selectors';
+import { Stack, StackProps } from 'leather-styles/jsx';
+
+export function Brc20AssetListLayout({ children }: StackProps) {
+ return (
+
+ {children}
+
+ );
+}
diff --git a/src/app/components/crypto-assets/choose-crypto-asset/crypto-asset-list-item.tsx b/src/app/components/crypto-assets/choose-crypto-asset/crypto-asset-list-item.tsx
index e98f899f166..81eb70a1ecc 100644
--- a/src/app/components/crypto-assets/choose-crypto-asset/crypto-asset-list-item.tsx
+++ b/src/app/components/crypto-assets/choose-crypto-asset/crypto-asset-list-item.tsx
@@ -1,7 +1,6 @@
import type { AllTransferableCryptoAssetBalances } from '@shared/models/crypto-asset-balance.model';
-import { CryptoCurrencyAssetItem } from '@app/components/crypto-assets/crypto-currency-asset/crypto-currency-asset-item';
-
+import { CryptoCurrencyAssetItemLayout } from '../crypto-currency-asset/crypto-currency-asset-item.layout';
import { CryptoCurrencyAssetIcon } from './crypto-currency-asset-icon';
import { FungibleTokenAssetItem } from './fungible-token-asset-item';
@@ -16,7 +15,7 @@ export function CryptoAssetListItem(props: CryptoAssetListItemProps) {
switch (type) {
case 'crypto-currency':
return (
- }
isPressable
diff --git a/src/app/components/crypto-assets/choose-crypto-asset/crypto-asset-list.layout.tsx b/src/app/components/crypto-assets/choose-crypto-asset/crypto-asset-list.layout.tsx
index e1803d73421..c5c9418347b 100644
--- a/src/app/components/crypto-assets/choose-crypto-asset/crypto-asset-list.layout.tsx
+++ b/src/app/components/crypto-assets/choose-crypto-asset/crypto-asset-list.layout.tsx
@@ -3,12 +3,7 @@ import { Stack, StackProps } from 'leather-styles/jsx';
export function CryptoAssetListLayout({ children }: StackProps) {
return (
-
+
{children}
);
diff --git a/src/app/components/crypto-assets/choose-crypto-asset/crypto-asset-list.tsx b/src/app/components/crypto-assets/choose-crypto-asset/crypto-asset-list.tsx
index fd1fb2f0e0f..d40c64ede1f 100644
--- a/src/app/components/crypto-assets/choose-crypto-asset/crypto-asset-list.tsx
+++ b/src/app/components/crypto-assets/choose-crypto-asset/crypto-asset-list.tsx
@@ -8,7 +8,7 @@ import { Brc20TokensLoader } from '@app/components/brc20-tokens-loader';
import { Brc20TokenAssetList } from '@app/components/crypto-assets/bitcoin/brc20-token-asset-list/brc20-token-asset-list';
import { BtcIcon } from '@app/ui/components/icons/btc-icon';
-import { CryptoCurrencyAssetItem } from '../crypto-currency-asset/crypto-currency-asset-item';
+import { CryptoCurrencyAssetItemLayout } from '../crypto-currency-asset/crypto-currency-asset-item.layout';
import { CryptoAssetListItem } from './crypto-asset-list-item';
import { CryptoAssetListLayout } from './crypto-asset-list.layout';
@@ -25,7 +25,7 @@ export function CryptoAssetList({ cryptoAssetBalances, onItemClick }: CryptoAsse
{signer => (
{balance => (
- }
onClick={() => onItemClick(balance)}
diff --git a/src/app/components/crypto-assets/choose-crypto-asset/fungible-token-asset-item.tsx b/src/app/components/crypto-assets/choose-crypto-asset/fungible-token-asset-item.tsx
index 0ce6b98c4ff..8d7389ea14f 100644
--- a/src/app/components/crypto-assets/choose-crypto-asset/fungible-token-asset-item.tsx
+++ b/src/app/components/crypto-assets/choose-crypto-asset/fungible-token-asset-item.tsx
@@ -2,7 +2,7 @@ import { FlexProps } from 'leather-styles/jsx';
import type { StacksFungibleTokenAssetBalance } from '@shared/models/crypto-asset-balance.model';
-import { StacksFungibleTokenAssetItem } from '@app/components/crypto-assets/stacks/fungible-token-asset/stacks-fungible-token-asset-item';
+import { StacksFungibleTokenAssetItemLayout } from '../stacks/fungible-token-asset/stacks-fungible-token-asset-item.layout';
interface FungibleTokenAssetItemProps extends FlexProps {
assetBalance: StacksFungibleTokenAssetBalance;
@@ -14,7 +14,11 @@ export function FungibleTokenAssetItem({ assetBalance, onClick }: FungibleTokenA
switch (blockchain) {
case 'stacks':
return (
-
+
);
default:
return null;
diff --git a/src/app/components/crypto-assets/components/asset-caption.tsx b/src/app/components/crypto-assets/components/asset-caption.tsx
deleted file mode 100644
index 25116dfecf0..00000000000
--- a/src/app/components/crypto-assets/components/asset-caption.tsx
+++ /dev/null
@@ -1,37 +0,0 @@
-import { Flex, HStack, styled } from 'leather-styles/jsx';
-
-import { BulletOperator } from '@app/ui/components/bullet-separator/bullet-separator';
-import { InfoIcon } from '@app/ui/components/icons/info-icon';
-import { BasicTooltip } from '@app/ui/components/tooltip/basic-tooltip';
-
-interface AssetCaptionProps {
- caption: string;
- isUnanchored?: boolean;
-}
-export function AssetCaption({ caption, isUnanchored }: AssetCaptionProps) {
- return (
-
- {caption}{' '}
- {isUnanchored ? (
- <>
-
- Microblock
-
-
-
-
-
-
-
-
- >
- ) : (
- ''
- )}
-
- );
-}
diff --git a/src/app/components/crypto-assets/components/asset-row-grid.tsx b/src/app/components/crypto-assets/components/asset-row-grid.tsx
deleted file mode 100644
index 3508934ab32..00000000000
--- a/src/app/components/crypto-assets/components/asset-row-grid.tsx
+++ /dev/null
@@ -1,40 +0,0 @@
-import { Grid, GridItem } from 'leather-styles/jsx';
-
-interface AssetRowGridProps {
- title: React.ReactNode;
- balance: React.ReactNode;
- caption: React.ReactNode;
- usdBalance?: React.ReactNode;
- rightElement?: React.ReactNode;
-}
-export function AssetRowGrid({
- title,
- balance,
- caption,
- usdBalance,
- rightElement,
-}: AssetRowGridProps) {
- const balanceItem = rightElement ? (
-
- {rightElement}
-
- ) : (
- {balance}
- );
- return (
-
-
- {title}
-
- {balanceItem}
- {caption}
- {usdBalance && {usdBalance}}
-
- );
-}
diff --git a/src/app/components/crypto-assets/crypto-currency-asset/crypto-currency-asset-item.layout.tsx b/src/app/components/crypto-assets/crypto-currency-asset/crypto-currency-asset-item.layout.tsx
index bf4a65215d5..d5ff281458e 100644
--- a/src/app/components/crypto-assets/crypto-currency-asset/crypto-currency-asset-item.layout.tsx
+++ b/src/app/components/crypto-assets/crypto-currency-asset/crypto-currency-asset-item.layout.tsx
@@ -1,103 +1,73 @@
-import { CryptoAssetSelectors } from '@tests/selectors/crypto-asset.selectors';
-import { Flex, styled } from 'leather-styles/jsx';
+import { ReactNode } from 'react';
-import { CryptoCurrencies } from '@shared/models/currencies.model';
-import { Money } from '@shared/models/money.model';
+import { AllCryptoCurrencyAssetBalances } from '@shared/models/crypto-asset-balance.model';
-import { formatBalance } from '@app/common/format-balance';
-import { ftDecimals } from '@app/common/stacks-utils';
-import { usePressable } from '@app/components/item-hover';
-import { Flag } from '@app/components/layout/flag';
+import { Item } from '@app/ui/components/items/item';
+import { ItemDefaultLayout } from '@app/ui/components/items/item-default.layout';
import { BasicTooltip } from '@app/ui/components/tooltip/basic-tooltip';
-import { truncateMiddle } from '@app/ui/utils/truncate-middle';
-import { AssetRowGrid } from '../components/asset-row-grid';
+import { parseCryptoCurrencyAssetBalance } from './crypto-currency-asset.utils';
interface CryptoCurrencyAssetItemLayoutProps {
- balance: Money;
- caption: string;
+ additionalBalanceInfo?: ReactNode;
+ additionalUsdBalanceInfo?: ReactNode;
+ address?: string;
+ assetBalance: AllCryptoCurrencyAssetBalances;
icon: React.ReactNode;
- copyIcon?: React.ReactNode;
isPressable?: boolean;
- title: string;
- usdBalance?: string;
- address?: string;
- isHovered?: boolean;
- currency?: CryptoCurrencies;
- additionalBalanceInfo?: React.ReactNode;
- additionalUsdBalanceInfo?: React.ReactNode;
- rightElement?: React.ReactNode;
onClick?(): void;
+ rightElement?: React.ReactNode;
+ usdBalance?: string;
}
export function CryptoCurrencyAssetItemLayout({
- balance,
- caption,
- icon,
- copyIcon,
- isPressable,
- title,
- usdBalance,
- address = '',
- isHovered = false,
additionalBalanceInfo,
additionalUsdBalanceInfo,
- rightElement,
+ address = '',
+ assetBalance,
+ icon,
+ isPressable,
onClick,
+ rightElement,
+ usdBalance,
}: CryptoCurrencyAssetItemLayoutProps) {
- const [component, bind] = usePressable(isPressable);
-
- const amount = balance.decimals
- ? ftDecimals(balance.amount, balance.decimals)
- : balance.amount.toString();
- const dataTestId = CryptoAssetSelectors.CryptoAssetListItem.replace(
- '{symbol}',
- balance.symbol.toLowerCase()
- );
- const formattedBalance = formatBalance(amount);
+ const { balance, dataTestId, formattedBalance, title } =
+ parseCryptoCurrencyAssetBalance(assetBalance);
return (
-
-
-
- {isHovered ? truncateMiddle(address, 6) : title}
-
- }
- balance={
-
-
- {formattedBalance.value} {additionalBalanceInfo}
-
-
- }
- caption={
-
- {caption}
-
+
+
+ {title}}
+ contentLeftBottom={{balance.symbol}}
+ contentRightTop={
+ rightElement ? (
+ rightElement
+ ) : (
+
+
+ {formattedBalance.value} {additionalBalanceInfo}
+
+
+ )
}
- usdBalance={
-
- {balance.amount.toNumber() > 0 && address ? (
-
- {usdBalance}
-
- ) : null}
- {additionalUsdBalanceInfo}
-
+ contentRightBottom={
+ !rightElement && (
+ <>
+ {balance.amount.toNumber() > 0 && address ? (
+ {usdBalance}
+ ) : null}
+ {additionalUsdBalanceInfo}
+ >
+ )
}
- rightElement={rightElement}
/>
-
- {component}
-
+
+
);
}
diff --git a/src/app/components/crypto-assets/crypto-currency-asset/crypto-currency-asset-item.tsx b/src/app/components/crypto-assets/crypto-currency-asset/crypto-currency-asset-item.tsx
deleted file mode 100644
index bddbae243ac..00000000000
--- a/src/app/components/crypto-assets/crypto-currency-asset/crypto-currency-asset-item.tsx
+++ /dev/null
@@ -1,46 +0,0 @@
-import type { AllCryptoCurrencyAssetBalances } from '@shared/models/crypto-asset-balance.model';
-
-import { spamFilter } from '@app/common/utils/spam-filter';
-
-import { CryptoCurrencyAssetItemLayout } from './crypto-currency-asset-item.layout';
-
-interface CryptoCurrencyAssetItemProps {
- assetBalance: AllCryptoCurrencyAssetBalances;
- icon: React.ReactNode;
- usdBalance?: string;
- address?: string;
- isPressable?: boolean;
- additionalBalanceInfo?: React.ReactNode;
- additionalUsdBalanceInfo?: React.ReactNode;
- rightElement?: React.ReactNode;
- onClick?(): void;
-}
-export function CryptoCurrencyAssetItem({
- additionalBalanceInfo,
- additionalUsdBalanceInfo,
- address,
- assetBalance,
- icon,
- isPressable,
- onClick,
- rightElement,
- usdBalance,
-}: CryptoCurrencyAssetItemProps) {
- const { balance, asset } = assetBalance;
-
- return (
-
- );
-}
diff --git a/src/app/components/crypto-assets/crypto-currency-asset/crypto-currency-asset.utils.ts b/src/app/components/crypto-assets/crypto-currency-asset/crypto-currency-asset.utils.ts
new file mode 100644
index 00000000000..041ca288ea5
--- /dev/null
+++ b/src/app/components/crypto-assets/crypto-currency-asset/crypto-currency-asset.utils.ts
@@ -0,0 +1,28 @@
+import { CryptoAssetSelectors } from '@tests/selectors/crypto-asset.selectors';
+
+import { AllCryptoCurrencyAssetBalances } from '@shared/models/crypto-asset-balance.model';
+
+import { formatBalance } from '@app/common/format-balance';
+import { ftDecimals } from '@app/common/stacks-utils';
+import { spamFilter } from '@app/common/utils/spam-filter';
+
+export function parseCryptoCurrencyAssetBalance(assetBalance: AllCryptoCurrencyAssetBalances) {
+ const { asset, balance } = assetBalance;
+
+ const amount = balance.decimals
+ ? ftDecimals(balance.amount, balance.decimals)
+ : balance.amount.toString();
+ const dataTestId = CryptoAssetSelectors.CryptoAssetListItem.replace(
+ '{symbol}',
+ balance.symbol.toLowerCase()
+ );
+ const formattedBalance = formatBalance(amount);
+ const title = spamFilter(asset.name);
+
+ return {
+ balance,
+ dataTestId,
+ formattedBalance,
+ title,
+ };
+}
diff --git a/src/app/components/crypto-assets/stacks/components/stacks-asset-avatar.tsx b/src/app/components/crypto-assets/stacks/components/stacks-asset-avatar.tsx
index 13fd5147231..2c728e417cb 100644
--- a/src/app/components/crypto-assets/stacks/components/stacks-asset-avatar.tsx
+++ b/src/app/components/crypto-assets/stacks/components/stacks-asset-avatar.tsx
@@ -18,7 +18,7 @@ export function StacksAssetAvatar({
imageCanonicalUri,
isStx,
isUnanchored,
- size = '36',
+ size = '40',
...props
}: StacksAssetAvatarProps) {
if (isStx) return ;
diff --git a/src/app/components/crypto-assets/stacks/fungible-token-asset/stacks-fungible-token-asset-item.tsx b/src/app/components/crypto-assets/stacks/fungible-token-asset/fungible-token-asset.utils.ts
similarity index 51%
rename from src/app/components/crypto-assets/stacks/fungible-token-asset/stacks-fungible-token-asset-item.tsx
rename to src/app/components/crypto-assets/stacks/fungible-token-asset/fungible-token-asset.utils.ts
index 3e62de0873e..8ca2943e28e 100644
--- a/src/app/components/crypto-assets/stacks/fungible-token-asset/stacks-fungible-token-asset-item.tsx
+++ b/src/app/components/crypto-assets/stacks/fungible-token-asset/fungible-token-asset.utils.ts
@@ -1,48 +1,41 @@
import { CryptoAssetSelectors } from '@tests/selectors/crypto-asset.selectors';
-import { FlexProps } from 'leather-styles/jsx';
-import type { StacksFungibleTokenAssetBalance } from '@shared/models/crypto-asset-balance.model';
-import { Money } from '@shared/models/money.model';
+import { StacksFungibleTokenAssetBalance } from '@shared/models/crypto-asset-balance.model';
import { getImageCanonicalUri } from '@app/common/crypto-assets/stacks-crypto-asset.utils';
+import { formatBalance } from '@app/common/format-balance';
+import { ftDecimals } from '@app/common/stacks-utils';
import { formatContractId, getTicker } from '@app/common/utils';
import { spamFilter } from '@app/common/utils/spam-filter';
import { getAssetName } from '@app/ui/utils/get-asset-name';
-import { StacksFungibleTokenAssetItemLayout } from './stacks-fungible-token-asset-item.layout';
-
-interface StacksFungibleTokenAssetItemProps extends FlexProps {
- assetBalance: StacksFungibleTokenAssetBalance;
- unanchoredAssetBalance?: Money;
- isPressable?: boolean;
- onClick?(): void;
-}
-export function StacksFungibleTokenAssetItem({
- assetBalance,
- isPressable,
- onClick,
-}: StacksFungibleTokenAssetItemProps) {
+export function parseStacksFungibleTokenAssetBalance(
+ assetBalance: StacksFungibleTokenAssetBalance
+) {
const { asset, balance } = assetBalance;
const { contractAddress, contractAssetName, contractName, name, symbol } = asset;
+ const amount = balance.decimals
+ ? ftDecimals(balance.amount, balance.decimals || 0)
+ : balance.amount.toString();
const avatar = `${formatContractId(contractAddress, contractName)}::${contractAssetName}`;
const dataTestId =
symbol && CryptoAssetSelectors.CryptoAssetListItem.replace('{symbol}', symbol.toLowerCase());
+ const formattedBalance = formatBalance(amount);
const friendlyName =
name ||
(contractAssetName.includes('::') ? getAssetName(contractAssetName) : contractAssetName);
const imageCanonicalUri = getImageCanonicalUri(asset.imageCanonicalUri, asset.name);
const caption = symbol || getTicker(friendlyName);
- return (
-
- );
+ const title = spamFilter(friendlyName);
+
+ return {
+ amount,
+ avatar,
+ caption,
+ dataTestId,
+ formattedBalance,
+ imageCanonicalUri,
+ title,
+ };
}
diff --git a/src/app/components/crypto-assets/stacks/fungible-token-asset/stacks-fungible-token-asset-item.layout.tsx b/src/app/components/crypto-assets/stacks/fungible-token-asset/stacks-fungible-token-asset-item.layout.tsx
index a259f6fbf0b..d47bb580151 100644
--- a/src/app/components/crypto-assets/stacks/fungible-token-asset/stacks-fungible-token-asset-item.layout.tsx
+++ b/src/app/components/crypto-assets/stacks/fungible-token-asset/stacks-fungible-token-asset-item.layout.tsx
@@ -1,71 +1,52 @@
-import { Flex, FlexProps, styled } from 'leather-styles/jsx';
+import { StacksFungibleTokenAssetBalance } from '@shared/models/crypto-asset-balance.model';
-import type { Money } from '@shared/models/money.model';
-
-import { formatBalance } from '@app/common/format-balance';
-import { ftDecimals } from '@app/common/stacks-utils';
import { StacksAssetAvatar } from '@app/components/crypto-assets/stacks/components/stacks-asset-avatar';
-import { usePressable } from '@app/components/item-hover';
-import { Flag } from '@app/components/layout/flag';
+import { Item } from '@app/ui/components/items/item';
+import { ItemDefaultLayout } from '@app/ui/components/items/item-default.layout';
import { BasicTooltip } from '@app/ui/components/tooltip/basic-tooltip';
-import { AssetCaption } from '../../components/asset-caption';
-import { AssetRowGrid } from '../../components/asset-row-grid';
+import { parseStacksFungibleTokenAssetBalance } from './fungible-token-asset.utils';
-interface StacksFungibleTokenAssetItemLayoutProps extends FlexProps {
- avatar: string;
- balance: Money;
- caption: string;
- imageCanonicalUri?: string;
+interface StacksFungibleTokenAssetItemLayoutProps {
+ assetBalance: StacksFungibleTokenAssetBalance;
isPressable?: boolean;
- title: string;
onClick?(): void;
}
export function StacksFungibleTokenAssetItemLayout({
- avatar,
- balance,
- caption,
- imageCanonicalUri,
+ assetBalance,
isPressable,
- title,
onClick,
}: StacksFungibleTokenAssetItemLayoutProps) {
- const [component, bind] = usePressable(isPressable);
-
- const amount = balance.decimals
- ? ftDecimals(balance.amount, balance.decimals || 0)
- : balance.amount.toString();
- const formattedBalance = formatBalance(amount);
+ const { amount, avatar, caption, dataTestId, formattedBalance, imageCanonicalUri, title } =
+ parseStacksFungibleTokenAssetBalance(assetBalance);
return (
-
-
- {title[0]}
-
- }
- spacing="space.04"
- width="100%"
- >
- {title}}
- balance={
-
-
- {formattedBalance.value}
-
+
+
+
+ {title[0]}
+
+ }
+ contentLeftTop={{title}}
+ contentLeftBottom={{caption}}
+ contentRightTop={
+
+ {formattedBalance.value}
}
- caption={}
/>
- {component}
-
-
+
+
);
}
diff --git a/src/app/components/stacks-transaction-item/increase-fee-button.tsx b/src/app/components/stacks-transaction-item/increase-fee-button.tsx
index 7cb383c8fef..cf924f4e988 100644
--- a/src/app/components/stacks-transaction-item/increase-fee-button.tsx
+++ b/src/app/components/stacks-transaction-item/increase-fee-button.tsx
@@ -4,13 +4,12 @@ import { ChevronsRightIcon } from '@app/ui/components/icons/chevrons-right-icon'
interface IncreaseFeeButtonProps {
isEnabled?: boolean;
- isHovered: boolean;
isSelected: boolean;
onIncreaseFee(): void;
}
export function IncreaseFeeButton(props: IncreaseFeeButtonProps) {
- const { isEnabled, isHovered, isSelected, onIncreaseFee } = props;
- const isActive = isEnabled && isHovered && !isSelected;
+ const { isEnabled, isSelected, onIncreaseFee } = props;
+ const isActive = isEnabled && !isSelected;
return (
@@ -99,16 +95,12 @@ export function StacksTransactionItem({
return (
}
txValue={txValue}
- belowCaptionEl={increaseFeeButton}
- {...bind}
- {...rest}
- >
- {component}
-
+ />
);
}
diff --git a/src/app/components/transaction-item/transaction-item.layout.tsx b/src/app/components/transaction-item/transaction-item.layout.tsx
index bb961b93108..4248f843e04 100644
--- a/src/app/components/transaction-item/transaction-item.layout.tsx
+++ b/src/app/components/transaction-item/transaction-item.layout.tsx
@@ -1,49 +1,43 @@
import { ReactNode } from 'react';
-import { Box, Flex, HStack } from 'leather-styles/jsx';
+import { Item } from '@app/ui/components/items/item';
+import { ItemDefaultLayout } from '@app/ui/components/items/item-default.layout';
interface TransactionItemLayoutProps {
openTxLink(): void;
+ rightElement?: ReactNode;
txCaption: ReactNode;
txTitle: ReactNode;
txValue: ReactNode;
txIcon?: ReactNode;
txStatus?: ReactNode;
- belowCaptionEl?: ReactNode;
children?: ReactNode;
}
export function TransactionItemLayout({
openTxLink,
+ rightElement,
txCaption,
txIcon,
txStatus,
txTitle,
txValue,
- belowCaptionEl,
- children,
- ...rest
}: TransactionItemLayoutProps) {
return (
-
-
- {txIcon && txIcon}
-
-
- {txTitle} {txValue}
-
-
- {txCaption} {txStatus && txStatus}
- {belowCaptionEl ? belowCaptionEl : null}
-
-
-
- {children}
-
+
+
+ {txTitle}}
+ contentLeftBottom={
+ <>
+ {txCaption}
+ {txStatus && txStatus}
+ >
+ }
+ contentRightTop={rightElement ? rightElement : {txValue}}
+ />
+
+
);
}
diff --git a/src/app/features/activity-list/components/pending-transaction-list/pending-transaction-list.layout.tsx b/src/app/features/activity-list/components/pending-transaction-list/pending-transaction-list.layout.tsx
index 3d2fb787563..44acfa0e297 100644
--- a/src/app/features/activity-list/components/pending-transaction-list/pending-transaction-list.layout.tsx
+++ b/src/app/features/activity-list/components/pending-transaction-list/pending-transaction-list.layout.tsx
@@ -11,7 +11,7 @@ export function PendingTransactionListLayout({ children }: PendingTransactionLis
Pending
-
+
{children}
>
diff --git a/src/app/features/activity-list/components/pending-transaction-list/pending-transaction-list.tsx b/src/app/features/activity-list/components/pending-transaction-list/pending-transaction-list.tsx
index 6fa22d355a1..2fe58970c23 100644
--- a/src/app/features/activity-list/components/pending-transaction-list/pending-transaction-list.tsx
+++ b/src/app/features/activity-list/components/pending-transaction-list/pending-transaction-list.tsx
@@ -2,9 +2,9 @@ import { MempoolTransaction } from '@stacks/stacks-blockchain-api-types';
import { BitcoinTx } from '@shared/models/transactions/bitcoin-transaction.model';
+import { BitcoinTransactionItem } from '@app/components/bitcoin-transaction-item/bitcoin-transaction-item';
import { StacksTransactionItem } from '@app/components/stacks-transaction-item/stacks-transaction-item';
-import { BitcoinTransactionItem } from '../../../../components/bitcoin-transaction-item/bitcoin-transaction-item';
import { PendingTransactionListLayout } from './pending-transaction-list.layout';
interface PendingTransactionListProps {
diff --git a/src/app/features/activity-list/components/transaction-list/transaction-list.layout.tsx b/src/app/features/activity-list/components/transaction-list/transaction-list.layout.tsx
index eed16bebc44..7d3f7c2e50a 100644
--- a/src/app/features/activity-list/components/transaction-list/transaction-list.layout.tsx
+++ b/src/app/features/activity-list/components/transaction-list/transaction-list.layout.tsx
@@ -6,9 +6,5 @@ interface TransactionListLayoutProps {
children: ReactNode;
}
export function TransactionListLayout({ children }: TransactionListLayoutProps) {
- return (
-
- {children}
-
- );
+ return {children};
}
diff --git a/src/app/features/activity-list/components/transaction-list/transactions-by-date.layout.tsx b/src/app/features/activity-list/components/transaction-list/transactions-by-date.layout.tsx
index 117d795d447..f9b3bde1d8d 100644
--- a/src/app/features/activity-list/components/transaction-list/transactions-by-date.layout.tsx
+++ b/src/app/features/activity-list/components/transaction-list/transactions-by-date.layout.tsx
@@ -17,9 +17,7 @@ export function TransactionsByDateLayout({
{displayDate}
-
- {children}
-
+ {children}
);
}
diff --git a/src/app/features/asset-list/asset-list.tsx b/src/app/features/asset-list/asset-list.tsx
index d6703e2c534..096f86f20b0 100644
--- a/src/app/features/asset-list/asset-list.tsx
+++ b/src/app/features/asset-list/asset-list.tsx
@@ -7,7 +7,7 @@ import { useBtcAssetBalance } from '@app/common/hooks/balance/btc/use-btc-balanc
import { useWalletType } from '@app/common/use-wallet-type';
import { BitcoinContractEntryPoint } from '@app/components/bitcoin-contract-entry-point/bitcoin-contract-entry-point';
import { Brc20TokensLoader } from '@app/components/brc20-tokens-loader';
-import { CryptoCurrencyAssetItem } from '@app/components/crypto-assets/crypto-currency-asset/crypto-currency-asset-item';
+import { CryptoCurrencyAssetItemLayout } from '@app/components/crypto-assets/crypto-currency-asset/crypto-currency-asset-item.layout';
import { CurrentStacksAccountLoader } from '@app/components/loaders/stacks-account-loader';
import { useHasBitcoinLedgerKeychain } from '@app/store/accounts/blockchain/bitcoin/bitcoin.ledger';
import { useCurrentAccountNativeSegwitAddressIndexZero } from '@app/store/accounts/blockchain/bitcoin/native-segwit-account.hooks';
@@ -31,10 +31,10 @@ export function AssetsList() {
const { whenWallet } = useWalletType();
return (
-
+
{whenWallet({
software: (
- }
@@ -42,7 +42,7 @@ export function AssetsList() {
/>
),
ledger: (
- }
diff --git a/src/app/features/asset-list/components/add-stacks-ledger-keys-item.tsx b/src/app/features/asset-list/components/add-stacks-ledger-keys-item.tsx
index a09c669a03f..ff0f0ed281a 100644
--- a/src/app/features/asset-list/components/add-stacks-ledger-keys-item.tsx
+++ b/src/app/features/asset-list/components/add-stacks-ledger-keys-item.tsx
@@ -1,6 +1,6 @@
import BigNumber from 'bignumber.js';
-import { CryptoCurrencyAssetItem } from '@app/components/crypto-assets/crypto-currency-asset/crypto-currency-asset-item';
+import { CryptoCurrencyAssetItemLayout } from '@app/components/crypto-assets/crypto-currency-asset/crypto-currency-asset-item.layout';
import { StxAvatar } from '@app/components/crypto-assets/stacks/components/stx-avatar';
import { createStacksCryptoCurrencyAssetTypeWrapper } from '@app/query/stacks/balance/stacks-ft-balances.utils';
@@ -8,7 +8,7 @@ import { ConnectLedgerAssetBtn } from './connect-ledger-asset-button';
export function AddStacksLedgerKeysItem() {
return (
- }
rightElement={}
diff --git a/src/app/features/asset-list/components/bitcoin-fungible-tokens-asset-list.tsx b/src/app/features/asset-list/components/bitcoin-fungible-tokens-asset-list.tsx
index c8ce70ef4b9..e71915b3ee3 100644
--- a/src/app/features/asset-list/components/bitcoin-fungible-tokens-asset-list.tsx
+++ b/src/app/features/asset-list/components/bitcoin-fungible-tokens-asset-list.tsx
@@ -1,6 +1,6 @@
import { Stack } from 'leather-styles/jsx';
-import { Brc20TokenAssetItem } from '@app/components/crypto-assets/bitcoin/brc20-token-asset-list/components/brc20-token-asset-item';
+import { Brc20TokenAssetItemLayout } from '@app/components/crypto-assets/bitcoin/brc20-token-asset-list/components/brc20-token-asset-item.layout';
import { Brc20Token } from '@app/query/bitcoin/ordinals/brc20/brc20-tokens.query';
interface BitcoinFungibleTokenAssetListProps {
@@ -12,7 +12,7 @@ export function BitcoinFungibleTokenAssetList({ brc20Tokens }: BitcoinFungibleTo
return (
{brc20Tokens.map(token => (
-
+
))}
);
diff --git a/src/app/features/asset-list/components/stacks-asset-list.tsx b/src/app/features/asset-list/components/stacks-asset-list.tsx
index 40207ba6177..d4a64017141 100644
--- a/src/app/features/asset-list/components/stacks-asset-list.tsx
+++ b/src/app/features/asset-list/components/stacks-asset-list.tsx
@@ -2,7 +2,7 @@ import { styled } from 'leather-styles/jsx';
import { useStxBalance } from '@app/common/hooks/balance/stx/use-stx-balance';
import { ftDecimals } from '@app/common/stacks-utils';
-import { CryptoCurrencyAssetItem } from '@app/components/crypto-assets/crypto-currency-asset/crypto-currency-asset-item';
+import { CryptoCurrencyAssetItemLayout } from '@app/components/crypto-assets/crypto-currency-asset/crypto-currency-asset-item.layout';
import { StxAvatar } from '@app/components/crypto-assets/stacks/components/stx-avatar';
import { useStacksFungibleTokenAssetBalancesAnchoredWithMetadata } from '@app/query/stacks/balance/stacks-ft-balances.hooks';
import { StacksAccount } from '@app/store/accounts/blockchain/stacks/stacks-account.models';
@@ -38,7 +38,7 @@ export function StacksAssetList({ account }: StacksAssetListProps) {
return (
<>
-
+
{assetBalances.map(assetBalance => (
-
diff --git a/src/app/features/collectibles/components/collectibes.layout.tsx b/src/app/features/collectibles/components/collectibes.layout.tsx
index ccdae07f19e..091fa5196c5 100644
--- a/src/app/features/collectibles/components/collectibes.layout.tsx
+++ b/src/app/features/collectibles/components/collectibes.layout.tsx
@@ -23,7 +23,7 @@ export function CollectiblesLayout({
}: CollectiblesLayoutProps) {
return (
<>
-
+
{title}
{isLoading ? (
diff --git a/src/app/features/increase-fee-drawer/components/increase-btc-fee-form.tsx b/src/app/features/increase-fee-drawer/components/increase-btc-fee-form.tsx
index c9520a69f7b..bb6bff62f2a 100644
--- a/src/app/features/increase-fee-drawer/components/increase-btc-fee-form.tsx
+++ b/src/app/features/increase-fee-drawer/components/increase-btc-fee-form.tsx
@@ -50,7 +50,7 @@ export function IncreaseBtcFeeForm({ btcTx }: IncreaseBtcFeeFormProps) {
validationSchema={validationSchema}
>
- {btcTx && }
+ {btcTx && }
diff --git a/src/app/features/increase-fee-drawer/components/increase-stx-fee-form.tsx b/src/app/features/increase-fee-drawer/components/increase-stx-fee-form.tsx
index e7620a13e18..010d495168a 100644
--- a/src/app/features/increase-fee-drawer/components/increase-stx-fee-form.tsx
+++ b/src/app/features/increase-fee-drawer/components/increase-stx-fee-form.tsx
@@ -93,7 +93,7 @@ export function IncreaseStxFeeForm() {
>
{props => (
- {tx && }
+ {tx && }
{balances?.stx.unlockedStx.amount && (
diff --git a/src/app/features/leather-intro-dialog/leather-intro-steps.tsx b/src/app/features/leather-intro-dialog/leather-intro-steps.tsx
index 29b62738bfc..48d8f80131a 100644
--- a/src/app/features/leather-intro-dialog/leather-intro-steps.tsx
+++ b/src/app/features/leather-intro-dialog/leather-intro-steps.tsx
@@ -16,7 +16,7 @@ export function LeatherIntroDialog({ children }: HasChildren) {
return (
e.preventDefault()}
onInteractOutside={e => e.preventDefault()}
className={css({ maxWidth: '500px', backgroundColor: 'accent.background-primary' })}
diff --git a/src/app/features/switch-account-drawer/components/switch-account-list-item.tsx b/src/app/features/switch-account-drawer/components/switch-account-list-item.tsx
index 761823e7bc1..d5e77122983 100644
--- a/src/app/features/switch-account-drawer/components/switch-account-list-item.tsx
+++ b/src/app/features/switch-account-drawer/components/switch-account-list-item.tsx
@@ -5,7 +5,6 @@ import { useSwitchAccount } from '@app/common/hooks/account/use-switch-account';
import { useLoading } from '@app/common/hooks/use-loading';
import { AccountTotalBalance } from '@app/components/account-total-balance';
import { AccountListItemLayout } from '@app/components/account/account-list-item-layout';
-import { usePressable } from '@app/components/item-hover';
import { useNativeSegwitSigner } from '@app/store/accounts/blockchain/bitcoin/native-segwit-account.hooks';
import { useStacksAccounts } from '@app/store/accounts/blockchain/stacks/stacks-account.hooks';
@@ -28,7 +27,6 @@ export const SwitchAccountListItem = memo(
'SWITCH_ACCOUNTS' + stacksAddress || bitcoinAddress
);
const { handleSwitchAccount } = useSwitchAccount(handleClose);
- const [component, bind] = usePressable(true);
const name = useAccountDisplayName({ address: stacksAddress, index });
const handleClick = async () => {
@@ -41,9 +39,7 @@ export const SwitchAccountListItem = memo(
return (
{name}}
avatar={
}
- onSelectAccount={handleClick}
- accountName={{name}}
balanceLabel={
}
- mt="space.05"
- {...bind}
- >
- {component}
-
+ index={index}
+ isLoading={isLoading}
+ isPressable
+ isSelected={currentAccountIndex === index}
+ onSelectAccount={handleClick}
+ />
);
}
);
diff --git a/src/app/features/switch-account-drawer/components/switch-account-list.tsx b/src/app/features/switch-account-drawer/components/switch-account-list.tsx
index 809dccd3804..0cafce7570f 100644
--- a/src/app/features/switch-account-drawer/components/switch-account-list.tsx
+++ b/src/app/features/switch-account-drawer/components/switch-account-list.tsx
@@ -23,7 +23,7 @@ export const SwitchAccountList = memo(
style={{ paddingTop: '24px', height: '70vh' }}
totalCount={addressesNum}
itemContent={index => (
-
+
{
- const [component, bind] = usePressable(true);
const name = useAccountDisplayName(account);
const btcAddress = useNativeSegwitAccountIndexAddressIndexZero(account.index);
const accountSlug = useMemo(() => slugify(`Account ${account?.index + 1}`), [account?.index]);
return (
- // Padding required on outer element to prevent jumpy list behaviours in
- // virtualised list library
+ // Padding required on outer element to prevent jumpy virtualized list
}>
{name}
@@ -65,13 +61,12 @@ const ChooseAccountItem = memo(
balanceLabel={
}
+ data-testid={`account-${accountSlug}-${account.index}`}
+ index={account.index}
isLoading={isLoading}
+ isSelected={false}
onSelectAccount={() => onSelectAccount(account.index)}
- data-testid={`account-${accountSlug}-${account.index}`}
- {...bind}
- >
- {component}
-
+ />
);
}
diff --git a/src/app/pages/receive/components/receive-item.tsx b/src/app/pages/receive/components/receive-item.tsx
index 95d97cec82c..a6d7e11f078 100644
--- a/src/app/pages/receive/components/receive-item.tsx
+++ b/src/app/pages/receive/components/receive-item.tsx
@@ -1,9 +1,10 @@
-import { Flex, HStack, Stack, styled } from 'leather-styles/jsx';
+import { HStack } from 'leather-styles/jsx';
-import { Flag } from '@app/components/layout/flag';
import { Button } from '@app/ui/components/button/button';
import { CopyIcon } from '@app/ui/components/icons/copy-icon';
import { QrCodeIcon } from '@app/ui/components/icons/qr-code-icon';
+import { Item } from '@app/ui/components/items/item';
+import { ItemWithButtonsLayout } from '@app/ui/components/items/item-with-buttons.layout';
import { truncateMiddle } from '@app/ui/utils/truncate-middle';
interface ReceiveItemProps {
@@ -24,23 +25,33 @@ export function ReceiveItem({
}: ReceiveItemProps) {
if (!address) return null;
return (
-
-
-
- {title}
- {truncateMiddle(address, 6)}
-
-
-
- {onClickQrCode && (
-
- )}
-
-
-
+
+
+ {title}}
+ contentLeftBottom={
+ {truncateMiddle(address, 6)}
+ }
+ contentRightTop={
+
+
+ {onClickQrCode && (
+
+ )}
+
+ }
+ />
+
+
);
}
diff --git a/src/app/pages/receive/components/receive-items.tsx b/src/app/pages/receive/components/receive-items.tsx
index d1c2d674ae9..4f89ecee966 100644
--- a/src/app/pages/receive/components/receive-items.tsx
+++ b/src/app/pages/receive/components/receive-items.tsx
@@ -13,7 +13,7 @@ export function ReceiveItemList({ children, title }: ReceiveItemListProps) {
)}
-
+
{children}
>
diff --git a/src/app/pages/send/send-crypto-asset-form/components/recipient-accounts-drawer/account-list-item.tsx b/src/app/pages/send/send-crypto-asset-form/components/recipient-accounts-drawer/account-list-item.tsx
index 971e0aa5bb5..c55652b6522 100644
--- a/src/app/pages/send/send-crypto-asset-form/components/recipient-accounts-drawer/account-list-item.tsx
+++ b/src/app/pages/send/send-crypto-asset-form/components/recipient-accounts-drawer/account-list-item.tsx
@@ -9,7 +9,6 @@ import { AccountTotalBalance } from '@app/components/account-total-balance';
import { AccountAvatarItem } from '@app/components/account/account-avatar-item';
import { AccountListItemLayout } from '@app/components/account/account-list-item-layout';
import { AccountNameLayout } from '@app/components/account/account-name';
-import { usePressable } from '@app/components/item-hover';
import { useNativeSegwitSigner } from '@app/store/accounts/blockchain/bitcoin/native-segwit-account.hooks';
import { StacksAccount } from '@app/store/accounts/blockchain/stacks/stacks-account.models';
@@ -22,7 +21,6 @@ export const AccountListItem = memo(({ index, stacksAccount, onClose }: AccountL
const { setFieldValue, values, setFieldTouched } = useFormikContext<
BitcoinSendFormValues | StacksSendFormValues
>();
- const [component, bind] = usePressable(true);
const stacksAddress = stacksAccount?.address || '';
const name = useAccountDisplayName({ address: stacksAddress, index });
@@ -48,13 +46,10 @@ export const AccountListItem = memo(({ index, stacksAccount, onClose }: AccountL
}
balanceLabel={}
index={index}
- isActive={false}
+ isSelected={false}
+ isPressable
isLoading={false}
- mt="space.05"
onSelectAccount={onSelectAccount}
- {...bind}
- >
- {component}
-
+ />
);
});
diff --git a/src/app/pages/swap/swap-choose-asset/components/swap-asset-item.layout.tsx b/src/app/pages/swap/swap-choose-asset/components/swap-asset-item.layout.tsx
deleted file mode 100644
index 6f85aeafa2c..00000000000
--- a/src/app/pages/swap/swap-choose-asset/components/swap-asset-item.layout.tsx
+++ /dev/null
@@ -1,16 +0,0 @@
-import { Stack } from 'leather-styles/jsx';
-
-import { HasChildren } from '@app/common/has-children';
-import { Flag } from '@app/components/layout/flag';
-
-export function SwapAssetItemLayout({
- children,
- icon,
- ...rest
-}: HasChildren & { icon: React.JSX.Element }) {
- return (
-
- {children}
-
- );
-}
diff --git a/src/app/pages/swap/swap-choose-asset/components/swap-asset-item.tsx b/src/app/pages/swap/swap-choose-asset/components/swap-asset-item.tsx
index 06b608c7194..2ef1aecb9df 100644
--- a/src/app/pages/swap/swap-choose-asset/components/swap-asset-item.tsx
+++ b/src/app/pages/swap/swap-choose-asset/components/swap-asset-item.tsx
@@ -1,19 +1,20 @@
-import { HStack, styled } from 'leather-styles/jsx';
+import { styled } from 'leather-styles/jsx';
import { formatMoneyWithoutSymbol } from '@app/common/money/format-money';
-import { usePressable } from '@app/components/item-hover';
import { useGetFungibleTokenMetadataQuery } from '@app/query/stacks/tokens/fungible-tokens/fungible-token-metadata.query';
import { isFtAsset } from '@app/query/stacks/tokens/token-metadata.utils';
+import { Item } from '@app/ui/components/items/item';
+import { ItemDefaultLayout } from '@app/ui/components/items/item-default.layout';
import { useAlexSdkBalanceAsFiat } from '../../hooks/use-alex-sdk-fiat-price';
import { SwapAsset } from '../../hooks/use-swap-form';
-import { SwapAssetItemLayout } from './swap-asset-item.layout';
interface SwapAssetItemProps {
asset: SwapAsset;
+ dataTestId: string;
+ onClick(): void;
}
-export function SwapAssetItem({ asset }: SwapAssetItemProps) {
- const [component, bind] = usePressable(true);
+export function SwapAssetItem({ asset, dataTestId, onClick }: SwapAssetItemProps) {
const balanceAsFiat = useAlexSdkBalanceAsFiat(asset.balance, asset.price);
const { data: ftMetadata } = useGetFungibleTokenMetadataQuery(asset.principal);
@@ -21,21 +22,17 @@ export function SwapAssetItem({ asset }: SwapAssetItemProps) {
const displayName = asset.displayName ?? ftMetadataName;
return (
- }
- {...bind}
- >
-
- {displayName}
- {formatMoneyWithoutSymbol(asset.balance)}
-
-
- {asset.name}
-
- {balanceAsFiat}
-
-
- {component}
-
+
+
+ }
+ contentLeftTop={{displayName}}
+ contentLeftBottom={{asset.name}}
+ contentRightTop={{formatMoneyWithoutSymbol(asset.balance)}}
+ contentRightBottom={{balanceAsFiat}}
+ />
+
+
);
}
diff --git a/src/app/pages/swap/swap-choose-asset/components/swap-asset-list.layout.tsx b/src/app/pages/swap/swap-choose-asset/components/swap-asset-list.layout.tsx
index e7feee61700..eb40bfdcbf4 100644
--- a/src/app/pages/swap/swap-choose-asset/components/swap-asset-list.layout.tsx
+++ b/src/app/pages/swap/swap-choose-asset/components/swap-asset-list.layout.tsx
@@ -2,7 +2,7 @@ import { Stack, StackProps } from 'leather-styles/jsx';
export function SwapAssetListLayout({ children }: StackProps) {
return (
-
+
{children}
);
diff --git a/src/app/pages/swap/swap-choose-asset/components/swap-asset-list.tsx b/src/app/pages/swap/swap-choose-asset/components/swap-asset-list.tsx
index bc2875175ff..5419a1c2629 100644
--- a/src/app/pages/swap/swap-choose-asset/components/swap-asset-list.tsx
+++ b/src/app/pages/swap/swap-choose-asset/components/swap-asset-list.tsx
@@ -3,7 +3,6 @@ import { useNavigate } from 'react-router-dom';
import { SwapSelectors } from '@tests/selectors/swap.selectors';
import BigNumber from 'bignumber.js';
import { useFormikContext } from 'formik';
-import { styled } from 'leather-styles/jsx';
import { createMoney } from '@shared/models/money.model';
import { isUndefined } from '@shared/utils';
@@ -68,15 +67,12 @@ export function SwapAssetList({ assets }: SwapAssetList) {
return (
{selectableAssets.map(asset => (
- onChooseAsset(asset)}
- textAlign="left"
- type="button"
- >
-
-
+ />
))}
);
diff --git a/src/app/ui/components/button/button.stories.tsx b/src/app/ui/components/button/button.stories.tsx
index c0d2277c292..796d2a6a9a0 100644
--- a/src/app/ui/components/button/button.stories.tsx
+++ b/src/app/ui/components/button/button.stories.tsx
@@ -26,6 +26,18 @@ export const Button: Story = {
},
};
+export const Disabled: Story = {
+ parameters: {
+ controls: { include: ['size', 'variant'] },
+ },
+ args: {
+ disabled: true,
+ children: 'Button',
+ size: 'md',
+ variant: 'solid',
+ },
+};
+
// TODO: Remove invert code
export const InvertSolid: Story = {
parameters: {
diff --git a/src/app/ui/components/dynamic-color-circle.tsx b/src/app/ui/components/dynamic-color-circle.tsx
index e0d62e71db4..2b4d0f0e495 100644
--- a/src/app/ui/components/dynamic-color-circle.tsx
+++ b/src/app/ui/components/dynamic-color-circle.tsx
@@ -6,7 +6,7 @@ interface DynamicColorCircleProps extends CircleProps {
}
export function DynamicColorCircle({
children,
- sizeParam = '36',
+ sizeParam = '40',
value,
...props
}: DynamicColorCircleProps) {
diff --git a/src/app/ui/components/items/item-default.layout.tsx b/src/app/ui/components/items/item-default.layout.tsx
new file mode 100644
index 00000000000..ad9b7b0e7f5
--- /dev/null
+++ b/src/app/ui/components/items/item-default.layout.tsx
@@ -0,0 +1,59 @@
+import { ReactNode } from 'react';
+
+import { Grid, GridItem } from 'leather-styles/jsx';
+
+import { ChevronUpIcon } from '../icons/chevron-up-icon';
+
+interface ItemDefaultLayoutProps {
+ contentLeftTop: ReactNode;
+ contentLeftBottom?: ReactNode;
+ contentRightTop: ReactNode;
+ contentRightBottom?: ReactNode;
+ flagImg?: ReactNode;
+ isDisabled?: boolean;
+ isPressable?: boolean;
+}
+export function ItemDefaultLayout({
+ contentLeftTop,
+ contentLeftBottom,
+ contentRightTop,
+ contentRightBottom,
+ flagImg,
+ isDisabled,
+ isPressable,
+}: ItemDefaultLayoutProps) {
+ return (
+
+
+ {flagImg}
+
+
+ {contentLeftTop}
+
+
+ {contentRightTop}
+
+
+ {isPressable && (
+
+ )}
+
+ {/* Using scroll here for now bc possibly shows addresses */}
+
+ {contentLeftBottom}
+
+ {contentRightBottom && {contentRightBottom}}
+
+ );
+}
diff --git a/src/app/ui/components/items/item-with-buttons.layout.tsx b/src/app/ui/components/items/item-with-buttons.layout.tsx
new file mode 100644
index 00000000000..7900957c59e
--- /dev/null
+++ b/src/app/ui/components/items/item-with-buttons.layout.tsx
@@ -0,0 +1,45 @@
+import { ReactNode } from 'react';
+
+import { Grid, GridItem } from 'leather-styles/jsx';
+
+interface ItemWithButtonsLayoutProps {
+ contentLeftTop: ReactNode;
+ contentLeftBottom?: ReactNode;
+ contentRightTop: ReactNode;
+ contentRightBottom?: ReactNode;
+ flagImg?: ReactNode;
+ isDisabled?: boolean;
+ isPressable?: boolean;
+}
+export function ItemWithButtonsLayout({
+ contentLeftTop,
+ contentLeftBottom,
+ contentRightTop,
+ flagImg,
+}: ItemWithButtonsLayoutProps) {
+ return (
+
+
+ {flagImg}
+
+
+ {contentLeftTop}
+
+
+ {contentRightTop}
+
+ {/* Using scroll here for now bc possibly shows addresses */}
+
+ {contentLeftBottom}
+
+
+ );
+}
diff --git a/src/app/ui/components/items/item.stories.tsx b/src/app/ui/components/items/item.stories.tsx
new file mode 100644
index 00000000000..f739c1ce5b9
--- /dev/null
+++ b/src/app/ui/components/items/item.stories.tsx
@@ -0,0 +1,147 @@
+import type { Meta, StoryObj } from '@storybook/react';
+import { HStack } from 'leather-styles/jsx';
+
+import { Button } from '../button/button';
+import { BtcIcon } from '../icons/btc-icon';
+import { CheckmarkIcon } from '../icons/checkmark-icon';
+import { CopyIcon } from '../icons/copy-icon';
+import { QrCodeIcon } from '../icons/qr-code-icon';
+import { Item } from './item';
+import { ItemDefaultLayout } from './item-default.layout';
+import { ItemWithButtonsLayout } from './item-with-buttons.layout';
+
+const meta: Meta = {
+ component: Item.Root,
+ tags: ['autodocs'],
+ title: 'Item',
+};
+
+export default meta;
+type Story = StoryObj;
+
+export const Default: Story = {
+ render: () => (
+ // eslint-disable-next-line no-console
+
+
+ }
+ contentLeftTop={Label}
+ contentLeftBottom={Caption}
+ contentRightTop={1,000.00 ABC}
+ contentRightBottom={$1,000.00}
+ />
+
+
+ ),
+};
+
+export const DefaultPressable: Story = {
+ render: () => (
+ // eslint-disable-next-line no-console
+ console.log(`I'm interactive`)}>
+
+ }
+ contentLeftTop={Label}
+ contentLeftBottom={Caption}
+ contentRightTop={1,000.00 ABC}
+ contentRightBottom={$1,000.00}
+ />
+
+
+ ),
+};
+
+export const Selected: Story = {
+ render: () => (
+ // eslint-disable-next-line no-console
+ console.log(`I'm interactive`)}>
+
+ }
+ contentLeftTop={
+
+ Label
+
+
+ }
+ contentLeftBottom={Caption}
+ contentRightTop={1,000.00 ABC}
+ contentRightBottom={$1,000.00}
+ />
+
+
+ ),
+};
+
+export const Disabled: Story = {
+ render: () => (
+ // eslint-disable-next-line no-console
+ console.log(`I'm interactive`)}>
+
+ }
+ contentLeftTop={Label}
+ contentLeftBottom={Caption}
+ contentRightTop={1,000.00 ABC}
+ contentRightBottom={$1,000.00}
+ />
+
+
+ ),
+};
+
+export const WithButtons: Story = {
+ render: () => (
+ // eslint-disable-next-line no-console
+
+
+ }
+ contentLeftTop={Label}
+ contentLeftBottom={Caption}
+ contentRightTop={
+
+
+
+
+ }
+ />
+
+
+ ),
+};
+
+export const WithButtonsDisabled: Story = {
+ render: () => (
+ // eslint-disable-next-line no-console
+
+
+ }
+ contentLeftTop={Label}
+ contentLeftBottom={Caption}
+ contentRightTop={
+
+
+
+
+ }
+ />
+
+
+ ),
+};
diff --git a/src/app/ui/components/items/item.tsx b/src/app/ui/components/items/item.tsx
new file mode 100644
index 00000000000..eec33764e1e
--- /dev/null
+++ b/src/app/ui/components/items/item.tsx
@@ -0,0 +1,137 @@
+import { type ComponentProps, KeyboardEvent } from 'react';
+
+import { sva } from 'leather-styles/css';
+import { BoxProps, styled } from 'leather-styles/jsx';
+
+import { createStyleContext } from '@app/ui/utils/style-context';
+
+const item = sva({
+ slots: ['root', 'content', 'text', 'caption'],
+ base: {
+ root: {
+ bg: 'accent.background-primary',
+ height: 'auto',
+ userSelect: 'none',
+ p: 'space.03',
+ width: '100%',
+ },
+ content: {
+ width: '100%',
+ },
+ text: {
+ color: 'accent.text-primary',
+ fontWeight: 500,
+ textStyle: 'label.02',
+ },
+ caption: {
+ color: 'accent.text-subdued',
+ },
+ },
+ variants: {
+ isDisabled: {
+ true: {
+ root: {
+ _active: { bg: 'unset' },
+ _focus: { border: 'unset' },
+ _hover: { bg: 'unset' },
+ cursor: 'not-allowed',
+ },
+ content: {},
+ text: {
+ color: 'accent.non-interactive',
+ },
+ caption: {
+ color: 'accent.non-interactive',
+ },
+ },
+ },
+ isPressable: {
+ true: {
+ root: {
+ _active: {
+ bg: 'accent.component-background-pressed',
+ },
+ _focus: {
+ border: '2px solid {focus}',
+ },
+ _hover: {
+ bg: 'accent.component-background-hover',
+ },
+ },
+ },
+ },
+ isSelected: {
+ true: {
+ root: {
+ _active: {
+ bg: 'unset',
+ },
+ _focus: {
+ border: 'unset',
+ },
+ _hover: {
+ bg: 'unset',
+ },
+ },
+ },
+ },
+ },
+ compoundVariants: [
+ {
+ isDisabled: true,
+ isPressable: true,
+ css: {
+ root: {
+ _active: { bg: 'unset' },
+ _focus: { border: 'unset' },
+ _hover: { bg: 'unset' },
+ cursor: 'not-allowed',
+ },
+ },
+ },
+ ],
+});
+
+const { withProvider, withContext } = createStyleContext(item);
+
+const RootBase = withProvider('div', 'root');
+
+interface RootProps extends ComponentProps<'div'> {
+ isDisabled?: boolean;
+ isPressable?: boolean;
+ isSelected?: boolean;
+ onClick?(): void;
+}
+function Root({ isDisabled, isPressable, isSelected, onClick, ...props }: RootProps & BoxProps) {
+ const isRoleButton = !!onClick;
+
+ function onKeyDown(e: KeyboardEvent) {
+ if (!isRoleButton) return;
+ if (e.key === 'enter' || e.key === ' ') {
+ e.preventDefault();
+ onClick();
+ }
+ }
+
+ return (
+
+ );
+}
+
+const Content = withContext(styled('div'), 'content');
+
+const Text = withContext(styled('span'), 'text');
+
+const Caption = withContext(styled('span'), 'caption');
+
+export const Item = { Root, Content, Text, Caption };
diff --git a/src/app/ui/components/link/link.stories.tsx b/src/app/ui/components/link/link.stories.tsx
index b92fd85c7b8..8048b8a5091 100644
--- a/src/app/ui/components/link/link.stories.tsx
+++ b/src/app/ui/components/link/link.stories.tsx
@@ -22,6 +22,18 @@ export const Link: Story = {
},
};
+export const Disabled: Story = {
+ parameters: {
+ controls: { include: ['size', 'variant'] },
+ },
+ args: {
+ children: 'Link',
+ disabled: true,
+ size: 'md',
+ variant: 'underlined',
+ },
+};
+
// TODO: Remove invert code
export const InvertLink: Story = {
parameters: {
diff --git a/src/app/ui/components/link/link.tsx b/src/app/ui/components/link/link.tsx
index ff714540708..6890f139d90 100644
--- a/src/app/ui/components/link/link.tsx
+++ b/src/app/ui/components/link/link.tsx
@@ -9,12 +9,12 @@ type LinkProps = Omit, keyof LinkVariant
LinkVariantProps;
export const Link = forwardRef((props: LinkProps, ref: ForwardedRef) => {
- const { children, fullWidth, invert, size, variant, ...rest } = props;
+ const { children, disabled, fullWidth, invert, size, variant, ...rest } = props;
return (
diff --git a/src/app/ui/components/tooltip/basic-tooltip.tsx b/src/app/ui/components/tooltip/basic-tooltip.tsx
index e10e2d61616..56fcead792f 100644
--- a/src/app/ui/components/tooltip/basic-tooltip.tsx
+++ b/src/app/ui/components/tooltip/basic-tooltip.tsx
@@ -11,7 +11,6 @@ interface BasicTooltipProps {
side?: RadixTooltip.TooltipContentProps['side'];
asChild?: boolean;
}
-
export function BasicTooltip({ children, label, disabled, side, asChild }: BasicTooltipProps) {
const isDisabled = !label || disabled;
return (
diff --git a/theme/recipes/button-recipe.ts b/theme/recipes/button-recipe.ts
index 5e26e867360..36da0ac019b 100644
--- a/theme/recipes/button-recipe.ts
+++ b/theme/recipes/button-recipe.ts
@@ -30,9 +30,6 @@ export const buttonRecipe = defineRecipe({
className: 'button',
jsx: ['Button'],
base: {
- _disabled: {
- cursor: 'not-allowed',
- },
position: 'relative',
rounded: 'xs',
textStyle: 'label.02',
@@ -54,8 +51,10 @@ export const buttonRecipe = defineRecipe({
bg: 'accent.action-primary-default',
},
_disabled: {
+ _hover: { bg: 'accent.background-secondary' },
bg: 'accent.background-secondary',
color: 'accent.non-interactive',
+ cursor: 'not-allowed',
},
_focus: {
_before: {
@@ -74,6 +73,12 @@ export const buttonRecipe = defineRecipe({
_active: {
bg: 'accent.component-background-pressed',
},
+ _disabled: {
+ _hover: { bg: 'unset' },
+ border: '1px solid {colors.accent.non-interactive}',
+ color: 'accent.non-interactive',
+ cursor: 'not-allowed',
+ },
_focus: {
_before: {
border: '3px solid {colors.focus}',
@@ -90,6 +95,11 @@ export const buttonRecipe = defineRecipe({
_active: {
bg: 'accent.component-background-pressed',
},
+ _disabled: {
+ _hover: { bg: 'unset' },
+ color: 'accent.non-interactive',
+ cursor: 'not-allowed',
+ },
_focus: {
_before: {
border: '3px solid {focus}',
diff --git a/theme/recipes/link-recipe.ts b/theme/recipes/link-recipe.ts
index 88ba089d678..f64aba9716a 100644
--- a/theme/recipes/link-recipe.ts
+++ b/theme/recipes/link-recipe.ts
@@ -6,11 +6,7 @@ export const linkRecipe = defineRecipe({
className: 'link',
jsx: ['Link'],
base: {
- _disabled: {
- cursor: 'not-allowed',
- },
appearance: 'none',
- color: 'accent.text-primary',
display: 'inline',
mb: 'space.01',
p: 'unset',
@@ -48,12 +44,6 @@ export const linkRecipe = defineRecipe({
},
color: 'accent.text-primary',
},
- _disabled: {
- _before: {
- background: 'accent.non-interactive',
- },
- color: 'accent.non-interactive',
- },
_focus: {
_before: { background: 'focus' },
color: 'accent.text-primary',
@@ -64,6 +54,7 @@ export const linkRecipe = defineRecipe({
background: 'accent.action-primary-hover',
},
},
+ color: 'accent.text-primary',
},
text: {
@@ -84,13 +75,6 @@ export const linkRecipe = defineRecipe({
color: 'accent.text-primary',
visibility: 'visible',
},
- _disabled: {
- _before: {
- background: 'accent.non-interactive',
- visibility: 'visible',
- },
- color: 'accent.non-interactive',
- },
_focus: {
_before: {
background: 'focus',
@@ -105,12 +89,13 @@ export const linkRecipe = defineRecipe({
visibility: 'visible',
},
},
+ color: 'accent.text-primary',
},
},
// TODO: Remove invert code
invert: { true: {} },
-
+ disabled: { true: {} },
fullWidth: { true: { width: '100%' } },
},
@@ -122,8 +107,6 @@ export const linkRecipe = defineRecipe({
// TODO: Remove invert code
compoundVariants: [
{
- variant: 'underlined',
- invert: true,
css: {
_focus: {
_before: {
@@ -140,6 +123,43 @@ export const linkRecipe = defineRecipe({
},
color: 'accent.background-secondary',
},
+ invert: true,
+ variant: 'underlined',
+ },
+ {
+ css: {
+ _before: {
+ content: '""',
+ background: 'accent.non-interactive',
+ bottom: '-2px',
+ height: '2px',
+ left: 0,
+ position: 'absolute',
+ right: 0,
+ },
+ color: 'accent.non-interactive',
+ cursor: 'not-allowed',
+ },
+ disabled: true,
+ variant: 'underlined',
+ },
+ {
+ css: {
+ _before: {
+ content: '""',
+ background: 'accent.non-interactive',
+ bottom: '-2px',
+ height: '2px',
+ left: 0,
+ position: 'absolute',
+ right: 0,
+ visibility: 'visible',
+ },
+ color: 'accent.non-interactive',
+ cursor: 'not-allowed',
+ },
+ disabled: true,
+ variant: 'text',
},
],
});