186 lines
5.2 KiB
TypeScript
186 lines
5.2 KiB
TypeScript
|
// Copyright 2024 Signal Messenger, LLC
|
||
|
// SPDX-License-Identifier: AGPL-3.0-only
|
||
|
import type { FormEvent } from 'react';
|
||
|
import React, { useCallback, useMemo, useState } from 'react';
|
||
|
import uuid from 'uuid';
|
||
|
import { z } from 'zod';
|
||
|
import { Modal } from './Modal';
|
||
|
import type { LocalizerType } from '../types/I18N';
|
||
|
import { Avatar, AvatarSize } from './Avatar';
|
||
|
import type {
|
||
|
ConversationType,
|
||
|
NicknameAndNote,
|
||
|
} from '../state/ducks/conversations';
|
||
|
import { Input } from './Input';
|
||
|
import { AutoSizeTextArea } from './AutoSizeTextArea';
|
||
|
import { Button, ButtonVariant } from './Button';
|
||
|
|
||
|
const formSchema = z.object({
|
||
|
nickname: z
|
||
|
.object({
|
||
|
givenName: z.string(),
|
||
|
familyName: z.string().nullable(),
|
||
|
})
|
||
|
.nullable(),
|
||
|
note: z.string().nullable(),
|
||
|
});
|
||
|
|
||
|
function toOptionalStringValue(value: string): string | null {
|
||
|
const trimmed = value.trim();
|
||
|
return trimmed === '' ? null : trimmed;
|
||
|
}
|
||
|
|
||
|
export type EditNicknameAndNoteModalProps = Readonly<{
|
||
|
conversation: ConversationType;
|
||
|
i18n: LocalizerType;
|
||
|
onSave: (result: NicknameAndNote) => void;
|
||
|
onClose: () => void;
|
||
|
}>;
|
||
|
|
||
|
export function EditNicknameAndNoteModal({
|
||
|
conversation,
|
||
|
i18n,
|
||
|
onSave,
|
||
|
onClose,
|
||
|
}: EditNicknameAndNoteModalProps): JSX.Element {
|
||
|
const [givenName, setGivenName] = useState(
|
||
|
conversation.nicknameGivenName ?? ''
|
||
|
);
|
||
|
const [familyName, setFamilyName] = useState(
|
||
|
conversation.nicknameFamilyName ?? ''
|
||
|
);
|
||
|
const [note, setNote] = useState(conversation.note ?? '');
|
||
|
|
||
|
const [formId] = useState(() => uuid());
|
||
|
const [givenNameId] = useState(() => uuid());
|
||
|
const [familyNameId] = useState(() => uuid());
|
||
|
const [noteId] = useState(() => uuid());
|
||
|
|
||
|
const formResult = useMemo(() => {
|
||
|
const givenNameValue = toOptionalStringValue(givenName);
|
||
|
const familyNameValue = toOptionalStringValue(familyName);
|
||
|
const noteValue = toOptionalStringValue(note);
|
||
|
const hasEitherName = givenNameValue != null || familyNameValue != null;
|
||
|
return formSchema.safeParse({
|
||
|
nickname: hasEitherName
|
||
|
? { givenName: givenNameValue, familyName: familyNameValue }
|
||
|
: null,
|
||
|
note: noteValue,
|
||
|
});
|
||
|
}, [givenName, familyName, note]);
|
||
|
|
||
|
const handleSubmit = useCallback(
|
||
|
(event: MouseEvent | FormEvent) => {
|
||
|
event.preventDefault();
|
||
|
if (formResult.success) {
|
||
|
onSave(formResult.data);
|
||
|
onClose();
|
||
|
}
|
||
|
},
|
||
|
[formResult, onSave, onClose]
|
||
|
);
|
||
|
|
||
|
return (
|
||
|
<Modal
|
||
|
modalName="EditNicknameAndNoteModal"
|
||
|
moduleClassName="EditNicknameAndNoteModal"
|
||
|
i18n={i18n}
|
||
|
onClose={onClose}
|
||
|
title={i18n('icu:EditNicknameAndNoteModal__Title')}
|
||
|
hasXButton
|
||
|
modalFooter={
|
||
|
<>
|
||
|
<Button variant={ButtonVariant.Secondary} onClick={onClose}>
|
||
|
{i18n('icu:cancel')}
|
||
|
</Button>
|
||
|
<Button
|
||
|
variant={ButtonVariant.Primary}
|
||
|
type="submit"
|
||
|
form={formId}
|
||
|
aria-disabled={!formResult.success}
|
||
|
onClick={handleSubmit}
|
||
|
>
|
||
|
{i18n('icu:save')}
|
||
|
</Button>
|
||
|
</>
|
||
|
}
|
||
|
>
|
||
|
<p className="EditNicknameAndNoteModal__Description">
|
||
|
{i18n('icu:EditNicknameAndNoteModal__Description')}
|
||
|
</p>
|
||
|
<div className="EditNicknameAndNoteModal__Avatar">
|
||
|
<Avatar
|
||
|
{...conversation}
|
||
|
conversationType={conversation.type}
|
||
|
i18n={i18n}
|
||
|
size={AvatarSize.EIGHTY}
|
||
|
badge={undefined}
|
||
|
theme={undefined}
|
||
|
/>
|
||
|
</div>
|
||
|
<form onSubmit={handleSubmit}>
|
||
|
<label
|
||
|
htmlFor={givenNameId}
|
||
|
className="EditNicknameAndNoteModal__Label"
|
||
|
>
|
||
|
{i18n('icu:EditNicknameAndNoteModal__FirstName__Label')}
|
||
|
</label>
|
||
|
<Input
|
||
|
id={givenNameId}
|
||
|
i18n={i18n}
|
||
|
placeholder={i18n(
|
||
|
'icu:EditNicknameAndNoteModal__FirstName__Placeholder'
|
||
|
)}
|
||
|
value={givenName}
|
||
|
hasClearButton
|
||
|
maxLengthCount={26}
|
||
|
maxByteCount={128}
|
||
|
onChange={value => {
|
||
|
setGivenName(value);
|
||
|
}}
|
||
|
/>
|
||
|
<label
|
||
|
htmlFor={familyNameId}
|
||
|
className="EditNicknameAndNoteModal__Label"
|
||
|
>
|
||
|
{i18n('icu:EditNicknameAndNoteModal__LastName__Label')}
|
||
|
</label>
|
||
|
<Input
|
||
|
id={familyNameId}
|
||
|
i18n={i18n}
|
||
|
placeholder={i18n(
|
||
|
'icu:EditNicknameAndNoteModal__LastName__Placeholder'
|
||
|
)}
|
||
|
value={familyName}
|
||
|
hasClearButton
|
||
|
maxLengthCount={26}
|
||
|
maxByteCount={128}
|
||
|
onChange={value => {
|
||
|
setFamilyName(value);
|
||
|
}}
|
||
|
/>
|
||
|
|
||
|
<label htmlFor={noteId} className="EditNicknameAndNoteModal__Label">
|
||
|
{i18n('icu:EditNicknameAndNoteModal__Note__Label')}
|
||
|
</label>
|
||
|
<AutoSizeTextArea
|
||
|
i18n={i18n}
|
||
|
id={noteId}
|
||
|
placeholder={i18n('icu:EditNicknameAndNoteModal__Note__Placeholder')}
|
||
|
value={note}
|
||
|
maxByteCount={240}
|
||
|
maxLengthCount={240}
|
||
|
whenToShowRemainingCount={140}
|
||
|
whenToWarnRemainingCount={235}
|
||
|
onChange={value => {
|
||
|
setNote(value);
|
||
|
}}
|
||
|
/>
|
||
|
<button type="submit" hidden>
|
||
|
{i18n('icu:submit')}
|
||
|
</button>
|
||
|
</form>
|
||
|
</Modal>
|
||
|
);
|
||
|
}
|