2022-08-31 02:35:06 -07:00
|
|
|
import classNames from 'clsx';
|
2022-01-10 14:17:52 -08:00
|
|
|
import React, { useEffect, useState } from 'react';
|
2022-10-10 14:25:19 -07:00
|
|
|
import { useIntl, FormattedMessage } from 'react-intl';
|
2022-01-10 14:01:24 -08:00
|
|
|
import { usePopper } from 'react-popper';
|
2022-03-22 05:42:26 -07:00
|
|
|
import { useHistory } from 'react-router-dom';
|
2022-01-10 14:25:06 -08:00
|
|
|
|
2020-09-10 17:09:27 -07:00
|
|
|
import { fetchRelationships } from 'soapbox/actions/accounts';
|
2020-09-11 08:05:52 -07:00
|
|
|
import {
|
|
|
|
closeProfileHoverCard,
|
|
|
|
updateProfileHoverCard,
|
2022-11-16 05:32:32 -08:00
|
|
|
} from 'soapbox/actions/profile-hover-card';
|
2022-01-10 14:17:52 -08:00
|
|
|
import Badge from 'soapbox/components/badge';
|
2022-05-03 06:27:15 -07:00
|
|
|
import ActionButton from 'soapbox/features/ui/components/action-button';
|
2022-11-16 05:32:32 -08:00
|
|
|
import BundleContainer from 'soapbox/features/ui/containers/bundle-container';
|
2022-01-10 14:17:52 -08:00
|
|
|
import { UserPanel } from 'soapbox/features/ui/util/async-components';
|
2022-05-01 11:11:20 -07:00
|
|
|
import { useAppSelector, useAppDispatch } from 'soapbox/hooks';
|
2022-01-10 14:17:52 -08:00
|
|
|
import { makeGetAccount } from 'soapbox/selectors';
|
2022-10-11 11:22:54 -07:00
|
|
|
import { isLocal } from 'soapbox/utils/accounts';
|
2020-09-10 17:09:27 -07:00
|
|
|
|
2022-11-15 12:46:23 -08:00
|
|
|
import { showProfileHoverCard } from './hover-ref-wrapper';
|
2022-10-10 14:25:19 -07:00
|
|
|
import { Card, CardBody, HStack, Icon, Stack, Text } from './ui';
|
2022-03-21 11:09:01 -07:00
|
|
|
|
2022-05-01 11:11:20 -07:00
|
|
|
import type { AppDispatch } from 'soapbox/store';
|
|
|
|
import type { Account } from 'soapbox/types/entities';
|
|
|
|
|
2020-09-10 17:09:27 -07:00
|
|
|
const getAccount = makeGetAccount();
|
|
|
|
|
2022-05-01 11:11:20 -07:00
|
|
|
const getBadges = (account: Account): JSX.Element[] => {
|
2021-08-03 10:10:42 -07:00
|
|
|
const badges = [];
|
2021-07-15 09:35:50 -07:00
|
|
|
|
2022-04-01 17:35:57 -07:00
|
|
|
if (account.admin) {
|
2021-07-15 09:35:50 -07:00
|
|
|
badges.push(<Badge key='admin' slug='admin' title='Admin' />);
|
2022-04-01 17:35:57 -07:00
|
|
|
} else if (account.moderator) {
|
2021-07-15 09:35:50 -07:00
|
|
|
badges.push(<Badge key='moderator' slug='moderator' title='Moderator' />);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (account.getIn(['patron', 'is_patron'])) {
|
|
|
|
badges.push(<Badge key='patron' slug='patron' title='Patron' />);
|
|
|
|
}
|
|
|
|
|
2020-09-10 17:09:27 -07:00
|
|
|
return badges;
|
|
|
|
};
|
|
|
|
|
2022-05-01 11:11:20 -07:00
|
|
|
const handleMouseEnter = (dispatch: AppDispatch): React.MouseEventHandler => {
|
|
|
|
return () => {
|
2020-09-11 08:05:52 -07:00
|
|
|
dispatch(updateProfileHoverCard());
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2022-05-01 11:11:20 -07:00
|
|
|
const handleMouseLeave = (dispatch: AppDispatch): React.MouseEventHandler => {
|
|
|
|
return () => {
|
2020-09-11 08:05:52 -07:00
|
|
|
dispatch(closeProfileHoverCard(true));
|
2020-09-10 20:08:17 -07:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2022-05-01 11:11:20 -07:00
|
|
|
interface IProfileHoverCard {
|
|
|
|
visible: boolean,
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Popup profile preview that appears when hovering avatars and display names. */
|
|
|
|
export const ProfileHoverCard: React.FC<IProfileHoverCard> = ({ visible = true }) => {
|
|
|
|
const dispatch = useAppDispatch();
|
2022-03-22 05:42:26 -07:00
|
|
|
const history = useHistory();
|
2022-10-10 14:25:19 -07:00
|
|
|
const intl = useIntl();
|
2020-09-10 17:09:27 -07:00
|
|
|
|
2022-05-01 11:11:20 -07:00
|
|
|
const [popperElement, setPopperElement] = useState<HTMLElement | null>(null);
|
2020-09-10 17:09:27 -07:00
|
|
|
|
2022-05-01 11:11:20 -07:00
|
|
|
const me = useAppSelector(state => state.me);
|
2022-06-12 11:15:34 -07:00
|
|
|
const accountId: string | undefined = useAppSelector(state => state.profile_hover_card.accountId || undefined);
|
2022-05-01 11:11:20 -07:00
|
|
|
const account = useAppSelector(state => accountId && getAccount(state, accountId));
|
2022-06-12 11:15:34 -07:00
|
|
|
const targetRef = useAppSelector(state => state.profile_hover_card.ref?.current);
|
2020-09-10 17:09:27 -07:00
|
|
|
const badges = account ? getBadges(account) : [];
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
if (accountId) dispatch(fetchRelationships([accountId]));
|
|
|
|
}, [dispatch, accountId]);
|
|
|
|
|
2022-03-21 11:09:01 -07:00
|
|
|
useEffect(() => {
|
|
|
|
const unlisten = history.listen(() => {
|
|
|
|
showProfileHoverCard.cancel();
|
|
|
|
dispatch(closeProfileHoverCard());
|
|
|
|
});
|
|
|
|
|
|
|
|
return () => {
|
|
|
|
unlisten();
|
|
|
|
};
|
|
|
|
}, []);
|
|
|
|
|
2020-09-11 10:17:32 -07:00
|
|
|
const { styles, attributes } = usePopper(targetRef, popperElement);
|
2020-09-10 17:09:27 -07:00
|
|
|
|
|
|
|
if (!account) return null;
|
2022-05-01 11:11:20 -07:00
|
|
|
const accountBio = { __html: account.note_emojified };
|
2022-10-10 14:25:19 -07:00
|
|
|
const memberSinceDate = intl.formatDate(account.created_at, { month: 'long', year: 'numeric' });
|
2022-06-04 00:22:36 -07:00
|
|
|
const followedBy = me !== account.id && account.relationship?.followed_by === true;
|
2020-09-10 17:09:27 -07:00
|
|
|
|
|
|
|
return (
|
2022-03-21 11:09:01 -07:00
|
|
|
<div
|
|
|
|
className={classNames({
|
2022-08-25 05:52:11 -07:00
|
|
|
'absolute transition-opacity w-[320px] z-[101] top-0 left-0': true,
|
2022-03-21 11:09:01 -07:00
|
|
|
'opacity-100': visible,
|
|
|
|
'opacity-0 pointer-events-none': !visible,
|
|
|
|
})}
|
|
|
|
ref={setPopperElement}
|
|
|
|
style={styles.popper}
|
|
|
|
{...attributes.popper}
|
|
|
|
onMouseEnter={handleMouseEnter(dispatch)}
|
|
|
|
onMouseLeave={handleMouseLeave(dispatch)}
|
|
|
|
>
|
2022-11-27 09:50:55 -08:00
|
|
|
<Card variant='rounded' className='relative isolate'>
|
2022-03-21 11:09:01 -07:00
|
|
|
<CardBody>
|
|
|
|
<Stack space={2}>
|
|
|
|
<BundleContainer fetchComponent={UserPanel}>
|
|
|
|
{Component => (
|
|
|
|
<Component
|
|
|
|
accountId={account.get('id')}
|
|
|
|
action={<ActionButton account={account} small />}
|
|
|
|
badges={badges}
|
|
|
|
/>
|
|
|
|
)}
|
|
|
|
</BundleContainer>
|
|
|
|
|
2022-10-10 14:25:19 -07:00
|
|
|
{isLocal(account) ? (
|
|
|
|
<HStack alignItems='center' space={0.5}>
|
|
|
|
<Icon
|
|
|
|
src={require('@tabler/icons/calendar.svg')}
|
2023-02-01 14:13:42 -08:00
|
|
|
className='h-4 w-4 text-gray-800 dark:text-gray-200'
|
2022-10-10 14:25:19 -07:00
|
|
|
/>
|
|
|
|
|
|
|
|
<Text size='sm'>
|
|
|
|
<FormattedMessage
|
|
|
|
id='account.member_since' defaultMessage='Joined {date}' values={{
|
|
|
|
date: memberSinceDate,
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
</Text>
|
|
|
|
</HStack>
|
|
|
|
) : null}
|
|
|
|
|
2022-11-22 09:33:00 -08:00
|
|
|
{account.note.length > 0 && (
|
2022-03-21 11:09:01 -07:00
|
|
|
<Text size='sm' dangerouslySetInnerHTML={accountBio} />
|
|
|
|
)}
|
|
|
|
</Stack>
|
|
|
|
|
|
|
|
{followedBy && (
|
|
|
|
<div className='absolute top-2 left-2'>
|
|
|
|
<Badge
|
|
|
|
slug='opaque'
|
2022-05-24 03:24:26 -07:00
|
|
|
title={<FormattedMessage id='account.follows_you' defaultMessage='Follows you' />}
|
2022-03-21 11:09:01 -07:00
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
</CardBody>
|
|
|
|
</Card>
|
2020-09-10 17:09:27 -07:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
2022-03-22 05:42:26 -07:00
|
|
|
export default ProfileHoverCard;
|