import _ from 'lodash' import Link from 'next/link' import { useEffect, useState } from 'react' import { Fold } from '../../common/fold' import { CreateFoldButton } from '../components/create-fold-button' import { FollowFoldButton } from '../components/follow-fold-button' import { Col } from '../components/layout/col' import { Row } from '../components/layout/row' import { Page } from '../components/page' import { SiteLink } from '../components/site-link' import { TagsList } from '../components/tags-list' import { Title } from '../components/title' import { UserLink } from '../components/user-page' import { useFolds } from '../hooks/use-fold' import { useUser } from '../hooks/use-user' import { foldPath, listAllFolds } from '../lib/firebase/folds' import { getUser, User } from '../lib/firebase/users' export async function getStaticProps() { const folds = await listAllFolds().catch((_) => []) const curators = await Promise.all( folds.map((fold) => getUser(fold.curatorId)) ) const curatorsDict = _.fromPairs( curators.map((curator) => [curator.id, curator]) ) return { props: { folds, curatorsDict, }, revalidate: 60, // regenerate after a minute } } export default function Folds(props: { folds: Fold[] curatorsDict: _.Dictionary<User> }) { const [curatorsDict, setCuratorsDict] = useState(props.curatorsDict) let folds = useFolds() ?? props.folds folds = _.sortBy(folds, (fold) => -1 * fold.followCount) const user = useUser() useEffect(() => { // Load User object for curator of new Folds. const newFolds = folds.filter(({ curatorId }) => !curatorsDict[curatorId]) if (newFolds.length > 0) { Promise.all(newFolds.map(({ curatorId }) => getUser(curatorId))).then( (newUsers) => { const newUsersDict = _.fromPairs( newUsers.map((user) => [user.id, user]) ) setCuratorsDict({ ...curatorsDict, ...newUsersDict }) } ) } }) return ( <Page> <Col className="items-center"> <Col className="max-w-xl w-full"> <Col className="px-4 sm:px-0"> <Row className="justify-between items-center"> <Title text="Explore communities" /> {user && <CreateFoldButton />} </Row> <div className="text-gray-500 mb-6"> Communities on Manifold are centered around a collection of markets. </div> </Col> <Col className="gap-4"> {folds.map((fold) => ( <FoldCard key={fold.id} fold={fold} curator={curatorsDict[fold.curatorId]} /> ))} </Col> </Col> </Col> </Page> ) } function FoldCard(props: { fold: Fold; curator: User | undefined }) { const { fold, curator } = props const tags = fold.tags.slice(1) return ( <Col key={fold.id} className="bg-white hover:bg-gray-100 p-8 rounded-xl gap-1 shadow-md relative" > <Link href={foldPath(fold)}> <a className="absolute left-0 right-0 top-0 bottom-0" /> </Link> <Row className="justify-between items-center gap-2"> <span className="text-xl">{fold.name}</span> <FollowFoldButton className="z-10 mb-1" fold={fold} /> </Row> <Row className="items-center gap-2 text-gray-500 text-sm"> <div>{fold.followCount} followers</div> <div>•</div> <Row> <div className="mr-1">Curated by</div> <UserLink className="text-neutral" name={curator?.name ?? ''} username={curator?.username ?? ''} /> </Row> </Row> <div className="text-gray-500 text-sm">{fold.about}</div> {tags.length > 0 && ( <TagsList className="mt-4" tags={tags} noLink noLabel /> )} </Col> ) }