mango-v4-ui/components/token/TokenPage.tsx

233 lines
7.7 KiB
TypeScript

import Change from '@components/shared/Change'
import DailyRange from '@components/shared/DailyRange'
import { useTranslation } from 'next-i18next'
import { useRouter } from 'next/router'
import { useMemo, useState } from 'react'
import FlipNumbers from 'react-flip-numbers'
import { formatCurrencyValue } from 'utils/numbers'
import Link from 'next/link'
import SheenLoader from '@components/shared/SheenLoader'
import useMangoGroup from 'hooks/useMangoGroup'
import useJupiterMints from 'hooks/useJupiterMints'
import useLocalStorageState from 'hooks/useLocalStorageState'
import { ANIMATION_SETTINGS_KEY } from 'utils/constants'
import { INITIAL_ANIMATION_SETTINGS } from '@components/settings/AnimationSettings'
import ActionPanel from './ActionPanel'
import ChartTabs from './ChartTabs'
import CoingeckoStats from './CoingeckoStats'
import { useQuery } from '@tanstack/react-query'
import FormatNumericValue from '@components/shared/FormatNumericValue'
import TopTokenAccounts from './TopTokenAccounts'
import TokenParams from './TokenParams'
import { formatTokenSymbol } from 'utils/tokens'
import TokenLogo from '@components/shared/TokenLogo'
import { ArrowLeftIcon } from '@heroicons/react/20/solid'
import RateCurveChart from './RateCurveChart'
const DEFAULT_COINGECKO_VALUES = {
ath: 0,
atl: 0,
ath_change_percentage: 0,
atl_change_percentage: 0,
ath_date: 0,
atl_date: 0,
high_24h: { usd: 0 },
circulating_supply: 0,
fully_diluted_valuation: 0,
low_24h: { usd: 0 },
market_cap: 0,
max_supply: 0,
price_change_percentage_24h: 0,
total_supply: 0,
total_volume: 0,
}
type CryptoStatsResponse = {
high_24h: Record<string, number>
low_24h: Record<string, number>
price_change_percentage_24h: number
}
export type CoingeckoDataType = {
market_data: CryptoStatsResponse
name: string
}
const fetchTokenInfo = async (tokenId: string | undefined) => {
if (!tokenId) return
const response = await fetch(
`https://api.coingecko.com/api/v3/coins/${tokenId}?localization=false&tickers=false&developer_data=false&sparkline=false&community_data=false
`,
)
const data = await response.json()
return data
}
const TokenPage = () => {
const { t } = useTranslation(['common', 'token'])
const [loading, setLoading] = useState(true)
const router = useRouter()
const { token } = router.query
const { group } = useMangoGroup()
const { mangoTokens } = useJupiterMints()
const [animationSettings] = useLocalStorageState(
ANIMATION_SETTINGS_KEY,
INITIAL_ANIMATION_SETTINGS,
)
const bankName = useMemo(() => {
if (!token) return
return token === 'wBTC'
? 'wBTC (Portal)'
: token === 'ETH'
? 'ETH (Portal)'
: token.toString()
}, [token])
const bank = useMemo(() => {
if (group && bankName) {
const bank = group.banksMapByName.get(bankName)
if (bank) {
return bank[0]
} else {
setLoading(false)
}
}
}, [group, bankName])
const coingeckoId = useMemo(() => {
if (bank && mangoTokens.length) {
return mangoTokens.find((t) => t.address === bank.mint.toString())
?.extensions?.coingeckoId
}
}, [bank, mangoTokens])
const { data: coingeckoTokenInfo, isLoading: loadingCoingeckoInfo } =
useQuery<CoingeckoDataType, Error>(
['coingecko-token-info', coingeckoId],
() => fetchTokenInfo(coingeckoId),
{
cacheTime: 1000 * 60 * 15,
staleTime: 1000 * 60 * 5,
retry: 3,
refetchOnWindowFocus: false,
enabled: !!coingeckoId,
},
)
const { high_24h, low_24h, price_change_percentage_24h } =
coingeckoTokenInfo?.market_data
? coingeckoTokenInfo.market_data
: DEFAULT_COINGECKO_VALUES
return (
<>
<div className="flex h-14 items-center space-x-4 border-b border-th-bkg-3">
<button
className="flex h-14 w-14 flex-shrink-0 items-center justify-center border-r border-th-bkg-3 focus-visible:bg-th-bkg-3 md:hover:bg-th-bkg-2"
onClick={() =>
router.push(router.pathname, undefined, { shallow: true })
}
>
<ArrowLeftIcon className="h-5 w-5" />
</button>
{bank ? (
<span className="text-lg font-bold text-th-fgd-1">
{formatTokenSymbol(bank.name)}
</span>
) : null}
</div>
{bank && bankName ? (
<>
<div className="flex flex-col border-b border-th-bkg-3 px-6 py-5 md:flex-row md:items-center md:justify-between">
<div className="mb-4 md:mb-1">
<div className="mb-1.5 flex items-center space-x-2">
<TokenLogo bank={bank} />
{coingeckoTokenInfo ? (
<h1 className="text-base font-normal">
{coingeckoTokenInfo.name}
</h1>
) : (
<h1 className="text-base font-normal">{bank.name}</h1>
)}
</div>
<div className="flex flex-wrap items-end font-display text-4xl text-th-fgd-1 sm:text-5xl">
<div className="mb-0.5 mr-3 sm:mb-2">
{animationSettings['number-scroll'] ? (
<FlipNumbers
height={48}
width={35}
play
delay={0.05}
duration={1}
numbers={formatCurrencyValue(bank.uiPrice)}
/>
) : (
<FormatNumericValue value={bank.uiPrice} isUsd />
)}
</div>
{coingeckoTokenInfo?.market_data ? (
<div className="mb-2">
<Change change={price_change_percentage_24h} suffix="%" />
</div>
) : null}
</div>
{high_24h.usd && low_24h.usd ? (
<DailyRange
high={high_24h.usd}
low={low_24h.usd}
price={bank.uiPrice}
/>
) : null}
</div>
<ActionPanel bank={bank} />
</div>
<ChartTabs bank={bank} />
<div className="border-y border-th-bkg-3 px-6 pb-2 pt-6">
<RateCurveChart bank={bank} />
</div>
<TopTokenAccounts bank={bank} />
{coingeckoTokenInfo?.market_data ? (
<CoingeckoStats
bank={bank}
coingeckoData={coingeckoTokenInfo.market_data}
/>
) : loadingCoingeckoInfo && coingeckoId ? (
<div className="p-6">
<SheenLoader className="flex flex-1">
<div className="h-72 w-full rounded-lg bg-th-bkg-2 md:h-80" />
</SheenLoader>
</div>
) : (
<div className="flex flex-col items-center p-6">
<span className="mb-0.5 text-2xl">🦎</span>
<p>No CoinGecko data...</p>
</div>
)}
<TokenParams bank={bank} />
</>
) : loading ? (
<div className="space-y-3 px-6 py-4">
<SheenLoader className="flex flex-1">
<div className="h-32 w-full rounded-lg bg-th-bkg-2" />
</SheenLoader>
<SheenLoader className="flex flex-1">
<div className="h-72 w-full rounded-lg bg-th-bkg-2" />
</SheenLoader>
</div>
) : (
<div className="-mt-8 flex h-screen flex-col items-center justify-center">
<p className="text-3xl">😔</p>
<h2 className="mb-1">{t('token:token-not-found')}</h2>
<p className="mb-2">
{t('token:token-not-found-desc', { token: token })}
</p>
<Link href="/">{t('token:go-to-account')}</Link>
</div>
)}
</>
)
}
export default TokenPage