118 lines
3 KiB
TypeScript
118 lines
3 KiB
TypeScript
|
// Copyright 2021 Signal Messenger, LLC
|
||
|
// SPDX-License-Identifier: AGPL-3.0-only
|
||
|
|
||
|
import React, { MouseEvent, useEffect, useState } from 'react';
|
||
|
import { noop } from 'lodash';
|
||
|
import { AvatarDataType } from '../types/Avatar';
|
||
|
import { BetterAvatarBubble } from './BetterAvatarBubble';
|
||
|
import { LocalizerType } from '../types/Util';
|
||
|
import { Spinner } from './Spinner';
|
||
|
import { avatarDataToArrayBuffer } from '../util/avatarDataToArrayBuffer';
|
||
|
|
||
|
type AvatarSize = 48 | 80;
|
||
|
|
||
|
export type PropsType = {
|
||
|
avatarData: AvatarDataType;
|
||
|
i18n: LocalizerType;
|
||
|
isSelected?: boolean;
|
||
|
onClick: (avatarBuffer: ArrayBuffer | undefined) => unknown;
|
||
|
onDelete: () => unknown;
|
||
|
size?: AvatarSize;
|
||
|
};
|
||
|
|
||
|
export const BetterAvatar = ({
|
||
|
avatarData,
|
||
|
i18n,
|
||
|
isSelected,
|
||
|
onClick,
|
||
|
onDelete,
|
||
|
size = 48,
|
||
|
}: PropsType): JSX.Element => {
|
||
|
const [avatarBuffer, setAvatarBuffer] = useState<ArrayBuffer | undefined>(
|
||
|
avatarData.buffer
|
||
|
);
|
||
|
const [avatarURL, setAvatarURL] = useState<string | undefined>(undefined);
|
||
|
|
||
|
useEffect(() => {
|
||
|
let shouldCancel = false;
|
||
|
|
||
|
async function makeAvatar() {
|
||
|
const buffer = await avatarDataToArrayBuffer(avatarData);
|
||
|
if (!shouldCancel) {
|
||
|
setAvatarBuffer(buffer);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// If we don't have this we'll get lots of flashing because avatarData
|
||
|
// changes too much. Once we have a buffer set we don't need to reload.
|
||
|
if (avatarBuffer) {
|
||
|
return noop;
|
||
|
}
|
||
|
|
||
|
makeAvatar();
|
||
|
|
||
|
return () => {
|
||
|
shouldCancel = true;
|
||
|
};
|
||
|
}, [avatarBuffer, avatarData]);
|
||
|
|
||
|
// Convert avatar's ArrayBuffer to a URL object
|
||
|
useEffect(() => {
|
||
|
if (avatarBuffer) {
|
||
|
const url = URL.createObjectURL(new Blob([avatarBuffer]));
|
||
|
|
||
|
setAvatarURL(url);
|
||
|
}
|
||
|
}, [avatarBuffer]);
|
||
|
|
||
|
// Clean up any remaining object URLs
|
||
|
useEffect(() => {
|
||
|
return () => {
|
||
|
if (avatarURL) {
|
||
|
URL.revokeObjectURL(avatarURL);
|
||
|
}
|
||
|
};
|
||
|
}, [avatarURL]);
|
||
|
|
||
|
const isEditable = Boolean(avatarData.color);
|
||
|
const handleDelete = !avatarData.icon
|
||
|
? (ev: MouseEvent) => {
|
||
|
ev.preventDefault();
|
||
|
ev.stopPropagation();
|
||
|
onDelete();
|
||
|
}
|
||
|
: undefined;
|
||
|
|
||
|
return (
|
||
|
<BetterAvatarBubble
|
||
|
i18n={i18n}
|
||
|
isSelected={isSelected}
|
||
|
onDelete={handleDelete}
|
||
|
onSelect={() => {
|
||
|
onClick(avatarBuffer);
|
||
|
}}
|
||
|
style={{
|
||
|
backgroundImage: avatarURL ? `url(${avatarURL})` : undefined,
|
||
|
backgroundSize: size,
|
||
|
// +8 so that the size is the acutal size we want, 8 is the invisible
|
||
|
// padding around the bubble to make room for the selection border
|
||
|
height: size + 8,
|
||
|
width: size + 8,
|
||
|
}}
|
||
|
>
|
||
|
{isEditable && isSelected && (
|
||
|
<div className="BetterAvatarBubble--editable" />
|
||
|
)}
|
||
|
{!avatarURL && (
|
||
|
<div className="module-Avatar__spinner-container">
|
||
|
<Spinner
|
||
|
size={`${size - 8}px`}
|
||
|
svgSize="small"
|
||
|
direction="on-avatar"
|
||
|
/>
|
||
|
</div>
|
||
|
)}
|
||
|
</BetterAvatarBubble>
|
||
|
);
|
||
|
};
|