mango-ui-v3/components/MarketDetails.tsx

278 lines
9.2 KiB
TypeScript
Raw Normal View History

import React, { useCallback, useMemo, useState } from 'react'
2021-04-19 06:45:59 -07:00
import useMangoStore from '../stores/useMangoStore'
import usePrevious from '../hooks/usePrevious'
import useInterval from '../hooks/useInterval'
import ChartApi from '../utils/chartDataConnector'
2021-04-19 06:45:59 -07:00
import UiLock from './UiLock'
import ManualRefresh from './ManualRefresh'
2021-07-05 08:03:57 -07:00
import useOraclePrice from '../hooks/useOraclePrice'
import DayHighLow from './DayHighLow'
2021-08-13 11:54:09 -07:00
import { useEffect } from 'react'
import { getDecimalCount, usdFormatter } from '../utils'
import { PerpMarket } from '@blockworks-foundation/mango-client'
import BN from 'bn.js'
import { useViewport } from '../hooks/useViewport'
import { breakpoints } from './TradePageGrid'
import { useTranslation } from 'next-i18next'
2021-08-13 11:54:09 -07:00
2021-08-31 13:31:35 -07:00
const SECONDS = 1000
function calculateFundingRate(perpStats, perpMarket) {
2021-08-13 11:54:09 -07:00
const oldestStat = perpStats[perpStats.length - 1]
const latestStat = perpStats[0]
if (!latestStat || !(perpMarket instanceof PerpMarket)) return 0.0
2021-08-13 11:54:09 -07:00
// Averaging long and short funding excludes socialized loss
const startFunding =
(parseFloat(oldestStat.longFunding) + parseFloat(oldestStat.shortFunding)) /
2
const endFunding =
(parseFloat(latestStat.longFunding) + parseFloat(latestStat.shortFunding)) /
2
const fundingDifference = endFunding - startFunding
2021-08-13 11:54:09 -07:00
const fundingInQuoteDecimals =
fundingDifference / Math.pow(10, perpMarket.quoteDecimals)
const avgPrice =
(parseFloat(latestStat.baseOraclePrice) +
parseFloat(oldestStat.baseOraclePrice)) /
2
const basePriceInBaseLots = avgPrice * perpMarket.baseLotsToNumber(new BN(1))
2021-08-13 11:54:09 -07:00
return (fundingInQuoteDecimals / basePriceInBaseLots) * 100
}
function parseOpenInterest(perpMarket: PerpMarket) {
2022-01-12 14:46:41 -08:00
if (!perpMarket || !(perpMarket instanceof PerpMarket)) return 0
2021-08-13 12:06:17 -07:00
return perpMarket.baseLotsToNumber(perpMarket.openInterest) / 2
2021-08-13 11:54:09 -07:00
}
const MarketDetails = () => {
const { t } = useTranslation('common')
2021-07-05 08:03:57 -07:00
const oraclePrice = useOraclePrice()
2021-08-20 05:10:59 -07:00
const groupConfig = useMangoStore((s) => s.selectedMangoGroup.config)
const marketConfig = useMangoStore((s) => s.selectedMarket.config)
2021-08-13 11:54:09 -07:00
const selectedMarket = useMangoStore((s) => s.selectedMarket.current)
2021-06-23 08:32:33 -07:00
const baseSymbol = marketConfig.baseSymbol
const selectedMarketName = marketConfig.name
2021-08-20 05:10:59 -07:00
const isPerpMarket = marketConfig.kind === 'perp'
const previousMarketName: string = usePrevious(selectedMarketName)
const connected = useMangoStore((s) => s.wallet.connected)
const { width } = useViewport()
const isMobile = width ? width < breakpoints.sm : false
const [ohlcv, setOhlcv] = useState(null)
const [, setLoading] = useState(false)
2021-08-13 11:54:09 -07:00
const [perpStats, setPerpStats] = useState([])
2021-09-02 21:33:10 -07:00
const [perpVolume, setPerpVolume] = useState(0)
const change = ohlcv ? ((ohlcv.c[0] - ohlcv.o[0]) / ohlcv.o[0]) * 100 : ''
2021-08-13 14:30:39 -07:00
const fetchPerpStats = useCallback(async () => {
2021-08-25 10:40:32 -07:00
const urlParams = new URLSearchParams({ mangoGroup: groupConfig.name })
urlParams.append('market', selectedMarketName)
2021-08-13 11:54:09 -07:00
const perpStats = await fetch(
2021-08-25 10:40:32 -07:00
`https://mango-stats-v3.herokuapp.com/perp/funding_rate?` + urlParams
2021-08-13 11:54:09 -07:00
)
const parsedPerpStats = await perpStats.json()
setPerpStats(parsedPerpStats)
2021-09-02 21:33:10 -07:00
const perpVolume = await fetch(
`https://event-history-api.herokuapp.com/stats/perps/${marketConfig.publicKey.toString()}`
)
const parsedPerpVolume = await perpVolume.json()
setPerpVolume(parsedPerpVolume?.data?.volume)
}, [selectedMarketName, marketConfig, groupConfig.name])
2021-08-13 11:54:09 -07:00
2021-08-31 13:31:35 -07:00
useInterval(() => {
if (isPerpMarket) {
fetchPerpStats()
}
}, 120 * SECONDS)
2021-08-13 14:30:39 -07:00
useEffect(() => {
2021-08-20 05:10:59 -07:00
if (isPerpMarket) {
2021-08-13 11:54:09 -07:00
fetchPerpStats()
}
}, [isPerpMarket, fetchPerpStats])
2021-08-13 11:54:09 -07:00
const fetchOhlcv = useCallback(async () => {
if (!selectedMarketName) return
// calculate from and to date (0:00UTC to 23:59:59UTC)
const date = new Date()
2021-05-20 04:21:36 -07:00
const utcFrom = new Date(
2021-06-05 11:40:07 -07:00
Date.UTC(
date.getFullYear(),
date.getUTCMonth(),
date.getUTCDate(),
0,
0,
0
)
)
2021-05-20 04:21:36 -07:00
const utcTo = new Date(
2021-06-05 11:40:07 -07:00
Date.UTC(
date.getFullYear(),
date.getUTCMonth(),
date.getUTCDate(),
23,
59,
59
)
)
2021-06-05 11:40:07 -07:00
const from = utcFrom.getTime() / 1000
const to = utcTo.getTime() / 1000
const ohlcv = await ChartApi.getOhlcv(selectedMarketName, '1D', from, to)
if (ohlcv) {
setOhlcv(ohlcv)
setLoading(false)
}
}, [selectedMarketName])
2021-11-06 06:54:24 -07:00
// TODO: don't spam db
// useInterval(async () => {
// fetchOhlcv()
// }, 5000)
useMemo(() => {
if (previousMarketName !== selectedMarketName) {
setLoading(true)
2021-11-06 06:54:24 -07:00
fetchOhlcv()
}
}, [selectedMarketName])
2021-04-19 06:45:59 -07:00
2021-10-18 10:15:37 -07:00
const funding1h = calculateFundingRate(perpStats, selectedMarket)
const [funding1hStr, fundingAprStr] = funding1h
? [funding1h.toFixed(4), (funding1h * 24 * 365).toFixed(2)]
: ['-', '-']
2021-04-19 06:45:59 -07:00
return (
<div
2022-01-31 07:52:28 -08:00
className={`flex flex-col relative md:pb-2 md:pt-3 md:px-3 lg:flex-row lg:items-center lg:justify-between`}
>
2022-01-31 07:52:28 -08:00
<div className="flex flex-col lg:flex-row lg:items-center">
<div className="hidden md:block md:pb-4 md:pr-6 lg:pb-0">
<div className="flex items-center">
<img
alt=""
width="24"
height="24"
src={`/assets/icons/${baseSymbol.toLowerCase()}.svg`}
className={`mr-2.5`}
/>
2021-06-23 06:16:37 -07:00
<div className="font-semibold pr-0.5 text-xl">{baseSymbol}</div>
<span className="text-th-fgd-4 text-xl">
2021-08-20 05:10:59 -07:00
{isPerpMarket ? '-' : '/'}
2021-06-23 06:16:37 -07:00
</span>
<div className="font-semibold pl-0.5 text-xl">
2021-08-20 05:10:59 -07:00
{isPerpMarket ? 'PERP' : groupConfig.quoteSymbol}
</div>
</div>
</div>
2022-01-31 07:52:28 -08:00
<div className="grid grid-flow-row grid-cols-1 md:grid-cols-3 gap-3 lg:grid-cols-none lg:grid-flow-col lg:grid-rows-1 lg:gap-6">
<div className="flex items-center justify-between md:block">
<div className="text-th-fgd-3 tiny-text pb-0.5">
{t('oracle-price')}
</div>
<div className="text-th-fgd-1 md:text-xs">
{oraclePrice && selectedMarket
? oraclePrice.toFixed(getDecimalCount(selectedMarket.tickSize))
: '--'}
</div>
</div>
<div className="flex items-center justify-between md:block">
<div className="text-th-fgd-3 tiny-text pb-0.5">
{t('daily-change')}
</div>
2021-08-28 19:02:22 -07:00
{change || change === 0 ? (
2021-08-13 14:45:22 -07:00
<div
className={`md:text-xs ${
change > 0
2021-08-13 14:45:22 -07:00
? `text-th-green`
: change < 0
2021-08-13 14:45:22 -07:00
? `text-th-red`
: `text-th-fgd-1`
2021-08-13 14:45:22 -07:00
}`}
>
2021-08-28 19:02:22 -07:00
{change.toFixed(2) + '%'}
2021-08-13 14:45:22 -07:00
</div>
) : (
<MarketDataLoader />
)}
</div>
2021-09-02 21:33:10 -07:00
{isPerpMarket ? (
<div className="flex items-center justify-between md:block">
<div className="text-th-fgd-3 tiny-text pb-0.5">
{t('daily-volume')}
</div>
<div className="text-th-fgd-1 md:text-xs">
2021-09-02 21:33:10 -07:00
{perpVolume ? (
usdFormatter(perpVolume, 0)
) : (
2021-09-02 21:33:10 -07:00
<MarketDataLoader />
)}
</div>
</div>
2021-09-02 21:33:10 -07:00
) : null}
{isPerpMarket && selectedMarket instanceof PerpMarket ? (
<>
<div className="flex items-center justify-between md:block">
2021-08-16 06:31:25 -07:00
<div className="text-th-fgd-3 tiny-text pb-0.5">
{t('average-funding')}
2021-08-13 12:10:50 -07:00
</div>
<div className="text-th-fgd-1 md:text-xs">
2021-08-25 10:43:17 -07:00
{selectedMarket ? (
2021-10-18 10:15:37 -07:00
`${funding1hStr}% (${fundingAprStr}% APR)`
2021-08-21 06:02:51 -07:00
) : (
<MarketDataLoader />
)}
</div>
</div>
<div className="flex items-center justify-between md:block">
2021-08-16 06:31:25 -07:00
<div className="text-th-fgd-3 tiny-text pb-0.5">
{t('open-interest')}
2021-08-16 06:31:25 -07:00
</div>
<div className="text-th-fgd-1 md:text-xs">
{selectedMarket ? (
`${parseOpenInterest(
selectedMarket as PerpMarket
)} ${baseSymbol}`
2021-08-21 06:02:51 -07:00
) : (
<MarketDataLoader />
)}
2021-08-13 11:54:09 -07:00
</div>
</div>
</>
) : null}
2021-08-25 11:30:16 -07:00
<DayHighLow
high={ohlcv?.h[0]}
low={ohlcv?.l[0]}
latest={oraclePrice?.toNumber()}
/>
</div>
2021-04-19 06:45:59 -07:00
</div>
2021-08-31 05:46:47 -07:00
<div className="absolute right-4 bottom-0 sm:bottom-auto lg:right-6 flex items-center justify-end">
2021-10-29 05:05:55 -07:00
{!isMobile ? (
2021-11-04 05:25:11 -07:00
<div id="layout-tip">
2021-10-29 05:05:55 -07:00
<UiLock />
</div>
) : null}
2021-11-04 05:25:11 -07:00
<div className="ml-2" id="data-refresh-tip">
2021-12-17 01:51:54 -08:00
{!isMobile && connected ? <ManualRefresh /> : null}
2021-10-29 05:05:55 -07:00
</div>
2021-04-19 06:45:59 -07:00
</div>
</div>
)
}
export default MarketDetails
2021-04-19 06:45:59 -07:00
2021-08-21 06:02:51 -07:00
export const MarketDataLoader = () => (
<div className="animate-pulse bg-th-bkg-3 h-3.5 mt-0.5 w-10 rounded-sm" />
2021-04-19 06:45:59 -07:00
)