Various group & mobile ux improvements
This commit is contained in:
parent
ebc4bd6bcf
commit
8ced159d9a
|
@ -1,7 +1,7 @@
|
|||
import { Row } from 'web/components/layout/row'
|
||||
import { Col } from 'web/components/layout/col'
|
||||
import { User } from 'common/user'
|
||||
import React, { useEffect, memo, useState } from 'react'
|
||||
import React, { useEffect, memo, useState, useMemo } from 'react'
|
||||
import { Avatar } from 'web/components/avatar'
|
||||
import { Group } from 'common/group'
|
||||
import { Comment, createCommentOnGroup } from 'web/lib/firebase/comments'
|
||||
|
@ -19,7 +19,7 @@ import { UserLink } from 'web/components/user-page'
|
|||
import { groupPath } from 'web/lib/firebase/groups'
|
||||
import { CopyLinkDateTimeComponent } from 'web/components/feed/copy-link-date-time'
|
||||
|
||||
export function Discussion(props: {
|
||||
export function GroupChat(props: {
|
||||
messages: Comment[]
|
||||
user: User | null | undefined
|
||||
group: Group
|
||||
|
@ -34,8 +34,30 @@ export function Discussion(props: {
|
|||
useState<HTMLDivElement | null>(null)
|
||||
const [replyToUsername, setReplyToUsername] = useState('')
|
||||
const [inputRef, setInputRef] = useState<HTMLTextAreaElement | null>(null)
|
||||
const [groupedMessages, setGroupedMessages] = useState<Comment[]>([])
|
||||
const router = useRouter()
|
||||
|
||||
useMemo(() => {
|
||||
// Group messages with createdTime within 2 minutes of each other.
|
||||
const tempMessages = []
|
||||
for (let i = 0; i < messages.length; i++) {
|
||||
const message = messages[i]
|
||||
if (i === 0) tempMessages.push({ ...message })
|
||||
else {
|
||||
const prevMessage = messages[i - 1]
|
||||
const diff = message.createdTime - prevMessage.createdTime
|
||||
const creatorsMatch = message.userId === prevMessage.userId
|
||||
if (diff < 2 * 60 * 1000 && creatorsMatch) {
|
||||
tempMessages[tempMessages.length - 1].text += `\n${message.text}`
|
||||
} else {
|
||||
tempMessages.push({ ...message })
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
setGroupedMessages(tempMessages)
|
||||
}, [messages])
|
||||
|
||||
useEffect(() => {
|
||||
scrollToMessageRef?.scrollIntoView()
|
||||
}, [scrollToMessageRef])
|
||||
|
@ -78,7 +100,7 @@ export function Discussion(props: {
|
|||
}
|
||||
ref={setScrollToBottomRef}
|
||||
>
|
||||
{messages.map((message) => (
|
||||
{groupedMessages.map((message) => (
|
||||
<GroupMessage
|
||||
user={user}
|
||||
key={message.id}
|
||||
|
@ -142,8 +164,8 @@ const GroupMessage = memo(function GroupMessage_(props: {
|
|||
<Col
|
||||
ref={setRef}
|
||||
className={clsx(
|
||||
isCreatorsComment ? 'mr-2 self-end' : ' ml-2',
|
||||
'w-fit max-w-md gap-1 space-x-3 rounded-md bg-white p-2 p-2 px-4 text-sm text-gray-500 transition-all duration-1000',
|
||||
isCreatorsComment ? 'mr-2 self-end' : '',
|
||||
'w-fit max-w-sm gap-1 space-x-3 rounded-md bg-white p-1 text-sm text-gray-500 transition-colors duration-1000 sm:max-w-md sm:p-3 sm:leading-[1.3rem]',
|
||||
highlight ? `-m-1 bg-indigo-500/[0.2] p-2` : ''
|
||||
)}
|
||||
>
|
||||
|
@ -151,8 +173,8 @@ const GroupMessage = memo(function GroupMessage_(props: {
|
|||
{!isCreatorsComment && (
|
||||
<Col>
|
||||
<Avatar
|
||||
className={'mx-2 ml-0'}
|
||||
size={'sm'}
|
||||
className={'mx-2 ml-2.5'}
|
||||
size={'xs'}
|
||||
username={userUsername}
|
||||
avatarUrl={userAvatarUrl}
|
||||
/>
|
||||
|
@ -161,7 +183,7 @@ const GroupMessage = memo(function GroupMessage_(props: {
|
|||
{!isCreatorsComment ? (
|
||||
<UserLink username={userUsername} name={userName} />
|
||||
) : (
|
||||
<span>{'You'}</span>
|
||||
<span className={'ml-2.5'}>{'You'}</span>
|
||||
)}
|
||||
<CopyLinkDateTimeComponent
|
||||
prefix={'group'}
|
|
@ -2,6 +2,7 @@ import { useEffect, useState } from 'react'
|
|||
import {
|
||||
Comment,
|
||||
listenForCommentsOnContract,
|
||||
listenForCommentsOnGroup,
|
||||
listenForRecentComments,
|
||||
} from 'web/lib/firebase/comments'
|
||||
|
||||
|
@ -14,6 +15,15 @@ export const useComments = (contractId: string) => {
|
|||
|
||||
return comments
|
||||
}
|
||||
export const useCommentsOnGroup = (groupId: string | undefined) => {
|
||||
const [comments, setComments] = useState<Comment[] | undefined>()
|
||||
|
||||
useEffect(() => {
|
||||
if (groupId) return listenForCommentsOnGroup(groupId, setComments)
|
||||
}, [groupId])
|
||||
|
||||
return comments
|
||||
}
|
||||
|
||||
export const useRecentComments = () => {
|
||||
const [recentComments, setRecentComments] = useState<Comment[] | undefined>()
|
||||
|
|
|
@ -17,7 +17,7 @@ const groupCollection = collection(db, 'groups')
|
|||
|
||||
export function groupPath(
|
||||
groupSlug: string,
|
||||
subpath?: 'edit' | 'questions' | 'details' | 'discussion'
|
||||
subpath?: 'edit' | 'questions' | 'details' | 'chat'
|
||||
) {
|
||||
return `/group/${groupSlug}${subpath ? `/${subpath}` : ''}`
|
||||
}
|
||||
|
|
|
@ -1,7 +1,6 @@
|
|||
import { take, sortBy, debounce } from 'lodash'
|
||||
|
||||
import { Group } from 'common/group'
|
||||
import { Comment } from 'common/comment'
|
||||
import { Page } from 'web/components/page'
|
||||
import { Title } from 'web/components/title'
|
||||
import { listAllBets } from 'web/lib/firebase/bets'
|
||||
|
@ -32,14 +31,14 @@ import { Tabs } from 'web/components/layout/tabs'
|
|||
import { ContractsGrid } from 'web/components/contract/contracts-list'
|
||||
import { CreateQuestionButton } from 'web/components/create-question-button'
|
||||
import React, { useEffect, useState } from 'react'
|
||||
import { Discussion } from 'web/components/groups/discussion'
|
||||
import { listenForCommentsOnGroup } from 'web/lib/firebase/comments'
|
||||
import { GroupChat } from 'web/components/groups/group-chat'
|
||||
import { LoadingIndicator } from 'web/components/loading-indicator'
|
||||
import { Modal } from 'web/components/layout/modal'
|
||||
import { PlusIcon } from '@heroicons/react/outline'
|
||||
import { checkAgainstQuery } from 'web/hooks/use-sort-and-query-params'
|
||||
import { ChoicesToggleGroup } from 'web/components/choices-toggle-group'
|
||||
import { toast } from 'react-hot-toast'
|
||||
import { useCommentsOnGroup } from 'web/hooks/use-comments'
|
||||
|
||||
export const getStaticProps = fromPropz(getStaticPropz)
|
||||
export async function getStaticPropz(props: { params: { slugs: string[] } }) {
|
||||
|
@ -92,7 +91,7 @@ async function toTopUsers(userScores: { [userId: string]: number }) {
|
|||
export async function getStaticPaths() {
|
||||
return { paths: [], fallback: 'blocking' }
|
||||
}
|
||||
const groupSubpages = [undefined, 'discussion', 'questions', 'details'] as const
|
||||
const groupSubpages = [undefined, 'chat', 'questions', 'details'] as const
|
||||
|
||||
export default function GroupPage(props: {
|
||||
group: Group | null
|
||||
|
@ -115,14 +114,11 @@ export default function GroupPage(props: {
|
|||
|
||||
const router = useRouter()
|
||||
const { slugs } = router.query as { slugs: string[] }
|
||||
const page = (slugs?.[1] ?? 'discussion') as typeof groupSubpages[number]
|
||||
const page = (slugs?.[1] ?? 'chat') as typeof groupSubpages[number]
|
||||
|
||||
const group = useGroup(props.group?.id) ?? props.group
|
||||
const [messages, setMessages] = useState<Comment[] | undefined>(undefined)
|
||||
const [contracts, setContracts] = useState<Contract[] | undefined>(undefined)
|
||||
useEffect(() => {
|
||||
if (group) listenForCommentsOnGroup(group.id, setMessages)
|
||||
}, [group])
|
||||
const messages = useCommentsOnGroup(group?.id)
|
||||
|
||||
useEffect(() => {
|
||||
if (group)
|
||||
|
@ -189,7 +185,9 @@ export default function GroupPage(props: {
|
|||
<Row className={' items-center justify-between gap-4 '}>
|
||||
<div className={'mb-1'}>
|
||||
<Title className={'line-clamp-2'} text={group.name} />
|
||||
<span className={'text-gray-700'}>{group.about}</span>
|
||||
<span className={'hidden text-gray-700 sm:block'}>
|
||||
{group.about}
|
||||
</span>
|
||||
</div>
|
||||
{isMember && (
|
||||
<CreateQuestionButton
|
||||
|
@ -209,13 +207,13 @@ export default function GroupPage(props: {
|
|||
defaultIndex={page === 'details' ? 2 : page === 'questions' ? 1 : 0}
|
||||
tabs={[
|
||||
{
|
||||
title: 'Discussion',
|
||||
title: 'Chat',
|
||||
content: messages ? (
|
||||
<Discussion messages={messages} user={user} group={group} />
|
||||
<GroupChat messages={messages} user={user} group={group} />
|
||||
) : (
|
||||
<LoadingIndicator />
|
||||
),
|
||||
href: groupPath(group.slug, 'discussion'),
|
||||
href: groupPath(group.slug, 'chat'),
|
||||
},
|
||||
{
|
||||
title: 'Questions',
|
||||
|
|
Loading…
Reference in New Issue
Block a user