242 lines
7.1 KiB
TypeScript
242 lines
7.1 KiB
TypeScript
|
import clsx from 'clsx'
|
||
|
import { getOutcomeProbability, getTopAnswer } from 'common/calculate'
|
||
|
import { getExpectedValue } from 'common/calculate-dpm'
|
||
|
import {
|
||
|
Contract,
|
||
|
FullContract,
|
||
|
CPMM,
|
||
|
DPM,
|
||
|
Binary,
|
||
|
NumericContract,
|
||
|
FreeResponse,
|
||
|
} from 'common/contract'
|
||
|
import { formatMoney } from 'common/util/format'
|
||
|
import toast from 'react-hot-toast'
|
||
|
import { useUser } from 'web/hooks/use-user'
|
||
|
import { useUserContractBets } from 'web/hooks/use-user-bets'
|
||
|
import { placeBet } from 'web/lib/firebase/api-call'
|
||
|
import { getBinaryProb } from 'web/lib/firebase/contracts'
|
||
|
import TriangleDownFillIcon from 'web/lib/icons/triangle-down-fill-icon'
|
||
|
import TriangleFillIcon from 'web/lib/icons/triangle-fill-icon'
|
||
|
import { Col } from '../layout/col'
|
||
|
import { OUTCOME_TO_COLOR } from '../outcome-label'
|
||
|
import { useSaveShares } from '../use-save-shares'
|
||
|
import {
|
||
|
BinaryResolutionOrChance,
|
||
|
NumericResolutionOrExpectation,
|
||
|
FreeResponseResolutionOrChance,
|
||
|
} from './contract-card'
|
||
|
|
||
|
export function QuickBet(props: { contract: Contract }) {
|
||
|
const { contract } = props
|
||
|
|
||
|
const user = useUser()
|
||
|
const userBets = useUserContractBets(user?.id, contract.id)
|
||
|
const { yesFloorShares, noFloorShares, yesShares, noShares } = useSaveShares(
|
||
|
contract as FullContract<CPMM | DPM, Binary>,
|
||
|
userBets
|
||
|
)
|
||
|
// TODO: For some reason, Floor Shares are inverted for non-BINARY markets
|
||
|
const hasUpShares =
|
||
|
contract.outcomeType === 'BINARY' ? yesFloorShares : noFloorShares
|
||
|
const hasDownShares =
|
||
|
contract.outcomeType === 'BINARY' ? noFloorShares : yesFloorShares
|
||
|
|
||
|
const color = getColor(contract)
|
||
|
|
||
|
async function placeQuickBet(direction: 'UP' | 'DOWN') {
|
||
|
const betPromise = async () => {
|
||
|
const outcome = quickOutcome(contract, direction)
|
||
|
return await placeBet({
|
||
|
amount: 10,
|
||
|
outcome,
|
||
|
contractId: contract.id,
|
||
|
})
|
||
|
}
|
||
|
const shortQ = contract.question.slice(0, 20)
|
||
|
toast.promise(betPromise(), {
|
||
|
loading: `${formatMoney(10)} on "${shortQ}"...`,
|
||
|
success: `${formatMoney(10)} on "${shortQ}"...`,
|
||
|
error: (err) => `${err.message}`,
|
||
|
})
|
||
|
}
|
||
|
|
||
|
function quickOutcome(contract: Contract, direction: 'UP' | 'DOWN') {
|
||
|
if (contract.outcomeType === 'BINARY') {
|
||
|
return direction === 'UP' ? 'YES' : 'NO'
|
||
|
}
|
||
|
if (contract.outcomeType === 'FREE_RESPONSE') {
|
||
|
// TODO: Implement shorting of free response answers
|
||
|
if (direction === 'DOWN') {
|
||
|
throw new Error("Can't short free response answers")
|
||
|
}
|
||
|
return getTopAnswer(contract)?.id
|
||
|
}
|
||
|
if (contract.outcomeType === 'NUMERIC') {
|
||
|
// TODO: Ideally an 'UP' bet would be a uniform bet between [current, max]
|
||
|
throw new Error("Can't quick bet on numeric markets")
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return (
|
||
|
<Col
|
||
|
className={clsx(
|
||
|
'relative -my-4 -mr-5 min-w-[6rem] justify-center gap-2 pr-5 pl-3 align-middle',
|
||
|
// Use this for colored QuickBet panes
|
||
|
// `bg-opacity-10 bg-${color}`
|
||
|
'bg-gray-50'
|
||
|
)}
|
||
|
>
|
||
|
{/* Up bet triangle */}
|
||
|
<div>
|
||
|
<div
|
||
|
className="peer absolute top-0 left-0 right-0 h-[50%]"
|
||
|
onClick={() => placeQuickBet('UP')}
|
||
|
></div>
|
||
|
<div className="mt-2 text-center text-xs text-transparent peer-hover:text-gray-400">
|
||
|
{formatMoney(10)}
|
||
|
</div>
|
||
|
|
||
|
{hasUpShares > 0 ? (
|
||
|
<TriangleFillIcon
|
||
|
className={clsx(
|
||
|
'mx-auto h-5 w-5',
|
||
|
`text-${color} text-opacity-70 peer-hover:text-gray-400`
|
||
|
)}
|
||
|
/>
|
||
|
) : (
|
||
|
<TriangleFillIcon className="mx-auto h-5 w-5 text-gray-200 peer-hover:text-gray-400" />
|
||
|
)}
|
||
|
</div>
|
||
|
|
||
|
<QuickOutcomeView contract={contract} />
|
||
|
|
||
|
{/* Down bet triangle */}
|
||
|
<div>
|
||
|
<div
|
||
|
className="peer absolute bottom-0 left-0 right-0 h-[50%]"
|
||
|
onClick={() => placeQuickBet('DOWN')}
|
||
|
></div>
|
||
|
{hasDownShares > 0 ? (
|
||
|
<TriangleDownFillIcon
|
||
|
className={clsx(
|
||
|
'mx-auto h-5 w-5',
|
||
|
`text-${color} text-opacity-70 peer-hover:text-gray-400`
|
||
|
)}
|
||
|
/>
|
||
|
) : (
|
||
|
<TriangleDownFillIcon className="mx-auto h-5 w-5 text-gray-200 peer-hover:text-gray-400" />
|
||
|
)}
|
||
|
<div className="mb-2 text-center text-xs text-transparent peer-hover:text-gray-400">
|
||
|
{formatMoney(10)}
|
||
|
</div>
|
||
|
</div>
|
||
|
</Col>
|
||
|
)
|
||
|
}
|
||
|
|
||
|
export function ProbBar(props: { contract: Contract }) {
|
||
|
const { contract } = props
|
||
|
const color = getColor(contract)
|
||
|
const prob = getProb(contract)
|
||
|
return (
|
||
|
<>
|
||
|
<div
|
||
|
className={clsx(
|
||
|
'absolute right-0 top-0 w-2 rounded-tr-md transition-all',
|
||
|
'bg-gray-200'
|
||
|
)}
|
||
|
style={{ height: `${100 * (1 - prob)}%` }}
|
||
|
></div>
|
||
|
<div
|
||
|
className={clsx(
|
||
|
'absolute right-0 bottom-0 w-2 rounded-br-md transition-all',
|
||
|
`bg-${color}`,
|
||
|
// If we're showing the full bar, also round the top
|
||
|
prob === 1 ? 'rounded-tr-md' : ''
|
||
|
)}
|
||
|
style={{ height: `${100 * prob}%` }}
|
||
|
></div>
|
||
|
</>
|
||
|
)
|
||
|
}
|
||
|
|
||
|
export function QuickOutcomeView(props: { contract: Contract }) {
|
||
|
const { contract } = props
|
||
|
const { outcomeType } = contract
|
||
|
return (
|
||
|
<>
|
||
|
{outcomeType === 'BINARY' && (
|
||
|
<BinaryResolutionOrChance
|
||
|
className="items-center"
|
||
|
contract={contract}
|
||
|
hideText
|
||
|
/>
|
||
|
)}
|
||
|
|
||
|
{outcomeType === 'NUMERIC' && (
|
||
|
<NumericResolutionOrExpectation
|
||
|
className="items-center"
|
||
|
contract={contract as NumericContract}
|
||
|
hideText
|
||
|
/>
|
||
|
)}
|
||
|
|
||
|
{outcomeType === 'FREE_RESPONSE' && (
|
||
|
<FreeResponseResolutionOrChance
|
||
|
className="self-end text-gray-600"
|
||
|
contract={contract as FullContract<DPM, FreeResponse>}
|
||
|
truncate="long"
|
||
|
hideText
|
||
|
/>
|
||
|
)}
|
||
|
</>
|
||
|
)
|
||
|
}
|
||
|
|
||
|
// Return a number from 0 to 1 for this contract
|
||
|
// Resolved contracts are set to 1, for coloring purposes (even if NO)
|
||
|
function getProb(contract: Contract) {
|
||
|
const { outcomeType, resolution } = contract
|
||
|
return resolution
|
||
|
? 1
|
||
|
: outcomeType === 'BINARY'
|
||
|
? getBinaryProb(contract)
|
||
|
: outcomeType === 'FREE_RESPONSE'
|
||
|
? getOutcomeProbability(contract, getTopAnswer(contract)?.id || '')
|
||
|
: outcomeType === 'NUMERIC'
|
||
|
? getNumericScale(contract as NumericContract)
|
||
|
: 1 // Should not happen
|
||
|
}
|
||
|
|
||
|
function getNumericScale(contract: NumericContract) {
|
||
|
const { min, max } = contract
|
||
|
const ev = getExpectedValue(contract)
|
||
|
return (ev - min) / (max - min)
|
||
|
}
|
||
|
|
||
|
export function getColor(contract: Contract) {
|
||
|
// TODO: Not sure why eg green-400 doesn't work here; try upgrading Tailwind
|
||
|
// TODO: Try injecting a gradient here
|
||
|
// return 'primary'
|
||
|
const { resolution } = contract
|
||
|
if (resolution) {
|
||
|
return (
|
||
|
// @ts-ignore; TODO: Have better typing for contract.resolution?
|
||
|
OUTCOME_TO_COLOR[resolution] ||
|
||
|
// If resolved to a FR answer, use 'primary'
|
||
|
'primary'
|
||
|
)
|
||
|
}
|
||
|
if (contract.outcomeType === 'NUMERIC') {
|
||
|
return 'blue-400'
|
||
|
}
|
||
|
|
||
|
const marketClosed = (contract.closeTime || Infinity) < Date.now()
|
||
|
return marketClosed
|
||
|
? 'gray-400'
|
||
|
: getProb(contract) >= 0.5
|
||
|
? 'primary'
|
||
|
: 'red-400'
|
||
|
}
|