import type { MouseEventHandler } from 'react'; import { useCallback } from 'react'; import { FormattedMessage, defineMessages, useIntl } from 'react-intl'; import classNames from 'classnames'; import { Link } from 'react-router-dom'; import { followAccount, unfollowAccount, unblockAccount, unmuteAccount, } from 'mastodon/actions/accounts'; import { openModal } from 'mastodon/actions/modal'; import { Avatar } from 'mastodon/components/avatar'; import { Button } from 'mastodon/components/button'; import { DisplayName } from 'mastodon/components/display_name'; import { ShortNumber } from 'mastodon/components/short_number'; import { autoPlayGif, me } from 'mastodon/initial_state'; import type { Account } from 'mastodon/models/account'; import { makeGetAccount } from 'mastodon/selectors'; import { useAppDispatch, useAppSelector } from 'mastodon/store'; const messages = defineMessages({ unfollow: { id: 'account.unfollow', defaultMessage: 'Unfollow' }, follow: { id: 'account.follow', defaultMessage: 'Follow' }, cancel_follow_request: { id: 'account.cancel_follow_request', defaultMessage: 'Withdraw follow request', }, cancelFollowRequestConfirm: { id: 'confirmations.cancel_follow_request.confirm', defaultMessage: 'Withdraw request', }, requested: { id: 'account.requested', defaultMessage: 'Awaiting approval. Click to cancel follow request', }, unblock: { id: 'account.unblock_short', defaultMessage: 'Unblock' }, unmute: { id: 'account.unmute_short', defaultMessage: 'Unmute' }, unfollowConfirm: { id: 'confirmations.unfollow.confirm', defaultMessage: 'Unfollow', }, edit_profile: { id: 'account.edit_profile', defaultMessage: 'Edit profile' }, }); const getAccount = makeGetAccount(); export const AccountCard: React.FC<{ accountId: string }> = ({ accountId }) => { const intl = useIntl(); const account = useAppSelector((s) => getAccount(s, accountId)); const dispatch = useAppDispatch(); const handleMouseEnter = useCallback( ({ currentTarget }) => { if (autoPlayGif) { return; } const emojis = currentTarget.querySelectorAll('.custom-emoji'); emojis.forEach((emoji) => { const original = emoji.getAttribute('data-original'); if (original) emoji.src = original; }); }, [], ); const handleMouseLeave = useCallback( ({ currentTarget }) => { if (autoPlayGif) { return; } const emojis = currentTarget.querySelectorAll('.custom-emoji'); emojis.forEach((emoji) => { const staticUrl = emoji.getAttribute('data-static'); if (staticUrl) emoji.src = staticUrl; }); }, [], ); const handleFollow = useCallback(() => { if (!account) return; if (account.getIn(['relationship', 'following'])) { dispatch( openModal({ modalType: 'CONFIRM', modalProps: { message: ( @{account.get('acct')} }} /> ), confirm: intl.formatMessage(messages.unfollowConfirm), onConfirm: () => { dispatch(unfollowAccount(account.get('id'))); }, }, }), ); } else if (account.getIn(['relationship', 'requested'])) { dispatch( openModal({ modalType: 'CONFIRM', modalProps: { message: ( @{account.get('acct')} }} /> ), confirm: intl.formatMessage(messages.cancelFollowRequestConfirm), onConfirm: () => { dispatch(unfollowAccount(account.get('id'))); }, }, }), ); } else { dispatch(followAccount(account.get('id'))); } }, [account, dispatch, intl]); const handleBlock = useCallback(() => { if (account?.relationship?.blocking) { dispatch(unblockAccount(account.get('id'))); } }, [account, dispatch]); const handleMute = useCallback(() => { if (account?.relationship?.muting) { dispatch(unmuteAccount(account.get('id'))); } }, [account, dispatch]); const handleEditProfile = useCallback(() => { window.open('/settings/profile', '_blank'); }, []); if (!account) return null; let actionBtn; if (me !== account.get('id')) { if (!account.get('relationship')) { // Wait until the relationship is loaded actionBtn = ''; } else if (account.getIn(['relationship', 'requested'])) { actionBtn = (