2022-10-05 06:16:56 +00:00
|
|
|
import React, { memo, useEffect, useRef, useState } from 'react'
|
|
|
|
import { Editor } from '@tiptap/react'
|
|
|
|
import { useRouter } from 'next/router'
|
|
|
|
import { sum } from 'lodash'
|
|
|
|
import clsx from 'clsx'
|
|
|
|
|
2022-08-19 08:06:40 +00:00
|
|
|
import { ContractComment } from 'common/comment'
|
2022-05-18 14:42:13 +00:00
|
|
|
import { Contract } from 'common/contract'
|
|
|
|
import { useUser } from 'web/hooks/use-user'
|
|
|
|
import { formatMoney } from 'common/util/format'
|
|
|
|
import { Row } from 'web/components/layout/row'
|
|
|
|
import { Avatar } from 'web/components/avatar'
|
|
|
|
import { OutcomeLabel } from 'web/components/outcome-label'
|
|
|
|
import { CopyLinkDateTimeComponent } from 'web/components/feed/copy-link-date-time'
|
|
|
|
import { firebaseLogin } from 'web/lib/firebase/users'
|
2022-09-07 22:09:20 +00:00
|
|
|
import { createCommentOnContract } from 'web/lib/firebase/comments'
|
2022-05-18 14:42:13 +00:00
|
|
|
import { Col } from 'web/components/layout/col'
|
2022-06-15 21:34:34 +00:00
|
|
|
import { track } from 'web/lib/service/analytics'
|
2022-06-18 03:28:16 +00:00
|
|
|
import { Tipper } from '../tipper'
|
2022-10-05 06:16:56 +00:00
|
|
|
import { CommentTipMap } from 'web/hooks/use-tip-txns'
|
|
|
|
import { useEvent } from 'web/hooks/use-event'
|
2022-09-07 22:09:20 +00:00
|
|
|
import { Content } from '../editor'
|
2022-08-30 15:38:59 +00:00
|
|
|
import { UserLink } from 'web/components/user-link'
|
2022-09-07 22:09:20 +00:00
|
|
|
import { CommentInput } from '../comment-input'
|
2022-09-30 15:27:42 +00:00
|
|
|
import { AwardBountyButton } from 'web/components/award-bounty-button'
|
2022-10-12 18:05:58 +00:00
|
|
|
import { ReplyIcon } from '@heroicons/react/solid'
|
|
|
|
import { Button } from '../button'
|
|
|
|
import { ReplyToggle } from '../comments/reply-toggle'
|
2022-05-18 14:42:13 +00:00
|
|
|
|
2022-09-22 19:40:44 +00:00
|
|
|
export type ReplyTo = { id: string; username: string }
|
|
|
|
|
2022-05-18 14:42:13 +00:00
|
|
|
export function FeedCommentThread(props: {
|
|
|
|
contract: Contract
|
2022-08-30 09:41:47 +00:00
|
|
|
threadComments: ContractComment[]
|
2022-06-18 03:28:16 +00:00
|
|
|
tips: CommentTipMap
|
2022-08-19 08:06:40 +00:00
|
|
|
parentComment: ContractComment
|
2022-05-18 14:42:13 +00:00
|
|
|
}) {
|
2022-09-20 22:25:58 +00:00
|
|
|
const { contract, threadComments, tips, parentComment } = props
|
2022-09-22 19:40:44 +00:00
|
|
|
const [replyTo, setReplyTo] = useState<ReplyTo>()
|
2022-10-12 22:49:04 +00:00
|
|
|
const [seeReplies, setSeeReplies] = useState(true)
|
2022-08-06 20:39:52 +00:00
|
|
|
|
2022-10-05 06:16:56 +00:00
|
|
|
const user = useUser()
|
|
|
|
const onSubmitComment = useEvent(() => setReplyTo(undefined))
|
|
|
|
const onReplyClick = useEvent((comment: ContractComment) => {
|
|
|
|
setReplyTo({ id: comment.id, username: comment.userUsername })
|
|
|
|
})
|
|
|
|
|
2022-05-18 14:42:13 +00:00
|
|
|
return (
|
2022-08-30 09:41:47 +00:00
|
|
|
<Col className="relative w-full items-stretch gap-3 pb-4">
|
2022-10-12 18:05:58 +00:00
|
|
|
<ParentFeedComment
|
|
|
|
key={parentComment.id}
|
|
|
|
contract={contract}
|
|
|
|
comment={parentComment}
|
|
|
|
myTip={user ? tips[parentComment.id]?.[user.id] : undefined}
|
|
|
|
totalTip={sum(Object.values(tips[parentComment.id] ?? {}))}
|
|
|
|
showTip={true}
|
|
|
|
seeReplies={seeReplies}
|
|
|
|
numComments={threadComments.length}
|
|
|
|
onSeeReplyClick={() => setSeeReplies(!seeReplies)}
|
|
|
|
onReplyClick={() =>
|
|
|
|
setReplyTo({
|
|
|
|
id: parentComment.id,
|
|
|
|
username: parentComment.userUsername,
|
|
|
|
})
|
|
|
|
}
|
2022-06-08 13:24:12 +00:00
|
|
|
/>
|
2022-10-12 18:05:58 +00:00
|
|
|
{seeReplies &&
|
|
|
|
threadComments.map((comment, _commentIdx) => (
|
|
|
|
<FeedComment
|
|
|
|
key={comment.id}
|
|
|
|
contract={contract}
|
|
|
|
comment={comment}
|
|
|
|
myTip={user ? tips[comment.id]?.[user.id] : undefined}
|
|
|
|
totalTip={sum(Object.values(tips[comment.id] ?? {}))}
|
|
|
|
showTip={true}
|
|
|
|
onReplyClick={onReplyClick}
|
|
|
|
/>
|
|
|
|
))}
|
2022-09-22 19:40:44 +00:00
|
|
|
{replyTo && (
|
2022-08-30 09:41:47 +00:00
|
|
|
<Col className="-pb-2 relative ml-6">
|
2022-09-07 22:09:20 +00:00
|
|
|
<ContractCommentInput
|
2022-06-08 13:24:12 +00:00
|
|
|
contract={contract}
|
|
|
|
parentCommentId={parentComment.id}
|
2022-09-22 19:40:44 +00:00
|
|
|
replyTo={replyTo}
|
2022-10-05 06:16:56 +00:00
|
|
|
onSubmitComment={onSubmitComment}
|
2022-06-08 13:24:12 +00:00
|
|
|
/>
|
2022-07-21 05:46:56 +00:00
|
|
|
</Col>
|
2022-06-08 13:24:12 +00:00
|
|
|
)}
|
2022-07-21 05:46:56 +00:00
|
|
|
</Col>
|
2022-06-08 13:24:12 +00:00
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2022-10-12 18:05:58 +00:00
|
|
|
export function ParentFeedComment(props: {
|
2022-05-18 14:42:13 +00:00
|
|
|
contract: Contract
|
2022-08-19 08:06:40 +00:00
|
|
|
comment: ContractComment
|
2022-10-05 06:16:56 +00:00
|
|
|
showTip?: boolean
|
|
|
|
myTip?: number
|
|
|
|
totalTip?: number
|
2022-10-12 18:05:58 +00:00
|
|
|
seeReplies: boolean
|
|
|
|
numComments: number
|
2022-10-05 06:16:56 +00:00
|
|
|
onReplyClick?: (comment: ContractComment) => void
|
2022-10-12 18:05:58 +00:00
|
|
|
onSeeReplyClick: () => void
|
2022-05-18 14:42:13 +00:00
|
|
|
}) {
|
|
|
|
const {
|
2022-10-12 18:05:58 +00:00
|
|
|
contract,
|
|
|
|
comment,
|
|
|
|
myTip,
|
|
|
|
totalTip,
|
|
|
|
showTip,
|
|
|
|
onReplyClick,
|
|
|
|
onSeeReplyClick,
|
|
|
|
seeReplies,
|
|
|
|
numComments,
|
|
|
|
} = props
|
|
|
|
const { text, content, userUsername, userAvatarUrl } = comment
|
2022-05-18 14:42:13 +00:00
|
|
|
|
2022-10-07 19:40:38 +00:00
|
|
|
const { isReady, asPath } = useRouter()
|
|
|
|
const [highlighted, setHighlighted] = useState(false)
|
2022-09-22 19:58:40 +00:00
|
|
|
const commentRef = useRef<HTMLDivElement>(null)
|
|
|
|
|
2022-05-18 14:42:13 +00:00
|
|
|
useEffect(() => {
|
2022-10-07 19:40:38 +00:00
|
|
|
if (isReady && asPath.endsWith(`#${comment.id}`)) {
|
|
|
|
setHighlighted(true)
|
|
|
|
}
|
|
|
|
}, [isReady, asPath, comment.id])
|
|
|
|
|
2022-10-12 18:05:58 +00:00
|
|
|
useEffect(() => {
|
|
|
|
if (highlighted && commentRef.current) {
|
|
|
|
commentRef.current.scrollIntoView(true)
|
|
|
|
}
|
|
|
|
}, [highlighted])
|
|
|
|
return (
|
|
|
|
<Row
|
|
|
|
ref={commentRef}
|
|
|
|
id={comment.id}
|
|
|
|
className={clsx(
|
|
|
|
'hover:bg-greyscale-1 ml-3 gap-2 transition-colors',
|
|
|
|
highlighted ? `-m-1.5 rounded bg-indigo-500/[0.2] p-1.5` : ''
|
|
|
|
)}
|
|
|
|
>
|
|
|
|
<Col className="-ml-3.5">
|
|
|
|
<Avatar size="sm" username={userUsername} avatarUrl={userAvatarUrl} />
|
|
|
|
</Col>
|
|
|
|
<Col className="w-full">
|
|
|
|
<FeedCommentHeader comment={comment} contract={contract} />
|
|
|
|
<Content
|
|
|
|
className="text-greyscale-7 mt-2 grow text-[14px]"
|
|
|
|
content={content || text}
|
|
|
|
smallImage
|
|
|
|
/>
|
|
|
|
<Row className="justify-between">
|
|
|
|
<ReplyToggle
|
|
|
|
seeReplies={seeReplies}
|
|
|
|
numComments={numComments}
|
|
|
|
onClick={onSeeReplyClick}
|
|
|
|
/>
|
|
|
|
<Row className="grow justify-end gap-2">
|
|
|
|
{onReplyClick && (
|
|
|
|
<Button
|
|
|
|
size={'sm'}
|
|
|
|
className={clsx(
|
|
|
|
'hover:bg-greyscale-2 mt-0 mb-1 max-w-xs px-0 py-0'
|
|
|
|
)}
|
|
|
|
color={'gray-white'}
|
|
|
|
onClick={() => onReplyClick(comment)}
|
|
|
|
>
|
|
|
|
<ReplyIcon className="h-5 w-5" />
|
|
|
|
</Button>
|
|
|
|
)}
|
|
|
|
{showTip && (
|
|
|
|
<Tipper
|
|
|
|
comment={comment}
|
|
|
|
myTip={myTip ?? 0}
|
|
|
|
totalTip={totalTip ?? 0}
|
|
|
|
/>
|
|
|
|
)}
|
|
|
|
{(contract.openCommentBounties ?? 0) > 0 && (
|
|
|
|
<AwardBountyButton comment={comment} contract={contract} />
|
|
|
|
)}
|
|
|
|
</Row>
|
|
|
|
</Row>
|
|
|
|
</Col>
|
|
|
|
</Row>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
export const FeedComment = memo(function FeedComment(props: {
|
|
|
|
contract: Contract
|
|
|
|
comment: ContractComment
|
|
|
|
showTip?: boolean
|
|
|
|
myTip?: number
|
|
|
|
totalTip?: number
|
|
|
|
onReplyClick?: (comment: ContractComment) => void
|
|
|
|
}) {
|
|
|
|
const { contract, comment, myTip, totalTip, showTip, onReplyClick } = props
|
|
|
|
const { text, content, userUsername, userAvatarUrl } = comment
|
|
|
|
const { isReady, asPath } = useRouter()
|
|
|
|
const [highlighted, setHighlighted] = useState(false)
|
|
|
|
const commentRef = useRef<HTMLDivElement>(null)
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
if (isReady && asPath.endsWith(`#${comment.id}`)) {
|
|
|
|
setHighlighted(true)
|
|
|
|
}
|
|
|
|
}, [isReady, asPath, comment.id])
|
|
|
|
|
2022-10-07 19:40:38 +00:00
|
|
|
useEffect(() => {
|
|
|
|
if (highlighted && commentRef.current) {
|
2022-09-22 19:58:40 +00:00
|
|
|
commentRef.current.scrollIntoView(true)
|
2022-05-18 14:42:13 +00:00
|
|
|
}
|
2022-09-22 19:58:40 +00:00
|
|
|
}, [highlighted])
|
2022-05-18 14:42:13 +00:00
|
|
|
|
|
|
|
return (
|
|
|
|
<Row
|
2022-09-22 19:58:40 +00:00
|
|
|
ref={commentRef}
|
2022-08-30 09:41:47 +00:00
|
|
|
id={comment.id}
|
2022-05-18 14:42:13 +00:00
|
|
|
className={clsx(
|
2022-10-12 18:05:58 +00:00
|
|
|
'hover:bg-greyscale-1 ml-10 gap-2 transition-colors',
|
|
|
|
highlighted ? `-m-1.5 rounded bg-indigo-500/[0.2] p-1.5` : ''
|
2022-05-18 14:42:13 +00:00
|
|
|
)}
|
|
|
|
>
|
2022-10-12 18:05:58 +00:00
|
|
|
<Col className="-ml-3">
|
|
|
|
<Avatar size="xs" username={userUsername} avatarUrl={userAvatarUrl} />
|
2022-08-30 09:41:47 +00:00
|
|
|
<span
|
2022-10-12 18:05:58 +00:00
|
|
|
className="bg-greyscale-3 mx-auto h-full w-[1.5px]"
|
2022-08-30 09:41:47 +00:00
|
|
|
aria-hidden="true"
|
|
|
|
/>
|
2022-10-12 18:05:58 +00:00
|
|
|
</Col>
|
|
|
|
<Col className="w-full">
|
|
|
|
<FeedCommentHeader comment={comment} contract={contract} />
|
2022-08-30 09:41:47 +00:00
|
|
|
<Content
|
2022-10-12 18:05:58 +00:00
|
|
|
className="text-greyscale-7 mt-2 grow text-[14px]"
|
2022-08-30 09:41:47 +00:00
|
|
|
content={content || text}
|
|
|
|
smallImage
|
|
|
|
/>
|
2022-10-12 18:05:58 +00:00
|
|
|
<Row className="grow justify-end gap-2">
|
2022-06-18 03:28:16 +00:00
|
|
|
{onReplyClick && (
|
2022-10-12 18:05:58 +00:00
|
|
|
<Button
|
|
|
|
size={'sm'}
|
|
|
|
className={clsx(
|
|
|
|
'hover:bg-greyscale-2 mt-0 mb-1 max-w-xs px-0 py-0'
|
|
|
|
)}
|
|
|
|
color={'gray-white'}
|
2022-10-05 06:16:56 +00:00
|
|
|
onClick={() => onReplyClick(comment)}
|
2022-06-18 03:28:16 +00:00
|
|
|
>
|
2022-10-12 18:05:58 +00:00
|
|
|
<ReplyIcon className="h-5 w-5" />
|
|
|
|
</Button>
|
2022-06-18 03:28:16 +00:00
|
|
|
)}
|
2022-10-05 06:16:56 +00:00
|
|
|
{showTip && (
|
|
|
|
<Tipper
|
|
|
|
comment={comment}
|
|
|
|
myTip={myTip ?? 0}
|
|
|
|
totalTip={totalTip ?? 0}
|
|
|
|
/>
|
|
|
|
)}
|
2022-10-01 20:51:08 +00:00
|
|
|
{(contract.openCommentBounties ?? 0) > 0 && (
|
|
|
|
<AwardBountyButton comment={comment} contract={contract} />
|
|
|
|
)}
|
2022-06-18 03:28:16 +00:00
|
|
|
</Row>
|
2022-10-12 18:05:58 +00:00
|
|
|
</Col>
|
2022-05-18 14:42:13 +00:00
|
|
|
</Row>
|
|
|
|
)
|
2022-10-05 06:16:56 +00:00
|
|
|
})
|
2022-05-17 15:55:26 +00:00
|
|
|
|
2022-05-19 01:38:02 +00:00
|
|
|
function CommentStatus(props: {
|
|
|
|
contract: Contract
|
|
|
|
outcome: string
|
|
|
|
prob?: number
|
|
|
|
}) {
|
|
|
|
const { contract, outcome, prob } = props
|
2022-05-18 14:42:13 +00:00
|
|
|
return (
|
|
|
|
<>
|
2022-09-18 00:10:34 +00:00
|
|
|
{` predicting `}
|
2022-05-18 14:42:13 +00:00
|
|
|
<OutcomeLabel outcome={outcome} contract={contract} truncate="short" />
|
2022-05-19 01:38:02 +00:00
|
|
|
{prob && ' at ' + Math.round(prob * 100) + '%'}
|
2022-05-18 14:42:13 +00:00
|
|
|
</>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2022-09-07 22:09:20 +00:00
|
|
|
export function ContractCommentInput(props: {
|
2022-05-18 14:42:13 +00:00
|
|
|
contract: Contract
|
2022-08-30 09:41:47 +00:00
|
|
|
className?: string
|
2022-09-07 22:09:20 +00:00
|
|
|
parentAnswerOutcome?: string | undefined
|
2022-09-22 19:40:44 +00:00
|
|
|
replyTo?: ReplyTo
|
2022-06-08 13:24:12 +00:00
|
|
|
parentCommentId?: string
|
|
|
|
onSubmitComment?: () => void
|
2022-05-18 14:42:13 +00:00
|
|
|
}) {
|
|
|
|
const user = useUser()
|
2022-09-30 15:27:42 +00:00
|
|
|
const { contract, parentAnswerOutcome, parentCommentId, replyTo, className } =
|
|
|
|
props
|
|
|
|
const { openCommentBounties } = contract
|
2022-09-20 22:25:58 +00:00
|
|
|
async function onSubmitComment(editor: Editor) {
|
2022-05-18 14:42:13 +00:00
|
|
|
if (!user) {
|
2022-06-15 21:34:34 +00:00
|
|
|
track('sign in to comment')
|
2022-05-18 14:42:13 +00:00
|
|
|
return await firebaseLogin()
|
|
|
|
}
|
2022-06-22 16:35:50 +00:00
|
|
|
await createCommentOnContract(
|
2022-09-30 15:27:42 +00:00
|
|
|
contract.id,
|
2022-08-06 20:39:52 +00:00
|
|
|
editor.getJSON(),
|
2022-05-18 14:42:13 +00:00
|
|
|
user,
|
2022-09-30 15:27:42 +00:00
|
|
|
!!openCommentBounties,
|
|
|
|
parentAnswerOutcome,
|
|
|
|
parentCommentId
|
2022-05-18 14:42:13 +00:00
|
|
|
)
|
2022-09-07 22:09:20 +00:00
|
|
|
props.onSubmitComment?.()
|
2022-05-18 14:42:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
2022-09-12 21:50:31 +00:00
|
|
|
<CommentInput
|
2022-09-30 15:27:42 +00:00
|
|
|
replyTo={replyTo}
|
|
|
|
parentAnswerOutcome={parentAnswerOutcome}
|
|
|
|
parentCommentId={parentCommentId}
|
2022-09-12 21:50:31 +00:00
|
|
|
onSubmitComment={onSubmitComment}
|
2022-10-11 19:52:27 +00:00
|
|
|
pageId={contract.id}
|
2022-09-30 15:27:42 +00:00
|
|
|
className={className}
|
2022-09-12 21:50:31 +00:00
|
|
|
/>
|
2022-05-18 14:42:13 +00:00
|
|
|
)
|
|
|
|
}
|
2022-10-12 18:05:58 +00:00
|
|
|
|
|
|
|
export function FeedCommentHeader(props: {
|
|
|
|
comment: ContractComment
|
|
|
|
contract: Contract
|
|
|
|
}) {
|
|
|
|
const { comment, contract } = props
|
|
|
|
const {
|
|
|
|
userUsername,
|
|
|
|
userName,
|
|
|
|
commenterPositionProb,
|
|
|
|
commenterPositionShares,
|
|
|
|
commenterPositionOutcome,
|
|
|
|
createdTime,
|
|
|
|
bountiesAwarded,
|
|
|
|
} = comment
|
|
|
|
const betOutcome = comment.betOutcome
|
|
|
|
let bought: string | undefined
|
|
|
|
let money: string | undefined
|
|
|
|
if (comment.betAmount != null) {
|
|
|
|
bought = comment.betAmount >= 0 ? 'bought' : 'sold'
|
|
|
|
money = formatMoney(Math.abs(comment.betAmount))
|
|
|
|
}
|
|
|
|
const totalAwarded = bountiesAwarded ?? 0
|
|
|
|
return (
|
|
|
|
<Row>
|
|
|
|
<div className="text-greyscale-6 mt-0.5 text-xs">
|
|
|
|
<UserLink username={userUsername} name={userName} />{' '}
|
|
|
|
<span className="text-greyscale-4">
|
|
|
|
{comment.betId == null &&
|
|
|
|
commenterPositionProb != null &&
|
|
|
|
commenterPositionOutcome != null &&
|
|
|
|
commenterPositionShares != null &&
|
|
|
|
commenterPositionShares > 0 &&
|
|
|
|
contract.outcomeType !== 'NUMERIC' && (
|
|
|
|
<>
|
|
|
|
{'is '}
|
|
|
|
<CommentStatus
|
|
|
|
prob={commenterPositionProb}
|
|
|
|
outcome={commenterPositionOutcome}
|
|
|
|
contract={contract}
|
|
|
|
/>
|
|
|
|
</>
|
|
|
|
)}
|
|
|
|
{bought} {money}
|
|
|
|
{contract.outcomeType !== 'FREE_RESPONSE' && betOutcome && (
|
|
|
|
<>
|
|
|
|
{' '}
|
|
|
|
of{' '}
|
|
|
|
<OutcomeLabel
|
|
|
|
outcome={betOutcome ? betOutcome : ''}
|
|
|
|
contract={contract}
|
|
|
|
truncate="short"
|
|
|
|
/>
|
|
|
|
</>
|
|
|
|
)}
|
|
|
|
</span>
|
|
|
|
<CopyLinkDateTimeComponent
|
|
|
|
prefix={contract.creatorUsername}
|
|
|
|
slug={contract.slug}
|
|
|
|
createdTime={createdTime}
|
|
|
|
elementId={comment.id}
|
|
|
|
/>
|
|
|
|
{totalAwarded > 0 && (
|
|
|
|
<span className=" text-primary ml-2 text-sm">
|
|
|
|
+{formatMoney(totalAwarded)}
|
|
|
|
</span>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
</Row>
|
|
|
|
)
|
|
|
|
}
|