manifold/web/components/manaboard.tsx

306 lines
8.6 KiB
TypeScript
Raw Normal View History

import clsx from 'clsx'
2022-03-31 19:55:40 +00:00
import React, { useEffect, useState } from 'react'
import { ENV_CONFIG } from '../../common/envs/constants'
import { User } from '../../common/user'
2022-03-31 19:55:40 +00:00
import { formatMoney } from '../../common/util/format'
import { useUser } from '../hooks/use-user'
2022-04-01 04:49:10 +00:00
import { buyLeaderboardSlot } from '../lib/firebase/api-call'
import {
SlotData,
Transaction,
writeTransaction,
} from '../lib/firebase/transactions'
import { loadFakeBalance } from '../pages/leaderboards'
import { AddFundsButton } from './add-funds-button'
import { AmountInput } from './amount-input'
import { Avatar } from './avatar'
import { Col } from './layout/col'
import { Modal } from './layout/modal'
import { Row } from './layout/row'
import { SiteLink } from './site-link'
import { Title } from './title'
export function Manaboard(props: {
title: string
users: User[]
2022-04-01 05:16:22 +00:00
values: number[]
createdTimes: number[]
className?: string
}) {
// TODO: Ideally, highlight your own entry on the leaderboard
let { title, users, className, values, createdTimes } = props
2022-04-01 05:41:40 +00:00
const [expanded, setExpanded] = useState(false)
if (!expanded) {
users = users.slice(0, 25)
values = values.slice(0, 25)
}
return (
<div className={clsx('w-full px-1', className)}>
<Title text={title} className="!mt-0" />
{users.length === 0 ? (
<div className="ml-2 text-gray-500">None yet</div>
) : (
<div className="overflow-x-auto">
<table className="table-zebra table-compact table w-full text-gray-500">
<thead>
<tr className="p-2">
2022-03-31 22:54:40 +00:00
<th>
<div className="pl-2">#</div>
</th>
<th>Name</th>
2022-03-31 22:54:40 +00:00
<th>Value</th>
<th></th>
</tr>
</thead>
<tbody>
{users.map((user, index) => (
2022-04-01 15:06:04 +00:00
<tr key={user.id + index}>
2022-03-31 22:54:40 +00:00
<td>
<div className="pl-2">{index + 1}</div>
</td>
<td className="w-full" style={{ maxWidth: 190 }}>
<Row className="items-center gap-4">
<Avatar avatarUrl={user.avatarUrl} size={8} />
<div
className={clsx(
'truncate',
createdTimes[index]
? 'text-gray-600'
: 'text-gray-300'
)}
>
{user.name}
</div>
</Row>
</td>
<td>
2022-03-31 22:54:40 +00:00
<Row className="items-center gap-4">
2022-04-01 05:16:22 +00:00
{formatMoney(values[index])}
2022-03-31 22:54:40 +00:00
<BuySlotModal
slot={index + 1}
title={`${title}`}
holder={user}
2022-04-01 05:16:22 +00:00
value={values[index]}
createdTime={createdTimes[index]}
2022-04-01 16:24:33 +00:00
allSlots={users}
2022-03-31 22:54:40 +00:00
/>
</Row>
</td>
2022-03-31 22:54:40 +00:00
<td></td>
</tr>
))}
</tbody>
</table>
2022-04-01 05:41:40 +00:00
<button
className="btn btn-sm btn-outline m-2"
onClick={() => setExpanded(!expanded)}
>
{expanded ? 'Fewer slots' : 'More slots'}
2022-04-01 05:41:40 +00:00
</button>
</div>
)}
</div>
)
}
2022-03-31 19:55:40 +00:00
function Label(props: { children: React.ReactNode }) {
return <label className="-mb-3 text-sm">{props.children}</label>
}
export function BuySlotModal(props: {
title: string
holder: User
slot: number
2022-03-31 19:55:40 +00:00
value: number
createdTime: number
2022-04-01 16:24:33 +00:00
allSlots: User[]
}) {
2022-04-01 16:24:33 +00:00
const { slot, allSlots, holder, value, createdTime } = props
const user = useUser()
const [open, setOpen] = useState(false)
2022-03-31 19:55:40 +00:00
const [newValue, setNewValue] = useState(value)
const [message, setMessage] = useState('')
useEffect(() => {
if (user?.name) {
setMessage(user.name)
}
}, [user])
// const onBuy = async () => {
// // Feel free to change this. - James
// const slotId = `${title}-${slot}`
// await buyLeaderboardSlot({ slotId, reassessValue: newValue })
// }
2022-04-01 16:24:33 +00:00
// If the user already exists in a different slot, forbid them from buying this one
const userExists = allSlots.find(
(u, index) => u.id === user?.id && index + 1 !== slot
)
// Hm, existing leaderboard users may not be able to participate atm.
2022-04-01 16:24:33 +00:00
const errorMsg = userExists
? 'Sell your other slot first (by revaluing it to M$ 0)'
: ''
async function onBuy() {
if (user) {
// Start transactions, but don't block
const buyData = { slot, newValue, message }
const buyTxn = buyTransaction({
buyer: user,
holder,
amount: value,
buyData,
})
await Promise.all([
writeTransaction(buyTxn),
writeTransaction(taxTransaction({ holder, slot, value, createdTime })),
])
setOpen(false)
}
2022-04-01 04:49:10 +00:00
}
const fakeBalance = loadFakeBalance()
const noFundsMsg =
value > fakeBalance && holder.id !== user?.id
? `You only have ${formatMoney(fakeBalance)}!`
: ''
return (
<>
<Modal open={open} setOpen={setOpen}>
2022-03-31 19:55:40 +00:00
<Col className="gap-5 rounded-md bg-white p-6 text-gray-500">
<Title text={`Buy slot #${slot}`} className="!mt-0" />
2022-03-31 19:55:40 +00:00
<Label>Current value: {formatMoney(value)}</Label>
2022-03-31 22:54:40 +00:00
{user && (
2022-04-01 01:03:27 +00:00
<Row className="items-center gap-4 rounded-md bg-gray-100 p-2 text-sm">
<div className="pl-2">{slot}</div>
2022-03-31 22:54:40 +00:00
<Avatar avatarUrl={user.avatarUrl} size={8} />
<div className="truncate">{message}</div>
</Row>
)}
2022-03-31 19:55:40 +00:00
<Label>(Optional) set message</Label>
<input
type="text"
className="input input-bordered w-full max-w-xs"
onChange={(e) => {
setMessage(e.target.value)
}}
value={message}
/>
2022-03-31 22:54:40 +00:00
<Label>Reassess value</Label>
<AmountInput
amount={newValue}
2022-04-01 16:12:42 +00:00
onChange={(amount) =>
setNewValue(amount && amount >= 1 ? amount : 0)
}
2022-04-01 16:24:33 +00:00
error={errorMsg}
2022-03-31 22:54:40 +00:00
label={ENV_CONFIG.moneyMoniker}
/>
{noFundsMsg ? (
<div className="alert alert-error">
{noFundsMsg}{' '}
<span className="!text-gray-600">
<AddFundsButton />
</span>
</div>
) : (
<Col>
2022-04-01 16:24:33 +00:00
<button
className="btn btn-primary"
onClick={onBuy}
disabled={!!errorMsg}
>
Buy Slot ({formatMoney(value)})
</button>
<div className="mt-2 text-sm">
Additional fees: {formatMoney(newValue * 0.25)} per hour
</div>
</Col>
)}
</Col>
</Modal>
2022-03-31 22:54:40 +00:00
<button
className="btn btn-outline btn-sm normal-case"
onClick={() => setOpen(true)}
>
Buy
</button>
</>
)
}
function buyTransaction(options: {
buyer: User
holder: User
buyData: SlotData
amount: number
}): Transaction {
const { buyer, holder, buyData, amount } = options
return {
id: '',
createdTime: Date.now(),
fromId: buyer.id,
fromName: buyer.name,
fromUsername: buyer.username,
fromAvatarUrl: buyer.avatarUrl,
toId: holder.id,
toName: holder.name,
toUsername: holder.username,
toAvatarUrl: holder.avatarUrl,
amount,
category: 'BUY_LEADERBOARD_SLOT',
description: `${buyer.name} bought a slot from ${holder.name}`,
data: buyData,
}
}
function taxTransaction(options: {
holder: User
slot: number
value: number
createdTime: number
}): Transaction {
const { holder, slot, value, createdTime } = options
2022-04-01 15:59:17 +00:00
const APRIL_FOOLS_9AM_PT = 1648828800000
const elapsedMs = Date.now() - (createdTime || APRIL_FOOLS_9AM_PT)
const elapsedHours = elapsedMs / 1000 / 60 / 60
2022-04-01 06:35:46 +00:00
const tax = elapsedHours * value * 0.25
return {
id: '',
createdTime: Date.now(),
fromId: holder.id,
fromName: holder.name,
fromUsername: holder.username,
fromAvatarUrl: holder.avatarUrl,
// Send fee to Manifold Markets official account
toId: 'IPTOzEqrpkWmEzh6hwvAyY9PqFb2',
toName: 'Manifold Markets',
toUsername: 'ManifoldMarkets',
toAvatarUrl: 'https://manifold.markets/logo-bg-white.png',
amount: tax,
category: 'LEADERBOARD_TAX',
description: `${holder.name} paid M$ 10 in fees`,
data: {
slot,
},
}
}