import { ClockIcon } from '@heroicons/react/outline' import clsx from 'clsx' import { Editor } from '@tiptap/react' import dayjs from 'dayjs' import Link from 'next/link' import { Row } from '../layout/row' import { formatMoney } from 'common/util/format' import { Contract, updateContract } from 'web/lib/firebase/contracts' import { DateTimeTooltip } from '../datetime-tooltip' import { fromNow } from 'web/lib/util/time' import { Avatar } from '../avatar' import { useState } from 'react' import NewContractBadge from '../new-contract-badge' import { MiniUserFollowButton } from '../follow-button' import { DAY_MS } from 'common/util/time' import { useUser } from 'web/hooks/use-user' import { exhibitExts } from 'common/util/parse' import { Button } from 'web/components/button' import { Modal } from 'web/components/layout/modal' import { Col } from 'web/components/layout/col' import { ContractGroupsList } from 'web/components/groups/contract-groups-list' import { linkClass } from 'web/components/site-link' import { getGroupLinkToDisplay, groupPath } from 'web/lib/firebase/groups' import { insertContent } from '../editor/utils' import { contractMetrics } from 'common/contract-details' import { UserLink } from 'web/components/user-link' import { FeaturedContractBadge } from 'web/components/contract/featured-contract-badge' import { Tooltip } from 'web/components/tooltip' import { ExtraContractActionsRow } from './extra-contract-actions-row' import { PlusCircleIcon } from '@heroicons/react/solid' import { GroupLink } from 'common/group' import { Subtitle } from '../subtitle' import { useIsMobile } from 'web/hooks/use-is-mobile' import { BountiedContractBadge, BountiedContractSmallBadge, } from 'web/components/contract/bountied-contract-badge' export type ShowTime = 'resolve-date' | 'close-date' export function MiscDetails(props: { contract: Contract showTime?: ShowTime hideGroupLink?: boolean }) { const { contract, showTime, hideGroupLink } = props const { volume, closeTime, isResolved, createdTime, resolutionTime } = contract const isNew = createdTime > Date.now() - DAY_MS && !isResolved const groupToDisplay = getGroupLinkToDisplay(contract) return ( <Row className="items-center gap-3 truncate text-sm text-gray-400"> {showTime === 'close-date' ? ( <Row className="gap-0.5 whitespace-nowrap"> <ClockIcon className="h-5 w-5" /> {(closeTime || 0) < Date.now() ? 'Closed' : 'Closes'}{' '} {fromNow(closeTime || 0)} </Row> ) : showTime === 'resolve-date' && resolutionTime !== undefined ? ( <Row className="gap-0.5"> <ClockIcon className="h-5 w-5" /> {'Resolved '} {fromNow(resolutionTime || 0)} </Row> ) : (contract?.featuredOnHomeRank ?? 0) > 0 ? ( <FeaturedContractBadge /> ) : (contract.openCommentBounties ?? 0) > 0 ? ( <BountiedContractBadge /> ) : volume > 0 || !isNew ? ( <Row className={'shrink-0'}>{formatMoney(volume)} bet</Row> ) : ( <NewContractBadge /> )} {!hideGroupLink && groupToDisplay && ( <Link prefetch={false} href={groupPath(groupToDisplay.slug)}> <a className={clsx(linkClass, 'truncate text-sm text-gray-400')}> {groupToDisplay.name} </a> </Link> )} </Row> ) } export function AvatarDetails(props: { contract: Contract className?: string short?: boolean noLink?: boolean }) { const { contract, short, className, noLink } = props const { creatorName, creatorUsername, creatorAvatarUrl } = contract return ( <Row className={clsx('items-center gap-2 text-sm text-gray-400', className)} > <Avatar username={creatorUsername} avatarUrl={creatorAvatarUrl} size={6} noLink={noLink} /> <UserLink name={creatorName} username={creatorUsername} short={short} noLink={noLink} /> </Row> ) } export function ContractDetails(props: { contract: Contract disabled?: boolean }) { const { contract, disabled } = props const isMobile = useIsMobile() return ( <Col> <Row className="justify-between"> <MarketSubheader contract={contract} disabled={disabled} /> <div className="mt-0"> <ExtraContractActionsRow contract={contract} /> </div> </Row> {/* GROUPS */} {isMobile && ( <Row className="mt-2 gap-1"> <BountiedContractSmallBadge contract={contract} /> <MarketGroups contract={contract} disabled={disabled} /> </Row> )} </Col> ) } export function MarketSubheader(props: { contract: Contract disabled?: boolean }) { const { contract, disabled } = props const { creatorName, creatorUsername, creatorId, creatorAvatarUrl } = contract const { resolvedDate } = contractMetrics(contract) const user = useUser() const isCreator = user?.id === creatorId const isMobile = useIsMobile() return ( <Row> <Avatar username={creatorUsername} avatarUrl={creatorAvatarUrl} noLink={disabled} size={9} className="mr-1.5" /> {!disabled && ( <div className="absolute mt-3 ml-[11px]"> <MiniUserFollowButton userId={creatorId} /> </div> )} <Col className="text-greyscale-6 ml-2 flex-1 flex-wrap text-sm"> <Row className="w-full justify-between "> {disabled ? ( creatorName ) : ( <UserLink className="my-auto whitespace-nowrap" name={creatorName} username={creatorUsername} short={isMobile} /> )} </Row> <Row className="text-2xs text-greyscale-4 gap-2 sm:text-xs"> <CloseOrResolveTime contract={contract} resolvedDate={resolvedDate} isCreator={isCreator} /> {!isMobile && ( <Row className={'gap-1'}> <BountiedContractSmallBadge contract={contract} /> <MarketGroups contract={contract} disabled={disabled} /> </Row> )} </Row> </Col> </Row> ) } export function CloseOrResolveTime(props: { contract: Contract resolvedDate: any isCreator: boolean }) { const { contract, resolvedDate, isCreator } = props const { resolutionTime, closeTime } = contract if (!!closeTime || !!resolvedDate) { return ( <Row className="select-none items-center gap-1"> {resolvedDate && resolutionTime ? ( <> <DateTimeTooltip text="Market resolved:" time={resolutionTime}> <Row> <div>resolved </div> {resolvedDate} </Row> </DateTimeTooltip> </> ) : null} {!resolvedDate && closeTime && ( <Row> {dayjs().isBefore(closeTime) && <div>closes </div>} {!dayjs().isBefore(closeTime) && <div>closed </div>} <EditableCloseDate closeTime={closeTime} contract={contract} isCreator={isCreator ?? false} /> </Row> )} </Row> ) } else return <></> } export function MarketGroups(props: { contract: Contract disabled?: boolean }) { const [open, setOpen] = useState(false) const user = useUser() const { contract, disabled } = props const groupToDisplay = getGroupLinkToDisplay(contract) return ( <> <Row className="items-center gap-1"> <GroupDisplay groupToDisplay={groupToDisplay} /> {!disabled && user && ( <button className="text-greyscale-4 hover:text-greyscale-3" onClick={() => setOpen(true)} > <PlusCircleIcon className="h-[18px]" /> </button> )} </Row> <Modal open={open} setOpen={setOpen} size={'md'}> <Col className={ 'max-h-[70vh] min-h-[20rem] overflow-auto rounded bg-white p-6' } > <ContractGroupsList contract={contract} user={user} /> </Col> </Modal> </> ) } export function ExtraMobileContractDetails(props: { contract: Contract forceShowVolume?: boolean }) { const { contract, forceShowVolume } = props const { volume, resolutionTime, closeTime, creatorId, uniqueBettorCount } = contract const user = useUser() const uniqueBettors = uniqueBettorCount ?? 0 const { resolvedDate } = contractMetrics(contract) const volumeTranslation = volume > 800 || uniqueBettors >= 20 ? 'High' : volume > 300 || uniqueBettors >= 10 ? 'Medium' : 'Low' return ( <Row className={clsx( 'items-center justify-around md:hidden', user ? 'w-full' : '' )} > {resolvedDate && resolutionTime ? ( <Col className={'items-center text-sm'}> <Row className={'text-gray-500'}> <DateTimeTooltip text="Market resolved:" time={resolutionTime}> {resolvedDate} </DateTimeTooltip> </Row> <Row className={'text-gray-400'}>Ended</Row> </Col> ) : ( !resolvedDate && closeTime && ( <Col className={'items-center text-sm text-gray-500'}> <Row className={'text-gray-400'}>Closes </Row> <EditableCloseDate closeTime={closeTime} contract={contract} isCreator={creatorId === user?.id} /> </Col> ) )} {(user || forceShowVolume) && ( <Col className={'items-center text-sm text-gray-500'}> <Tooltip text={`${formatMoney( volume )} bet - ${uniqueBettors} unique traders`} > {volumeTranslation} </Tooltip> <Row className={'text-gray-400'}>Activity</Row> </Col> )} </Row> ) } export function GroupDisplay(props: { groupToDisplay?: GroupLink | null }) { const { groupToDisplay } = props if (groupToDisplay) { return ( <Link prefetch={false} href={groupPath(groupToDisplay.slug)}> <a className="bg-greyscale-4 hover:bg-greyscale-3 max-w-[140px] truncate rounded-full py-0.5 px-2 text-xs text-white sm:max-w-[250px]"> {groupToDisplay.name} </a> </Link> ) } else return ( <div className="bg-greyscale-4 truncate rounded-full py-0.5 px-2 text-xs text-white"> No Group </div> ) } function EditableCloseDate(props: { closeTime: number contract: Contract isCreator: boolean }) { const { closeTime, contract, isCreator } = props const dayJsCloseTime = dayjs(closeTime) const dayJsNow = dayjs() const [isEditingCloseTime, setIsEditingCloseTime] = useState(false) const [closeDate, setCloseDate] = useState( closeTime && dayJsCloseTime.format('YYYY-MM-DD') ) const [closeHoursMinutes, setCloseHoursMinutes] = useState( closeTime && dayJsCloseTime.format('HH:mm') ) const isSameYear = dayJsCloseTime.isSame(dayJsNow, 'year') const isSameDay = dayJsCloseTime.isSame(dayJsNow, 'day') let newCloseTime = closeDate ? dayjs(`${closeDate}T${closeHoursMinutes}`).valueOf() : undefined function onSave(customTime?: number) { if (customTime) { newCloseTime = customTime setCloseDate(dayjs(newCloseTime).format('YYYY-MM-DD')) setCloseHoursMinutes(dayjs(newCloseTime).format('HH:mm')) } if (!newCloseTime) return if (newCloseTime === closeTime) setIsEditingCloseTime(false) else { const content = contract.description const formattedCloseDate = dayjs(newCloseTime).format('YYYY-MM-DD h:mm a') const editor = new Editor({ content, extensions: exhibitExts }) editor.commands.focus('end') insertContent( editor, `<br><p>Close date updated to ${formattedCloseDate}</p>` ) updateContract(contract.id, { closeTime: newCloseTime, description: editor.getJSON(), }) setIsEditingCloseTime(false) } } return ( <> <Modal size="sm" open={isEditingCloseTime} setOpen={setIsEditingCloseTime} position="top" > <Col className="rounded bg-white px-8 pb-8"> <Subtitle text="Edit Close Date" /> <Row className="z-10 mr-2 w-full shrink-0 flex-wrap items-center gap-2"> <input type="date" className="input input-bordered w-full shrink-0 sm:w-fit" onClick={(e) => e.stopPropagation()} onChange={(e) => setCloseDate(e.target.value)} min={Date.now()} value={closeDate} /> <input type="time" className="input input-bordered w-full shrink-0 sm:w-max" onClick={(e) => e.stopPropagation()} onChange={(e) => setCloseHoursMinutes(e.target.value)} min="00:00" value={closeHoursMinutes} /> </Row> <Button className="mt-4" size={'xs'} color={'indigo'} onClick={() => onSave()} > Done </Button> <Button className="mt-4" size={'xs'} color={'gray-white'} onClick={() => onSave(Date.now())} > Close Now </Button> </Col> </Modal> <DateTimeTooltip text={closeTime > Date.now() ? 'Trading ends:' : 'Trading ended:'} time={closeTime} > <span className={isCreator ? 'cursor-pointer' : ''} onClick={() => isCreator && setIsEditingCloseTime(true)} > {isSameDay ? ( <span className={'capitalize'}> {fromNow(closeTime)}</span> ) : isSameYear ? ( dayJsCloseTime.format('MMM D') ) : ( dayJsCloseTime.format('MMM D, YYYY') )} </span> </DateTimeTooltip> </> ) }