signal-desktop/js/views/recorder_view.js

154 lines
4.2 KiB
JavaScript
Raw Normal View History

2020-10-30 20:34:04 +00:00
// Copyright 2016-2020 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
/* global $, Whisper, moment, WebAudioRecorder */
/* eslint-disable more/no-then */
// eslint-disable-next-line func-names
(function () {
2018-04-27 21:25:04 +00:00
window.Whisper = window.Whisper || {};
2016-08-15 22:36:29 +00:00
2018-04-27 21:25:04 +00:00
Whisper.RecorderView = Whisper.View.extend({
className: 'recorder clearfix',
template: () => $('#recorder').html(),
initialize() {
2018-04-27 21:25:04 +00:00
this.startTime = Date.now();
this.interval = setInterval(this.updateTime.bind(this), 1000);
this.onSwitchAwayBound = this.onSwitchAway.bind(this);
$(window).on('blur', this.onSwitchAwayBound);
2019-11-07 21:36:16 +00:00
this.handleKeyDownBound = this.handleKeyDown.bind(this);
this.$el.on('keydown', this.handleKeyDownBound);
2018-04-27 21:25:04 +00:00
this.start();
},
events: {
'click .close': 'close',
'click .finish': 'finish',
close: 'close',
},
onSwitchAway() {
2020-02-20 22:51:25 +00:00
this.lostFocus = true;
this.recorder.finishRecording();
this.close();
},
2019-11-07 21:36:16 +00:00
handleKeyDown(event) {
if (event.key === 'Escape') {
this.close();
event.preventDefault();
event.stopPropagation();
}
},
updateTime() {
const duration = moment.duration(Date.now() - this.startTime, 'ms');
const minutes = `${Math.trunc(duration.asMinutes())}`;
let seconds = `${duration.seconds()}`;
2018-04-27 21:25:04 +00:00
if (seconds.length < 2) {
seconds = `0${seconds}`;
2018-04-27 21:25:04 +00:00
}
this.$('.time').text(`${minutes}:${seconds}`);
2018-04-27 21:25:04 +00:00
},
close() {
2018-04-27 21:25:04 +00:00
// Note: the 'close' event can be triggered by InboxView, when the user clicks
// anywhere outside the recording pane.
2018-04-27 21:25:04 +00:00
if (this.recorder.isRecording()) {
this.recorder.cancelRecording();
}
// Reach in and terminate the web worker used by WebAudioRecorder, otherwise
// it gets leaked due to a reference cycle with its onmessage listener
this.recorder.worker.terminate();
this.recorder = null;
2018-04-27 21:25:04 +00:00
if (this.interval) {
clearInterval(this.interval);
}
this.interval = null;
2018-04-27 21:25:04 +00:00
if (this.source) {
this.source.disconnect();
}
this.source = null;
2018-04-27 21:25:04 +00:00
if (this.context) {
this.context.close().then(() => {
window.log.info('audio context closed');
2018-04-27 21:25:04 +00:00
});
}
this.context = null;
2018-04-27 21:25:04 +00:00
this.remove();
this.trigger('closed');
$(window).off('blur', this.onSwitchAwayBound);
2019-11-07 21:36:16 +00:00
this.$el.off('keydown', this.handleKeyDownBound);
2018-04-27 21:25:04 +00:00
},
finish() {
this.clickedFinish = true;
2018-04-27 21:25:04 +00:00
this.recorder.finishRecording();
this.close();
},
handleBlob(recorder, blob) {
if (blob && this.clickedFinish) {
2018-04-27 21:25:04 +00:00
this.trigger('send', blob);
2020-02-20 22:51:25 +00:00
} else if (blob) {
this.trigger('confirm', blob, this.lostFocus);
} else {
this.close();
2018-04-27 21:25:04 +00:00
}
},
start() {
2020-02-20 22:51:25 +00:00
this.lostFocus = false;
this.clickedFinish = false;
2018-04-27 21:25:04 +00:00
this.context = new AudioContext();
this.input = this.context.createGain();
this.recorder = new WebAudioRecorder(this.input, {
encoding: 'mp3',
workerDir: 'js/', // must end with slash
});
this.recorder.onComplete = this.handleBlob.bind(this);
this.recorder.onError = this.onError.bind(this);
2020-02-20 22:51:25 +00:00
this.recorder.onTimeout = this.onTimeout.bind(this);
2018-04-27 21:25:04 +00:00
navigator.webkitGetUserMedia(
{ audio: true },
stream => {
2018-04-27 21:25:04 +00:00
this.source = this.context.createMediaStreamSource(stream);
this.source.connect(this.input);
},
2018-04-27 21:25:04 +00:00
this.onError.bind(this)
);
this.recorder.startRecording();
},
2020-02-20 22:51:25 +00:00
onTimeout() {
this.recorder.finishRecording();
this.close();
},
onError(error) {
// Protect against out-of-band errors, which can happen if the user revokes media
// permissions after successfully accessing the microphone.
if (!this.recorder) {
return;
}
2018-04-27 21:25:04 +00:00
this.close();
if (error && error.name === 'NotAllowedError') {
window.log.warn(
'RecorderView.onError: Microphone access is not allowed!'
);
window.showPermissionsPopup();
} else {
window.log.error(
'RecorderView.onError:',
error && error.stack ? error.stack : error
);
}
2018-04-27 21:25:04 +00:00
},
});
2016-08-15 22:36:29 +00:00
})();