Make leaderboard code generic on entry type

This commit is contained in:
Marshall Polaris 2022-09-21 15:02:18 -07:00
parent 91f92bd362
commit 74a046f6d6
4 changed files with 24 additions and 18 deletions

View File

@ -49,7 +49,7 @@ export function ContractLeaderboard(props: {
return users && users.length > 0 ? ( return users && users.length > 0 ? (
<Leaderboard <Leaderboard
title={`🏅 Top ${BETTORS}`} title={`🏅 Top ${BETTORS}`}
users={users || []} entries={users || []}
columns={[ columns={[
{ {
header: 'Total profit', header: 'Total profit',

View File

@ -1,28 +1,34 @@
import clsx from 'clsx' import clsx from 'clsx'
import { User } from 'common/user'
import { Avatar } from './avatar' import { Avatar } from './avatar'
import { Row } from './layout/row' import { Row } from './layout/row'
import { SiteLink } from './site-link' import { SiteLink } from './site-link'
import { Title } from './title' import { Title } from './title'
export function Leaderboard(props: { interface LeaderboardEntry {
id: string
username: string
name: string
avatarUrl?: string
}
export function Leaderboard<T extends LeaderboardEntry>(props: {
title: string title: string
users: User[] entries: T[]
columns: { columns: {
header: string header: string
renderCell: (user: User) => any renderCell: (entry: T) => any
}[] }[]
className?: string className?: string
maxToShow?: number maxToShow?: number
}) { }) {
// TODO: Ideally, highlight your own entry on the leaderboard // TODO: Ideally, highlight your own entry on the leaderboard
const { title, columns, className } = props const { title, columns, className } = props
const maxToShow = props.maxToShow ?? props.users.length const maxToShow = props.maxToShow ?? props.entries.length
const users = props.users.slice(0, maxToShow) const entries = props.entries.slice(0, maxToShow)
return ( return (
<div className={clsx('w-full px-1', className)}> <div className={clsx('w-full px-1', className)}>
<Title text={title} className="!mt-0" /> <Title text={title} className="!mt-0" />
{users.length === 0 ? ( {entries.length === 0 ? (
<div className="ml-2 text-gray-500">None yet</div> <div className="ml-2 text-gray-500">None yet</div>
) : ( ) : (
<div className="overflow-x-auto"> <div className="overflow-x-auto">
@ -37,19 +43,19 @@ export function Leaderboard(props: {
</tr> </tr>
</thead> </thead>
<tbody> <tbody>
{users.map((user, index) => ( {entries.map((entry, index) => (
<tr key={user.id}> <tr key={entry.id}>
<td>{index + 1}</td> <td>{index + 1}</td>
<td className="max-w-[190px]"> <td className="max-w-[190px]">
<SiteLink className="relative" href={`/${user.username}`}> <SiteLink className="relative" href={`/${entry.username}`}>
<Row className="items-center gap-4"> <Row className="items-center gap-4">
<Avatar avatarUrl={user.avatarUrl} size={8} /> <Avatar avatarUrl={entry.avatarUrl} size={8} />
<div className="truncate">{user.name}</div> <div className="truncate">{entry.name}</div>
</Row> </Row>
</SiteLink> </SiteLink>
</td> </td>
{columns.map((column) => ( {columns.map((column) => (
<td key={column.header}>{column.renderCell(user)}</td> <td key={column.header}>{column.renderCell(entry)}</td>
))} ))}
</tr> </tr>
))} ))}

View File

@ -451,7 +451,7 @@ function GroupLeaderboard(props: {
return ( return (
<Leaderboard <Leaderboard
className="max-w-xl" className="max-w-xl"
users={topUsers.map((t) => t.user)} entries={topUsers.map((t) => t.user)}
title={title} title={title}
columns={[ columns={[
{ header, renderCell: (user) => formatMoney(scoresByUser[user.id]) }, { header, renderCell: (user) => formatMoney(scoresByUser[user.id]) },

View File

@ -81,7 +81,7 @@ export default function Leaderboards(_props: {
<Col className="mx-4 items-center gap-10 lg:flex-row"> <Col className="mx-4 items-center gap-10 lg:flex-row">
<Leaderboard <Leaderboard
title={`🏅 Top ${BETTORS}`} title={`🏅 Top ${BETTORS}`}
users={topTraders} entries={topTraders}
columns={[ columns={[
{ {
header: 'Total profit', header: 'Total profit',
@ -92,7 +92,7 @@ export default function Leaderboards(_props: {
<Leaderboard <Leaderboard
title="🏅 Top creators" title="🏅 Top creators"
users={topCreators} entries={topCreators}
columns={[ columns={[
{ {
header: 'Total bet', header: 'Total bet',
@ -106,7 +106,7 @@ export default function Leaderboards(_props: {
<Col className="mx-4 my-10 items-center gap-10 lg:mx-0 lg:w-1/2 lg:flex-row"> <Col className="mx-4 my-10 items-center gap-10 lg:mx-0 lg:w-1/2 lg:flex-row">
<Leaderboard <Leaderboard
title="🏅 Top followed" title="🏅 Top followed"
users={topFollowed} entries={topFollowed}
columns={[ columns={[
{ {
header: 'Total followers', header: 'Total followers',