import clsx from 'clsx' import React, { useEffect, useState } from 'react' import { ENV_CONFIG } from '../../common/envs/constants' import { User } from '../../common/user' import { formatMoney } from '../../common/util/format' import { useUser } from '../hooks/use-user' 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[] values: number[] createdTimes: number[] className?: string }) { // TODO: Ideally, highlight your own entry on the leaderboard let { title, users, className, values, createdTimes } = props const [expanded, setExpanded] = useState(false) if (!expanded) { users = users.slice(0, 25) values = values.slice(0, 25) } return (
{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"> <th> <div className="pl-2">#</div> </th> <th>Name</th> <th>Value</th> <th></th> </tr> </thead> <tbody> {users.map((user, index) => ( <tr key={user.id + index}> <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> <Row className="items-center gap-4"> {formatMoney(values[index])} <BuySlotModal slot={index + 1} title={`${title}`} holder={user} value={values[index]} createdTime={createdTimes[index]} allSlots={users} /> </Row> </td> <td></td> </tr> ))} </tbody> </table> <button className="btn btn-sm btn-outline m-2" onClick={() => setExpanded(!expanded)} > {expanded ? 'Fewer slots' : 'More slots'} </button> </div> )} </div> ) } 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 value: number createdTime: number allSlots: User[] }) { const { slot, allSlots, holder, value, createdTime } = props const user = useUser() const [open, setOpen] = useState(false) 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 }) // } // 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. 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) } } const fakeBalance = loadFakeBalance() const noFundsMsg = value > fakeBalance && holder.id !== user?.id ? `You only have ${formatMoney(fakeBalance)}!` : '' return ( <> <Modal open={open} setOpen={setOpen}> <Col className="gap-5 rounded-md bg-white p-6 text-gray-500"> <Title text={`Buy slot #${slot}`} className="!mt-0" /> <Label>Current value: {formatMoney(value)}</Label> {user && ( <Row className="items-center gap-4 rounded-md bg-gray-100 p-2 text-sm"> <div className="pl-2">{slot}</div> <Avatar avatarUrl={user.avatarUrl} size={8} /> <div className="truncate">{message}</div> </Row> )} <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} /> <Label>Reassess value</Label> <AmountInput amount={newValue} onChange={(amount) => setNewValue(amount && amount >= 1 ? amount : 0) } error={errorMsg} label={ENV_CONFIG.moneyMoniker} /> {noFundsMsg ? ( <div className="alert alert-error"> {noFundsMsg}{' '} <span className="!text-gray-600"> <AddFundsButton /> </span> </div> ) : ( <Col> <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> <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 const APRIL_FOOLS_9AM_PT = 1648828800000 const elapsedMs = Date.now() - (createdTime || APRIL_FOOLS_9AM_PT) const elapsedHours = elapsedMs / 1000 / 60 / 60 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, }, } }