import * as React from 'react'; import { Link, useHistory } from 'react-router-dom'; import HoverRefWrapper from 'soapbox/components/hover_ref_wrapper'; import VerificationBadge from 'soapbox/components/verification_badge'; import ActionButton from 'soapbox/features/ui/components/action-button'; import { useAppSelector, useOnScreen } from 'soapbox/hooks'; import { getAcct } from 'soapbox/utils/accounts'; import { displayFqn } from 'soapbox/utils/state'; import RelativeTimestamp from './relative_timestamp'; import { Avatar, Emoji, HStack, Icon, IconButton, Stack, Text } from './ui'; import type { Account as AccountEntity } from 'soapbox/types/entities'; interface IInstanceFavicon { account: AccountEntity, } const InstanceFavicon: React.FC = ({ account }) => { const history = useHistory(); const handleClick: React.MouseEventHandler = (e) => { e.stopPropagation(); history.push(`/timeline/${account.domain}`); }; return ( ); }; interface IProfilePopper { condition: boolean, wrapper: (children: any) => React.ReactElement } const ProfilePopper: React.FC = ({ condition, wrapper, children }): any => condition ? wrapper(children) : children; interface IAccount { account: AccountEntity, action?: React.ReactElement, actionAlignment?: 'center' | 'top', actionIcon?: string, actionTitle?: string, /** Override other actions for specificity like mute/unmute. */ actionType?: 'muting' | 'blocking' | 'follow_request', avatarSize?: number, hidden?: boolean, hideActions?: boolean, id?: string, onActionClick?: (account: any) => void, showProfileHoverCard?: boolean, timestamp?: string, timestampUrl?: string, futureTimestamp?: boolean, withAccountNote?: boolean, withDate?: boolean, withLinkToProfile?: boolean, withRelationship?: boolean, showEdit?: boolean, emoji?: string, } const Account = ({ account, actionType, action, actionIcon, actionTitle, actionAlignment = 'center', avatarSize = 42, hidden = false, hideActions = false, onActionClick, showProfileHoverCard = true, timestamp, timestampUrl, futureTimestamp = false, withAccountNote = false, withDate = false, withLinkToProfile = true, withRelationship = true, showEdit = false, emoji, }: IAccount) => { const overflowRef = React.useRef(null); const actionRef = React.useRef(null); // @ts-ignore const isOnScreen = useOnScreen(overflowRef); const [style, setStyle] = React.useState({ visibility: 'hidden' }); const me = useAppSelector((state) => state.me); const username = useAppSelector((state) => account ? getAcct(account, displayFqn(state)) : null); const handleAction = () => { // @ts-ignore onActionClick(account); }; const renderAction = () => { if (action) { return action; } if (hideActions) { return null; } if (onActionClick && actionIcon) { return ( ); } if (account.id !== me) { return ; } return null; }; React.useEffect(() => { const style: React.CSSProperties = {}; const actionWidth = actionRef.current?.clientWidth || 0; if (overflowRef.current) { style.maxWidth = overflowRef.current.clientWidth - 30 - avatarSize - actionWidth; } else { style.visibility = 'hidden'; } setStyle(style); }, [isOnScreen, overflowRef, actionRef]); if (!account) { return null; } if (hidden) { return ( <> {account.display_name} {account.username} ); } if (withDate) timestamp = account.created_at; const LinkEl: any = withLinkToProfile ? Link : 'div'; return (
{children}} > event.stopPropagation()} > {emoji && ( )}
{children}} > event.stopPropagation()} >
{account.verified && }
@{username} {account.favicon && ( )} {(timestamp) ? ( <> · {timestampUrl ? ( ) : ( )} ) : null} {showEdit ? ( <> · ) : null} {withAccountNote && ( )}
{withRelationship ? renderAction() : null}
); }; export default Account;