From 1713cd56a86311e0e4c11303a49418d83f3a5559 Mon Sep 17 00:00:00 2001 From: Bruno Menezes Date: Tue, 14 Jan 2025 15:55:31 +1300 Subject: [PATCH] Feat: Ether deposit validation on change and display balance (#294) --- .../index.tsx} | 122 ++++-- packages/ui/src/hooks/useAccountBalance.ts | 42 ++ packages/ui/src/hooks/useFormattedBalance.ts | 15 - packages/ui/src/index.tsx | 8 +- packages/ui/test/EtherDepositForm.test.tsx | 395 ++++++++++-------- .../ui/test/hooks/useAccountBalance.test.ts | 60 +++ .../ui/test/hooks/useFormattedBalance.test.ts | 46 -- packages/ui/test/utils/helpers.ts | 25 ++ 8 files changed, 430 insertions(+), 283 deletions(-) rename packages/ui/src/{EtherDepositForm.tsx => EtherDepositForm/index.tsx} (64%) create mode 100644 packages/ui/src/hooks/useAccountBalance.ts delete mode 100644 packages/ui/src/hooks/useFormattedBalance.ts create mode 100644 packages/ui/test/hooks/useAccountBalance.test.ts delete mode 100644 packages/ui/test/hooks/useFormattedBalance.test.ts create mode 100644 packages/ui/test/utils/helpers.ts diff --git a/packages/ui/src/EtherDepositForm.tsx b/packages/ui/src/EtherDepositForm/index.tsx similarity index 64% rename from packages/ui/src/EtherDepositForm.tsx rename to packages/ui/src/EtherDepositForm/index.tsx index ec279348e..6564dec3e 100644 --- a/packages/ui/src/EtherDepositForm.tsx +++ b/packages/ui/src/EtherDepositForm/index.tsx @@ -7,12 +7,14 @@ import { Autocomplete, Button, Collapse, + Flex, Group, Loader, Stack, Text, - Textarea, TextInput, + Textarea, + UnstyledButton, } from "@mantine/core"; import { useForm } from "@mantine/form"; import { useDisclosure } from "@mantine/hooks"; @@ -25,18 +27,18 @@ import { } from "react-icons/tb"; import { BaseError, - getAddress, Hex, + getAddress, isAddress, isHex, parseUnits, zeroAddress, } from "viem"; import { useAccount, useWaitForTransactionReceipt } from "wagmi"; -import { TransactionProgress } from "./TransactionProgress"; -import useUndeployedApplication from "./hooks/useUndeployedApplication"; -import { TransactionFormSuccessData } from "./DepositFormTypes"; -import { useFormattedBalance } from "./hooks/useFormattedBalance"; +import { TransactionFormSuccessData } from "../DepositFormTypes"; +import { TransactionProgress } from "../TransactionProgress"; +import { useAccountBalance } from "../hooks/useAccountBalance"; +import useUndeployedApplication from "../hooks/useUndeployedApplication"; export interface EtherDepositFormProps { applications: string[]; @@ -54,11 +56,12 @@ export const EtherDepositForm: FC = (props) => { } = props; const [advanced, { toggle: toggleAdvanced }] = useDisclosure(false); const { chain } = useAccount(); - const balance = useFormattedBalance(); + const accountBalance = useAccountBalance(); const form = useForm({ - validateInputOnBlur: true, + validateInputOnChange: true, initialValues: { + accountBalance: accountBalance, application: "", amount: "", execLayerData: "0x", @@ -66,10 +69,14 @@ export const EtherDepositForm: FC = (props) => { validate: { application: (value) => value !== "" && isAddress(value) ? null : "Invalid application", - amount: (value) => { + amount: (value, values) => { if (value !== "" && Number(value) > 0) { - if (Number(value) > Number(balance)) { - return `The amount ${value} exceeds your current balance of ${balance} ETH`; + const val = parseUnits( + value, + values.accountBalance.decimals, + ); + if (val > values.accountBalance.value) { + return `The amount ${value} exceeds your current balance of ${values.accountBalance.formatted} ETH`; } return null; } else { @@ -79,22 +86,25 @@ export const EtherDepositForm: FC = (props) => { execLayerData: (value) => isHex(value) ? null : "Invalid hex string", }, - transformValues: (values) => ({ - address: isAddress(values.application) - ? getAddress(values.application) - : zeroAddress, - amount: - values.amount !== "" - ? parseUnits( - values.amount, - chain?.nativeCurrency.decimals ?? 18, - ) - : undefined, - execLayerData: values.execLayerData - ? (values.execLayerData as Hex) - : "0x", - }), + transformValues: (values) => { + return { + address: isAddress(values.application) + ? getAddress(values.application) + : zeroAddress, + amount: + values.amount !== "" + ? parseUnits( + values.amount, + chain?.nativeCurrency.decimals ?? 18, + ) + : undefined, + execLayerData: values.execLayerData + ? (values.execLayerData as Hex) + : "0x", + }; + }, }); + const { address, amount, execLayerData } = form.getTransformedValues(); const prepare = useSimulateEtherPortalDepositEther({ args: [address, execLayerData], @@ -118,9 +128,19 @@ export const EtherDepositForm: FC = (props) => { form.reset(); execute.reset(); onSearchApplications(""); + accountBalance.refetch(); } // eslint-disable-next-line react-hooks/exhaustive-deps - }, [wait, onSearchApplications, onSuccess]); + }, [wait, onSearchApplications, onSuccess, accountBalance]); + + useEffect(() => { + form.setValues({ accountBalance: accountBalance }); + + if (form.isDirty("amount")) { + form.validateField("amount"); + } + // eslint-disable-next-line react-hooks/exhaustive-deps + }, [accountBalance]); return (
@@ -128,6 +148,7 @@ export const EtherDepositForm: FC = (props) => { = (props) => { )} - ETH} - withAsterisk - {...form.getInputProps("amount")} - /> + + ETH} + withAsterisk + {...form.getInputProps("amount")} + /> + + + Balance: {accountBalance.formatted} + {accountBalance.value > 0 && ( + { + form.setFieldValue( + "amount", + accountBalance.formatted, + ); + }} + data-testid="max-button" + > + Max + + )} + +