apps/meteor/client/components/message/variants/SystemMessage.tsx
import type { IMessage } from '@rocket.chat/core-typings';
import {
MessageSystem,
MessageSystemBody,
MessageSystemContainer,
MessageSystemLeftContainer,
MessageSystemName,
MessageSystemTimestamp,
MessageSystemBlock,
CheckBox,
MessageUsername,
MessageNameContainer,
} from '@rocket.chat/fuselage';
import { UserAvatar } from '@rocket.chat/ui-avatar';
import type { TranslationKey } from '@rocket.chat/ui-contexts';
import { useTranslation } from '@rocket.chat/ui-contexts';
import type { ComponentProps, ReactElement, KeyboardEvent } from 'react';
import React, { memo } from 'react';
import { MessageTypes } from '../../../../app/ui-utils/client';
import { getUserDisplayName } from '../../../../lib/getUserDisplayName';
import { useFormatDateAndTime } from '../../../hooks/useFormatDateAndTime';
import { useFormatTime } from '../../../hooks/useFormatTime';
import { useUserData } from '../../../hooks/useUserData';
import type { UserPresence } from '../../../lib/presence';
import {
useIsSelecting,
useToggleSelect,
useIsSelectedMessage,
useCountSelected,
} from '../../../views/room/MessageList/contexts/SelectedMessagesContext';
import { useUserCard } from '../../../views/room/contexts/UserCardContext';
import Attachments from '../content/Attachments';
import MessageActions from '../content/MessageActions';
import { useMessageListShowRealName, useMessageListShowUsername } from '../list/MessageListContext';
type SystemMessageProps = {
message: IMessage;
showUserAvatar: boolean;
} & ComponentProps<typeof MessageSystem>;
const SystemMessage = ({ message, showUserAvatar, ...props }: SystemMessageProps): ReactElement => {
const t = useTranslation();
const formatTime = useFormatTime();
const formatDateAndTime = useFormatDateAndTime();
const { triggerProps, openUserCard } = useUserCard();
const showRealName = useMessageListShowRealName();
const user: UserPresence = { ...message.u, roles: [], ...useUserData(message.u._id) };
const usernameAndRealNameAreSame = !user.name || user.username === user.name;
const showUsername = useMessageListShowUsername() && showRealName && !usernameAndRealNameAreSame;
const messageType = MessageTypes.getType(message);
const isSelecting = useIsSelecting();
const toggleSelected = useToggleSelect(message._id);
const isSelected = useIsSelectedMessage(message._id);
useCountSelected();
return (
<MessageSystem
role='listitem'
aria-roledescription={t('system_message')}
tabIndex={0}
onClick={isSelecting ? toggleSelected : undefined}
isSelected={isSelected}
data-qa-selected={isSelected}
data-qa='system-message'
data-system-message-type={message.t}
{...props}
>
<MessageSystemLeftContainer>
{!isSelecting && showUserAvatar && <UserAvatar username={message.u.username} size='x18' />}
{isSelecting && <CheckBox checked={isSelected} onChange={toggleSelected} />}
</MessageSystemLeftContainer>
<MessageSystemContainer>
<MessageSystemBlock>
<MessageNameContainer
tabIndex={0}
role='button'
onClick={(e) => user.username && openUserCard(e, user.username)}
onKeyDown={(e: KeyboardEvent<HTMLSpanElement>) => {
(e.code === 'Enter' || e.code === 'Space') && openUserCard(e, message.u.username);
}}
style={{ cursor: 'pointer' }}
{...triggerProps}
>
<MessageSystemName>{getUserDisplayName(user.name, user.username, showRealName)}</MessageSystemName>
{showUsername && <MessageUsername data-username={user.username}>@{user.username}</MessageUsername>}
</MessageNameContainer>
{messageType && (
<MessageSystemBody
data-qa-type='system-message-body'
dangerouslySetInnerHTML={{
__html: messageType.render
? messageType.render(message)
: t(messageType.message, messageType.data ? messageType.data(message) : {}),
}}
/>
)}
<MessageSystemTimestamp title={formatDateAndTime(message.ts)}>{formatTime(message.ts)}</MessageSystemTimestamp>
</MessageSystemBlock>
{message.attachments && (
<MessageSystemBlock>
<Attachments attachments={message.attachments} />
</MessageSystemBlock>
)}
{message.actionLinks?.length && (
<MessageActions
message={message}
actions={message.actionLinks.map(({ method_id: methodId, i18nLabel, ...action }) => ({
methodId,
i18nLabel: i18nLabel as TranslationKey,
...action,
}))}
/>
)}
</MessageSystemContainer>
</MessageSystem>
);
};
export default memo(SystemMessage);