pleroma/app/soapbox/features/chats/components/chat-list-item.tsx

161 lines
6 KiB
TypeScript
Raw Normal View History

2022-09-23 06:04:01 -07:00
import React, { useMemo } from 'react';
import { defineMessages, useIntl } from 'react-intl';
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';
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';
import VerificationBadge from 'soapbox/components/verification-badge';
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
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 {
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();
const features = useFeatures();
const history = useHistory();
2022-09-23 06:04:01 -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',
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
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)}
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'
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}>
{features.chatsDelete && (
2023-02-01 14:13:42 -08:00
<div className='hidden text-gray-600 hover:text-gray-100 group-hover:block'>
<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'
iconClassName='h-4 w-4'
2022-12-08 11:42:03 -08:00
/>
</DropdownMenu>
</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;