256 lines
8.0 KiB
TypeScript
256 lines
8.0 KiB
TypeScript
import React from 'react';
|
|
import { Text, View } from 'react-native';
|
|
import { Audio, InterruptionModeAndroid, InterruptionModeIOS } from 'expo-av';
|
|
import { BorderlessButton } from 'react-native-gesture-handler';
|
|
import { getInfoAsync } from 'expo-file-system';
|
|
import { activateKeepAwake, deactivateKeepAwake } from 'expo-keep-awake';
|
|
|
|
import styles from './styles';
|
|
import I18n from '../../i18n';
|
|
import { themes } from '../../lib/constants';
|
|
import { CustomIcon } from '../CustomIcon';
|
|
import { events, logEvent } from '../../lib/methods/helpers/log';
|
|
import { TSupportedThemes } from '../../theme';
|
|
|
|
interface IMessageBoxRecordAudioProps {
|
|
theme: TSupportedThemes;
|
|
permissionToUpload: boolean;
|
|
recordingCallback: Function;
|
|
onFinish: Function;
|
|
onStart: Function;
|
|
}
|
|
|
|
const RECORDING_EXTENSION = '.aac';
|
|
const RECORDING_SETTINGS = {
|
|
android: {
|
|
// Settings related to audio encoding.
|
|
extension: RECORDING_EXTENSION,
|
|
outputFormat: Audio.RECORDING_OPTION_ANDROID_OUTPUT_FORMAT_AAC_ADTS,
|
|
audioEncoder: Audio.RECORDING_OPTION_ANDROID_AUDIO_ENCODER_AAC,
|
|
// Settings related to audio quality.
|
|
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: {
|
|
// Settings related to audio encoding.
|
|
extension: RECORDING_EXTENSION,
|
|
audioQuality: Audio.RECORDING_OPTION_IOS_AUDIO_QUALITY_MEDIUM,
|
|
outputFormat: Audio.RECORDING_OPTION_IOS_OUTPUT_FORMAT_MPEG4AAC,
|
|
// Settings related to audio quality.
|
|
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
|
|
},
|
|
web: {}
|
|
};
|
|
|
|
const RECORDING_MODE = {
|
|
allowsRecordingIOS: true,
|
|
playsInSilentModeIOS: true,
|
|
staysActiveInBackground: true,
|
|
shouldDuckAndroid: true,
|
|
playThroughEarpieceAndroid: false,
|
|
interruptionModeIOS: InterruptionModeIOS.DoNotMix,
|
|
interruptionModeAndroid: InterruptionModeAndroid.DoNotMix
|
|
};
|
|
|
|
const formatTime = function (time: number) {
|
|
const minutes = Math.floor(time / 60);
|
|
const seconds = time % 60;
|
|
const min = minutes < 10 ? `0${minutes}` : minutes;
|
|
const sec = seconds < 10 ? `0${seconds}` : seconds;
|
|
return `${min}:${sec}`;
|
|
};
|
|
|
|
export default class RecordAudio extends React.PureComponent<IMessageBoxRecordAudioProps, any> {
|
|
private isRecorderBusy: boolean;
|
|
private recording!: Audio.Recording;
|
|
private LastDuration: number;
|
|
|
|
constructor(props: IMessageBoxRecordAudioProps) {
|
|
super(props);
|
|
this.isRecorderBusy = false;
|
|
this.LastDuration = 0;
|
|
this.state = {
|
|
isRecording: false,
|
|
isRecorderActive: false,
|
|
recordingDurationMillis: 0
|
|
};
|
|
}
|
|
|
|
componentDidUpdate() {
|
|
const { recordingCallback } = this.props;
|
|
const { isRecorderActive } = this.state;
|
|
|
|
recordingCallback(isRecorderActive);
|
|
}
|
|
|
|
componentWillUnmount() {
|
|
if (this.recording) {
|
|
this.cancelRecordingAudio();
|
|
}
|
|
}
|
|
|
|
get duration() {
|
|
const { recordingDurationMillis } = this.state;
|
|
return formatTime(Math.floor(recordingDurationMillis / 1000));
|
|
}
|
|
|
|
get GetLastDuration() {
|
|
return formatTime(Math.floor(this.LastDuration / 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: Audio.RecordingStatus) => {
|
|
this.setState({
|
|
isRecording: status.isRecording,
|
|
recordingDurationMillis: status.durationMillis
|
|
});
|
|
this.LastDuration = status.durationMillis;
|
|
};
|
|
|
|
startRecordingAudio = async () => {
|
|
const { onStart } = this.props;
|
|
onStart();
|
|
|
|
logEvent(events.ROOM_AUDIO_RECORD);
|
|
if (!this.isRecorderBusy) {
|
|
this.isRecorderBusy = true;
|
|
this.LastDuration = 0;
|
|
try {
|
|
const canRecord = await this.isRecordingPermissionGranted();
|
|
if (canRecord) {
|
|
await Audio.setAudioModeAsync(RECORDING_MODE);
|
|
|
|
this.setState({ isRecorderActive: true });
|
|
this.recording = new Audio.Recording();
|
|
await this.recording.prepareToRecordAsync(RECORDING_SETTINGS);
|
|
this.recording.setOnRecordingStatusUpdate(this.onRecordingStatusUpdate);
|
|
|
|
await this.recording.startAsync();
|
|
activateKeepAwake();
|
|
} else {
|
|
await Audio.requestPermissionsAsync();
|
|
}
|
|
} catch (error) {
|
|
logEvent(events.ROOM_AUDIO_RECORD_F);
|
|
}
|
|
this.isRecorderBusy = false;
|
|
}
|
|
};
|
|
|
|
finishRecordingAudio = async () => {
|
|
logEvent(events.ROOM_AUDIO_FINISH);
|
|
if (!this.isRecorderBusy) {
|
|
const { onFinish } = this.props;
|
|
|
|
this.isRecorderBusy = true;
|
|
try {
|
|
await this.recording.stopAndUnloadAsync();
|
|
|
|
const fileURI = this.recording.getURI();
|
|
const fileData = await getInfoAsync(fileURI as string);
|
|
const fileInfo = {
|
|
name: `${Date.now()}.aac`,
|
|
mime: 'audio/aac',
|
|
type: 'audio/aac',
|
|
store: 'Uploads',
|
|
path: fileURI,
|
|
size: fileData.exists ? fileData.size : null
|
|
};
|
|
|
|
onFinish(fileInfo);
|
|
} catch (error) {
|
|
logEvent(events.ROOM_AUDIO_FINISH_F);
|
|
}
|
|
this.setState({ isRecording: false, isRecorderActive: false, recordingDurationMillis: 0 });
|
|
deactivateKeepAwake();
|
|
this.isRecorderBusy = false;
|
|
}
|
|
};
|
|
|
|
cancelRecordingAudio = async () => {
|
|
logEvent(events.ROOM_AUDIO_CANCEL);
|
|
if (!this.isRecorderBusy) {
|
|
this.isRecorderBusy = true;
|
|
try {
|
|
await this.recording.stopAndUnloadAsync();
|
|
} catch (error) {
|
|
logEvent(events.ROOM_AUDIO_CANCEL_F);
|
|
}
|
|
this.setState({ isRecording: false, isRecorderActive: false, recordingDurationMillis: 0 });
|
|
deactivateKeepAwake();
|
|
this.isRecorderBusy = false;
|
|
}
|
|
};
|
|
|
|
render() {
|
|
const { theme, permissionToUpload } = this.props;
|
|
const { isRecording, isRecorderActive } = this.state;
|
|
if (!permissionToUpload) {
|
|
return null;
|
|
}
|
|
if (!isRecording && !isRecorderActive) {
|
|
return (
|
|
<BorderlessButton onPress={this.startRecordingAudio} style={styles.actionButton} testID='messagebox-send-audio'>
|
|
<View accessible accessibilityLabel={I18n.t('Send_audio_message')} accessibilityRole='button'>
|
|
<CustomIcon name='microphone' size={24} color={themes[theme].auxiliaryTintColor} />
|
|
</View>
|
|
</BorderlessButton>
|
|
);
|
|
}
|
|
|
|
if (!isRecording && isRecorderActive) {
|
|
return (
|
|
<View style={styles.recordingContent}>
|
|
<View style={styles.textArea}>
|
|
<BorderlessButton onPress={this.cancelRecordingAudio} style={styles.actionButton}>
|
|
<View accessible accessibilityLabel={I18n.t('Cancel_recording')} accessibilityRole='button'>
|
|
<CustomIcon size={24} color={themes[theme].dangerColor} name='delete' />
|
|
</View>
|
|
</BorderlessButton>
|
|
<Text style={[styles.recordingDurationText, { color: themes[theme].titleText }]}>{this.GetLastDuration}</Text>
|
|
</View>
|
|
<BorderlessButton onPress={this.finishRecordingAudio} style={styles.actionButton}>
|
|
<View accessible accessibilityLabel={I18n.t('Finish_recording')} accessibilityRole='button'>
|
|
<CustomIcon size={24} color={themes[theme].tintColor} name='send-filled' />
|
|
</View>
|
|
</BorderlessButton>
|
|
</View>
|
|
);
|
|
}
|
|
|
|
return (
|
|
<View style={styles.recordingContent}>
|
|
<View style={styles.textArea}>
|
|
<BorderlessButton onPress={this.cancelRecordingAudio} style={styles.actionButton}>
|
|
<View accessible accessibilityLabel={I18n.t('Cancel_recording')} accessibilityRole='button'>
|
|
<CustomIcon size={24} color={themes[theme].dangerColor} name='delete' />
|
|
</View>
|
|
</BorderlessButton>
|
|
<Text style={[styles.recordingDurationText, { color: themes[theme].titleText }]}>{this.duration}</Text>
|
|
<CustomIcon size={24} color={themes[theme].dangerColor} name='record' />
|
|
</View>
|
|
<BorderlessButton onPress={this.finishRecordingAudio} style={styles.actionButton}>
|
|
<View accessible accessibilityLabel={I18n.t('Finish_recording')} accessibilityRole='button'>
|
|
<CustomIcon size={24} color={themes[theme].tintColor} name='send-filled' />
|
|
</View>
|
|
</BorderlessButton>
|
|
</View>
|
|
);
|
|
}
|
|
}
|