2022-10-04 23:17:15 +00:00
|
|
|
// Copyright 2022 Signal Messenger, LLC
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
|
|
|
import React, { useEffect } from 'react';
|
|
|
|
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';
|
2023-05-10 00:40:19 +00:00
|
|
|
import type { HydratedBodyRangesType } from '../types/BodyRange';
|
2022-10-04 23:17:15 +00:00
|
|
|
|
|
|
|
export type Props = {
|
|
|
|
i18n: LocalizerType;
|
|
|
|
onClose: () => void;
|
2023-05-10 00:40:19 +00:00
|
|
|
onSubmit: (
|
|
|
|
text: string,
|
|
|
|
bodyRanges: HydratedBodyRangesType | undefined
|
|
|
|
) => void;
|
2022-10-04 23:17:15 +00:00
|
|
|
draftText: string;
|
2023-05-10 00:40:19 +00:00
|
|
|
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,
|
2023-05-10 00:40:19 +00:00
|
|
|
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('');
|
2023-05-10 00:40:19 +00:00
|
|
|
const [bodyRanges, setBodyRanges] = React.useState<
|
|
|
|
HydratedBodyRangesType | undefined
|
|
|
|
>();
|
2022-10-04 23:17:15 +00:00
|
|
|
|
|
|
|
const [isScrolledTop, setIsScrolledTop] = React.useState(true);
|
|
|
|
const [isScrolledBottom, setIsScrolledBottom] = React.useState(true);
|
|
|
|
|
|
|
|
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) {
|
|
|
|
setIsScrolledTop(scrollerEl.scrollTop === 0);
|
|
|
|
setIsScrolledBottom(
|
|
|
|
scrollerEl.scrollHeight - scrollerEl.scrollTop ===
|
|
|
|
scrollerEl.clientHeight
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}, [scrollerRef]);
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
updateScrollState();
|
|
|
|
}, [updateScrollState]);
|
|
|
|
|
|
|
|
const handleSubmit = React.useCallback(() => {
|
2023-05-10 00:40:19 +00:00
|
|
|
onSubmit(messageText, bodyRanges);
|
|
|
|
}, [bodyRanges, messageText, onSubmit]);
|
2022-10-04 23:17:15 +00:00
|
|
|
|
|
|
|
return (
|
|
|
|
<Modal
|
|
|
|
i18n={i18n}
|
|
|
|
modalName="AddCaptionModal"
|
|
|
|
hasXButton
|
|
|
|
hasHeaderDivider={!isScrolledTop}
|
|
|
|
hasFooterDivider={!isScrolledBottom}
|
|
|
|
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')}
|
2023-05-10 00:40:19 +00:00
|
|
|
onChange={(updatedMessageText, updatedBodyRanges) => {
|
|
|
|
setMessageText(updatedMessageText);
|
|
|
|
setBodyRanges(updatedBodyRanges);
|
|
|
|
}}
|
2022-10-04 23:17:15 +00:00
|
|
|
scrollerRef={scrollerRef}
|
|
|
|
draftText={draftText}
|
2023-05-10 00:40:19 +00:00
|
|
|
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
|
|
|
}
|