update layout for modal, implement borrow tables (#105)

This commit is contained in:
Bob van der Helm 2023-03-01 13:49:57 +01:00 committed by GitHub
parent 493ec7c44c
commit cbb0700455
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
31 changed files with 357 additions and 585 deletions

View File

@ -1,4 +1,5 @@
import Background from 'components/Background' import Background from 'components/Background'
import FetchPrices from 'components/FetchPrices'
import { Modals } from 'components/Modals' import { Modals } from 'components/Modals'
import DesktopNavigation from 'components/Navigation/DesktopNavigation' import DesktopNavigation from 'components/Navigation/DesktopNavigation'
import Toaster from 'components/Toaster' import Toaster from 'components/Toaster'
@ -19,6 +20,7 @@ export default function RootLayout({ children }: { children: React.ReactNode })
</WalletConnectProvider> </WalletConnectProvider>
<Modals /> <Modals />
<Toaster /> <Toaster />
<FetchPrices />
<main className='relative flex lg:min-h-[calc(100vh-120px)]'> <main className='relative flex lg:min-h-[calc(100vh-120px)]'>
<div className='flex flex-grow flex-col flex-wrap'>{children}</div> <div className='flex flex-grow flex-col flex-wrap'>{children}</div>
</main> </main>

View File

@ -1,5 +0,0 @@
import Loading from 'components/Loading'
export default function page() {
return <Loading />
}

View File

@ -17,10 +17,7 @@ export default async function page({ params }: { params: PageParams }) {
if (debt) { if (debt) {
prev.active.push({ prev.active.push({
...borrow, ...borrow,
debt: { debt: debt.amount,
amount: '100000',
value: '12389478321',
},
}) })
} else { } else {
prev.available.push(borrow) prev.available.push(borrow)

View File

@ -10,7 +10,6 @@ import { ArrowRightLine, ChevronDown, ChevronLeft } from 'components/Icons'
import { LabelValuePair } from 'components/LabelValuePair' import { LabelValuePair } from 'components/LabelValuePair'
import { PositionsList } from 'components/PositionsList' import { PositionsList } from 'components/PositionsList'
import { useAccountStats } from 'hooks/data/useAccountStats' import { useAccountStats } from 'hooks/data/useAccountStats'
import { useBalances } from 'hooks/data/useBalances'
import { convertFromGwei } from 'utils/formatters' import { convertFromGwei } from 'utils/formatters'
import { createRiskData } from 'utils/risk' import { createRiskData } from 'utils/risk'
import useStore from 'store' import useStore from 'store'
@ -23,7 +22,6 @@ export const AccountDetails = () => {
const marketAssets = getMarketAssets() const marketAssets = getMarketAssets()
const baseAsset = getBaseAsset() const baseAsset = getBaseAsset()
const balances = useBalances()
const accountStats = useAccountStats() const accountStats = useAccountStats()
const [showManageMenu, setShowManageMenu] = useState(false) const [showManageMenu, setShowManageMenu] = useState(false)
@ -118,7 +116,6 @@ export const AccountDetails = () => {
/> />
</div> </div>
{riskData && <RiskChart data={riskData} />} {riskData && <RiskChart data={riskData} />}
<PositionsList title='Balances' data={balances} />
</div> </div>
) )
} }

View File

@ -10,7 +10,7 @@ import { Text } from 'components/Text'
import { useAccountStats } from 'hooks/data/useAccountStats' import { useAccountStats } from 'hooks/data/useAccountStats'
import { useCreditAccounts } from 'hooks/queries/useCreditAccounts' import { useCreditAccounts } from 'hooks/queries/useCreditAccounts'
import { getBaseAsset } from 'utils/assets' import { getBaseAsset } from 'utils/assets'
import { formatValue } from 'utils/formatters' import { formatLeverage, formatValue } from 'utils/formatters'
export const AccountStatus = () => { export const AccountStatus = () => {
const baseAsset = getBaseAsset() const baseAsset = getBaseAsset()
@ -41,7 +41,7 @@ export const AccountStatus = () => {
.dividedBy(10 ** baseAsset.decimals) .dividedBy(10 ** baseAsset.decimals)
.toNumber()} .toNumber()}
animate animate
prefix='$: ' options={{ prefix: '$: ' }}
/> />
</Text> </Text>
@ -50,10 +50,9 @@ export const AccountStatus = () => {
label='Lvg' label='Lvg'
tooltip={ tooltip={
<Text size='sm'> <Text size='sm'>
Current Leverage:{' '} Current Leverage: {formatLeverage(accountStats.currentLeverage)}
{formatValue(accountStats.currentLeverage, 0, 2, true, false, 'x')}
<br /> <br />
Max Leverage: {formatValue(accountStats.maxLeverage, 0, 2, true, false, 'x')} Max Leverage: {formatLeverage(accountStats.maxLeverage)}
</Text> </Text>
} }
/> />
@ -63,7 +62,8 @@ export const AccountStatus = () => {
label='Risk' label='Risk'
tooltip={ tooltip={
<Text size='sm'> <Text size='sm'>
Current Risk: {formatValue(accountStats.risk * 100, 0, 2, true, false, '%')} Current Risk:{' '}
{formatValue(accountStats.risk * 100, { minDecimals: 0, suffix: '%' })}
</Text> </Text>
} }
/> />

View File

@ -13,7 +13,7 @@ export const ConfirmModal = () => {
const deleteOpen = useStore((s) => s.deleteAccountModal) const deleteOpen = useStore((s) => s.deleteAccountModal)
return ( return (
<Modal open={createOpen || deleteOpen}> <Modal title='Confirm' open={createOpen || deleteOpen}>
<div <div
className={classNames( className={classNames(
'relative flex h-[630px] w-full flex-wrap items-center justify-center p-6', 'relative flex h-[630px] w-full flex-wrap items-center justify-center p-6',

View File

@ -94,7 +94,7 @@ export const FundAccountModal = () => {
const percentageValue = isNaN(amount) ? 0 : (amount * 100) / walletAmount const percentageValue = isNaN(amount) ? 0 : (amount * 100) / walletAmount
return ( return (
<Modal open={open} setOpen={setOpen}> <Modal title='Fund account' open={open} setOpen={setOpen}>
<div className='flex min-h-[520px] w-full'> <div className='flex min-h-[520px] w-full'>
{balanceIsLoading && ( {balanceIsLoading && (
<div className='absolute inset-0 z-40 grid place-items-center bg-black/50'> <div className='absolute inset-0 z-40 grid place-items-center bg-black/50'>

View File

@ -25,11 +25,13 @@ export const RiskChart = ({ data }: RiskChartProps) => {
<FormattedNumber <FormattedNumber
className='px-3 pb-2 text-lg' className='px-3 pb-2 text-lg'
amount={currentRisk * 100} amount={currentRisk * 100}
maxDecimals={0} options={{
minDecimals={0} maxDecimals: 0,
minDecimals: 0,
prefix: 'Risk score: ',
suffix: '/100',
}}
animate animate
prefix='Risk Score: '
suffix='/100'
/> />
<div className='-ml-6 h-[100px] w-[412px]'> <div className='-ml-6 h-[100px] w-[412px]'>
<ResponsiveContainer width='100%' height='100%'> <ResponsiveContainer width='100%' height='100%'>
@ -77,7 +79,9 @@ export const RiskChart = ({ data }: RiskChartProps) => {
return ( return (
<div className='max-w-[320px] rounded-lg px-4 py-2 shadow-tooltip gradient-tooltip '> <div className='max-w-[320px] rounded-lg px-4 py-2 shadow-tooltip gradient-tooltip '>
<Text size='sm'>{moment(label).format('MM-DD-YYYY')}</Text> <Text size='sm'>{moment(label).format('MM-DD-YYYY')}</Text>
<Text size='sm'>Risk: {formatValue(risk, 0, 0, true, false, '%')}</Text> <Text size='sm'>
Risk: {formatValue(risk, { minDecimals: 0, maxDecimals: 0, suffix: '%' })}
</Text>
</div> </div>
) )
} }

View File

@ -5,7 +5,7 @@ import React, { useEffect, useMemo, useState } from 'react'
import { toast } from 'react-toastify' import { toast } from 'react-toastify'
import { BorrowCapacity } from 'components/BorrowCapacity' import { BorrowCapacity } from 'components/BorrowCapacity'
import { convertFromGwei, formatValue } from 'utils/formatters' import { convertFromGwei, formatLeverage, formatValue } from 'utils/formatters'
import { getTokenDecimals, getTokenSymbol } from 'utils/tokens' import { getTokenDecimals, getTokenSymbol } from 'utils/tokens'
import { CircularProgress } from 'components/CircularProgress' import { CircularProgress } from 'components/CircularProgress'
import { Button } from 'components/Button' import { Button } from 'components/Button'
@ -17,7 +17,6 @@ import { LabelValuePair } from 'components/LabelValuePair'
import { Modal } from 'components/Modal' import { Modal } from 'components/Modal'
import { PositionsList } from 'components/PositionsList' import { PositionsList } from 'components/PositionsList'
import { useAccountStats } from 'hooks/data/useAccountStats' import { useAccountStats } from 'hooks/data/useAccountStats'
import { useBalances } from 'hooks/data/useBalances'
import { useCalculateMaxWithdrawAmount } from 'hooks/data/useCalculateMaxWithdrawAmount' import { useCalculateMaxWithdrawAmount } from 'hooks/data/useCalculateMaxWithdrawAmount'
import { useWithdrawFunds } from 'hooks/mutations/useWithdrawFunds' import { useWithdrawFunds } from 'hooks/mutations/useWithdrawFunds'
import { useCreditAccountPositions } from 'hooks/queries/useCreditAccountPositions' import { useCreditAccountPositions } from 'hooks/queries/useCreditAccountPositions'
@ -46,7 +45,6 @@ export const WithdrawModal = () => {
// EXTERNAL HOOKS // EXTERNAL HOOKS
// --------------- // ---------------
const { data: tokenPrices } = useTokenPrices() const { data: tokenPrices } = useTokenPrices()
const balances = useBalances()
const selectedTokenSymbol = getTokenSymbol(selectedToken, marketAssets) const selectedTokenSymbol = getTokenSymbol(selectedToken, marketAssets)
const selectedTokenDecimals = getTokenDecimals(selectedToken, marketAssets) const selectedTokenDecimals = getTokenDecimals(selectedToken, marketAssets)
@ -153,7 +151,7 @@ export const WithdrawModal = () => {
} }
return ( return (
<Modal open={open} setOpen={setOpen}> <Modal title='Withdraw' open={open} setOpen={setOpen}>
<div className='flex min-h-[470px] w-full flex-wrap'> <div className='flex min-h-[470px] w-full flex-wrap'>
{isLoading && ( {isLoading && (
<div className='absolute inset-0 z-40 grid place-items-center bg-black/50'> <div className='absolute inset-0 z-40 grid place-items-center bg-black/50'>
@ -205,7 +203,7 @@ export const WithdrawModal = () => {
</div> </div>
</div> </div>
<Text size='xs' uppercase className='mb-2 text-white/60'> <Text size='xs' uppercase className='mb-2 text-white/60'>
Available: {formatValue(maxWithdrawAmount, 0, 4, true, false, false, false, false)} Available: {formatValue(maxWithdrawAmount, { minDecimals: 0, maxDecimals: 4 })}
</Text> </Text>
<Slider <Slider
className='mb-6' className='mb-6'
@ -265,7 +263,7 @@ export const WithdrawModal = () => {
amount={BigNumber(accountStats.netWorth) amount={BigNumber(accountStats.netWorth)
.dividedBy(10 ** baseAsset.decimals) .dividedBy(10 ** baseAsset.decimals)
.toNumber()} .toNumber()}
prefix='$: ' options={{ prefix: '$: ' }}
animate animate
/> />
</Text> </Text>
@ -275,11 +273,9 @@ export const WithdrawModal = () => {
label='Lvg' label='Lvg'
tooltip={ tooltip={
<Text size='sm'> <Text size='sm'>
Current Leverage:{' '} Current Leverage: {formatLeverage(accountStats.currentLeverage)}
{formatValue(accountStats.currentLeverage, 0, 2, true, false, 'x')}
<br /> <br />
Max Leverage:{' '} Max Leverage: {formatLeverage(accountStats.maxLeverage)}
{formatValue(accountStats.maxLeverage, 0, 2, true, false, 'x')}
</Text> </Text>
} }
/> />
@ -288,7 +284,8 @@ export const WithdrawModal = () => {
label='Risk' label='Risk'
tooltip={ tooltip={
<Text size='sm'> <Text size='sm'>
Current Risk: {formatValue(accountStats.risk * 100, 0, 2, true, false, '%')} Current Risk:{' '}
{formatValue(accountStats.risk * 100, { minDecimals: 0, suffix: '%' })}
</Text> </Text>
} }
/> />
@ -331,7 +328,6 @@ export const WithdrawModal = () => {
}} }}
/> />
</div> </div>
<PositionsList title='Balances' data={balances} />
</div> </div>
</div> </div>
</div> </div>

View File

@ -0,0 +1,27 @@
import { FormattedNumber } from './FormattedNumber'
import TitleAndSubCell from './TitleAndSubCell'
interface Props {
asset: Asset
amount: string
}
export default function AmountAndValue(props: Props) {
return (
<TitleAndSubCell
title={
<FormattedNumber
amount={props.amount}
options={{ decimals: props.asset.decimals, abbreviated: true }}
/>
}
sub={
<FormattedNumber
amount={props.amount}
options={{ prefix: '$', abbreviated: true, decimals: props.asset.decimals }}
/>
}
className='justify-end'
/>
)
}

View File

@ -1,11 +1,14 @@
'use client'
import React from 'react' import React from 'react'
import { Row } from '@tanstack/react-table' import { Row } from '@tanstack/react-table'
import { getMarketAssets } from 'utils/assets' import { getMarketAssets } from 'utils/assets'
import { Button } from 'components/Button' import { Button } from 'components/Button'
import useStore from 'store'
type AssetRowProps = { type AssetRowProps = {
row: Row<BorrowAsset> row: Row<BorrowAsset | BorrowAssetActive>
onBorrowClick: () => void onBorrowClick: () => void
onRepayClick: () => void onRepayClick: () => void
resetExpanded: (defaultState?: boolean | undefined) => void resetExpanded: (defaultState?: boolean | undefined) => void
@ -14,9 +17,22 @@ type AssetRowProps = {
export default function AssetExpanded(props: AssetRowProps) { export default function AssetExpanded(props: AssetRowProps) {
const marketAssets = getMarketAssets() const marketAssets = getMarketAssets()
const asset = marketAssets.find((asset) => asset.denom === props.row.original.denom) const asset = marketAssets.find((asset) => asset.denom === props.row.original.denom)
let isActive: boolean = false
if ((props.row.original as BorrowAssetActive)?.debt) {
isActive = true
}
if (!asset) return null if (!asset) return null
function borrowHandler() {
useStore.setState({ borrowModal: true })
}
function repayHandler() {
useStore.setState({ repayModal: true })
}
return ( return (
<tr <tr
key={props.row.id} key={props.row.id}
@ -28,10 +44,14 @@ export default function AssetExpanded(props: AssetRowProps) {
!isExpanded && props.row.toggleExpanded() !isExpanded && props.row.toggleExpanded()
}} }}
> >
<td colSpan={4}> <td colSpan={isActive ? 5 : 4}>
<div className='flex justify-end p-4'> <div className='flex justify-end p-4'>
<Button color='secondary' text='CLick me' onClick={() => {}} /> <Button
<Button color='primary' text='CLick me' /> onClick={borrowHandler}
color='primary'
text={isActive ? 'Borrow more' : 'Borrow'}
/>
{isActive && <Button color='primary' text='Repay' />}
</div> </div>
</td> </td>
</tr> </tr>

View File

@ -4,7 +4,7 @@ import { flexRender, Row } from '@tanstack/react-table'
import { getMarketAssets } from 'utils/assets' import { getMarketAssets } from 'utils/assets'
type AssetRowProps = { type AssetRowProps = {
row: Row<BorrowAsset> row: Row<BorrowAsset | BorrowAssetActive>
resetExpanded: (defaultState?: boolean | undefined) => void resetExpanded: (defaultState?: boolean | undefined) => void
} }

View File

@ -15,6 +15,11 @@ import classNames from 'classnames'
import { AssetRow } from 'components/Borrow/AssetRow' import { AssetRow } from 'components/Borrow/AssetRow'
import { ChevronDown, ChevronUp } from 'components/Icons' import { ChevronDown, ChevronUp } from 'components/Icons'
import { getMarketAssets } from 'utils/assets' import { getMarketAssets } from 'utils/assets'
import { Text } from 'components/Text'
import TitleAndSubCell from 'components/TitleAndSubCell'
import { FormattedNumber } from 'components/FormattedNumber'
import AmountAndValue from 'components/AmountAndValue'
import { formatPercent } from 'utils/formatters'
import AssetExpanded from './AssetExpanded' import AssetExpanded from './AssetExpanded'
@ -26,7 +31,7 @@ export const BorrowTable = (props: Props) => {
const [sorting, setSorting] = React.useState<SortingState>([]) const [sorting, setSorting] = React.useState<SortingState>([])
const marketAssets = getMarketAssets() const marketAssets = getMarketAssets()
const columns = React.useMemo<ColumnDef<BorrowAsset>[]>( const columns = React.useMemo<ColumnDef<BorrowAsset | BorrowAssetActive>[]>(
() => [ () => [
{ {
header: 'Asset', header: 'Asset',
@ -37,12 +42,9 @@ export const BorrowTable = (props: Props) => {
if (!asset) return null if (!asset) return null
return ( return (
<div className='flex flex-1 items-center'> <div className='flex flex-1 items-center gap-3'>
<Image src={asset.logo} alt='token' width={32} height={32} /> <Image src={asset.logo} alt='token' width={32} height={32} />
<div className='pl-2'> <TitleAndSubCell title={asset.symbol} sub={asset.name} />
<div>{asset.symbol}</div>
<div className='text-xs'>{asset.name}</div>
</div>
</div> </div>
) )
}, },
@ -50,17 +52,38 @@ export const BorrowTable = (props: Props) => {
{ {
accessorKey: 'borrowRate', accessorKey: 'borrowRate',
header: 'Borrow Rate', header: 'Borrow Rate',
cell: ({ row }) => <div>{(Number(row.original.borrowRate) * 100).toFixed(2)}%</div>, cell: ({ row }) => (
<Text className='justify-end' size='sm'>
{formatPercent(row.original.borrowRate)}
</Text>
),
}, },
...((props.data[0] as BorrowAssetActive)?.debt
? [
{
accessorKey: 'debt',
header: 'Borrowed',
cell: (info: any) => {
const borrowAsset = info.row.original as BorrowAssetActive
const asset = marketAssets.find((asset) => asset.denom === borrowAsset.denom)
if (!asset) return null
return <AmountAndValue asset={asset} amount={borrowAsset.debt} />
},
},
]
: []),
{ {
accessorKey: 'liquidity', accessorKey: 'liquidity',
header: 'Liquidity Available', header: 'Liquidity Available',
cell: ({ row }) => ( cell: ({ row }) => {
<div className='items-right flex flex-col'> const asset = marketAssets.find((asset) => asset.denom === row.original.denom)
<div className=''>{row.original.liquidity.amount}</div>
<div className='text-xs opacity-60'>${row.original.liquidity.value}</div> if (!asset) return null
</div>
), return <AmountAndValue asset={asset} amount={row.original.liquidity.amount} />
},
}, },
{ {
accessorKey: 'status', accessorKey: 'status',
@ -91,7 +114,7 @@ export const BorrowTable = (props: Props) => {
return ( return (
<table className='w-full'> <table className='w-full'>
<thead className='bg-white/5'> <thead className='bg-black/20'>
{table.getHeaderGroups().map((headerGroup) => ( {table.getHeaderGroups().map((headerGroup) => (
<tr key={headerGroup.id}> <tr key={headerGroup.id}>
{headerGroup.headers.map((header, index) => { {headerGroup.headers.map((header, index) => {
@ -100,7 +123,7 @@ export const BorrowTable = (props: Props) => {
key={header.id} key={header.id}
onClick={header.column.getToggleSortingHandler()} onClick={header.column.getToggleSortingHandler()}
className={classNames( className={classNames(
'px-4 py-2', 'px-4 py-3',
header.column.getCanSort() && 'cursor-pointer', header.column.getCanSort() && 'cursor-pointer',
header.id === 'symbol' ? 'text-left' : 'text-right', header.id === 'symbol' ? 'text-left' : 'text-right',
)} )}
@ -109,6 +132,7 @@ export const BorrowTable = (props: Props) => {
className={classNames( className={classNames(
'flex', 'flex',
header.id === 'symbol' ? 'justify-start' : 'justify-end', header.id === 'symbol' ? 'justify-start' : 'justify-end',
'align-center',
)} )}
> >
{header.column.getCanSort() {header.column.getCanSort()
@ -124,7 +148,9 @@ export const BorrowTable = (props: Props) => {
), ),
}[header.column.getIsSorted() as string] ?? null }[header.column.getIsSorted() as string] ?? null
: null} : null}
<span>{flexRender(header.column.columnDef.header, header.getContext())}</span> <Text tag='span' size='sm' className='font-normal text-white/40'>
{flexRender(header.column.columnDef.header, header.getContext())}
</Text>
</div> </div>
</th> </th>
) )

View File

@ -121,11 +121,12 @@ export const BorrowCapacity = ({
<FormattedNumber <FormattedNumber
className='text-white' className='text-white'
animate animate
options={{
minDecimals: decimals,
maxDecimals: decimals,
suffix: '%',
}}
amount={percentOfMaxRound} amount={percentOfMaxRound}
minDecimals={decimals}
maxDecimals={decimals}
suffix='%'
abbreviated={false}
/> />
)} )}
</span> </span>

View File

@ -1,314 +1,25 @@
import { Dialog, Switch, Transition } from '@headlessui/react'
import BigNumber from 'bignumber.js'
import React, { useMemo, useState } from 'react'
import { NumericFormat } from 'react-number-format'
import { Button } from 'components/Button'
import { CircularProgress } from 'components/CircularProgress'
import { ContainerSecondary } from 'components/ContainerSecondary'
import { Gauge } from 'components/Gauge'
import { PositionsList } from 'components/PositionsList'
import { ProgressBar } from 'components/ProgressBar'
import { Slider } from 'components/Slider'
import { Text } from 'components/Text'
import { Tooltip } from 'components/Tooltip'
import { useAccountStats } from 'hooks/data/useAccountStats'
import { useBalances } from 'hooks/data/useBalances'
import { useCalculateMaxBorrowAmount } from 'hooks/data/useCalculateMaxBorrowAmount'
import { useBorrowFunds } from 'hooks/mutations/useBorrowFunds'
import { useAllBalances } from 'hooks/queries/useAllBalances'
import { useMarkets } from 'hooks/queries/useMarkets'
import { useTokenPrices } from 'hooks/queries/useTokenPrices'
import useStore from 'store' import useStore from 'store'
import { getBaseAsset, getMarketAssets } from 'utils/assets'
import { formatCurrency, formatValue } from 'utils/formatters'
import { getTokenDecimals, getTokenSymbol } from 'utils/tokens'
type Props = { import { Modal } from './Modal'
show: boolean import TitleAndSubCell from './TitleAndSubCell'
onClose: () => void
tokenDenom: string
}
export const BorrowModal = ({ show, onClose, tokenDenom }: Props) => { export default function BorrowModal() {
const [amount, setAmount] = useState(0) const open = useStore((s) => s.borrowModal)
const [isBorrowToCreditAccount, setIsBorrowToCreditAccount] = useState(false)
const selectedAccount = useStore((s) => s.selectedAccount) function setOpen(isOpen: boolean) {
const marketAssets = getMarketAssets() useStore.setState({ borrowModal: isOpen })
const baseAsset = getBaseAsset()
const balances = useBalances()
const { actions, borrowAmount } = useMemo(() => {
const borrowAmount = BigNumber(amount)
.times(10 ** getTokenDecimals(tokenDenom, marketAssets))
.toNumber()
const withdrawAmount = isBorrowToCreditAccount ? 0 : borrowAmount
return {
borrowAmount,
withdrawAmount,
actions: [
{
type: 'borrow',
amount: borrowAmount,
denom: tokenDenom,
},
{
type: 'withdraw',
amount: withdrawAmount,
denom: tokenDenom,
},
] as AccountStatsAction[],
}
}, [amount, isBorrowToCreditAccount, tokenDenom, marketAssets])
const accountStats = useAccountStats(actions)
const tokenSymbol = getTokenSymbol(tokenDenom, marketAssets)
const { mutate, isLoading } = useBorrowFunds(borrowAmount, tokenDenom, !isBorrowToCreditAccount, {
onSuccess: () => {
onClose()
useStore.setState({
toast: { message: `${amount} ${tokenSymbol} successfully Borrowed` },
})
},
})
const { data: tokenPrices } = useTokenPrices()
const { data: balancesData } = useAllBalances()
const { data: marketsData } = useMarkets()
const handleSubmit = () => {
mutate()
}
const walletAmount = useMemo(() => {
return BigNumber(balancesData?.find((balance) => balance.denom === tokenDenom)?.amount ?? 0)
.div(10 ** getTokenDecimals(tokenDenom, marketAssets))
.toNumber()
}, [balancesData, tokenDenom, marketAssets])
const tokenPrice = tokenPrices?.[tokenDenom] ?? 0
const borrowRate = Number(marketsData?.[tokenDenom]?.borrow_rate)
const maxValue = useCalculateMaxBorrowAmount(tokenDenom, isBorrowToCreditAccount)
const percentageValue = useMemo(() => {
if (isNaN(amount) || maxValue === 0) return 0
return (amount * 100) / maxValue
}, [amount, maxValue])
const handleValueChange = (value: number) => {
if (value > maxValue) {
setAmount(maxValue)
return
}
setAmount(value)
}
const handleSliderValueChange = (value: number[]) => {
const decimal = value[0] / 100
const tokenDecimals = getTokenDecimals(tokenDenom, marketAssets)
// limit decimal precision based on token contract decimals
const newAmount = Number((decimal * maxValue).toFixed(tokenDecimals))
setAmount(newAmount)
}
const handleBorrowTargetChange = () => {
setIsBorrowToCreditAccount((c) => !c)
// reset amount due to max value calculations changing depending on borrow target
setAmount(0)
} }
return ( return (
<Transition appear show={show} as={React.Fragment}> <Modal open={open} setOpen={setOpen} title='Borrow OSMO'>
<Dialog as='div' className='relative z-10' onClose={onClose}> <div className='flex gap-3'>
<Transition.Child <TitleAndSubCell title='10.00%' sub={'Borrow rate'} />
as={React.Fragment} <div className='h-100 w-[1px] bg-white/10'></div>
enter='ease-out duration-300' <TitleAndSubCell title='$200' sub={'Borrowed'} />
enterFrom='opacity-0' <div className='h-100 w-[1px] bg-white/10'></div>
enterTo='opacity-100' <TitleAndSubCell title='10.5M ($105M)' sub={'Liquidity available'} />
leave='ease-in duration-200' </div>
leaveFrom='opacity-100' <div className='flex'></div>
leaveTo='opacity-0' </Modal>
>
<div className='fixed inset-0 bg-black bg-opacity-80' />
</Transition.Child>
<div className='fixed inset-0 overflow-y-auto'>
<div className='flex min-h-full items-center justify-center p-4'>
<Transition.Child
as={React.Fragment}
enter='ease-out duration-300'
enterFrom='opacity-0 scale-95'
enterTo='opacity-100 scale-100'
leave='ease-in duration-200'
leaveFrom='opacity-100 scale-100'
leaveTo='opacity-0 scale-95'
>
<Dialog.Panel className='flex w-full max-w-3xl transform overflow-hidden rounded-2xl bg-[#585A74] align-middle shadow-xl transition-all'>
{isLoading && (
<div className='absolute inset-0 z-40 grid place-items-center bg-black/50'>
<CircularProgress />
</div>
)}
<div className='flex flex-1 flex-col p-4'>
<Dialog.Title as='h3' className='mb-4 text-center font-medium'>
Borrow {tokenSymbol}
</Dialog.Title>
<div className='mb-4 flex flex-col gap-2 text-sm'>
<ContainerSecondary>
<p className='mb-1'>
In wallet: {walletAmount.toLocaleString()} {tokenSymbol}
</p>
<p className='mb-5'>Borrow Rate: {(borrowRate * 100).toFixed(2)}%</p>
<div className='mb-7'>
<p className='mb-2 font-semibold uppercase tracking-widest'>Amount</p>
<NumericFormat
className='mb-2 h-[32px] w-full rounded-lg border border-black/50 bg-transparent px-2'
value={amount}
placeholder='0'
allowNegative={false}
onValueChange={(v) => handleValueChange(v.floatValue || 0)}
suffix={` ${tokenSymbol}`}
decimalScale={getTokenDecimals(tokenDenom, marketAssets)}
/>
<div className='flex justify-between text-xs tracking-widest'>
<div>
1 {tokenSymbol} = {formatCurrency(tokenPrice)}
</div>
<div>{formatCurrency(tokenPrice * amount)}</div>
</div>
</div>
<Slider
className='mb-6'
value={percentageValue}
onChange={handleSliderValueChange}
onMaxClick={() => setAmount(maxValue)}
/>
</ContainerSecondary>
<ContainerSecondary className='flex items-center justify-between'>
<div className='flex'>
Borrow to Credit Account{' '}
<Tooltip
className='ml-2'
content={
<>
<Text size='sm' className='mb-2'>
OFF = Borrow directly into your wallet by using your account Assets
as collateral. The borrowed asset will become a liability in your
account.
</Text>
<Text size='sm'>
ON = Borrow into your Account. The borrowed asset will be available
in the account as an Asset and appear also as a liability in your
account.
</Text>
</>
}
/>
</div>
<Switch
checked={isBorrowToCreditAccount}
onChange={handleBorrowTargetChange}
className={`${
isBorrowToCreditAccount ? 'bg-blue-600' : 'bg-gray-400'
} relative inline-flex h-6 w-11 items-center rounded-full`}
>
<span
className={`${
isBorrowToCreditAccount ? 'translate-x-6' : 'translate-x-1'
} inline-block h-4 w-4 transform rounded-full bg-white transition`}
/>
</Switch>
</ContainerSecondary>
</div>
<Button
className='mt-auto'
onClick={handleSubmit}
disabled={amount === 0 || !amount}
>
Borrow
</Button>
</div>
<div className='flex w-1/2 flex-col justify-center bg-[#4A4C60] p-4'>
<p className='text-bold mb-3 text-xs uppercase text-white/50'>About</p>
<h4 className='mb-4 text-xl'>Account {selectedAccount}</h4>
<div className='mb-2 rounded-md border border-white/20 p-3'>
{accountStats && (
<div className='flex items-center gap-x-3'>
<p className='flex-1 text-xs'>
{formatCurrency(
BigNumber(accountStats.netWorth)
.dividedBy(10 ** baseAsset.decimals)
.toNumber(),
)}
</p>
<Gauge
value={accountStats.currentLeverage / accountStats.maxLeverage}
label='Lvg'
tooltip={
<Text size='sm'>
Current Leverage:{' '}
{formatValue(accountStats.currentLeverage, 0, 2, true, false, 'x')}
<br />
Max Leverage:{' '}
{formatValue(accountStats.maxLeverage, 0, 2, true, false, 'x')}
</Text>
}
/>
<Gauge
value={accountStats.risk}
label='Risk'
tooltip={
<Text size='sm'>
Current Risk:{' '}
{formatValue(accountStats.risk * 100, 0, 2, true, false, '%')}
</Text>
}
/>
<ProgressBar value={accountStats.health} />
</div>
)}
</div>
<div className='mb-2 rounded-md border border-white/20 p-3 text-sm'>
<div className='mb-1 flex justify-between'>
<div>Total Position:</div>
<div className='font-semibold'>
{formatCurrency(
BigNumber(accountStats?.totalPosition ?? 0)
.dividedBy(10 ** baseAsset.decimals)
.toNumber(),
)}
</div>
</div>
<div className='flex justify-between'>
<div>Total Liabilities:</div>
<div className='font-semibold'>
{formatCurrency(
BigNumber(accountStats?.totalDebt ?? 0)
.dividedBy(10 ** baseAsset.decimals)
.toNumber(),
)}
</div>
</div>
</div>
<PositionsList title='Balances' data={balances} />
</div>
</Dialog.Panel>
</Transition.Child>
</div>
</div>
</Dialog>
</Transition>
) )
} }

View File

@ -1,6 +1,8 @@
import classNames from 'classnames' import classNames from 'classnames'
import { ReactNode } from 'react' import { ReactNode } from 'react'
import { Text } from 'components/Text'
interface Props { interface Props {
title: string title: string
children: ReactNode children: ReactNode
@ -15,8 +17,10 @@ export const Card = (props: Props) => {
'h-fit w-full max-w-full overflow-hidden rounded-md border-[1px] border-white/20', 'h-fit w-full max-w-full overflow-hidden rounded-md border-[1px] border-white/20',
)} )}
> >
<div className='bg-white/10 p-4 font-semibold'>{props.title}</div> <Text size='lg' className='bg-white/10 p-4 font-semibold'>
<div className=''>{props.children}</div> {props.title}
</Text>
<div>{props.children}</div>
</section> </section>
) )
} }

View File

@ -0,0 +1,15 @@
'use client'
import useSWR from 'swr'
import useStore from 'store'
import { getPrices } from 'utils/api'
export default function FetchPrices() {
useSWR('prices', getPrices, {
refreshInterval: 30000,
onSuccess: (prices) => useStore.setState({ prices }),
})
return null
}

View File

@ -5,64 +5,60 @@ import React, { useEffect, useRef } from 'react'
import { animated, useSpring } from 'react-spring' import { animated, useSpring } from 'react-spring'
import useStore from 'store' import useStore from 'store'
import { formatValue } from 'utils/formatters' import { FormatOptions, formatValue } from 'utils/formatters'
export const FormattedNumber = React.memo( interface Props {
({ amount: number | string
amount, options?: FormatOptions
animate = false, className?: string
className, animate?: boolean
minDecimals = 2, }
maxDecimals = 2,
thousandSeparator = true,
prefix = false,
suffix = false,
rounded = false,
abbreviated = false,
}: FormattedNumberProps) => {
const enableAnimations = useStore((s) => s.enableAnimations)
const prevAmountRef = useRef<number>(0)
useEffect(() => { export const FormattedNumber = React.memo((props: Props) => {
if (prevAmountRef.current !== Number(amount)) prevAmountRef.current = Number(amount) const enableAnimations = useStore((s) => s.enableAnimations)
}, [amount]) const prevAmountRef = useRef<number>(0)
const springAmount = useSpring({ useEffect(() => {
number: Number(amount), if (prevAmountRef.current !== Number(props.amount)) prevAmountRef.current = Number(props.amount)
from: { number: prevAmountRef.current }, }, [props.amount])
config: { duration: 1000 },
})
return (prevAmountRef.current === amount && amount === 0) || !animate || !enableAnimations ? ( const springAmount = useSpring({
<span className={classNames('number', className)}> number: Number(props.amount),
{formatValue( from: { number: prevAmountRef.current },
amount, config: { duration: 1000 },
minDecimals, })
maxDecimals,
thousandSeparator, return (prevAmountRef.current === props.amount && props.amount === 0) ||
prefix, !props.animate ||
suffix, !enableAnimations ? (
rounded, <span className={classNames('number', props.className)}>
abbreviated, {formatValue(props.amount, {
)} minDecimals: props.options?.minDecimals,
</span> maxDecimals: props.options?.maxDecimals,
) : ( thousandSeparator: props.options?.thousandSeparator,
<animated.span className={classNames('number', className)}> prefix: props.options?.prefix,
{springAmount.number.to((num) => suffix: props.options?.suffix,
formatValue( rounded: props.options?.rounded,
num, abbreviated: props.options?.abbreviated,
minDecimals, decimals: props.options?.decimals,
maxDecimals, })}
thousandSeparator, </span>
prefix, ) : (
suffix, <animated.span className={classNames('number', props.className)}>
rounded, {springAmount.number.to((num) =>
abbreviated, formatValue(num, {
), minDecimals: props.options?.minDecimals,
)} maxDecimals: props.options?.maxDecimals,
</animated.span> thousandSeparator: props.options?.thousandSeparator,
) prefix: props.options?.prefix,
}, suffix: props.options?.suffix,
) rounded: props.options?.rounded,
abbreviated: props.options?.abbreviated,
decimals: props.options?.decimals,
}),
)}
</animated.span>
)
})
FormattedNumber.displayName = 'FormattedNumber' FormattedNumber.displayName = 'FormattedNumber'

View File

@ -2,9 +2,12 @@ import classNames from 'classnames'
import { ReactNode } from 'react' import { ReactNode } from 'react'
import { Close } from 'components/Icons' import { Close } from 'components/Icons'
import { Card } from 'components/Card' import { Text } from 'components/Text'
import { Button } from './Button'
interface Props { interface Props {
title: string
children?: ReactNode | string children?: ReactNode | string
content?: ReactNode | string content?: ReactNode | string
className?: string className?: string
@ -12,31 +15,28 @@ interface Props {
setOpen?: (open: boolean) => void setOpen?: (open: boolean) => void
} }
export const Modal = ({ children, content, className, open, setOpen }: Props) => { export const Modal = (props: Props) => {
const onClickAway = () => { const onClickAway = () => {
if (setOpen) setOpen(false) if (props.setOpen) props.setOpen(false)
} }
return open ? ( return props.open ? (
<div className='fixed top-0 left-0 z-20 h-screen w-screen'> <div className='fixed top-0 left-0 z-20 h-screen w-screen'>
<div className='relative flex h-full w-full items-center justify-center'> <div className='relative flex h-full w-full items-center justify-center'>
<Card <section
title='Modal' className={classNames(
className={classNames('relative z-40 w-[790px] max-w-full p-0', className)} 'relative z-40 w-[790px] max-w-full rounded-md border-[1px] border-white/20 bg-white/5 p-6 backdrop-blur-3xl ',
> props.className,
{setOpen && (
<span
className='absolute top-4 right-4 z-50 w-[32px] text-white opacity-60 hover:cursor-pointer hover:opacity-100'
onClick={onClickAway}
role='button'
>
<Close />
</span>
)} )}
{children ? children : content} >
</Card> <div className='flex justify-between pb-6'>
<Text>{props.title}</Text>
<Button onClick={onClickAway} text='X' color='tertiary' />
</div>
<div>{props.children ? props.children : props.content}</div>
</section>
<div <div
className='fixed top-0 left-0 z-30 block h-full w-full bg-black/70 backdrop-blur hover:cursor-pointer' className='fixed top-0 left-0 z-30 block h-full w-full bg-black/50 hover:cursor-pointer'
onClick={onClickAway} onClick={onClickAway}
role='button' role='button'
/> />

View File

@ -3,10 +3,13 @@
import { ConfirmModal } from 'components/Account/ConfirmModal' import { ConfirmModal } from 'components/Account/ConfirmModal'
import { FundAccountModal } from 'components/Account/FundAccountModal' import { FundAccountModal } from 'components/Account/FundAccountModal'
import BorrowModal from './BorrowModal'
export const Modals = () => ( export const Modals = () => (
<> <>
<FundAccountModal /> <FundAccountModal />
{/* <WithdrawModal /> */} {/* <WithdrawModal /> */}
<ConfirmModal /> <ConfirmModal />
<BorrowModal />
</> </>
) )

View File

@ -13,7 +13,6 @@ import { useRepayFunds } from 'hooks/mutations/useRepayFunds'
import { useAllBalances } from 'hooks/queries/useAllBalances' import { useAllBalances } from 'hooks/queries/useAllBalances'
import { useCreditAccountPositions } from 'hooks/queries/useCreditAccountPositions' import { useCreditAccountPositions } from 'hooks/queries/useCreditAccountPositions'
import { useTokenPrices } from 'hooks/queries/useTokenPrices' import { useTokenPrices } from 'hooks/queries/useTokenPrices'
import { formatCurrency } from 'utils/formatters'
import { getTokenDecimals, getTokenSymbol } from 'utils/tokens' import { getTokenDecimals, getTokenSymbol } from 'utils/tokens'
import { getMarketAssets } from 'utils/assets' import { getMarketAssets } from 'utils/assets'
import useStore from 'store' import useStore from 'store'
@ -33,6 +32,7 @@ export const RepayModal = ({ show, onClose, tokenDenom }: Props) => {
const selectedAccount = useStore((s) => s.selectedAccount) const selectedAccount = useStore((s) => s.selectedAccount)
const { data: positionsData } = useCreditAccountPositions(selectedAccount ?? '') const { data: positionsData } = useCreditAccountPositions(selectedAccount ?? '')
const marketAssets = getMarketAssets() const marketAssets = getMarketAssets()
const formatCurrency = useStore((s) => s.formatCurrency)
const tokenSymbol = getTokenSymbol(tokenDenom, marketAssets) const tokenSymbol = getTokenSymbol(tokenDenom, marketAssets)
@ -155,9 +155,11 @@ export const RepayModal = ({ show, onClose, tokenDenom }: Props) => {
/> />
<div className='flex justify-between text-xs tracking-widest'> <div className='flex justify-between text-xs tracking-widest'>
<div> <div>
1 {tokenSymbol} = {formatCurrency(tokenPrice)} 1 {tokenSymbol} = {formatCurrency({ denom: tokenDenom, amount: '1' })}
</div>
<div>
{formatCurrency({ denom: tokenDenom, amount: amount.toString() })}
</div> </div>
<div>{formatCurrency(tokenPrice * amount)}</div>
</div> </div>
</div> </div>

View File

@ -29,6 +29,7 @@ export const Text = ({
<HtmlElement <HtmlElement
className={classNames( className={classNames(
className, className,
'flex items-center',
uppercase ? `text-${sizeClass}-caps` : `text-${sizeClass}`, uppercase ? `text-${sizeClass}-caps` : `text-${sizeClass}`,
monospace && 'number', monospace && 'number',
)} )}

View File

@ -0,0 +1,20 @@
import { Text } from 'components/Text'
interface Props {
title: string | React.ReactNode
sub: string | React.ReactNode
className?: string
}
export default function TitleAndSubCell(props: Props) {
return (
<div className='flex flex-col gap-[0.5]'>
<Text className={props.className} size='sm'>
{props.title}
</Text>
<Text size='sm' className={'text-white/50 ' + props.className}>
{props.sub}
</Text>
</div>
)
}

View File

@ -104,7 +104,7 @@ export default function ConnectedButton() {
{isLoading ? ( {isLoading ? (
<CircularProgress size={12} /> <CircularProgress size={12} />
) : ( ) : (
`${formatValue(walletAmount, 2, 2, true, false, ` ${baseAsset.symbol}`)}` `${formatValue(walletAmount, { suffix: baseAsset.symbol })}`
)} )}
</div> </div>
</button> </button>

View File

@ -1,36 +0,0 @@
import { useEffect, useState } from 'react'
import { useCreditAccountPositions } from 'hooks/queries/useCreditAccountPositions'
import { useMarkets } from 'hooks/queries/useMarkets'
import { useTokenPrices } from 'hooks/queries/useTokenPrices'
import { formatBalances } from 'utils/balances'
import useStore from 'store'
import { getMarketAssets } from 'utils/assets'
export const useBalances = () => {
const [balanceData, setBalanceData] = useState<PositionsData[]>()
const { data: marketsData } = useMarkets()
const { data: tokenPrices } = useTokenPrices()
const selectedAccount = useStore((s) => s.selectedAccount)
const marketAssets = getMarketAssets()
const { data: positionsData, isLoading: isLoadingPositions } = useCreditAccountPositions(
selectedAccount ?? '',
)
useEffect(() => {
const balances =
positionsData?.coins && tokenPrices
? formatBalances(positionsData.coins, tokenPrices, false, marketAssets)
: []
const debtBalances =
positionsData?.debts && tokenPrices
? formatBalances(positionsData.debts, tokenPrices, true, marketAssets, marketsData)
: []
setBalanceData([...balances, ...debtBalances])
}, [positionsData, marketsData, tokenPrices, marketAssets])
return balanceData
}

View File

@ -12,7 +12,7 @@ export default async function handler(req: NextApiRequest, res: NextApiResponse)
const account = await (await fetch(`${URL_API}/accounts/${accountId}`)).json() const account = await (await fetch(`${URL_API}/accounts/${accountId}`)).json()
if (account) { if (account) {
return res.status(200).json(account.debts) return res.status(200).json([{ denom: 'uosmo', amount: '123876' }])
} }
return res.status(404) return res.status(404)

View File

@ -1,23 +1,52 @@
import { Coin } from '@cosmjs/stargate'
import BigNumber from 'bignumber.js'
import { GetState, SetState } from 'zustand' import { GetState, SetState } from 'zustand'
import { getMarketAssets } from 'utils/assets'
import { formatValue } from 'utils/formatters'
export interface CommonSlice { export interface CommonSlice {
borrowModal: boolean
createAccountModal: boolean createAccountModal: boolean
deleteAccountModal: boolean deleteAccountModal: boolean
enableAnimations: boolean enableAnimations: boolean
repayModal: boolean
fundAccountModal: boolean fundAccountModal: boolean
prices: Coin[]
isOpen: boolean isOpen: boolean
selectedAccount: string | null selectedAccount: string | null
withdrawModal: boolean withdrawModal: boolean
formatCurrency: (coin: Coin) => string
} }
export function createCommonSlice(set: SetState<CommonSlice>, get: GetState<CommonSlice>) { export function createCommonSlice(set: SetState<CommonSlice>, get: GetState<CommonSlice>) {
return { return {
borrowModal: false,
createAccountModal: false, createAccountModal: false,
deleteAccountModal: false, deleteAccountModal: false,
repayModal: false,
enableAnimations: true, enableAnimations: true,
fundAccountModal: false, fundAccountModal: false,
prices: [],
isOpen: true, isOpen: true,
selectedAccount: null, selectedAccount: null,
withdrawModal: false, withdrawModal: false,
formatCurrency: (coin: Coin) => {
const price = get().prices.find((price) => price.denom === coin.denom)
const marketAsset = getMarketAssets().find((asset) => asset.denom === coin.denom)
if (!price || !marketAsset) return ''
return formatValue(
new BigNumber(coin.amount)
.times(price.amount)
.dividedBy(10 ** marketAsset.decimals)
.toNumber(),
{
minDecimals: 0,
prefix: '$',
},
)
},
} }
} }

View File

@ -66,20 +66,21 @@ a {
} }
/* ORBS */ /* ORBS */
@mixin orbs($count) { @mixin orbs($count, $hue) {
$text-shadow: (); $text-shadow: ();
@for $i from 0 through $count { @for $i from 0 through $count {
$text-shadow: $text-shadow, $text-shadow: $text-shadow,
(-0.5+ (random()) * 3) + (-0.5+ (random()) * 3) + em (-0.5+ (random()) * 3) + em 10px rgb(92, 5, 92);
em // hsla((random() * 50)+$hue, 100%, 45%);
(-0.5+ (random()) * 3) +
em
7px
hsla((random() * 50)+210, 100%, 45%, 0.3);
} }
text-shadow: $text-shadow; text-shadow: $text-shadow;
} }
@mixin newOrbs($count, $color) {
filter: blur(4px);
background: radial-gradient(circle at center, rgba($color, 0.25) 0%, rgba($color, 0) 20%);
}
.background { .background {
font-family: serif; font-family: serif;
font-size: 90px; font-size: 90px;
@ -98,18 +99,27 @@ a {
width: 3em; width: 3em;
height: 3em; height: 3em;
content: '.'; content: '.';
color: transparent;
mix-blend-mode: screen; mix-blend-mode: screen;
} }
&:after {
top: 10%;
left: 10%;
@include newOrbs(1, rgb(177, 47, 37));
}
&:before { &:before {
@include orbs(15); top: 80%;
left: 80%;
@include newOrbs(1, rgb(83, 7, 129));
animation-duration: 300s; animation-duration: 300s;
animation-delay: -50s; // animation-delay: -50s;
animation: 180s -15s move infinite ease-in-out alternate; animation: 180s -15s move infinite ease-in-out alternate;
} }
&:after { &:after {
@include orbs(25); // @include orbs(15, 260);
animation-duration: 600s; animation-duration: 600s;
animation: 180s 0s move infinite ease-in-out alternate; animation: 180s 0s move infinite ease-in-out alternate;
} }

View File

@ -26,8 +26,5 @@ interface BorrowAsset {
} }
interface BorrowAssetActive extends BorrowAsset { interface BorrowAssetActive extends BorrowAsset {
debt: { debt: string
amount: string
value: string
}
} }

View File

@ -1,43 +0,0 @@
import { Coin } from '@cosmjs/stargate'
import { convertFromGwei, getTokenTotalUSDValue } from 'utils/formatters'
import { getTokenSymbol } from 'utils/tokens'
export const formatBalances = (
positionData: Coin[],
tokenPrices: KeyValuePair,
debt: boolean,
marketAssets: Asset[],
marketsData?: MarketData,
): PositionsData[] => {
const balances: PositionsData[] = []
positionData.forEach((coin) => {
const dataEntry: PositionsData = {
asset: {
amount: getTokenSymbol(coin.denom, marketAssets),
type: debt ? 'debt' : undefined,
},
value: {
amount: getTokenTotalUSDValue(coin.amount, coin.denom, marketAssets, tokenPrices),
format: 'number',
prefix: '$',
},
size: {
amount: convertFromGwei(coin.amount, coin.denom, marketAssets),
format: 'number',
maxDecimals: 4,
minDecimals: 0,
},
apy: {
amount: debt ? Number(marketsData?.[coin.denom].borrow_rate) * 100 : '-',
format: debt ? 'number' : 'string',
suffix: '%',
minDecimals: 0,
},
}
balances.push(dataEntry)
})
return balances
}

View File

@ -10,28 +10,6 @@ export function truncate(text = '', [h, t]: [number, number] = [6, 6]): string {
return text.length > h + t ? [head, tail].join('...') : text return text.length > h + t ? [head, tail].join('...') : text
} }
export const formatCurrency = (value: string | number) => {
return Number(value).toLocaleString('en-US', {
style: 'currency',
currency: 'USD',
})
}
export const getTokenTotalUSDValue = (
amount: string,
denom: string,
marketAssets: Asset[],
tokenPrices?: KeyValuePair,
) => {
if (!tokenPrices) return 0
return (
BigNumber(amount)
.div(10 ** getTokenDecimals(denom, marketAssets))
.toNumber() * tokenPrices[denom]
)
}
export const convertFromGwei = (amount: string | number, denom: string, marketAssets: Asset[]) => { export const convertFromGwei = (amount: string | number, denom: string, marketAssets: Asset[]) => {
return BigNumber(amount) return BigNumber(amount)
.div(10 ** getTokenDecimals(denom, marketAssets)) .div(10 ** getTokenDecimals(denom, marketAssets))
@ -44,26 +22,34 @@ export const convertToGwei = (amount: string | number, denom: string, marketAsse
.toNumber() .toNumber()
} }
export const formatValue = ( export interface FormatOptions {
amount: number | string, decimals?: number
minDecimals = 2, minDecimals?: number
maxDecimals = 2, maxDecimals?: number
thousandSeparator = true, thousandSeparator?: boolean
prefix: boolean | string = false, prefix?: string
suffix: boolean | string = false, suffix?: string
rounded = false, rounded?: boolean
abbreviated = false, abbreviated?: boolean
): string => { }
export const formatValue = (amount: number | string, options?: FormatOptions): string => {
let numberOfZeroDecimals: number | null = null let numberOfZeroDecimals: number | null = null
const minDecimals = options?.minDecimals ?? 2
const maxDecimals = options?.maxDecimals ?? 2
const thousandSeparator = options?.thousandSeparator ?? true
if (typeof amount === 'string') { if (typeof amount === 'string') {
const decimals = amount.split('.')[1] ?? null const decimals = amount.split('.')[1] ?? null
if (decimals && Number(decimals) === 0) { if (decimals && Number(decimals) === 0) {
numberOfZeroDecimals = decimals.length numberOfZeroDecimals = decimals.length
} }
} }
let convertedAmount: number | string = +amount || 0 let convertedAmount: number | string = new BigNumber(amount)
.dividedBy(10 ** (options?.decimals ?? 0))
.toNumber()
const amountSuffix = abbreviated const amountSuffix = options?.abbreviated
? convertedAmount >= 1_000_000_000 ? convertedAmount >= 1_000_000_000
? 'B' ? 'B'
: convertedAmount >= 1_000_000 : convertedAmount >= 1_000_000
@ -73,19 +59,17 @@ export const formatValue = (
: false : false
: '' : ''
const amountPrefix = prefix
if (amountSuffix === 'B') { if (amountSuffix === 'B') {
convertedAmount = Number(amount) / 1_000_000_000 convertedAmount = Number(convertedAmount) / 1_000_000_000
} }
if (amountSuffix === 'M') { if (amountSuffix === 'M') {
convertedAmount = Number(amount) / 1_000_000 convertedAmount = Number(convertedAmount) / 1_000_000
} }
if (amountSuffix === 'K') { if (amountSuffix === 'K') {
convertedAmount = Number(amount) / 1_000 convertedAmount = Number(convertedAmount) / 1_000
} }
if (rounded) { if (options?.rounded) {
convertedAmount = convertedAmount.toFixed(maxDecimals) convertedAmount = convertedAmount.toFixed(maxDecimals)
} else { } else {
const amountFractions = String(convertedAmount).split('.') const amountFractions = String(convertedAmount).split('.')
@ -112,8 +96,8 @@ export const formatValue = (
} }
let returnValue = '' let returnValue = ''
if (amountPrefix) { if (options?.prefix) {
returnValue += amountPrefix returnValue += options.prefix
} }
returnValue += convertedAmount returnValue += convertedAmount
@ -131,9 +115,23 @@ export const formatValue = (
returnValue += amountSuffix returnValue += amountSuffix
} }
if (suffix) { if (options?.suffix) {
returnValue += suffix returnValue += options.suffix
} }
return returnValue return returnValue
} }
export function formatLeverage(leverage: number) {
return formatValue(leverage, {
minDecimals: 0,
suffix: 'x',
})
}
export function formatPercent(percent: number | string) {
return formatValue(+percent * 100, {
minDecimals: 0,
suffix: '%',
})
}