import React from 'react';
import PropTypes from 'prop-types';
import { View, Text } from 'react-native';
import { Audio } from 'expo-av';
import { BorderlessButton } from 'react-native-gesture-handler';
import { getInfoAsync } from 'expo-file-system';
import { deactivateKeepAwake, activateKeepAwake } from 'expo-keep-awake';
import styles from './styles';
import I18n from '../../i18n';
import { themes } from '../../constants/colors';
import { CustomIcon } from '../../lib/Icons';
import { logEvent, events } from '../../utils/log';
const RECORDING_EXTENSION = '.aac';
const RECORDING_SETTINGS = {
android: {
extension: RECORDING_EXTENSION,
outputFormat: Audio.RECORDING_OPTION_ANDROID_OUTPUT_FORMAT_AAC_ADTS,
audioEncoder: Audio.RECORDING_OPTION_ANDROID_AUDIO_ENCODER_AAC,
sampleRate: Audio.RECORDING_OPTIONS_PRESET_LOW_QUALITY.android.sampleRate,
numberOfChannels: Audio.RECORDING_OPTIONS_PRESET_LOW_QUALITY.android.numberOfChannels,
bitRate: Audio.RECORDING_OPTIONS_PRESET_LOW_QUALITY.android.bitRate
},
ios: {
audioQuality: Audio.RECORDING_OPTION_IOS_AUDIO_QUALITY_MIN,
sampleRate: Audio.RECORDING_OPTIONS_PRESET_LOW_QUALITY.ios.sampleRate,
numberOfChannels: Audio.RECORDING_OPTIONS_PRESET_LOW_QUALITY.ios.numberOfChannels,
bitRate: Audio.RECORDING_OPTIONS_PRESET_LOW_QUALITY.ios.bitRate,
outputFormat: Audio.RECORDING_OPTION_IOS_OUTPUT_FORMAT_MPEG4AAC
}
};
const RECORDING_MODE = {
allowsRecordingIOS: true,
playsInSilentModeIOS: true,
staysActiveInBackground: false,
shouldDuckAndroid: true,
playThroughEarpieceAndroid: false,
interruptionModeIOS: Audio.INTERRUPTION_MODE_IOS_DO_NOT_MIX,
interruptionModeAndroid: Audio.INTERRUPTION_MODE_ANDROID_DO_NOT_MIX
const formatTime = function(seconds) {
let minutes = Math.floor(seconds / 60);
seconds %= 60;
if (minutes < 10) { minutes = `0${ minutes }`; }
if (seconds < 10) { seconds = `0${ seconds }`; }
return `${ minutes }:${ seconds }`;
export default class RecordAudio extends React.PureComponent {
static propTypes = {
theme: PropTypes.string,
recordingCallback: PropTypes.func,
onFinish: PropTypes.func
constructor(props) {
super(props);
this.isRecorderBusy = false;
this.state = {
isRecording: false,
recordingDurationMillis: 0
componentDidUpdate() {
const { recordingCallback } = this.props;
const { isRecording } = this.state;
recordingCallback(isRecording);
componentWillUnmount() {
if (this.recording) {
this.cancelRecordingAudio();
get duration() {
const { recordingDurationMillis } = this.state;
return formatTime(Math.floor(recordingDurationMillis / 1000));
isRecordingPermissionGranted = async() => {
try {
const permission = await Audio.getPermissionsAsync();
if (permission.status === 'granted') {
return true;
await Audio.requestPermissionsAsync();
} catch {
// Do nothing
return false;
onRecordingStatusUpdate = (status) => {
this.setState({
isRecording: status.isRecording,
recordingDurationMillis: status.durationMillis
});
startRecordingAudio = async() => {
logEvent(events.ROOM_AUDIO_RECORD);
if (!this.isRecorderBusy) {
this.isRecorderBusy = true;
const canRecord = await this.isRecordingPermissionGranted();
if (canRecord) {
await Audio.setAudioModeAsync(RECORDING_MODE);
this.recording = new Audio.Recording();
await this.recording.prepareToRecordAsync(RECORDING_SETTINGS);
this.recording.setOnRecordingStatusUpdate(this.onRecordingStatusUpdate);
await this.recording.startAsync();
activateKeepAwake();
} else {
} catch (error) {
logEvent(events.ROOM_AUDIO_RECORD_F);
finishRecordingAudio = async() => {
logEvent(events.ROOM_AUDIO_FINISH);
const { onFinish } = this.props;
await this.recording.stopAndUnloadAsync();
const fileURI = this.recording.getURI();
const fileData = await getInfoAsync(fileURI);
const fileInfo = {
name: `${ Date.now() }.aac`,
mime: 'audio/aac',
type: 'audio/aac',
store: 'Uploads',
path: fileURI,
size: fileData.size
onFinish(fileInfo);
logEvent(events.ROOM_AUDIO_FINISH_F);
this.setState({ isRecording: false, recordingDurationMillis: 0 });
deactivateKeepAwake();
cancelRecordingAudio = async() => {
logEvent(events.ROOM_AUDIO_CANCEL);
logEvent(events.ROOM_AUDIO_CANCEL_F);
render() {
const { theme } = this.props;
if (!isRecording) {
return (
<BorderlessButton
onPress={this.startRecordingAudio}
style={styles.actionButton}
testID='messagebox-send-audio'
accessibilityLabel={I18n.t('Send_audio_message')}
accessibilityTraits='button'
>
<CustomIcon name='microphone' size={23} color={themes[theme].tintColor} />
</BorderlessButton>
);
<View style={styles.recordingContent}>
<View style={styles.textArea}>
onPress={this.cancelRecordingAudio}
accessibilityLabel={I18n.t('Cancel_recording')}
<CustomIcon
size={22}
color={themes[theme].dangerColor}
name='close'
/>
<Text
style={[styles.recordingCancelText, { color: themes[theme].titleText }]}
{this.duration}
</Text>
</View>
onPress={this.finishRecordingAudio}
accessibilityLabel={I18n.t('Finish_recording')}
color={themes[theme].successColor}
name='check'