Compare commits

...

6 Commits

Author SHA1 Message Date
jahooma
624231f223 Prevent header jump on mobile. 2022-01-18 12:55:24 -06:00
jahooma
3b4910eb95 Add medal emoji 2022-01-17 21:16:31 -06:00
jahooma
275b773466 Put leaderboards side-by-side on large screens 2022-01-17 21:10:25 -06:00
jahooma
7ffd41c859 Add leaderboards menu option 2022-01-17 21:02:16 -06:00
jahooma
df2f8046b0 Filter out SG from traders. Center leaderboard. Use Nextjs Image. 2022-01-17 20:56:37 -06:00
jahooma
129ef23a4e Simple leaderboards 2022-01-17 20:28:46 -06:00
5 changed files with 146 additions and 5 deletions

View File

@ -2,12 +2,15 @@ import Link from 'next/link'
import Image from 'next/image' import Image from 'next/image'
import clsx from 'clsx' import clsx from 'clsx'
export function ManifoldLogo(props: { darkBackground?: boolean }) { export function ManifoldLogo(props: {
const { darkBackground } = props className?: string
darkBackground?: boolean
}) {
const { darkBackground, className } = props
return ( return (
<Link href="/"> <Link href="/">
<a className="flex flex-row gap-4 flex-shrink-0"> <a className={clsx('flex flex-row gap-4 flex-shrink-0', className)}>
<Image <Image
className="hover:rotate-12 transition-all" className="hover:rotate-12 transition-all"
src={darkBackground ? '/logo-white.svg' : '/logo.svg'} src={darkBackground ? '/logo-white.svg' : '/logo.svg'}

View File

@ -28,7 +28,7 @@ export function NavBar(props: {
wide ? 'max-w-6xl' : 'max-w-4xl' wide ? 'max-w-6xl' : 'max-w-4xl'
)} )}
> >
<ManifoldLogo darkBackground={darkBackground} /> <ManifoldLogo className="my-1" darkBackground={darkBackground} />
<Row className="items-center gap-6 sm:gap-8 ml-6"> <Row className="items-center gap-6 sm:gap-8 ml-6">
{(user || user === null) && ( {(user || user === null) && (

View File

@ -51,6 +51,10 @@ function getNavigationOptions(user: User, options: { mobile: boolean }) {
name: 'Your markets', name: 'Your markets',
href: `/${user.username}`, href: `/${user.username}`,
}, },
{
name: 'Leaderboards',
href: '/leaderboards',
},
{ {
name: 'Discord', name: 'Discord',
href: 'https://discord.gg/eHQBNBqXuh', href: 'https://discord.gg/eHQBNBqXuh',

View File

@ -10,6 +10,7 @@ import {
where, where,
limit, limit,
getDocs, getDocs,
orderBy,
} from 'firebase/firestore' } from 'firebase/firestore'
import { getAuth } from 'firebase/auth' import { getAuth } from 'firebase/auth'
import { ref, getStorage, uploadBytes, getDownloadURL } from 'firebase/storage' import { ref, getStorage, uploadBytes, getDownloadURL } from 'firebase/storage'
@ -20,7 +21,7 @@ import {
} from 'firebase/auth' } from 'firebase/auth'
import { User } from '../../../common/user' import { User } from '../../../common/user'
import { listenForValues } from './utils' import { getValues, listenForValues } from './utils'
export type { User } export type { User }
export const STARTING_BALANCE = 1000 export const STARTING_BALANCE = 1000
@ -127,3 +128,24 @@ export function listenForAllUsers(setUsers: (users: User[]) => void) {
const q = query(userCollection) const q = query(userCollection)
listenForValues(q, setUsers) listenForValues(q, setUsers)
} }
const topTradersQuery = query(
collection(db, 'users'),
orderBy('totalPnLCached', 'desc'),
limit(21)
)
export async function getTopTraders() {
const users = await getValues<User>(topTradersQuery)
return users.filter((user) => user.username !== 'SG').slice(0, 20)
}
const topCreatorsQuery = query(
collection(db, 'users'),
orderBy('creatorVolumeCached', 'desc'),
limit(20)
)
export function getTopCreators() {
return getValues<User>(topCreatorsQuery)
}

112
web/pages/leaderboards.tsx Normal file
View File

@ -0,0 +1,112 @@
import _ from 'lodash'
import Image from 'next/image'
import { Col } from '../components/layout/col'
import { Row } from '../components/layout/row'
import { Page } from '../components/page'
import { SiteLink } from '../components/site-link'
import { Title } from '../components/title'
import { getTopCreators, getTopTraders, User } from '../lib/firebase/users'
import { formatMoney } from '../lib/util/format'
export async function getStaticProps() {
const [topTraders, topCreators] = await Promise.all([
getTopTraders().catch((_) => {}),
getTopCreators().catch((_) => {}),
])
return {
props: {
topTraders,
topCreators,
},
revalidate: 60, // regenerate after a minute
}
}
export default function Leaderboards(props: {
topTraders: User[]
topCreators: User[]
}) {
const { topTraders, topCreators } = props
return (
<Page>
<Col className="items-center lg:flex-row gap-10">
<Leaderboard
title="🏅 Top traders"
users={topTraders}
columns={[
{
header: 'Total profit',
renderCell: (user) => formatMoney(user.totalPnLCached),
},
]}
/>
<Leaderboard
title="🏅 Top creators"
users={topCreators}
columns={[
{
header: 'Market volume',
renderCell: (user) => formatMoney(user.creatorVolumeCached),
},
]}
/>
</Col>
</Page>
)
}
function Leaderboard(props: {
title: string
users: User[]
columns: {
header: string
renderCell: (user: User) => any
}[]
}) {
const { title, users, columns } = props
return (
<div className="max-w-xl w-full px-1">
<Title text={title} />
<div className="overflow-x-auto">
<table className="table table-zebra table-compact text-gray-500 w-full">
<thead>
<tr className="p-2">
<th>#</th>
<th>Name</th>
{columns.map((column) => (
<th key={column.header}>{column.header}</th>
))}
</tr>
</thead>
<tbody>
{users.map((user, index) => (
<tr key={user.id}>
<td>{index + 1}</td>
<td>
<SiteLink className="relative" href={`/${user.username}`}>
<Row className="items-center gap-4">
<Image
className="rounded-full bg-gray-400 flex-shrink-0 ring-8 ring-gray-50"
src={user.avatarUrl}
alt=""
width={32}
height={32}
/>
<div>{user.name}</div>
</Row>
</SiteLink>
</td>
{columns.map((column) => (
<td key={column.header}>{column.renderCell(user)}</td>
))}
</tr>
))}
</tbody>
</table>
</div>
</div>
)
}