Add discover tab of users based on markets you have bet on
This commit is contained in:
parent
ac763de16b
commit
ad6594f0bc
|
@ -3,11 +3,12 @@ import { PencilIcon } from '@heroicons/react/outline'
|
||||||
import { User } from 'common/user'
|
import { User } from 'common/user'
|
||||||
import { useEffect, useState } from 'react'
|
import { useEffect, useState } from 'react'
|
||||||
import { useFollowers, useFollows } from 'web/hooks/use-follows'
|
import { useFollowers, useFollows } from 'web/hooks/use-follows'
|
||||||
import { prefetchUsers } from 'web/hooks/use-user'
|
import { prefetchUsers, useUser } from 'web/hooks/use-user'
|
||||||
import { FollowList } from './follow-list'
|
import { FollowList } from './follow-list'
|
||||||
import { Col } from './layout/col'
|
import { Col } from './layout/col'
|
||||||
import { Modal } from './layout/modal'
|
import { Modal } from './layout/modal'
|
||||||
import { Tabs } from './layout/tabs'
|
import { Tabs } from './layout/tabs'
|
||||||
|
import { useDiscoverUsers } from 'web/hooks/use-users'
|
||||||
|
|
||||||
export function FollowingButton(props: { user: User }) {
|
export function FollowingButton(props: { user: User }) {
|
||||||
const { user } = props
|
const { user } = props
|
||||||
|
@ -114,6 +115,13 @@ function FollowingFollowersDialog(props: {
|
||||||
prefetchUsers([...followingIds, ...followerIds])
|
prefetchUsers([...followingIds, ...followerIds])
|
||||||
}, [followingIds, followerIds])
|
}, [followingIds, followerIds])
|
||||||
|
|
||||||
|
const currentUser = useUser()
|
||||||
|
|
||||||
|
const discoverUserIds = useDiscoverUsers()
|
||||||
|
useEffect(() => {
|
||||||
|
prefetchUsers(discoverUserIds)
|
||||||
|
}, [discoverUserIds])
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Modal open={isOpen} setOpen={setIsOpen}>
|
<Modal open={isOpen} setOpen={setIsOpen}>
|
||||||
<Col className="rounded bg-white p-6">
|
<Col className="rounded bg-white p-6">
|
||||||
|
@ -129,6 +137,14 @@ function FollowingFollowersDialog(props: {
|
||||||
title: 'Followers',
|
title: 'Followers',
|
||||||
content: <FollowList userIds={followerIds} />,
|
content: <FollowList userIds={followerIds} />,
|
||||||
},
|
},
|
||||||
|
...(currentUser
|
||||||
|
? [
|
||||||
|
{
|
||||||
|
title: 'Discover',
|
||||||
|
content: <FollowList userIds={discoverUserIds} />,
|
||||||
|
},
|
||||||
|
]
|
||||||
|
: []),
|
||||||
]}
|
]}
|
||||||
defaultIndex={defaultTab === 'following' ? 0 : 1}
|
defaultIndex={defaultTab === 'following' ? 0 : 1}
|
||||||
/>
|
/>
|
||||||
|
|
|
@ -5,6 +5,10 @@ import {
|
||||||
listenForAllUsers,
|
listenForAllUsers,
|
||||||
listenForPrivateUsers,
|
listenForPrivateUsers,
|
||||||
} from 'web/lib/firebase/users'
|
} from 'web/lib/firebase/users'
|
||||||
|
import { useUser } from './use-user'
|
||||||
|
import { groupBy, sortBy, difference } from 'lodash'
|
||||||
|
import { getContractsOfUserBets } from 'web/lib/firebase/bets'
|
||||||
|
import { useFollows } from './use-follows'
|
||||||
|
|
||||||
export const useUsers = () => {
|
export const useUsers = () => {
|
||||||
const [users, setUsers] = useState<User[]>([])
|
const [users, setUsers] = useState<User[]>([])
|
||||||
|
@ -37,3 +41,29 @@ export const usePrivateUsers = () => {
|
||||||
|
|
||||||
return users
|
return users
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export const useDiscoverUsers = () => {
|
||||||
|
const user = useUser()
|
||||||
|
|
||||||
|
const [discoverUserIds, setDiscoverUserIds] = useState<string[]>([])
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (user)
|
||||||
|
getContractsOfUserBets(user.id).then((contracts) => {
|
||||||
|
const creatorCounts = Object.entries(
|
||||||
|
groupBy(contracts, 'creatorId')
|
||||||
|
).map(([id, contracts]) => [id, contracts.length] as const)
|
||||||
|
|
||||||
|
const topCreatorIds = sortBy(creatorCounts, ([_, i]) => i)
|
||||||
|
.map(([id]) => id)
|
||||||
|
.reverse()
|
||||||
|
|
||||||
|
setDiscoverUserIds(topCreatorIds)
|
||||||
|
})
|
||||||
|
}, [user])
|
||||||
|
|
||||||
|
const followedUserIds = useFollows(user?.id)
|
||||||
|
const nonSuggestions = [user?.id ?? '', ...(followedUserIds ?? [])]
|
||||||
|
|
||||||
|
return difference(discoverUserIds, nonSuggestions).slice(0, 50)
|
||||||
|
}
|
||||||
|
|
|
@ -5,12 +5,14 @@ import {
|
||||||
where,
|
where,
|
||||||
orderBy,
|
orderBy,
|
||||||
} from 'firebase/firestore'
|
} from 'firebase/firestore'
|
||||||
import { range } from 'lodash'
|
import { range, uniq } from 'lodash'
|
||||||
|
|
||||||
import { db } from './init'
|
import { db } from './init'
|
||||||
import { Bet } from 'common/bet'
|
import { Bet } from 'common/bet'
|
||||||
import { Contract } from 'common/contract'
|
import { Contract } from 'common/contract'
|
||||||
import { getValues, listenForValues } from './utils'
|
import { getValues, listenForValues } from './utils'
|
||||||
|
import { getContractFromId } from './contracts'
|
||||||
|
import { filterDefined } from 'common/util/array'
|
||||||
export type { Bet }
|
export type { Bet }
|
||||||
|
|
||||||
function getBetsCollection(contractId: string) {
|
function getBetsCollection(contractId: string) {
|
||||||
|
@ -76,6 +78,15 @@ export async function getUserBets(
|
||||||
.catch((reason) => reason)
|
.catch((reason) => reason)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export async function getContractsOfUserBets(userId: string) {
|
||||||
|
const bets: Bet[] = await getUserBets(userId, { includeRedemptions: false })
|
||||||
|
const contractIds = uniq(bets.map((bet) => bet.contractId))
|
||||||
|
const contracts = await Promise.all(
|
||||||
|
contractIds.map((contractId) => getContractFromId(contractId))
|
||||||
|
)
|
||||||
|
return filterDefined(contracts)
|
||||||
|
}
|
||||||
|
|
||||||
export function listenForUserBets(
|
export function listenForUserBets(
|
||||||
userId: string,
|
userId: string,
|
||||||
setBets: (bets: Bet[]) => void,
|
setBets: (bets: Bet[]) => void,
|
||||||
|
|
Loading…
Reference in New Issue
Block a user