|
|
|
@ -7,15 +7,15 @@ import { getTimestamp } from './SessionConversationManager';
|
|
|
|
|
import { SessionIconButton, SessionIconSize, SessionIconType } from '../icon';
|
|
|
|
|
import {
|
|
|
|
|
SessionButton,
|
|
|
|
|
SessionButtonType,
|
|
|
|
|
SessionButtonColor,
|
|
|
|
|
SessionButtonType,
|
|
|
|
|
} from '../SessionButton';
|
|
|
|
|
import { Constants } from '../../../session';
|
|
|
|
|
|
|
|
|
|
interface Props {
|
|
|
|
|
sendVoiceMessage: any;
|
|
|
|
|
onLoadVoiceNoteView: any;
|
|
|
|
|
onExitVoiceNoteView: any;
|
|
|
|
|
onLoadVoiceNoteView: any;
|
|
|
|
|
sendVoiceMessage: any;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
interface State {
|
|
|
|
@ -56,9 +56,9 @@ interface State {
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export class SessionRecording extends React.Component<Props, State> {
|
|
|
|
|
private visualisationRef: React.RefObject<HTMLDivElement>;
|
|
|
|
|
private visualisationCanvas: React.RefObject<HTMLCanvasElement>;
|
|
|
|
|
private playbackCanvas: React.RefObject<HTMLCanvasElement>;
|
|
|
|
|
private readonly visualisationRef: React.RefObject<HTMLDivElement>;
|
|
|
|
|
private readonly visualisationCanvas: React.RefObject<HTMLCanvasElement>;
|
|
|
|
|
private readonly playbackCanvas: React.RefObject<HTMLCanvasElement>;
|
|
|
|
|
|
|
|
|
|
constructor(props: any) {
|
|
|
|
|
super(props);
|
|
|
|
@ -125,12 +125,17 @@ export class SessionRecording extends React.Component<Props, State> {
|
|
|
|
|
|
|
|
|
|
public async componentWillMount() {
|
|
|
|
|
// This turns on the microphone on the system. Later we need to turn it off.
|
|
|
|
|
this.initiateRecordingStream();
|
|
|
|
|
await this.initiateRecordingStream();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public componentDidMount() {
|
|
|
|
|
window.addEventListener('resize', this.updateCanvasDimensions);
|
|
|
|
|
this.updateCanvasDimensions();
|
|
|
|
|
|
|
|
|
|
// Callback to parent on load complete
|
|
|
|
|
if (this.props.onLoadVoiceNoteView) {
|
|
|
|
|
this.props.onLoadVoiceNoteView();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public componentWillUnmount() {
|
|
|
|
@ -138,12 +143,12 @@ export class SessionRecording extends React.Component<Props, State> {
|
|
|
|
|
window.removeEventListener('resize', this.updateCanvasDimensions);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public componentDidUpdate() {
|
|
|
|
|
public async componentDidUpdate() {
|
|
|
|
|
const { audioElement, isPlaying } = this.state;
|
|
|
|
|
|
|
|
|
|
if (audioElement) {
|
|
|
|
|
if (isPlaying) {
|
|
|
|
|
audioElement.play();
|
|
|
|
|
await audioElement.play();
|
|
|
|
|
} else {
|
|
|
|
|
audioElement.pause();
|
|
|
|
|
}
|
|
|
|
@ -226,8 +231,8 @@ export class SessionRecording extends React.Component<Props, State> {
|
|
|
|
|
className="session-recording--visualisation"
|
|
|
|
|
ref={this.visualisationRef}
|
|
|
|
|
>
|
|
|
|
|
{!isRecording && <canvas ref={this.playbackCanvas}></canvas>}
|
|
|
|
|
{isRecording && <canvas ref={this.visualisationCanvas}></canvas>}
|
|
|
|
|
{!isRecording && <canvas ref={this.playbackCanvas} />}
|
|
|
|
|
{isRecording && <canvas ref={this.visualisationCanvas} />}
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div
|
|
|
|
@ -237,9 +242,7 @@ export class SessionRecording extends React.Component<Props, State> {
|
|
|
|
|
)}
|
|
|
|
|
>
|
|
|
|
|
{displayTimeString}
|
|
|
|
|
{isRecording && (
|
|
|
|
|
<div className="session-recording--timer-light"></div>
|
|
|
|
|
)}
|
|
|
|
|
{isRecording && <div className="session-recording--timer-light" />}
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
{!isRecording && (
|
|
|
|
@ -282,13 +285,13 @@ export class SessionRecording extends React.Component<Props, State> {
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private timerUpdate() {
|
|
|
|
|
private async timerUpdate() {
|
|
|
|
|
const { nowTimestamp, startTimestamp } = this.state;
|
|
|
|
|
const elapsedTime = nowTimestamp - startTimestamp;
|
|
|
|
|
|
|
|
|
|
// Prevent voice messages exceeding max length.
|
|
|
|
|
if (elapsedTime >= Constants.CONVERSATION.MAX_VOICE_MESSAGE_DURATION) {
|
|
|
|
|
this.stopRecordingStream();
|
|
|
|
|
await this.stopRecordingStream();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
this.setState({
|
|
|
|
@ -321,19 +324,19 @@ export class SessionRecording extends React.Component<Props, State> {
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const audioURL = window.URL.createObjectURL(mediaBlob.data);
|
|
|
|
|
const audioElement = new Audio(audioURL);
|
|
|
|
|
const audioElementN = new Audio(audioURL);
|
|
|
|
|
|
|
|
|
|
audioElement.loop = false;
|
|
|
|
|
audioElementN.loop = false;
|
|
|
|
|
|
|
|
|
|
audioElement.oncanplaythrough = async () => {
|
|
|
|
|
audioElementN.oncanplaythrough = async () => {
|
|
|
|
|
const duration = recordDuration;
|
|
|
|
|
|
|
|
|
|
if (duration && audioElement.currentTime < duration) {
|
|
|
|
|
await audioElement.play();
|
|
|
|
|
if (duration && audioElementN.currentTime < duration) {
|
|
|
|
|
await audioElementN.play();
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
return audioElement;
|
|
|
|
|
return audioElementN;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const audioElement = this.state.audioElement || generateAudioElement();
|
|
|
|
@ -422,9 +425,7 @@ export class SessionRecording extends React.Component<Props, State> {
|
|
|
|
|
|
|
|
|
|
// Is the audio file > attachment filesize limit
|
|
|
|
|
if (audioBlob.size > Constants.CONVERSATION.MAX_ATTACHMENT_FILESIZE) {
|
|
|
|
|
console.log(
|
|
|
|
|
`[send] Voice message too large: ${audioBlob.size / 1000000} MB`
|
|
|
|
|
);
|
|
|
|
|
// TODO VINCE: warn the user that it's too big
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|