apps/meteor/client/views/room/UserCard/UserCardWithData.tsx
import type { IRoom } from '@rocket.chat/core-typings';
import { useEffectEvent } from '@rocket.chat/fuselage-hooks';
import { useSetting, useRolesDescription, useTranslation } from '@rocket.chat/ui-contexts';
import type { ReactElement } from 'react';
import React, { useMemo } from 'react';
import { getUserDisplayName } from '../../../../lib/getUserDisplayName';
import GenericMenu from '../../../components/GenericMenu/GenericMenu';
import LocalTime from '../../../components/LocalTime';
import { UserCard, UserCardAction, UserCardRole, UserCardSkeleton } from '../../../components/UserCard';
import { ReactiveUserStatus } from '../../../components/UserStatus';
import { useUserInfoQuery } from '../../../hooks/useUserInfoQuery';
import { useMemberExists } from '../../hooks/useMemberExists';
import { useUserInfoActions } from '../hooks/useUserInfoActions';
type UserCardWithDataProps = {
username: string;
rid: IRoom['_id'];
onOpenUserInfo: () => void;
onClose: () => void;
};
const UserCardWithData = ({ username, rid, onOpenUserInfo, onClose }: UserCardWithDataProps) => {
const t = useTranslation();
const getRoles = useRolesDescription();
const showRealNames = Boolean(useSetting('UI_Use_Real_Name'));
const { data, isLoading: isUserInfoLoading } = useUserInfoQuery({ username });
const {
data: isMemberData,
refetch,
isSuccess: membershipCheckSuccess,
isLoading: isMembershipStatusLoading,
} = useMemberExists({ roomId: rid, username });
const isLoading = isUserInfoLoading || isMembershipStatusLoading;
const isMember = membershipCheckSuccess && isMemberData?.isMember;
const user = useMemo(() => {
const defaultValue = isLoading ? undefined : null;
const {
_id,
name,
roles = defaultValue,
statusText = defaultValue,
bio = defaultValue,
utcOffset = defaultValue,
nickname,
avatarETag,
} = data?.user || {};
return {
_id,
name: getUserDisplayName(name, username, showRealNames),
username,
roles: roles && getRoles(roles).map((role, index) => <UserCardRole key={index}>{role}</UserCardRole>),
bio,
etag: avatarETag,
localTime: utcOffset && Number.isInteger(utcOffset) && <LocalTime utcOffset={utcOffset} />,
status: _id && <ReactiveUserStatus uid={_id} />,
customStatus: statusText,
nickname,
};
}, [data, username, showRealNames, isLoading, getRoles]);
const handleOpenUserInfo = useEffectEvent(() => {
onOpenUserInfo();
onClose();
});
const { actions: actionsDefinition, menuActions: menuOptions } = useUserInfoActions(
{ _id: user._id ?? '', username: user.username, name: user.name },
rid,
refetch,
undefined,
isMember,
);
const menu = useMemo(() => {
if (!menuOptions?.length) {
return null;
}
return (
<GenericMenu
title={t('More')}
key='menu'
data-qa-id='menu'
sections={menuOptions}
placement='bottom-start'
callbackAction={onClose}
/>
);
}, [menuOptions, onClose, t]);
const actions = useMemo(() => {
const mapAction = ([key, { content, icon, onClick }]: any): ReactElement => (
<UserCardAction key={key} label={content} aria-label={content} onClick={onClick} icon={icon} />
);
return [...actionsDefinition.map(mapAction), menu].filter(Boolean);
}, [actionsDefinition, menu]);
if (isLoading) {
return <UserCardSkeleton />;
}
return <UserCard {...user} onClose={onClose} onOpenUserInfo={handleOpenUserInfo} actions={actions} />;
};
export default UserCardWithData;