matrix-react-sdk
Version:
SDK for matrix.org using React
245 lines (189 loc) • 29.3 kB
JavaScript
"use strict";
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = void 0;
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _AccessibleTooltipButton = _interopRequireDefault(require("../elements/AccessibleTooltipButton"));
var _languageHandler = require("../../../languageHandler");
var _react = _interopRequireDefault(require("react"));
var _VoiceRecording = require("../../../voice/VoiceRecording");
var _MatrixClientPeg = require("../../../MatrixClientPeg");
var _classnames = _interopRequireDefault(require("classnames"));
var _LiveRecordingWaveform = _interopRequireDefault(require("../voice_messages/LiveRecordingWaveform"));
var _replaceableComponent = require("../../../utils/replaceableComponent");
var _LiveRecordingClock = _interopRequireDefault(require("../voice_messages/LiveRecordingClock"));
var _VoiceRecordingStore = require("../../../stores/VoiceRecordingStore");
var _AsyncStore = require("../../../stores/AsyncStore");
var _RecordingPlayback = _interopRequireDefault(require("../voice_messages/RecordingPlayback"));
var _event = require("matrix-js-sdk/src/@types/event");
var _Modal = _interopRequireDefault(require("../../../Modal"));
var _ErrorDialog = _interopRequireDefault(require("../dialogs/ErrorDialog"));
var _CallMediaHandler = _interopRequireDefault(require("../../../CallMediaHandler"));
var _dec, _class, _temp;
let VoiceRecordComposerTile = (
/**
* Container tile for rendering the voice message recorder in the composer.
*/
_dec = (0, _replaceableComponent.replaceableComponent)("views.rooms.VoiceRecordComposerTile"), _dec(_class = (_temp = class VoiceRecordComposerTile extends _react.default.PureComponent
/*:: <IProps, IState>*/
{
constructor(props) {
super(props);
(0, _defineProperty2.default)(this, "onCancel", async () => {
await this.disposeRecording();
});
(0, _defineProperty2.default)(this, "onRecordStartEndClick", async () => {
if (this.state.recorder) {
await this.state.recorder.stop();
return;
} // The "microphone access error" dialogs are used a lot, so let's functionify them
const accessError = () => {
_Modal.default.createTrackedDialog('Microphone Access Error', '', _ErrorDialog.default, {
title: (0, _languageHandler._t)("Unable to access your microphone"),
description: /*#__PURE__*/_react.default.createElement(_react.default.Fragment, null, /*#__PURE__*/_react.default.createElement("p", null, (0, _languageHandler._t)("We were unable to access your microphone. Please check your browser settings and try again.")))
});
}; // Do a sanity test to ensure we're about to grab a valid microphone reference. Things might
// change between this and recording, but at least we will have tried.
try {
const devices = await _CallMediaHandler.default.getDevices();
if (!devices?.['audioinput']?.length) {
_Modal.default.createTrackedDialog('No Microphone Error', '', _ErrorDialog.default, {
title: (0, _languageHandler._t)("No microphone found"),
description: /*#__PURE__*/_react.default.createElement(_react.default.Fragment, null, /*#__PURE__*/_react.default.createElement("p", null, (0, _languageHandler._t)("We didn't find a microphone on your device. Please check your settings and try again.")))
});
return;
} // else we probably have a device that is good enough
} catch (e) {
console.error("Error getting devices: ", e);
accessError();
return;
}
try {
const recorder = _VoiceRecordingStore.VoiceRecordingStore.instance.startRecording();
await recorder.start(); // We don't need to remove the listener: the recorder will clean that up for us.
recorder.on(_AsyncStore.UPDATE_EVENT, (ev
/*: RecordingState*/
) => {
if (ev === _VoiceRecording.RecordingState.EndingSoon) return; // ignore this state: it has no UI purpose here
this.setState({
recordingPhase: ev
});
});
this.setState({
recorder,
recordingPhase: _VoiceRecording.RecordingState.Started
});
} catch (e) {
console.error("Error starting recording: ", e);
accessError(); // noinspection ES6MissingAwait - if this goes wrong we don't want it to affect the call stack
_VoiceRecordingStore.VoiceRecordingStore.instance.disposeRecording();
}
});
this.state = {
recorder: null // no recording started by default
};
}
async componentWillUnmount() {
await _VoiceRecordingStore.VoiceRecordingStore.instance.disposeRecording();
} // called by composer
async send() {
if (!this.state.recorder) {
throw new Error("No recording started - cannot send anything");
}
await this.state.recorder.stop();
const mxc = await this.state.recorder.upload();
_MatrixClientPeg.MatrixClientPeg.get().sendMessage(this.props.room.roomId, {
"body": "Voice message",
//"msgtype": "org.matrix.msc2516.voice",
"msgtype": _event.MsgType.Audio,
"url": mxc,
"info": {
duration: Math.round(this.state.recorder.durationSeconds * 1000),
mimetype: this.state.recorder.contentType,
size: this.state.recorder.contentLength
},
// MSC1767 experiment
"org.matrix.msc1767.text": "Voice message",
"org.matrix.msc1767.file": {
url: mxc,
name: "Voice message.ogg",
mimetype: this.state.recorder.contentType,
size: this.state.recorder.contentLength
},
"org.matrix.msc1767.audio": {
duration: Math.round(this.state.recorder.durationSeconds * 1000),
// Events can't have floats, so we try to maintain resolution by using 1024
// as a maximum value. The waveform contains values between zero and 1, so this
// should come out largely sane.
//
// We're expecting about one data point per second of audio.
waveform: this.state.recorder.getPlayback().waveform.map(v => Math.round(v * 1024))
},
"org.matrix.msc2516.voice": {} // No content, this is a rendering hint
});
await this.disposeRecording();
}
async disposeRecording() {
await _VoiceRecordingStore.VoiceRecordingStore.instance.disposeRecording(); // Reset back to no recording, which means no phase (ie: restart component entirely)
this.setState({
recorder: null,
recordingPhase: null
});
}
renderWaveformArea()
/*: ReactNode*/
{
if (!this.state.recorder) return null; // no recorder means we're not recording: no waveform
if (this.state.recordingPhase !== _VoiceRecording.RecordingState.Started) {
// TODO: @@ TR: Should we disable this during upload? What does a failed upload look like?
return /*#__PURE__*/_react.default.createElement(_RecordingPlayback.default, {
playback: this.state.recorder.getPlayback()
});
} // only other UI is the recording-in-progress UI
return /*#__PURE__*/_react.default.createElement("div", {
className: "mx_VoiceMessagePrimaryContainer mx_VoiceRecordComposerTile_recording"
}, /*#__PURE__*/_react.default.createElement(_LiveRecordingClock.default, {
recorder: this.state.recorder
}), /*#__PURE__*/_react.default.createElement(_LiveRecordingWaveform.default, {
recorder: this.state.recorder
}));
}
render()
/*: ReactNode*/
{
let recordingInfo;
let deleteButton;
if (!this.state.recordingPhase || this.state.recordingPhase === _VoiceRecording.RecordingState.Started) {
const classes = (0, _classnames.default)({
'mx_MessageComposer_button': !this.state.recorder,
'mx_MessageComposer_voiceMessage': !this.state.recorder,
'mx_VoiceRecordComposerTile_stop': this.state.recorder?.isRecording
});
let tooltip = (0, _languageHandler._t)("Record a voice message");
if (!!this.state.recorder) {
tooltip = (0, _languageHandler._t)("Stop the recording");
}
let stopOrRecordBtn = /*#__PURE__*/_react.default.createElement(_AccessibleTooltipButton.default, {
className: classes,
onClick: this.onRecordStartEndClick,
title: tooltip
});
if (this.state.recorder && !this.state.recorder?.isRecording) {
stopOrRecordBtn = null;
}
recordingInfo = stopOrRecordBtn;
}
if (this.state.recorder && this.state.recordingPhase !== _VoiceRecording.RecordingState.Uploading) {
deleteButton = /*#__PURE__*/_react.default.createElement(_AccessibleTooltipButton.default, {
className: "mx_VoiceRecordComposerTile_delete",
title: (0, _languageHandler._t)("Delete recording"),
onClick: this.onCancel
});
}
return /*#__PURE__*/_react.default.createElement(_react.default.Fragment, null, deleteButton, this.renderWaveformArea(), recordingInfo);
}
}, _temp)) || _class);
exports.default = VoiceRecordComposerTile;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/views/rooms/VoiceRecordComposerTile.tsx"],"names":["VoiceRecordComposerTile","React","PureComponent","constructor","props","disposeRecording","state","recorder","stop","accessError","Modal","createTrackedDialog","ErrorDialog","title","description","devices","CallMediaHandler","getDevices","length","e","console","error","VoiceRecordingStore","instance","startRecording","start","on","UPDATE_EVENT","ev","RecordingState","EndingSoon","setState","recordingPhase","Started","componentWillUnmount","send","Error","mxc","upload","MatrixClientPeg","get","sendMessage","room","roomId","MsgType","Audio","duration","Math","round","durationSeconds","mimetype","contentType","size","contentLength","url","name","waveform","getPlayback","map","v","renderWaveformArea","render","recordingInfo","deleteButton","classes","isRecording","tooltip","stopOrRecordBtn","onRecordStartEndClick","Uploading","onCancel"],"mappings":"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","sourcesContent":["/*\nCopyright 2021 The Matrix.org Foundation C.I.C.\n\nLicensed under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License.\nYou may obtain a copy of the License at\n\n    http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software\ndistributed under the License is distributed on an \"AS IS\" BASIS,\nWITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\nSee the License for the specific language governing permissions and\nlimitations under the License.\n*/\n\nimport AccessibleTooltipButton from \"../elements/AccessibleTooltipButton\";\nimport {_t} from \"../../../languageHandler\";\nimport React, {ReactNode} from \"react\";\nimport {RecordingState, VoiceRecording} from \"../../../voice/VoiceRecording\";\nimport {Room} from \"matrix-js-sdk/src/models/room\";\nimport {MatrixClientPeg} from \"../../../MatrixClientPeg\";\nimport classNames from \"classnames\";\nimport LiveRecordingWaveform from \"../voice_messages/LiveRecordingWaveform\";\nimport {replaceableComponent} from \"../../../utils/replaceableComponent\";\nimport LiveRecordingClock from \"../voice_messages/LiveRecordingClock\";\nimport {VoiceRecordingStore} from \"../../../stores/VoiceRecordingStore\";\nimport {UPDATE_EVENT} from \"../../../stores/AsyncStore\";\nimport RecordingPlayback from \"../voice_messages/RecordingPlayback\";\nimport {MsgType} from \"matrix-js-sdk/src/@types/event\";\nimport Modal from \"../../../Modal\";\nimport ErrorDialog from \"../dialogs/ErrorDialog\";\nimport CallMediaHandler from \"../../../CallMediaHandler\";\n\ninterface IProps {\n    room: Room;\n}\n\ninterface IState {\n    recorder?: VoiceRecording;\n    recordingPhase?: RecordingState;\n}\n\n/**\n * Container tile for rendering the voice message recorder in the composer.\n */\n@replaceableComponent(\"views.rooms.VoiceRecordComposerTile\")\nexport default class VoiceRecordComposerTile extends React.PureComponent<IProps, IState> {\n    public constructor(props) {\n        super(props);\n\n        this.state = {\n            recorder: null, // no recording started by default\n        };\n    }\n\n    public async componentWillUnmount() {\n        await VoiceRecordingStore.instance.disposeRecording();\n    }\n\n    // called by composer\n    public async send() {\n        if (!this.state.recorder) {\n            throw new Error(\"No recording started - cannot send anything\");\n        }\n\n        await this.state.recorder.stop();\n        const mxc = await this.state.recorder.upload();\n        MatrixClientPeg.get().sendMessage(this.props.room.roomId, {\n            \"body\": \"Voice message\",\n            //\"msgtype\": \"org.matrix.msc2516.voice\",\n            \"msgtype\": MsgType.Audio,\n            \"url\": mxc,\n            \"info\": {\n                duration: Math.round(this.state.recorder.durationSeconds * 1000),\n                mimetype: this.state.recorder.contentType,\n                size: this.state.recorder.contentLength,\n            },\n\n            // MSC1767 experiment\n            \"org.matrix.msc1767.text\": \"Voice message\",\n            \"org.matrix.msc1767.file\": {\n                url: mxc,\n                name: \"Voice message.ogg\",\n                mimetype: this.state.recorder.contentType,\n                size: this.state.recorder.contentLength,\n            },\n            \"org.matrix.msc1767.audio\": {\n                duration: Math.round(this.state.recorder.durationSeconds * 1000),\n\n                // Events can't have floats, so we try to maintain resolution by using 1024\n                // as a maximum value. The waveform contains values between zero and 1, so this\n                // should come out largely sane.\n                //\n                // We're expecting about one data point per second of audio.\n                waveform: this.state.recorder.getPlayback().waveform.map(v => Math.round(v * 1024)),\n            },\n            \"org.matrix.msc2516.voice\": {}, // No content, this is a rendering hint\n        });\n        await this.disposeRecording();\n    }\n\n    private async disposeRecording() {\n        await VoiceRecordingStore.instance.disposeRecording();\n\n        // Reset back to no recording, which means no phase (ie: restart component entirely)\n        this.setState({recorder: null, recordingPhase: null});\n    }\n\n    private onCancel = async () => {\n        await this.disposeRecording();\n    };\n\n    private onRecordStartEndClick = async () => {\n        if (this.state.recorder) {\n            await this.state.recorder.stop();\n            return;\n        }\n\n        // The \"microphone access error\" dialogs are used a lot, so let's functionify them\n        const accessError = () => {\n            Modal.createTrackedDialog('Microphone Access Error', '', ErrorDialog, {\n                title: _t(\"Unable to access your microphone\"),\n                description: <>\n                    <p>{_t(\n                        \"We were unable to access your microphone. Please check your browser settings and try again.\",\n                    )}</p>\n                </>,\n            });\n        };\n\n        // Do a sanity test to ensure we're about to grab a valid microphone reference. Things might\n        // change between this and recording, but at least we will have tried.\n        try {\n            const devices = await CallMediaHandler.getDevices();\n            if (!devices?.['audioinput']?.length) {\n                Modal.createTrackedDialog('No Microphone Error', '', ErrorDialog, {\n                    title: _t(\"No microphone found\"),\n                    description: <>\n                        <p>{_t(\n                            \"We didn't find a microphone on your device. Please check your settings and try again.\",\n                        )}</p>\n                    </>,\n                });\n                return;\n            }\n            // else we probably have a device that is good enough\n        } catch (e) {\n            console.error(\"Error getting devices: \", e);\n            accessError();\n            return;\n        }\n\n        try {\n            const recorder = VoiceRecordingStore.instance.startRecording();\n            await recorder.start();\n\n            // We don't need to remove the listener: the recorder will clean that up for us.\n            recorder.on(UPDATE_EVENT, (ev: RecordingState) => {\n                if (ev === RecordingState.EndingSoon) return; // ignore this state: it has no UI purpose here\n                this.setState({recordingPhase: ev});\n            });\n\n            this.setState({recorder, recordingPhase: RecordingState.Started});\n        } catch (e) {\n            console.error(\"Error starting recording: \", e);\n            accessError();\n\n            // noinspection ES6MissingAwait - if this goes wrong we don't want it to affect the call stack\n            VoiceRecordingStore.instance.disposeRecording();\n        }\n    };\n\n    private renderWaveformArea(): ReactNode {\n        if (!this.state.recorder) return null; // no recorder means we're not recording: no waveform\n\n        if (this.state.recordingPhase !== RecordingState.Started) {\n            // TODO: @@ TR: Should we disable this during upload? What does a failed upload look like?\n            return <RecordingPlayback playback={this.state.recorder.getPlayback()} />;\n        }\n\n        // only other UI is the recording-in-progress UI\n        return <div className=\"mx_VoiceMessagePrimaryContainer mx_VoiceRecordComposerTile_recording\">\n            <LiveRecordingClock recorder={this.state.recorder} />\n            <LiveRecordingWaveform recorder={this.state.recorder} />\n        </div>;\n    }\n\n    public render(): ReactNode {\n        let recordingInfo;\n        let deleteButton;\n        if (!this.state.recordingPhase || this.state.recordingPhase === RecordingState.Started) {\n            const classes = classNames({\n                'mx_MessageComposer_button': !this.state.recorder,\n                'mx_MessageComposer_voiceMessage': !this.state.recorder,\n                'mx_VoiceRecordComposerTile_stop': this.state.recorder?.isRecording,\n            });\n\n            let tooltip = _t(\"Record a voice message\");\n            if (!!this.state.recorder) {\n                tooltip = _t(\"Stop the recording\");\n            }\n\n            let stopOrRecordBtn = <AccessibleTooltipButton\n                className={classes}\n                onClick={this.onRecordStartEndClick}\n                title={tooltip}\n            />;\n            if (this.state.recorder && !this.state.recorder?.isRecording) {\n                stopOrRecordBtn = null;\n            }\n\n            recordingInfo = stopOrRecordBtn;\n        }\n\n        if (this.state.recorder && this.state.recordingPhase !== RecordingState.Uploading) {\n            deleteButton = <AccessibleTooltipButton\n                className='mx_VoiceRecordComposerTile_delete'\n                title={_t(\"Delete recording\")}\n                onClick={this.onCancel}\n            />;\n        }\n\n        return (<>\n            {deleteButton}\n            {this.renderWaveformArea()}\n            {recordingInfo}\n        </>);\n    }\n}\n"]}