// 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 () { window.Whisper = window.Whisper || {}; Whisper.RecorderView = Whisper.View.extend({ className: 'recorder clearfix', template: () => $('#recorder').html(), initialize() { this.startTime = Date.now(); this.interval = setInterval(this.updateTime.bind(this), 1000); this.onSwitchAwayBound = this.onSwitchAway.bind(this); $(window).on('blur', this.onSwitchAwayBound); this.handleKeyDownBound = this.handleKeyDown.bind(this); this.$el.on('keydown', this.handleKeyDownBound); this.start(); }, events: { 'click .close': 'remove', 'click .finish': 'finish', close: 'remove', }, onSwitchAway() { this.lostFocus = true; this.recorder.finishRecording(); }, handleKeyDown(event) { if (event.key === 'Escape') { this.remove(); event.preventDefault(); event.stopPropagation(); } }, updateTime() { const duration = moment.duration(Date.now() - this.startTime, 'ms'); const minutes = `${Math.trunc(duration.asMinutes())}`; let seconds = `${duration.seconds()}`; if (seconds.length < 2) { seconds = `0${seconds}`; } this.$('.time').text(`${minutes}:${seconds}`); }, remove() { // Note: the 'close' event can be triggered by InboxView, when the user clicks // anywhere outside the recording pane. 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; if (this.interval) { clearInterval(this.interval); } this.interval = null; if (this.source) { this.source.disconnect(); } this.source = null; if (this.context) { this.context.close().then(() => { window.log.info('audio context closed'); }); } this.context = null; Whisper.View.prototype.remove.call(this); this.trigger('closed'); $(window).off('blur', this.onSwitchAwayBound); this.$el.off('keydown', this.handleKeyDownBound); }, finish() { this.clickedFinish = true; this.recorder.finishRecording(); }, handleBlob(recorder, blob) { if (blob && this.clickedFinish) { this.trigger('send', blob); } else if (blob) { this.trigger('confirm', blob, this.lostFocus); } this.remove(); }, start() { this.lostFocus = false; this.clickedFinish = false; 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); this.recorder.onTimeout = this.onTimeout.bind(this); navigator.webkitGetUserMedia( { audio: true }, stream => { this.source = this.context.createMediaStreamSource(stream); this.source.connect(this.input); }, this.onError.bind(this) ); this.recorder.startRecording(); }, onTimeout() { this.recorder.finishRecording(); }, 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; } this.remove(); 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 ); } }, }); })();