You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
	
	
		
			178 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			TypeScript
		
	
			
		
		
	
	
			178 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			TypeScript
		
	
| 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 <AvatarPlaceHolder diameter={size} name={userName} pubkey={pubkey} />;
 | |
| };
 | |
| 
 | |
| const CrownWrapper = styled.div`
 | |
|   display: flex;
 | |
|   align-items: center;
 | |
|   justify-content: center;
 | |
|   position: absolute;
 | |
|   bottom: 0%;
 | |
|   right: 12%;
 | |
|   height: 20px;
 | |
|   width: 20px;
 | |
|   transform: translate(25%, 25%);
 | |
|   color: #f7c347;
 | |
|   background: var(--background-primary-color);
 | |
|   border-radius: 50%;
 | |
|   /* TODO Theming update? */
 | |
|   filter: drop-shadow(0px 0px 4px rgba(0, 0, 0, 0.3));
 | |
| `;
 | |
| 
 | |
| export const CrownIcon = () => {
 | |
|   return (
 | |
|     <CrownWrapper>
 | |
|       <SessionIcon iconSize={'small'} iconType="crown" iconPadding="1px 0 0 0 " />
 | |
|     </CrownWrapper>
 | |
|   );
 | |
| };
 | |
| 
 | |
| const NoImage = (
 | |
|   props: Pick<Props, 'forcedName' | 'size' | 'pubkey' | 'onAvatarClick'> & {
 | |
|     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 (
 | |
|       <ClosedGroupAvatar size={size} closedGroupId={pubkey} onAvatarClick={props.onAvatarClick} />
 | |
|     );
 | |
|   }
 | |
| 
 | |
|   return <Identicon size={size} forcedName={forcedName} pubkey={pubkey} />;
 | |
| };
 | |
| 
 | |
| 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 (
 | |
|     <img
 | |
|       onError={handleImageError}
 | |
|       onDragStart={disableDrag}
 | |
|       alt={window.i18n('contactAvatarAlt', [name || 'avatar'])}
 | |
|       src={dataToDisplay}
 | |
|       data-testid={datatestId}
 | |
|     />
 | |
|   );
 | |
| };
 | |
| 
 | |
| 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 (
 | |
|     <div
 | |
|       className={classNames(
 | |
|         'module-avatar',
 | |
|         `module-avatar--${size}`,
 | |
|         hasImage ? 'module-avatar--with-image' : 'module-avatar--no-image',
 | |
|         isClickable && 'module-avatar-clickable'
 | |
|       )}
 | |
|       onClick={e => {
 | |
|         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
 | |
|         <AvatarImage
 | |
|           avatarPath={urlToLoad}
 | |
|           base64Data={base64Data}
 | |
|           imageBroken={imageBroken}
 | |
|           name={forcedName || name}
 | |
|           handleImageError={handleImageError}
 | |
|           datatestId={dataTestId ? `img-${dataTestId}` : undefined}
 | |
|         />
 | |
|       ) : (
 | |
|         <NoImage {...props} isClosedGroup={isClosedGroupAvatar} />
 | |
|       )}
 | |
|     </div>
 | |
|   );
 | |
| };
 | |
| 
 | |
| export const Avatar = React.memo(AvatarInner, isEqual);
 |