add volume chart

This commit is contained in:
saml33 2023-07-04 13:24:13 +10:00
parent 24d927e156
commit bbcb50f8fb
4 changed files with 294 additions and 8 deletions

View File

@ -49,7 +49,8 @@ import {
TotalAccountFundingItem, TotalAccountFundingItem,
} from 'types' } from 'types'
import { useQuery } from '@tanstack/react-query' import { useQuery } from '@tanstack/react-query'
import FundingDetails from './FundingDetails' import FundingChart from './FundingChart'
import VolumeChart from './VolumeChart'
const TABS = ['account-value', 'account:assets-liabilities'] const TABS = ['account-value', 'account:assets-liabilities']
@ -172,6 +173,7 @@ export type ChartToShow =
| 'cumulative-interest-value' | 'cumulative-interest-value'
| 'pnl' | 'pnl'
| 'hourly-funding' | 'hourly-funding'
| 'hourly-volume'
const AccountPage = () => { const AccountPage = () => {
const { t } = useTranslation(['common', 'account']) const { t } = useTranslation(['common', 'account'])
@ -396,7 +398,11 @@ const AccountPage = () => {
}, [mangoAccount, group]) }, [mangoAccount, group])
const handleChartToShow = ( const handleChartToShow = (
chartName: 'pnl' | 'cumulative-interest-value' | 'hourly-funding' chartName:
| 'pnl'
| 'cumulative-interest-value'
| 'hourly-funding'
| 'hourly-volume'
) => { ) => {
if ( if (
(chartName === 'cumulative-interest-value' && interestTotalValue > 1) || (chartName === 'cumulative-interest-value' && interestTotalValue > 1) ||
@ -410,6 +416,9 @@ const AccountPage = () => {
if (chartName === 'hourly-funding') { if (chartName === 'hourly-funding') {
setChartToShow(chartName) setChartToShow(chartName)
} }
if (chartName === 'hourly-volume') {
setChartToShow(chartName)
}
} }
const latestAccountData = useMemo(() => { const latestAccountData = useMemo(() => {
@ -737,9 +746,28 @@ const AccountPage = () => {
</div> </div>
<div className="col-span-6 border-t border-th-bkg-3 py-3 pl-6 pr-4 md:col-span-3 md:border-l lg:col-span-2 lg:border-l-0 xl:col-span-1 xl:border-l xl:border-t-0"> <div className="col-span-6 border-t border-th-bkg-3 py-3 pl-6 pr-4 md:col-span-3 md:border-l lg:col-span-2 lg:border-l-0 xl:col-span-1 xl:border-l xl:border-t-0">
<div id="account-step-six"> <div id="account-step-six">
<p className="text-sm text-th-fgd-3 xl:text-base"> <div className="flex w-full items-center justify-between">
{t('account:lifetime-volume')} <p className="text-sm text-th-fgd-3 xl:text-base">
</p> {t('account:lifetime-volume')}
</p>
{mangoAccountAddress &&
hourlyVolumeData &&
hourlyVolumeData.length ? (
<Tooltip
className="hidden md:block"
content="Funding Chart"
delay={100}
>
<IconButton
className="text-th-fgd-3"
hideBg
onClick={() => handleChartToShow('hourly-volume')}
>
<ChartBarIcon className="h-5 w-5" />
</IconButton>
</Tooltip>
) : null}
</div>
{loadingTotalVolume && mangoAccountAddress ? ( {loadingTotalVolume && mangoAccountAddress ? (
<SheenLoader className="mt-1"> <SheenLoader className="mt-1">
<div className="h-7 w-16 bg-th-bkg-2" /> <div className="h-7 w-16 bg-th-bkg-2" />
@ -885,7 +913,13 @@ const AccountPage = () => {
yKey="pnl" yKey="pnl"
/> />
) : chartToShow === 'hourly-funding' ? ( ) : chartToShow === 'hourly-funding' ? (
<FundingDetails hideChart={handleHideChart} /> <FundingChart hideChart={handleHideChart} />
) : chartToShow === 'hourly-volume' ? (
<VolumeChart
chartData={hourlyVolumeData}
hideChart={handleHideChart}
loading={loadingHourlyVolume}
/>
) : ( ) : (
<AccountChart <AccountChart
chartToShow="cumulative-interest-value" chartToShow="cumulative-interest-value"

View File

@ -61,7 +61,7 @@ const fetchHourlyFunding = async (mangoAccountPk: string) => {
} }
} }
const FundingDetails = ({ hideChart }: { hideChart: () => void }) => { const FundingChart = ({ hideChart }: { hideChart: () => void }) => {
const { t } = useTranslation('common') const { t } = useTranslation('common')
const { mangoAccountAddress } = useMangoAccount() const { mangoAccountAddress } = useMangoAccount()
const [daysToShow, setDaysToShow] = useState('30') const [daysToShow, setDaysToShow] = useState('30')
@ -334,4 +334,4 @@ const FundingDetails = ({ hideChart }: { hideChart: () => void }) => {
) )
} }
export default FundingDetails export default FundingChart

View File

@ -0,0 +1,251 @@
/* eslint-disable @typescript-eslint/no-explicit-any */
import dayjs from 'dayjs'
import { useMemo, useState } from 'react'
import { FormattedHourlyAccountVolumeData, HourlyFundingChartData } from 'types'
import { formatCurrencyValue } from 'utils/numbers'
import { TooltipProps } from 'recharts/types/component/Tooltip'
import {
Cell,
XAxis,
YAxis,
Tooltip,
BarChart,
Bar,
ReferenceLine,
ResponsiveContainer,
} from 'recharts'
import { useTheme } from 'next-themes'
import { COLORS } from 'styles/colors'
import { formatDateAxis } from '@components/shared/DetailedAreaOrBarChart'
import { formatYAxis } from 'utils/formatting'
import ChartRangeButtons from '@components/shared/ChartRangeButtons'
import { useTranslation } from 'next-i18next'
import { IconButton } from '@components/shared/Button'
import { ArrowLeftIcon } from '@heroicons/react/20/solid'
import { FadeInFadeOut } from '@components/shared/Transitions'
import ContentBox from '@components/shared/ContentBox'
import SheenLoader from '@components/shared/SheenLoader'
const VolumeChart = ({
chartData,
hideChart,
loading,
}: {
chartData: FormattedHourlyAccountVolumeData[] | undefined
hideChart: () => void
loading: boolean
}) => {
const { t } = useTranslation(['common', 'stats'])
const [daysToShow, setDaysToShow] = useState('30')
const { theme } = useTheme()
const CustomTooltip = ({
active,
payload,
label,
}: TooltipProps<number, string>) => {
if (active && payload && payload.length) {
const load = payload[0].payload
return (
<div className="rounded-md bg-th-bkg-2 p-4">
<h3 className="mb-3 text-sm">
{daysToShow === '30'
? dayjs(label).format('DD MMM YY')
: dayjs(label).format('DD MMM YY, h:mma')}
</h3>
<div className="space-y-1">
{Object.keys(load.markets)
.filter((market) => load.markets[market] >= 0.01)
.sort((a, b) => a.localeCompare(b))
.map((mkt) => (
<div className="flex items-center justify-between" key={mkt}>
<p>{mkt}</p>
<p className="pl-4 font-mono text-th-fgd-2">
{formatCurrencyValue(load.markets[mkt])}
</p>
</div>
))}
</div>
<div className="mt-3 flex justify-between border-t border-th-bkg-4 pt-3">
<p>{t('total')}</p>
<p className="pl-4 font-mono text-th-fgd-2">
{formatCurrencyValue(load['total_volume_usd'])}
</p>
</div>
</div>
)
}
return null
}
const chunkDataByDay = (data: FormattedHourlyAccountVolumeData[]) => {
const chunkedData: FormattedHourlyAccountVolumeData[] = []
// Iterate over the data array
for (const entry of data) {
const { time, total_volume_usd, markets } = entry
// Extract the date portion from the timestamp
const date = time.substr(0, 10)
// Find the existing chunk for the date or create a new one
let chunk = chunkedData.find((chunk) => chunk.time === date)
if (!chunk) {
chunk = {
time: date,
total_volume_usd: 0,
markets: {},
}
chunkedData.push(chunk)
}
// Update the total volume for the day
chunk.total_volume_usd += total_volume_usd
// Update the market values for the day
for (const market in markets) {
if (Object.prototype.hasOwnProperty.call(markets, market)) {
const value = markets[market]
// Initialize the market value if it doesn't exist
if (!chunk.markets[market]) {
chunk.markets[market] = 0
}
// Add the value to the market value for the day
chunk.markets[market] += value
}
}
}
return chunkedData
}
const filteredData: HourlyFundingChartData[] = useMemo(() => {
if (!chartData || !chartData.length) return []
const start = Number(daysToShow) * 86400000
const filtered = chartData.filter((d: any) => {
const date = new Date()
if (daysToShow === '30') {
date.setHours(0, 0, 0, 0)
} else {
date.setMinutes(0, 0, 0)
}
const dataTime = new Date(d.time).getTime()
const now = date.getTime()
const limit = now - start
return dataTime >= limit
})
if (daysToShow === '30') {
return chunkDataByDay(filtered)
}
return filtered
}, [chartData, daysToShow])
return (
<FadeInFadeOut show={true}>
<ContentBox className="px-6 pt-4" hideBorder hidePadding>
{loading ? (
<SheenLoader className="flex flex-1">
<div
className={`h-[calc(100vh-116px)] w-full rounded-lg bg-th-bkg-2`}
/>
</SheenLoader>
) : filteredData.length ? (
<div>
<div className="flex items-center justify-between">
<div className="flex items-center space-x-4 md:space-x-6">
<IconButton onClick={hideChart} size="medium">
<ArrowLeftIcon className="h-5 w-5" />
</IconButton>
<h2 className="text-lg">{t('stats:volume')}</h2>
</div>
<ChartRangeButtons
activeValue={daysToShow}
names={['24H', '7D', '30D']}
values={['1', '7', '30']}
onChange={(v) => setDaysToShow(v)}
/>
</div>
<div className="-mx-6 mt-6 h-[calc(100vh-170px)]">
<ResponsiveContainer width="100%" height="100%">
<BarChart data={filteredData}>
<Tooltip
cursor={{
fill: 'var(--bkg-2)',
opacity: 0.5,
}}
content={<CustomTooltip />}
/>
<defs>
<linearGradient
id="greenGradientBar"
x1="0"
y1="0"
x2="0"
y2="1"
>
<stop
offset="0%"
stopColor={COLORS.UP[theme]}
stopOpacity={1}
/>
<stop
offset="100%"
stopColor={COLORS.UP[theme]}
stopOpacity={0.7}
/>
</linearGradient>
</defs>
<Bar dataKey="total_volume_usd">
{filteredData.map((entry, index) => {
return (
<Cell
key={`cell-${index}`}
fill="url(#greenGradientBar)"
/>
)
})}
</Bar>
<XAxis
dataKey="time"
axisLine={false}
dy={10}
minTickGap={20}
padding={{ left: 20, right: 20 }}
tick={{
fill: 'var(--fgd-4)',
fontSize: 10,
}}
tickLine={false}
tickFormatter={(v) =>
formatDateAxis(v, parseInt(daysToShow))
}
/>
<YAxis
dataKey="total_volume_usd"
interval="preserveStartEnd"
axisLine={false}
dx={-10}
padding={{ top: 20, bottom: 20 }}
tick={{
fill: 'var(--fgd-4)',
fontSize: 10,
}}
tickLine={false}
tickFormatter={(v) => formatYAxis(v)}
type="number"
/>
<ReferenceLine y={0} stroke={COLORS.BKG4[theme]} />
</BarChart>
</ResponsiveContainer>
</div>
</div>
) : null}
</ContentBox>
</FadeInFadeOut>
)
}
export default VolumeChart

View File

@ -16,6 +16,7 @@ export async function getStaticProps({ locale }: { locale: string }) {
'search', 'search',
'settings', 'settings',
'swap', 'swap',
'stats',
'token', 'token',
'trade', 'trade',
'close-account', 'close-account',