Algo feed (#77)
* Implement algo feed * Remove 'See more...' from feed items * Fix problem with useUpdatedContracts. * Tweak some params
This commit is contained in:
parent
7c11df6147
commit
ec49a73c74
94
common/recommended-contracts.ts
Normal file
94
common/recommended-contracts.ts
Normal file
|
@ -0,0 +1,94 @@
|
||||||
|
import _ from 'lodash'
|
||||||
|
import { Contract } from './contract'
|
||||||
|
import { filterDefined } from './util/array'
|
||||||
|
import { addObjects } from './util/object'
|
||||||
|
|
||||||
|
export const getRecommendedContracts = (
|
||||||
|
contractsById: { [contractId: string]: Contract },
|
||||||
|
yourBetOnContractIds: string[]
|
||||||
|
) => {
|
||||||
|
const contracts = Object.values(contractsById)
|
||||||
|
const yourContracts = filterDefined(
|
||||||
|
yourBetOnContractIds.map((contractId) => contractsById[contractId])
|
||||||
|
)
|
||||||
|
|
||||||
|
const yourContractIds = new Set(yourContracts.map((c) => c.id))
|
||||||
|
const notYourContracts = contracts.filter((c) => !yourContractIds.has(c.id))
|
||||||
|
|
||||||
|
const yourWordFrequency = contractsToWordFrequency(yourContracts)
|
||||||
|
const otherWordFrequency = contractsToWordFrequency(notYourContracts)
|
||||||
|
const words = _.union(
|
||||||
|
Object.keys(yourWordFrequency),
|
||||||
|
Object.keys(otherWordFrequency)
|
||||||
|
)
|
||||||
|
|
||||||
|
const yourWeightedFrequency = _.fromPairs(
|
||||||
|
_.map(words, (word) => {
|
||||||
|
const [yourFreq, otherFreq] = [
|
||||||
|
yourWordFrequency[word] ?? 0,
|
||||||
|
otherWordFrequency[word] ?? 0,
|
||||||
|
]
|
||||||
|
|
||||||
|
const score = yourFreq / (yourFreq + otherFreq + 0.0001)
|
||||||
|
|
||||||
|
return [word, score]
|
||||||
|
})
|
||||||
|
)
|
||||||
|
|
||||||
|
// console.log(
|
||||||
|
// 'your weighted frequency',
|
||||||
|
// _.sortBy(_.toPairs(yourWeightedFrequency), ([, freq]) => -freq)
|
||||||
|
// )
|
||||||
|
|
||||||
|
const scoredContracts = contracts.map((contract) => {
|
||||||
|
const wordFrequency = contractToWordFrequency(contract)
|
||||||
|
|
||||||
|
const score = _.sumBy(Object.keys(wordFrequency), (word) => {
|
||||||
|
const wordFreq = wordFrequency[word] ?? 0
|
||||||
|
const weight = yourWeightedFrequency[word] ?? 0
|
||||||
|
return wordFreq * weight
|
||||||
|
})
|
||||||
|
|
||||||
|
return {
|
||||||
|
contract,
|
||||||
|
score,
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
return _.sortBy(scoredContracts, (scored) => -scored.score).map(
|
||||||
|
(scored) => scored.contract
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
const contractToText = (contract: Contract) => {
|
||||||
|
const { description, question, tags, creatorUsername } = contract
|
||||||
|
return `${creatorUsername} ${question} ${tags.join(' ')} ${description}`
|
||||||
|
}
|
||||||
|
|
||||||
|
const getWordsCount = (text: string) => {
|
||||||
|
const normalizedText = text.replace(/[^a-zA-Z]/g, ' ').toLowerCase()
|
||||||
|
const words = normalizedText.split(' ').filter((word) => word)
|
||||||
|
|
||||||
|
const counts: { [word: string]: number } = {}
|
||||||
|
for (const word of words) {
|
||||||
|
if (counts[word]) counts[word]++
|
||||||
|
else counts[word] = 1
|
||||||
|
}
|
||||||
|
return counts
|
||||||
|
}
|
||||||
|
|
||||||
|
const toFrequency = (counts: { [word: string]: number }) => {
|
||||||
|
const total = _.sum(Object.values(counts))
|
||||||
|
return _.mapValues(counts, (count) => count / total)
|
||||||
|
}
|
||||||
|
|
||||||
|
const contractToWordFrequency = (contract: Contract) =>
|
||||||
|
toFrequency(getWordsCount(contractToText(contract)))
|
||||||
|
|
||||||
|
const contractsToWordFrequency = (contracts: Contract[]) => {
|
||||||
|
const frequencySum = contracts
|
||||||
|
.map(contractToWordFrequency)
|
||||||
|
.reduce(addObjects, {})
|
||||||
|
|
||||||
|
return toFrequency(frequencySum)
|
||||||
|
}
|
6
common/util/math.ts
Normal file
6
common/util/math.ts
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
export const logInterpolation = (min: number, max: number, value: number) => {
|
||||||
|
if (value <= min) return 0
|
||||||
|
if (value >= max) return 1
|
||||||
|
|
||||||
|
return Math.log(value - min + 1) / Math.log(max - min + 1)
|
||||||
|
}
|
|
@ -340,14 +340,6 @@ export function FeedQuestion(props: {
|
||||||
>
|
>
|
||||||
{question}
|
{question}
|
||||||
</SiteLink>
|
</SiteLink>
|
||||||
{!showDescription && (
|
|
||||||
<SiteLink
|
|
||||||
href={contractPath(contract)}
|
|
||||||
className="relative top-4 self-end text-sm sm:self-start"
|
|
||||||
>
|
|
||||||
<div className="pb-1.5 text-gray-400">See more...</div>
|
|
||||||
</SiteLink>
|
|
||||||
)}
|
|
||||||
</Col>
|
</Col>
|
||||||
{(isBinary || resolution) && (
|
{(isBinary || resolution) && (
|
||||||
<ResolutionOrChance className="items-center" contract={contract} />
|
<ResolutionOrChance className="items-center" contract={contract} />
|
||||||
|
|
|
@ -59,7 +59,10 @@ export function findActiveContracts(
|
||||||
}
|
}
|
||||||
|
|
||||||
let activeContracts = allContracts.filter(
|
let activeContracts = allContracts.filter(
|
||||||
(contract) => contract.visibility === 'public' && !contract.isResolved
|
(contract) =>
|
||||||
|
contract.visibility === 'public' &&
|
||||||
|
!contract.isResolved &&
|
||||||
|
(contract.closeTime ?? Infinity) > Date.now()
|
||||||
)
|
)
|
||||||
activeContracts = _.sortBy(
|
activeContracts = _.sortBy(
|
||||||
activeContracts,
|
activeContracts,
|
||||||
|
|
172
web/hooks/use-algo-feed.ts
Normal file
172
web/hooks/use-algo-feed.ts
Normal file
|
@ -0,0 +1,172 @@
|
||||||
|
import _ from 'lodash'
|
||||||
|
import { useState, useEffect, useMemo } from 'react'
|
||||||
|
import { Bet } from '../../common/bet'
|
||||||
|
import { Comment } from '../../common/comment'
|
||||||
|
import { Contract } from '../../common/contract'
|
||||||
|
import { User } from '../../common/user'
|
||||||
|
import { logInterpolation } from '../../common/util/math'
|
||||||
|
import { getRecommendedContracts } from '../../common/recommended-contracts'
|
||||||
|
import { useSeenContracts } from './use-seen-contracts'
|
||||||
|
import { useGetUserBetContractIds, useUserBetContracts } from './use-user-bets'
|
||||||
|
|
||||||
|
const MAX_FEED_CONTRACTS = 75
|
||||||
|
|
||||||
|
export const useAlgoFeed = (
|
||||||
|
user: User | null | undefined,
|
||||||
|
contracts: Contract[] | undefined,
|
||||||
|
recentBets: Bet[] | undefined,
|
||||||
|
recentComments: Comment[] | undefined
|
||||||
|
) => {
|
||||||
|
const initialContracts = useMemo(() => contracts, [!!contracts])
|
||||||
|
const initialBets = useMemo(() => recentBets, [!!recentBets])
|
||||||
|
const initialComments = useMemo(() => recentComments, [!!recentComments])
|
||||||
|
|
||||||
|
const yourBetContractIds = useGetUserBetContractIds(user?.id)
|
||||||
|
// Update user bet contracts in local storage.
|
||||||
|
useUserBetContracts(user?.id)
|
||||||
|
|
||||||
|
const seenContracts = useSeenContracts()
|
||||||
|
|
||||||
|
const [algoFeed, setAlgoFeed] = useState<Contract[]>([])
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (initialContracts && initialBets && initialComments) {
|
||||||
|
const eligibleContracts = initialContracts.filter(
|
||||||
|
(c) => !c.isResolved && (c.closeTime ?? Infinity) > Date.now()
|
||||||
|
)
|
||||||
|
const contracts = getAlgoFeed(
|
||||||
|
eligibleContracts,
|
||||||
|
initialBets,
|
||||||
|
initialComments,
|
||||||
|
yourBetContractIds,
|
||||||
|
seenContracts
|
||||||
|
)
|
||||||
|
setAlgoFeed(contracts)
|
||||||
|
}
|
||||||
|
}, [
|
||||||
|
initialBets,
|
||||||
|
initialComments,
|
||||||
|
initialContracts,
|
||||||
|
seenContracts,
|
||||||
|
yourBetContractIds,
|
||||||
|
])
|
||||||
|
|
||||||
|
return algoFeed
|
||||||
|
}
|
||||||
|
|
||||||
|
const getAlgoFeed = (
|
||||||
|
contracts: Contract[],
|
||||||
|
recentBets: Bet[],
|
||||||
|
recentComments: Comment[],
|
||||||
|
yourBetContractIds: string[],
|
||||||
|
seenContracts: { [contractId: string]: number }
|
||||||
|
) => {
|
||||||
|
const contractsById = _.keyBy(contracts, (c) => c.id)
|
||||||
|
|
||||||
|
const recommended = getRecommendedContracts(contractsById, yourBetContractIds)
|
||||||
|
const confidence = logInterpolation(0, 100, yourBetContractIds.length)
|
||||||
|
const recommendedScores = _.fromPairs(
|
||||||
|
recommended.map((c, index) => {
|
||||||
|
const score = 1 - index / recommended.length
|
||||||
|
const withConfidence = score * confidence + (1 - confidence)
|
||||||
|
return [c.id, withConfidence] as [string, number]
|
||||||
|
})
|
||||||
|
)
|
||||||
|
|
||||||
|
const seenScores = _.fromPairs(
|
||||||
|
contracts.map(
|
||||||
|
(c) => [c.id, getSeenContractsScore(c, seenContracts)] as [string, number]
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
const activityScores = getContractsActivityScores(
|
||||||
|
contracts,
|
||||||
|
recentComments,
|
||||||
|
recentBets,
|
||||||
|
seenContracts
|
||||||
|
)
|
||||||
|
|
||||||
|
const combinedScores = contracts.map((contract) => {
|
||||||
|
const score =
|
||||||
|
(recommendedScores[contract.id] ?? 0) *
|
||||||
|
(seenScores[contract.id] ?? 0) *
|
||||||
|
(activityScores[contract.id] ?? 0)
|
||||||
|
return { contract, score }
|
||||||
|
})
|
||||||
|
|
||||||
|
const sorted = _.sortBy(combinedScores, (c) => -c.score)
|
||||||
|
return sorted.map((c) => c.contract).slice(0, MAX_FEED_CONTRACTS)
|
||||||
|
}
|
||||||
|
|
||||||
|
function getContractsActivityScores(
|
||||||
|
contracts: Contract[],
|
||||||
|
recentComments: Comment[],
|
||||||
|
recentBets: Bet[],
|
||||||
|
seenContracts: { [contractId: string]: number }
|
||||||
|
) {
|
||||||
|
const contractBets = _.groupBy(recentBets, (bet) => bet.contractId)
|
||||||
|
const contractMostRecentBet = _.mapValues(
|
||||||
|
contractBets,
|
||||||
|
(bets) => _.maxBy(bets, (bet) => bet.createdTime) as Bet
|
||||||
|
)
|
||||||
|
|
||||||
|
const contractComments = _.groupBy(
|
||||||
|
recentComments,
|
||||||
|
(comment) => comment.contractId
|
||||||
|
)
|
||||||
|
const contractMostRecentComment = _.mapValues(
|
||||||
|
contractComments,
|
||||||
|
(comments) => _.maxBy(comments, (c) => c.createdTime) as Comment
|
||||||
|
)
|
||||||
|
|
||||||
|
const scoredContracts = contracts.map((contract) => {
|
||||||
|
const seenTime = seenContracts[contract.id]
|
||||||
|
const lastCommentTime = contractMostRecentComment[contract.id]?.createdTime
|
||||||
|
const hasNewComments =
|
||||||
|
!seenTime || (lastCommentTime && lastCommentTime > seenTime)
|
||||||
|
const newCommentScore = hasNewComments ? 1 : 0.75
|
||||||
|
|
||||||
|
const commentCount = contractComments[contract.id]?.length ?? 0
|
||||||
|
const betCount = contractBets[contract.id]?.length ?? 0
|
||||||
|
const activtyCount = betCount + commentCount * 5
|
||||||
|
const activityCountScore =
|
||||||
|
0.5 + 0.5 * logInterpolation(0, 200, activtyCount)
|
||||||
|
|
||||||
|
const lastBetTime = contractMostRecentBet[contract.id]?.createdTime
|
||||||
|
const timeSinceLastBet = !lastBetTime
|
||||||
|
? contract.createdTime
|
||||||
|
: Date.now() - lastBetTime
|
||||||
|
const daysAgo = timeSinceLastBet / oneDayMs
|
||||||
|
const timeAgoScore = 1 - logInterpolation(0, 3, daysAgo)
|
||||||
|
|
||||||
|
const score = newCommentScore * activityCountScore * timeAgoScore
|
||||||
|
|
||||||
|
// Map score to [0.5, 1] since no recent activty is not a deal breaker.
|
||||||
|
const mappedScore = 0.5 + score / 2
|
||||||
|
return [contract.id, mappedScore] as [string, number]
|
||||||
|
})
|
||||||
|
|
||||||
|
return _.fromPairs(scoredContracts)
|
||||||
|
}
|
||||||
|
|
||||||
|
const oneDayMs = 24 * 60 * 60 * 1000
|
||||||
|
|
||||||
|
function getSeenContractsScore(
|
||||||
|
contract: Contract,
|
||||||
|
seenContracts: { [contractId: string]: number }
|
||||||
|
) {
|
||||||
|
const lastSeen = seenContracts[contract.id]
|
||||||
|
if (lastSeen === undefined) {
|
||||||
|
return 1
|
||||||
|
}
|
||||||
|
|
||||||
|
const daysAgo = (Date.now() - lastSeen) / oneDayMs
|
||||||
|
|
||||||
|
if (daysAgo < 0.5) {
|
||||||
|
const frac = logInterpolation(0, 0.5, daysAgo)
|
||||||
|
return 0.5 * frac
|
||||||
|
}
|
||||||
|
|
||||||
|
const frac = logInterpolation(0.5, 14, daysAgo)
|
||||||
|
return 0.5 + 0.5 * frac
|
||||||
|
}
|
|
@ -2,7 +2,6 @@ import { useEffect, useState } from 'react'
|
||||||
import { Contract } from '../../common/contract'
|
import { Contract } from '../../common/contract'
|
||||||
import {
|
import {
|
||||||
Bet,
|
Bet,
|
||||||
getRecentBets,
|
|
||||||
listenForBets,
|
listenForBets,
|
||||||
listenForRecentBets,
|
listenForRecentBets,
|
||||||
withoutAnteBets,
|
withoutAnteBets,
|
||||||
|
@ -37,11 +36,3 @@ export const useRecentBets = () => {
|
||||||
useEffect(() => listenForRecentBets(setRecentBets), [])
|
useEffect(() => listenForRecentBets(setRecentBets), [])
|
||||||
return recentBets
|
return recentBets
|
||||||
}
|
}
|
||||||
|
|
||||||
export const useGetRecentBets = () => {
|
|
||||||
const [recentBets, setRecentBets] = useState<Bet[] | undefined>()
|
|
||||||
useEffect(() => {
|
|
||||||
getRecentBets().then(setRecentBets)
|
|
||||||
}, [])
|
|
||||||
return recentBets
|
|
||||||
}
|
|
||||||
|
|
|
@ -39,19 +39,6 @@ export const useInactiveContracts = () => {
|
||||||
return contracts
|
return contracts
|
||||||
}
|
}
|
||||||
|
|
||||||
export const useUpdatedContracts = (initialContracts: Contract[]) => {
|
|
||||||
const [contracts, setContracts] = useState(initialContracts)
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
return listenForContracts((newContracts) => {
|
|
||||||
const contractMap = _.fromPairs(newContracts.map((c) => [c.id, c]))
|
|
||||||
setContracts(initialContracts.map((c) => contractMap[c.id]))
|
|
||||||
})
|
|
||||||
}, [initialContracts])
|
|
||||||
|
|
||||||
return contracts
|
|
||||||
}
|
|
||||||
|
|
||||||
export const useTaggedContracts = (tags: string[] | undefined) => {
|
export const useTaggedContracts = (tags: string[] | undefined) => {
|
||||||
const [contracts, setContracts] = useState<Contract[] | undefined>(
|
const [contracts, setContracts] = useState<Contract[] | undefined>(
|
||||||
tags && tags.length === 0 ? [] : undefined
|
tags && tags.length === 0 ? [] : undefined
|
||||||
|
|
|
@ -1,150 +0,0 @@
|
||||||
import _ from 'lodash'
|
|
||||||
import { useMemo, useRef } from 'react'
|
|
||||||
|
|
||||||
import { Fold } from '../../common/fold'
|
|
||||||
import { User } from '../../common/user'
|
|
||||||
import { filterDefined } from '../../common/util/array'
|
|
||||||
import { findActiveContracts } from '../components/feed/find-active-contracts'
|
|
||||||
import { Bet } from '../lib/firebase/bets'
|
|
||||||
import { Comment, getRecentComments } from '../lib/firebase/comments'
|
|
||||||
import { Contract, getActiveContracts } from '../lib/firebase/contracts'
|
|
||||||
import { listAllFolds } from '../lib/firebase/folds'
|
|
||||||
import { useInactiveContracts } from './use-contracts'
|
|
||||||
import { useFollowedFoldIds } from './use-fold'
|
|
||||||
import { useSeenContracts } from './use-seen-contracts'
|
|
||||||
import { useUserBetContracts } from './use-user-bets'
|
|
||||||
|
|
||||||
// used in static props
|
|
||||||
export const getAllContractInfo = async () => {
|
|
||||||
let [contracts, folds] = await Promise.all([
|
|
||||||
getActiveContracts().catch((_) => []),
|
|
||||||
listAllFolds().catch(() => []),
|
|
||||||
])
|
|
||||||
|
|
||||||
const recentComments = await getRecentComments()
|
|
||||||
|
|
||||||
return { contracts, recentComments, folds }
|
|
||||||
}
|
|
||||||
|
|
||||||
const defaultExcludedTags = [
|
|
||||||
'meta',
|
|
||||||
'test',
|
|
||||||
'trolling',
|
|
||||||
'spam',
|
|
||||||
'transaction',
|
|
||||||
'personal',
|
|
||||||
]
|
|
||||||
const includedWithDefaultFeed = (contract: Contract) => {
|
|
||||||
const { lowercaseTags } = contract
|
|
||||||
|
|
||||||
if (lowercaseTags.length === 0) return false
|
|
||||||
if (lowercaseTags.some((tag) => defaultExcludedTags.includes(tag)))
|
|
||||||
return false
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
export const useFilterYourContracts = (
|
|
||||||
user: User | undefined | null,
|
|
||||||
folds: Fold[],
|
|
||||||
contracts: Contract[]
|
|
||||||
) => {
|
|
||||||
const followedFoldIds = useFollowedFoldIds(user)
|
|
||||||
|
|
||||||
const followedFolds = filterDefined(
|
|
||||||
(followedFoldIds ?? []).map((id) => folds.find((fold) => fold.id === id))
|
|
||||||
)
|
|
||||||
|
|
||||||
// Save the initial followed fold slugs.
|
|
||||||
const followedFoldSlugsRef = useRef<string[] | undefined>()
|
|
||||||
if (followedFoldIds && !followedFoldSlugsRef.current)
|
|
||||||
followedFoldSlugsRef.current = followedFolds.map((f) => f.slug)
|
|
||||||
const initialFollowedFoldSlugs = followedFoldSlugsRef.current
|
|
||||||
|
|
||||||
const tagSet = new Set(
|
|
||||||
_.flatten(followedFolds.map((fold) => fold.lowercaseTags))
|
|
||||||
)
|
|
||||||
|
|
||||||
const yourBetContractIds = useUserBetContracts(user?.id)
|
|
||||||
const yourBetContracts = yourBetContractIds
|
|
||||||
? new Set(yourBetContractIds)
|
|
||||||
: undefined
|
|
||||||
|
|
||||||
// Show no contracts before your info is loaded.
|
|
||||||
let yourContracts: Contract[] = []
|
|
||||||
if (yourBetContracts && followedFoldIds) {
|
|
||||||
// Show default contracts if no folds are followed.
|
|
||||||
if (followedFoldIds.length === 0)
|
|
||||||
yourContracts = contracts.filter(
|
|
||||||
(contract) =>
|
|
||||||
includedWithDefaultFeed(contract) || yourBetContracts.has(contract.id)
|
|
||||||
)
|
|
||||||
else
|
|
||||||
yourContracts = contracts.filter(
|
|
||||||
(contract) =>
|
|
||||||
contract.lowercaseTags.some((tag) => tagSet.has(tag)) ||
|
|
||||||
yourBetContracts.has(contract.id)
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
yourContracts,
|
|
||||||
initialFollowedFoldSlugs,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export const useFindActiveContracts = (props: {
|
|
||||||
contracts: Contract[]
|
|
||||||
recentBets: Bet[]
|
|
||||||
recentComments: Comment[]
|
|
||||||
}) => {
|
|
||||||
const { contracts, recentBets, recentComments } = props
|
|
||||||
|
|
||||||
const seenContracts = useSeenContracts()
|
|
||||||
|
|
||||||
const activeContracts = findActiveContracts(
|
|
||||||
contracts,
|
|
||||||
recentComments,
|
|
||||||
recentBets,
|
|
||||||
seenContracts
|
|
||||||
)
|
|
||||||
|
|
||||||
const betsByContract = _.groupBy(recentBets, (bet) => bet.contractId)
|
|
||||||
|
|
||||||
const activeBets = activeContracts.map(
|
|
||||||
(contract) => betsByContract[contract.id] ?? []
|
|
||||||
)
|
|
||||||
|
|
||||||
const commentsByContract = _.groupBy(
|
|
||||||
recentComments,
|
|
||||||
(comment) => comment.contractId
|
|
||||||
)
|
|
||||||
|
|
||||||
const activeComments = activeContracts.map(
|
|
||||||
(contract) => commentsByContract[contract.id] ?? []
|
|
||||||
)
|
|
||||||
|
|
||||||
return {
|
|
||||||
activeContracts,
|
|
||||||
activeBets,
|
|
||||||
activeComments,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export const useExploreContracts = (maxContracts = 75) => {
|
|
||||||
const inactiveContracts = useInactiveContracts()
|
|
||||||
|
|
||||||
const contractsDict = _.fromPairs(
|
|
||||||
(inactiveContracts ?? []).map((c) => [c.id, c])
|
|
||||||
)
|
|
||||||
|
|
||||||
// Preserve random ordering once inactiveContracts loaded.
|
|
||||||
const exploreContractIds = useMemo(
|
|
||||||
() => _.shuffle(Object.keys(contractsDict)),
|
|
||||||
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
||||||
[!!inactiveContracts]
|
|
||||||
).slice(0, maxContracts)
|
|
||||||
|
|
||||||
if (!inactiveContracts) return undefined
|
|
||||||
|
|
||||||
return filterDefined(exploreContractIds.map((id) => contractsDict[id]))
|
|
||||||
}
|
|
|
@ -52,3 +52,17 @@ export const useUserBetContracts = (userId: string | undefined) => {
|
||||||
|
|
||||||
return contractIds
|
return contractIds
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export const useGetUserBetContractIds = (userId: string | undefined) => {
|
||||||
|
const [contractIds, setContractIds] = useState<string[]>([])
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const key = `user-bet-contractIds-${userId}`
|
||||||
|
const userBetContractJson = localStorage.getItem(key)
|
||||||
|
if (userBetContractJson) {
|
||||||
|
setContractIds(JSON.parse(userBetContractJson))
|
||||||
|
}
|
||||||
|
}, [userId])
|
||||||
|
|
||||||
|
return contractIds
|
||||||
|
}
|
||||||
|
|
|
@ -124,7 +124,7 @@ const activeContractsQuery = query(
|
||||||
contractCollection,
|
contractCollection,
|
||||||
where('isResolved', '==', false),
|
where('isResolved', '==', false),
|
||||||
where('visibility', '==', 'public'),
|
where('visibility', '==', 'public'),
|
||||||
where('volume24Hours', '>', 0)
|
where('volume7Days', '>', 0)
|
||||||
)
|
)
|
||||||
|
|
||||||
export function getActiveContracts() {
|
export function getActiveContracts() {
|
||||||
|
|
|
@ -2,96 +2,56 @@ import React from 'react'
|
||||||
import Router from 'next/router'
|
import Router from 'next/router'
|
||||||
import _ from 'lodash'
|
import _ from 'lodash'
|
||||||
|
|
||||||
import { Contract } from '../lib/firebase/contracts'
|
|
||||||
import { Page } from '../components/page'
|
import { Page } from '../components/page'
|
||||||
import { ActivityFeed } from '../components/feed/activity-feed'
|
import { ActivityFeed } from '../components/feed/activity-feed'
|
||||||
import { Comment } from '../lib/firebase/comments'
|
|
||||||
import FeedCreate from '../components/feed-create'
|
import FeedCreate from '../components/feed-create'
|
||||||
import { Spacer } from '../components/layout/spacer'
|
import { Spacer } from '../components/layout/spacer'
|
||||||
import { Col } from '../components/layout/col'
|
import { Col } from '../components/layout/col'
|
||||||
import { useUser } from '../hooks/use-user'
|
import { useUser } from '../hooks/use-user'
|
||||||
import { Fold } from '../../common/fold'
|
|
||||||
import { LoadingIndicator } from '../components/loading-indicator'
|
import { LoadingIndicator } from '../components/loading-indicator'
|
||||||
import {
|
import { useRecentBets } from '../hooks/use-bets'
|
||||||
getAllContractInfo,
|
|
||||||
useFilterYourContracts,
|
|
||||||
useFindActiveContracts,
|
|
||||||
} from '../hooks/use-find-active-contracts'
|
|
||||||
import { fromPropz, usePropz } from '../hooks/use-propz'
|
|
||||||
import { useGetRecentBets, useRecentBets } from '../hooks/use-bets'
|
|
||||||
import { useActiveContracts } from '../hooks/use-contracts'
|
import { useActiveContracts } from '../hooks/use-contracts'
|
||||||
import { useRecentComments } from '../hooks/use-comments'
|
import { useRecentComments } from '../hooks/use-comments'
|
||||||
|
import { useAlgoFeed } from '../hooks/use-algo-feed'
|
||||||
|
|
||||||
export const getStaticProps = fromPropz(getStaticPropz)
|
const Home = () => {
|
||||||
export async function getStaticPropz() {
|
|
||||||
const contractInfo = await getAllContractInfo()
|
|
||||||
|
|
||||||
return {
|
|
||||||
props: contractInfo,
|
|
||||||
revalidate: 60, // regenerate after a minute
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const Home = (props: {
|
|
||||||
contracts: Contract[]
|
|
||||||
folds: Fold[]
|
|
||||||
recentComments: Comment[]
|
|
||||||
}) => {
|
|
||||||
props = usePropz(props, getStaticPropz) ?? {
|
|
||||||
contracts: [],
|
|
||||||
folds: [],
|
|
||||||
recentComments: [],
|
|
||||||
}
|
|
||||||
const { folds } = props
|
|
||||||
const user = useUser()
|
const user = useUser()
|
||||||
|
|
||||||
const contracts = useActiveContracts() ?? props.contracts
|
const contracts = useActiveContracts()
|
||||||
const { yourContracts } = useFilterYourContracts(user, folds, contracts)
|
const contractsDict = _.keyBy(contracts, 'id')
|
||||||
|
|
||||||
const initialRecentBets = useGetRecentBets()
|
const recentBets = useRecentBets()
|
||||||
const recentBets = useRecentBets() ?? initialRecentBets
|
const recentComments = useRecentComments()
|
||||||
const recentComments = useRecentComments() ?? props.recentComments
|
|
||||||
|
|
||||||
const { activeContracts } = useFindActiveContracts({
|
const feedContracts = useAlgoFeed(user, contracts, recentBets, recentComments)
|
||||||
contracts: yourContracts,
|
|
||||||
recentBets: initialRecentBets ?? [],
|
const updatedContracts = feedContracts.map(
|
||||||
recentComments: props.recentComments,
|
(contract) => contractsDict[contract.id] ?? contract
|
||||||
})
|
)
|
||||||
|
|
||||||
if (user === null) {
|
if (user === null) {
|
||||||
Router.replace('/')
|
Router.replace('/')
|
||||||
return <></>
|
return <></>
|
||||||
}
|
}
|
||||||
|
|
||||||
const activityContent = recentBets ? (
|
const activityContent =
|
||||||
<ActivityFeed
|
contracts && recentBets && recentComments ? (
|
||||||
contracts={activeContracts}
|
<ActivityFeed
|
||||||
recentBets={recentBets}
|
contracts={updatedContracts}
|
||||||
recentComments={recentComments}
|
recentBets={recentBets}
|
||||||
mode="only-recent"
|
recentComments={recentComments}
|
||||||
/>
|
mode="only-recent"
|
||||||
) : (
|
/>
|
||||||
<LoadingIndicator className="mt-4" />
|
) : (
|
||||||
)
|
<LoadingIndicator className="mt-4" />
|
||||||
|
)
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Page assertUser="signed-in">
|
<Page assertUser="signed-in">
|
||||||
<Col className="items-center">
|
<Col className="items-center">
|
||||||
<Col className="w-full max-w-[700px]">
|
<Col className="w-full max-w-[700px]">
|
||||||
<FeedCreate user={user ?? undefined} />
|
<FeedCreate user={user ?? undefined} />
|
||||||
<Spacer h={6} />
|
<Spacer h={10} />
|
||||||
|
|
||||||
{/* {initialFollowedFoldSlugs !== undefined &&
|
|
||||||
initialFollowedFoldSlugs.length === 0 &&
|
|
||||||
!IS_PRIVATE_MANIFOLD && (
|
|
||||||
<FastFoldFollowing
|
|
||||||
user={user}
|
|
||||||
followedFoldSlugs={initialFollowedFoldSlugs}
|
|
||||||
/>
|
|
||||||
)} */}
|
|
||||||
|
|
||||||
<Spacer h={5} />
|
|
||||||
|
|
||||||
{activityContent}
|
{activityContent}
|
||||||
</Col>
|
</Col>
|
||||||
</Col>
|
</Col>
|
||||||
|
|
Loading…
Reference in New Issue
Block a user