signal-desktop/ts/components/ProfileEditor.tsx

890 lines
26 KiB
TypeScript
Raw Normal View History

2023-01-03 19:55:46 +00:00
// Copyright 2021 Signal Messenger, LLC
2021-07-19 19:26:06 +00:00
// SPDX-License-Identifier: AGPL-3.0-only
import React, { useCallback, useEffect, useRef, useState } from 'react';
import { useSpring, animated } from '@react-spring/web';
2021-07-19 19:26:06 +00:00
import type { AvatarColorType } from '../types/Colors';
import { AvatarColors } from '../types/Colors';
import type {
2021-08-06 00:17:05 +00:00
AvatarDataType,
AvatarUpdateOptionsType,
2021-08-06 00:17:05 +00:00
DeleteAvatarFromDiskActionType,
ReplaceAvatarActionType,
SaveAvatarToDiskActionType,
} from '../types/Avatar';
import { AvatarEditor } from './AvatarEditor';
import { AvatarPreview } from './AvatarPreview';
2021-07-19 19:26:06 +00:00
import { Button, ButtonVariant } from './Button';
2021-08-06 00:17:05 +00:00
import { ConfirmDiscardDialog } from './ConfirmDiscardDialog';
2021-07-19 19:26:06 +00:00
import { Emoji } from './emoji/Emoji';
import type { Props as EmojiButtonProps } from './emoji/EmojiButton';
2022-10-18 17:12:02 +00:00
import { EmojiButton, EmojiButtonVariant } from './emoji/EmojiButton';
import type { EmojiPickDataType } from './emoji/EmojiPicker';
2021-07-19 19:26:06 +00:00
import { Input } from './Input';
2022-10-18 17:12:02 +00:00
import type { LocalizerType } from '../types/Util';
2021-08-06 00:17:05 +00:00
import { Modal } from './Modal';
2021-07-19 19:26:06 +00:00
import { PanelRow } from './conversation/conversation-details/PanelRow';
2023-07-20 03:14:08 +00:00
import type {
ProfileDataType,
SaveAttachmentActionCreatorType,
} from '../state/ducks/conversations';
2022-10-18 17:12:02 +00:00
import { UsernameEditState } from '../state/ducks/usernameEnums';
2023-07-20 03:14:08 +00:00
import type { UsernameLinkState } from '../state/ducks/usernameEnums';
import { ToastType } from '../types/Toast';
import type { ShowToastAction } from '../state/ducks/toast';
2021-07-19 19:26:06 +00:00
import { getEmojiData, unifiedToEmoji } from './emoji/lib';
2022-10-18 17:12:02 +00:00
import { assertDev } from '../util/assert';
2021-07-19 19:26:06 +00:00
import { missingCaseError } from '../util/missingCaseError';
import { ConfirmationDialog } from './ConfirmationDialog';
2022-10-18 17:12:02 +00:00
import { ContextMenu } from './ContextMenu';
2023-07-20 03:14:08 +00:00
import { UsernameLinkModalBody } from './UsernameLinkModalBody';
import {
ConversationDetailsIcon,
IconType,
} from './conversation/conversation-details/ConversationDetailsIcon';
2022-01-26 21:58:00 +00:00
import { isWhitespace, trim } from '../util/whitespaceStringUtil';
2023-04-20 17:03:43 +00:00
import { UserText } from './UserText';
2023-07-20 03:14:08 +00:00
import { Tooltip, TooltipPlacement } from './Tooltip';
import { offsetDistanceModifier } from '../util/popperUtil';
2021-07-19 19:26:06 +00:00
export enum EditState {
None = 'None',
2021-08-06 00:17:05 +00:00
BetterAvatar = 'BetterAvatar',
2021-07-19 19:26:06 +00:00
ProfileName = 'ProfileName',
Bio = 'Bio',
Username = 'Username',
2023-07-20 03:14:08 +00:00
UsernameLink = 'UsernameLink',
}
2021-07-19 19:26:06 +00:00
type PropsExternalType = {
onEditStateChanged: (editState: EditState) => unknown;
onProfileChanged: (
profileData: ProfileDataType,
avatarUpdateOptions: AvatarUpdateOptionsType
2021-07-19 19:26:06 +00:00
) => unknown;
2024-02-06 18:35:59 +00:00
renderEditUsernameModalBody: (props: {
isRootModal: boolean;
onClose: () => void;
}) => JSX.Element;
2021-07-19 19:26:06 +00:00
};
export type PropsDataType = {
aboutEmoji?: string;
aboutText?: string;
profileAvatarPath?: string;
2021-08-06 00:17:05 +00:00
color?: AvatarColorType;
conversationId: string;
2021-07-19 19:26:06 +00:00
familyName?: string;
firstName: string;
2023-07-20 03:14:08 +00:00
hasCompletedUsernameLinkOnboarding: boolean;
2021-07-19 19:26:06 +00:00
i18n: LocalizerType;
userAvatarData: ReadonlyArray<AvatarDataType>;
username?: string;
initialEditState?: EditState;
usernameCorrupted: boolean;
2022-10-18 17:12:02 +00:00
usernameEditState: UsernameEditState;
2023-07-20 03:14:08 +00:00
usernameLinkState: UsernameLinkState;
usernameLinkColor?: number;
usernameLink?: string;
usernameLinkCorrupted: boolean;
2024-02-14 20:30:32 +00:00
isUsernameDeletionEnabled: boolean;
2021-07-19 19:26:06 +00:00
} & Pick<EmojiButtonProps, 'recentEmojis' | 'skinTone'>;
type PropsActionType = {
2021-08-06 00:17:05 +00:00
deleteAvatarFromDisk: DeleteAvatarFromDiskActionType;
2023-07-20 03:14:08 +00:00
markCompletedUsernameLinkOnboarding: () => void;
2021-07-19 19:26:06 +00:00
onSetSkinTone: (tone: number) => unknown;
2021-08-06 00:17:05 +00:00
replaceAvatar: ReplaceAvatarActionType;
2023-07-20 03:14:08 +00:00
saveAttachment: SaveAttachmentActionCreatorType;
2021-08-06 00:17:05 +00:00
saveAvatarToDisk: SaveAvatarToDiskActionType;
2022-10-18 17:12:02 +00:00
setUsernameEditState: (editState: UsernameEditState) => void;
2023-07-20 03:14:08 +00:00
setUsernameLinkColor: (color: number) => void;
2024-02-06 18:35:59 +00:00
toggleProfileEditor: () => void;
2023-07-20 03:14:08 +00:00
resetUsernameLink: () => void;
2022-10-18 17:12:02 +00:00
deleteUsername: () => void;
showToast: ShowToastAction;
2022-10-18 17:12:02 +00:00
openUsernameReservationModal: () => void;
2021-07-19 19:26:06 +00:00
};
export type PropsType = PropsDataType & PropsActionType & PropsExternalType;
type DefaultBio = {
i18nLabel: string;
shortName: string;
};
function getDefaultBios(i18n: LocalizerType): Array<DefaultBio> {
return [
{
2023-03-30 00:03:25 +00:00
i18nLabel: i18n('icu:Bio--speak-freely'),
shortName: 'wave',
},
{
2023-03-30 00:03:25 +00:00
i18nLabel: i18n('icu:Bio--encrypted'),
shortName: 'zipper_mouth_face',
},
{
2023-03-30 00:03:25 +00:00
i18nLabel: i18n('icu:Bio--free-to-chat'),
shortName: '+1',
},
{
2023-03-30 00:03:25 +00:00
i18nLabel: i18n('icu:Bio--coffee-lover'),
shortName: 'coffee',
},
{
2023-03-30 00:03:25 +00:00
i18nLabel: i18n('icu:Bio--taking-break'),
shortName: 'mobile_phone_off',
},
];
}
2021-07-19 19:26:06 +00:00
2022-11-18 00:45:19 +00:00
export function ProfileEditor({
2021-07-19 19:26:06 +00:00
aboutEmoji,
aboutText,
2021-08-06 00:17:05 +00:00
color,
conversationId,
deleteAvatarFromDisk,
2022-10-18 17:12:02 +00:00
deleteUsername,
2021-07-19 19:26:06 +00:00
familyName,
firstName,
2023-07-20 03:14:08 +00:00
hasCompletedUsernameLinkOnboarding,
2021-07-19 19:26:06 +00:00
i18n,
initialEditState = EditState.None,
2023-07-20 03:14:08 +00:00
markCompletedUsernameLinkOnboarding,
2021-07-19 19:26:06 +00:00
onEditStateChanged,
onProfileChanged,
onSetSkinTone,
2022-10-18 17:12:02 +00:00
openUsernameReservationModal,
profileAvatarPath,
2021-07-19 19:26:06 +00:00
recentEmojis,
2022-10-18 17:12:02 +00:00
renderEditUsernameModalBody,
2021-08-06 00:17:05 +00:00
replaceAvatar,
2023-07-20 03:14:08 +00:00
resetUsernameLink,
2024-02-06 18:35:59 +00:00
toggleProfileEditor,
2023-07-20 03:14:08 +00:00
saveAttachment,
2021-08-06 00:17:05 +00:00
saveAvatarToDisk,
2022-10-18 17:12:02 +00:00
setUsernameEditState,
2023-07-20 03:14:08 +00:00
setUsernameLinkColor,
2022-10-18 17:12:02 +00:00
showToast,
2021-07-19 19:26:06 +00:00
skinTone,
2021-08-06 00:17:05 +00:00
userAvatarData,
username,
usernameCorrupted,
2022-10-18 17:12:02 +00:00
usernameEditState,
2023-07-20 03:14:08 +00:00
usernameLinkState,
usernameLinkColor,
usernameLink,
usernameLinkCorrupted,
2024-02-14 20:30:32 +00:00
isUsernameDeletionEnabled,
2022-11-18 00:45:19 +00:00
}: PropsType): JSX.Element {
2021-07-19 19:26:06 +00:00
const focusInputRef = useRef<HTMLInputElement | null>(null);
const [editState, setEditState] = useState<EditState>(initialEditState);
2021-07-19 19:26:06 +00:00
const [confirmDiscardAction, setConfirmDiscardAction] = useState<
(() => unknown) | undefined
>(undefined);
// This is here to avoid component re-render jitters in the time it takes
// redux to come back with the correct state
const [fullName, setFullName] = useState({
familyName,
firstName,
});
const [fullBio, setFullBio] = useState({
aboutEmoji,
aboutText,
});
const [startingAvatarPath, setStartingAvatarPath] =
useState(profileAvatarPath);
const [oldAvatarBuffer, setOldAvatarBuffer] = useState<
Uint8Array | undefined
>(undefined);
2021-09-24 00:49:05 +00:00
const [avatarBuffer, setAvatarBuffer] = useState<Uint8Array | undefined>(
2021-07-19 19:26:06 +00:00
undefined
);
const [stagedProfile, setStagedProfile] = useState<ProfileDataType>({
aboutEmoji,
aboutText,
familyName,
firstName,
});
const [isResettingUsername, setIsResettingUsername] = useState(false);
2024-02-14 20:30:32 +00:00
const [isUsernameNoticeVisible, setIsUsernameNoticeVisible] = useState(false);
2024-02-06 18:35:59 +00:00
const [isResettingUsernameLink, setIsResettingUsernameLink] = useState(false);
2021-07-19 19:26:06 +00:00
2022-10-18 17:12:02 +00:00
// Reset username edit state when leaving
useEffect(() => {
return () => {
setUsernameEditState(UsernameEditState.Editing);
};
}, [setUsernameEditState]);
// To make AvatarEditor re-render less often
2021-07-19 19:26:06 +00:00
const handleBack = useCallback(() => {
setEditState(EditState.None);
onEditStateChanged(EditState.None);
}, [setEditState, onEditStateChanged]);
// To make EmojiButton re-render less often
2021-07-19 19:26:06 +00:00
const setAboutEmoji = useCallback(
(ev: EmojiPickDataType) => {
const emojiData = getEmojiData(ev.shortName, skinTone);
setStagedProfile(profileData => ({
...profileData,
aboutEmoji: unifiedToEmoji(emojiData.unified),
}));
},
[setStagedProfile, skinTone]
);
// To make AvatarEditor re-render less often
2021-07-19 19:26:06 +00:00
const handleAvatarChanged = useCallback(
2021-09-24 00:49:05 +00:00
(avatar: Uint8Array | undefined) => {
// Do not display stale avatar from disk anymore.
setStartingAvatarPath(undefined);
2021-08-06 00:17:05 +00:00
setAvatarBuffer(avatar);
setEditState(EditState.None);
2022-01-26 21:58:00 +00:00
onProfileChanged(
{
...stagedProfile,
firstName: trim(stagedProfile.firstName),
familyName: stagedProfile.familyName
? trim(stagedProfile.familyName)
: undefined,
},
{
keepAvatar: false,
avatarUpdate: { oldAvatar: oldAvatarBuffer, newAvatar: avatar },
}
2022-01-26 21:58:00 +00:00
);
setOldAvatarBuffer(avatar);
2021-07-19 19:26:06 +00:00
},
[onProfileChanged, stagedProfile, oldAvatarBuffer]
2021-07-19 19:26:06 +00:00
);
2021-08-06 00:17:05 +00:00
const getFullNameText = () => {
return [fullName.firstName, fullName.familyName].filter(Boolean).join(' ');
};
2021-07-19 19:26:06 +00:00
useEffect(() => {
const focusNode = focusInputRef.current;
if (!focusNode) {
return;
}
focusNode.focus();
2021-10-01 19:27:34 +00:00
focusNode.setSelectionRange(focusNode.value.length, focusNode.value.length);
2021-07-19 19:26:06 +00:00
}, [editState]);
2021-08-06 00:17:05 +00:00
useEffect(() => {
onEditStateChanged(editState);
}, [editState, onEditStateChanged]);
2024-02-06 18:35:59 +00:00
useEffect(() => {
// If we opened at a nested sub-modal - close when leaving it.
if (editState === EditState.None && initialEditState !== EditState.None) {
toggleProfileEditor();
}
}, [initialEditState, editState, toggleProfileEditor]);
// To make AvatarEditor re-render less often
const handleAvatarLoaded = useCallback(
avatar => {
setAvatarBuffer(avatar);
setOldAvatarBuffer(avatar);
},
[setAvatarBuffer, setOldAvatarBuffer]
);
2021-08-06 00:17:05 +00:00
let content: JSX.Element;
if (editState === EditState.BetterAvatar) {
content = (
<AvatarEditor
avatarColor={color || AvatarColors[0]}
avatarPath={startingAvatarPath}
2021-08-06 00:17:05 +00:00
avatarValue={avatarBuffer}
conversationId={conversationId}
conversationTitle={getFullNameText()}
deleteAvatarFromDisk={deleteAvatarFromDisk}
i18n={i18n}
onCancel={handleBack}
onSave={handleAvatarChanged}
userAvatarData={userAvatarData}
replaceAvatar={replaceAvatar}
saveAvatarToDisk={saveAvatarToDisk}
/>
);
} else if (editState === EditState.ProfileName) {
const shouldDisableSave =
!stagedProfile.firstName ||
(stagedProfile.firstName === fullName.firstName &&
2022-01-26 21:58:00 +00:00
stagedProfile.familyName === fullName.familyName) ||
isWhitespace(stagedProfile.firstName);
2021-07-19 19:26:06 +00:00
content = (
<>
<Input
i18n={i18n}
maxLengthCount={26}
maxByteCount={128}
whenToShowRemainingCount={0}
2021-07-19 19:26:06 +00:00
onChange={newFirstName => {
setStagedProfile(profileData => ({
...profileData,
firstName: String(newFirstName),
}));
}}
2023-03-30 00:03:25 +00:00
placeholder={i18n('icu:ProfileEditor--first-name')}
2021-07-19 19:26:06 +00:00
ref={focusInputRef}
value={stagedProfile.firstName}
/>
<Input
i18n={i18n}
maxLengthCount={26}
maxByteCount={128}
whenToShowRemainingCount={0}
2021-07-19 19:26:06 +00:00
onChange={newFamilyName => {
setStagedProfile(profileData => ({
...profileData,
familyName: newFamilyName,
}));
}}
2023-03-30 00:03:25 +00:00
placeholder={i18n('icu:ProfileEditor--last-name')}
2021-07-19 19:26:06 +00:00
value={stagedProfile.familyName}
/>
2021-08-06 00:17:05 +00:00
<Modal.ButtonFooter>
2021-07-19 19:26:06 +00:00
<Button
onClick={() => {
const handleCancel = () => {
handleBack();
setStagedProfile(profileData => ({
...profileData,
familyName,
firstName,
}));
};
const hasChanges =
stagedProfile.familyName !== fullName.familyName ||
stagedProfile.firstName !== fullName.firstName;
if (hasChanges) {
setConfirmDiscardAction(() => handleCancel);
} else {
handleCancel();
}
}}
variant={ButtonVariant.Secondary}
>
2023-03-30 00:03:25 +00:00
{i18n('icu:cancel')}
2021-07-19 19:26:06 +00:00
</Button>
<Button
disabled={shouldDisableSave}
2021-07-19 19:26:06 +00:00
onClick={() => {
if (!stagedProfile.firstName) {
return;
}
setFullName({
firstName: stagedProfile.firstName,
familyName: stagedProfile.familyName,
2021-07-19 19:26:06 +00:00
});
onProfileChanged(stagedProfile, { keepAvatar: true });
2021-07-19 19:26:06 +00:00
handleBack();
}}
>
2023-03-30 00:03:25 +00:00
{i18n('icu:save')}
2021-07-19 19:26:06 +00:00
</Button>
2021-08-06 00:17:05 +00:00
</Modal.ButtonFooter>
2021-07-19 19:26:06 +00:00
</>
);
} else if (editState === EditState.Bio) {
const shouldDisableSave =
stagedProfile.aboutText === fullBio.aboutText &&
stagedProfile.aboutEmoji === fullBio.aboutEmoji;
const defaultBios = getDefaultBios(i18n);
2021-07-19 19:26:06 +00:00
content = (
<>
<Input
expandable
hasClearButton
i18n={i18n}
icon={
<div className="module-composition-area__button-cell">
<EmojiButton
2022-10-18 17:12:02 +00:00
variant={EmojiButtonVariant.ProfileEditor}
2021-07-19 19:26:06 +00:00
closeOnPick
emoji={stagedProfile.aboutEmoji}
i18n={i18n}
onPickEmoji={setAboutEmoji}
onSetSkinTone={onSetSkinTone}
recentEmojis={recentEmojis}
skinTone={skinTone}
/>
</div>
}
maxLengthCount={140}
maxByteCount={512}
2021-07-19 19:26:06 +00:00
moduleClassName="ProfileEditor__about-input"
onChange={value => {
if (value) {
setStagedProfile(profileData => ({
...profileData,
aboutEmoji: stagedProfile.aboutEmoji,
2021-10-01 19:27:34 +00:00
aboutText: value.replace(/(\r\n|\n|\r)/gm, ''),
2021-07-19 19:26:06 +00:00
}));
} else {
setStagedProfile(profileData => ({
...profileData,
aboutEmoji: undefined,
aboutText: '',
}));
}
}}
ref={focusInputRef}
2023-03-30 00:03:25 +00:00
placeholder={i18n('icu:ProfileEditor--about-placeholder')}
2021-07-19 19:26:06 +00:00
value={stagedProfile.aboutText}
whenToShowRemainingCount={40}
/>
{defaultBios.map(defaultBio => (
2021-07-19 19:26:06 +00:00
<PanelRow
className="ProfileEditor__row"
key={defaultBio.shortName}
icon={
<div className="ProfileEditor__icon--container">
<Emoji shortName={defaultBio.shortName} size={24} />
</div>
}
label={defaultBio.i18nLabel}
2021-07-19 19:26:06 +00:00
onClick={() => {
const emojiData = getEmojiData(defaultBio.shortName, skinTone);
setStagedProfile(profileData => ({
...profileData,
aboutEmoji: unifiedToEmoji(emojiData.unified),
aboutText: defaultBio.i18nLabel,
2021-07-19 19:26:06 +00:00
}));
}}
/>
))}
2021-08-06 00:17:05 +00:00
<Modal.ButtonFooter>
2021-07-19 19:26:06 +00:00
<Button
onClick={() => {
const handleCancel = () => {
handleBack();
setStagedProfile(profileData => ({
...profileData,
...fullBio,
}));
};
const hasChanges =
stagedProfile.aboutText !== fullBio.aboutText ||
stagedProfile.aboutEmoji !== fullBio.aboutEmoji;
if (hasChanges) {
setConfirmDiscardAction(() => handleCancel);
} else {
handleCancel();
}
}}
variant={ButtonVariant.Secondary}
>
2023-03-30 00:03:25 +00:00
{i18n('icu:cancel')}
2021-07-19 19:26:06 +00:00
</Button>
<Button
disabled={shouldDisableSave}
2021-07-19 19:26:06 +00:00
onClick={() => {
setFullBio({
aboutEmoji: stagedProfile.aboutEmoji,
aboutText: stagedProfile.aboutText,
});
onProfileChanged(stagedProfile, { keepAvatar: true });
2021-07-19 19:26:06 +00:00
handleBack();
}}
>
2023-03-30 00:03:25 +00:00
{i18n('icu:save')}
2021-07-19 19:26:06 +00:00
</Button>
2021-08-06 00:17:05 +00:00
</Modal.ButtonFooter>
2021-07-19 19:26:06 +00:00
</>
);
} else if (editState === EditState.Username) {
2022-10-18 17:12:02 +00:00
content = renderEditUsernameModalBody({
2024-02-06 18:35:59 +00:00
isRootModal: initialEditState === editState,
2022-10-18 17:12:02 +00:00
onClose: () => setEditState(EditState.None),
});
2023-07-20 03:14:08 +00:00
} else if (editState === EditState.UsernameLink) {
content = (
<UsernameLinkModalBody
i18n={i18n}
link={usernameLink}
username={username ?? ''}
colorId={usernameLinkColor}
usernameLinkCorrupted={usernameLinkCorrupted}
2023-07-20 03:14:08 +00:00
usernameLinkState={usernameLinkState}
setUsernameLinkColor={setUsernameLinkColor}
resetUsernameLink={resetUsernameLink}
saveAttachment={saveAttachment}
showToast={showToast}
onBack={() => setEditState(EditState.None)}
2023-07-20 03:14:08 +00:00
/>
);
2022-10-18 17:12:02 +00:00
} else if (editState === EditState.None) {
2024-02-08 00:34:31 +00:00
let actions: JSX.Element | undefined;
let alwaysShowActions = false;
if (usernameEditState === UsernameEditState.Deleting) {
actions = (
<ConversationDetailsIcon
ariaLabel={i18n('icu:ProfileEditor--username--deleting-username')}
icon={IconType.spinner}
disabled
fakeButton
/>
);
} else {
const menuOptions = [
{
group: 'copy',
icon: 'ProfileEditor__username-menu__copy-icon',
label: i18n('icu:ProfileEditor--username--copy'),
onClick: () => {
assertDev(
username !== undefined,
'Should not be visible without username'
);
void window.navigator.clipboard.writeText(username);
showToast({ toastType: ToastType.CopiedUsername });
},
},
{
// Different group to display a divider above it
group: 'delete',
2022-10-18 17:12:02 +00:00
2024-02-08 00:34:31 +00:00
icon: 'ProfileEditor__username-menu__trash-icon',
label: i18n('icu:ProfileEditor--username--delete'),
onClick: () => {
2024-02-14 20:30:32 +00:00
if (isUsernameDeletionEnabled) {
setUsernameEditState(UsernameEditState.ConfirmingDelete);
} else {
setIsUsernameNoticeVisible(true);
}
2024-02-08 00:34:31 +00:00
},
},
];
if (usernameCorrupted) {
2022-10-18 17:12:02 +00:00
actions = (
2024-02-08 00:34:31 +00:00
<i
className="ProfileEditor__error-icon"
title={i18n('icu:ProfileEditor__username__error-icon')}
/>
);
alwaysShowActions = true;
} else if (username) {
actions = (
<ContextMenu
i18n={i18n}
menuOptions={menuOptions}
popperOptions={{ placement: 'bottom', strategy: 'absolute' }}
moduleClassName="ProfileEditor__username-menu"
ariaLabel={i18n('icu:ProfileEditor--username--context-menu')}
2022-10-18 17:12:02 +00:00
/>
);
}
2024-02-08 00:34:31 +00:00
}
2022-10-18 17:12:02 +00:00
2024-02-08 00:34:31 +00:00
let maybeUsernameLinkRow: JSX.Element | undefined;
if (username && !usernameCorrupted) {
let linkActions: JSX.Element | undefined;
2024-02-06 18:35:59 +00:00
2024-02-08 00:34:31 +00:00
if (usernameLinkCorrupted) {
linkActions = (
<i
className="ProfileEditor__error-icon"
title={i18n('icu:ProfileEditor__username-link__error-icon')}
2023-07-20 03:14:08 +00:00
/>
);
2024-02-08 00:34:31 +00:00
}
2023-07-20 03:14:08 +00:00
2024-02-08 00:34:31 +00:00
maybeUsernameLinkRow = (
<PanelRow
className="ProfileEditor__row"
icon={
<i className="ProfileEditor__icon--container ProfileEditor__icon ProfileEditor__icon--username-link" />
}
label={i18n('icu:ProfileEditor__username-link')}
onClick={() => {
2024-02-16 22:01:06 +00:00
markCompletedUsernameLinkOnboarding();
2024-02-08 00:34:31 +00:00
if (usernameLinkCorrupted) {
setIsResettingUsernameLink(true);
return;
}
2023-07-20 03:14:08 +00:00
2024-02-08 00:34:31 +00:00
setEditState(EditState.UsernameLink);
}}
alwaysShowActions
actions={linkActions}
/>
);
2023-07-20 03:14:08 +00:00
2024-02-16 22:01:06 +00:00
if (!hasCompletedUsernameLinkOnboarding && !usernameLink) {
2024-02-08 00:34:31 +00:00
maybeUsernameLinkRow = (
<UsernameLinkTooltip
handleClose={markCompletedUsernameLinkOnboarding}
i18n={i18n}
2024-02-08 00:34:31 +00:00
>
{maybeUsernameLinkRow}
</UsernameLinkTooltip>
2024-02-08 00:34:31 +00:00
);
2023-07-20 03:14:08 +00:00
}
2024-02-08 00:34:31 +00:00
}
2023-07-20 03:14:08 +00:00
2024-02-08 00:34:31 +00:00
const usernameRows = (
<>
<hr className="ProfileEditor__divider" />
<PanelRow
className="ProfileEditor__row"
icon={
<i className="ProfileEditor__icon--container ProfileEditor__icon ProfileEditor__icon--username" />
}
label={
(!usernameCorrupted && username) ||
i18n('icu:ProfileEditor--username')
}
onClick={() => {
if (usernameCorrupted) {
setIsResettingUsername(true);
return;
}
2024-02-08 00:34:31 +00:00
openUsernameReservationModal();
setEditState(EditState.Username);
}}
alwaysShowActions={alwaysShowActions}
actions={actions}
/>
{maybeUsernameLinkRow}
<div className="ProfileEditor__info">
{username
? i18n('icu:ProfileEditor--info--pnp')
: i18n('icu:ProfileEditor--info--pnp--no-username')}
</div>
</>
);
2021-07-19 19:26:06 +00:00
content = (
<>
2021-08-06 00:17:05 +00:00
<AvatarPreview
avatarColor={color}
avatarPath={startingAvatarPath}
2021-08-06 00:17:05 +00:00
avatarValue={avatarBuffer}
conversationTitle={getFullNameText()}
2021-07-19 19:26:06 +00:00
i18n={i18n}
2021-08-06 00:17:05 +00:00
onAvatarLoaded={handleAvatarLoaded}
onClick={() => {
setEditState(EditState.BetterAvatar);
}}
style={{
2021-08-06 21:35:25 +00:00
height: 80,
width: 80,
2021-07-19 19:26:06 +00:00
}}
/>
2024-02-06 18:35:59 +00:00
<div className="ProfileEditor__EditPhotoContainer">
<Button
onClick={() => {
setEditState(EditState.BetterAvatar);
}}
variant={ButtonVariant.Secondary}
className="ProfileEditor__EditPhoto"
>
{i18n('icu:ProfileEditor--edit-photo')}
</Button>
</div>
2021-07-19 19:26:06 +00:00
<PanelRow
className="ProfileEditor__row"
icon={
<i className="ProfileEditor__icon--container ProfileEditor__icon ProfileEditor__icon--name" />
}
2023-04-20 17:03:43 +00:00
label={<UserText text={getFullNameText()} />}
2021-07-19 19:26:06 +00:00
onClick={() => {
setEditState(EditState.ProfileName);
}}
/>
<PanelRow
className="ProfileEditor__row"
icon={
fullBio.aboutEmoji ? (
<div className="ProfileEditor__icon--container">
<Emoji emoji={fullBio.aboutEmoji} size={24} />
</div>
) : (
<i className="ProfileEditor__icon--container ProfileEditor__icon ProfileEditor__icon--bio" />
)
}
label={
2023-04-20 17:03:43 +00:00
<UserText
2023-03-30 00:03:25 +00:00
text={fullBio.aboutText || i18n('icu:ProfileEditor--about')}
/>
}
2021-07-19 19:26:06 +00:00
onClick={() => {
setEditState(EditState.Bio);
}}
/>
<div className="ProfileEditor__info">
2024-02-06 18:35:59 +00:00
{i18n('icu:ProfileEditor--info--general')}
2021-07-19 19:26:06 +00:00
</div>
2024-02-08 00:34:31 +00:00
{usernameRows}
2021-07-19 19:26:06 +00:00
</>
);
} else {
throw missingCaseError(editState);
}
return (
<>
{usernameEditState === UsernameEditState.ConfirmingDelete && (
<ConfirmationDialog
2022-09-27 20:24:21 +00:00
dialogName="ProfileEditor.confirmDeleteUsername"
i18n={i18n}
onClose={() => setUsernameEditState(UsernameEditState.Editing)}
actions={[
{
2023-03-30 00:03:25 +00:00
text: i18n('icu:ProfileEditor--username--confirm-delete-button'),
style: 'negative',
action: () => deleteUsername(),
},
]}
>
2023-07-20 03:14:08 +00:00
{i18n('icu:ProfileEditor--username--confirm-delete-body-2', {
2024-03-04 18:03:11 +00:00
username: username ?? '',
2023-07-20 03:14:08 +00:00
})}
</ConfirmationDialog>
)}
2022-10-18 17:12:02 +00:00
2024-02-14 20:30:32 +00:00
{isUsernameNoticeVisible && (
<ConfirmationDialog
dialogName="ProfileEditor.confirmDeleteUsername"
i18n={i18n}
onClose={() => setIsUsernameNoticeVisible(false)}
cancelText={i18n('icu:ok')}
>
{i18n('icu:ProfileEditor--username--delete-unavailable-notice')}
</ConfirmationDialog>
)}
2021-07-19 19:26:06 +00:00
{confirmDiscardAction && (
2021-08-06 00:17:05 +00:00
<ConfirmDiscardDialog
2021-07-19 19:26:06 +00:00
i18n={i18n}
2021-08-06 00:17:05 +00:00
onDiscard={confirmDiscardAction}
2021-07-19 19:26:06 +00:00
onClose={() => setConfirmDiscardAction(undefined)}
2021-08-06 00:17:05 +00:00
/>
2021-07-19 19:26:06 +00:00
)}
2024-02-06 18:35:59 +00:00
{isResettingUsernameLink && (
<ConfirmationDialog
i18n={i18n}
dialogName="UsernameLinkModal__error"
onClose={() => setIsResettingUsernameLink(false)}
cancelButtonVariant={ButtonVariant.Secondary}
cancelText={i18n('icu:cancel')}
actions={[
{
action: () => {
setIsResettingUsernameLink(false);
setEditState(EditState.UsernameLink);
},
style: 'affirmative',
text: i18n('icu:UsernameLinkModalBody__error__fix-now'),
},
]}
>
{i18n('icu:UsernameLinkModalBody__error__text')}
</ConfirmationDialog>
)}
{isResettingUsername && (
<ConfirmationDialog
dialogName="ProfileEditor.confirmResetUsername"
moduleClassName="ProfileEditor__reset-username-modal"
i18n={i18n}
onClose={() => setIsResettingUsername(false)}
actions={[
{
2024-02-06 18:35:59 +00:00
text: i18n('icu:ProfileEditor--username--corrupted--fix-button'),
style: 'affirmative',
action: () => {
openUsernameReservationModal();
setEditState(EditState.Username);
},
},
]}
>
{i18n('icu:ProfileEditor--username--corrupted--body')}
</ConfirmationDialog>
)}
2021-07-19 19:26:06 +00:00
<div className="ProfileEditor">{content}</div>
</>
);
2022-11-18 00:45:19 +00:00
}
function UsernameLinkTooltip({
handleClose,
children,
i18n,
}: {
handleClose: VoidFunction;
children: React.ReactNode;
i18n: LocalizerType;
}) {
const animatedStyles = useSpring({
from: { opacity: 0, scale: 0.25 },
to: { opacity: 1, scale: 1 },
config: { mass: 1, tension: 280, friction: 25 },
delay: 200,
});
const tooltip = (
<animated.div
className="ProfileEditor__username-link__tooltip__container"
style={animatedStyles}
>
<div className="ProfileEditor__username-link__tooltip__icon" />
<div className="ProfileEditor__username-link__tooltip__content">
<h3>{i18n('icu:ProfileEditor__username-link__tooltip__title')}</h3>
<p>{i18n('icu:ProfileEditor__username-link__tooltip__body')}</p>
</div>
<button
type="button"
className="ProfileEditor__username-link__tooltip__close"
onClick={handleClose}
aria-label={i18n('icu:close')}
/>
<div className="ProfileEditor__username-link__tooltip__arrow" />
</animated.div>
);
return (
<Tooltip
className="ProfileEditor__username-link__tooltip"
direction={TooltipPlacement.Bottom}
sticky
content={tooltip}
// By default tooltip has its distance modified, here we clear that
popperModifiers={[offsetDistanceModifier(0)]}
hideArrow
>
{children}
</Tooltip>
);
}