Revert "Switch comments/chat to rich text editor (#703)"

This reverts commit f52da72115.
This commit is contained in:
James Grugett 2022-08-04 16:49:59 -07:00
parent f52da72115
commit 33906adfe4
12 changed files with 266 additions and 196 deletions

View File

@ -1,5 +1,3 @@
import type { JSONContent } from '@tiptap/core'
// Currently, comments are created after the bet, not atomically with the bet. // Currently, comments are created after the bet, not atomically with the bet.
// They're uniquely identified by the pair contractId/betId. // They're uniquely identified by the pair contractId/betId.
export type Comment = { export type Comment = {
@ -11,9 +9,7 @@ export type Comment = {
replyToCommentId?: string replyToCommentId?: string
userId: string userId: string
/** @deprecated - content now stored as JSON in content*/ text: string
text?: string
content: JSONContent
createdTime: number createdTime: number
// Denormalized, for rendering comments // Denormalized, for rendering comments

View File

@ -7,7 +7,7 @@ import {
} from '../../common/notification' } from '../../common/notification'
import { User } from '../../common/user' import { User } from '../../common/user'
import { Contract } from '../../common/contract' import { Contract } from '../../common/contract'
import { getValues } from './utils' import { getUserByUsername, getValues } from './utils'
import { Comment } from '../../common/comment' import { Comment } from '../../common/comment'
import { uniq } from 'lodash' import { uniq } from 'lodash'
import { Bet, LimitBet } from '../../common/bet' import { Bet, LimitBet } from '../../common/bet'
@ -17,7 +17,6 @@ import { removeUndefinedProps } from '../../common/util/object'
import { TipTxn } from '../../common/txn' import { TipTxn } from '../../common/txn'
import { Group, GROUP_CHAT_SLUG } from '../../common/group' import { Group, GROUP_CHAT_SLUG } from '../../common/group'
import { Challenge } from '../../common/challenge' import { Challenge } from '../../common/challenge'
import { richTextToString } from 'common/util/parse'
const firestore = admin.firestore() const firestore = admin.firestore()
type user_to_reason_texts = { type user_to_reason_texts = {
@ -156,6 +155,17 @@ export const createNotification = async (
} }
} }
/** @deprecated parse from rich text instead */
const parseMentions = async (source: string) => {
const mentions = source.match(/@\w+/g)
if (!mentions) return []
return Promise.all(
mentions.map(
async (username) => (await getUserByUsername(username.slice(1)))?.id
)
)
}
const notifyTaggedUsers = ( const notifyTaggedUsers = (
userToReasonTexts: user_to_reason_texts, userToReasonTexts: user_to_reason_texts,
userIds: (string | undefined)[] userIds: (string | undefined)[]
@ -291,7 +301,8 @@ export const createNotification = async (
if (sourceType === 'comment') { if (sourceType === 'comment') {
if (recipients?.[0] && relatedSourceType) if (recipients?.[0] && relatedSourceType)
notifyRepliedUser(userToReasonTexts, recipients[0], relatedSourceType) notifyRepliedUser(userToReasonTexts, recipients[0], relatedSourceType)
if (sourceText) notifyTaggedUsers(userToReasonTexts, recipients ?? []) if (sourceText)
notifyTaggedUsers(userToReasonTexts, await parseMentions(sourceText))
} }
await notifyContractCreator(userToReasonTexts, sourceContract) await notifyContractCreator(userToReasonTexts, sourceContract)
await notifyOtherAnswerersOnContract(userToReasonTexts, sourceContract) await notifyOtherAnswerersOnContract(userToReasonTexts, sourceContract)
@ -416,7 +427,7 @@ export const createGroupCommentNotification = async (
sourceUserName: fromUser.name, sourceUserName: fromUser.name,
sourceUserUsername: fromUser.username, sourceUserUsername: fromUser.username,
sourceUserAvatarUrl: fromUser.avatarUrl, sourceUserAvatarUrl: fromUser.avatarUrl,
sourceText: richTextToString(comment.content), sourceText: comment.text,
sourceSlug, sourceSlug,
sourceTitle: `${group.name}`, sourceTitle: `${group.name}`,
isSeenOnHref: sourceSlug, isSeenOnHref: sourceSlug,

View File

@ -17,7 +17,6 @@ import { formatNumericProbability } from '../../common/pseudo-numeric'
import { sendTemplateEmail } from './send-email' import { sendTemplateEmail } from './send-email'
import { getPrivateUser, getUser } from './utils' import { getPrivateUser, getUser } from './utils'
import { getFunctionUrl } from '../../common/api' import { getFunctionUrl } from '../../common/api'
import { richTextToString } from 'common/util/parse'
const UNSUBSCRIBE_ENDPOINT = getFunctionUrl('unsubscribe') const UNSUBSCRIBE_ENDPOINT = getFunctionUrl('unsubscribe')
@ -292,8 +291,7 @@ export const sendNewCommentEmail = async (
const unsubscribeUrl = `${UNSUBSCRIBE_ENDPOINT}?id=${userId}&type=${emailType}` const unsubscribeUrl = `${UNSUBSCRIBE_ENDPOINT}?id=${userId}&type=${emailType}`
const { name: commentorName, avatarUrl: commentorAvatarUrl } = commentCreator const { name: commentorName, avatarUrl: commentorAvatarUrl } = commentCreator
const { content } = comment const { text } = comment
const text = richTextToString(content)
let betDescription = '' let betDescription = ''
if (bet) { if (bet) {

View File

@ -1,13 +1,13 @@
import * as functions from 'firebase-functions' import * as functions from 'firebase-functions'
import * as admin from 'firebase-admin' import * as admin from 'firebase-admin'
import { compact, uniq } from 'lodash' import { uniq } from 'lodash'
import { getContract, getUser, getValues } from './utils' import { getContract, getUser, getValues } from './utils'
import { Comment } from '../../common/comment' import { Comment } from '../../common/comment'
import { sendNewCommentEmail } from './emails' import { sendNewCommentEmail } from './emails'
import { Bet } from '../../common/bet' import { Bet } from '../../common/bet'
import { Answer } from '../../common/answer' import { Answer } from '../../common/answer'
import { createNotification } from './create-notification' import { createNotification } from './create-notification'
import { parseMentions, richTextToString } from 'common/util/parse'
const firestore = admin.firestore() const firestore = admin.firestore()
@ -71,10 +71,7 @@ export const onCreateCommentOnContract = functions
const repliedUserId = comment.replyToCommentId const repliedUserId = comment.replyToCommentId
? comments.find((c) => c.id === comment.replyToCommentId)?.userId ? comments.find((c) => c.id === comment.replyToCommentId)?.userId
: answer?.userId : answer?.userId
const recipients = repliedUserId ? [repliedUserId] : []
const recipients = uniq(
compact([...parseMentions(comment.content), repliedUserId])
)
await createNotification( await createNotification(
comment.id, comment.id,
@ -82,7 +79,7 @@ export const onCreateCommentOnContract = functions
'created', 'created',
commentCreator, commentCreator,
eventId, eventId,
richTextToString(comment.content), comment.text,
{ contract, relatedSourceType, recipients } { contract, relatedSourceType, recipients }
) )

View File

@ -8,8 +8,8 @@ import { RelativeTimestamp } from './relative-timestamp'
import { UserLink } from './user-page' import { UserLink } from './user-page'
import { User } from 'common/user' import { User } from 'common/user'
import { Col } from './layout/col' import { Col } from './layout/col'
import { Linkify } from './linkify'
import { groupBy } from 'lodash' import { groupBy } from 'lodash'
import { Content } from './editor'
export function UserCommentsList(props: { export function UserCommentsList(props: {
user: User user: User
@ -50,8 +50,7 @@ export function UserCommentsList(props: {
function ProfileComment(props: { comment: Comment; className?: string }) { function ProfileComment(props: { comment: Comment; className?: string }) {
const { comment, className } = props const { comment, className } = props
const { text, content, userUsername, userName, userAvatarUrl, createdTime } = const { text, userUsername, userName, userAvatarUrl, createdTime } = comment
comment
// TODO: find and attach relevant bets by comment betId at some point // TODO: find and attach relevant bets by comment betId at some point
return ( return (
<Row className={className}> <Row className={className}>
@ -65,7 +64,7 @@ function ProfileComment(props: { comment: Comment; className?: string }) {
/>{' '} />{' '}
<RelativeTimestamp time={createdTime} /> <RelativeTimestamp time={createdTime} />
</p> </p>
<Content content={content || text} /> <Linkify text={text} />
</div> </div>
</Row> </Row>
) )

View File

@ -107,6 +107,7 @@ export function ContractTopTrades(props: {
comment={commentsById[topCommentId]} comment={commentsById[topCommentId]}
tips={tips[topCommentId]} tips={tips[topCommentId]}
betsBySameUser={[betsById[topCommentId]]} betsBySameUser={[betsById[topCommentId]]}
truncate={false}
smallAvatar={false} smallAvatar={false}
/> />
</div> </div>

View File

@ -41,16 +41,14 @@ export function useTextEditor(props: {
max?: number max?: number
defaultValue?: Content defaultValue?: Content
disabled?: boolean disabled?: boolean
simple?: boolean
}) { }) {
const { placeholder, max, defaultValue = '', disabled, simple } = props const { placeholder, max, defaultValue = '', disabled } = props
const users = useUsers() const users = useUsers()
const editorClass = clsx( const editorClass = clsx(
proseClass, proseClass,
!simple && 'min-h-[6em]', 'min-h-[6em] resize-none outline-none border-none pt-3 px-4 focus:ring-0'
'outline-none pt-2 px-4'
) )
const editor = useEditor( const editor = useEditor(
@ -58,8 +56,7 @@ export function useTextEditor(props: {
editorProps: { attributes: { class: editorClass } }, editorProps: { attributes: { class: editorClass } },
extensions: [ extensions: [
StarterKit.configure({ StarterKit.configure({
heading: simple ? false : { levels: [1, 2, 3] }, heading: { levels: [1, 2, 3] },
horizontalRule: simple ? false : {},
}), }),
Placeholder.configure({ Placeholder.configure({
placeholder, placeholder,
@ -123,9 +120,8 @@ function isValidIframe(text: string) {
export function TextEditor(props: { export function TextEditor(props: {
editor: Editor | null editor: Editor | null
upload: ReturnType<typeof useUploadMutation> upload: ReturnType<typeof useUploadMutation>
children?: React.ReactNode // additional toolbar buttons
}) { }) {
const { editor, upload, children } = props const { editor, upload } = props
const [iframeOpen, setIframeOpen] = useState(false) const [iframeOpen, setIframeOpen] = useState(false)
return ( return (
@ -147,10 +143,20 @@ export function TextEditor(props: {
images! images!
</FloatingMenu> </FloatingMenu>
)} )}
<div className="rounded-lg border border-gray-300 shadow-sm focus-within:border-indigo-500 focus-within:ring-1 focus-within:ring-indigo-500"> <div className="overflow-hidden rounded-lg border border-gray-300 shadow-sm focus-within:border-indigo-500 focus-within:ring-1 focus-within:ring-indigo-500">
<EditorContent editor={editor} /> <EditorContent editor={editor} />
{/* Toolbar, with buttons for images and embeds */} {/* Spacer element to match the height of the toolbar */}
<div className="flex h-9 items-center gap-5 pl-4 pr-1"> <div className="py-2" aria-hidden="true">
{/* Matches height of button in toolbar (1px border + 36px content height) */}
<div className="py-px">
<div className="h-9" />
</div>
</div>
</div>
{/* Toolbar, with buttons for image and embeds */}
<div className="absolute inset-x-0 bottom-0 flex justify-between py-2 pl-3 pr-2">
<div className="flex items-center space-x-5">
<div className="flex items-center"> <div className="flex items-center">
<FileUploadButton <FileUploadButton
onFiles={upload.mutate} onFiles={upload.mutate}
@ -175,8 +181,6 @@ export function TextEditor(props: {
<span className="sr-only">Embed an iframe</span> <span className="sr-only">Embed an iframe</span>
</button> </button>
</div> </div>
<div className="ml-auto" />
{children}
</div> </div>
</div> </div>
</div> </div>

View File

@ -31,9 +31,9 @@ export function FeedAnswerCommentGroup(props: {
const { answer, contract, comments, tips, bets, user } = props const { answer, contract, comments, tips, bets, user } = props
const { username, avatarUrl, name, text } = answer const { username, avatarUrl, name, text } = answer
const [replyToUser, setReplyToUser] = const [replyToUsername, setReplyToUsername] = useState('')
useState<Pick<User, 'id' | 'username'>>()
const [showReply, setShowReply] = useState(false) const [showReply, setShowReply] = useState(false)
const [inputRef, setInputRef] = useState<HTMLTextAreaElement | null>(null)
const [highlighted, setHighlighted] = useState(false) const [highlighted, setHighlighted] = useState(false)
const router = useRouter() const router = useRouter()
@ -70,14 +70,9 @@ export function FeedAnswerCommentGroup(props: {
const scrollAndOpenReplyInput = useEvent( const scrollAndOpenReplyInput = useEvent(
(comment?: Comment, answer?: Answer) => { (comment?: Comment, answer?: Answer) => {
setReplyToUser( setReplyToUsername(comment?.userUsername ?? answer?.username ?? '')
comment
? { id: comment.userId, username: comment.userUsername }
: answer
? { id: answer.userId, username: answer.username }
: undefined
)
setShowReply(true) setShowReply(true)
inputRef?.focus()
} }
) )
@ -85,7 +80,7 @@ export function FeedAnswerCommentGroup(props: {
// Only show one comment input for a bet at a time // Only show one comment input for a bet at a time
if ( if (
betsByCurrentUser.length > 1 && betsByCurrentUser.length > 1 &&
// inputRef?.textContent?.length === 0 && //TODO: editor.isEmpty inputRef?.textContent?.length === 0 &&
betsByCurrentUser.sort((a, b) => b.createdTime - a.createdTime)[0] betsByCurrentUser.sort((a, b) => b.createdTime - a.createdTime)[0]
?.outcome !== answer.number.toString() ?.outcome !== answer.number.toString()
) )
@ -94,6 +89,10 @@ export function FeedAnswerCommentGroup(props: {
// eslint-disable-next-line react-hooks/exhaustive-deps // eslint-disable-next-line react-hooks/exhaustive-deps
}, [betsByCurrentUser.length, user, answer.number]) }, [betsByCurrentUser.length, user, answer.number])
useEffect(() => {
if (showReply && inputRef) inputRef.focus()
}, [inputRef, showReply])
useEffect(() => { useEffect(() => {
if (router.asPath.endsWith(`#${answerElementId}`)) { if (router.asPath.endsWith(`#${answerElementId}`)) {
setHighlighted(true) setHighlighted(true)
@ -155,6 +154,7 @@ export function FeedAnswerCommentGroup(props: {
commentsList={commentsList} commentsList={commentsList}
betsByUserId={betsByUserId} betsByUserId={betsByUserId}
smallAvatar={true} smallAvatar={true}
truncate={false}
bets={bets} bets={bets}
tips={tips} tips={tips}
scrollAndOpenReplyInput={scrollAndOpenReplyInput} scrollAndOpenReplyInput={scrollAndOpenReplyInput}
@ -172,8 +172,12 @@ export function FeedAnswerCommentGroup(props: {
betsByCurrentUser={betsByCurrentUser} betsByCurrentUser={betsByCurrentUser}
commentsByCurrentUser={commentsByCurrentUser} commentsByCurrentUser={commentsByCurrentUser}
parentAnswerOutcome={answer.number.toString()} parentAnswerOutcome={answer.number.toString()}
replyToUser={replyToUser} replyToUsername={replyToUsername}
onSubmitComment={() => setShowReply(false)} setRef={setInputRef}
onSubmitComment={() => {
setShowReply(false)
setReplyToUsername('')
}}
/> />
</div> </div>
)} )}

View File

@ -13,22 +13,25 @@ import { Avatar } from 'web/components/avatar'
import { UserLink } from 'web/components/user-page' import { UserLink } from 'web/components/user-page'
import { OutcomeLabel } from 'web/components/outcome-label' import { OutcomeLabel } from 'web/components/outcome-label'
import { CopyLinkDateTimeComponent } from 'web/components/feed/copy-link-date-time' import { CopyLinkDateTimeComponent } from 'web/components/feed/copy-link-date-time'
import { contractPath } from 'web/lib/firebase/contracts'
import { firebaseLogin } from 'web/lib/firebase/users' import { firebaseLogin } from 'web/lib/firebase/users'
import { import {
createCommentOnContract, createCommentOnContract,
MAX_COMMENT_LENGTH, MAX_COMMENT_LENGTH,
} from 'web/lib/firebase/comments' } from 'web/lib/firebase/comments'
import Textarea from 'react-expanding-textarea'
import { Linkify } from 'web/components/linkify'
import { SiteLink } from 'web/components/site-link'
import { BetStatusText } from 'web/components/feed/feed-bets' import { BetStatusText } from 'web/components/feed/feed-bets'
import { Col } from 'web/components/layout/col' import { Col } from 'web/components/layout/col'
import { getProbability } from 'common/calculate' import { getProbability } from 'common/calculate'
import { LoadingIndicator } from 'web/components/loading-indicator' import { LoadingIndicator } from 'web/components/loading-indicator'
import { PaperAirplaneIcon } from '@heroicons/react/outline' import { PaperAirplaneIcon } from '@heroicons/react/outline'
import { track } from 'web/lib/service/analytics' import { track } from 'web/lib/service/analytics'
import { useEvent } from 'web/hooks/use-event'
import { Tipper } from '../tipper' import { Tipper } from '../tipper'
import { CommentTipMap, CommentTips } from 'web/hooks/use-tip-txns' import { CommentTipMap, CommentTips } from 'web/hooks/use-tip-txns'
import { useWindowSize } from 'web/hooks/use-window-size' import { useWindowSize } from 'web/hooks/use-window-size'
import { Content, TextEditor, useTextEditor } from '../editor'
import { Editor } from '@tiptap/react'
export function FeedCommentThread(props: { export function FeedCommentThread(props: {
contract: Contract contract: Contract
@ -36,12 +39,20 @@ export function FeedCommentThread(props: {
tips: CommentTipMap tips: CommentTipMap
parentComment: Comment parentComment: Comment
bets: Bet[] bets: Bet[]
truncate?: boolean
smallAvatar?: boolean smallAvatar?: boolean
}) { }) {
const { contract, comments, bets, tips, smallAvatar, parentComment } = props const {
contract,
comments,
bets,
tips,
truncate,
smallAvatar,
parentComment,
} = props
const [showReply, setShowReply] = useState(false) const [showReply, setShowReply] = useState(false)
const [replyToUser, setReplyToUser] = const [replyToUsername, setReplyToUsername] = useState('')
useState<{ id: string; username: string }>()
const betsByUserId = groupBy(bets, (bet) => bet.userId) const betsByUserId = groupBy(bets, (bet) => bet.userId)
const user = useUser() const user = useUser()
const commentsList = comments.filter( const commentsList = comments.filter(
@ -49,12 +60,15 @@ export function FeedCommentThread(props: {
parentComment.id && comment.replyToCommentId === parentComment.id parentComment.id && comment.replyToCommentId === parentComment.id
) )
commentsList.unshift(parentComment) commentsList.unshift(parentComment)
const [inputRef, setInputRef] = useState<HTMLTextAreaElement | null>(null)
function scrollAndOpenReplyInput(comment: Comment) { function scrollAndOpenReplyInput(comment: Comment) {
setReplyToUser({ id: comment.userId, username: comment.userUsername }) setReplyToUsername(comment.userUsername)
setShowReply(true) setShowReply(true)
inputRef?.focus()
} }
useEffect(() => {
if (showReply && inputRef) inputRef.focus()
}, [inputRef, showReply])
return ( return (
<Col className={'w-full gap-3 pr-1'}> <Col className={'w-full gap-3 pr-1'}>
<span <span
@ -67,6 +81,7 @@ export function FeedCommentThread(props: {
betsByUserId={betsByUserId} betsByUserId={betsByUserId}
tips={tips} tips={tips}
smallAvatar={smallAvatar} smallAvatar={smallAvatar}
truncate={truncate}
bets={bets} bets={bets}
scrollAndOpenReplyInput={scrollAndOpenReplyInput} scrollAndOpenReplyInput={scrollAndOpenReplyInput}
/> />
@ -83,9 +98,13 @@ export function FeedCommentThread(props: {
(c) => c.userId === user?.id (c) => c.userId === user?.id
)} )}
parentCommentId={parentComment.id} parentCommentId={parentComment.id}
replyToUser={replyToUser} replyToUsername={replyToUsername}
parentAnswerOutcome={comments[0].answerOutcome} parentAnswerOutcome={comments[0].answerOutcome}
onSubmitComment={() => setShowReply(false)} setRef={setInputRef}
onSubmitComment={() => {
setShowReply(false)
setReplyToUsername('')
}}
/> />
</Col> </Col>
)} )}
@ -102,12 +121,14 @@ export function CommentRepliesList(props: {
bets: Bet[] bets: Bet[]
treatFirstIndexEqually?: boolean treatFirstIndexEqually?: boolean
smallAvatar?: boolean smallAvatar?: boolean
truncate?: boolean
}) { }) {
const { const {
contract, contract,
commentsList, commentsList,
betsByUserId, betsByUserId,
tips, tips,
truncate,
smallAvatar, smallAvatar,
bets, bets,
scrollAndOpenReplyInput, scrollAndOpenReplyInput,
@ -147,6 +168,7 @@ export function CommentRepliesList(props: {
: undefined : undefined
} }
smallAvatar={smallAvatar} smallAvatar={smallAvatar}
truncate={truncate}
/> />
</div> </div>
))} ))}
@ -160,6 +182,7 @@ export function FeedComment(props: {
tips: CommentTips tips: CommentTips
betsBySameUser: Bet[] betsBySameUser: Bet[]
probAtCreatedTime?: number probAtCreatedTime?: number
truncate?: boolean
smallAvatar?: boolean smallAvatar?: boolean
onReplyClick?: (comment: Comment) => void onReplyClick?: (comment: Comment) => void
}) { }) {
@ -169,10 +192,10 @@ export function FeedComment(props: {
tips, tips,
betsBySameUser, betsBySameUser,
probAtCreatedTime, probAtCreatedTime,
truncate,
onReplyClick, onReplyClick,
} = props } = props
const { text, content, userUsername, userName, userAvatarUrl, createdTime } = const { text, userUsername, userName, userAvatarUrl, createdTime } = comment
comment
let betOutcome: string | undefined, let betOutcome: string | undefined,
bought: string | undefined, bought: string | undefined,
money: string | undefined money: string | undefined
@ -253,9 +276,11 @@ export function FeedComment(props: {
elementId={comment.id} elementId={comment.id}
/> />
</div> </div>
<div className="mt-2 text-[15px] text-gray-700"> <TruncatedComment
<Content content={content || text} /> comment={text}
</div> moreHref={contractPath(contract)}
shouldTruncate={truncate}
/>
<Row className="mt-2 items-center gap-6 text-xs text-gray-500"> <Row className="mt-2 items-center gap-6 text-xs text-gray-500">
<Tipper comment={comment} tips={tips ?? {}} /> <Tipper comment={comment} tips={tips ?? {}} />
{onReplyClick && ( {onReplyClick && (
@ -320,7 +345,8 @@ export function CommentInput(props: {
contract: Contract contract: Contract
betsByCurrentUser: Bet[] betsByCurrentUser: Bet[]
commentsByCurrentUser: Comment[] commentsByCurrentUser: Comment[]
replyToUser?: { id: string; username: string } replyToUsername?: string
setRef?: (ref: HTMLTextAreaElement) => void
// Reply to a free response answer // Reply to a free response answer
parentAnswerOutcome?: string parentAnswerOutcome?: string
// Reply to another comment // Reply to another comment
@ -333,18 +359,12 @@ export function CommentInput(props: {
commentsByCurrentUser, commentsByCurrentUser,
parentAnswerOutcome, parentAnswerOutcome,
parentCommentId, parentCommentId,
replyToUser, replyToUsername,
onSubmitComment, onSubmitComment,
setRef,
} = props } = props
const user = useUser() const user = useUser()
const { editor, upload } = useTextEditor({ const [comment, setComment] = useState('')
simple: true,
max: MAX_COMMENT_LENGTH,
placeholder:
!!parentCommentId || !!parentAnswerOutcome
? 'Write a reply...'
: 'Write a comment...',
})
const [isSubmitting, setIsSubmitting] = useState(false) const [isSubmitting, setIsSubmitting] = useState(false)
const mostRecentCommentableBet = getMostRecentCommentableBet( const mostRecentCommentableBet = getMostRecentCommentableBet(
@ -360,17 +380,18 @@ export function CommentInput(props: {
track('sign in to comment') track('sign in to comment')
return await firebaseLogin() return await firebaseLogin()
} }
if (!editor || editor.isEmpty || isSubmitting) return if (!comment || isSubmitting) return
setIsSubmitting(true) setIsSubmitting(true)
await createCommentOnContract( await createCommentOnContract(
contract.id, contract.id,
editor.getJSON(), comment,
user, user,
betId, betId,
parentAnswerOutcome, parentAnswerOutcome,
parentCommentId parentCommentId
) )
onSubmitComment?.() onSubmitComment?.()
setComment('')
setIsSubmitting(false) setIsSubmitting(false)
} }
@ -425,12 +446,14 @@ export function CommentInput(props: {
)} )}
</div> </div>
<CommentInputTextArea <CommentInputTextArea
editor={editor} commentText={comment}
upload={upload} setComment={setComment}
replyToUser={replyToUser} isReply={!!parentCommentId || !!parentAnswerOutcome}
replyToUsername={replyToUsername ?? ''}
user={user} user={user}
submitComment={submitComment} submitComment={submitComment}
isSubmitting={isSubmitting} isSubmitting={isSubmitting}
setRef={setRef}
presetId={id} presetId={id}
/> />
</div> </div>
@ -442,89 +465,94 @@ export function CommentInput(props: {
export function CommentInputTextArea(props: { export function CommentInputTextArea(props: {
user: User | undefined | null user: User | undefined | null
replyToUser?: { id: string; username: string } isReply: boolean
editor: Editor | null replyToUsername: string
upload: Parameters<typeof TextEditor>[0]['upload'] commentText: string
setComment: (text: string) => void
submitComment: (id?: string) => void submitComment: (id?: string) => void
isSubmitting: boolean isSubmitting: boolean
submitOnEnter?: boolean setRef?: (ref: HTMLTextAreaElement) => void
presetId?: string presetId?: string
enterToSubmitOnDesktop?: boolean
}) { }) {
const { const {
isReply,
setRef,
user, user,
editor, commentText,
upload, setComment,
submitComment, submitComment,
presetId, presetId,
isSubmitting, isSubmitting,
submitOnEnter, replyToUsername,
replyToUser, enterToSubmitOnDesktop,
} = props } = props
const isMobile = (useWindowSize().width ?? 0) < 768 // TODO: base off input device (keybord vs touch) const { width } = useWindowSize()
const memoizedSetComment = useEvent(setComment)
useEffect(() => { useEffect(() => {
editor?.setEditable(!isSubmitting) if (!replyToUsername || !user || replyToUsername === user.username) return
}, [isSubmitting, editor]) const replacement = `@${replyToUsername} `
memoizedSetComment(replacement + commentText.replace(replacement, ''))
const submit = () => {
submitComment(presetId)
editor?.commands?.clearContent()
}
useEffect(() => {
if (!editor) {
return
}
// submit on Enter key
editor.setOptions({
editorProps: {
handleKeyDown: (view, event) => {
if (
submitOnEnter &&
event.key === 'Enter' &&
!event.shiftKey &&
(!isMobile || event.ctrlKey || event.metaKey) &&
// mention list is closed
!(view.state as any).mention$.active
) {
submit()
event.preventDefault()
return true
}
return false
},
},
})
// insert at mention
if (replyToUser) {
editor.commands.insertContentAt(0, {
type: 'mention',
attrs: { label: replyToUser.username, id: replyToUser.id },
})
editor.commands.focus()
}
// eslint-disable-next-line react-hooks/exhaustive-deps // eslint-disable-next-line react-hooks/exhaustive-deps
}, [editor]) }, [user, replyToUsername, memoizedSetComment])
return ( return (
<> <>
<div> <Row className="gap-1.5 text-gray-700">
<TextEditor editor={editor} upload={upload}> <Textarea
ref={setRef}
value={commentText}
onChange={(e) => setComment(e.target.value)}
className={clsx('textarea textarea-bordered w-full resize-none')}
// Make room for floating submit button.
style={{ paddingRight: 48 }}
placeholder={
isReply
? 'Write a reply... '
: enterToSubmitOnDesktop
? 'Send a message'
: 'Write a comment...'
}
autoFocus={false}
maxLength={MAX_COMMENT_LENGTH}
disabled={isSubmitting}
onKeyDown={(e) => {
if (
(enterToSubmitOnDesktop &&
e.key === 'Enter' &&
!e.shiftKey &&
width &&
width > 768) ||
(e.key === 'Enter' && (e.ctrlKey || e.metaKey))
) {
e.preventDefault()
submitComment(presetId)
e.currentTarget.blur()
}
}}
/>
<Col className={clsx('relative justify-end')}>
{user && !isSubmitting && ( {user && !isSubmitting && (
<button <button
className="btn btn-ghost btn-sm px-2 disabled:bg-inherit disabled:text-gray-300" className={clsx(
disabled={!editor || editor.isEmpty} 'btn btn-ghost btn-sm absolute right-2 bottom-2 flex-row pl-2 capitalize',
onClick={submit} !commentText && 'pointer-events-none text-gray-500'
)}
onClick={() => {
submitComment(presetId)
}}
> >
<PaperAirplaneIcon className="m-0 h-[25px] min-w-[22px] rotate-90 p-0" /> <PaperAirplaneIcon
className={'m-0 min-w-[22px] rotate-90 p-0 '}
height={25}
/>
</button> </button>
)} )}
{isSubmitting && ( {isSubmitting && (
<LoadingIndicator spinnerClassName={'border-gray-500'} /> <LoadingIndicator spinnerClassName={'border-gray-500'} />
)} )}
</TextEditor> </Col>
</div> </Row>
<Row> <Row>
{!user && ( {!user && (
<button <button
@ -539,6 +567,38 @@ export function CommentInputTextArea(props: {
) )
} }
export function TruncatedComment(props: {
comment: string
moreHref: string
shouldTruncate?: boolean
}) {
const { comment, moreHref, shouldTruncate } = props
let truncated = comment
// Keep descriptions to at most 400 characters
const MAX_CHARS = 400
if (shouldTruncate && truncated.length > MAX_CHARS) {
truncated = truncated.slice(0, MAX_CHARS)
// Make sure to end on a space
const i = truncated.lastIndexOf(' ')
truncated = truncated.slice(0, i)
}
return (
<div
className="mt-2 whitespace-pre-line break-words text-gray-700"
style={{ fontSize: 15 }}
>
<Linkify text={truncated} />
{truncated != comment && (
<SiteLink href={moreHref} className="text-indigo-700">
... (show more)
</SiteLink>
)}
</div>
)
}
function getBettorsLargestPositionBeforeTime( function getBettorsLargestPositionBeforeTime(
contract: Contract, contract: Contract,
createdTime: number, createdTime: number,

View File

@ -5,19 +5,24 @@ import React, { useEffect, memo, useState, useMemo } from 'react'
import { Avatar } from 'web/components/avatar' import { Avatar } from 'web/components/avatar'
import { Group } from 'common/group' import { Group } from 'common/group'
import { Comment, createCommentOnGroup } from 'web/lib/firebase/comments' import { Comment, createCommentOnGroup } from 'web/lib/firebase/comments'
import { CommentInputTextArea } from 'web/components/feed/feed-comments' import {
CommentInputTextArea,
TruncatedComment,
} from 'web/components/feed/feed-comments'
import { track } from 'web/lib/service/analytics' import { track } from 'web/lib/service/analytics'
import { firebaseLogin } from 'web/lib/firebase/users' import { firebaseLogin } from 'web/lib/firebase/users'
import { useRouter } from 'next/router' import { useRouter } from 'next/router'
import clsx from 'clsx' import clsx from 'clsx'
import { UserLink } from 'web/components/user-page' import { UserLink } from 'web/components/user-page'
import { groupPath } from 'web/lib/firebase/groups'
import { CopyLinkDateTimeComponent } from 'web/components/feed/copy-link-date-time' import { CopyLinkDateTimeComponent } from 'web/components/feed/copy-link-date-time'
import { CommentTipMap, CommentTips } from 'web/hooks/use-tip-txns' import { CommentTipMap, CommentTips } from 'web/hooks/use-tip-txns'
import { Tipper } from 'web/components/tipper' import { Tipper } from 'web/components/tipper'
import { sum } from 'lodash' import { sum } from 'lodash'
import { formatMoney } from 'common/util/format' import { formatMoney } from 'common/util/format'
import { useWindowSize } from 'web/hooks/use-window-size' import { useWindowSize } from 'web/hooks/use-window-size'
import { Content, useTextEditor } from 'web/components/editor'
import { useUnseenPreferredNotifications } from 'web/hooks/use-notifications' import { useUnseenPreferredNotifications } from 'web/hooks/use-notifications'
import { ChatIcon, ChevronDownIcon } from '@heroicons/react/outline' import { ChatIcon, ChevronDownIcon } from '@heroicons/react/outline'
import { setNotificationsAsSeen } from 'web/pages/notifications' import { setNotificationsAsSeen } from 'web/pages/notifications'
@ -29,18 +34,16 @@ export function GroupChat(props: {
tips: CommentTipMap tips: CommentTipMap
}) { }) {
const { messages, user, group, tips } = props const { messages, user, group, tips } = props
const { editor, upload } = useTextEditor({ const [messageText, setMessageText] = useState('')
simple: true,
placeholder: 'Send a message',
})
const [isSubmitting, setIsSubmitting] = useState(false) const [isSubmitting, setIsSubmitting] = useState(false)
const [scrollToBottomRef, setScrollToBottomRef] = const [scrollToBottomRef, setScrollToBottomRef] =
useState<HTMLDivElement | null>(null) useState<HTMLDivElement | null>(null)
const [scrollToMessageId, setScrollToMessageId] = useState('') const [scrollToMessageId, setScrollToMessageId] = useState('')
const [scrollToMessageRef, setScrollToMessageRef] = const [scrollToMessageRef, setScrollToMessageRef] =
useState<HTMLDivElement | null>(null) useState<HTMLDivElement | null>(null)
const [replyToUser, setReplyToUser] = useState<any>() const [replyToUsername, setReplyToUsername] = useState('')
const [inputRef, setInputRef] = useState<HTMLTextAreaElement | null>(null)
const [groupedMessages, setGroupedMessages] = useState<Comment[]>([])
const router = useRouter() const router = useRouter()
const isMember = user && group.memberIds.includes(user?.id) const isMember = user && group.memberIds.includes(user?.id)
@ -51,26 +54,25 @@ export function GroupChat(props: {
const remainingHeight = const remainingHeight =
(height ?? 0) - (containerRef?.offsetTop ?? 0) - bottomBarHeight (height ?? 0) - (containerRef?.offsetTop ?? 0) - bottomBarHeight
// array of groups, where each group is an array of messages that are displayed as one useMemo(() => {
const groupedMessages = useMemo(() => {
// Group messages with createdTime within 2 minutes of each other. // Group messages with createdTime within 2 minutes of each other.
const tempGrouped: Comment[][] = [] const tempMessages = []
for (let i = 0; i < messages.length; i++) { for (let i = 0; i < messages.length; i++) {
const message = messages[i] const message = messages[i]
if (i === 0) tempGrouped.push([message]) if (i === 0) tempMessages.push({ ...message })
else { else {
const prevMessage = messages[i - 1] const prevMessage = messages[i - 1]
const diff = message.createdTime - prevMessage.createdTime const diff = message.createdTime - prevMessage.createdTime
const creatorsMatch = message.userId === prevMessage.userId const creatorsMatch = message.userId === prevMessage.userId
if (diff < 2 * 60 * 1000 && creatorsMatch) { if (diff < 2 * 60 * 1000 && creatorsMatch) {
tempGrouped.at(-1)?.push(message) tempMessages[tempMessages.length - 1].text += `\n${message.text}`
} else { } else {
tempGrouped.push([message]) tempMessages.push({ ...message })
} }
} }
} }
return tempGrouped setGroupedMessages(tempMessages)
}, [messages]) }, [messages])
useEffect(() => { useEffect(() => {
@ -92,12 +94,11 @@ export function GroupChat(props: {
useEffect(() => { useEffect(() => {
// is mobile? // is mobile?
if (width && width > 720) focusInput() if (inputRef && width && width > 720) inputRef.focus()
// eslint-disable-next-line react-hooks/exhaustive-deps }, [inputRef, width])
}, [width])
function onReplyClick(comment: Comment) { function onReplyClick(comment: Comment) {
setReplyToUser({ id: comment.userId, username: comment.userUsername }) setReplyToUsername(comment.userUsername)
} }
async function submitMessage() { async function submitMessage() {
@ -105,16 +106,13 @@ export function GroupChat(props: {
track('sign in to comment') track('sign in to comment')
return await firebaseLogin() return await firebaseLogin()
} }
if (!editor || editor.isEmpty || isSubmitting) return if (!messageText || isSubmitting) return
setIsSubmitting(true) setIsSubmitting(true)
await createCommentOnGroup(group.id, editor.getJSON(), user) await createCommentOnGroup(group.id, messageText, user)
editor.commands.clearContent() setMessageText('')
setIsSubmitting(false) setIsSubmitting(false)
setReplyToUser(undefined) setReplyToUsername('')
focusInput() inputRef?.focus()
}
function focusInput() {
editor?.commands.focus()
} }
return ( return (
@ -125,20 +123,20 @@ export function GroupChat(props: {
} }
ref={setScrollToBottomRef} ref={setScrollToBottomRef}
> >
{groupedMessages.map((messages) => ( {groupedMessages.map((message) => (
<GroupMessage <GroupMessage
user={user} user={user}
key={`group ${messages[0].id}`} key={message.id}
comments={messages} comment={message}
group={group} group={group}
onReplyClick={onReplyClick} onReplyClick={onReplyClick}
highlight={messages[0].id === scrollToMessageId} highlight={message.id === scrollToMessageId}
setRef={ setRef={
scrollToMessageId === messages[0].id scrollToMessageId === message.id
? setScrollToMessageRef ? setScrollToMessageRef
: undefined : undefined
} }
tips={tips[messages[0].id] ?? {}} tips={tips[message.id] ?? {}}
/> />
))} ))}
{messages.length === 0 && ( {messages.length === 0 && (
@ -146,7 +144,7 @@ export function GroupChat(props: {
No messages yet. Why not{isMember ? ` ` : ' join and '} No messages yet. Why not{isMember ? ` ` : ' join and '}
<button <button
className={'cursor-pointer font-bold text-gray-700'} className={'cursor-pointer font-bold text-gray-700'}
onClick={focusInput} onClick={() => inputRef?.focus()}
> >
add one? add one?
</button> </button>
@ -164,13 +162,15 @@ export function GroupChat(props: {
</div> </div>
<div className={'flex-1'}> <div className={'flex-1'}>
<CommentInputTextArea <CommentInputTextArea
editor={editor} commentText={messageText}
upload={upload} setComment={setMessageText}
isReply={false}
user={user} user={user}
replyToUser={replyToUser} replyToUsername={replyToUsername}
submitComment={submitMessage} submitComment={submitMessage}
isSubmitting={isSubmitting} isSubmitting={isSubmitting}
submitOnEnter enterToSubmitOnDesktop={true}
setRef={setInputRef}
/> />
</div> </div>
</div> </div>
@ -292,18 +292,16 @@ function GroupChatNotificationsIcon(props: {
const GroupMessage = memo(function GroupMessage_(props: { const GroupMessage = memo(function GroupMessage_(props: {
user: User | null | undefined user: User | null | undefined
comments: Comment[] comment: Comment
group: Group group: Group
onReplyClick?: (comment: Comment) => void onReplyClick?: (comment: Comment) => void
setRef?: (ref: HTMLDivElement) => void setRef?: (ref: HTMLDivElement) => void
highlight?: boolean highlight?: boolean
tips: CommentTips tips: CommentTips
}) { }) {
const { comments, onReplyClick, group, setRef, highlight, user, tips } = props const { comment, onReplyClick, group, setRef, highlight, user, tips } = props
const first = comments[0] const { text, userUsername, userName, userAvatarUrl, createdTime } = comment
const { id, userUsername, userName, userAvatarUrl, createdTime } = first const isCreatorsComment = user && comment.userId === user.id
const isCreatorsComment = user && first.userId === user.id
return ( return (
<Col <Col
ref={setRef} ref={setRef}
@ -333,21 +331,23 @@ const GroupMessage = memo(function GroupMessage_(props: {
prefix={'group'} prefix={'group'}
slug={group.slug} slug={group.slug}
createdTime={createdTime} createdTime={createdTime}
elementId={id} elementId={comment.id}
/>
</Row>
<Row className={'text-black'}>
<TruncatedComment
comment={text}
moreHref={groupPath(group.slug)}
shouldTruncate={false}
/> />
</Row> </Row>
<div className="mt-2 text-black">
{comments.map((comment) => (
<Content content={comment.content || comment.text} />
))}
</div>
<Row> <Row>
{!isCreatorsComment && onReplyClick && ( {!isCreatorsComment && onReplyClick && (
<button <button
className={ className={
'self-start py-1 text-xs font-bold text-gray-500 hover:underline' 'self-start py-1 text-xs font-bold text-gray-500 hover:underline'
} }
onClick={() => onReplyClick(first)} onClick={() => onReplyClick(comment)}
> >
Reply Reply
</button> </button>
@ -357,7 +357,7 @@ const GroupMessage = memo(function GroupMessage_(props: {
{formatMoney(sum(Object.values(tips)))} {formatMoney(sum(Object.values(tips)))}
</span> </span>
)} )}
{!isCreatorsComment && <Tipper comment={first} tips={tips} />} {!isCreatorsComment && <Tipper comment={comment} tips={tips} />}
</Row> </Row>
</Col> </Col>
) )

View File

@ -14,7 +14,6 @@ import { User } from 'common/user'
import { Comment } from 'common/comment' import { Comment } from 'common/comment'
import { removeUndefinedProps } from 'common/util/object' import { removeUndefinedProps } from 'common/util/object'
import { track } from '@amplitude/analytics-browser' import { track } from '@amplitude/analytics-browser'
import { JSONContent } from '@tiptap/react'
export type { Comment } export type { Comment }
@ -22,7 +21,7 @@ export const MAX_COMMENT_LENGTH = 10000
export async function createCommentOnContract( export async function createCommentOnContract(
contractId: string, contractId: string,
content: JSONContent, text: string,
commenter: User, commenter: User,
betId?: string, betId?: string,
answerOutcome?: string, answerOutcome?: string,
@ -35,7 +34,7 @@ export async function createCommentOnContract(
id: ref.id, id: ref.id,
contractId, contractId,
userId: commenter.id, userId: commenter.id,
content: content, text: text.slice(0, MAX_COMMENT_LENGTH),
createdTime: Date.now(), createdTime: Date.now(),
userName: commenter.name, userName: commenter.name,
userUsername: commenter.username, userUsername: commenter.username,
@ -54,7 +53,7 @@ export async function createCommentOnContract(
} }
export async function createCommentOnGroup( export async function createCommentOnGroup(
groupId: string, groupId: string,
content: JSONContent, text: string,
user: User, user: User,
replyToCommentId?: string replyToCommentId?: string
) { ) {
@ -63,7 +62,7 @@ export async function createCommentOnGroup(
id: ref.id, id: ref.id,
groupId, groupId,
userId: user.id, userId: user.id,
content: content, text: text.slice(0, MAX_COMMENT_LENGTH),
createdTime: Date.now(), createdTime: Date.now(),
userName: user.name, userName: user.name,
userUsername: user.username, userUsername: user.username,

View File

@ -354,6 +354,7 @@ function ContractTopTrades(props: {
comment={commentsById[topCommentId]} comment={commentsById[topCommentId]}
tips={tips[topCommentId]} tips={tips[topCommentId]}
betsBySameUser={[betsById[topCommentId]]} betsBySameUser={[betsById[topCommentId]]}
truncate={false}
smallAvatar={false} smallAvatar={false}
/> />
</div> </div>