| 
									
										
										
										
											2022-10-04 17:17:15 -06: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-09 17:40:19 -07:00
										 |  |  | import type { HydratedBodyRangesType } from '../types/BodyRange'; | 
					
						
							| 
									
										
										
										
											2022-10-04 17:17:15 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | export type Props = { | 
					
						
							|  |  |  |   i18n: LocalizerType; | 
					
						
							|  |  |  |   onClose: () => void; | 
					
						
							| 
									
										
										
										
											2023-05-09 17:40:19 -07:00
										 |  |  |   onSubmit: ( | 
					
						
							|  |  |  |     text: string, | 
					
						
							|  |  |  |     bodyRanges: HydratedBodyRangesType | undefined | 
					
						
							|  |  |  |   ) => void; | 
					
						
							| 
									
										
										
										
											2022-10-04 17:17:15 -06:00
										 |  |  |   draftText: string; | 
					
						
							| 
									
										
										
										
											2023-05-09 17:40:19 -07:00
										 |  |  |   draftBodyRanges: HydratedBodyRangesType | undefined; | 
					
						
							| 
									
										
										
										
											2022-10-04 17:17:15 -06:00
										 |  |  |   theme: ThemeType; | 
					
						
							|  |  |  |   RenderCompositionTextArea: ( | 
					
						
							|  |  |  |     props: SmartCompositionTextAreaProps | 
					
						
							|  |  |  |   ) => JSX.Element; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  | export function AddCaptionModal({ | 
					
						
							| 
									
										
										
										
											2022-10-04 17:17:15 -06:00
										 |  |  |   i18n, | 
					
						
							|  |  |  |   onClose, | 
					
						
							|  |  |  |   onSubmit, | 
					
						
							|  |  |  |   draftText, | 
					
						
							| 
									
										
										
										
											2023-05-09 17:40:19 -07:00
										 |  |  |   draftBodyRanges, | 
					
						
							| 
									
										
										
										
											2022-10-04 17:17:15 -06:00
										 |  |  |   RenderCompositionTextArea, | 
					
						
							|  |  |  |   theme, | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  | }: Props): JSX.Element { | 
					
						
							| 
									
										
										
										
											2022-10-04 17:17:15 -06:00
										 |  |  |   const [messageText, setMessageText] = React.useState(''); | 
					
						
							| 
									
										
										
										
											2023-05-09 17:40:19 -07:00
										 |  |  |   const [bodyRanges, setBodyRanges] = React.useState< | 
					
						
							|  |  |  |     HydratedBodyRangesType | undefined | 
					
						
							|  |  |  |   >(); | 
					
						
							| 
									
										
										
										
											2022-10-04 17:17:15 -06: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-09 17:40:19 -07:00
										 |  |  |     onSubmit(messageText, bodyRanges); | 
					
						
							|  |  |  |   }, [bodyRanges, messageText, onSubmit]); | 
					
						
							| 
									
										
										
										
											2022-10-04 17:17:15 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return ( | 
					
						
							|  |  |  |     <Modal | 
					
						
							|  |  |  |       i18n={i18n} | 
					
						
							|  |  |  |       modalName="AddCaptionModal" | 
					
						
							|  |  |  |       hasXButton | 
					
						
							|  |  |  |       hasHeaderDivider={!isScrolledTop} | 
					
						
							|  |  |  |       hasFooterDivider={!isScrolledBottom} | 
					
						
							|  |  |  |       moduleClassName="AddCaptionModal" | 
					
						
							|  |  |  |       padded={false} | 
					
						
							| 
									
										
										
										
											2023-03-29 17:03:25 -07:00
										 |  |  |       title={i18n('icu:AddCaptionModal__title')} | 
					
						
							| 
									
										
										
										
											2022-10-04 17:17:15 -06:00
										 |  |  |       onClose={onClose} | 
					
						
							|  |  |  |       modalFooter={ | 
					
						
							|  |  |  |         <Button onClick={handleSubmit}> | 
					
						
							| 
									
										
										
										
											2023-03-29 17:03:25 -07:00
										 |  |  |           {i18n('icu:AddCaptionModal__submit-button')} | 
					
						
							| 
									
										
										
										
											2022-10-04 17:17:15 -06:00
										 |  |  |         </Button> | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     > | 
					
						
							|  |  |  |       <RenderCompositionTextArea | 
					
						
							|  |  |  |         maxLength={1500} | 
					
						
							|  |  |  |         whenToShowRemainingCount={1450} | 
					
						
							| 
									
										
										
										
											2023-03-29 17:03:25 -07:00
										 |  |  |         placeholder={i18n('icu:AddCaptionModal__placeholder')} | 
					
						
							| 
									
										
										
										
											2023-05-09 17:40:19 -07:00
										 |  |  |         onChange={(updatedMessageText, updatedBodyRanges) => { | 
					
						
							|  |  |  |           setMessageText(updatedMessageText); | 
					
						
							|  |  |  |           setBodyRanges(updatedBodyRanges); | 
					
						
							|  |  |  |         }} | 
					
						
							| 
									
										
										
										
											2022-10-04 17:17:15 -06:00
										 |  |  |         scrollerRef={scrollerRef} | 
					
						
							|  |  |  |         draftText={draftText} | 
					
						
							| 
									
										
										
										
											2023-05-09 17:40:19 -07:00
										 |  |  |         bodyRanges={draftBodyRanges} | 
					
						
							| 
									
										
										
										
											2022-10-04 17:17:15 -06:00
										 |  |  |         onSubmit={noop} | 
					
						
							|  |  |  |         onScroll={updateScrollState} | 
					
						
							|  |  |  |         theme={theme} | 
					
						
							|  |  |  |       /> | 
					
						
							|  |  |  |     </Modal> | 
					
						
							|  |  |  |   ); | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  | } |