feat: created AccountPerpsPositions for the AccountDetails component (#745)

* feat: created AccountPerpsPositions for the AccountDetails component

* fix: type fixing

* feat: added ToolTip

* tidy: refactor

* fix: added unrealized PnL placeholder

* tidy: finetune on the border color

* fix: fixed tables

* fix: adjustments according to feedback

* fix: fixed lent assets table

* fix: fixed typing

* refactor: streamline size to amount
This commit is contained in:
Linkie Link 2024-01-23 15:14:48 +01:00 committed by GitHub
parent 254df8cd6e
commit 0ccb29154a
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
50 changed files with 488 additions and 152 deletions

View File

@ -17,7 +17,7 @@ export default async function getHLSStakingAccounts(
if (account.deposits.length === 0) return if (account.deposits.length === 0) return
const strategy = hlsStrategies.find( const strategy = hlsStrategies.find(
(strategy) => strategy.denoms.deposit === account.deposits.at(0).denom, (strategy) => strategy.denoms.deposit === account.deposits[0].denom,
) )
if (!strategy) return if (!strategy) return

View File

@ -1,10 +1,14 @@
import { getPerpsQueryClient } from 'api/cosmwasm-client' import { getPerpsQueryClient } from 'api/cosmwasm-client'
import { BNCoin } from 'types/classes/BNCoin' import { BNCoin } from 'types/classes/BNCoin'
export default async function getOpeningFee(chainConfig: ChainConfig, denom: string, size: string) { export default async function getOpeningFee(
chainConfig: ChainConfig,
denom: string,
amount: string,
) {
const perpsClient = await getPerpsQueryClient(chainConfig) const perpsClient = await getPerpsQueryClient(chainConfig)
return perpsClient return perpsClient
.openingFee({ denom, size: size as any }) .openingFee({ denom, size: amount as any })
.then((resp) => BNCoin.fromCoin(resp.fee)) .then((resp) => BNCoin.fromCoin(resp.fee))
} }

View File

@ -1,22 +1,23 @@
import { useCallback, useState } from 'react' import { useCallback, useState } from 'react'
import Modal from 'components/Modals/Modal'
import CurrentAccountSummary from 'components/account/CurrentAccountSummary' import CurrentAccountSummary from 'components/account/CurrentAccountSummary'
import AssetImage from 'components/common/assets/AssetImage'
import Button from 'components/common/Button' import Button from 'components/common/Button'
import Card from 'components/common/Card' import Card from 'components/common/Card'
import Divider from 'components/common/Divider' import Divider from 'components/common/Divider'
import { ArrowRight } from 'components/common/Icons' import { ArrowRight } from 'components/common/Icons'
import Modal from 'components/Modals/Modal'
import Text from 'components/common/Text' import Text from 'components/common/Text'
import TokenInputWithSlider from 'components/common/TokenInput/TokenInputWithSlider' import TokenInputWithSlider from 'components/common/TokenInput/TokenInputWithSlider'
import AssetImage from 'components/common/assets/AssetImage'
import { BN_ZERO } from 'constants/math' import { BN_ZERO } from 'constants/math'
import { BNCoin } from 'types/classes/BNCoin'
import { byDenom } from 'utils/array' import { byDenom } from 'utils/array'
import { BN } from 'utils/helpers' import { BN } from 'utils/helpers'
interface Props { interface Props {
asset: Asset asset: Asset
title: string title: string
coinBalances: Coin[] coinBalances: BNCoin[]
actionButtonText: string actionButtonText: string
contentHeader?: JSX.Element contentHeader?: JSX.Element
onClose: () => void onClose: () => void

View File

@ -28,7 +28,7 @@ export default function Repay(props: Props) {
const repay = useStore((s) => s.repay) const repay = useStore((s) => s.repay)
const currentDebt: BigNumber = useMemo( const currentDebt: BigNumber = useMemo(
() => props.account.debts.find(byDenom(props.borrowAsset.denom)).amount || BN_ZERO, () => props.account.debts.find(byDenom(props.borrowAsset.denom))?.amount || BN_ZERO,
[props.account.debts, props.borrowAsset.denom], [props.account.debts, props.borrowAsset.denom],
) )

View File

@ -1,4 +1,4 @@
import React, { useCallback, useMemo } from 'react' import { useCallback, useMemo } from 'react'
import Button from 'components/common/Button' import Button from 'components/common/Button'
import TokenInputWithSlider from 'components/common/TokenInput/TokenInputWithSlider' import TokenInputWithSlider from 'components/common/TokenInput/TokenInputWithSlider'
@ -42,6 +42,7 @@ export default function Withdraw(props: Props) {
const onClick = useCallback(() => { const onClick = useCallback(() => {
useStore.setState({ hlsManageModal: null }) useStore.setState({ hlsManageModal: null })
if (!removedDeposit) return
withdraw({ withdraw({
accountId: props.account.id, accountId: props.account.id,
coins: [{ coin: removedDeposit }], coins: [{ coin: removedDeposit }],

View File

@ -15,7 +15,7 @@ function DetailsHeader({ data }: Props) {
const balanceInWallet = useCurrentWalletBalance(asset.denom) const balanceInWallet = useCurrentWalletBalance(asset.denom)
return ( return (
<div className='flex gap-6 border-b border-white/5 px-6 py-4 gradient-header'> <div className='flex gap-6 px-6 py-4 border-b border-white/5 gradient-header'>
{assetApy && ( {assetApy && (
<> <>
<TitleAndSubCell <TitleAndSubCell
@ -40,7 +40,7 @@ function DetailsHeader({ data }: Props) {
<TitleAndSubCell <TitleAndSubCell
title={ title={
<DisplayCurrency <DisplayCurrency
coin={new BNCoin({ denom: asset.denom, amount: accountLentAmount })} coin={new BNCoin({ denom: asset.denom, amount: accountLentAmount.toString() })}
/> />
} }
sub={'Deposited'} sub={'Deposited'}

View File

@ -7,7 +7,7 @@ interface Props {
apy: number apy: number
markets: Market[] markets: Market[]
denom: string denom: string
type: 'deposits' | 'borrowing' | 'lending' | 'vault' type: PositionType
} }
export default function Apr(props: Props) { export default function Apr(props: Props) {
@ -20,7 +20,7 @@ export default function Apr(props: Props) {
<AssetRate <AssetRate
className='justify-end text-xs' className='justify-end text-xs'
rate={apy} rate={apy}
isEnabled={type !== 'lending' || isEnabled} isEnabled={type !== 'lend' || isEnabled}
type='apy' type='apy'
orientation='ltr' orientation='ltr'
/> />

View File

@ -3,19 +3,16 @@ export const ASSET_META = { accessorKey: 'symbol', header: 'Asset', id: 'symbol'
interface Props { interface Props {
symbol: string symbol: string
type: 'deposits' | 'borrowing' | 'lending' | 'vault' type: PositionType
} }
export const borderColor = (type: Props['type']): string =>
type === 'borrowing' ? 'border-loss' : 'border-profit'
export default function Asset(props: Props) { export default function Asset(props: Props) {
const { symbol, type } = props const { symbol, type } = props
return ( return (
<Text size='xs'> <Text size='xs'>
{symbol} {symbol}
{type === 'borrowing' && <span className='ml-1 text-loss'>(debt)</span>} {type === 'borrow' && <span className='ml-1 text-loss'>(debt)</span>}
{type === 'lending' && <span className='ml-1 text-profit'>(lent)</span>} {type === 'lend' && <span className='ml-1 text-profit'>(lent)</span>}
{type === 'vault' && <span className='ml-1 text-profit'>(farm)</span>} {type === 'vault' && <span className='ml-1 text-profit'>(farm)</span>}
</Text> </Text>
) )

View File

@ -20,7 +20,7 @@ interface Props {
amount: number amount: number
computeLiquidationPrice: (denom: string, kind: LiquidationPriceKind) => number | null computeLiquidationPrice: (denom: string, kind: LiquidationPriceKind) => number | null
denom: string denom: string
type: 'deposits' | 'borrowing' | 'lending' | 'vault' type: PositionType
account: Account account: Account
} }
@ -31,7 +31,7 @@ export default function LiqPrice(props: Props) {
const liqPrice = useMemo(() => { const liqPrice = useMemo(() => {
if (type === 'vault' || amount === 0) return 0 if (type === 'vault' || amount === 0) return 0
return computeLiquidationPrice(denom, type === 'borrowing' ? 'debt' : 'asset') return computeLiquidationPrice(denom, type === 'borrow' ? 'debt' : 'asset')
}, [amount, computeLiquidationPrice, denom, type]) }, [amount, computeLiquidationPrice, denom, type])
const { liquidationPrice } = useLiquidationPrice(liqPrice) const { liquidationPrice } = useLiquidationPrice(liqPrice)

View File

@ -8,7 +8,7 @@ export const PRICE_META = { id: 'price', header: 'Price', meta: { className: 'w-
interface Props { interface Props {
amount: number amount: number
denom: string denom: string
type: 'deposits' | 'borrowing' | 'lending' | 'vault' type: PositionType
} }
export default function Price(props: Props) { export default function Price(props: Props) {

View File

@ -12,7 +12,7 @@ interface Props {
size: number size: number
amountChange: BigNumber amountChange: BigNumber
denom: string denom: string
type: 'deposits' | 'borrowing' | 'lending' | 'vault' type: PositionType
} }
export const sizeSortingFn = (a: Row<AccountBalanceRow>, b: Row<AccountBalanceRow>): number => { export const sizeSortingFn = (a: Row<AccountBalanceRow>, b: Row<AccountBalanceRow>): number => {

View File

@ -12,10 +12,19 @@ export const VALUE_META = { accessorKey: 'value', header: 'Value' }
interface Props { interface Props {
amountChange: BigNumber amountChange: BigNumber
value: string value: string
type: 'deposits' | 'borrowing' | 'lending' | 'vault' type: PositionType
} }
export const valueSortingFn = (a: Row<AccountBalanceRow>, b: Row<AccountBalanceRow>): number => { export const valueBalancesSortingFn = (
a: Row<AccountBalanceRow>,
b: Row<AccountBalanceRow>,
): number => {
const valueA = BN(a.original.value)
const valueB = BN(b.original.value)
return valueA.minus(valueB).toNumber()
}
export const valuePerpSortingFn = (a: Row<AccountPerpRow>, b: Row<AccountPerpRow>): number => {
const valueA = BN(a.original.value) const valueA = BN(a.original.value)
const valueB = BN(b.original.value) const valueB = BN(b.original.value)
return valueA.minus(valueB).toNumber() return valueA.minus(valueB).toNumber()
@ -30,10 +39,6 @@ export default function Value(props: Props) {
}) })
return ( return (
<DisplayCurrency <DisplayCurrency coin={coin} className={classNames('text-xs text-right', color)} showZero />
coin={coin}
className={classNames('text-xs text-right', color)}
showZero={true}
/>
) )
} }

View File

@ -11,7 +11,7 @@ import Size, {
} from 'components/account/AccountBalancesTable/Columns/Size' } from 'components/account/AccountBalancesTable/Columns/Size'
import Value, { import Value, {
VALUE_META, VALUE_META,
valueSortingFn, valueBalancesSortingFn,
} from 'components/account/AccountBalancesTable/Columns/Value' } from 'components/account/AccountBalancesTable/Columns/Value'
import useMarketAssets from 'hooks/markets/useMarketAssets' import useMarketAssets from 'hooks/markets/useMarketAssets'
import useHealthComputer from 'hooks/useHealthComputer' import useHealthComputer from 'hooks/useHealthComputer'
@ -41,7 +41,7 @@ export default function useAccountBalancesColumns(
type={row.original.type} type={row.original.type}
/> />
), ),
sortingFn: valueSortingFn, sortingFn: valueBalancesSortingFn,
}, },
{ {
...SIZE_META, ...SIZE_META,

View File

@ -3,7 +3,7 @@ import { BNCoin } from 'types/classes/BNCoin'
import { demagnify, getCoinValue } from 'utils/formatters' import { demagnify, getCoinValue } from 'utils/formatters'
export function getAssetAccountBalanceRow( export function getAssetAccountBalanceRow(
type: 'deposits' | 'borrowing' | 'lending', type: 'deposit' | 'borrow' | 'lend',
asset: Asset, asset: Asset,
prices: BNCoin[], prices: BNCoin[],
assets: Asset[], assets: Asset[],
@ -57,11 +57,9 @@ export function getVaultAccountBalanceRow(
} }
} }
export function getAmountChangeColor( export function getAmountChangeColor(type: PositionType, amount: BigNumber) {
type: 'deposits' | 'borrowing' | 'lending' | 'vault', if (type === 'perp') return ''
amount: BigNumber, if (type === 'borrow') {
) {
if (type === 'borrowing') {
if (amount.isGreaterThan(0)) return 'text-loss' if (amount.isGreaterThan(0)) return 'text-loss'
if (amount.isLessThan(0)) return 'text-profit' if (amount.isLessThan(0)) return 'text-profit'
} }

View File

@ -89,7 +89,7 @@ export default function AccountBalancesTable(props: Props) {
initialSorting={[]} initialSorting={[]}
spacingClassName='p-2' spacingClassName='p-2'
hideCard={hideCard} hideCard={hideCard}
isBalancesTable type='balances'
/> />
) )
} }

View File

@ -38,7 +38,7 @@ export default function useAccountBalanceData(props: Props) {
: 0 : 0
const prevDeposit = updatedAccount ? account?.deposits.find(byDenom(deposit.denom)) : deposit const prevDeposit = updatedAccount ? account?.deposits.find(byDenom(deposit.denom)) : deposit
deposits.push( deposits.push(
getAssetAccountBalanceRow('deposits', asset, prices, assets, deposit, apy, prevDeposit), getAssetAccountBalanceRow('deposit', asset, prices, assets, deposit, apy, prevDeposit),
) )
}) })
@ -50,7 +50,7 @@ export default function useAccountBalanceData(props: Props) {
const prevLending = updatedAccount const prevLending = updatedAccount
? account?.lends.find((position) => position.denom === lending.denom) ? account?.lends.find((position) => position.denom === lending.denom)
: lending : lending
return getAssetAccountBalanceRow('lending', asset, prices, assets, lending, apy, prevLending) return getAssetAccountBalanceRow('lend', asset, prices, assets, lending, apy, prevLending)
}) })
const vaults = accountVaults.map((vault) => { const vaults = accountVaults.map((vault) => {
@ -67,7 +67,7 @@ export default function useAccountBalanceData(props: Props) {
const prevDebt = updatedAccount const prevDebt = updatedAccount
? account?.debts.find((position) => position.denom === debt.denom) ? account?.debts.find((position) => position.denom === debt.denom)
: debt : debt
return getAssetAccountBalanceRow('borrowing', asset, prices, assets, debt, apy, prevDebt) return getAssetAccountBalanceRow('borrow', asset, prices, assets, debt, apy, prevDebt)
}) })
return [...deposits, ...lends, ...vaults, ...debts] return [...deposits, ...lends, ...vaults, ...debts]
}, [ }, [

View File

@ -6,6 +6,7 @@ import AccountBalancesTable from 'components/account/AccountBalancesTable'
import AccountComposition from 'components/account/AccountComposition' import AccountComposition from 'components/account/AccountComposition'
import AccountDetailsLeverage from 'components/account/AccountDetails/AccountDetailsLeverage' import AccountDetailsLeverage from 'components/account/AccountDetails/AccountDetailsLeverage'
import Skeleton from 'components/account/AccountDetails/Skeleton' import Skeleton from 'components/account/AccountDetails/Skeleton'
import AccountPerpPositionTable from 'components/account/AccountPerpPositionTable'
import { HealthGauge } from 'components/account/Health/HealthGauge' import { HealthGauge } from 'components/account/Health/HealthGauge'
import EscButton from 'components/common/Button/EscButton' import EscButton from 'components/common/Button/EscButton'
import { glowElement } from 'components/common/Button/utils' import { glowElement } from 'components/common/Button/utils'
@ -24,8 +25,8 @@ import useLocalStorage from 'hooks/localStorage/useLocalStorage'
import useAccountId from 'hooks/useAccountId' import useAccountId from 'hooks/useAccountId'
import useBorrowMarketAssetsTableData from 'hooks/useBorrowMarketAssetsTableData' import useBorrowMarketAssetsTableData from 'hooks/useBorrowMarketAssetsTableData'
import useCurrentAccount from 'hooks/useCurrentAccount' import useCurrentAccount from 'hooks/useCurrentAccount'
import useHealthComputer from 'hooks/useHealthComputer'
import useHLSStakingAssets from 'hooks/useHLSStakingAssets' import useHLSStakingAssets from 'hooks/useHLSStakingAssets'
import useHealthComputer from 'hooks/useHealthComputer'
import useLendingMarketAssetsTableData from 'hooks/useLendingMarketAssetsTableData' import useLendingMarketAssetsTableData from 'hooks/useLendingMarketAssetsTableData'
import usePrices from 'hooks/usePrices' import usePrices from 'hooks/usePrices'
import useStore from 'store' import useStore from 'store'
@ -114,7 +115,10 @@ function AccountDetails(props: Props) {
), ),
[account, assets, borrowAssetsData, hlsStrategies, lendingAssetsData, prices, updatedAccount], [account, assets, borrowAssetsData, hlsStrategies, lendingAssetsData, prices, updatedAccount],
) )
const isFullWidth = location.pathname.includes('trade') || location.pathname === '/' const isFullWidth =
location.pathname.includes('trade') ||
location.pathname === '/' ||
location.pathname.includes('perps')
function AccountDetailsHeader() { function AccountDetailsHeader() {
const onClose = useCallback(() => useStore.setState({ accountDetailsExpanded: false }), []) const onClose = useCallback(() => useStore.setState({ accountDetailsExpanded: false }), [])
@ -215,6 +219,12 @@ function AccountDetails(props: Props) {
lendingData={lendingAssetsData} lendingData={lendingAssetsData}
hideCard hideCard
/> />
{account.perps.length > 0 && (
<>
<Text className='w-full px-4 py-2 text-white bg-white/10'>Perp Positions</Text>
<AccountPerpPositionTable account={account} hideCard />
</>
)}
</Card> </Card>
</div> </div>
</div> </div>

View File

@ -0,0 +1,71 @@
import { ReactNode } from 'react'
import DisplayCurrency from 'components/common/DisplayCurrency'
import { FormattedNumber } from 'components/common/FormattedNumber'
import Text from 'components/common/Text'
import { Tooltip } from 'components/common/Tooltip'
import TradeDirection from 'components/perps/BalancesTable/Columns/TradeDirection'
import { BN_ZERO } from 'constants/math'
import usePerpsEnabledAssets from 'hooks/assets/usePerpsEnabledAssets'
import { BNCoin } from 'types/classes/BNCoin'
import { demagnify } from 'utils/formatters'
export const ASSET_META = { accessorKey: 'symbol', header: 'Asset', id: 'symbol' }
interface Props {
row: AccountPerpRow
}
function LabelAndValue(props: { label: string; children: ReactNode; className?: string }) {
const { label, children, className } = props
return (
<div className='flex items-center justify-between'>
<Text size='sm' className='text-white/60'>
{label}
</Text>
{children}
</div>
)
}
function TooltipContent(props: Props) {
const { row } = props
const assets = usePerpsEnabledAssets()
const asset = assets.find((asset) => asset.symbol === row.symbol)
if (!asset) return null
return (
<div className='flex flex-col flex-wrap gap-1 w-50'>
<LabelAndValue label='Entry Price'>
<FormattedNumber amount={row.entryPrice.toNumber()} options={{ prefix: '$' }} />
</LabelAndValue>
<LabelAndValue label='Size'>
<Text size='sm'>{demagnify(row.amount, asset)}</Text>
</LabelAndValue>
<LabelAndValue label='Realized PnL'>
<DisplayCurrency
className='text-sm text-right number'
coin={BNCoin.fromDenomAndBigNumber('usd', BN_ZERO)}
showZero
/>
</LabelAndValue>
<LabelAndValue label='Unrealized PnL'>
<DisplayCurrency coin={row.pnl} options={{ abbreviated: false }} showZero isProfitOrLoss />
</LabelAndValue>
</div>
)
}
export default function Asset(props: Props) {
const { row } = props
return (
<Tooltip content={<TooltipContent row={row} />} type='info'>
<Text size='xs' className='flex items-center gap-1 no-wrap group/asset hover:cursor-help'>
<span className='pb-[1px] border-b border-white/20 border-dashed group-hover/asset:border-transparent'>
{row.symbol}
</span>
<TradeDirection tradeDirection={row.tradeDirection} />
</Text>
</Tooltip>
)
}

View File

@ -0,0 +1,22 @@
import DisplayCurrency from 'components/common/DisplayCurrency'
import { BNCoin } from 'types/classes/BNCoin'
export const PNL_META = { id: 'pnl', header: 'Total PnL', meta: { className: 'w-30' } }
interface Props {
pnl: BNCoin
}
export default function TotalPnL(props: Props) {
const { pnl } = props
return (
<DisplayCurrency
className='text-xs text-right number'
coin={pnl}
options={{ abbreviated: false }}
isProfitOrLoss
showZero
/>
)
}

View File

@ -0,0 +1,52 @@
import { ColumnDef } from '@tanstack/react-table'
import { useMemo } from 'react'
import LiqPrice, { LIQ_META } from 'components/account/AccountBalancesTable/Columns/LiqPrice'
import Value, {
VALUE_META,
valuePerpSortingFn,
} from 'components/account/AccountBalancesTable/Columns/Value'
import Asset, { ASSET_META } from 'components/account/AccountPerpPositionTable/Columns/Asset'
import TotalPnL, { PNL_META } from 'components/account/AccountPerpPositionTable/Columns/TotalPnL'
import useHealthComputer from 'hooks/useHealthComputer'
import useStore from 'store'
export default function useAccountPerpsColumns(account: Account, showLiquidationPrice?: boolean) {
const updatedAccount = useStore((s) => s.updatedAccount)
const { computeLiquidationPrice } = useHealthComputer(updatedAccount ?? account)
return useMemo<ColumnDef<AccountPerpRow>[]>(() => {
return [
{
...ASSET_META,
cell: ({ row }) => <Asset row={row.original} />,
},
{
...VALUE_META,
cell: ({ row }) => (
<Value amountChange={row.original.amountChange} value={row.original.value} type='perp' />
),
sortingFn: valuePerpSortingFn,
},
{
...LIQ_META,
header: 'Liq. Price',
enableSorting: false,
cell: ({ row }) => (
<LiqPrice
denom={row.original.denom}
computeLiquidationPrice={computeLiquidationPrice}
type='perp'
amount={row.original.amount.toNumber()}
account={updatedAccount ?? account}
/>
),
},
{
...PNL_META,
cell: ({ row }) => <TotalPnL pnl={row.original.pnl} />,
},
]
}, [computeLiquidationPrice, account, updatedAccount])
}

View File

@ -0,0 +1,20 @@
import { BNCoin } from 'types/classes/BNCoin'
import { getCoinValue } from 'utils/formatters'
export function getAssetAccountPerpRow(
asset: Asset,
prices: BNCoin[],
position: PerpsPosition,
assets: Asset[],
prev?: BNCoin,
): AccountPerpRow {
const { denom, amount } = position
const amountChange = !prev ? position.amount : position.amount.minus(prev.amount)
return {
symbol: asset.symbol,
value: getCoinValue(BNCoin.fromDenomAndBigNumber(denom, amount), prices, assets).toString(),
amountChange,
...position,
}
}

View File

@ -0,0 +1,37 @@
import classNames from 'classnames'
import useAccountPerpsColumns from 'components/account/AccountPerpPositionTable/Columns/useAccountPerpsColumns'
import useAccountPerpData from 'components/account/AccountPerpPositionTable/useAccountPerpData'
import Table from 'components/common/Table'
import useStore from 'store'
interface Props {
account: Account
tableBodyClassName?: string
showLiquidationPrice?: boolean
hideCard?: boolean
}
export default function AccountPerpPositionTable(props: Props) {
const { account, tableBodyClassName, showLiquidationPrice, hideCard } = props
const updatedAccount = useStore((s) => s.updatedAccount)
const accountPerpData = useAccountPerpData({
account,
updatedAccount,
})
const columns = useAccountPerpsColumns(account, showLiquidationPrice)
return (
<Table
title='Perp Positions'
columns={columns}
data={accountPerpData}
tableBodyClassName={classNames(tableBodyClassName, 'text-white/60')}
initialSorting={[]}
spacingClassName='p-2'
hideCard={hideCard}
type='perps'
/>
)
}

View File

@ -0,0 +1,30 @@
import { useMemo } from 'react'
import { getAssetAccountPerpRow } from 'components/account/AccountPerpPositionTable/functions'
import useAllAssets from 'hooks/assets/useAllAssets'
import usePrices from 'hooks/usePrices'
import { byDenom } from 'utils/array'
interface Props {
account: Account
updatedAccount?: Account
}
export default function useAccountPerpData(props: Props) {
const { account, updatedAccount } = props
const { data: prices } = usePrices()
const assets = useAllAssets()
return useMemo<AccountPerpRow[]>(() => {
const usedAccount = updatedAccount ?? account
const accountPerps = usedAccount?.perps ?? []
return accountPerps.map((perp) => {
const asset = assets.find(byDenom(perp.denom)) ?? assets[0]
const prevPerp = updatedAccount
? account?.perps.find((position) => position.denom === perp.denom)
: perp
return getAssetAccountPerpRow(asset, prices, perp, assets, prevPerp)
})
}, [updatedAccount, account, prices, assets])
}

View File

@ -1,10 +1,11 @@
import classNames from 'classnames' import classNames from 'classnames'
import { HTMLAttributes, useCallback, useMemo } from 'react' import { HTMLAttributes, useCallback, useMemo } from 'react'
import Accordion from 'components/common/Accordion'
import AccountBalancesTable from 'components/account/AccountBalancesTable' import AccountBalancesTable from 'components/account/AccountBalancesTable'
import AccountComposition from 'components/account/AccountComposition' import AccountComposition from 'components/account/AccountComposition'
import AccountPerpPositionTable from 'components/account/AccountPerpPositionTable'
import HealthBar from 'components/account/Health/HealthBar' import HealthBar from 'components/account/Health/HealthBar'
import Accordion from 'components/common/Accordion'
import Card from 'components/common/Card' import Card from 'components/common/Card'
import DisplayCurrency from 'components/common/DisplayCurrency' import DisplayCurrency from 'components/common/DisplayCurrency'
import { FormattedNumber } from 'components/common/FormattedNumber' import { FormattedNumber } from 'components/common/FormattedNumber'
@ -150,6 +151,16 @@ export default function AccountSummary(props: Props) {
toggleOpen: (index: number) => handleToggle(index), toggleOpen: (index: number) => handleToggle(index),
renderSubTitle: () => <></>, renderSubTitle: () => <></>,
}, },
{
title: 'Perp Positions',
renderContent: () =>
props.account && props.account.perps.length > 0 ? (
<AccountPerpPositionTable account={props.account} hideCard />
) : null,
isOpen: accountSummaryTabs[2] ?? false,
toggleOpen: (index: number) => handleToggle(index),
renderSubTitle: () => <></>,
},
]} ]}
allowMultipleOpen allowMultipleOpen
/> />

View File

@ -22,6 +22,7 @@ interface Props {
} }
export default function DisplayCurrency(props: Props) { export default function DisplayCurrency(props: Props) {
const { coin, className, isApproximation, parentheses, showZero, options, isProfitOrLoss } = props
const displayCurrencies = useDisplayCurrencyAssets() const displayCurrencies = useDisplayCurrencyAssets()
const assets = useAllAssets() const assets = useAllAssets()
const [displayCurrency] = useDisplayCurrency() const [displayCurrency] = useDisplayCurrency()
@ -36,7 +37,7 @@ export default function DisplayCurrency(props: Props) {
const isUSD = displayCurrencyAsset.id === 'USD' const isUSD = displayCurrencyAsset.id === 'USD'
const [amount, absoluteAmount] = useMemo(() => { const [amount, absoluteAmount] = useMemo(() => {
const coinValue = getCoinValue(props.coin, prices, assets) const coinValue = getCoinValue(coin, prices, assets)
if (displayCurrency === ORACLE_DENOM) return [coinValue.toNumber(), coinValue.abs().toNumber()] if (displayCurrency === ORACLE_DENOM) return [coinValue.toNumber(), coinValue.abs().toNumber()]
@ -50,18 +51,29 @@ export default function DisplayCurrency(props: Props) {
const amount = coinValue.div(displayPrice).toNumber() const amount = coinValue.div(displayPrice).toNumber()
return [amount, Math.abs(amount)] return [amount, Math.abs(amount)]
}, [assets, displayCurrency, displayCurrencyAsset.decimals, prices, props.coin]) }, [assets, displayCurrency, displayCurrencyAsset.decimals, prices, coin])
const isLessThanACent = useMemo( const isLessThanACent = useMemo(
() => isUSD && absoluteAmount < 0.01 && absoluteAmount > 0, () => isUSD && absoluteAmount < 0.01 && absoluteAmount > 0,
[absoluteAmount, isUSD], [absoluteAmount, isUSD],
) )
const smallerThanPrefix = isLessThanACent && !props.showZero ? '< ' : '' const prefix = useMemo(() => {
const positiveOrNegativePrefix = isProfitOrLoss
? amount > 0
? '+'
: amount < 0
? '-'
: ''
: ''
const approximationPrefix = isApproximation ? '~ ' : ''
const smallerThanPrefix = isLessThanACent && !showZero ? '< ' : ''
return isUSD
? `${approximationPrefix}${smallerThanPrefix}${positiveOrNegativePrefix}$`
: `${approximationPrefix}${smallerThanPrefix}${positiveOrNegativePrefix}`
}, [isUSD, isApproximation, showZero, isProfitOrLoss, amount, isLessThanACent])
const prefix = isUSD
? `${props.isApproximation ? '~ ' : smallerThanPrefix}$`
: `${props.isApproximation ? '~ ' : ''}`
const suffix = isUSD const suffix = isUSD
? '' ? ''
: ` ${displayCurrencyAsset.symbol ? ` ${displayCurrencyAsset.symbol}` : ''}` : ` ${displayCurrencyAsset.symbol ? ` ${displayCurrencyAsset.symbol}` : ''}`
@ -69,19 +81,19 @@ export default function DisplayCurrency(props: Props) {
return ( return (
<FormattedNumber <FormattedNumber
className={classNames( className={classNames(
props.className, className,
props.parentheses && 'before:content-["("] after:content-[")"]', parentheses && 'before:content-["("] after:content-[")"]',
props.isProfitOrLoss && (amount < 0 ? 'text-error' : amount === 0 ? '' : 'text-success'), isProfitOrLoss && amount < 0 && 'text-loss',
props.isProfitOrLoss && amount < 0 && 'before:content-["-"]', isProfitOrLoss && amount > 0 && 'text-profit',
)} )}
amount={isLessThanACent ? 0.01 : absoluteAmount} amount={isLessThanACent ? 0.01 : absoluteAmount}
options={{ options={{
minDecimals: isUSD ? 2 : 0, minDecimals: isUSD ? 2 : 0,
maxDecimals: 2, maxDecimals: 2,
abbreviated: true, abbreviated: true,
prefix,
suffix, suffix,
...props.options, ...options,
prefix,
}} }}
animate animate
/> />

View File

@ -7,51 +7,56 @@ interface Props<T> {
renderExpanded?: (row: TanstackRow<T>, table: TanstackTable<T>) => JSX.Element renderExpanded?: (row: TanstackRow<T>, table: TanstackTable<T>) => JSX.Element
rowClassName?: string rowClassName?: string
spacingClassName?: string spacingClassName?: string
isBalancesTable?: boolean
className?: string className?: string
isSelectable?: boolean isSelectable?: boolean
type?: TableType
} }
function getBorderColor(row: AccountBalanceRow) { function getBorderColor(type: TableType, row: AccountBalanceRow | AccountPerpRow) {
return row.type === 'borrowing' ? 'border-loss' : 'border-profit' if (type === 'balances') {
const balancesRow = row as AccountBalanceRow
return balancesRow.type === 'borrow' ? 'border-loss' : 'border-profit'
}
const perpRow = row as AccountPerpRow
return perpRow.tradeDirection === 'short' ? 'border-loss' : 'border-profit'
} }
export default function Row<T>(props: Props<T>) { export default function Row<T>(props: Props<T>) {
const canExpand = !!props.renderExpanded const { renderExpanded, table, row, type, spacingClassName, isSelectable } = props
const canExpand = !!renderExpanded
return ( return (
<> <>
<tr <tr
key={`${props.row.id}-row`} key={`${row.id}-row`}
className={classNames( className={classNames(
'group/row transition-bg', 'group/row transition-bg',
(props.renderExpanded || props.isSelectable) && 'hover:cursor-pointer', (renderExpanded || isSelectable) && 'hover:cursor-pointer',
canExpand && props.row.getIsExpanded() ? 'is-expanded bg-black/20' : 'hover:bg-white/5', canExpand && row.getIsExpanded() ? 'is-expanded bg-black/20' : 'hover:bg-white/5',
)} )}
onClick={(e) => { onClick={(e) => {
e.preventDefault() e.preventDefault()
if (props.isSelectable) { if (isSelectable) {
props.row.toggleSelected() row.toggleSelected()
} }
if (canExpand) { if (canExpand) {
const isExpanded = props.row.getIsExpanded() const isExpanded = row.getIsExpanded()
props.table.resetExpanded() table.resetExpanded()
!isExpanded && props.row.toggleExpanded() !isExpanded && row.toggleExpanded()
} }
}} }}
> >
{props.row.getVisibleCells().map((cell) => { {row.getVisibleCells().map((cell) => {
const isSymbolOrName = cell.column.id === 'symbol' || cell.column.id === 'name' const isSymbolOrName = cell.column.id === 'symbol' || cell.column.id === 'name'
const borderClasses =
props.isBalancesTable && isSymbolOrName
? classNames('border-l', getBorderColor(cell.row.original as AccountBalanceRow))
: ''
return ( return (
<td <td
key={cell.id} key={cell.id}
className={classNames( className={classNames(
isSymbolOrName ? 'text-left' : 'text-right', isSymbolOrName ? 'text-left' : 'text-right',
props.spacingClassName ?? 'px-3 py-4', spacingClassName ?? 'px-3 py-4',
borderClasses, type && isSymbolOrName && 'border-l',
type && getBorderColor(type, cell.row.original as any),
cell.column.columnDef.meta?.className, cell.column.columnDef.meta?.className,
)} )}
> >
@ -60,9 +65,7 @@ export default function Row<T>(props: Props<T>) {
) )
})} })}
</tr> </tr>
{props.row.getIsExpanded() && {row.getIsExpanded() && renderExpanded && renderExpanded(row, table)}
props.renderExpanded &&
props.renderExpanded(props.row, props.table)}
</> </>
) )
} }

View File

@ -27,7 +27,7 @@ interface Props<T> {
renderExpanded?: (row: TanstackRow<T>, table: TanstackTable<T>) => JSX.Element renderExpanded?: (row: TanstackRow<T>, table: TanstackTable<T>) => JSX.Element
tableBodyClassName?: string tableBodyClassName?: string
spacingClassName?: string spacingClassName?: string
isBalancesTable?: boolean type?: TableType
hideCard?: boolean hideCard?: boolean
setRowSelection?: OnChangeFn<RowSelectionState> setRowSelection?: OnChangeFn<RowSelectionState>
selectedRows?: RowSelectionState selectedRows?: RowSelectionState
@ -55,7 +55,7 @@ export default function Table<T>(props: Props<T>) {
condition={!props.hideCard} condition={!props.hideCard}
wrapper={(children) => ( wrapper={(children) => (
<Card <Card
className={classNames('w-full', !props.isBalancesTable && 'h-fit bg-white/5')} className={classNames('w-full', props.type !== 'balances' && 'h-fit bg-white/5')}
title={props.title} title={props.title}
> >
{children} {children}
@ -118,8 +118,8 @@ export default function Table<T>(props: Props<T>) {
table={table} table={table}
renderExpanded={props.renderExpanded} renderExpanded={props.renderExpanded}
spacingClassName={props.spacingClassName} spacingClassName={props.spacingClassName}
isBalancesTable={props.isBalancesTable}
isSelectable={!!props.setRowSelection} isSelectable={!!props.setRowSelection}
type={props.type}
/> />
))} ))}
</tbody> </tbody>

View File

@ -20,13 +20,10 @@ export const depositedSortingFn = (
interface Props { interface Props {
asset: Asset asset: Asset
lentAmount?: string lentAmount?: BigNumber
} }
export default function DepositValue(props: Props) { export default function DepositValue(props: Props) {
return ( return (
<AmountAndValue <AmountAndValue asset={props.asset} amount={props.lentAmount ? props.lentAmount : BN_ZERO} />
asset={props.asset}
amount={props.lentAmount ? BN(props.lentAmount) : BN_ZERO}
/>
) )
} }

View File

@ -1,9 +1,8 @@
import React, { useMemo } from 'react' import { useMemo } from 'react'
import { useSearchParams } from 'react-router-dom' import { useSearchParams } from 'react-router-dom'
import DropDownButton from 'components/common/Button/DropDownButton' import DropDownButton from 'components/common/Button/DropDownButton'
import { Cross, Edit } from 'components/common/Icons' import { Cross, Edit } from 'components/common/Icons'
import { PerpPositionRow } from 'components/perps/BalancesTable/usePerpsBalancesData'
import useCurrentAccount from 'hooks/useCurrentAccount' import useCurrentAccount from 'hooks/useCurrentAccount'
import useStore from 'store' import useStore from 'store'
import { SearchParams } from 'types/enums/searchParams' import { SearchParams } from 'types/enums/searchParams'

View File

@ -22,7 +22,7 @@ export default function PnL(props: Props) {
type='info' type='info'
underline underline
> >
<DisplayCurrency className='inline text-xs' coin={props.pnl} isProfitOrLoss /> <DisplayCurrency className='inline text-xs' coin={props.pnl} isProfitOrLoss showZero />
</Tooltip> </Tooltip>
) )
} }
@ -34,13 +34,18 @@ type PnLTooltipProps = {
function PnLTooltip(props: PnLTooltipProps) { function PnLTooltip(props: PnLTooltipProps) {
return ( return (
<div className='flex flex-col gap-2 w-full'> <div className='flex flex-col w-full gap-2'>
{[props.realized, props.unrealized].map((coin, i) => ( {[props.realized, props.unrealized].map((coin, i) => (
<div key={i} className='flex w-full text-white/60 space-between items-center gap-8'> <div key={i} className='flex items-center w-full gap-8 space-between'>
<Text className='mr-auto' size='sm'> <Text className='mr-auto text-white/60' size='sm'>
{i === 0 ? 'Realized' : 'Unrealized'} PnL {i === 0 ? 'Realized' : 'Unrealized'} PnL
</Text> </Text>
<DisplayCurrency coin={coin} className='self-end text-end' isProfitOrLoss /> <DisplayCurrency
coin={coin}
className='self-end text-sm text-end'
isProfitOrLoss
showZero
/>
</div> </div>
))} ))}
</div> </div>

View File

@ -19,7 +19,7 @@ export const SIZE_META = {
} }
type Props = { type Props = {
size: BigNumber amount: BigNumber
asset: Asset asset: Asset
} }
@ -27,8 +27,8 @@ export default function Size(props: Props) {
const price = usePrice(props.asset.denom) const price = usePrice(props.asset.denom)
const amount = useMemo( const amount = useMemo(
() => demagnify(props.size.toString(), props.asset), () => demagnify(props.amount.toString(), props.asset),
[props.asset, props.size], [props.asset, props.amount],
) )
const value = useMemo(() => price.times(amount).toNumber(), [amount, price]) const value = useMemo(() => price.times(amount).toNumber(), [amount, price])
return ( return (

View File

@ -1,25 +1,24 @@
import classNames from 'classnames' import classNames from 'classnames'
import Text from 'components/common/Text'
export const PERP_TYPE_META = { accessorKey: 'tradeDirection', header: 'Side' } export const PERP_TYPE_META = { accessorKey: 'tradeDirection', header: 'Side' }
type Props = { type Props = {
tradeDirection: TradeDirection tradeDirection: TradeDirection
className?: string
} }
export default function TradeDirection(props: Props) { export default function TradeDirection(props: Props) {
const { tradeDirection } = props const { tradeDirection, className } = props
return ( return (
<Text <span
size='xs'
className={classNames( className={classNames(
'capitalize px-1 py-0.5 rounded-sm inline-block', 'capitalize px-1 py-0.5 rounded-sm inline-block text-xs',
tradeDirection === 'short' && 'text-error bg-error/20', tradeDirection === 'short' && 'text-error bg-error/20',
tradeDirection === 'long' && 'text-success bg-success/20', tradeDirection === 'long' && 'text-success bg-success/20',
className,
)} )}
> >
{tradeDirection} {tradeDirection}
</Text> </span>
) )
} }

View File

@ -10,7 +10,6 @@ import Size, { SIZE_META } from 'components/perps/BalancesTable/Columns/Size'
import TradeDirection, { import TradeDirection, {
PERP_TYPE_META, PERP_TYPE_META,
} from 'components/perps/BalancesTable/Columns/TradeDirection' } from 'components/perps/BalancesTable/Columns/TradeDirection'
import { PerpPositionRow } from 'components/perps/BalancesTable/usePerpsBalancesData'
export default function usePerpsBalancesTable() { export default function usePerpsBalancesTable() {
return useMemo<ColumnDef<PerpPositionRow>[]>(() => { return useMemo<ColumnDef<PerpPositionRow>[]>(() => {
@ -25,7 +24,7 @@ export default function usePerpsBalancesTable() {
}, },
{ {
...SIZE_META, ...SIZE_META,
cell: ({ row }) => <Size size={row.original.size} asset={row.original.asset} />, cell: ({ row }) => <Size amount={row.original.amount} asset={row.original.asset} />,
}, },
{ {
...LEVERAGE_META, ...LEVERAGE_META,

View File

@ -5,7 +5,6 @@ import useAllAssets from 'hooks/assets/useAllAssets'
import usePerpsEnabledAssets from 'hooks/assets/usePerpsEnabledAssets' import usePerpsEnabledAssets from 'hooks/assets/usePerpsEnabledAssets'
import useCurrentAccount from 'hooks/useCurrentAccount' import useCurrentAccount from 'hooks/useCurrentAccount'
import usePrices from 'hooks/usePrices' import usePrices from 'hooks/usePrices'
import { BNCoin } from 'types/classes/BNCoin'
import { getAccountNetValue } from 'utils/accounts' import { getAccountNetValue } from 'utils/accounts'
import { byDenom } from 'utils/array' import { byDenom } from 'utils/array'
import { demagnify } from 'utils/formatters' import { demagnify } from 'utils/formatters'
@ -24,25 +23,16 @@ export default function usePerpsBalancesTable() {
return currentAccount.perps.map((position) => { return currentAccount.perps.map((position) => {
const price = prices.find(byDenom(position.denom))?.amount ?? BN_ZERO const price = prices.find(byDenom(position.denom))?.amount ?? BN_ZERO
const asset = perpAssets.find(byDenom(position.denom))! const asset = perpAssets.find(byDenom(position.denom))!
return { return {
asset, asset,
tradeDirection: position.tradeDirection, tradeDirection: position.tradeDirection,
size: position.size, amount: position.amount,
pnl: position.pnl, pnl: position.pnl,
entryPrice: position.entryPrice, entryPrice: position.entryPrice,
liquidationPrice: position.entryPrice, // TODO: 📈 Get actual liquidation price from HC liquidationPrice: position.entryPrice, // TODO: 📈 Get actual liquidation price from HC
leverage: price.times(demagnify(position.size, asset)).div(netValue).plus(1).toNumber(), leverage: price.times(demagnify(position.amount, asset)).div(netValue).plus(1).toNumber(),
} as PerpPositionRow } as PerpPositionRow
}) })
}, [allAssets, currentAccount, perpAssets, prices]) }, [allAssets, currentAccount, perpAssets, prices])
} }
export type PerpPositionRow = {
asset: Asset
tradeDirection: TradeDirection
size: BigNumber
pnl: BNCoin
entryPrice: BigNumber
liquidationPrice: BigNumber
leverage: number
}

View File

@ -35,7 +35,7 @@ export default function usePerpsManageModule(amount: BigNumber | null) {
}) })
}, [searchParams, setSearchParams]) }, [searchParams, setSearchParams])
const previousAmount = useMemo(() => perpPosition?.size ?? BN_ZERO, [perpPosition?.size]) const previousAmount = useMemo(() => perpPosition?.amount ?? BN_ZERO, [perpPosition?.amount])
const previousTradeDirection = useMemo( const previousTradeDirection = useMemo(
() => perpPosition?.tradeDirection || 'long', () => perpPosition?.tradeDirection || 'long',
[perpPosition?.tradeDirection], [perpPosition?.tradeDirection],

View File

@ -1,4 +1,4 @@
import PerpsBalancesTable from './BalancesTable' import PerpsBalancesTable from 'components/perps/BalancesTable'
export function PerpsPositions() { export function PerpsPositions() {
return <PerpsBalancesTable /> return <PerpsBalancesTable />

View File

@ -0,0 +1,50 @@
import React, { Suspense } from 'react'
import AccountPerpPositionTable from 'components/account/AccountPerpPositionTable'
import Card from 'components/common/Card'
import TableSkeleton from 'components/common/Table/TableSkeleton'
import Text from 'components/common/Text'
import useAccount from 'hooks/accounts/useAccount'
interface Props {
accountId: string
}
function Content(props: Props) {
const { data: account } = useAccount(props.accountId, true)
if (!account || account.perps.length === 0) return null
return (
<Skeleton>
<AccountPerpPositionTable account={account} showLiquidationPrice hideCard />
</Skeleton>
)
}
export default function PerpPositions(props: Props) {
return (
<Suspense fallback={<Skeleton />}>
<Content {...props} />
</Suspense>
)
}
interface SkeletonProps {
children?: React.ReactNode
}
function Skeleton(props: SkeletonProps) {
return (
<div className='flex flex-wrap w-full gap-4'>
<Text size='2xl'>Perp Positions</Text>
<Card className='w-full bg-white/5'>
{props.children ? (
props.children
) : (
<TableSkeleton labels={['Asset', 'Value', 'Liq. Price', 'Total PnL']} rowCount={3} />
)}
</Card>
</div>
)
}

View File

@ -6,12 +6,11 @@ import Overlay from 'components/common/Overlay'
import SearchBar from 'components/common/SearchBar' import SearchBar from 'components/common/SearchBar'
import Text from 'components/common/Text' import Text from 'components/common/Text'
import AssetList from 'components/trade/TradeModule/AssetSelector/AssetList' import AssetList from 'components/trade/TradeModule/AssetSelector/AssetList'
import StablesFilter from 'components/trade/TradeModule/AssetSelector/AssetOverlay/StablesFilter'
import PairsList from 'components/trade/TradeModule/AssetSelector/PairsList' import PairsList from 'components/trade/TradeModule/AssetSelector/PairsList'
import useAllAssets from 'hooks/assets/useAllAssets' import useAllAssets from 'hooks/assets/useAllAssets'
import useFilteredAssets from 'hooks/useFilteredAssets' import useFilteredAssets from 'hooks/useFilteredAssets'
import StablesFilter from './StablesFilter'
interface Props { interface Props {
state: OverlayState state: OverlayState
buyAsset: Asset buyAsset: Asset

View File

@ -8,7 +8,7 @@ const enabledMarketAssets = useStore
.chainConfig.assets.filter((asset) => asset.isEnabled && asset.isMarket) .chainConfig.assets.filter((asset) => asset.isEnabled && asset.isMarket)
export const DEFAULT_SETTINGS: Settings = { export const DEFAULT_SETTINGS: Settings = {
accountSummaryTabs: [true, true], accountSummaryTabs: [true, true, false],
reduceMotion: false, reduceMotion: false,
enableAutoLendGlobal: true, enableAutoLendGlobal: true,
tradingPairSimple: { tradingPairSimple: {

View File

@ -1,9 +1,8 @@
import { useMemo } from 'react' import { useMemo } from 'react'
import useCurrentAccount from 'hooks/useCurrentAccount'
import { byDenom } from 'utils/array' import { byDenom } from 'utils/array'
import useCurrentAccount from '../useCurrentAccount'
export default function usePerpPosition(denom: string): PerpsPosition | undefined { export default function usePerpPosition(denom: string): PerpsPosition | undefined {
const account = useCurrentAccount() const account = useCurrentAccount()

View File

@ -1,5 +1,6 @@
import { useMemo } from 'react' import { useMemo } from 'react'
import { BN_ZERO } from 'constants/math'
import useAllAssets from 'hooks/assets/useAllAssets' import useAllAssets from 'hooks/assets/useAllAssets'
import useMarketDeposits from 'hooks/markets/useMarketDeposits' import useMarketDeposits from 'hooks/markets/useMarketDeposits'
import useMarketLiquidities from 'hooks/markets/useMarketLiquidities' import useMarketLiquidities from 'hooks/markets/useMarketLiquidities'
@ -29,7 +30,7 @@ function useLendingMarketAssetsTableData(): {
const asset = assets.find(byDenom(denom)) as Asset const asset = assets.find(byDenom(denom)) as Asset
const marketDepositAmount = BN(marketDeposits.find(byDenom(denom))?.amount ?? 0) const marketDepositAmount = BN(marketDeposits.find(byDenom(denom))?.amount ?? 0)
const marketLiquidityAmount = BN(marketLiquidities.find(byDenom(denom))?.amount ?? 0) const marketLiquidityAmount = BN(marketLiquidities.find(byDenom(denom))?.amount ?? 0)
const accountLentAmount = accountLentAmounts.find(byDenom(denom))?.amount const accountLentAmount = accountLentAmounts.find(byDenom(denom))?.amount ?? BN_ZERO
const accountLentValue = accountLentAmount const accountLentValue = accountLentAmount
? convertAmount(asset, accountLentAmount) ? convertAmount(asset, accountLentAmount)
: undefined : undefined
@ -46,9 +47,11 @@ function useLendingMarketAssetsTableData(): {
cap, cap,
} }
;(lendingMarketAsset.accountLentValue ? accountLentAssets : availableAssets).push( if (lendingMarketAsset.accountLentAmount?.isZero()) {
lendingMarketAsset, availableAssets.push(lendingMarketAsset)
) } else {
accountLentAssets.push(lendingMarketAsset)
}
}) })
return { return {

View File

@ -1,14 +1,17 @@
import { useNavigate, useParams, useSearchParams } from 'react-router-dom' import { useNavigate, useParams, useSearchParams } from 'react-router-dom'
import MigrationBanner from 'components/common/MigrationBanner' import MigrationBanner from 'components/common/MigrationBanner'
import ShareBar from 'components/common/ShareBar'
import Balances from 'components/portfolio/Account/Balances' import Balances from 'components/portfolio/Account/Balances'
import BreadCrumbs from 'components/portfolio/Account/BreadCrumbs' import BreadCrumbs from 'components/portfolio/Account/BreadCrumbs'
import PerpPositions from 'components/portfolio/Account/PerpPositions'
import Summary from 'components/portfolio/Account/Summary' import Summary from 'components/portfolio/Account/Summary'
import ShareBar from 'components/common/ShareBar'
import useAccountId from 'hooks/useAccountId' import useAccountId from 'hooks/useAccountId'
import useStore from 'store'
import { getRoute } from 'utils/route' import { getRoute } from 'utils/route'
export default function PortfolioAccountPage() { export default function PortfolioAccountPage() {
const chainConfig = useStore((s) => s.chainConfig)
const selectedAccountId = useAccountId() const selectedAccountId = useAccountId()
const { address, accountId } = useParams() const { address, accountId } = useParams()
const navigate = useNavigate() const navigate = useNavigate()
@ -25,6 +28,7 @@ export default function PortfolioAccountPage() {
<BreadCrumbs accountId={accountId} /> <BreadCrumbs accountId={accountId} />
<Summary accountId={accountId} /> <Summary accountId={accountId} />
<Balances accountId={accountId} /> <Balances accountId={accountId} />
{chainConfig.perps && <PerpPositions accountId={accountId} />}
<ShareBar text={`Have a look at Credit Account ${accountId} on @mars_protocol!`} /> <ShareBar text={`Have a look at Credit Account ${accountId} on @mars_protocol!`} />
</div> </div>
) )

View File

@ -1 +1,3 @@
export { ActionCoin } from 'types/generated/mars-credit-manager/MarsCreditManager.types' type ActionCoin = import('types/generated/mars-credit-manager/MarsCreditManager.types').ActionCoin
type BNCoin = import('types/classes/BNCoin').BNCoin

View File

@ -264,7 +264,7 @@ export interface UserDebtResponse {
} }
export type ArrayOfUserDebtResponse = UserDebtResponse[] export type ArrayOfUserDebtResponse = UserDebtResponse[]
export type UserHealthStatus = export type UserHealthStatus =
| 'not_borrowing' | 'not_borrow'
| { | {
borrowing: { borrowing: {
liq_threshold_hf: Decimal liq_threshold_hf: Decimal

View File

@ -1,4 +1,7 @@
interface Account extends AccountChange { type PositionType = 'deposit' | 'borrow' | 'lend' | 'vault' | 'perp'
type TableType = 'balances' | 'perps'
interface Account {
id: string id: string
deposits: BNCoin[] deposits: BNCoin[]
debts: BNCoin[] debts: BNCoin[]
@ -8,7 +11,7 @@ interface Account extends AccountChange {
kind: AccountKind kind: AccountKind
} }
interface AccountChange { interface AccountChange extends Account {
deposits?: BNCoin[] deposits?: BNCoin[]
debts?: BNCoin[] debts?: BNCoin[]
lends?: BNCoin[] lends?: BNCoin[]
@ -21,7 +24,14 @@ interface AccountBalanceRow {
denom: string denom: string
size: number size: number
symbol: string symbol: string
type: 'deposits' | 'borrowing' | 'lending' | 'vault' type: PositionType
value: string
amountChange: BigNumber
}
interface AccountPerpRow extends PerpsPosition {
amount: BigNumber
symbol: string
value: string value: string
amountChange: BigNumber amountChange: BigNumber
} }

View File

@ -59,7 +59,7 @@ interface BorrowMarketTableData extends MarketTableData {
} }
interface LendingMarketTableData extends MarketTableData { interface LendingMarketTableData extends MarketTableData {
accountLentAmount?: string accountLentAmount?: BigNumber
accountLentValue?: BigNumber accountLentValue?: BigNumber
borrowEnabled: boolean borrowEnabled: boolean
cap: DepositCap cap: DepositCap

View File

@ -1,5 +1,3 @@
const BNCoin = import('types/classes/BNCoin').BNCoin
type TradeDirection = 'long' | 'short' type TradeDirection = 'long' | 'short'
// TODO: 📈Remove this type when healthcomputer is implemented // TODO: 📈Remove this type when healthcomputer is implemented
@ -8,9 +6,18 @@ type PositionsWithoutPerps = Omit<
'perps' 'perps'
> >
type PerpsPosition = { interface PerpsPosition {
denom: string denom: string
baseDenom: string baseDenom: string
tradeDirection: TradeDirection tradeDirection: TradeDirection
size: BigNumber amount: BigNumber
closingFee: BNCoin
pnl: BNCoin
entryPrice: BigNumber
}
interface PerpPositionRow extends PerpsPosition {
asset: Asset
liquidationPrice: BigNumber
leverage: number
} }

View File

@ -1,6 +1,3 @@
const BNCoin = import('types/classes/BNCoin').BNCoin
const ActionCoin = import('types/generated').ActionCoin
interface BroadcastResult { interface BroadcastResult {
result?: import('@delphi-labs/shuttle-react').BroadcastResult result?: import('@delphi-labs/shuttle-react').BroadcastResult
error?: string error?: string

View File

@ -53,14 +53,16 @@ export const calculateAccountValue = (
) )
} }
return account[type]?.reduce((acc, position) => { return (
const asset = assets.find(byDenom(position.denom)) account[type]?.reduce((acc, position) => {
if (!asset) return acc const asset = assets.find(byDenom(position.denom))
const price = prices.find((price) => price.denom === position.denom)?.amount ?? 0 if (!asset) return acc
const amount = BN(position.amount).shiftedBy(-asset.decimals) const price = prices.find((price) => price.denom === position.denom)?.amount ?? 0
const positionValue = amount.multipliedBy(price) const amount = BN(position.amount).shiftedBy(-asset.decimals)
return acc.plus(positionValue) const positionValue = amount.multipliedBy(price)
}, BN_ZERO) return acc.plus(positionValue)
}, BN_ZERO) ?? BN_ZERO
)
} }
export const calculateAccountApr = ( export const calculateAccountApr = (

View File

@ -83,7 +83,7 @@ export function resolvePerpsPositions(perpPositions: Positions['perps']): PerpsP
return { return {
denom: position.denom, denom: position.denom,
baseDenom: position.base_denom, baseDenom: position.base_denom,
size: BN(position.size as any).abs(), amount: BN(position.size as any).abs(),
tradeDirection: BN(position.size as any).isNegative() ? 'short' : 'long', tradeDirection: BN(position.size as any).isNegative() ? 'short' : 'long',
closingFee: BNCoin.fromCoin(position.pnl.coins.closing_fee), closingFee: BNCoin.fromCoin(position.pnl.coins.closing_fee),
pnl: getPnlCoin(position.pnl.coins.pnl, position.base_denom), pnl: getPnlCoin(position.pnl.coins.pnl, position.base_denom),