2022-09-23 06:04:01 -07:00
import React , { useMemo } from 'react' ;
import { defineMessages , useIntl } from 'react-intl' ;
2022-11-03 12:29:53 -07:00
import { useHistory } from 'react-router-dom' ;
2022-09-12 13:46:19 -07:00
2022-09-23 06:04:01 -07:00
import { openModal } from 'soapbox/actions/modals' ;
2023-02-09 09:42:46 -08:00
import DropdownMenu from 'soapbox/components/dropdown-menu' ;
2022-09-12 13:46:19 -07:00
import RelativeTimestamp from 'soapbox/components/relative-timestamp' ;
2022-12-08 11:42:03 -08:00
import { Avatar , HStack , IconButton , Stack , Text } from 'soapbox/components/ui' ;
2022-11-17 07:58:34 -08:00
import VerificationBadge from 'soapbox/components/verification-badge' ;
2022-11-03 12:29:53 -07:00
import { useChatContext } from 'soapbox/contexts/chat-context' ;
2022-11-07 08:16:47 -08:00
import { useAppDispatch , useAppSelector , useFeatures } from 'soapbox/hooks' ;
2022-10-25 05:23:33 -07:00
import { IChat , useChatActions } from 'soapbox/queries/chats' ;
2022-09-12 13:46:19 -07:00
2022-11-17 07:58:34 -08:00
import type { Menu } from 'soapbox/components/dropdown-menu' ;
2022-09-23 06:04:01 -07:00
const messages = defineMessages ( {
2022-11-07 08:16:47 -08:00
blockedYou : { id : 'chat_list_item.blocked_you' , defaultMessage : 'This user has blocked you' } ,
2022-11-17 10:13:22 -08:00
blocking : { id : 'chat_list_item.blocking' , defaultMessage : 'You have blocked this user' } ,
2022-09-27 07:18:12 -07:00
leaveMessage : { id : 'chat_settings.leave.message' , defaultMessage : 'Are you sure you want to leave this chat? Messages will be deleted for you and this chat will be removed from your inbox.' } ,
2022-09-23 06:04:01 -07:00
leaveHeading : { id : 'chat_settings.leave.heading' , defaultMessage : 'Leave Chat' } ,
leaveConfirm : { id : 'chat_settings.leave.confirm' , defaultMessage : 'Leave Chat' } ,
leaveChat : { id : 'chat_settings.options.leave_chat' , defaultMessage : 'Leave Chat' } ,
} ) ;
2022-09-12 13:46:19 -07:00
interface IChatListItemInterface {
2023-02-15 13:26:27 -08:00
chat : IChat
onClick : ( chat : any ) = > void
2022-09-12 13:46:19 -07:00
}
2022-10-25 05:23:33 -07:00
const ChatListItem : React.FC < IChatListItemInterface > = ( { chat , onClick } ) = > {
2022-09-23 06:04:01 -07:00
const dispatch = useAppDispatch ( ) ;
const intl = useIntl ( ) ;
2022-11-02 12:14:16 -07:00
const features = useFeatures ( ) ;
2022-11-03 12:29:53 -07:00
const history = useHistory ( ) ;
2022-09-23 06:04:01 -07:00
2022-11-03 12:29:53 -07:00
const { isUsingMainChatPage } = useChatContext ( ) ;
2022-09-28 13:20:59 -07:00
const { deleteChat } = useChatActions ( chat ? . id as string ) ;
2022-11-07 08:16:47 -08:00
const isBlocked = useAppSelector ( ( state ) = > state . getIn ( [ 'relationships' , chat . account . id , 'blocked_by' ] ) ) ;
2022-11-17 10:13:22 -08:00
const isBlocking = useAppSelector ( ( state ) = > state . getIn ( [ 'relationships' , chat ? . account ? . id , 'blocking' ] ) ) ;
2022-09-23 06:04:01 -07:00
2022-10-25 05:23:33 -07:00
const menu = useMemo ( ( ) : Menu = > [ {
text : intl.formatMessage ( messages . leaveChat ) ,
action : ( event ) = > {
event . stopPropagation ( ) ;
dispatch ( openModal ( 'CONFIRM' , {
heading : intl.formatMessage ( messages . leaveHeading ) ,
message : intl.formatMessage ( messages . leaveMessage ) ,
confirm : intl.formatMessage ( messages . leaveConfirm ) ,
confirmationTheme : 'primary' ,
2022-11-03 12:29:53 -07:00
onConfirm : ( ) = > {
deleteChat . mutate ( undefined , {
onSuccess() {
if ( isUsingMainChatPage ) {
history . push ( '/chats' ) ;
}
} ,
} ) ;
} ,
2022-10-25 05:23:33 -07:00
} ) ) ;
} ,
icon : require ( '@tabler/icons/logout.svg' ) ,
} ] , [ ] ) ;
2022-09-23 06:04:01 -07:00
2023-07-23 10:22:15 -07:00
const handleKeyDown : React.KeyboardEventHandler < HTMLDivElement > = ( event ) = > {
if ( event . key === 'Enter' || event . key === ' ' ) {
onClick ( chat ) ;
}
} ;
2022-09-12 13:46:19 -07:00
return (
2022-09-29 10:54:18 -07:00
< div
role = 'button'
2022-09-12 13:46:19 -07:00
key = { chat . id }
onClick = { ( ) = > onClick ( chat ) }
2023-07-23 10:22:15 -07:00
onKeyDown = { handleKeyDown }
2023-02-01 14:13:42 -08:00
className = 'group flex w-full flex-col rounded-lg px-2 py-3 hover:bg-gray-100 focus:shadow-inset-ring dark:hover:bg-gray-800'
2022-10-03 08:03:43 -07:00
data - testid = 'chat-list-item'
2023-07-23 10:22:15 -07:00
tabIndex = { 0 }
2022-09-12 13:46:19 -07:00
>
< HStack alignItems = 'center' justifyContent = 'between' space = { 2 } className = 'w-full' >
< HStack alignItems = 'center' space = { 2 } className = 'overflow-hidden' >
< Avatar src = { chat . account ? . avatar } size = { 40 } className = 'flex-none' / >
< Stack alignItems = 'start' className = 'overflow-hidden' >
2023-02-01 14:13:42 -08:00
< div className = 'flex w-full grow items-center space-x-1' >
2022-09-12 13:46:19 -07:00
< Text weight = 'bold' size = 'sm' align = 'left' truncate > { chat . account ? . display_name || ` @ ${ chat . account . username } ` } < / Text >
{ chat . account ? . verified && < VerificationBadge / > }
< / div >
2022-11-17 10:13:22 -08:00
{ ( isBlocked || isBlocking ) ? (
2022-09-12 13:46:19 -07:00
< Text
align = 'left'
size = 'sm'
weight = 'medium'
2022-11-07 08:16:47 -08:00
theme = 'muted'
2022-09-12 13:46:19 -07:00
truncate
2023-02-01 14:13:42 -08:00
className = 'pointer-events-none h-5 w-full italic'
2022-09-12 13:46:19 -07:00
data - testid = 'chat-last-message'
2022-11-07 08:16:47 -08:00
>
2022-11-17 10:13:22 -08:00
{ intl . formatMessage ( isBlocked ? messages.blockedYou : messages.blocking ) }
2022-11-07 08:16:47 -08:00
< / Text >
) : (
< >
{ chat . last_message ? . content && (
< Text
align = 'left'
size = 'sm'
weight = 'medium'
theme = { chat . last_message . unread ? 'default' : 'muted' }
truncate
2023-02-01 14:13:42 -08:00
className = 'truncate-child pointer-events-none h-5 w-full'
2022-11-07 08:16:47 -08:00
data - testid = 'chat-last-message'
dangerouslySetInnerHTML = { { __html : chat.last_message?.content } }
/ >
) }
< / >
2022-09-12 13:46:19 -07:00
) }
< / Stack >
< / HStack >
2022-09-16 05:57:09 -07:00
< HStack alignItems = 'center' space = { 2 } >
2022-11-02 12:14:16 -07:00
{ features . chatsDelete && (
2023-02-01 14:13:42 -08:00
< div className = 'hidden text-gray-600 hover:text-gray-100 group-hover:block' >
2023-02-09 09:42:46 -08:00
< DropdownMenu items = { menu } >
2022-12-08 11:42:03 -08:00
< IconButton
src = { require ( '@tabler/icons/dots.svg' ) }
title = 'Settings'
2023-02-01 14:13:42 -08:00
className = 'text-gray-600 hover:text-gray-700 dark:text-gray-600 dark:hover:text-gray-500'
2023-02-15 13:26:27 -08:00
iconClassName = 'h-4 w-4'
2022-12-08 11:42:03 -08:00
/ >
2023-02-09 09:42:46 -08:00
< / DropdownMenu >
2022-11-02 12:14:16 -07:00
< / div >
) }
2022-09-23 06:04:01 -07:00
2022-09-16 05:57:09 -07:00
{ chat . last_message && (
< >
{ chat . last_message . unread && (
< div
2023-02-01 14:13:42 -08:00
className = 'h-2 w-2 rounded-full bg-secondary-500'
2022-09-16 05:57:09 -07:00
data - testid = 'chat-unread-indicator'
/ >
) }
2022-09-12 13:46:19 -07:00
2022-09-16 05:57:09 -07:00
< RelativeTimestamp
timestamp = { chat . last_message . created_at }
align = 'right'
size = 'xs'
2022-11-07 07:53:44 -08:00
theme = { chat . last_message . unread ? 'default' : 'muted' }
2022-09-16 05:57:09 -07:00
truncate
/ >
< / >
) }
< / HStack >
2022-09-12 13:46:19 -07:00
< / HStack >
2022-09-29 10:54:18 -07:00
< / div >
2022-09-12 13:46:19 -07:00
) ;
} ;
export default ChatListItem ;