mango-ui-v3/components/SwapTokenInfo.tsx

836 lines
35 KiB
TypeScript
Raw Normal View History

import { FunctionComponent, useEffect, useMemo, useState } from 'react'
2022-01-05 18:59:05 -08:00
import { ExternalLinkIcon, EyeOffIcon } from '@heroicons/react/outline'
2021-12-31 04:17:48 -08:00
import { ChevronDownIcon } from '@heroicons/react/solid'
import { Disclosure } from '@headlessui/react'
2021-12-30 03:44:14 -08:00
import dayjs from 'dayjs'
2022-01-02 16:52:03 -08:00
import relativeTime from 'dayjs/plugin/relativeTime'
2021-12-30 03:44:14 -08:00
import { AreaChart, Area, XAxis, YAxis, Tooltip } from 'recharts'
import useDimensions from 'react-cool-dimensions'
2021-12-31 04:17:48 -08:00
import { IconButton } from './Button'
2021-12-31 19:07:14 -08:00
import { LineChartIcon } from './icons'
import { useTranslation } from 'next-i18next'
2021-12-31 19:07:14 -08:00
dayjs.extend(relativeTime)
2021-12-30 03:44:14 -08:00
interface SwapTokenInfoProps {
inputTokenId?: string
outputTokenId?: string
}
2022-01-02 04:20:42 -08:00
export const numberFormatter = Intl.NumberFormat('en', {
minimumFractionDigits: 1,
maximumFractionDigits: 5,
2022-01-02 04:20:42 -08:00
})
2022-01-10 14:59:12 -08:00
export const numberCompacter = Intl.NumberFormat('en', {
notation: 'compact',
maximumFractionDigits: 2,
})
2021-12-30 03:44:14 -08:00
const SwapTokenInfo: FunctionComponent<SwapTokenInfoProps> = ({
inputTokenId,
outputTokenId,
}) => {
2021-12-31 04:17:48 -08:00
const [chartData, setChartData] = useState([])
const [hideChart, setHideChart] = useState(false)
2022-01-10 18:40:20 -08:00
const [baseTokenId, setBaseTokenId] = useState('')
const [quoteTokenId, setQuoteTokenId] = useState('')
2021-12-31 19:07:14 -08:00
const [inputTokenInfo, setInputTokenInfo] = useState(null)
2021-12-30 03:44:14 -08:00
const [outputTokenInfo, setOutputTokenInfo] = useState(null)
const [mouseData, setMouseData] = useState<string | null>(null)
2022-01-02 04:20:42 -08:00
const [daysToShow, setDaysToShow] = useState(1)
2022-01-05 18:59:05 -08:00
const [topHolders, setTopHolders] = useState(null)
2021-12-30 03:44:14 -08:00
const { observe, width, height } = useDimensions()
const { t } = useTranslation(['common', 'swap'])
2021-12-30 03:44:14 -08:00
2022-01-05 18:59:05 -08:00
const getTopHolders = async (inputMint, outputMint) => {
const inputResponse = await fetch(
`https://public-api.solscan.io/token/holders?tokenAddress=${inputMint}&offset=0&limit=10`
)
const outputResponse = await fetch(
`https://public-api.solscan.io/token/holders?tokenAddress=${outputMint}&offset=0&limit=10`
)
const inputData = await inputResponse.json()
const outputData = await outputResponse.json()
setTopHolders({
inputHolders: inputData.data,
outputHolders: outputData.data,
})
}
useEffect(() => {
if (inputTokenInfo && outputTokenInfo) {
getTopHolders(
inputTokenInfo.contract_address,
outputTokenInfo.contract_address
)
}
}, [inputTokenInfo, outputTokenInfo])
2021-12-30 03:44:14 -08:00
const handleMouseMove = (coords) => {
if (coords.activePayload) {
setMouseData(coords.activePayload[0].payload)
}
}
const handleMouseLeave = () => {
setMouseData(null)
}
useEffect(() => {
if (['usd-coin', 'tether'].includes(inputTokenId)) {
setBaseTokenId(outputTokenId)
setQuoteTokenId(inputTokenId)
} else {
setBaseTokenId(inputTokenId)
setQuoteTokenId(outputTokenId)
}
}, [inputTokenId, outputTokenId])
2022-01-02 04:20:42 -08:00
// Use ohlc data
2021-12-31 04:17:48 -08:00
2021-12-30 03:44:14 -08:00
const getChartData = async () => {
const inputResponse = await fetch(
`https://api.coingecko.com/api/v3/coins/${baseTokenId}/ohlc?vs_currency=usd&days=${daysToShow}`
2021-12-30 03:44:14 -08:00
)
const outputResponse = await fetch(
`https://api.coingecko.com/api/v3/coins/${quoteTokenId}/ohlc?vs_currency=usd&days=${daysToShow}`
2021-12-30 03:44:14 -08:00
)
const inputData = await inputResponse.json()
const outputData = await outputResponse.json()
let data = []
if (Array.isArray(inputData)) {
data = data.concat(inputData)
}
if (Array.isArray(outputData)) {
data = data.concat(outputData)
}
2021-12-30 03:44:14 -08:00
const formattedData = data.reduce((a, c) => {
2022-01-02 04:20:42 -08:00
const found = a.find((price) => price.time === c[0])
2021-12-30 03:44:14 -08:00
if (found) {
2022-01-09 21:21:05 -08:00
if (['usd-coin', 'tether'].includes(quoteTokenId)) {
found.price = found.inputPrice / c[4]
} else {
found.price = c[4] / found.inputPrice
}
2021-12-30 03:44:14 -08:00
} else {
2022-01-02 04:20:42 -08:00
a.push({ time: c[0], inputPrice: c[4] })
2021-12-30 03:44:14 -08:00
}
return a
}, [])
2022-01-02 04:20:42 -08:00
formattedData[formattedData.length - 1].time = Date.now()
2021-12-30 03:44:14 -08:00
setChartData(formattedData.filter((d) => d.price))
}
2022-01-02 04:20:42 -08:00
// Alternative chart data. Needs a timestamp tolerance to get data points for each asset
// const getChartData = async () => {
// const now = Date.now() / 1000
// const inputResponse = await fetch(
// `https://api.coingecko.com/api/v3/coins/${inputTokenId}/market_chart/range?vs_currency=usd&from=${
// now - 1 * 86400
// }&to=${now}`
// )
// const outputResponse = await fetch(
// `https://api.coingecko.com/api/v3/coins/${outputTokenId}/market_chart/range?vs_currency=usd&from=${
// now - 1 * 86400
// }&to=${now}`
// )
// const inputData = await inputResponse.json()
// const outputData = await outputResponse.json()
// const data = inputData?.prices.concat(outputData?.prices)
// const formattedData = data.reduce((a, c) => {
// const found = a.find(
// (price) => c[0] >= price.time - 120000 && c[0] <= price.time + 120000
// )
// if (found) {
// found.price = found.inputPrice / c[1]
// } else {
// a.push({ time: c[0], inputPrice: c[1] })
// }
// return a
// }, [])
// setChartData(formattedData.filter((d) => d.price))
// }
2021-12-31 19:07:14 -08:00
const getInputTokenInfo = async () => {
const response = await fetch(
2022-01-09 21:21:05 -08:00
`https://api.coingecko.com/api/v3/coins/${inputTokenId}?localization=false&tickers=false&developer_data=false&sparkline=false
2021-12-31 19:07:14 -08:00
`
)
const data = await response.json()
setInputTokenInfo(data)
}
2021-12-30 03:44:14 -08:00
const getOutputTokenInfo = async () => {
const response = await fetch(
2022-01-09 21:21:05 -08:00
`https://api.coingecko.com/api/v3/coins/${outputTokenId}?localization=false&tickers=false&developer_data=false&sparkline=false
2021-12-30 03:44:14 -08:00
`
)
const data = await response.json()
setOutputTokenInfo(data)
}
useMemo(() => {
if (baseTokenId && quoteTokenId) {
2021-12-30 03:44:14 -08:00
getChartData()
}
}, [daysToShow, baseTokenId, quoteTokenId])
2021-12-30 03:44:14 -08:00
useMemo(() => {
if (baseTokenId) {
2021-12-31 19:07:14 -08:00
getInputTokenInfo()
}
if (quoteTokenId) {
2021-12-30 03:44:14 -08:00
getOutputTokenInfo()
}
}, [baseTokenId, quoteTokenId])
2021-12-30 03:44:14 -08:00
2021-12-31 04:17:48 -08:00
const chartChange = chartData.length
? ((chartData[chartData.length - 1]['price'] - chartData[0]['price']) /
chartData[0]['price']) *
100
: 0
2021-12-30 03:44:14 -08:00
return (
<div>
{chartData.length && baseTokenId && quoteTokenId ? (
2022-01-09 21:21:05 -08:00
<div className="pb-6">
2022-01-02 04:20:42 -08:00
<div className="flex items-start justify-between">
2021-12-30 03:44:14 -08:00
<div>
{inputTokenInfo && outputTokenInfo ? (
2022-01-09 21:21:05 -08:00
<div className="text-th-fgd-3 text-sm">
2022-01-23 12:03:59 -08:00
{`${outputTokenInfo?.symbol?.toUpperCase()}/${inputTokenInfo?.symbol?.toUpperCase()}`}
2022-01-09 21:21:05 -08:00
</div>
2021-12-30 03:44:14 -08:00
) : null}
{mouseData ? (
<>
<div className="font-bold text-lg text-th-fgd-1">
2021-12-31 19:07:14 -08:00
{numberFormatter.format(mouseData['price'])}
2021-12-31 04:17:48 -08:00
<span
2022-01-02 04:20:42 -08:00
className={`ml-2 text-xs ${
2021-12-31 04:17:48 -08:00
chartChange >= 0 ? 'text-th-green' : 'text-th-red'
}`}
>
{chartChange.toFixed(2)}%
</span>
2021-12-30 03:44:14 -08:00
</div>
2022-01-02 04:20:42 -08:00
<div className="text-xs font-normal text-th-fgd-3">
2021-12-30 03:44:14 -08:00
{dayjs(mouseData['time']).format('DD MMM YY, h:mma')}
</div>
</>
) : (
<>
<div className="font-bold text-lg text-th-fgd-1">
2021-12-31 19:07:14 -08:00
{numberFormatter.format(
chartData[chartData.length - 1]['price']
2021-12-30 03:44:14 -08:00
)}
2021-12-31 04:17:48 -08:00
<span
2022-01-01 04:34:46 -08:00
className={`ml-2 text-xs ${
2021-12-31 04:17:48 -08:00
chartChange >= 0 ? 'text-th-green' : 'text-th-red'
}`}
>
{chartChange.toFixed(2)}%
</span>
2021-12-30 03:44:14 -08:00
</div>
2022-01-01 04:34:46 -08:00
<div className="text-xs font-normal text-th-fgd-3">
2021-12-30 03:44:14 -08:00
{dayjs(chartData[chartData.length - 1]['time']).format(
'DD MMM YY, h:mma'
)}
</div>
</>
)}
</div>
2021-12-31 04:17:48 -08:00
<IconButton onClick={() => setHideChart(!hideChart)}>
{hideChart ? (
2021-12-31 19:07:14 -08:00
<LineChartIcon className="w-4 h-4" />
2021-12-31 04:17:48 -08:00
) : (
<EyeOffIcon className="w-4 h-4" />
)}
</IconButton>
2021-12-30 03:44:14 -08:00
</div>
2021-12-31 04:17:48 -08:00
{!hideChart ? (
<div className="h-52 mt-4 w-full" ref={observe}>
<AreaChart
width={width}
height={height}
data={chartData}
onMouseMove={handleMouseMove}
onMouseLeave={handleMouseLeave}
>
<Tooltip
cursor={{
strokeOpacity: 0,
}}
content={<></>}
/>
<defs>
<linearGradient id="gradientArea" x1="0" y1="0" x2="0" y2="1">
2022-01-06 18:35:48 -08:00
<stop offset="0%" stopColor="#ffba24" stopOpacity={0.9} />
<stop offset="90%" stopColor="#ffba24" stopOpacity={0} />
2021-12-31 04:17:48 -08:00
</linearGradient>
</defs>
<Area
isAnimationActive={true}
type="monotone"
dataKey="price"
2022-01-06 18:35:48 -08:00
stroke="#ffba24"
2021-12-31 04:17:48 -08:00
fill="url(#gradientArea)"
/>
<XAxis dataKey="time" hide />
<YAxis
dataKey="price"
type="number"
domain={['dataMin', 'dataMax']}
hide
/>
</AreaChart>
2022-01-02 04:20:42 -08:00
<div className="flex justify-end">
<button
className={`default-transition font-bold px-3 py-2 text-th-fgd-1 text-xs hover:text-th-primary focus:outline-none ${
daysToShow === 1 && 'text-th-primary'
}`}
onClick={() => setDaysToShow(1)}
>
24H
</button>
<button
className={`default-transition font-bold px-3 py-2 text-th-fgd-1 text-xs hover:text-th-primary focus:outline-none ${
daysToShow === 7 && 'text-th-primary'
}`}
onClick={() => setDaysToShow(7)}
>
7D
</button>
<button
className={`default-transition font-bold px-3 py-2 text-th-fgd-1 text-xs hover:text-th-primary focus:outline-none ${
daysToShow === 30 && 'text-th-primary'
}`}
onClick={() => setDaysToShow(30)}
>
30D
</button>
</div>
2021-12-31 04:17:48 -08:00
</div>
) : null}
2021-12-30 03:44:14 -08:00
</div>
) : (
2022-01-02 04:20:42 -08:00
<div className="bg-th-bkg-3 mt-4 md:mt-0 p-4 rounded-md text-center text-th-fgd-3">
<LineChartIcon className="h-6 mx-auto text-th-primary w-6" />
{t('swap:chart-not-available')}
2021-12-30 03:44:14 -08:00
</div>
)}
2021-12-31 04:17:48 -08:00
{inputTokenInfo && outputTokenInfo && baseTokenId ? (
2022-01-01 04:34:46 -08:00
<div className="w-full">
2021-12-31 04:17:48 -08:00
<Disclosure>
{({ open }) => (
<>
<Disclosure.Button
2022-01-02 04:20:42 -08:00
className={`border border-th-bkg-4 default-transition flex items-center justify-between mt-6 p-3 rounded-md w-full hover:bg-th-bkg-2 ${
2021-12-31 04:17:48 -08:00
open
? 'border-b-transparent rounded-b-none'
: 'transform rotate-360'
}`}
>
2021-12-31 19:07:14 -08:00
<div className="flex items-center">
{inputTokenInfo.image?.small ? (
<img
2022-01-09 21:21:05 -08:00
className="rounded-full"
2021-12-31 19:07:14 -08:00
src={inputTokenInfo.image?.small}
width="32"
height="32"
alt={inputTokenInfo.name}
/>
) : null}
<div className="ml-2.5 text-left">
<h2 className="font-bold text-base text-th-fgd-1">
{inputTokenInfo?.symbol?.toUpperCase()}
2021-12-31 19:07:14 -08:00
</h2>
<div className="font-normal text-th-fgd-3 text-xs">
{inputTokenInfo.name}
</div>
</div>
</div>
<div className="flex items-center">
2022-01-01 04:34:46 -08:00
<div className="flex items-center space-x-3">
2021-12-31 19:07:14 -08:00
{inputTokenInfo.market_data?.current_price?.usd ? (
<div className="font-normal text-th-fgd-1">
$
{numberFormatter.format(
inputTokenInfo.market_data?.current_price.usd
2021-12-31 19:07:14 -08:00
)}
</div>
) : null}
{inputTokenInfo.market_data
?.price_change_percentage_24h ? (
<div
className={`font-normal text-th-fgd-1 ${
inputTokenInfo.market_data
.price_change_percentage_24h >= 0
? 'text-th-green'
: 'text-th-red'
}`}
>
{inputTokenInfo.market_data.price_change_percentage_24h.toFixed(
2
)}
%
</div>
) : null}
</div>
<ChevronDownIcon
className={`default-transition h-6 ml-2 w-6 text-th-fgd-3 ${
open ? 'transform rotate-180' : 'transform rotate-360'
}`}
/>
</div>
2021-12-31 04:17:48 -08:00
</Disclosure.Button>
<Disclosure.Panel>
2022-01-05 18:59:05 -08:00
<div className="border border-th-bkg-4 border-t-0 p-3 rounded-b-md">
<div className="font-bold m-1 mt-0 pb-2 text-th-fgd-1 text-base">
Market Data
</div>
<div className="grid grid-cols-1 sm:grid-cols-2 md:grid-cols-3 lg:grid-cols-2 xl:grid-cols-3 grid-flow-row">
{inputTokenInfo.market_cap_rank ? (
<div className="border border-th-bkg-4 m-1 p-3 rounded-md">
<div className="text-th-fgd-3 text-xs">
2022-01-17 17:54:21 -08:00
{t('swap:market-cap-rank')}
2022-01-05 18:59:05 -08:00
</div>
<div className="font-bold text-th-fgd-1 text-lg">
#{inputTokenInfo.market_cap_rank}
</div>
2021-12-31 19:07:14 -08:00
</div>
2022-01-05 18:59:05 -08:00
) : null}
2022-01-10 14:59:12 -08:00
{inputTokenInfo.market_data?.market_cap &&
inputTokenInfo.market_data?.market_cap?.usd !== 0 ? (
2022-01-05 18:59:05 -08:00
<div className="border border-th-bkg-4 m-1 p-3 rounded-md">
<div className="text-th-fgd-3 text-xs">
2022-01-17 17:54:21 -08:00
{t('swap:market-cap')}
2022-01-05 18:59:05 -08:00
</div>
<div className="font-bold text-th-fgd-1 text-lg">
$
2022-01-10 14:59:12 -08:00
{numberCompacter.format(
2022-01-05 18:59:05 -08:00
inputTokenInfo.market_data?.market_cap?.usd
2021-12-31 19:07:14 -08:00
)}
</div>
</div>
2022-01-05 18:59:05 -08:00
) : null}
{inputTokenInfo.market_data?.total_volume?.usd ? (
2022-01-05 18:59:05 -08:00
<div className="border border-th-bkg-4 m-1 p-3 rounded-md">
<div className="text-th-fgd-3 text-xs">
2022-01-17 17:54:21 -08:00
{t('daily-volume')}
2022-01-05 18:59:05 -08:00
</div>
2021-12-31 19:07:14 -08:00
<div className="font-bold text-th-fgd-1 text-lg">
$
2022-01-10 14:59:12 -08:00
{numberCompacter.format(
inputTokenInfo.market_data?.total_volume?.usd
2021-12-31 19:07:14 -08:00
)}
</div>
</div>
2022-01-05 18:59:05 -08:00
) : null}
{inputTokenInfo.market_data?.circulating_supply ? (
<div className="border border-th-bkg-4 m-1 p-3 rounded-md">
<div className="text-th-fgd-3 text-xs">
2022-01-17 17:54:21 -08:00
{t('swap:token-supply')}
2021-12-31 19:07:14 -08:00
</div>
<div className="font-bold text-th-fgd-1 text-lg">
2022-01-10 14:59:12 -08:00
{numberCompacter.format(
2022-01-05 18:59:05 -08:00
inputTokenInfo.market_data.circulating_supply
2021-12-31 19:07:14 -08:00
)}
</div>
2022-01-05 18:59:05 -08:00
{inputTokenInfo.market_data?.max_supply ? (
<div className="text-th-fgd-2 text-xs">
2022-01-17 17:54:21 -08:00
{t('swap:max-supply')}:{' '}
2022-01-10 14:59:12 -08:00
{numberCompacter.format(
2022-01-05 18:59:05 -08:00
inputTokenInfo.market_data.max_supply
)}
</div>
) : null}
</div>
) : null}
{inputTokenInfo.market_data?.ath?.usd ? (
<div className="border border-th-bkg-4 m-1 p-3 rounded-md">
<div className="text-th-fgd-3 text-xs">
2022-01-17 17:54:21 -08:00
{t('swap:ath')}
2022-01-05 18:59:05 -08:00
</div>
<div className="flex">
<div className="font-bold text-th-fgd-1 text-lg">
$
{numberFormatter.format(
inputTokenInfo.market_data.ath.usd
2021-12-31 19:07:14 -08:00
)}
2022-01-05 18:59:05 -08:00
</div>
{inputTokenInfo.market_data?.ath_change_percentage
?.usd ? (
<div
className={`ml-1.5 mt-2 text-xs ${
inputTokenInfo.market_data
?.ath_change_percentage?.usd >= 0
? 'text-th-green'
: 'text-th-red'
}`}
>
{(inputTokenInfo.market_data?.ath_change_percentage?.usd).toFixed(
2
)}
%
</div>
) : null}
</div>
{inputTokenInfo.market_data?.ath_date?.usd ? (
<div className="text-th-fgd-2 text-xs">
{dayjs(
inputTokenInfo.market_data.ath_date.usd
).fromNow()}
2021-12-31 19:07:14 -08:00
</div>
) : null}
</div>
2022-01-05 18:59:05 -08:00
) : null}
{inputTokenInfo.market_data?.atl?.usd ? (
<div className="border border-th-bkg-4 m-1 p-3 rounded-md">
<div className="text-th-fgd-3 text-xs">
2022-01-17 17:54:21 -08:00
{t('swap:atl')}
2022-01-05 18:59:05 -08:00
</div>
<div className="flex">
<div className="font-bold text-th-fgd-1 text-lg">
$
{numberFormatter.format(
inputTokenInfo.market_data.atl.usd
)}
</div>
{inputTokenInfo.market_data?.atl_change_percentage
?.usd ? (
<div
className={`ml-1.5 mt-2 text-xs ${
inputTokenInfo.market_data
?.atl_change_percentage?.usd >= 0
? 'text-th-green'
: 'text-th-red'
}`}
>
{(inputTokenInfo.market_data?.atl_change_percentage?.usd).toLocaleString(
undefined,
{
minimumFractionDigits: 0,
maximumFractionDigits: 2,
}
)}
%
</div>
) : null}
2021-12-31 19:07:14 -08:00
</div>
2022-01-05 18:59:05 -08:00
{inputTokenInfo.market_data?.atl_date?.usd ? (
<div className="text-th-fgd-2 text-xs">
{dayjs(
inputTokenInfo.market_data.atl_date.usd
).fromNow()}
</div>
) : null}
</div>
) : null}
</div>
{topHolders?.inputHolders ? (
<div className="pt-4">
<div className="font-bold m-1 pb-3 text-th-fgd-1 text-base">
Top 10 Holders
</div>
{topHolders.inputHolders.map((holder) => (
<a
className="border-t border-th-bkg-4 default transition flex justify-between mx-1 px-2 py-2.5 text-th-fgd-3 hover:bg-th-bkg-2"
href={`https://explorer.solana.com/address/${holder.owner}`}
target="_blank"
rel="noopener noreferrer"
key={holder.owner}
>
<div className="text-th-fgd-3">
{holder.owner.slice(0, 5) +
'…' +
holder.owner.slice(-5)}
</div>
<div className="flex items-center">
<div className="text-th-fgd-1">
{numberFormatter.format(
holder.amount / Math.pow(10, holder.decimals)
)}
</div>
<ExternalLinkIcon className="h-4 ml-2 w-4" />
</div>
</a>
))}
2021-12-31 19:07:14 -08:00
</div>
) : null}
</div>
</Disclosure.Panel>
</>
)}
</Disclosure>
</div>
) : (
2022-01-02 04:20:42 -08:00
<div className="bg-th-bkg-3 mt-3 p-4 rounded-md text-center text-th-fgd-3">
{t('swap:input-info-unavailable')}
2021-12-31 19:07:14 -08:00
</div>
)}
2022-01-09 21:21:05 -08:00
{outputTokenInfo && quoteTokenId ? (
2022-01-01 04:34:46 -08:00
<div className="w-full">
2021-12-31 19:07:14 -08:00
<Disclosure>
{({ open }) => (
<>
<Disclosure.Button
2022-01-01 04:34:46 -08:00
className={`border border-th-bkg-4 default-transition flex items-center justify-between mt-3 p-3 rounded-md w-full hover:bg-th-bkg-2 ${
2021-12-31 19:07:14 -08:00
open
? 'border-b-transparent rounded-b-none'
: 'transform rotate-360'
}`}
>
<div className="flex items-center">
{outputTokenInfo.image?.small ? (
<img
2022-01-09 21:21:05 -08:00
className="rounded-full"
2021-12-31 19:07:14 -08:00
src={outputTokenInfo.image?.small}
width="32"
height="32"
alt={outputTokenInfo.name}
/>
) : null}
<div className="ml-2.5 text-left">
<h2 className="font-bold text-base text-th-fgd-1">
{outputTokenInfo?.symbol?.toUpperCase()}
2021-12-31 19:07:14 -08:00
</h2>
<div className="font-normal text-th-fgd-3 text-xs">
{outputTokenInfo.name}
</div>
</div>
</div>
<div className="flex items-center">
2022-01-01 04:34:46 -08:00
<div className="flex items-center space-x-3">
2021-12-31 19:07:14 -08:00
{outputTokenInfo.market_data?.current_price?.usd ? (
<div className="font-normal text-th-fgd-1">
$
{numberFormatter.format(
outputTokenInfo.market_data.current_price.usd
)}
</div>
) : null}
{outputTokenInfo.market_data
?.price_change_percentage_24h ? (
<div
className={`font-normal text-th-fgd-1 ${
outputTokenInfo.market_data
.price_change_percentage_24h >= 0
? 'text-th-green'
: 'text-th-red'
}`}
>
{outputTokenInfo.market_data.price_change_percentage_24h.toFixed(
2
)}
%
</div>
) : null}
</div>
<ChevronDownIcon
className={`default-transition h-6 ml-2 w-6 text-th-fgd-3 ${
open ? 'transform rotate-180' : 'transform rotate-360'
}`}
/>
</div>
</Disclosure.Button>
<Disclosure.Panel>
2022-01-05 18:59:05 -08:00
<div className="border border-th-bkg-4 border-t-0 p-3 rounded-b-md">
<div className="font-bold m-1 mt-0 pb-2 text-th-fgd-1 text-base">
Market Data
</div>
<div className="grid grid-cols-1 sm:grid-cols-2 md:grid-cols-3 lg:grid-cols-2 xl:grid-cols-3 grid-flow-row">
{outputTokenInfo.market_cap_rank ? (
<div className="border border-th-bkg-4 m-1 p-3 rounded-md">
<div className="text-th-fgd-3 text-xs">
2022-01-17 17:54:21 -08:00
{t('swap:market-cap-rank')}
2022-01-05 18:59:05 -08:00
</div>
<div className="font-bold text-th-fgd-1 text-lg">
#{outputTokenInfo.market_cap_rank}
</div>
2021-12-31 19:07:14 -08:00
</div>
2022-01-05 18:59:05 -08:00
) : null}
2022-01-10 14:59:12 -08:00
{outputTokenInfo.market_data?.market_cap &&
outputTokenInfo.market_data?.market_cap?.usd !== 0 ? (
2022-01-05 18:59:05 -08:00
<div className="border border-th-bkg-4 m-1 p-3 rounded-md">
<div className="text-th-fgd-3 text-xs">
2022-01-17 17:54:21 -08:00
{t('swap:market-cap')}
2022-01-05 18:59:05 -08:00
</div>
<div className="font-bold text-th-fgd-1 text-lg">
$
2022-01-10 14:59:12 -08:00
{numberCompacter.format(
2022-01-05 18:59:05 -08:00
outputTokenInfo.market_data?.market_cap?.usd
2021-12-31 19:07:14 -08:00
)}
</div>
</div>
2022-01-05 18:59:05 -08:00
) : null}
{outputTokenInfo.market_data?.total_volume?.usd ? (
2022-01-05 18:59:05 -08:00
<div className="border border-th-bkg-4 m-1 p-3 rounded-md">
<div className="text-th-fgd-3 text-xs">
2022-01-17 17:54:21 -08:00
{t('daily-volume')}
2022-01-05 18:59:05 -08:00
</div>
2021-12-31 19:07:14 -08:00
<div className="font-bold text-th-fgd-1 text-lg">
$
2022-01-10 14:59:12 -08:00
{numberCompacter.format(
outputTokenInfo.market_data?.total_volume?.usd
2021-12-31 19:07:14 -08:00
)}
</div>
</div>
2022-01-05 18:59:05 -08:00
) : null}
{outputTokenInfo.market_data?.circulating_supply ? (
<div className="border border-th-bkg-4 m-1 p-3 rounded-md">
<div className="text-th-fgd-3 text-xs">
2022-01-17 17:54:21 -08:00
{t('swap:token-supply')}
2021-12-31 19:07:14 -08:00
</div>
<div className="font-bold text-th-fgd-1 text-lg">
2022-01-10 14:59:12 -08:00
{numberCompacter.format(
2022-01-05 18:59:05 -08:00
outputTokenInfo.market_data.circulating_supply
2021-12-31 19:07:14 -08:00
)}
</div>
2022-01-05 18:59:05 -08:00
{outputTokenInfo.market_data?.max_supply ? (
<div className="text-th-fgd-2 text-xs">
2022-01-17 17:54:21 -08:00
{t('swap:max-supply')}:{' '}
2022-01-10 14:59:12 -08:00
{numberCompacter.format(
2022-01-05 18:59:05 -08:00
outputTokenInfo.market_data.max_supply
)}
</div>
) : null}
</div>
) : null}
{outputTokenInfo.market_data?.ath?.usd ? (
<div className="border border-th-bkg-4 m-1 p-3 rounded-md">
<div className="text-th-fgd-3 text-xs">
2022-01-17 17:54:21 -08:00
{t('swap:ath')}
2022-01-05 18:59:05 -08:00
</div>
<div className="flex">
<div className="font-bold text-th-fgd-1 text-lg">
$
{numberFormatter.format(
outputTokenInfo.market_data.ath.usd
2021-12-31 19:07:14 -08:00
)}
2022-01-05 18:59:05 -08:00
</div>
{outputTokenInfo.market_data?.ath_change_percentage
?.usd ? (
<div
className={`ml-1.5 mt-2 text-xs ${
outputTokenInfo.market_data
?.ath_change_percentage?.usd >= 0
? 'text-th-green'
: 'text-th-red'
}`}
>
{(outputTokenInfo.market_data?.ath_change_percentage?.usd).toFixed(
2
)}
%
</div>
) : null}
</div>
{outputTokenInfo.market_data?.ath_date?.usd ? (
<div className="text-th-fgd-2 text-xs">
{dayjs(
outputTokenInfo.market_data.ath_date.usd
).fromNow()}
2021-12-31 19:07:14 -08:00
</div>
) : null}
</div>
2022-01-05 18:59:05 -08:00
) : null}
{outputTokenInfo.market_data?.atl?.usd ? (
<div className="border border-th-bkg-4 m-1 p-3 rounded-md">
<div className="text-th-fgd-3 text-xs">
2022-01-17 17:54:21 -08:00
{t('swap:atl')}
2022-01-05 18:59:05 -08:00
</div>
<div className="flex">
<div className="font-bold text-th-fgd-1 text-lg">
$
{numberFormatter.format(
outputTokenInfo.market_data.atl.usd
)}
</div>
{outputTokenInfo.market_data?.atl_change_percentage
?.usd ? (
<div
className={`ml-1.5 mt-2 text-xs ${
outputTokenInfo.market_data
?.atl_change_percentage?.usd >= 0
? 'text-th-green'
: 'text-th-red'
}`}
>
{(outputTokenInfo.market_data?.atl_change_percentage?.usd).toLocaleString(
undefined,
{
minimumFractionDigits: 0,
maximumFractionDigits: 2,
}
)}
%
</div>
) : null}
2021-12-31 19:07:14 -08:00
</div>
2022-01-05 18:59:05 -08:00
{outputTokenInfo.market_data?.atl_date?.usd ? (
<div className="text-th-fgd-2 text-xs">
{dayjs(
outputTokenInfo.market_data.atl_date.usd
).fromNow()}
</div>
) : null}
</div>
) : null}
</div>
{topHolders?.outputHolders ? (
<div className="pt-4">
<div className="font-bold m-1 pb-3 text-th-fgd-1 text-base">
Top 10 Holders
</div>
{topHolders.outputHolders.map((holder) => (
<a
className="border-t border-th-bkg-4 default transition flex justify-between mx-1 px-2 py-2.5 text-th-fgd-3 hover:bg-th-bkg-2"
href={`https://explorer.solana.com/address/${holder.owner}`}
target="_blank"
rel="noopener noreferrer"
key={holder.owner}
>
<div className="text-th-fgd-3">
{holder.owner.slice(0, 5) +
'…' +
holder.owner.slice(-5)}
</div>
<div className="flex items-center">
<div className="text-th-fgd-1">
{numberFormatter.format(
holder.amount / Math.pow(10, holder.decimals)
)}
</div>
<ExternalLinkIcon className="h-4 ml-2 w-4" />
</div>
</a>
))}
2021-12-31 19:07:14 -08:00
</div>
) : null}
2021-12-31 04:17:48 -08:00
</div>
</Disclosure.Panel>
</>
)}
</Disclosure>
2021-12-30 03:44:14 -08:00
</div>
2021-12-31 19:07:14 -08:00
) : (
2022-01-02 04:20:42 -08:00
<div className="bg-th-bkg-3 mt-3 p-4 rounded-md text-center text-th-fgd-3">
{t('swap:output-info-unavailable')}
2021-12-31 19:07:14 -08:00
</div>
)}
2021-12-30 03:44:14 -08:00
</div>
)
}
export default SwapTokenInfo