Merge FundAccount and AccountFund (#431)

*  Merge FundAccount and AccountFund

* fix build

* 🐛 fundAccount not showing, small typos/text corrections
This commit is contained in:
Bob van der Helm 2023-09-07 10:20:19 +02:00 committed by GitHub
parent d3c73303c5
commit 93e725fc59
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
16 changed files with 143 additions and 263 deletions

View File

@ -12,7 +12,7 @@ import { useLocation, useNavigate } from 'react-router-dom'
import { getAmountChangeColor } from 'components/Account/AccountBalancesTable/functions' import { getAmountChangeColor } from 'components/Account/AccountBalancesTable/functions'
import useAccountBalanceData from 'components/Account/AccountBalancesTable/useAccountBalanceData' import useAccountBalanceData from 'components/Account/AccountBalancesTable/useAccountBalanceData'
import AccountFund from 'components/Account/AccountFund' import AccountFundFullPage from 'components/Account/AccountFund/AccountFundFullPage'
import ActionButton from 'components/Button/ActionButton' import ActionButton from 'components/Button/ActionButton'
import DisplayCurrency from 'components/DisplayCurrency' import DisplayCurrency from 'components/DisplayCurrency'
import { FormattedNumber } from 'components/FormattedNumber' import { FormattedNumber } from 'components/FormattedNumber'
@ -147,7 +147,7 @@ export default function Index(props: Props) {
} }
useStore.setState({ useStore.setState({
focusComponent: { focusComponent: {
component: <AccountFund />, component: <AccountFundFullPage />,
onClose: () => { onClose: () => {
useStore.setState({ getStartedModal: true }) useStore.setState({ getStartedModal: true })
}, },

View File

@ -1,7 +1,7 @@
import { useCallback, useEffect } from 'react' import { useCallback, useEffect } from 'react'
import { useLocation, useNavigate } from 'react-router-dom' import { useLocation, useNavigate } from 'react-router-dom'
import AccountFund from 'components/Account/AccountFund' import AccountFundFullPage from 'components/Account/AccountFund/AccountFundFullPage'
import FullOverlayContent from 'components/FullOverlayContent' import FullOverlayContent from 'components/FullOverlayContent'
import WalletSelect from 'components/Wallet/WalletSelect' import WalletSelect from 'components/Wallet/WalletSelect'
import useToggle from 'hooks/useToggle' import useToggle from 'hooks/useToggle'
@ -27,7 +27,7 @@ export default function AccountCreateFirst() {
navigate(getRoute(getPage(pathname), address, accountId)) navigate(getRoute(getPage(pathname), address, accountId))
useStore.setState({ useStore.setState({
focusComponent: { focusComponent: {
component: <AccountFund />, component: <AccountFundFullPage />,
onClose: () => { onClose: () => {
useStore.setState({ getStartedModal: true }) useStore.setState({ getStartedModal: true })
}, },

View File

@ -1,20 +1,18 @@
import classNames from 'classnames'
import React, { useCallback, useEffect, useMemo, useState } from 'react' import React, { useCallback, useEffect, useMemo, useState } from 'react'
import { useParams } from 'react-router-dom'
import Button from 'components/Button' import Button from 'components/Button'
import Card from 'components/Card'
import DepositCapMessage from 'components/DepositCapMessage' import DepositCapMessage from 'components/DepositCapMessage'
import FullOverlayContent from 'components/FullOverlayContent' import { ArrowRight, Plus } from 'components/Icons'
import { Plus } from 'components/Icons'
import SwitchAutoLend from 'components/Switch/SwitchAutoLend' import SwitchAutoLend from 'components/Switch/SwitchAutoLend'
import Text from 'components/Text' import Text from 'components/Text'
import TokenInputWithSlider from 'components/TokenInput/TokenInputWithSlider' import TokenInputWithSlider from 'components/TokenInput/TokenInputWithSlider'
import WalletBridges from 'components/Wallet/WalletBridges' import WalletBridges from 'components/Wallet/WalletBridges'
import { BN_ZERO } from 'constants/math' import { BN_ZERO } from 'constants/math'
import useAccounts from 'hooks/useAccounts' import useAutoLend from 'hooks/useAutoLend'
import useCurrentAccount from 'hooks/useCurrentAccount'
import useMarketAssets from 'hooks/useMarketAssets' import useMarketAssets from 'hooks/useMarketAssets'
import useToggle from 'hooks/useToggle' import useToggle from 'hooks/useToggle'
import { useUpdatedAccount } from 'hooks/useUpdatedAccount'
import useWalletBalances from 'hooks/useWalletBalances' import useWalletBalances from 'hooks/useWalletBalances'
import useStore from 'store' import useStore from 'store'
import { BNCoin } from 'types/classes/BNCoin' import { BNCoin } from 'types/classes/BNCoin'
@ -24,56 +22,78 @@ import { defaultFee } from 'utils/constants'
import { getCapLeftWithBuffer } from 'utils/generic' import { getCapLeftWithBuffer } from 'utils/generic'
import { BN } from 'utils/helpers' import { BN } from 'utils/helpers'
export default function AccountFund() { interface Props {
const address = useStore((s) => s.address) account?: Account
address: string
accountId: string
isFullPage?: boolean
}
export default function AccountFundContent(props: Props) {
const deposit = useStore((s) => s.deposit) const deposit = useStore((s) => s.deposit)
const walletAssetModal = useStore((s) => s.walletAssetsModal) const walletAssetModal = useStore((s) => s.walletAssetsModal)
const { accountId } = useParams()
const { data: accounts } = useAccounts(address)
const currentAccount = useCurrentAccount()
const [isFunding, setIsFunding] = useToggle(false) const [isFunding, setIsFunding] = useToggle(false)
const [selectedAccountId, setSelectedAccountId] = useState<null | string>(null)
const [fundingAssets, setFundingAssets] = useState<BNCoin[]>([]) const [fundingAssets, setFundingAssets] = useState<BNCoin[]>([])
const { data: walletBalances } = useWalletBalances(address) const { data: marketAssets } = useMarketAssets()
const { data: walletBalances } = useWalletBalances(props.address)
const { autoLendEnabledAccountIds } = useAutoLend()
const [isLending, toggleIsLending] = useToggle(false)
const { simulateDeposits } = useUpdatedAccount(props.account)
const baseAsset = getBaseAsset() const baseAsset = getBaseAsset()
const hasAssetSelected = fundingAssets.length > 0 const hasAssetSelected = fundingAssets.length > 0
const hasFundingAssets = const hasFundingAssets =
fundingAssets.length > 0 && fundingAssets.every((a) => a.toCoin().amount !== '0') fundingAssets.length > 0 && fundingAssets.every((a) => a.toCoin().amount !== '0')
const { data: marketAssets } = useMarketAssets()
const [isLending, toggleIsLending] = useToggle(false)
const baseBalance = useMemo(
() => walletBalances.find(byDenom(baseAsset.denom))?.amount ?? '0',
[walletBalances, baseAsset],
)
const balances = walletBalances.map((coin) => new BNCoin(coin)) const balances = walletBalances.map((coin) => new BNCoin(coin))
const selectedDenoms = useMemo(() => { const selectedDenoms = useMemo(() => {
return walletAssetModal?.selectedDenoms ?? [] return walletAssetModal?.selectedDenoms ?? []
}, [walletAssetModal?.selectedDenoms]) }, [walletAssetModal?.selectedDenoms])
const handleClick = useCallback(async () => { const baseBalance = useMemo(
setIsFunding(true) () => walletBalances.find(byDenom(baseAsset.denom))?.amount ?? '0',
if (!accountId) return [walletBalances, baseAsset],
const result = await deposit({ )
accountId,
coins: fundingAssets,
lend: isLending,
})
setIsFunding(false)
if (result) useStore.setState({ focusComponent: null, walletAssetsModal: null })
}, [setIsFunding, accountId, deposit, fundingAssets, isLending])
const handleSelectAssetsClick = useCallback(() => { const handleSelectAssetsClick = useCallback(() => {
useStore.setState({ useStore.setState({
walletAssetsModal: { walletAssetsModal: {
isOpen: true, isOpen: true,
selectedDenoms, selectedDenoms,
isBorrow: false,
}, },
}) })
}, [selectedDenoms]) }, [selectedDenoms])
const handleClick = useCallback(async () => {
setIsFunding(true)
if (!props.accountId) return
const result = await deposit({
accountId: props.accountId,
coins: fundingAssets,
lend: isLending,
})
setIsFunding(false)
if (result)
useStore.setState({
fundAndWithdrawModal: null,
walletAssetsModal: null,
focusComponent: null,
})
}, [setIsFunding, props.accountId, deposit, fundingAssets, isLending])
useEffect(() => {
if (BN(baseBalance).isLessThan(defaultFee.amount[0].amount)) {
useStore.setState({ focusComponent: { component: <WalletBridges /> } })
}
}, [baseBalance])
useEffect(() => {
simulateDeposits(isLending ? 'lend' : 'deposit', fundingAssets)
}, [isLending, fundingAssets, simulateDeposits])
useEffect(() => { useEffect(() => {
const currentSelectedDenom = fundingAssets.map((asset) => asset.denom) const currentSelectedDenom = fundingAssets.map((asset) => asset.denom)
@ -101,15 +121,8 @@ export default function AccountFund() {
}, []) }, [])
useEffect(() => { useEffect(() => {
if (BN(baseBalance).isLessThan(defaultFee.amount[0].amount)) { toggleIsLending(autoLendEnabledAccountIds.includes(props.accountId))
useStore.setState({ focusComponent: { component: <WalletBridges /> } }) }, [props.accountId, autoLendEnabledAccountIds, toggleIsLending])
}
}, [baseBalance])
useEffect(() => {
if (accounts && !selectedAccountId && accountId)
setSelectedAccountId(currentAccount?.id ?? accountId)
}, [accounts, selectedAccountId, accountId, currentAccount])
const depositCapReachedCoins = useMemo(() => { const depositCapReachedCoins = useMemo(() => {
const depositCapReachedCoins: BNCoin[] = [] const depositCapReachedCoins: BNCoin[] = []
@ -126,15 +139,9 @@ export default function AccountFund() {
return depositCapReachedCoins return depositCapReachedCoins
}, [fundingAssets, marketAssets]) }, [fundingAssets, marketAssets])
if (!selectedAccountId) return null
return ( return (
<FullOverlayContent <>
title={`Fund Credit Account #${selectedAccountId}`} <div>
copy='In order to start trading with this account, you need to deposit funds.'
docs='fund'
>
<Card className='w-full p-6 bg-white/5'>
{!hasAssetSelected && <Text>Please select an asset.</Text>} {!hasAssetSelected && <Text>Please select an asset.</Text>}
{fundingAssets.map((coin) => { {fundingAssets.map((coin) => {
const asset = getAssetByDenom(coin.denom) as Asset const asset = getAssetByDenom(coin.denom) as Asset
@ -143,7 +150,10 @@ export default function AccountFund() {
return ( return (
<div <div
key={asset.symbol} key={asset.symbol}
className='w-full p-4 border rounded-base border-white/20 bg-white/5' className={classNames(
'w-full mb-4',
props.isFullPage && 'w-full p-4 border rounded-base border-white/20 bg-white/5',
)}
> >
<TokenInputWithSlider <TokenInputWithSlider
asset={asset} asset={asset}
@ -175,20 +185,21 @@ export default function AccountFund() {
/> />
<SwitchAutoLend <SwitchAutoLend
className='pt-4 mt-4 border border-transparent border-t-white/10' className='pt-4 mt-4 border border-transparent border-t-white/10'
accountId={selectedAccountId} accountId={props.accountId}
value={isLending} value={isLending}
onChange={toggleIsLending} onChange={toggleIsLending}
/> />
<Button </div>
className='w-full mt-4' <Button
text='Fund account' className='w-full mt-4'
color='tertiary' text='Fund account'
disabled={!hasFundingAssets || depositCapReachedCoins.length > 0} disabled={!hasFundingAssets || depositCapReachedCoins.length > 0}
showProgressIndicator={isFunding} showProgressIndicator={isFunding}
onClick={handleClick} onClick={handleClick}
size='lg' color={props.isFullPage ? 'tertiary' : undefined}
/> size={props.isFullPage ? 'lg' : undefined}
</Card> rightIcon={props.isFullPage ? undefined : <ArrowRight />}
</FullOverlayContent> />
</>
) )
} }

View File

@ -0,0 +1,42 @@
import React, { useEffect, useState } from 'react'
import { useParams } from 'react-router-dom'
import AccountFundContent from 'components/Account/AccountFund/AccountFundContent'
import Card from 'components/Card'
import FullOverlayContent from 'components/FullOverlayContent'
import useAccounts from 'hooks/useAccounts'
import useCurrentAccount from 'hooks/useCurrentAccount'
import useStore from 'store'
export default function AccountFundFullPage() {
const address = useStore((s) => s.address)
const { accountId } = useParams()
const { data: accounts } = useAccounts(address)
const currentAccount = useCurrentAccount()
const [selectedAccountId, setSelectedAccountId] = useState<null | string>(null)
useEffect(() => {
if (accounts && !selectedAccountId && accountId)
setSelectedAccountId(currentAccount?.id ?? accountId)
}, [accounts, selectedAccountId, accountId, currentAccount])
if (!selectedAccountId || !address) return null
return (
<FullOverlayContent
title={`Fund Credit Account #${selectedAccountId}`}
copy='In order to start trading with this account, you need to deposit funds.'
docs='fund'
>
<Card className='w-full p-6 bg-white/5'>
<AccountFundContent
account={currentAccount}
address={address}
accountId={selectedAccountId}
isFullPage
/>
</Card>
</FullOverlayContent>
)
}

View File

@ -2,7 +2,7 @@ import classNames from 'classnames'
import { useCallback, useEffect } from 'react' import { useCallback, useEffect } from 'react'
import { useLocation, useNavigate, useParams } from 'react-router-dom' import { useLocation, useNavigate, useParams } from 'react-router-dom'
import AccountFund from 'components/Account/AccountFund' import AccountFundFullPage from 'components/Account/AccountFund/AccountFundFullPage'
import AccountStats from 'components/Account/AccountStats' import AccountStats from 'components/Account/AccountStats'
import Button from 'components/Button' import Button from 'components/Button'
import Card from 'components/Card' import Card from 'components/Card'
@ -94,7 +94,7 @@ export default function AccountList(props: Props) {
if (positionBalance.isLessThanOrEqualTo(0)) { if (positionBalance.isLessThanOrEqualTo(0)) {
useStore.setState({ useStore.setState({
focusComponent: { focusComponent: {
component: <AccountFund />, component: <AccountFundFullPage />,
onClose: () => { onClose: () => {
useStore.setState({ getStartedModal: true }) useStore.setState({ getStartedModal: true })
}, },

View File

@ -3,7 +3,7 @@ import { useCallback, useEffect } from 'react'
import { useLocation, useNavigate, useParams } from 'react-router-dom' import { useLocation, useNavigate, useParams } from 'react-router-dom'
import AccountCreateFirst from 'components/Account/AccountCreateFirst' import AccountCreateFirst from 'components/Account/AccountCreateFirst'
import AccountFund from 'components/Account/AccountFund' import AccountFund from 'components/Account/AccountFund/AccountFundFullPage'
import AccountList from 'components/Account/AccountList' import AccountList from 'components/Account/AccountList'
import Button from 'components/Button' import Button from 'components/Button'
import { CircularProgress } from 'components/CircularProgress' import { CircularProgress } from 'components/CircularProgress'

View File

@ -24,7 +24,7 @@ export default function AccoundDeleteAlertDialog(props: Props) {
title, title,
description, description,
negativeButton: { negativeButton: {
text: 'Close', text: 'Cancel',
icon: <Enter />, icon: <Enter />,
onClick: closeHandler, onClick: closeHandler,
}, },

View File

@ -95,7 +95,7 @@ function AccountDeleteModal(props: Props) {
description='Deleting your credit account is irreversible.' description='Deleting your credit account is irreversible.'
closeHandler={closeDeleteAccountModal} closeHandler={closeDeleteAccountModal}
positiveButton={{ positiveButton={{
text: 'Close Positions', text: 'Delete Account',
icon: <ArrowRight />, icon: <ArrowRight />,
isAsync: true, isAsync: true,
onClick: deleteAccountHandler, onClick: deleteAccountHandler,

View File

@ -1,25 +1,5 @@
import { useCallback, useEffect, useMemo, useState } from 'react' import AccountFundContent from 'components/Account/AccountFund/AccountFundContent'
import Button from 'components/Button'
import DepositCapMessage from 'components/DepositCapMessage'
import { ArrowRight, Plus } from 'components/Icons'
import SwitchAutoLend from 'components/Switch/SwitchAutoLend'
import Text from 'components/Text'
import TokenInputWithSlider from 'components/TokenInput/TokenInputWithSlider'
import WalletBridges from 'components/Wallet/WalletBridges'
import { BN_ZERO } from 'constants/math'
import useAutoLend from 'hooks/useAutoLend'
import useMarketAssets from 'hooks/useMarketAssets'
import useToggle from 'hooks/useToggle'
import { useUpdatedAccount } from 'hooks/useUpdatedAccount'
import useWalletBalances from 'hooks/useWalletBalances'
import useStore from 'store' import useStore from 'store'
import { BNCoin } from 'types/classes/BNCoin'
import { byDenom } from 'utils/array'
import { getAssetByDenom, getBaseAsset } from 'utils/assets'
import { defaultFee } from 'utils/constants'
import { getCapLeftWithBuffer } from 'utils/generic'
import { BN } from 'utils/helpers'
interface Props { interface Props {
account: Account account: Account
@ -27,161 +7,10 @@ interface Props {
export default function FundAccount(props: Props) { export default function FundAccount(props: Props) {
const { account } = props const { account } = props
const accountId = account.id
const address = useStore((s) => s.address) const address = useStore((s) => s.address)
const deposit = useStore((s) => s.deposit) const accountId = account.id
const walletAssetModal = useStore((s) => s.walletAssetsModal) if (!address) return null
const [isFunding, setIsFunding] = useToggle(false)
const [fundingAssets, setFundingAssets] = useState<BNCoin[]>([])
const { data: walletBalances } = useWalletBalances(address)
const baseAsset = getBaseAsset()
const hasAssetSelected = fundingAssets.length > 0
const hasFundingAssets =
fundingAssets.length > 0 && fundingAssets.every((a) => a.toCoin().amount !== '0')
const { autoLendEnabledAccountIds } = useAutoLend()
const [isLending, toggleIsLending] = useToggle(false)
const { simulateDeposits } = useUpdatedAccount(account)
const { data: marketAssets } = useMarketAssets()
const baseBalance = useMemo(
() => walletBalances.find(byDenom(baseAsset.denom))?.amount ?? '0',
[walletBalances, baseAsset],
)
const balances = walletBalances.map((coin) => new BNCoin(coin)) return <AccountFundContent account={account} address={address} accountId={accountId} />
const selectedDenoms = useMemo(() => {
return walletAssetModal?.selectedDenoms ?? []
}, [walletAssetModal?.selectedDenoms])
const handleClick = useCallback(async () => {
setIsFunding(true)
if (!accountId) return
const result = await deposit({
accountId,
coins: fundingAssets,
lend: isLending,
})
setIsFunding(false)
if (result) useStore.setState({ fundAndWithdrawModal: null, walletAssetsModal: null })
}, [setIsFunding, accountId, deposit, fundingAssets, isLending])
const handleSelectAssetsClick = useCallback(() => {
useStore.setState({
walletAssetsModal: {
isOpen: true,
selectedDenoms,
isBorrow: false,
},
})
}, [selectedDenoms])
useEffect(() => {
toggleIsLending(autoLendEnabledAccountIds.includes(accountId))
}, [accountId, autoLendEnabledAccountIds, toggleIsLending])
useEffect(() => {
const currentSelectedDenom = fundingAssets.map((asset) => asset.denom)
if (
selectedDenoms.every((denom) => currentSelectedDenom.includes(denom)) &&
selectedDenoms.length === currentSelectedDenom.length
)
return
const newFundingAssets = selectedDenoms.map((denom) =>
BNCoin.fromDenomAndBigNumber(denom, BN(fundingAssets.find(byDenom(denom))?.amount ?? '0')),
)
setFundingAssets(newFundingAssets)
}, [selectedDenoms, fundingAssets])
const updateFundingAssets = useCallback((amount: BigNumber, denom: string) => {
setFundingAssets((fundingAssets) => {
const updateIdx = fundingAssets.findIndex(byDenom(denom))
if (updateIdx === -1) return fundingAssets
fundingAssets[updateIdx].amount = amount
return [...fundingAssets]
})
}, [])
useEffect(() => {
simulateDeposits(isLending ? 'lend' : 'deposit', fundingAssets)
}, [isLending, fundingAssets, simulateDeposits])
useEffect(() => {
if (BN(baseBalance).isLessThan(defaultFee.amount[0].amount)) {
useStore.setState({ focusComponent: { component: <WalletBridges /> } })
}
}, [baseBalance])
const depositCapReachedCoins = useMemo(() => {
const depositCapReachedCoins: BNCoin[] = []
fundingAssets.forEach((asset) => {
const marketAsset = marketAssets.find(byDenom(asset.denom))
if (!marketAsset) return
const capLeft = getCapLeftWithBuffer(marketAsset.cap)
if (asset.amount.isLessThanOrEqualTo(capLeft)) return
depositCapReachedCoins.push(BNCoin.fromDenomAndBigNumber(asset.denom, capLeft))
})
return depositCapReachedCoins
}, [fundingAssets, marketAssets])
return (
<>
<div className='flex flex-wrap items-start'>
{!hasAssetSelected && <Text>Please select an asset.</Text>}
{fundingAssets.map((coin) => {
const asset = getAssetByDenom(coin.denom) as Asset
const balance = balances.find(byDenom(coin.denom))?.amount ?? BN_ZERO
return (
<TokenInputWithSlider
key={coin.denom}
asset={asset}
onChange={(amount) => updateFundingAssets(amount, asset.denom)}
amount={coin.amount ?? BN_ZERO}
max={balance}
balances={balances}
maxText='Max'
disabled={isFunding}
className='w-full mb-4'
/>
)
})}
<Button
className='w-full mt-4'
text='Select assets'
color='tertiary'
rightIcon={<Plus />}
iconClassName='w-3'
onClick={handleSelectAssetsClick}
disabled={isFunding}
/>
<DepositCapMessage
action='fund'
coins={depositCapReachedCoins}
className='pr-4 py-2 mt-4'
showIcon
/>
<SwitchAutoLend
className='pt-4 mt-4 border border-transparent border-t-white/10'
accountId={accountId}
onChange={toggleIsLending}
value={isLending}
/>
</div>
<Button
className='w-full mt-4'
text='Fund account'
rightIcon={<ArrowRight />}
disabled={!hasFundingAssets || depositCapReachedCoins.length > 0}
showProgressIndicator={isFunding}
onClick={handleClick}
/>
</>
)
} }

View File

@ -1,10 +1,7 @@
import { useState } from 'react'
import AccountSummary from 'components/Account/AccountSummary' import AccountSummary from 'components/Account/AccountSummary'
import Card from 'components/Card' import Card from 'components/Card'
import FundAccount from 'components/Modals/FundWithdraw/FundAccount' import FundAccount from 'components/Modals/FundWithdraw/FundAccount'
import WithdrawFromAccount from 'components/Modals/FundWithdraw/WithdrawFromAccount' import WithdrawFromAccount from 'components/Modals/FundWithdraw/WithdrawFromAccount'
import useStore from 'store'
interface Props { interface Props {
account: Account account: Account

View File

@ -20,6 +20,7 @@ import { formatAmountWithSymbol } from 'utils/formatters'
import getTokenOutFromSwapResponse from 'utils/getTokenOutFromSwapResponse' import getTokenOutFromSwapResponse from 'utils/getTokenOutFromSwapResponse'
import { BN } from 'utils/helpers' import { BN } from 'utils/helpers'
function generateExecutionMessage( function generateExecutionMessage(
sender: string | undefined = '', sender: string | undefined = '',
contract: string, contract: string,
@ -243,7 +244,7 @@ export default function createBroadcastSlice(
.join(' and ') .join(' and ')
handleResponseMessages( handleResponseMessages(
response, response,
`Deposited ${options.lend ? 'and lent ' : ''}${depositString} to Credit Credit Account ${ `Deposited ${options.lend ? 'and lent ' : ''}${depositString} to Credit Account ${
options.accountId options.accountId
}`, }`,
) )