// Copyright 2023 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import { noop } from 'lodash'; import React, { useEffect, useRef, useState } from 'react'; import { useEscapeHandling } from '../hooks/useEscapeHandling'; import type { HideToastAction, ShowToastAction } from '../state/ducks/toast'; import type { InMemoryAttachmentDraftType } from '../types/Attachment'; import { ErrorDialogAudioRecorderType } from '../types/AudioRecorder'; import type { LocalizerType } from '../types/Util'; import type { AnyToast } from '../types/Toast'; import { ToastType } from '../types/Toast'; import { DurationInSeconds, SECOND } from '../util/durations'; import { durationToPlaybackText } from '../util/durationToPlaybackText'; import { ConfirmationDialog } from './ConfirmationDialog'; import { RecordingComposer } from './RecordingComposer'; export type Props = { i18n: LocalizerType; onCancel: () => void; onSend: () => void; errorRecording: (e: ErrorDialogAudioRecorderType) => unknown; errorDialogAudioRecorderType?: ErrorDialogAudioRecorderType; addAttachment: ( conversationId: string, attachment: InMemoryAttachmentDraftType ) => unknown; completeRecording: ( conversationId: string, onRecordingComplete: (rec: InMemoryAttachmentDraftType) => unknown ) => unknown; saveDraftRecordingIfNeeded: () => void; showToast: ShowToastAction; hideToast: HideToastAction; }; export function CompositionRecording({ i18n, onCancel, onSend, errorRecording, errorDialogAudioRecorderType, saveDraftRecordingIfNeeded, showToast, hideToast, }: Props): JSX.Element { useEscapeHandling(onCancel); // switched to another app useEffect(() => { window.addEventListener('blur', saveDraftRecordingIfNeeded); return () => { window.removeEventListener('blur', saveDraftRecordingIfNeeded); }; }, [saveDraftRecordingIfNeeded]); useEffect(() => { const toast: AnyToast = { toastType: ToastType.VoiceNoteLimit }; showToast(toast); return () => hideToast(toast); }, [showToast, hideToast]); const startTime = useRef(Date.now()); const [duration, setDuration] = useState(0); const drift = useRef(0); // update recording duration useEffect(() => { const timeoutId = setTimeout(() => { const now = Date.now(); const newDurationMs = now - startTime.current; drift.current = newDurationMs % SECOND; setDuration(newDurationMs / SECOND); if ( DurationInSeconds.fromMillis(newDurationMs) >= DurationInSeconds.HOUR ) { errorRecording(ErrorDialogAudioRecorderType.Timeout); } }, SECOND - drift.current); return () => { clearTimeout(timeoutId); }; }, [duration, errorRecording]); let confirmationDialog: JSX.Element | undefined; if (errorDialogAudioRecorderType === ErrorDialogAudioRecorderType.Timeout) { confirmationDialog = ( {i18n('icu:voiceRecordingInterruptedMax')} ); } else if ( errorDialogAudioRecorderType === ErrorDialogAudioRecorderType.ErrorRecording ) { confirmationDialog = ( {i18n('icu:voiceNoteError')} ); } return ( {durationToPlaybackText(duration)} {confirmationDialog} ); }