import React, { useState } from 'react' import Link from 'next/link' import { useRouter } from 'next/router' import { sortBy, take } from 'lodash' import { toast } from 'react-hot-toast' import { Group, GROUP_CHAT_SLUG } from 'common/group' import { Page } from 'web/components/page' import { listAllBets } from 'web/lib/firebase/bets' import { Contract, listContractsByGroupSlug } from 'web/lib/firebase/contracts' import { addContractToGroup, getGroupBySlug, groupPath, joinGroup, listMemberIds, updateGroup, } from 'web/lib/firebase/groups' import { Row } from 'web/components/layout/row' import { firebaseLogin, getUser, User } from 'web/lib/firebase/users' import { Col } from 'web/components/layout/col' import { useUser } from 'web/hooks/use-user' import { useGroup, useGroupContractIds, useMemberIds, } from 'web/hooks/use-group' import { scoreCreators, scoreTraders } from 'common/scoring' import { Leaderboard } from 'web/components/leaderboard' import { formatMoney } from 'common/util/format' import { EditGroupButton } from 'web/components/groups/edit-group-button' import Custom404 from '../../404' import { SEO } from 'web/components/SEO' import { Linkify } from 'web/components/linkify' import { fromPropz, usePropz } from 'web/hooks/use-propz' import { Tabs } from 'web/components/layout/tabs' import { LoadingIndicator } from 'web/components/loading-indicator' import { Modal } from 'web/components/layout/modal' import { ChoicesToggleGroup } from 'web/components/choices-toggle-group' import { ContractSearch } from 'web/components/contract-search' import { JoinOrLeaveGroupButton } from 'web/components/groups/groups-button' import { CopyLinkButton } from 'web/components/copy-link-button' import { ENV_CONFIG } from 'common/envs/constants' import { useSaveReferral } from 'web/hooks/use-save-referral' import { Button } from 'web/components/button' import { listAllCommentsOnGroup } from 'web/lib/firebase/comments' import { GroupComment } from 'common/comment' import { REFERRAL_AMOUNT } from 'common/economy' import { UserLink } from 'web/components/user-link' import { GroupAboutPost } from 'web/components/groups/group-about-post' import { getPost } from 'web/lib/firebase/posts' import { Post } from 'common/post' import { Spacer } from 'web/components/layout/spacer' import { usePost } from 'web/hooks/use-post' import { useAdmin } from 'web/hooks/use-admin' import { track } from '@amplitude/analytics-browser' export const getStaticProps = fromPropz(getStaticPropz) export async function getStaticPropz(props: { params: { slugs: string[] } }) { const { slugs } = props.params const group = await getGroupBySlug(slugs[0]) const memberIds = group && (await listMemberIds(group)) const creatorPromise = group ? getUser(group.creatorId) : null const contracts = (group && (await listContractsByGroupSlug(group.slug))) ?? [] const now = Date.now() const suggestedFilter = contracts.filter((c) => (c.closeTime ?? 0) > now).length < 5 ? 'all' : 'open' const aboutPost = group && group.aboutPostId != null && (await getPost(group.aboutPostId)) const bets = await Promise.all( contracts.map((contract: Contract) => listAllBets(contract.id)) ) const messages = group && (await listAllCommentsOnGroup(group.id)) const creatorScores = scoreCreators(contracts) const traderScores = scoreTraders(contracts, bets) const [topCreators, topTraders] = (memberIds && [ await toTopUsers(creatorScores, memberIds), await toTopUsers(traderScores, memberIds), ]) ?? [] const creator = await creatorPromise // Only count unresolved markets const contractsCount = contracts.filter((c) => !c.isResolved).length return { props: { contractsCount, group, memberIds, creator, traderScores, topTraders, creatorScores, topCreators, messages, aboutPost, suggestedFilter, }, revalidate: 60, // regenerate after a minute } } function toTopUsers( userScores: { [userId: string]: number }, userIds: string[] ) { const topUserPairs = take( sortBy(Object.entries(userScores), ([_, score]) => -1 * score), 10 ).filter(([_, score]) => score >= 0.5) const topUserIds = topUserPairs .map(([userId]) => userIds.filter((uid) => uid === userId)[0]) .filter((userId) => userId != null) as string[] return Promise.all(topUserIds.map(getUser)) } export async function getStaticPaths() { return { paths: [], fallback: 'blocking' } } const groupSubpages = [ undefined, GROUP_CHAT_SLUG, 'markets', 'leaderboards', 'about', ] as const export default function GroupPage(props: { contractsCount: number group: Group | null memberIds: string[] creator: User traderScores: { [userId: string]: number } topTraders: User[] creatorScores: { [userId: string]: number } topCreators: User[] messages: GroupComment[] aboutPost: Post suggestedFilter: 'open' | 'all' }) { props = usePropz(props, getStaticPropz) ?? { contractsCount: 0, group: null, memberIds: [], creator: null, traderScores: {}, topTraders: [], creatorScores: {}, topCreators: [], messages: [], suggestedFilter: 'open', } const { contractsCount, creator, traderScores, topTraders, creatorScores, topCreators, suggestedFilter, } = props const router = useRouter() const { slugs } = router.query as { slugs: string[] } const page = slugs?.[1] as typeof groupSubpages[number] const group = useGroup(props.group?.id) ?? props.group const aboutPost = usePost(props.aboutPost?.id) ?? props.aboutPost const user = useUser() const isAdmin = useAdmin() const memberIds = useMemberIds(group?.id ?? null) ?? props.memberIds useSaveReferral(user, { defaultReferrerUsername: creator.username, groupId: group?.id, }) if (group === null || !groupSubpages.includes(page) || slugs[2]) { return } const isCreator = user && group && user.id === group.creatorId const isMember = user && memberIds.includes(user.id) const leaderboard = ( ) const aboutTab = ( {(group.aboutPostId != null || isCreator || isAdmin) && ( )} ) const questionsTab = ( ) const tabs = [ { badge: `${contractsCount}`, title: 'Markets', content: questionsTab, href: groupPath(group.slug, 'markets'), }, { title: 'Leaderboards', content: leaderboard, href: groupPath(group.slug, 'leaderboards'), }, { title: 'About', content: aboutTab, href: groupPath(group.slug, 'about'), }, ] const tabIndex = tabs .map((t) => t.title.toLowerCase()) .indexOf(page ?? 'markets') return (
{group.name}
0 ? tabIndex : 0} tabs={tabs} />
) } function JoinOrAddQuestionsButtons(props: { group: Group user: User | null | undefined isMember: boolean }) { const { group, user, isMember } = props return user && isMember ? ( ) : group.anyoneCanJoin ? ( ) : null } function GroupOverview(props: { group: Group creator: User user: User | null | undefined isCreator: boolean memberIds: string[] }) { const { group, creator, isCreator, user, memberIds } = props const anyoneCanJoinChoices: { [key: string]: string } = { Closed: 'false', Open: 'true', } const [anyoneCanJoin, setAnyoneCanJoin] = useState(group.anyoneCanJoin) function updateAnyoneCanJoin(newVal: boolean) { if (group.anyoneCanJoin == newVal || !isCreator) return setAnyoneCanJoin(newVal) toast.promise(updateGroup(group, { ...group, anyoneCanJoin: newVal }), { loading: 'Updating group...', success: 'Updated group!', error: "Couldn't update group", }) } const postFix = user ? '?referrer=' + user.username : '' const shareUrl = `https://${ENV_CONFIG.domain}${groupPath( group.slug )}${postFix}` const isMember = user ? memberIds.includes(user.id) : false return ( <>
Created by
{isCreator ? ( ) : ( user && ( ) )}
Membership {user && user.id === creator.id ? ( updateAnyoneCanJoin(choice.toString() === 'true') } toggleClassName={'h-10'} className={'ml-2'} /> ) : ( {anyoneCanJoin ? 'Open to all' : 'Closed (by invite only)'} )} {anyoneCanJoin && user && (
Invite
Invite a friend to this group and get M${REFERRAL_AMOUNT} if they sign up!
)} ) } function SortedLeaderboard(props: { users: User[] scoreFunction: (user: User) => number title: string header: string maxToShow?: number }) { const { users, scoreFunction, title, header, maxToShow } = props return ( formatMoney(scoreFunction(user)) }, ]} maxToShow={maxToShow} /> ) } function GroupLeaderboards(props: { traderScores: { [userId: string]: number } creatorScores: { [userId: string]: number } topTraders: User[] topCreators: User[] memberIds: string[] user: User | null | undefined }) { const { traderScores, creatorScores, memberIds, topTraders, topCreators } = props const maxToShow = 50 // Consider hiding M$0 // If it's just one member (curator), show all bettors, otherwise just show members return (
{memberIds.length > 1 ? ( <> traderScores[user.id] ?? 0} title="🏅 Top traders" header="Profit" maxToShow={maxToShow} /> creatorScores[user.id] ?? 0} title="🏅 Top creators" header="Market volume" maxToShow={maxToShow} /> ) : ( <> formatMoney(traderScores[user.id] ?? 0), }, ]} maxToShow={maxToShow} /> formatMoney(creatorScores[user.id] ?? 0), }, ]} maxToShow={maxToShow} /> )}
) } function AddContractButton(props: { group: Group; user: User }) { const { group, user } = props const [open, setOpen] = useState(false) const [contracts, setContracts] = useState([]) const [loading, setLoading] = useState(false) const groupContractIds = useGroupContractIds(group.id) async function addContractToCurrentGroup(contract: Contract) { if (contracts.map((c) => c.id).includes(contract.id)) { setContracts(contracts.filter((c) => c.id !== contract.id)) } else setContracts([...contracts, contract]) } async function doneAddingContracts() { Promise.all( contracts.map(async (contract) => { setLoading(true) await addContractToGroup(group, contract, user.id) }) ).then(() => { setLoading(false) setOpen(false) setContracts([]) }) } return ( <>
Add markets
Add pre-existing markets to this group, or{' '} create a new one .
{contracts.length > 0 && ( {!loading ? ( ) : ( )} )}
c.id), highlightClassName: '!bg-indigo-100 border-indigo-100 border-2', }} />
) } function JoinGroupButton(props: { group: Group user: User | null | undefined }) { const { group, user } = props const follow = async () => { track('join group') const userId = user ? user.id : (await firebaseLogin()).user.uid toast.promise(joinGroup(group, userId), { loading: 'Following group...', success: 'Followed', error: "Couldn't follow group, try again?", }) } return (
) }