matrix-react-sdk
Version:
SDK for matrix.org using React
115 lines (92 loc) • 12.3 kB
JavaScript
;
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = void 0;
var _extends2 = _interopRequireDefault(require("@babel/runtime/helpers/extends"));
var _react = _interopRequireDefault(require("react"));
var _replaceableComponent = require("../../../utils/replaceableComponent");
var _Playback = require("../../../voice/Playback");
var _MFileBody = _interopRequireDefault(require("./MFileBody"));
var _InlineSpinner = _interopRequireDefault(require("../elements/InlineSpinner"));
var _languageHandler = require("../../../languageHandler");
var _Media = require("../../../customisations/Media");
var _DecryptFile = require("../../../utils/DecryptFile");
var _RecordingPlayback = _interopRequireDefault(require("../voice_messages/RecordingPlayback"));
var _dec, _class;
let MVoiceMessageBody = (_dec = (0, _replaceableComponent.replaceableComponent)("views.messages.MVoiceMessageBody"), _dec(_class = class MVoiceMessageBody extends _react.default.PureComponent
/*:: <IProps, IState>*/
{
constructor(props
/*: IProps*/
) {
super(props);
this.state = {};
}
async componentDidMount() {
let buffer
/*: ArrayBuffer*/
;
const content = this.props.mxEvent.getContent();
const media = (0, _Media.mediaFromContent)(content);
if (media.isEncrypted) {
try {
const blob = await (0, _DecryptFile.decryptFile)(content.file);
buffer = await blob.arrayBuffer();
this.setState({
decryptedBlob: blob
});
} catch (e) {
this.setState({
error: e
});
console.warn("Unable to decrypt voice message", e);
return; // stop processing the audio file
}
} else {
try {
buffer = await media.downloadSource().then(r => r.blob()).then(r => r.arrayBuffer());
} catch (e) {
this.setState({
error: e
});
console.warn("Unable to download voice message", e);
return; // stop processing the audio file
}
}
const waveform = content?.["org.matrix.msc1767.audio"]?.waveform?.map(p => p / 1024); // We should have a buffer to work with now: let's set it up
const playback = new _Playback.Playback(buffer, waveform);
this.setState({
playback
}); // Note: the RecordingPlayback component will handle preparing the Playback class for us.
}
render() {
if (this.state.error) {
// TODO: @@TR: Verify error state
return /*#__PURE__*/_react.default.createElement("span", {
className: "mx_MVoiceMessageBody"
}, /*#__PURE__*/_react.default.createElement("img", {
src: require("../../../../res/img/warning.svg"),
width: "16",
height: "16"
}), (0, _languageHandler._t)("Error processing voice message"));
}
if (!this.state.playback) {
// TODO: @@TR: Verify loading/decrypting state
return /*#__PURE__*/_react.default.createElement("span", {
className: "mx_MVoiceMessageBody"
}, /*#__PURE__*/_react.default.createElement(_InlineSpinner.default, null));
} // At this point we should have a playable state
return /*#__PURE__*/_react.default.createElement("span", {
className: "mx_MVoiceMessageBody"
}, /*#__PURE__*/_react.default.createElement(_RecordingPlayback.default, {
playback: this.state.playback
}), /*#__PURE__*/_react.default.createElement(_MFileBody.default, (0, _extends2.default)({}, this.props, {
decryptedBlob: this.state.decryptedBlob,
showGenericPlaceholder: false
})));
}
}) || _class);
exports.default = MVoiceMessageBody;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/views/messages/MVoiceMessageBody.tsx"],"names":["MVoiceMessageBody","React","PureComponent","constructor","props","state","componentDidMount","buffer","content","mxEvent","getContent","media","isEncrypted","blob","file","arrayBuffer","setState","decryptedBlob","e","error","console","warn","downloadSource","then","r","waveform","map","p","playback","Playback","render","require"],"mappings":";;;;;;;;;;;AAgBA;;AAEA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;;;IAaqBA,iB,WADpB,gDAAqB,kCAArB,C,gBAAD,MACqBA,iBADrB,SAC+CC,eAAMC;AADrD;AACmF;AAC/EC,EAAAA,WAAW,CAACC;AAAD;AAAA,IAAgB;AACvB,UAAMA,KAAN;AAEA,SAAKC,KAAL,GAAa,EAAb;AACH;;AAED,QAAaC,iBAAb,GAAiC;AAC7B,QAAIC;AAAmB;AAAvB;AACA,UAAMC,OAAO,GAAG,KAAKJ,KAAL,CAAWK,OAAX,CAAmBC,UAAnB,EAAhB;AACA,UAAMC,KAAK,GAAG,6BAAiBH,OAAjB,CAAd;;AACA,QAAIG,KAAK,CAACC,WAAV,EAAuB;AACnB,UAAI;AACA,cAAMC,IAAI,GAAG,MAAM,8BAAYL,OAAO,CAACM,IAApB,CAAnB;AACAP,QAAAA,MAAM,GAAG,MAAMM,IAAI,CAACE,WAAL,EAAf;AACA,aAAKC,QAAL,CAAc;AAACC,UAAAA,aAAa,EAAEJ;AAAhB,SAAd;AACH,OAJD,CAIE,OAAOK,CAAP,EAAU;AACR,aAAKF,QAAL,CAAc;AAACG,UAAAA,KAAK,EAAED;AAAR,SAAd;AACAE,QAAAA,OAAO,CAACC,IAAR,CAAa,iCAAb,EAAgDH,CAAhD;AACA,eAHQ,CAGA;AACX;AACJ,KAVD,MAUO;AACH,UAAI;AACAX,QAAAA,MAAM,GAAG,MAAMI,KAAK,CAACW,cAAN,GAAuBC,IAAvB,CAA4BC,CAAC,IAAIA,CAAC,CAACX,IAAF,EAAjC,EAA2CU,IAA3C,CAAgDC,CAAC,IAAIA,CAAC,CAACT,WAAF,EAArD,CAAf;AACH,OAFD,CAEE,OAAOG,CAAP,EAAU;AACR,aAAKF,QAAL,CAAc;AAACG,UAAAA,KAAK,EAAED;AAAR,SAAd;AACAE,QAAAA,OAAO,CAACC,IAAR,CAAa,kCAAb,EAAiDH,CAAjD;AACA,eAHQ,CAGA;AACX;AACJ;;AAED,UAAMO,QAAQ,GAAGjB,OAAO,GAAG,0BAAH,CAAP,EAAuCiB,QAAvC,EAAiDC,GAAjD,CAAqDC,CAAC,IAAIA,CAAC,GAAG,IAA9D,CAAjB,CAxB6B,CA0B7B;;AACA,UAAMC,QAAQ,GAAG,IAAIC,kBAAJ,CAAatB,MAAb,EAAqBkB,QAArB,CAAjB;AACA,SAAKT,QAAL,CAAc;AAACY,MAAAA;AAAD,KAAd,EA5B6B,CA6B7B;AACH;;AAEME,EAAAA,MAAP,GAAgB;AACZ,QAAI,KAAKzB,KAAL,CAAWc,KAAf,EAAsB;AAClB;AACA,0BACI;AAAM,QAAA,SAAS,EAAC;AAAhB,sBACI;AAAK,QAAA,GAAG,EAAEY,OAAO,CAAC,iCAAD,CAAjB;AAAsD,QAAA,KAAK,EAAC,IAA5D;AAAiE,QAAA,MAAM,EAAC;AAAxE,QADJ,EAEM,yBAAG,gCAAH,CAFN,CADJ;AAMH;;AAED,QAAI,CAAC,KAAK1B,KAAL,CAAWuB,QAAhB,EAA0B;AACtB;AACA,0BACI;AAAM,QAAA,SAAS,EAAC;AAAhB,sBACI,6BAAC,sBAAD,OADJ,CADJ;AAKH,KAlBW,CAoBZ;;;AACA,wBACI;AAAM,MAAA,SAAS,EAAC;AAAhB,oBACI,6BAAC,0BAAD;AAAmB,MAAA,QAAQ,EAAE,KAAKvB,KAAL,CAAWuB;AAAxC,MADJ,eAEI,6BAAC,kBAAD,6BAAe,KAAKxB,KAApB;AAA2B,MAAA,aAAa,EAAE,KAAKC,KAAL,CAAWY,aAArD;AAAoE,MAAA,sBAAsB,EAAE;AAA5F,OAFJ,CADJ;AAMH;;AAlE8E,C","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 React from \"react\";\nimport {MatrixEvent} from \"matrix-js-sdk/src/models/event\";\nimport {replaceableComponent} from \"../../../utils/replaceableComponent\";\nimport {Playback} from \"../../../voice/Playback\";\nimport MFileBody from \"./MFileBody\";\nimport InlineSpinner from '../elements/InlineSpinner';\nimport {_t} from \"../../../languageHandler\";\nimport {mediaFromContent} from \"../../../customisations/Media\";\nimport {decryptFile} from \"../../../utils/DecryptFile\";\nimport RecordingPlayback from \"../voice_messages/RecordingPlayback\";\n\ninterface IProps {\n    mxEvent: MatrixEvent;\n}\n\ninterface IState {\n    error?: Error;\n    playback?: Playback;\n    decryptedBlob?: Blob;\n}\n\n@replaceableComponent(\"views.messages.MVoiceMessageBody\")\nexport default class MVoiceMessageBody extends React.PureComponent<IProps, IState> {\n    constructor(props: IProps) {\n        super(props);\n\n        this.state = {};\n    }\n\n    public async componentDidMount() {\n        let buffer: ArrayBuffer;\n        const content = this.props.mxEvent.getContent();\n        const media = mediaFromContent(content);\n        if (media.isEncrypted) {\n            try {\n                const blob = await decryptFile(content.file);\n                buffer = await blob.arrayBuffer();\n                this.setState({decryptedBlob: blob});\n            } catch (e) {\n                this.setState({error: e});\n                console.warn(\"Unable to decrypt voice message\", e);\n                return; // stop processing the audio file\n            }\n        } else {\n            try {\n                buffer = await media.downloadSource().then(r => r.blob()).then(r => r.arrayBuffer());\n            } catch (e) {\n                this.setState({error: e});\n                console.warn(\"Unable to download voice message\", e);\n                return; // stop processing the audio file\n            }\n        }\n\n        const waveform = content?.[\"org.matrix.msc1767.audio\"]?.waveform?.map(p => p / 1024);\n\n        // We should have a buffer to work with now: let's set it up\n        const playback = new Playback(buffer, waveform);\n        this.setState({playback});\n        // Note: the RecordingPlayback component will handle preparing the Playback class for us.\n    }\n\n    public render() {\n        if (this.state.error) {\n            // TODO: @@TR: Verify error state\n            return (\n                <span className=\"mx_MVoiceMessageBody\">\n                    <img src={require(\"../../../../res/img/warning.svg\")} width=\"16\" height=\"16\" />\n                    { _t(\"Error processing voice message\") }\n                </span>\n            );\n        }\n\n        if (!this.state.playback) {\n            // TODO: @@TR: Verify loading/decrypting state\n            return (\n                <span className=\"mx_MVoiceMessageBody\">\n                    <InlineSpinner />\n                </span>\n            );\n        }\n\n        // At this point we should have a playable state\n        return (\n            <span className=\"mx_MVoiceMessageBody\">\n                <RecordingPlayback playback={this.state.playback} />\n                <MFileBody {...this.props} decryptedBlob={this.state.decryptedBlob} showGenericPlaceholder={false} />\n            </span>\n        )\n    }\n}\n"]}