import React, { useState } from 'react'; import classNames from 'classnames'; import { useEncryptedFileFetch } from '../../hooks/useEncryptedFileFetch'; import { isEqual } from 'lodash'; import { useAvatarPath, useConversationUsername, useIsClosedGroup, } from '../../hooks/useParamSelector'; import { AvatarPlaceHolder } from './AvatarPlaceHolder/AvatarPlaceHolder'; import { ClosedGroupAvatar } from './AvatarPlaceHolder/ClosedGroupAvatar'; import { useDisableDrag } from '../../hooks/useDisableDrag'; import styled from 'styled-components'; import { SessionIcon } from '../icon'; import { useSelector } from 'react-redux'; import { isMessageSelectionMode } from '../../state/selectors/conversations'; export enum AvatarSize { XS = 28, S = 36, M = 48, L = 64, XL = 80, HUGE = 300, } type Props = { forcedAvatarPath?: string | null; forcedName?: string; pubkey: string; size: AvatarSize; base64Data?: string; // if this is not empty, it will be used to render the avatar with base64 encoded data onAvatarClick?: () => void; dataTestId?: string; }; const Identicon = (props: Props) => { const { size, forcedName, pubkey } = props; const displayName = useConversationUsername(pubkey); const userName = forcedName || displayName || '0'; return ; }; const CrownWrapper = styled.div` position: absolute; display: flex; bottom: 0%; right: 12%; height: 20px; width: 20px; transform: translate(25%, 25%); color: #f7c347; background: var(--color-inbox-background); border-radius: 50%; filter: drop-shadow(0px 0px 4px rgba(0, 0, 0, 0.3)); align-items: center; justify-content: center; `; export const CrownIcon = () => { return ( ); }; const NoImage = ( props: Pick & { isClosedGroup: boolean; } ) => { const { forcedName, size, pubkey, isClosedGroup } = props; // if no image but we have conversations set for the group, renders group members avatars if (pubkey && isClosedGroup) { return ( ); } return ; }; const AvatarImage = (props: { avatarPath?: string; base64Data?: string; name?: string; // display name, profileName or pubkey, whatever is set first imageBroken: boolean; datatestId?: string; handleImageError: () => any; }) => { const { avatarPath, base64Data, name, imageBroken, datatestId, handleImageError } = props; const disableDrag = useDisableDrag(); if ((!avatarPath && !base64Data) || imageBroken) { return null; } const dataToDisplay = base64Data ? `data:image/jpeg;base64,${base64Data}` : avatarPath; return ( {window.i18n('contactAvatarAlt', ); }; const AvatarInner = (props: Props) => { const { base64Data, size, pubkey, forcedAvatarPath, forcedName, dataTestId } = props; const [imageBroken, setImageBroken] = useState(false); const isSelectingMessages = useSelector(isMessageSelectionMode); const isClosedGroupAvatar = useIsClosedGroup(pubkey); const avatarPath = useAvatarPath(pubkey); const name = useConversationUsername(pubkey); // contentType is not important const { urlToLoad } = useEncryptedFileFetch(forcedAvatarPath || avatarPath || '', '', true); const handleImageError = () => { window.log.warn( 'Avatar: Image failed to load; failing over to placeholder', urlToLoad, forcedAvatarPath || avatarPath ); setImageBroken(true); }; const hasImage = (base64Data || urlToLoad) && !imageBroken && !isClosedGroupAvatar; const isClickable = !!props.onAvatarClick; return (
{ if (isSelectingMessages) { // we could toggle the selection of this message, // but this just disable opening the new Conversation dialog with that user while selecting messages return; } if (props.onAvatarClick) { e.stopPropagation(); e.preventDefault(); props.onAvatarClick?.(); } }} role="button" data-testid={dataTestId} > {hasImage ? ( // tslint:disable-next-line: use-simple-attributes ) : ( )}
); }; export const Avatar = React.memo(AvatarInner, isEqual);