// Copyright 2016-2020 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import React, { useCallback, useEffect, useMemo, useState } from 'react'; import * as moment from 'moment'; import { noop } from 'lodash'; import { AttachmentType } from '../../types/Attachment'; import { ConfirmationDialog } from '../ConfirmationDialog'; import { LocalizerType } from '../../types/Util'; import { ErrorDialogAudioRecorderType } from '../../state/ducks/audioRecorder'; import { ToastVoiceNoteLimit } from '../ToastVoiceNoteLimit'; import { ToastVoiceNoteMustBeOnlyAttachment } from '../ToastVoiceNoteMustBeOnlyAttachment'; import { useEscapeHandling } from '../../hooks/useEscapeHandling'; import { getStartRecordingShortcut, useKeyboardShortcuts, } from '../../hooks/useKeyboardShortcuts'; type OnSendAudioRecordingType = (rec: AttachmentType) => unknown; export type PropsType = { cancelRecording: () => unknown; conversationId: string; completeRecording: ( conversationId: string, onSendAudioRecording?: OnSendAudioRecordingType ) => unknown; draftAttachments: ReadonlyArray; errorDialogAudioRecorderType?: ErrorDialogAudioRecorderType; errorRecording: (e: ErrorDialogAudioRecorderType) => unknown; i18n: LocalizerType; isRecording: boolean; onSendAudioRecording: OnSendAudioRecordingType; startRecording: () => unknown; }; enum ToastType { VoiceNoteLimit, VoiceNoteMustBeOnlyAttachment, } const START_DURATION_TEXT = '0:00'; export const AudioCapture = ({ cancelRecording, completeRecording, conversationId, draftAttachments, errorDialogAudioRecorderType, errorRecording, i18n, isRecording, onSendAudioRecording, startRecording, }: PropsType): JSX.Element => { const [durationText, setDurationText] = useState(START_DURATION_TEXT); const [toastType, setToastType] = useState(); // Cancel recording if we switch away from this conversation, unmounting useEffect(() => { if (!isRecording) { return; } return () => { cancelRecording(); }; }, [cancelRecording, isRecording]); // Stop recording and show confirmation if user switches away from this app useEffect(() => { if (!isRecording) { return; } const handler = () => { errorRecording(ErrorDialogAudioRecorderType.Blur); }; window.addEventListener('blur', handler); return () => { window.removeEventListener('blur', handler); }; }, [isRecording, completeRecording, errorRecording]); const escapeRecording = useCallback(() => { if (!isRecording) { return; } cancelRecording(); }, [cancelRecording, isRecording]); useEscapeHandling(escapeRecording); const startRecordingShortcut = useMemo(() => { return getStartRecordingShortcut(startRecording); }, [startRecording]); useKeyboardShortcuts(startRecordingShortcut); // Update timestamp regularly, then timeout if recording goes over five minutes useEffect(() => { if (!isRecording) { return; } const startTime = Date.now(); const interval = setInterval(() => { const duration = moment.duration(Date.now() - startTime, 'ms'); const minutes = `${Math.trunc(duration.asMinutes())}`; let seconds = `${duration.seconds()}`; if (seconds.length < 2) { seconds = `0${seconds}`; } setDurationText(`${minutes}:${seconds}`); if (duration >= moment.duration(5, 'minutes')) { errorRecording(ErrorDialogAudioRecorderType.Timeout); } }, 1000); return () => { clearInterval(interval); }; }, [completeRecording, errorRecording, isRecording, setDurationText]); const clickCancel = useCallback(() => { cancelRecording(); }, [cancelRecording]); const clickSend = useCallback(() => { completeRecording(conversationId, onSendAudioRecording); }, [conversationId, completeRecording, onSendAudioRecording]); function closeToast() { setToastType(undefined); } let toastElement: JSX.Element | undefined; if (toastType === ToastType.VoiceNoteLimit) { toastElement = ; } else if (toastType === ToastType.VoiceNoteMustBeOnlyAttachment) { toastElement = ( ); } let confirmationDialog: JSX.Element | undefined; if ( errorDialogAudioRecorderType === ErrorDialogAudioRecorderType.Blur || errorDialogAudioRecorderType === ErrorDialogAudioRecorderType.Timeout ) { const confirmationDialogText = errorDialogAudioRecorderType === ErrorDialogAudioRecorderType.Blur ? i18n('voiceRecordingInterruptedBlur') : i18n('voiceRecordingInterruptedMax'); confirmationDialog = ( {confirmationDialogText} ); } else if ( errorDialogAudioRecorderType === ErrorDialogAudioRecorderType.ErrorRecording ) { confirmationDialog = ( {i18n('voiceNoteError')} ); } if (isRecording && !confirmationDialog) { return ( <>
{durationText}
{toastElement} ); } return ( <>
{toastElement} ); };