mango-v4-ui/components/TokenList.tsx

830 lines
28 KiB
TypeScript
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

import { Bank } from '@blockworks-foundation/mango-v4'
import { Disclosure, Popover, Transition } from '@headlessui/react'
import {
ChevronDownIcon,
EllipsisHorizontalIcon,
XMarkIcon,
} from '@heroicons/react/20/solid'
import { useTranslation } from 'next-i18next'
import { useRouter } from 'next/router'
import { Fragment, useCallback, useMemo, useState } from 'react'
import { useViewport } from '../hooks/useViewport'
import mangoStore from '@store/mangoStore'
import { breakpoints } from '../utils/theme'
import Switch from './forms/Switch'
import ContentBox from './shared/ContentBox'
import Tooltip from './shared/Tooltip'
import { formatTokenSymbol } from 'utils/tokens'
import useMangoAccount from 'hooks/useMangoAccount'
import {
SortableColumnHeader,
Table,
Td,
Th,
TrBody,
TrHead,
} from './shared/TableElements'
import DepositWithdrawModal from './modals/DepositWithdrawModal'
import BorrowRepayModal from './modals/BorrowRepayModal'
import { WRAPPED_SOL_MINT } from '@project-serum/serum/lib/token-instructions'
import {
SHOW_ZERO_BALANCES_KEY,
TOKEN_REDUCE_ONLY_OPTIONS,
USDC_MINT,
} from 'utils/constants'
import { PublicKey } from '@solana/web3.js'
import ActionsLinkButton from './account/ActionsLinkButton'
import FormatNumericValue from './shared/FormatNumericValue'
import BankAmountWithValue from './shared/BankAmountWithValue'
import useBanksWithBalances, {
BankWithBalance,
} from 'hooks/useBanksWithBalances'
import useUnownedAccount from 'hooks/useUnownedAccount'
import useLocalStorageState from 'hooks/useLocalStorageState'
import TokenLogo from './shared/TokenLogo'
import useHealthContributions from 'hooks/useHealthContributions'
import { useSortableData } from 'hooks/useSortableData'
import TableTokenName from './shared/TableTokenName'
import CloseBorrowModal from './modals/CloseBorrowModal'
import { floorToDecimal } from 'utils/numbers'
import SheenLoader from './shared/SheenLoader'
import useAccountInterest from 'hooks/useAccountInterest'
import { handleGoToTradePage } from 'utils/markets'
import TableRatesDisplay from './shared/TableRatesDisplay'
export const handleOpenCloseBorrowModal = (borrowBank: Bank) => {
const group = mangoStore.getState().group
const mangoAccount = mangoStore.getState().mangoAccount.current
let repayBank: Bank | undefined
if (borrowBank.name === 'USDC') {
const solBank = group?.getFirstBankByMint(WRAPPED_SOL_MINT)
repayBank = solBank
} else {
const usdcBank = group?.getFirstBankByMint(new PublicKey(USDC_MINT))
repayBank = usdcBank
}
if (mangoAccount && repayBank) {
const borrowBalance = mangoAccount.getTokenBalanceUi(borrowBank)
const roundedBorrowBalance = floorToDecimal(
borrowBalance,
borrowBank.mintDecimals,
).toNumber()
const repayBalance = mangoAccount.getTokenBalanceUi(repayBank)
const roundedRepayBalance = floorToDecimal(
repayBalance,
repayBank.mintDecimals,
).toNumber()
const hasSufficientRepayBalance =
Math.abs(roundedRepayBalance) * repayBank.uiPrice >
Math.abs(roundedBorrowBalance) * borrowBank.uiPrice
set((state) => {
state.swap.swapMode = hasSufficientRepayBalance ? 'ExactOut' : 'ExactIn'
state.swap.inputBank = repayBank
state.swap.outputBank = borrowBank
if (hasSufficientRepayBalance) {
state.swap.amountOut = Math.abs(roundedBorrowBalance).toString()
} else {
state.swap.amountIn = Math.abs(roundedRepayBalance).toString()
}
})
}
}
export const handleCloseBorrowModal = () => {
set((state) => {
state.swap.inputBank = undefined
state.swap.outputBank = undefined
state.swap.amountIn = ''
state.swap.amountOut = ''
state.swap.swapMode = 'ExactIn'
})
}
type TableData = {
bank: Bank
balance: number
symbol: string
interestAmount: number
interestValue: number
inOrders: number
unsettled: number
collateralValue: number
assetWeight: string
liabWeight: string
depositRate: number
borrowRate: number
}
const set = mangoStore.getState().set
const TokenList = () => {
const { t } = useTranslation(['common', 'token', 'trade'])
const [showCloseBorrowModal, setCloseBorrowModal] = useState(false)
const [closeBorrowBank, setCloseBorrowBank] = useState<Bank | undefined>()
const [showZeroBalances, setShowZeroBalances] = useLocalStorageState(
SHOW_ZERO_BALANCES_KEY,
true,
)
const { mangoAccountAddress } = useMangoAccount()
const { initContributions } = useHealthContributions()
const spotBalances = mangoStore((s) => s.mangoAccount.spotBalances)
const { width } = useViewport()
const showTableView = width ? width > breakpoints.md : false
const banks = useBanksWithBalances('balance')
const {
data: totalInterestData,
isInitialLoading: loadingTotalInterestData,
} = useAccountInterest()
const formattedTableData = useCallback(
(banks: BankWithBalance[]) => {
const formatted = []
for (const b of banks) {
const bank = b.bank
const roundedBalance = floorToDecimal(
b.balance,
bank.mintDecimals,
).toNumber()
let balance = roundedBalance
if (b.balance && !roundedBalance) {
balance = b.balance
}
const balanceValue = balance * bank.uiPrice
const symbol = bank.name === 'MSOL' ? 'mSOL' : bank.name
const hasInterestEarned = totalInterestData?.find(
(d) =>
d.symbol.toLowerCase() === symbol.toLowerCase() ||
(symbol === 'ETH (Portal)' && d.symbol === 'ETH'),
)
const interestAmount = hasInterestEarned
? hasInterestEarned.borrow_interest * -1 +
hasInterestEarned.deposit_interest
: 0
const interestValue = hasInterestEarned
? hasInterestEarned.borrow_interest_usd * -1 +
hasInterestEarned.deposit_interest_usd
: 0.0
const inOrders = spotBalances[bank.mint.toString()]?.inOrders || 0
const unsettled = spotBalances[bank.mint.toString()]?.unsettled || 0
const collateralValue =
initContributions.find((val) => val.asset === bank.name)
?.contribution || 0
const assetWeight = bank.scaledInitAssetWeight(bank.price).toFixed(2)
const liabWeight = bank.scaledInitLiabWeight(bank.price).toFixed(2)
const depositRate = bank.getDepositRateUi()
const borrowRate = bank.getBorrowRateUi()
const data = {
balance,
balanceValue,
bank,
symbol,
interestAmount,
interestValue,
inOrders,
unsettled,
collateralValue,
assetWeight,
liabWeight,
depositRate,
borrowRate,
}
formatted.push(data)
}
return formatted
},
[initContributions, spotBalances, totalInterestData],
)
const unsortedTableData = useMemo(() => {
if (!banks.length) return []
if (showZeroBalances || !mangoAccountAddress) {
return formattedTableData(banks)
} else {
const filtered = banks.filter((b) => Math.abs(b.balance) > 0)
return formattedTableData(filtered)
}
}, [banks, mangoAccountAddress, showZeroBalances, totalInterestData])
const {
items: tableData,
requestSort,
sortConfig,
} = useSortableData(unsortedTableData)
const openCloseBorrowModal = (borrowBank: Bank) => {
setCloseBorrowModal(true)
setCloseBorrowBank(borrowBank)
handleOpenCloseBorrowModal(borrowBank)
}
const closeBorrowModal = () => {
setCloseBorrowModal(false)
handleCloseBorrowModal()
}
const balancesNumber = useMemo(() => {
if (!banks.length) return 0
const activeBalancesNumber = banks.filter(
(bank) => Math.abs(bank.balance) > 0,
).length
return activeBalancesNumber
}, [banks])
return (
<ContentBox hideBorder hidePadding>
{mangoAccountAddress ? (
<div className="flex w-full items-center justify-between border-b border-th-bkg-3 px-6 py-3">
<p className="text-xs text-th-fgd-2">{`${balancesNumber} ${t(
'balances',
)}`}</p>
<Switch
checked={showZeroBalances}
disabled={!mangoAccountAddress}
onChange={() => setShowZeroBalances(!showZeroBalances)}
>
{t('account:zero-balances')}
</Switch>
</div>
) : null}
{showTableView ? (
<>
<Table>
<thead>
<TrHead>
<Th className="text-left">
<SortableColumnHeader
sortKey="symbol"
sort={() => requestSort('symbol')}
sortConfig={sortConfig}
title={t('token')}
/>
</Th>
<Th>
<div className="flex justify-end">
<Tooltip content="A negative balance represents a borrow">
<SortableColumnHeader
sortKey="balanceValue"
sort={() => requestSort('balanceValue')}
sortConfig={sortConfig}
title={t('balance')}
titleClass="tooltip-underline"
/>
</Tooltip>
</div>
</Th>
<Th>
<div className="flex justify-end">
<Tooltip content={t('tooltip-collateral-value')}>
<SortableColumnHeader
sortKey="collateralValue"
sort={() => requestSort('collateralValue')}
sortConfig={sortConfig}
title={t('collateral-value')}
titleClass="tooltip-underline"
/>
</Tooltip>
</div>
</Th>
<Th>
<div className="flex justify-end">
<SortableColumnHeader
sortKey="inOrders"
sort={() => requestSort('inOrders')}
sortConfig={sortConfig}
title={t('trade:in-orders')}
/>
</div>
</Th>
<Th>
<div className="flex justify-end">
<Tooltip content="The sum of interest earned and interest paid for each token">
<SortableColumnHeader
sortKey="interestValue"
sort={() => requestSort('interestValue')}
sortConfig={sortConfig}
title={t('interest-earned-paid')}
titleClass="tooltip-underline"
/>
</Tooltip>
</div>
</Th>
<Th>
<div className="flex justify-end">
<Tooltip content={t('tooltip-interest-rates')}>
<SortableColumnHeader
sortKey="depositRate"
sort={() => requestSort('depositRate')}
sortConfig={sortConfig}
title={t('rates')}
titleClass="tooltip-underline"
/>
</Tooltip>
</div>
</Th>
<Th className="text-right">
<span>{t('actions')}</span>
</Th>
</TrHead>
</thead>
<tbody>
{tableData.map((data) => {
const {
balance,
bank,
symbol,
interestValue,
inOrders,
collateralValue,
assetWeight,
liabWeight,
depositRate,
borrowRate,
} = data
const decimals = floorToDecimal(
balance,
bank.mintDecimals,
).toNumber()
? bank.mintDecimals
: undefined
return (
<TrBody key={symbol}>
<Td>
<TableTokenName bank={bank} symbol={symbol} />
</Td>
<Td className="text-right">
<BankAmountWithValue
amount={balance}
decimals={decimals}
bank={bank}
stacked
isPrivate
fixDecimals={false}
/>
</Td>
<Td className="text-right">
<p>
<FormatNumericValue
value={collateralValue}
decimals={2}
isUsd
isPrivate
/>
</p>
<p className="text-sm text-th-fgd-4">
<FormatNumericValue
value={
collateralValue <= -0.01 ? liabWeight : assetWeight
}
/>
x
</p>
</Td>
<Td className="text-right">
{inOrders ? (
<BankAmountWithValue
amount={inOrders}
bank={bank}
stacked
isPrivate
/>
) : (
<p className="text-th-fgd-4"></p>
)}
</Td>
<Td>
<div className="flex justify-end">
{loadingTotalInterestData ? (
<SheenLoader>
<div className="h-4 w-12 bg-th-bkg-2" />
</SheenLoader>
) : Math.abs(interestValue) > 0 ? (
<p>
<FormatNumericValue
value={interestValue}
isUsd
decimals={2}
isPrivate
/>
</p>
) : (
<p className="text-th-fgd-4"></p>
)}
</div>
</Td>
<Td>
<div className="flex justify-end space-x-1.5">
<TableRatesDisplay
borrowRate={borrowRate}
depositRate={depositRate}
/>
</div>
</Td>
<Td>
<div className="flex items-center justify-end space-x-2">
{balance < 0 ? (
<button
className="rounded-md border border-th-fgd-4 px-2 py-1.5 text-xs font-bold text-th-fgd-2 focus:outline-none md:hover:border-th-fgd-3"
onClick={() => openCloseBorrowModal(bank)}
>
{t('close-borrow', { token: '' })}
</button>
) : null}
<ActionsMenu bank={bank} />
</div>
</Td>
</TrBody>
)
})}
</tbody>
</Table>
</>
) : (
<div className="border-b border-th-bkg-3">
{tableData.map((data) => {
return <MobileTokenListItem key={data.bank.name} data={data} />
})}
</div>
)}
{showCloseBorrowModal ? (
<CloseBorrowModal
borrowBank={closeBorrowBank}
isOpen={showCloseBorrowModal}
onClose={closeBorrowModal}
/>
) : null}
</ContentBox>
)
}
export default TokenList
const MobileTokenListItem = ({ data }: { data: TableData }) => {
const { t } = useTranslation(['common', 'token'])
const { mangoAccount } = useMangoAccount()
const {
bank,
balance,
symbol,
interestAmount,
interestValue,
inOrders,
unsettled,
collateralValue,
assetWeight,
liabWeight,
depositRate,
borrowRate,
} = data
const decimals = floorToDecimal(balance, bank.mintDecimals).toNumber()
? bank.mintDecimals
: undefined
return (
<Disclosure>
{({ open }) => (
<>
<Disclosure.Button
className={`w-full border-t border-th-bkg-3 p-4 text-left first:border-t-0 focus:outline-none`}
>
<div className="flex items-center justify-between">
<TableTokenName bank={bank} symbol={symbol} />
<div className="flex items-center space-x-2">
<div className="text-right">
<p className="font-mono text-sm text-th-fgd-2">
<FormatNumericValue value={balance} decimals={decimals} />
</p>
<span className="font-mono text-xs text-th-fgd-3">
<FormatNumericValue
value={mangoAccount ? balance * bank.uiPrice : 0}
decimals={decimals ? 2 : undefined}
isUsd
isPrivate
/>
</span>
</div>
<ChevronDownIcon
className={`${
open ? 'rotate-180' : 'rotate-0'
} h-6 w-6 shrink-0 text-th-fgd-3`}
/>
</div>
</div>
</Disclosure.Button>
<Transition
enter="transition ease-in duration-200"
enterFrom="opacity-0"
enterTo="opacity-100"
>
<Disclosure.Panel>
<div className="mx-4 grid grid-cols-2 gap-4 border-t border-th-bkg-3 py-4">
<div className="col-span-1">
<Tooltip content={t('tooltip-collateral-value')}>
<p className="tooltip-underline text-xs text-th-fgd-3">
{t('collateral-value')}
</p>
</Tooltip>
<p className="font-mono text-th-fgd-2">
<FormatNumericValue
value={collateralValue}
decimals={2}
isUsd
isPrivate
/>
<span className="text-th-fgd-3">
{' '}
<FormatNumericValue
value={
collateralValue <= -0.01 ? liabWeight : assetWeight
}
/>
x
</span>
</p>
</div>
<div className="col-span-1">
<p className="text-xs text-th-fgd-3">
{t('trade:in-orders')}
</p>
<BankAmountWithValue
amount={inOrders}
bank={bank}
isPrivate
/>
</div>
<div className="col-span-1">
<p className="text-xs text-th-fgd-3">
{t('trade:unsettled')}
</p>
<BankAmountWithValue
amount={unsettled}
bank={bank}
isPrivate
/>
</div>
<div className="col-span-1">
<p className="text-xs text-th-fgd-3">
{t('interest-earned-paid')}
</p>
<BankAmountWithValue
amount={interestAmount}
bank={bank}
value={interestValue}
isPrivate
/>
</div>
<div className="col-span-1">
<p className="text-xs text-th-fgd-3">{t('rates')}</p>
<div className="flex space-x-1.5 font-mono">
<TableRatesDisplay
borrowRate={borrowRate}
depositRate={depositRate}
/>
</div>
</div>
<div className="col-span-1">
<ActionsMenu bank={bank} />
</div>
</div>
</Disclosure.Panel>
</Transition>
</>
)}
</Disclosure>
)
}
export const ActionsMenu = ({
bank,
showText,
}: {
bank: Bank
showText?: boolean
}) => {
const { t } = useTranslation('common')
const { mangoAccountAddress } = useMangoAccount()
const [showDepositModal, setShowDepositModal] = useState(false)
const [showWithdrawModal, setShowWithdrawModal] = useState(false)
const [showBorrowModal, setShowBorrowModal] = useState(false)
const [showRepayModal, setShowRepayModal] = useState(false)
const [selectedToken, setSelectedToken] = useState('')
const set = mangoStore.getState().set
const router = useRouter()
const spotMarkets = mangoStore((s) => s.serumMarkets)
const { isUnownedAccount } = useUnownedAccount()
const { isDesktop } = useViewport()
const hasSpotMarket = useMemo(() => {
const markets = spotMarkets.filter(
(m) => m.baseTokenIndex === bank?.tokenIndex,
)
if (markets?.length) return true
return false
}, [spotMarkets])
const handleShowActionModals = useCallback(
(token: string, action: 'borrow' | 'deposit' | 'withdraw' | 'repay') => {
setSelectedToken(token)
action === 'borrow'
? setShowBorrowModal(true)
: action === 'deposit'
? setShowDepositModal(true)
: action === 'withdraw'
? setShowWithdrawModal(true)
: setShowRepayModal(true)
},
[],
)
const balance = useMemo(() => {
if (!mangoAccountAddress || !bank) return 0
const mangoAccount = mangoStore.getState().mangoAccount.current
if (mangoAccount) {
return mangoAccount.getTokenBalanceUi(bank)
} else return 0
}, [bank, mangoAccountAddress])
const handleSwap = useCallback(() => {
const group = mangoStore.getState().group
if (balance > 0) {
if (bank.name === 'USDC') {
const solBank = group?.getFirstBankByMint(WRAPPED_SOL_MINT)
set((s) => {
s.swap.inputBank = bank
s.swap.outputBank = solBank
})
} else {
const usdcBank = group?.getFirstBankByMint(new PublicKey(USDC_MINT))
set((s) => {
s.swap.inputBank = bank
s.swap.outputBank = usdcBank
})
}
} else {
if (bank.name === 'USDC') {
const solBank = group?.getFirstBankByMint(WRAPPED_SOL_MINT)
set((s) => {
s.swap.inputBank = solBank
s.swap.outputBank = bank
})
} else {
const usdcBank = group?.getFirstBankByMint(new PublicKey(USDC_MINT))
set((s) => {
s.swap.inputBank = usdcBank
s.swap.outputBank = bank
})
}
}
router.push('/swap', undefined, { shallow: true })
}, [bank, router, set])
return (
<>
{isUnownedAccount ? null : (
<Popover>
{({ open }) => (
<div className="relative">
<Popover.Button
className={`flex items-center justify-center border border-th-button text-th-fgd-1 md:hover:border-th-button-hover md:hover:text-th-fgd-1 ${
showText || !isDesktop
? 'h-10 w-full rounded-md'
: 'h-8 w-8 rounded-full'
}`}
>
{showText || !isDesktop ? (
<span className="mr-2 font-display">{t('actions')}</span>
) : null}
{open ? (
<XMarkIcon className="h-5 w-5" />
) : (
<EllipsisHorizontalIcon className="h-5 w-5" />
)}
</Popover.Button>
<Transition
appear={true}
show={open}
as={Fragment}
enter="transition ease-in duration-100"
enterFrom="scale-90"
enterTo="scale-100"
leave="transition ease-out duration-100"
leaveFrom="opacity-100"
leaveTo="opacity-0"
>
<Popover.Panel
className={`thin-scroll absolute z-20 max-h-60 w-32 space-y-2 overflow-auto rounded-md bg-th-bkg-2 p-4 ${
isDesktop && !showText
? 'bottom-0 left-auto right-12 pt-2'
: 'bottom-12 left-0'
}`}
>
{!showText && isDesktop ? (
<div className="flex items-center justify-center border-b border-th-bkg-3 pb-2">
<div className="mr-2 flex shrink-0 items-center">
<TokenLogo bank={bank} size={20} />
</div>
<p className="font-body">
{formatTokenSymbol(bank.name)}
</p>
</div>
) : null}
{bank.reduceOnly !== TOKEN_REDUCE_ONLY_OPTIONS.ENABLED ? (
<ActionsLinkButton
onClick={() =>
handleShowActionModals(bank.name, 'deposit')
}
>
{t('deposit')}
</ActionsLinkButton>
) : null}
{balance < 0 ? (
<ActionsLinkButton
onClick={() => handleShowActionModals(bank.name, 'repay')}
>
{t('repay')}
</ActionsLinkButton>
) : null}
{balance && balance > 0 ? (
<ActionsLinkButton
onClick={() =>
handleShowActionModals(bank.name, 'withdraw')
}
>
{t('withdraw')}
</ActionsLinkButton>
) : null}
{bank.reduceOnly === TOKEN_REDUCE_ONLY_OPTIONS.DISABLED ? (
<ActionsLinkButton
onClick={() =>
handleShowActionModals(bank.name, 'borrow')
}
>
{t('borrow')}
</ActionsLinkButton>
) : null}
<ActionsLinkButton onClick={handleSwap}>
{t('swap')}
</ActionsLinkButton>
{hasSpotMarket ? (
<ActionsLinkButton
onClick={() =>
handleGoToTradePage(bank, spotMarkets, router)
}
>
{t('trade')}
</ActionsLinkButton>
) : null}
</Popover.Panel>
</Transition>
</div>
)}
</Popover>
)}
{showDepositModal ? (
<DepositWithdrawModal
action="deposit"
isOpen={showDepositModal}
onClose={() => setShowDepositModal(false)}
token={selectedToken}
/>
) : null}
{showWithdrawModal ? (
<DepositWithdrawModal
action="withdraw"
isOpen={showWithdrawModal}
onClose={() => setShowWithdrawModal(false)}
token={selectedToken}
/>
) : null}
{showBorrowModal ? (
<BorrowRepayModal
action="borrow"
isOpen={showBorrowModal}
onClose={() => setShowBorrowModal(false)}
token={selectedToken}
/>
) : null}
{showRepayModal ? (
<BorrowRepayModal
action="repay"
isOpen={showRepayModal}
onClose={() => setShowRepayModal(false)}
token={selectedToken}
/>
) : null}
</>
)
}