signal-desktop/ts/components/AddCaptionModal.tsx

96 lines
2.8 KiB
TypeScript
Raw Normal View History

2022-10-04 23:17:15 +00:00
// Copyright 2022 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
import React from 'react';
2022-10-04 23:17:15 +00:00
import { noop } from 'lodash';
import { Button } from './Button';
import { Modal } from './Modal';
import type { LocalizerType, ThemeType } from '../types/Util';
import type { SmartCompositionTextAreaProps } from '../state/smart/CompositionTextArea';
import type { HydratedBodyRangesType } from '../types/BodyRange';
import { isScrolled, isScrolledToBottom } from '../hooks/useSizeObserver';
2022-10-04 23:17:15 +00:00
export type Props = {
i18n: LocalizerType;
onClose: () => void;
onSubmit: (
text: string,
bodyRanges: HydratedBodyRangesType | undefined
) => void;
2022-10-04 23:17:15 +00:00
draftText: string;
draftBodyRanges: HydratedBodyRangesType | undefined;
2022-10-04 23:17:15 +00:00
theme: ThemeType;
RenderCompositionTextArea: (
props: SmartCompositionTextAreaProps
) => JSX.Element;
};
2022-11-18 00:45:19 +00:00
export function AddCaptionModal({
2022-10-04 23:17:15 +00:00
i18n,
onClose,
onSubmit,
draftText,
draftBodyRanges,
2022-10-04 23:17:15 +00:00
RenderCompositionTextArea,
theme,
2022-11-18 00:45:19 +00:00
}: Props): JSX.Element {
2022-10-04 23:17:15 +00:00
const [messageText, setMessageText] = React.useState('');
const [bodyRanges, setBodyRanges] = React.useState<
HydratedBodyRangesType | undefined
>();
2022-10-04 23:17:15 +00:00
const [scrolled, setScrolled] = React.useState(false);
// We don't know that this is true, but it most likely is
const [scrolledToBottom, setScrolledToBottom] = React.useState(true);
2022-10-04 23:17:15 +00:00
const scrollerRef = React.useRef<HTMLDivElement>(null);
// add footer/header dividers depending on the state of scroll
const updateScrollState = React.useCallback(() => {
const scrollerEl = scrollerRef.current;
if (scrollerEl) {
setScrolled(isScrolled(scrollerEl));
setScrolledToBottom(isScrolledToBottom(scrollerEl));
2022-10-04 23:17:15 +00:00
}
}, []);
2022-10-04 23:17:15 +00:00
const handleSubmit = React.useCallback(() => {
onSubmit(messageText, bodyRanges);
}, [bodyRanges, messageText, onSubmit]);
2022-10-04 23:17:15 +00:00
return (
<Modal
i18n={i18n}
modalName="AddCaptionModal"
hasXButton
hasHeaderDivider={scrolled}
hasFooterDivider={!scrolledToBottom}
2022-10-04 23:17:15 +00:00
moduleClassName="AddCaptionModal"
padded={false}
2023-03-30 00:03:25 +00:00
title={i18n('icu:AddCaptionModal__title')}
2022-10-04 23:17:15 +00:00
onClose={onClose}
modalFooter={
<Button onClick={handleSubmit}>
2023-03-30 00:03:25 +00:00
{i18n('icu:AddCaptionModal__submit-button')}
2022-10-04 23:17:15 +00:00
</Button>
}
>
<RenderCompositionTextArea
maxLength={1500}
whenToShowRemainingCount={1450}
2023-03-30 00:03:25 +00:00
placeholder={i18n('icu:AddCaptionModal__placeholder')}
onChange={(updatedMessageText, updatedBodyRanges) => {
setMessageText(updatedMessageText);
setBodyRanges(updatedBodyRanges);
}}
2022-10-04 23:17:15 +00:00
scrollerRef={scrollerRef}
draftText={draftText}
bodyRanges={draftBodyRanges}
2022-10-04 23:17:15 +00:00
onSubmit={noop}
onScroll={updateScrollState}
theme={theme}
/>
</Modal>
);
2022-11-18 00:45:19 +00:00
}