mango-v4-ui/components/trade/MarketSelectDropdown.tsx

534 lines
22 KiB
TypeScript
Raw Normal View History

2022-11-30 07:46:20 -08:00
import FavoriteMarketButton from '@components/shared/FavoriteMarketButton'
import { Popover } from '@headlessui/react'
2023-10-01 20:42:34 -07:00
import {
ChevronDownIcon,
ExclamationTriangleIcon,
MagnifyingGlassIcon,
} from '@heroicons/react/20/solid'
2023-04-03 19:48:31 -07:00
import useMangoGroup from 'hooks/useMangoGroup'
2022-11-30 07:46:20 -08:00
import useSelectedMarket from 'hooks/useSelectedMarket'
2023-02-22 11:04:18 -08:00
import { useTranslation } from 'next-i18next'
2022-11-30 07:46:20 -08:00
import Link from 'next/link'
import { ChangeEvent, useEffect, useMemo, useRef, useState } from 'react'
2023-06-18 22:39:18 -07:00
import {
2023-11-16 04:35:44 -08:00
countLeadingZeros,
2023-06-18 22:39:18 -07:00
floorToDecimal,
formatCurrencyValue,
formatNumericValue,
getDecimalCount,
2023-07-24 05:14:45 -07:00
numberCompacter,
2023-06-18 22:39:18 -07:00
} from 'utils/numbers'
2022-11-30 07:46:20 -08:00
import MarketLogos from './MarketLogos'
2023-06-14 17:43:33 -07:00
import SoonBadge from '@components/shared/SoonBadge'
2023-06-18 22:39:18 -07:00
import TabButtons from '@components/shared/TabButtons'
import { PerpMarket } from '@blockworks-foundation/mango-v4'
2023-06-29 21:00:28 -07:00
import Loading from '@components/shared/Loading'
2023-07-13 22:47:05 -07:00
import MarketChange from '@components/shared/MarketChange'
2023-07-24 05:14:45 -07:00
import SheenLoader from '@components/shared/SheenLoader'
2023-10-03 22:05:16 -07:00
import useListedMarketsWithMarketData from 'hooks/useListedMarketsWithMarketData'
import {
AllowedKeys,
sortPerpMarkets,
sortSpotMarkets,
startSearch,
} from 'utils/markets'
import Input from '@components/forms/Input'
2023-09-08 15:01:08 -07:00
import { useSortableData } from 'hooks/useSortableData'
import { SortableColumnHeader } from '@components/shared/TableElements'
import { useViewport } from 'hooks/useViewport'
2023-12-04 13:17:13 -08:00
import { useRouter } from 'next/router'
import { TOKEN_REDUCE_ONLY_OPTIONS } from 'utils/constants'
import { isBankVisibleForUser } from 'utils/bank'
import Decimal from 'decimal.js'
import useMangoAccount from 'hooks/useMangoAccount'
2023-05-03 18:41:18 -07:00
type Currencies = {
[key: string]: string
}
export const CURRENCY_SYMBOLS: Currencies = {
'wBTC (Portal)': '₿',
SOL: '◎',
}
2023-05-03 18:41:18 -07:00
const MARKET_LINK_CLASSES =
2023-09-11 21:17:59 -07:00
'grid grid-cols-3 sm:grid-cols-4 flex items-center w-full py-2 px-4 rounded-r-md focus:outline-none focus-visible:text-th-active md:hover:cursor-pointer md:hover:bg-th-bkg-3 md:hover:text-th-fgd-1'
2023-05-03 18:41:18 -07:00
const MARKET_LINK_DISABLED_CLASSES =
2023-07-24 05:14:45 -07:00
'flex w-full items-center justify-between py-2 px-4 md:hover:cursor-not-allowed'
export const DEFAULT_SORT_KEY: AllowedKeys = 'notionalQuoteVolume'
2022-11-30 07:46:20 -08:00
const MarketSelectDropdown = () => {
2023-09-24 17:17:19 -07:00
const { t } = useTranslation(['common', 'trade'])
2022-11-30 07:46:20 -08:00
const { selectedMarket } = useSelectedMarket()
const [spotOrPerp, setSpotOrPerp] = useState(
2023-07-21 11:47:53 -07:00
selectedMarket instanceof PerpMarket ? 'perp' : 'spot',
)
const [search, setSearch] = useState('')
const [isOpen, setIsOpen] = useState(false)
2023-04-03 19:48:31 -07:00
const { group } = useMangoGroup()
2023-06-18 22:39:18 -07:00
const [spotBaseFilter, setSpotBaseFilter] = useState('All')
2023-12-17 20:39:47 -08:00
const { perpMarketsWithData, serumMarketsWithData, isLoading } =
2023-07-24 05:14:45 -07:00
useListedMarketsWithMarketData()
const { isDesktop } = useViewport()
const focusRef = useRef<HTMLInputElement>(null)
2023-12-04 13:17:13 -08:00
const { query } = useRouter()
const { mangoAccount } = useMangoAccount()
2023-12-04 13:17:13 -08:00
// switch to spot tab on spot markets
useEffect(() => {
if (query?.name && !query.name.includes('PERP')) {
setSpotOrPerp('spot')
} else {
setSpotOrPerp('perp')
}
}, [query])
2022-11-30 07:46:20 -08:00
2023-09-08 15:01:08 -07:00
const unsortedPerpMarketsToShow = useMemo(() => {
2023-07-24 05:14:45 -07:00
if (!perpMarketsWithData.length) return []
return sortPerpMarkets(perpMarketsWithData, DEFAULT_SORT_KEY)
2023-09-08 15:01:08 -07:00
}, [perpMarketsWithData])
2023-01-19 20:17:43 -08:00
2023-10-04 18:06:56 -07:00
const spotQuoteTokens: string[] = useMemo(() => {
if (serumMarketsWithData.length && group) {
const quoteTokens: string[] = ['All']
2023-07-24 05:14:45 -07:00
serumMarketsWithData.map((m) => {
2023-10-04 18:06:56 -07:00
const quoteBank = group.getFirstBankByTokenIndex(m.quoteTokenIndex)
const quote = quoteBank.name
if (!quoteTokens.includes(quote)) {
quoteTokens.push(quote)
2023-06-18 22:39:18 -07:00
}
})
2023-10-04 18:06:56 -07:00
return quoteTokens.sort((a, b) => a.localeCompare(b))
2023-06-18 22:39:18 -07:00
}
return ['All']
2023-10-04 18:06:56 -07:00
}, [group, serumMarketsWithData])
2023-06-18 22:39:18 -07:00
2023-09-08 15:01:08 -07:00
const unsortedSerumMarketsToShow = useMemo(() => {
2023-10-04 18:06:56 -07:00
if (!serumMarketsWithData.length || !group) return []
2023-06-18 22:39:18 -07:00
if (spotBaseFilter !== 'All') {
2023-07-24 05:14:45 -07:00
const filteredMarkets = serumMarketsWithData.filter((m) => {
2023-10-04 18:06:56 -07:00
const quoteBank = group.getFirstBankByTokenIndex(m.quoteTokenIndex)
const quote = quoteBank.name
return quote === spotBaseFilter
2023-06-18 22:39:18 -07:00
})
return search
? startSearch(filteredMarkets, search)
: sortSpotMarkets(filteredMarkets, DEFAULT_SORT_KEY)
2023-06-18 22:39:18 -07:00
} else {
return search
? startSearch(serumMarketsWithData, search)
: sortSpotMarkets(serumMarketsWithData, DEFAULT_SORT_KEY)
2023-06-18 22:39:18 -07:00
}
2023-10-04 18:06:56 -07:00
}, [group, search, serumMarketsWithData, spotBaseFilter])
const handleUpdateSearch = (e: ChangeEvent<HTMLInputElement>) => {
setSearch(e.target.value)
}
2023-09-08 15:01:08 -07:00
const {
items: perpMarketsToShow,
requestSort: requestPerpSort,
sortConfig: perpSortConfig,
} = useSortableData(unsortedPerpMarketsToShow)
const {
items: serumMarketsToShow,
requestSort: requestSerumSort,
sortConfig: serumSortConfig,
} = useSortableData(unsortedSerumMarketsToShow)
const filteredSerumMarkets = serumMarketsToShow.filter((x) => {
const baseBank = group?.getFirstBankByTokenIndex(x.baseTokenIndex)
if (baseBank?.reduceOnly === TOKEN_REDUCE_ONLY_OPTIONS.ENABLED) {
if (!mangoAccount) {
return false
}
const borrowedAmount = mangoAccount
? new Decimal(mangoAccount.getTokenBorrowsUi(baseBank))
.toDecimalPlaces(baseBank.mintDecimals, Decimal.ROUND_UP)
.toNumber()
: 0
const balance = mangoAccount
? mangoAccount.getTokenBalanceUi(baseBank)
: 0
return isBankVisibleForUser(baseBank, borrowedAmount, balance)
} else {
return true
}
})
useEffect(() => {
if (focusRef?.current && spotOrPerp === 'spot' && isDesktop && isOpen) {
focusRef.current.focus()
}
}, [focusRef, isDesktop, isOpen, spotOrPerp])
2023-07-24 05:14:45 -07:00
2022-11-30 07:46:20 -08:00
return (
<Popover>
2023-02-22 11:04:18 -08:00
{({ open, close }) => (
2022-11-30 07:46:20 -08:00
<div
className="relative flex flex-col overflow-visible md:-ml-2"
2022-11-30 07:46:20 -08:00
id="trade-step-one"
>
2023-06-29 21:00:28 -07:00
<Popover.Button
className="-ml-4 flex h-12 items-center justify-between px-4 focus-visible:bg-th-bkg-3 disabled:cursor-not-allowed disabled:opacity-60 md:hover:bg-th-bkg-2 disabled:md:hover:bg-th-bkg-1"
disabled={!group}
onClick={() => setIsOpen(!isOpen)}
2023-06-29 21:00:28 -07:00
>
2022-12-06 21:25:37 -08:00
<div className="flex items-center">
2023-06-29 21:00:28 -07:00
{selectedMarket ? (
<MarketLogos market={selectedMarket} />
) : (
<Loading className="mr-2 h-5 w-5 shrink-0" />
2023-06-29 21:00:28 -07:00
)}
2023-10-01 20:42:34 -07:00
<div className="whitespace-nowrap text-left text-xl font-bold text-th-fgd-1 md:text-base">
2023-06-29 21:00:28 -07:00
{selectedMarket?.name || (
<span className="text-th-fgd-3">{t('loading')}</span>
)}
2023-10-01 20:42:34 -07:00
{selectedMarket?.reduceOnly ? (
<div className="flex items-center">
<ExclamationTriangleIcon className="mr-1 mt-0.5 h-3 w-3 text-th-warning" />
<p className="text-xxs leading-none text-th-warning">
{t('trade:reduce-only')}
</p>
</div>
) : null}
2022-12-06 21:25:37 -08:00
</div>
2022-11-30 07:46:20 -08:00
</div>
<ChevronDownIcon
className={`${
open ? 'rotate-180' : 'rotate-0'
} ml-2 mt-0.5 h-6 w-6 shrink-0 text-th-fgd-2`}
2022-11-30 07:46:20 -08:00
/>
</Popover.Button>
2023-09-11 21:17:59 -07:00
<Popover.Panel className="absolute -left-4 top-12 z-40 w-screen border-y border-th-bkg-3 bg-th-bkg-2 md:w-[580px] md:border-r">
2023-06-18 22:39:18 -07:00
<div className="border-b border-th-bkg-3">
<TabButtons
activeValue={spotOrPerp}
onChange={(v) => setSpotOrPerp(v)}
values={[
['perp', 0],
['spot', 0],
]}
fillWidth
/>
</div>
2023-10-04 18:06:56 -07:00
<div className="thin-scroll h-[calc(100vh-188px)] overflow-auto py-3 md:max-h-[calc(100vh-215px)]">
2023-07-24 05:14:45 -07:00
{spotOrPerp === 'perp' && perpMarketsToShow.length ? (
<>
2023-09-11 21:17:59 -07:00
<div className="mb-2 grid grid-cols-3 border-b border-th-bkg-3 pb-1 pl-4 pr-14 text-xxs sm:grid-cols-4">
<div className="col-span-1 flex-1">
2023-09-08 15:01:08 -07:00
<SortableColumnHeader
sortKey="name"
sort={() => requestPerpSort('name')}
sortConfig={perpSortConfig}
title={t('market')}
/>
</div>
<p className="col-span-1 flex justify-end">
<SortableColumnHeader
sortKey="marketData.last_price"
sort={() => requestPerpSort('marketData.last_price')}
sortConfig={perpSortConfig}
title={t('price')}
/>
</p>
<p className="col-span-1 flex justify-end">
<SortableColumnHeader
sortKey="rollingChange"
sort={() => requestPerpSort('rollingChange')}
sortConfig={perpSortConfig}
title={t('rolling-change')}
/>
2023-07-24 05:14:45 -07:00
</p>
2023-09-11 21:17:59 -07:00
<p className="col-span-1 hidden sm:flex sm:justify-end">
2023-09-08 15:01:08 -07:00
<SortableColumnHeader
sortKey="marketData.quote_volume_24h"
sort={() =>
requestPerpSort('marketData.quote_volume_24h')
}
sortConfig={perpSortConfig}
title={t('daily-volume')}
/>
2023-07-24 05:14:45 -07:00
</p>
</div>
{perpMarketsToShow.map((m) => {
2023-06-18 22:39:18 -07:00
const isComingSoon = m.oracleLastUpdatedSlot == 0
2023-07-24 05:14:45 -07:00
const volumeData = m?.marketData?.quote_volume_24h
const volume = volumeData ? volumeData : 0
2023-09-11 21:17:59 -07:00
const leverage = 1 / (m.maintBaseLiabWeight.toNumber() - 1)
2023-06-18 22:39:18 -07:00
return (
<div className="flex w-full items-center" key={m.name}>
2023-06-18 22:39:18 -07:00
{!isComingSoon ? (
<>
<Link
className={MARKET_LINK_CLASSES}
href={{
pathname: '/trade',
query: { name: m.name },
}}
2023-06-20 16:40:51 -07:00
onClick={() => {
close()
setSearch('')
2023-06-20 16:40:51 -07:00
}}
2023-06-18 22:39:18 -07:00
shallow={true}
>
2023-07-24 05:14:45 -07:00
<div className="col-span-1 flex items-center">
2023-09-11 21:17:59 -07:00
<div className="hidden sm:block">
<MarketLogos market={m} size="small" />
</div>
<span className="mr-1.5 whitespace-nowrap text-xs text-th-fgd-2">
2023-07-24 05:14:45 -07:00
{m.name}
</span>
2023-09-11 21:17:59 -07:00
{leverage ? (
<LeverageBadge leverage={leverage} />
) : null}
2023-06-18 22:39:18 -07:00
</div>
2023-07-24 05:14:45 -07:00
<div className="col-span-1 flex justify-end">
<span className="font-mono text-xs text-th-fgd-2">
2023-06-18 22:39:18 -07:00
{formatCurrencyValue(
m.uiPrice,
2023-07-21 11:47:53 -07:00
getDecimalCount(m.tickSize),
2023-06-18 22:39:18 -07:00
)}
</span>
2023-07-24 05:14:45 -07:00
</div>
<div className="col-span-1 flex justify-end">
2023-07-13 22:47:05 -07:00
<MarketChange market={m} size="small" />
2023-06-18 22:39:18 -07:00
</div>
2023-09-11 21:17:59 -07:00
<div className="col-span-1 hidden sm:flex sm:justify-end">
2023-12-17 20:39:47 -08:00
{isLoading ? (
2023-07-24 05:14:45 -07:00
<SheenLoader className="mt-0.5">
<div className="h-3.5 w-12 bg-th-bkg-2" />
</SheenLoader>
) : (
<span>
{volume ? (
<span className="font-mono text-xs text-th-fgd-2">
${numberCompacter.format(volume)}
</span>
) : (
<span className="font-mono text-xs text-th-fgd-2">
$0
</span>
)}
</span>
)}
</div>
2023-06-18 22:39:18 -07:00
</Link>
2023-07-24 05:14:45 -07:00
<div className="px-3">
<FavoriteMarketButton market={m} />
</div>
2023-06-18 22:39:18 -07:00
</>
) : (
<span className={MARKET_LINK_DISABLED_CLASSES}>
<div className="flex items-center">
2023-07-24 05:14:45 -07:00
<MarketLogos market={m} size="small" />
<span className="mr-2 text-xs">{m.name}</span>
2023-06-18 22:39:18 -07:00
<SoonBadge />
</div>
</span>
)}
</div>
)
2023-07-24 05:14:45 -07:00
})}
</>
) : null}
2023-09-24 17:17:19 -07:00
{spotOrPerp === 'spot' ? (
2023-06-18 22:39:18 -07:00
<>
<div className="mb-3 flex items-center justify-between px-4">
<div className="relative w-1/2">
<Input
className="h-8 pl-8"
type="text"
value={search}
onChange={handleUpdateSearch}
ref={focusRef}
/>
<MagnifyingGlassIcon className="absolute left-2 top-2 h-4 w-4" />
</div>
2023-07-24 05:14:45 -07:00
<div>
2023-10-04 18:06:56 -07:00
{spotQuoteTokens.map((tab) => (
2023-07-24 05:14:45 -07:00
<button
className={`rounded-md px-2.5 py-1.5 text-sm font-medium focus-visible:bg-th-bkg-3 focus-visible:text-th-fgd-1 ${
2023-07-24 05:14:45 -07:00
spotBaseFilter === tab
? 'bg-th-bkg-3 text-th-active md:hover:text-th-active'
: 'text-th-fgd-3 md:hover:text-th-fgd-2'
}`}
onClick={() => setSpotBaseFilter(tab)}
key={tab}
>
{t(tab)}
</button>
))}
</div>
</div>
2023-09-11 21:17:59 -07:00
<div className="mb-2 grid grid-cols-3 border-b border-th-bkg-3 pb-1 pl-4 pr-14 text-xxs sm:grid-cols-4">
<p className="col-span-1 flex">
2023-09-08 15:01:08 -07:00
<SortableColumnHeader
sortKey="name"
sort={() => requestSerumSort('name')}
sortConfig={serumSortConfig}
title={t('market')}
/>
</p>
<p className="col-span-1 flex justify-end">
<SortableColumnHeader
sortKey="marketData.last_price"
sort={() => requestSerumSort('marketData.last_price')}
sortConfig={serumSortConfig}
title={t('price')}
/>
2023-07-24 05:14:45 -07:00
</p>
2023-09-08 15:01:08 -07:00
<p className="col-span-1 flex justify-end">
<SortableColumnHeader
sortKey="rollingChange"
sort={() => requestSerumSort('rollingChange')}
sortConfig={serumSortConfig}
title={t('rolling-change')}
/>
</p>
2023-09-11 21:17:59 -07:00
<p className="col-span-1 hidden sm:flex sm:justify-end">
2023-09-08 15:01:08 -07:00
<SortableColumnHeader
sortKey="marketData.notionalQuoteVolume"
2023-09-08 15:01:08 -07:00
sort={() =>
requestSerumSort('marketData.notionalQuoteVolume')
2023-09-08 15:01:08 -07:00
}
sortConfig={serumSortConfig}
title={t('daily-volume')}
/>
2023-07-24 05:14:45 -07:00
</p>
2023-06-18 22:39:18 -07:00
</div>
{filteredSerumMarkets.length ? (
filteredSerumMarkets.map((m) => {
2023-09-24 17:17:19 -07:00
const baseBank = group?.getFirstBankByTokenIndex(
m.baseTokenIndex,
)
const quoteBank = group?.getFirstBankByTokenIndex(
m.quoteTokenIndex,
)
const market = group?.getSerum3ExternalMarket(
m.serumMarketExternal,
)
let leverage
if (group) {
leverage = m.maxBidLeverage(group)
}
let price
if (baseBank && market && quoteBank) {
price = floorToDecimal(
baseBank.uiPrice / quoteBank.uiPrice,
getDecimalCount(market.tickSize),
).toNumber()
}
2023-07-24 05:14:45 -07:00
2023-09-24 17:17:19 -07:00
const volumeData = m?.marketData?.quote_volume_24h
2023-07-24 05:14:45 -07:00
2023-09-24 17:17:19 -07:00
const volume = volumeData ? volumeData : 0
2023-07-24 05:14:45 -07:00
2023-09-24 17:17:19 -07:00
return (
<div className="flex w-full items-center" key={m.name}>
<Link
className={MARKET_LINK_CLASSES}
href={{
pathname: '/trade',
query: { name: m.name },
}}
onClick={() => {
close()
setSearch('')
}}
shallow={true}
>
<div className="col-span-1 flex items-center">
<div className="hidden sm:block">
<MarketLogos market={m} size="small" />
</div>
<span className="mr-1.5 text-xs text-th-fgd-2">
{m.name}
2023-07-24 05:14:45 -07:00
</span>
2023-09-24 17:17:19 -07:00
{leverage ? (
<LeverageBadge leverage={leverage} />
) : null}
</div>
<div className="col-span-1 flex justify-end">
{price && market?.tickSize ? (
<span className="font-mono text-xs text-th-fgd-2">
{quoteBank?.name === 'USDC' ? '$' : ''}
2023-11-16 04:35:44 -08:00
{countLeadingZeros(price) <= 4
2023-09-24 17:17:19 -07:00
? formatNumericValue(
price,
getDecimalCount(market.tickSize),
)
: price.toExponential(3)}
{quoteBank?.name &&
quoteBank.name !== 'USDC' ? (
2023-09-24 17:17:19 -07:00
<span className="font-body text-th-fgd-3">
{' '}
{CURRENCY_SYMBOLS[quoteBank.name] ||
quoteBank.name}
2023-09-24 17:17:19 -07:00
</span>
) : null}
</span>
) : null}
</div>
<div className="col-span-1 flex justify-end">
<MarketChange market={m} size="small" />
</div>
<div className="col-span-1 hidden sm:flex sm:justify-end">
2023-12-17 20:39:47 -08:00
{isLoading ? (
2023-09-24 17:17:19 -07:00
<SheenLoader className="mt-0.5">
<div className="h-3.5 w-12 bg-th-bkg-2" />
</SheenLoader>
) : (
<span className="font-mono text-xs text-th-fgd-2">
{quoteBank?.name === 'USDC' ? '$' : ''}
{volume ? numberCompacter.format(volume) : 0}
{quoteBank?.name &&
quoteBank.name !== 'USDC' ? (
2023-09-24 17:17:19 -07:00
<span className="font-body text-th-fgd-3">
{' '}
{CURRENCY_SYMBOLS[quoteBank.name] ||
quoteBank.name}
2023-09-24 17:17:19 -07:00
</span>
) : null}
</span>
)}
</div>
</Link>
<div className="px-3">
<FavoriteMarketButton market={m} />
2023-07-24 05:14:45 -07:00
</div>
2023-06-18 22:39:18 -07:00
</div>
2023-09-24 17:17:19 -07:00
)
})
) : (
<p className="mb-2 mt-4 text-center">
{t('trade:no-markets-found')}
</p>
)}
2023-06-18 22:39:18 -07:00
</>
) : null}
</div>
2022-11-30 07:46:20 -08:00
</Popover.Panel>
</div>
)}
</Popover>
)
}
export default MarketSelectDropdown
2023-09-11 21:17:59 -07:00
const LeverageBadge = ({ leverage }: { leverage: number }) => {
return (
<div className="rounded border border-th-fgd-4 px-1 py-0.5 text-xxs leading-none text-th-fgd-4">
<span>{leverage < 1 ? leverage.toFixed(1) : leverage.toFixed()}x</span>
</div>
)
}