Files
cinny/src/app/components/user-profile/UserHero.tsx
T

166 lines
5.1 KiB
TypeScript
Raw Normal View History

2025-08-24 18:06:45 +05:30
import React, { useState } from 'react';
import {
Avatar,
Box,
Icon,
Icons,
Modal,
Overlay,
OverlayBackdrop,
OverlayCenter,
Text,
} from 'folds';
2025-08-09 17:46:10 +05:30
import classNames from 'classnames';
2025-08-24 18:06:45 +05:30
import FocusTrap from 'focus-trap-react';
2025-08-09 17:46:10 +05:30
import * as css from './styles.css';
import { UserAvatar } from '../user-avatar';
import colorMXID from '../../../util/colorMXID';
import { getMxIdLocalPart } from '../../utils/matrix';
import { BreakWord, LineClamp2, LineClamp3 } from '../../styles/Text.css';
2025-08-09 17:46:10 +05:30
import { UserPresence } from '../../hooks/useUserPresence';
import { AvatarPresence, PresenceBadge } from '../presence';
2025-08-24 18:06:45 +05:30
import { ImageViewer } from '../image-viewer';
import { stopPropagation } from '../../utils/keyboard';
import { useSetting } from '../../state/hooks/settings';
import { settingsAtom } from '../../state/settings';
import { useLocalTime } from '../../hooks/useLocalTime';
2025-08-09 17:46:10 +05:30
type UserHeroProps = {
userId: string;
avatarUrl?: string;
presence?: UserPresence;
};
export function UserHero({ userId, avatarUrl, presence }: UserHeroProps) {
2025-08-24 18:06:45 +05:30
const [viewAvatar, setViewAvatar] = useState<string>();
2025-08-09 17:46:10 +05:30
return (
<Box direction="Column" className={css.UserHero}>
<div
className={css.UserHeroCoverContainer}
style={{
backgroundColor: colorMXID(userId),
filter: avatarUrl ? undefined : 'brightness(50%)',
}}
>
2025-08-24 18:06:45 +05:30
{avatarUrl && (
<img className={css.UserHeroCover} src={avatarUrl} alt={userId} draggable="false" />
)}
2025-08-09 17:46:10 +05:30
</div>
<div className={css.UserHeroAvatarContainer}>
<AvatarPresence
className={css.UserAvatarContainer}
badge={
presence && <PresenceBadge presence={presence.presence} status={presence.status} />
}
>
2025-08-24 18:06:45 +05:30
<Avatar
as={avatarUrl ? 'button' : 'div'}
onClick={avatarUrl ? () => setViewAvatar(avatarUrl) : undefined}
className={css.UserHeroAvatar}
size="500"
>
2025-08-09 17:46:10 +05:30
<UserAvatar
2025-08-24 18:06:45 +05:30
className={css.UserHeroAvatarImg}
2025-08-09 17:46:10 +05:30
userId={userId}
src={avatarUrl}
alt={userId}
renderFallback={() => <Icon size="500" src={Icons.User} filled />}
/>
</Avatar>
</AvatarPresence>
2025-08-24 18:06:45 +05:30
{viewAvatar && (
<Overlay open backdrop={<OverlayBackdrop />}>
<OverlayCenter>
<FocusTrap
focusTrapOptions={{
initialFocus: false,
onDeactivate: () => setViewAvatar(undefined),
clickOutsideDeactivates: true,
escapeDeactivates: stopPropagation,
}}
>
<Modal size="500" onContextMenu={(evt: any) => evt.stopPropagation()}>
<ImageViewer
src={viewAvatar}
alt={userId}
requestClose={() => setViewAvatar(undefined)}
/>
</Modal>
</FocusTrap>
</OverlayCenter>
</Overlay>
)}
2025-08-09 17:46:10 +05:30
</div>
</Box>
);
}
type UserHeroNameProps = {
displayName?: string;
userId: string;
status?: string;
pronouns?: string;
timezone?: string;
2025-08-09 17:46:10 +05:30
};
export function UserHeroName({
displayName,
userId,
status,
pronouns,
timezone,
}: UserHeroNameProps) {
2025-08-09 17:46:10 +05:30
const username = getMxIdLocalPart(userId);
const [hour24Clock] = useSetting(settingsAtom, 'hour24Clock');
const localTimeInfo = useLocalTime(timezone, !hour24Clock);
2025-08-09 17:46:10 +05:30
return (
<Box grow="Yes" direction="Column" gap="0">
<Box alignItems="Baseline" gap="200" wrap="Wrap">
<Text
size="H4"
className={classNames(BreakWord, LineClamp3)}
title={displayName ?? username}
>
{displayName ?? username ?? userId}
</Text>
</Box>
{pronouns && (
<Box alignItems="Center" gap="100" style={{ marginTop: '1px', overflow: 'hidden' }}>
<Text
size="T200"
className={classNames(BreakWord, LineClamp2)}
style={{ opacity: 0.6, fontStyle: 'italic' }}
>
{pronouns}
</Text>
</Box>
)}
2025-08-09 17:46:10 +05:30
<Box alignItems="Center" gap="100" wrap="Wrap">
<Text size="T200" className={classNames(BreakWord, LineClamp3)} title={username}>
@{username}
</Text>
</Box>
{localTimeInfo && (
<Box alignItems="Center" gap="100" style={{ marginTop: '1px', overflow: 'hidden' }}>
<Icon size="50" src={Icons.Clock} />
<Text size="T200" className={classNames(BreakWord, LineClamp2)} style={{ opacity: 0.6 }}>
{localTimeInfo.time}
{localTimeInfo.abbr && <span style={{ opacity: 0.7 }}>{` ${localTimeInfo.abbr}`}</span>}
</Text>
</Box>
)}
{status && (
<Box alignItems="Center" gap="100" style={{ marginTop: '2px', overflow: 'hidden' }}>
<Text
size="T200"
className={classNames(BreakWord, LineClamp2)}
style={{ opacity: 0.75, overflowWrap: 'anywhere' }}
>
{status}
</Text>
</Box>
)}
2025-08-09 17:46:10 +05:30
</Box>
);
}