From 9eb9a200308c9889d403595984eaef33bb93a31e Mon Sep 17 00:00:00 2001 From: Eugene Chybisov <18644653+chybisov@users.noreply.github.com> Date: Fri, 20 Dec 2024 14:36:42 +0100 Subject: [PATCH] fix: improve deposit flow card titles (#328) --- examples/deposit-flow/src/App.tsx | 18 ++-- .../src/components/DepositCard.tsx | 84 +++++++++++++++++++ examples/deposit-flow/src/config.ts | 4 + examples/deposit-flow/src/main.tsx | 6 +- package.json | 2 +- packages/wallet-management/package.json | 2 +- .../wallet-management/src/i18n/i18next.d.ts | 12 +-- packages/widget-embedded/src/config.ts | 1 + packages/widget/package.json | 2 +- .../components/AmountInput/AmountInput.tsx | 10 ++- .../ContractComponent/ItemPrice.tsx | 11 ++- .../components/ContractComponent/NFT/NFT.tsx | 11 ++- .../src/components/RouteCard/RouteCard.tsx | 4 +- .../widget/src/components/Routes/Routes.tsx | 14 +++- .../src/components/Routes/RoutesExpanded.tsx | 13 ++- .../SendToWallet/SendToWalletButton.tsx | 16 +++- packages/widget/src/components/Step/Step.tsx | 4 +- packages/widget/src/config/version.ts | 2 +- packages/widget/src/hooks/useGasRefuel.ts | 24 ++---- .../widget/src/hooks/useIsContractAddress.ts | 1 + packages/widget/src/hooks/useRoutes.ts | 10 ++- packages/widget/src/i18n/en.json | 6 +- packages/widget/src/i18n/i18next.d.ts | 12 +-- packages/widget/src/index.ts | 2 +- .../widget/src/pages/MainPage/MainPage.tsx | 17 ++-- packages/widget/src/types/widget.ts | 3 +- packages/widget/tsconfig.json | 6 +- 27 files changed, 207 insertions(+), 90 deletions(-) create mode 100644 examples/deposit-flow/src/components/DepositCard.tsx create mode 100644 examples/deposit-flow/src/config.ts diff --git a/examples/deposit-flow/src/App.tsx b/examples/deposit-flow/src/App.tsx index 727a9ad74..4888e834b 100644 --- a/examples/deposit-flow/src/App.tsx +++ b/examples/deposit-flow/src/App.tsx @@ -4,19 +4,14 @@ import { CoinKey, DisabledUI, HiddenUI, - ItemPrice, LiFiWidget, } from '@lifi/widget' import { useMemo } from 'react' +import { DepositCard } from './components/DepositCard' +import { contractTool } from './config' -const depositAddress = '0xdde759c7cf032b1d0e633a7e9cfa6653d1911a22' -const depositAmount = 5000000n - -export const contractTool = { - logoURI: - 'https://github.com/lifinance/widget/assets/18644653/eb043a91-18ba-4da7-91c4-029a53a25989', - name: 'Immutable', -} +// EXAMPLE CONTRACT, DON'T DEPOSIT +const depositAddress = '0x4bF3E32de155359D1D75e8B474b66848221142fc' const contractCalls: ContractCall[] = [] @@ -30,7 +25,7 @@ export function App() { }, subvariant: 'custom', subvariantOptions: { custom: 'deposit' }, - integrator: 'Immutable', + integrator: 'ProtocolName', disabledUI: [DisabledUI.ToAddress], hiddenUI: [HiddenUI.Appearance, HiddenUI.Language], useRecommendedRoute: true, @@ -47,7 +42,7 @@ export function App() { return ( diff --git a/examples/deposit-flow/src/components/DepositCard.tsx b/examples/deposit-flow/src/components/DepositCard.tsx new file mode 100644 index 000000000..c3b1fac6d --- /dev/null +++ b/examples/deposit-flow/src/components/DepositCard.tsx @@ -0,0 +1,84 @@ +import { + type ContractCall, + type TokenAmount, + useFieldActions, +} from '@lifi/widget' +import { Avatar, Box, Chip, Stack, Typography } from '@mui/material' +import { useEffect } from 'react' +import { contractTool } from '../config' + +export interface ItemPriceProps { + token: TokenAmount + contractCalls?: ContractCall[] +} + +export const DepositCard: React.FC = ({ + token, + contractCalls, +}) => { + const { setFieldValue } = useFieldActions() + + useEffect(() => { + if (token) { + setFieldValue('toChain', token.chainId, { isTouched: true }) + setFieldValue('toToken', token.address, { isTouched: true }) + } + if (contractCalls) { + setFieldValue('contractCalls', contractCalls, { + isTouched: true, + }) + } + }, [contractCalls, setFieldValue, token]) + return ( + + + + + Protocol Name + + + + + TVL + + $69M + + + + Incentives + + $4.2M + + + + + } /> + + } + /> + + + ) +} diff --git a/examples/deposit-flow/src/config.ts b/examples/deposit-flow/src/config.ts new file mode 100644 index 000000000..4aa4da344 --- /dev/null +++ b/examples/deposit-flow/src/config.ts @@ -0,0 +1,4 @@ +export const contractTool = { + logoURI: 'https://gravatar.com/avatar/any?d=wavatar', + name: 'Protocol Name', +} diff --git a/examples/deposit-flow/src/main.tsx b/examples/deposit-flow/src/main.tsx index daaa3ad04..fd76452db 100644 --- a/examples/deposit-flow/src/main.tsx +++ b/examples/deposit-flow/src/main.tsx @@ -2,7 +2,7 @@ import { QueryClient, QueryClientProvider } from '@tanstack/react-query' import React from 'react' import ReactDOM from 'react-dom/client' import { http, WagmiProvider, createConfig } from 'wagmi' -import { base, mainnet } from 'wagmi/chains' +import { arbitrum, base, mainnet, optimism } from 'wagmi/chains' import { injected } from 'wagmi/connectors' import { App } from './App' import { WalletHeader } from './components/WalletHeader' @@ -12,11 +12,13 @@ const queryClient = new QueryClient() const _projectId = import.meta.env.VITE_WALLET_CONNECT_PROJECT_ID const config = createConfig({ - chains: [mainnet], + chains: [mainnet, base, optimism, arbitrum], connectors: [injected()], transports: { [mainnet.id]: http(), [base.id]: http(), + [optimism.id]: http(), + [arbitrum.id]: http(), }, }) diff --git a/package.json b/package.json index acc6b2690..f1cfee050 100644 --- a/package.json +++ b/package.json @@ -1,5 +1,5 @@ { - "version": "3.12.6-alpha.0", + "version": "3.12.6-beta.0", "private": true, "sideEffects": false, "type": "module", diff --git a/packages/wallet-management/package.json b/packages/wallet-management/package.json index da3904f97..987855618 100644 --- a/packages/wallet-management/package.json +++ b/packages/wallet-management/package.json @@ -1,6 +1,6 @@ { "name": "@lifi/wallet-management", - "version": "3.4.8-alpha.0", + "version": "3.4.8-beta.0", "description": "LI.FI Wallet Management solution.", "type": "module", "main": "./src/index.ts", diff --git a/packages/wallet-management/src/i18n/i18next.d.ts b/packages/wallet-management/src/i18n/i18next.d.ts index 49e267140..ada8de6c2 100644 --- a/packages/wallet-management/src/i18n/i18next.d.ts +++ b/packages/wallet-management/src/i18n/i18next.d.ts @@ -2,13 +2,9 @@ import en from './en.json' with { type: 'json' } const defaultResource = { translation: en } -declare global { - namespace WalletTranslation { - declare module 'i18next' { - interface CustomTypeOptions { - defaultNS: 'translation' - resources: typeof defaultResource - } - } +declare module 'i18next' { + interface CustomTypeOptions { + defaultNS: 'translation' + resources: typeof defaultResource } } diff --git a/packages/widget-embedded/src/config.ts b/packages/widget-embedded/src/config.ts index cb0366583..c30152f8c 100644 --- a/packages/widget-embedded/src/config.ts +++ b/packages/widget-embedded/src/config.ts @@ -3,6 +3,7 @@ import './index.css' export const widgetBaseConfig: WidgetConfig = { subvariant: 'custom', + integrator: 'li.fi-playground', hiddenUI: ['history'], // buildUrl: true, diff --git a/packages/widget/package.json b/packages/widget/package.json index 40cf81e08..4b44eabb0 100644 --- a/packages/widget/package.json +++ b/packages/widget/package.json @@ -1,6 +1,6 @@ { "name": "@lifi/widget", - "version": "3.12.6-alpha.0", + "version": "3.12.6-beta.0", "description": "LI.FI Widget for cross-chain bridging and swapping. It will drive your multi-chain strategy and attract new users from everywhere.", "type": "module", "main": "./src/index.ts", diff --git a/packages/widget/src/components/AmountInput/AmountInput.tsx b/packages/widget/src/components/AmountInput/AmountInput.tsx index 603a9f42d..2ac23a3f6 100644 --- a/packages/widget/src/components/AmountInput/AmountInput.tsx +++ b/packages/widget/src/components/AmountInput/AmountInput.tsx @@ -70,6 +70,7 @@ export const AmountInputBase: React.FC< ...props }) => { const { t } = useTranslation() + const { subvariant, subvariantOptions } = useWidgetConfig() const ref = useRef(null) const amountKey = FormKeyHelper.getAmountKey(formType) const { onChange, onBlur, value } = useFieldController({ name: amountKey }) @@ -98,9 +99,16 @@ export const AmountInputBase: React.FC< } }, [value]) + const title = + subvariant === 'custom' + ? subvariantOptions?.custom === 'deposit' + ? t('header.amount') + : t('header.youPay') + : t('header.send') + return ( - {t('header.send')} + {title} diff --git a/packages/widget/src/components/ContractComponent/ItemPrice.tsx b/packages/widget/src/components/ContractComponent/ItemPrice.tsx index e02c83171..77899e541 100644 --- a/packages/widget/src/components/ContractComponent/ItemPrice.tsx +++ b/packages/widget/src/components/ContractComponent/ItemPrice.tsx @@ -1,5 +1,6 @@ import type { ContractCall } from '@lifi/sdk' import { useEffect } from 'react' +import { formatUnits } from 'viem' import { useFieldActions } from '../../stores/form/useFieldActions.js' import type { TokenAmount } from '../../types/token.js' import { Token } from '../Token/Token.js' @@ -19,9 +20,13 @@ export const ItemPrice: React.FC = ({ if (token) { setFieldValue('toChain', token.chainId, { isTouched: true }) setFieldValue('toToken', token.address, { isTouched: true }) - setFieldValue('toAmount', token.amount?.toString(), { - isTouched: true, - }) + setFieldValue( + 'toAmount', + token.amount ? formatUnits(token.amount, token.decimals) : '', + { + isTouched: true, + } + ) } if (contractCalls) { setFieldValue('contractCalls', contractCalls, { diff --git a/packages/widget/src/components/ContractComponent/NFT/NFT.tsx b/packages/widget/src/components/ContractComponent/NFT/NFT.tsx index adafc8eb0..5f66b1fdf 100644 --- a/packages/widget/src/components/ContractComponent/NFT/NFT.tsx +++ b/packages/widget/src/components/ContractComponent/NFT/NFT.tsx @@ -1,4 +1,5 @@ import { useEffect } from 'react' +import { formatUnits } from 'viem' import { useFieldActions } from '../../../stores/form/useFieldActions.js' import { NFTBase } from './NFTBase.js' import type { NFTProps } from './types.js' @@ -18,9 +19,13 @@ export const NFT: React.FC = ({ if (token) { setFieldValue('toChain', token.chainId, { isTouched: true }) setFieldValue('toToken', token.address, { isTouched: true }) - setFieldValue('toAmount', token.amount?.toString(), { - isTouched: true, - }) + setFieldValue( + 'toAmount', + token.amount ? formatUnits(token.amount, token.decimals) : '', + { + isTouched: true, + } + ) } if (contractCall) { setFieldValue('contractCalls', [contractCall], { diff --git a/packages/widget/src/components/RouteCard/RouteCard.tsx b/packages/widget/src/components/RouteCard/RouteCard.tsx index bf545d131..903e20d60 100644 --- a/packages/widget/src/components/RouteCard/RouteCard.tsx +++ b/packages/widget/src/components/RouteCard/RouteCard.tsx @@ -26,7 +26,7 @@ export const RouteCard: React.FC< ...other }) => { const { t } = useTranslation() - const { subvariant } = useWidgetConfig() + const { subvariant, subvariantOptions } = useWidgetConfig() const [cardExpanded, setCardExpanded] = useState(defaulExpanded) const handleExpand: MouseEventHandler = (e) => { @@ -35,7 +35,7 @@ export const RouteCard: React.FC< } const token: TokenAmount = - subvariant === 'custom' + subvariant === 'custom' && subvariantOptions?.custom !== 'deposit' ? { ...route.fromToken, amount: BigInt(route.fromAmount) } : { ...route.toToken, amount: BigInt(route.toAmount) } const impactToken: TokenAmount | undefined = diff --git a/packages/widget/src/components/Routes/Routes.tsx b/packages/widget/src/components/Routes/Routes.tsx index dc3777c5d..82881e45b 100644 --- a/packages/widget/src/components/Routes/Routes.tsx +++ b/packages/widget/src/components/Routes/Routes.tsx @@ -16,7 +16,8 @@ import { RouteNotFoundCard } from '../RouteCard/RouteNotFoundCard.js' export const Routes: React.FC = (props) => { const { t } = useTranslation() const navigate = useNavigate() - const { subvariant, useRecommendedRoute } = useWidgetConfig() + const { subvariant, subvariantOptions, useRecommendedRoute } = + useWidgetConfig() const { routes, isLoading, @@ -42,11 +43,16 @@ export const Routes: React.FC = (props) => { const showAll = !onlyRecommendedRoute && !routeNotFound && (routes?.length ?? 0) > 1 + const title = + subvariant === 'custom' + ? subvariantOptions?.custom === 'deposit' + ? t('header.receive') + : t('header.youPay') + : t('header.receive') + return ( - - {subvariant === 'custom' ? t('header.youPay') : t('header.receive')} - + {title} { export const RoutesExpandedElement = () => { const { t } = useTranslation() const navigate = useNavigate() - const { subvariant } = useWidgetConfig() + const { subvariant, subvariantOptions } = useWidgetConfig() const routesRef = useRef() const emitter = useWidgetEvents() const routesActiveRef = useRef(false) @@ -118,6 +118,13 @@ export const RoutesExpandedElement = () => { emitter.emit(WidgetEvent.WidgetExpanded, expanded) }, [emitter, expanded]) + const title = + subvariant === 'custom' + ? subvariantOptions?.custom === 'deposit' + ? t('header.deposit') + : t('header.youPay') + : t('header.receive') + return ( { flex: 1, }} > - {subvariant === 'custom' - ? t('header.youPay') - : t('header.receive')} + {title} = (props) => { const { t } = useTranslation() const navigate = useNavigate() - const { disabledUI, hiddenUI, toAddress, toAddresses } = useWidgetConfig() + const { + disabledUI, + hiddenUI, + toAddress, + toAddresses, + subvariant, + subvariantOptions, + } = useWidgetConfig() const { showSendToWallet } = useSendToWalletStore() const [toAddressFieldValue, toChainId, toTokenAddress] = useFieldValues( 'toAddress', @@ -114,6 +121,11 @@ export const SendToWalletButton: React.FC = (props) => { const isOpenCollapse = requiredToAddress || (showSendToWallet && !hiddenToAddress) + const title = + subvariant === 'custom' && subvariantOptions?.custom === 'deposit' + ? t('header.depositTo') + : t('header.sendToWallet') + return ( = (props) => { sx={{ width: '100%', ...props.sx }} > - {t('header.sendToWallet')} + {title} { const toChain = getChainById(toChainId) const fromChain = getChainById(fromChainId) - const { accounts } = useAccount() + const { account: toAccount } = useAccount({ chainType: toChain?.chainType }) - const fromAccount = accounts.find( - (account) => account.chainType === fromChain?.chainType - ) - - const toAccount = accounts.find( - (account) => account.chainType === toChain?.chainType - ) + const effectiveToAddress = toAddress || toAccount?.address - const isFromContractAddress = useIsContractAddress( - fromAccount?.address, - fromChainId, - fromAccount?.chainType - ) const isToContractAddress = useIsContractAddress( - toAddress, + effectiveToAddress, toChainId, toChain?.chainType ) const { token: destinationNativeToken } = useTokenBalance( - toAddress || toAccount?.address, + effectiveToAddress, toChainId ? toChain?.nativeToken : undefined ) @@ -55,9 +44,8 @@ export const useGasRefuel = () => { const isChainTypeSatisfied = fromChain?.chainType !== toChain?.chainType ? Boolean(toAddress) : true - const isToAddressSatisfied = isFromContractAddress - ? toAddress && toAddress !== fromAccount?.address && !isToContractAddress - : true + // We should not refuel to the contract address + const isToAddressSatisfied = effectiveToAddress && !isToContractAddress const enabled = useMemo(() => { if ( diff --git a/packages/widget/src/hooks/useIsContractAddress.ts b/packages/widget/src/hooks/useIsContractAddress.ts index 36def167a..fb0b6e6b7 100644 --- a/packages/widget/src/hooks/useIsContractAddress.ts +++ b/packages/widget/src/hooks/useIsContractAddress.ts @@ -16,6 +16,7 @@ export const useIsContractAddress = ( enabled: Boolean(chainType === ChainType.EVM && chainId), }, }) + const isContractAddress = !!contractCode return isContractAddress } diff --git a/packages/widget/src/hooks/useRoutes.ts b/packages/widget/src/hooks/useRoutes.ts index 9a34de458..3abd62254 100644 --- a/packages/widget/src/hooks/useRoutes.ts +++ b/packages/widget/src/hooks/useRoutes.ts @@ -177,6 +177,7 @@ export const useRoutes = ({ observableRoute }: RoutesProps = {}) => { signal, }) => { const fromAmount = parseUnits(fromTokenAmount, fromToken!.decimals) + const toAmount = parseUnits(toTokenAmount, toToken!.decimals) const formattedSlippage = Number.parseFloat(slippage) / 100 const allowBridges = swapOnly @@ -210,17 +211,18 @@ export const useRoutes = ({ observableRoute }: RoutesProps = {}) => { fromAddress, toAddress, fromAmount, + toAmount, slippage: formattedSlippage, }) - if (subvariant === 'custom' && contractCalls && toTokenAmount) { + if (subvariant === 'custom' && contractCalls && toAmount) { const contractCallQuote = await getContractCallsQuote( { // Contract calls are enabled only when fromAddress is set fromAddress: fromAddress as string, fromChain: fromChainId, fromToken: fromTokenAddress, - toAmount: toTokenAmount, + toAmount: toAmount.toString(), toChain: toChainId, toToken: toTokenAddress, contractCalls, @@ -265,8 +267,8 @@ export const useRoutes = ({ observableRoute }: RoutesProps = {}) => { fromAddress: contractCallQuote.action.fromAddress, toChainId: contractCallQuote.action.toChainId, toAmountUSD: contractCallQuote.estimate.toAmountUSD || '', - toAmount: toTokenAmount, - toAmountMin: toTokenAmount, + toAmount: contractCallQuote.estimate.toAmount, + toAmountMin: contractCallQuote.estimate.toAmountMin, toToken: toToken!, toAddress: contractCallQuote.action.toAddress || diff --git a/packages/widget/src/i18n/en.json b/packages/widget/src/i18n/en.json index a0d8dffa2..51a57ea24 100644 --- a/packages/widget/src/i18n/en.json +++ b/packages/widget/src/i18n/en.json @@ -52,12 +52,14 @@ }, "header": { "activeTransactions": "Active transactions", + "amount": "Amount", "bookmarkedWallets": "Bookmarked wallets", "bridge": "Bridge", "checkout": "Checkout", "checkoutDetails": "Checkout details", "deposit": "Deposit", "depositDetails": "Deposit details", + "depositTo": "Deposit to", "exchange": "Exchange", "from": "Exchange from", "gas": "Gas", @@ -265,13 +267,15 @@ "stepBridge": "Bridge", "stepBridgeAndBuy": "Bridge and buy", "stepBridgeAndDeposit": "Bridge and deposit", + "stepBuy": "Buy", + "stepDeposit": "Deposit", "stepDetails": "{{tool}} via LI.FI", "stepSwap": "Swap", "stepSwapAndBridge": "Swap and bridge", "stepSwapAndBuy": "Swap and buy", "stepSwapAndDeposit": "Swap and deposit", - "transferId": "Transfer ID", "swapStepDetails": "Swap on {{chain}} via {{tool}}", + "transferId": "Transfer ID", "tags": { "cheapest": "Best Return", "fastest": "Fastest" diff --git a/packages/widget/src/i18n/i18next.d.ts b/packages/widget/src/i18n/i18next.d.ts index 509d0fd72..ada8de6c2 100644 --- a/packages/widget/src/i18n/i18next.d.ts +++ b/packages/widget/src/i18n/i18next.d.ts @@ -2,13 +2,9 @@ import en from './en.json' with { type: 'json' } const defaultResource = { translation: en } -declare global { - namespace GeneralTranslation { - declare module 'i18next' { - interface CustomTypeOptions { - defaultNS: 'translation' - resources: typeof defaultResource - } - } +declare module 'i18next' { + interface CustomTypeOptions { + defaultNS: 'translation' + resources: typeof defaultResource } } diff --git a/packages/widget/src/index.ts b/packages/widget/src/index.ts index 31d7275d0..337237a9d 100644 --- a/packages/widget/src/index.ts +++ b/packages/widget/src/index.ts @@ -1,5 +1,5 @@ export type * from '@lifi/sdk' -export { ChainType, ChainId } from '@lifi/sdk' +export { ChainType, ChainId, CoinKey } from '@lifi/sdk' export { App as LiFiWidget } from './App.js' export type { WidgetDrawer } from './AppDrawer.js' export * from './components/ContractComponent/ItemPrice.js' diff --git a/packages/widget/src/pages/MainPage/MainPage.tsx b/packages/widget/src/pages/MainPage/MainPage.tsx index f6b290941..d4fdc5327 100644 --- a/packages/widget/src/pages/MainPage/MainPage.tsx +++ b/packages/widget/src/pages/MainPage/MainPage.tsx @@ -31,22 +31,23 @@ export const MainPage: React.FC = () => { : subvariant === 'refuel' ? t('header.gas') : t('header.exchange') + useHeader(title) + const marginSx = { marginBottom: 2 } + return ( - + {custom ? ( - - {contractComponent} - + {contractComponent} ) : null} - {!custom ? ( - + {!custom || subvariantOptions?.custom === 'deposit' ? ( + ) : null} - {!wideVariant ? : null} - + {!wideVariant ? : null} +