import _ from 'lodash'
import Link from 'next/link'

import { Fold } from '../../../../common/fold'
import { Comment } from '../../../../common/comment'
import { Page } from '../../../components/page'
import { Title } from '../../../components/title'
import { Bet, listAllBets } from '../../../lib/firebase/bets'
import { listAllComments } from '../../../lib/firebase/comments'
import { Contract } from '../../../lib/firebase/contracts'
import {
  foldPath,
  getFoldBySlug,
  getFoldContracts,
} from '../../../lib/firebase/folds'
import { ActivityFeed, findActiveContracts } from '../../activity'
import { TagsList } from '../../../components/tags-list'
import { Row } from '../../../components/layout/row'
import { UserLink } from '../../../components/user-page'
import { getUser, User } from '../../../lib/firebase/users'
import { Spacer } from '../../../components/layout/spacer'
import { Col } from '../../../components/layout/col'
import { useUser } from '../../../hooks/use-user'
import { useFold } from '../../../hooks/use-fold'
import { SearchableGrid } from '../../../components/contracts-list'
import { useQueryAndSortParams } from '../../../hooks/use-sort-and-query-params'
import { useRouter } from 'next/router'
import clsx from 'clsx'
import { scoreCreators, scoreTraders } from '../../../../common/scoring'
import { Leaderboard } from '../../../components/leaderboard'
import { formatMoney, toCamelCase } from '../../../../common/util/format'
import { EditFoldButton } from '../../../components/edit-fold-button'
import Custom404 from '../../404'
import { FollowFoldButton } from '../../../components/follow-fold-button'
import FeedCreate from '../../../components/feed-create'
import { SEO } from '../../../components/SEO'
import { useTaggedContracts } from '../../../hooks/use-contracts'

export async function getStaticProps(props: { params: { slugs: string[] } }) {
  const { slugs } = props.params

  const fold = await getFoldBySlug(slugs[0])
  const curatorPromise = fold ? getUser(fold.curatorId) : null

  const contracts = fold ? await getFoldContracts(fold).catch((_) => []) : []
  const contractComments = await Promise.all(
    contracts.map((contract) => listAllComments(contract.id).catch((_) => []))
  )

  let activeContracts = findActiveContracts(
    contracts,
    _.flatten(contractComments),
    [],
    365
  )
  const [resolved, unresolved] = _.partition(
    activeContracts,
    ({ isResolved }) => isResolved
  )
  activeContracts = [...unresolved, ...resolved]

  const activeContractBets = await Promise.all(
    activeContracts.map((contract) => listAllBets(contract.id).catch((_) => []))
  )
  const activeContractComments = activeContracts.map(
    (contract) =>
      contractComments[contracts.findIndex((c) => c.id === contract.id)]
  )

  const curator = await curatorPromise

  const bets = await Promise.all(
    contracts.map((contract) => listAllBets(contract.id))
  )

  const creatorScores = scoreCreators(contracts, bets)
  const [topCreators, topCreatorScores] = await toUserScores(creatorScores)

  const traderScores = scoreTraders(contracts, bets)
  const [topTraders, topTraderScores] = await toUserScores(traderScores)

  return {
    props: {
      fold,
      curator,
      contracts,
      activeContracts,
      activeContractBets,
      activeContractComments,
      topTraders,
      topTraderScores,
      topCreators,
      topCreatorScores,
    },

    revalidate: 60, // regenerate after a minute
  }
}

async function toUserScores(userScores: { [userId: string]: number }) {
  const topUserPairs = _.take(
    _.sortBy(Object.entries(userScores), ([_, score]) => -1 * score),
    10
  ).filter(([_, score]) => score > 0)

  const topUsers = await Promise.all(
    topUserPairs.map(([userId]) => getUser(userId))
  )
  const existingPairs = topUserPairs.filter(([id, _]) =>
    topUsers.find((user) => user?.id === id)
  )
  const topExistingUsers = existingPairs.map(
    ([id]) => topUsers.find((user) => user?.id === id) as User
  )
  const topUserScores = existingPairs.map(([_, score]) => score)
  return [topExistingUsers, topUserScores] as const
}

export async function getStaticPaths() {
  return { paths: [], fallback: 'blocking' }
}
const foldSubpages = [undefined, 'activity', 'markets', 'leaderboards'] as const

export default function FoldPage(props: {
  fold: Fold | null
  curator: User
  contracts: Contract[]
  activeContracts: Contract[]
  activeContractBets: Bet[][]
  activeContractComments: Comment[][]
  topTraders: User[]
  topTraderScores: number[]
  topCreators: User[]
  topCreatorScores: number[]
}) {
  const {
    curator,
    activeContractBets,
    activeContractComments,
    topTraders,
    topTraderScores,
    topCreators,
    topCreatorScores,
  } = props

  const router = useRouter()
  const { slugs } = router.query as { slugs: string[] }

  const page = (slugs[1] ?? 'activity') as typeof foldSubpages[number]

  const fold = useFold(props.fold?.id) ?? props.fold

  const { query, setQuery, sort, setSort } = useQueryAndSortParams({
    defaultSort: 'most-traded',
  })

  const user = useUser()
  const isCurator = user && fold && user.id === fold.curatorId

  const taggedContracts = useTaggedContracts(fold?.tags) ?? props.contracts
  const contractsMap = _.fromPairs(
    taggedContracts.map((contract) => [contract.id, contract])
  )

  const contracts = props.contracts.map((contract) => contractsMap[contract.id])
  const activeContracts = props.activeContracts.map(
    (contract) => contractsMap[contract.id]
  )

  if (fold === null || !foldSubpages.includes(page) || slugs[2]) {
    return <Custom404 />
  }

  return (
    <Page wide>
      <SEO
        title={fold.name}
        description={`Curated by ${curator.name}. ${fold.about}`}
        url={foldPath(fold)}
      />

      <div className="px-3 lg:px-1">
        <Row className="justify-between mb-6">
          <Title className="!m-0" text={fold.name} />
          {isCurator ? (
            <EditFoldButton className="ml-1" fold={fold} />
          ) : (
            <FollowFoldButton className="ml-1" fold={fold} />
          )}
        </Row>

        <Col className="md:hidden text-gray-500 gap-2 mb-6">
          <Row>
            <div className="mr-1">Curated by</div>
            <UserLink
              className="text-neutral"
              name={curator.name}
              username={curator.username}
            />
          </Row>
          <div>{fold.about}</div>
        </Col>
      </div>

      <div className="tabs mb-2">
        <Link href={foldPath(fold)} shallow>
          <a
            className={clsx(
              'tab tab-bordered',
              page === 'activity' && 'tab-active'
            )}
          >
            Activity
          </a>
        </Link>

        <Link href={foldPath(fold, 'markets')} shallow>
          <a
            className={clsx(
              'tab tab-bordered',
              page === 'markets' && 'tab-active'
            )}
          >
            Markets
          </a>
        </Link>
        <Link href={foldPath(fold, 'leaderboards')} shallow>
          <a
            className={clsx(
              'tab tab-bordered',
              page === 'leaderboards' && 'tab-active',
              page !== 'leaderboards' && 'md:hidden'
            )}
          >
            Leaderboards
          </a>
        </Link>
      </div>

      {(page === 'activity' || page === 'markets') && (
        <Row className={clsx(page === 'activity' ? 'gap-16' : 'gap-8')}>
          <Col className="flex-1">
            {user !== null && !fold.disallowMarketCreation && (
              <FeedCreate
                className={clsx('border-b-2', page !== 'activity' && 'hidden')}
                user={user}
                tag={toCamelCase(fold.name)}
                placeholder={`Type your question about ${fold.name}`}
              />
            )}
            {page === 'activity' ? (
              <>
                <ActivityFeed
                  contracts={activeContracts}
                  contractBets={activeContractBets}
                  contractComments={activeContractComments}
                />
                {activeContracts.length === 0 && (
                  <div className="text-gray-500 mt-4 mx-2 lg:mx-0">
                    No activity from matching markets.{' '}
                    {isCurator && 'Try editing to add more tags!'}
                  </div>
                )}
              </>
            ) : (
              <SearchableGrid
                contracts={contracts}
                query={query}
                setQuery={setQuery}
                sort={sort}
                setSort={setSort}
              />
            )}
          </Col>
          <Col className="hidden md:flex max-w-xs w-full gap-10">
            <FoldOverview fold={fold} curator={curator} />
            <FoldLeaderboards
              topTraders={topTraders}
              topTraderScores={topTraderScores}
              topCreators={topCreators}
              topCreatorScores={topCreatorScores}
            />
          </Col>
        </Row>
      )}

      {page === 'leaderboards' && (
        <Col className="gap-8 lg:flex-row">
          <FoldLeaderboards
            topTraders={topTraders}
            topTraderScores={topTraderScores}
            topCreators={topCreators}
            topCreatorScores={topCreatorScores}
          />
        </Col>
      )}
    </Page>
  )
}

function FoldOverview(props: { fold: Fold; curator: User }) {
  const { fold, curator } = props
  const { about, tags } = fold

  return (
    <Col>
      <div className="px-4 py-3 bg-indigo-500 text-white text-sm rounded-t">
        About community
      </div>
      <Col className="p-4 bg-white gap-2 rounded-b">
        <Row>
          <div className="text-gray-500 mr-1">Curated by</div>
          <UserLink
            className="text-neutral"
            name={curator.name}
            username={curator.username}
          />
        </Row>

        {about && (
          <>
            <Spacer h={2} />
            <div className="text-gray-500">{about}</div>
          </>
        )}

        <div className="divider" />

        <div className="text-gray-500 mb-2">
          Includes markets matching any of these tags:
        </div>

        <TagsList tags={tags.map((tag) => `#${tag}`)} noLabel />
      </Col>
    </Col>
  )
}

function FoldLeaderboards(props: {
  topTraders: User[]
  topTraderScores: number[]
  topCreators: User[]
  topCreatorScores: number[]
}) {
  const { topTraders, topTraderScores, topCreators, topCreatorScores } = props
  return (
    <>
      <Leaderboard
        className="max-w-xl"
        title="🏅 Top traders"
        users={topTraders}
        columns={[
          {
            header: 'Profit',
            renderCell: (user) =>
              formatMoney(topTraderScores[topTraders.indexOf(user)]),
          },
        ]}
      />
      <Leaderboard
        className="max-w-xl"
        title="🏅 Top creators"
        users={topCreators}
        columns={[
          {
            header: 'Market pool',
            renderCell: (user) =>
              formatMoney(topCreatorScores[topCreators.indexOf(user)]),
          },
        ]}
      />
    </>
  )
}