RocketChat/Rocket.Chat

View on GitHub
apps/meteor/client/views/room/UserCard/UserCardWithData.tsx

Summary

Maintainability
B
6 hrs
Test Coverage
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;