129 lines
3.5 KiB
TypeScript
129 lines
3.5 KiB
TypeScript
import clsx from 'clsx'
|
|
import Link from 'next/link'
|
|
import { Row } from '../components/layout/row'
|
|
import { formatMoney } from '../lib/util/format'
|
|
import { UserLink } from './user-page'
|
|
import { Linkify } from './linkify'
|
|
import { Contract, compute, path } from '../lib/firebase/contracts'
|
|
import { Col } from './layout/col'
|
|
import { parseTags } from '../lib/util/parse'
|
|
|
|
export function ContractCard(props: { contract: Contract }) {
|
|
const { contract } = props
|
|
const { question, resolution } = contract
|
|
const { probPercent } = compute(contract)
|
|
|
|
return (
|
|
<Link href={path(contract)}>
|
|
<a>
|
|
<li className="col-span-1 bg-white hover:bg-gray-100 shadow-md rounded-lg divide-y divide-gray-200">
|
|
<div className="card">
|
|
<div className="card-body p-6">
|
|
<Row className="justify-between gap-4 mb-2">
|
|
<p className="font-medium text-indigo-700">{question}</p>
|
|
<ResolutionOrChance
|
|
className="items-center"
|
|
resolution={resolution}
|
|
probPercent={probPercent}
|
|
/>
|
|
</Row>
|
|
<ContractDetails contract={contract} />
|
|
</div>
|
|
</div>
|
|
</li>
|
|
</a>
|
|
</Link>
|
|
)
|
|
}
|
|
|
|
export function ResolutionOrChance(props: {
|
|
resolution?: 'YES' | 'NO' | 'MKT' | 'CANCEL'
|
|
probPercent: string
|
|
large?: boolean
|
|
className?: string
|
|
}) {
|
|
const { resolution, probPercent, large, className } = props
|
|
|
|
const resolutionColor = {
|
|
YES: 'text-primary',
|
|
NO: 'text-red-400',
|
|
MKT: 'text-blue-400',
|
|
CANCEL: 'text-yellow-400',
|
|
'': '', // Empty if unresolved
|
|
}[resolution || '']
|
|
|
|
const resolutionText = {
|
|
YES: 'YES',
|
|
NO: 'NO',
|
|
MKT: 'MKT',
|
|
CANCEL: 'N/A',
|
|
'': '',
|
|
}[resolution || '']
|
|
|
|
return (
|
|
<Col className={clsx(large ? 'text-4xl' : 'text-3xl', className)}>
|
|
{resolution ? (
|
|
<>
|
|
<div
|
|
className={clsx('text-gray-500', large ? 'text-xl' : 'text-base')}
|
|
>
|
|
Resolved
|
|
</div>
|
|
<div className={resolutionColor}>{resolutionText}</div>
|
|
</>
|
|
) : (
|
|
<>
|
|
<div className="text-primary">{probPercent}</div>
|
|
<div
|
|
className={clsx('text-primary', large ? 'text-xl' : 'text-base')}
|
|
>
|
|
chance
|
|
</div>
|
|
</>
|
|
)}
|
|
</Col>
|
|
)
|
|
}
|
|
|
|
export function ContractDetails(props: {
|
|
contract: Contract
|
|
inlineTags?: boolean
|
|
}) {
|
|
const { contract, inlineTags } = props
|
|
const { question, description } = contract
|
|
const { truePool, createdDate, resolvedDate } = compute(contract)
|
|
|
|
const tags = parseTags(`${question} ${description}`).map((tag) => `#${tag}`)
|
|
|
|
return (
|
|
<Col
|
|
className={clsx(
|
|
'text-sm text-gray-500 gap-2',
|
|
inlineTags && 'sm:flex-row sm:flex-wrap'
|
|
)}
|
|
>
|
|
<Row className="gap-2 flex-wrap">
|
|
<div className="whitespace-nowrap">
|
|
<UserLink username={contract.creatorUsername} />
|
|
</div>
|
|
<div className="">•</div>
|
|
<div className="whitespace-nowrap">
|
|
{resolvedDate ? `${createdDate} - ${resolvedDate}` : createdDate}
|
|
</div>
|
|
<div className="">•</div>
|
|
<div className="whitespace-nowrap">{formatMoney(truePool)} pool</div>
|
|
</Row>
|
|
|
|
{inlineTags && <div className="hidden sm:block">•</div>}
|
|
|
|
<Row className="gap-2 flex-wrap">
|
|
{tags.map((tag) => (
|
|
<div className="bg-gray-100 px-1">
|
|
<Linkify text={tag} gray />
|
|
</div>
|
|
))}
|
|
</Row>
|
|
</Col>
|
|
)
|
|
}
|