matrix-react-sdk
Version:
SDK for matrix.org using React
188 lines (150 loc) • 26.3 kB
JavaScript
"use strict";
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
var _interopRequireWildcard = require("@babel/runtime/helpers/interopRequireWildcard");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = void 0;
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _react = _interopRequireDefault(require("react"));
var _propTypes = _interopRequireDefault(require("prop-types"));
var _SyntaxHighlight = _interopRequireDefault(require("../views/elements/SyntaxHighlight"));
var _languageHandler = require("../../languageHandler");
var sdk = _interopRequireWildcard(require("../../index"));
var _MatrixClientContext = _interopRequireDefault(require("../../contexts/MatrixClientContext"));
var _DevtoolsDialog = require("../views/dialogs/DevtoolsDialog");
var _EventUtils = require("../../utils/EventUtils");
var _MatrixClientPeg = require("../../MatrixClientPeg");
var _replaceableComponent = require("../../utils/replaceableComponent");
var _dec, _class, _class2, _temp;
let ViewSource = (_dec = (0, _replaceableComponent.replaceableComponent)("structures.ViewSource"), _dec(_class = (_temp = _class2 = class ViewSource extends _react.default.Component {
constructor(props) {
super(props);
this.state = {
isEditing: false
};
}
onBack() {
// TODO: refresh the "Event ID:" modal header
this.setState({
isEditing: false
});
}
onEdit() {
this.setState({
isEditing: true
});
} // returns the dialog body for viewing the event source
viewSourceContent() {
const mxEvent = this.props.mxEvent.replacingEvent() || this.props.mxEvent; // show the replacing event, not the original, if it is an edit
const isEncrypted = mxEvent.isEncrypted();
const decryptedEventSource = mxEvent._clearEvent; // FIXME: _clearEvent is private
const originalEventSource = mxEvent.event;
if (isEncrypted) {
return /*#__PURE__*/_react.default.createElement(_react.default.Fragment, null, /*#__PURE__*/_react.default.createElement("details", {
open: true,
className: "mx_ViewSource_details"
}, /*#__PURE__*/_react.default.createElement("summary", null, /*#__PURE__*/_react.default.createElement("span", {
className: "mx_ViewSource_heading"
}, (0, _languageHandler._t)("Decrypted event source"))), /*#__PURE__*/_react.default.createElement(_SyntaxHighlight.default, {
className: "json"
}, JSON.stringify(decryptedEventSource, null, 2))), /*#__PURE__*/_react.default.createElement("details", {
className: "mx_ViewSource_details"
}, /*#__PURE__*/_react.default.createElement("summary", null, /*#__PURE__*/_react.default.createElement("span", {
className: "mx_ViewSource_heading"
}, (0, _languageHandler._t)("Original event source"))), /*#__PURE__*/_react.default.createElement(_SyntaxHighlight.default, {
className: "json"
}, JSON.stringify(originalEventSource, null, 2))));
} else {
return /*#__PURE__*/_react.default.createElement(_react.default.Fragment, null, /*#__PURE__*/_react.default.createElement("div", {
className: "mx_ViewSource_heading"
}, (0, _languageHandler._t)("Original event source")), /*#__PURE__*/_react.default.createElement(_SyntaxHighlight.default, {
className: "json"
}, JSON.stringify(originalEventSource, null, 2)));
}
} // returns the id of the initial message, not the id of the previous edit
getBaseEventId() {
const mxEvent = this.props.mxEvent.replacingEvent() || this.props.mxEvent; // show the replacing event, not the original, if it is an edit
const isEncrypted = mxEvent.isEncrypted();
const baseMxEvent = this.props.mxEvent;
if (isEncrypted) {
// `relates_to` field is inside the encrypted event
return mxEvent.event.content["m.relates_to"]?.event_id ?? baseMxEvent.getId();
} else {
return mxEvent.getContent()["m.relates_to"]?.event_id ?? baseMxEvent.getId();
}
} // returns the SendCustomEvent component prefilled with the correct details
editSourceContent() {
const mxEvent = this.props.mxEvent.replacingEvent() || this.props.mxEvent; // show the replacing event, not the original, if it is an edit
const isStateEvent = mxEvent.isState();
const roomId = mxEvent.getRoomId();
const originalContent = mxEvent.getContent();
if (isStateEvent) {
return /*#__PURE__*/_react.default.createElement(_MatrixClientContext.default.Consumer, null, cli => /*#__PURE__*/_react.default.createElement(_DevtoolsDialog.SendCustomEvent, {
room: cli.getRoom(roomId),
forceStateEvent: true,
onBack: () => this.onBack(),
inputs: {
eventType: mxEvent.getType(),
evContent: JSON.stringify(originalContent, null, "\t"),
stateKey: mxEvent.getStateKey()
}
}));
} else {
// prefill an edit-message event
// keep only the `body` and `msgtype` fields of originalContent
const bodyToStartFrom = originalContent["m.new_content"]?.body ?? originalContent.body; // prefill the last edit body, to start editing from there
const newContent = {
"body": ` * ${bodyToStartFrom}`,
"msgtype": originalContent.msgtype,
"m.new_content": {
body: bodyToStartFrom,
msgtype: originalContent.msgtype
},
"m.relates_to": {
rel_type: "m.replace",
event_id: this.getBaseEventId()
}
};
return /*#__PURE__*/_react.default.createElement(_MatrixClientContext.default.Consumer, null, cli => /*#__PURE__*/_react.default.createElement(_DevtoolsDialog.SendCustomEvent, {
room: cli.getRoom(roomId),
forceStateEvent: false,
forceGeneralEvent: true,
onBack: () => this.onBack(),
inputs: {
eventType: mxEvent.getType(),
evContent: JSON.stringify(newContent, null, "\t")
}
}));
}
}
canSendStateEvent(mxEvent) {
const cli = _MatrixClientPeg.MatrixClientPeg.get();
const room = cli.getRoom(mxEvent.getRoomId());
return room.currentState.mayClientSendStateEvent(mxEvent.getType(), cli);
}
render() {
const BaseDialog = sdk.getComponent("views.dialogs.BaseDialog");
const mxEvent = this.props.mxEvent.replacingEvent() || this.props.mxEvent; // show the replacing event, not the original, if it is an edit
const isEditing = this.state.isEditing;
const roomId = mxEvent.getRoomId();
const eventId = mxEvent.getId();
const canEdit = mxEvent.isState() ? this.canSendStateEvent(mxEvent) : (0, _EventUtils.canEditContent)(this.props.mxEvent);
return /*#__PURE__*/_react.default.createElement(BaseDialog, {
className: "mx_ViewSource",
onFinished: this.props.onFinished,
title: (0, _languageHandler._t)("View Source")
}, /*#__PURE__*/_react.default.createElement("div", null, /*#__PURE__*/_react.default.createElement("div", null, "Room ID: ", roomId), /*#__PURE__*/_react.default.createElement("div", null, "Event ID: ", eventId), /*#__PURE__*/_react.default.createElement("div", {
className: "mx_ViewSource_separator"
}), isEditing ? this.editSourceContent() : this.viewSourceContent()), !isEditing && canEdit && /*#__PURE__*/_react.default.createElement("div", {
className: "mx_Dialog_buttons"
}, /*#__PURE__*/_react.default.createElement("button", {
onClick: () => this.onEdit()
}, (0, _languageHandler._t)("Edit"))));
}
}, (0, _defineProperty2.default)(_class2, "propTypes", {
onFinished: _propTypes.default.func.isRequired,
mxEvent: _propTypes.default.object.isRequired // the MatrixEvent associated with the context menu
}), _temp)) || _class);
exports.default = ViewSource;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/components/structures/ViewSource.js"],"names":["ViewSource","React","Component","constructor","props","state","isEditing","onBack","setState","onEdit","viewSourceContent","mxEvent","replacingEvent","isEncrypted","decryptedEventSource","_clearEvent","originalEventSource","event","JSON","stringify","getBaseEventId","baseMxEvent","content","event_id","getId","getContent","editSourceContent","isStateEvent","isState","roomId","getRoomId","originalContent","cli","getRoom","eventType","getType","evContent","stateKey","getStateKey","bodyToStartFrom","body","newContent","msgtype","rel_type","canSendStateEvent","MatrixClientPeg","get","room","currentState","mayClientSendStateEvent","render","BaseDialog","sdk","getComponent","eventId","canEdit","onFinished","PropTypes","func","isRequired","object"],"mappings":";;;;;;;;;;;;;AAkBA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;;;IAGqBA,U,WADpB,gDAAqB,uBAArB,C,mCAAD,MACqBA,UADrB,SACwCC,eAAMC,SAD9C,CACwD;AAMpDC,EAAAA,WAAW,CAACC,KAAD,EAAQ;AACf,UAAMA,KAAN;AAEA,SAAKC,KAAL,GAAa;AACTC,MAAAA,SAAS,EAAE;AADF,KAAb;AAGH;;AAEDC,EAAAA,MAAM,GAAG;AACL;AACA,SAAKC,QAAL,CAAc;AAAEF,MAAAA,SAAS,EAAE;AAAb,KAAd;AACH;;AAEDG,EAAAA,MAAM,GAAG;AACL,SAAKD,QAAL,CAAc;AAAEF,MAAAA,SAAS,EAAE;AAAb,KAAd;AACH,GArBmD,CAuBpD;;;AACAI,EAAAA,iBAAiB,GAAG;AAChB,UAAMC,OAAO,GAAG,KAAKP,KAAL,CAAWO,OAAX,CAAmBC,cAAnB,MAAuC,KAAKR,KAAL,CAAWO,OAAlE,CADgB,CAC2D;;AAC3E,UAAME,WAAW,GAAGF,OAAO,CAACE,WAAR,EAApB;AACA,UAAMC,oBAAoB,GAAGH,OAAO,CAACI,WAArC,CAHgB,CAGkC;;AAClD,UAAMC,mBAAmB,GAAGL,OAAO,CAACM,KAApC;;AAEA,QAAIJ,WAAJ,EAAiB;AACb,0BACI,yEACI;AAAS,QAAA,IAAI,MAAb;AAAc,QAAA,SAAS,EAAC;AAAxB,sBACI,2DACI;AAAM,QAAA,SAAS,EAAC;AAAhB,SAAyC,yBAAG,wBAAH,CAAzC,CADJ,CADJ,eAII,6BAAC,wBAAD;AAAiB,QAAA,SAAS,EAAC;AAA3B,SAAmCK,IAAI,CAACC,SAAL,CAAeL,oBAAf,EAAqC,IAArC,EAA2C,CAA3C,CAAnC,CAJJ,CADJ,eAOI;AAAS,QAAA,SAAS,EAAC;AAAnB,sBACI,2DACI;AAAM,QAAA,SAAS,EAAC;AAAhB,SAAyC,yBAAG,uBAAH,CAAzC,CADJ,CADJ,eAII,6BAAC,wBAAD;AAAiB,QAAA,SAAS,EAAC;AAA3B,SAAmCI,IAAI,CAACC,SAAL,CAAeH,mBAAf,EAAoC,IAApC,EAA0C,CAA1C,CAAnC,CAJJ,CAPJ,CADJ;AAgBH,KAjBD,MAiBO;AACH,0BACI,yEACI;AAAK,QAAA,SAAS,EAAC;AAAf,SAAwC,yBAAG,uBAAH,CAAxC,CADJ,eAEI,6BAAC,wBAAD;AAAiB,QAAA,SAAS,EAAC;AAA3B,SAAmCE,IAAI,CAACC,SAAL,CAAeH,mBAAf,EAAoC,IAApC,EAA0C,CAA1C,CAAnC,CAFJ,CADJ;AAMH;AACJ,GAvDmD,CAyDpD;;;AACAI,EAAAA,cAAc,GAAG;AACb,UAAMT,OAAO,GAAG,KAAKP,KAAL,CAAWO,OAAX,CAAmBC,cAAnB,MAAuC,KAAKR,KAAL,CAAWO,OAAlE,CADa,CAC8D;;AAC3E,UAAME,WAAW,GAAGF,OAAO,CAACE,WAAR,EAApB;AACA,UAAMQ,WAAW,GAAG,KAAKjB,KAAL,CAAWO,OAA/B;;AAEA,QAAIE,WAAJ,EAAiB;AACb;AACA,aAAOF,OAAO,CAACM,KAAR,CAAcK,OAAd,CAAsB,cAAtB,GAAuCC,QAAvC,IAAmDF,WAAW,CAACG,KAAZ,EAA1D;AACH,KAHD,MAGO;AACH,aAAOb,OAAO,CAACc,UAAR,GAAqB,cAArB,GAAsCF,QAAtC,IAAkDF,WAAW,CAACG,KAAZ,EAAzD;AACH;AACJ,GArEmD,CAuEpD;;;AACAE,EAAAA,iBAAiB,GAAG;AAChB,UAAMf,OAAO,GAAG,KAAKP,KAAL,CAAWO,OAAX,CAAmBC,cAAnB,MAAuC,KAAKR,KAAL,CAAWO,OAAlE,CADgB,CAC2D;;AAE3E,UAAMgB,YAAY,GAAGhB,OAAO,CAACiB,OAAR,EAArB;AACA,UAAMC,MAAM,GAAGlB,OAAO,CAACmB,SAAR,EAAf;AACA,UAAMC,eAAe,GAAGpB,OAAO,CAACc,UAAR,EAAxB;;AAEA,QAAIE,YAAJ,EAAkB;AACd,0BACI,6BAAC,4BAAD,CAAqB,QAArB,QACMK,GAAD,iBACG,6BAAC,+BAAD;AACI,QAAA,IAAI,EAAEA,GAAG,CAACC,OAAJ,CAAYJ,MAAZ,CADV;AAEI,QAAA,eAAe,EAAE,IAFrB;AAGI,QAAA,MAAM,EAAE,MAAM,KAAKtB,MAAL,EAHlB;AAII,QAAA,MAAM,EAAE;AACJ2B,UAAAA,SAAS,EAAEvB,OAAO,CAACwB,OAAR,EADP;AAEJC,UAAAA,SAAS,EAAElB,IAAI,CAACC,SAAL,CAAeY,eAAf,EAAgC,IAAhC,EAAsC,IAAtC,CAFP;AAGJM,UAAAA,QAAQ,EAAE1B,OAAO,CAAC2B,WAAR;AAHN;AAJZ,QAFR,CADJ;AAgBH,KAjBD,MAiBO;AACH;AACA;AACA,YAAMC,eAAe,GAAGR,eAAe,CAAC,eAAD,CAAf,EAAkCS,IAAlC,IAA0CT,eAAe,CAACS,IAAlF,CAHG,CAGqF;;AACxF,YAAMC,UAAU,GAAG;AACf,gBAAS,MAAKF,eAAgB,EADf;AAEf,mBAAWR,eAAe,CAACW,OAFZ;AAGf,yBAAiB;AACbF,UAAAA,IAAI,EAAED,eADO;AAEbG,UAAAA,OAAO,EAAEX,eAAe,CAACW;AAFZ,SAHF;AAOf,wBAAgB;AACZC,UAAAA,QAAQ,EAAE,WADE;AAEZpB,UAAAA,QAAQ,EAAE,KAAKH,cAAL;AAFE;AAPD,OAAnB;AAYA,0BACI,6BAAC,4BAAD,CAAqB,QAArB,QACMY,GAAD,iBACG,6BAAC,+BAAD;AACI,QAAA,IAAI,EAAEA,GAAG,CAACC,OAAJ,CAAYJ,MAAZ,CADV;AAEI,QAAA,eAAe,EAAE,KAFrB;AAGI,QAAA,iBAAiB,EAAE,IAHvB;AAII,QAAA,MAAM,EAAE,MAAM,KAAKtB,MAAL,EAJlB;AAKI,QAAA,MAAM,EAAE;AACJ2B,UAAAA,SAAS,EAAEvB,OAAO,CAACwB,OAAR,EADP;AAEJC,UAAAA,SAAS,EAAElB,IAAI,CAACC,SAAL,CAAesB,UAAf,EAA2B,IAA3B,EAAiC,IAAjC;AAFP;AALZ,QAFR,CADJ;AAgBH;AACJ;;AAEDG,EAAAA,iBAAiB,CAACjC,OAAD,EAAU;AACvB,UAAMqB,GAAG,GAAGa,iCAAgBC,GAAhB,EAAZ;;AACA,UAAMC,IAAI,GAAGf,GAAG,CAACC,OAAJ,CAAYtB,OAAO,CAACmB,SAAR,EAAZ,CAAb;AACA,WAAOiB,IAAI,CAACC,YAAL,CAAkBC,uBAAlB,CAA0CtC,OAAO,CAACwB,OAAR,EAA1C,EAA6DH,GAA7D,CAAP;AACH;;AAEDkB,EAAAA,MAAM,GAAG;AACL,UAAMC,UAAU,GAAGC,GAAG,CAACC,YAAJ,CAAiB,0BAAjB,CAAnB;AACA,UAAM1C,OAAO,GAAG,KAAKP,KAAL,CAAWO,OAAX,CAAmBC,cAAnB,MAAuC,KAAKR,KAAL,CAAWO,OAAlE,CAFK,CAEsE;;AAE3E,UAAML,SAAS,GAAG,KAAKD,KAAL,CAAWC,SAA7B;AACA,UAAMuB,MAAM,GAAGlB,OAAO,CAACmB,SAAR,EAAf;AACA,UAAMwB,OAAO,GAAG3C,OAAO,CAACa,KAAR,EAAhB;AACA,UAAM+B,OAAO,GAAG5C,OAAO,CAACiB,OAAR,KAAoB,KAAKgB,iBAAL,CAAuBjC,OAAvB,CAApB,GAAsD,gCAAe,KAAKP,KAAL,CAAWO,OAA1B,CAAtE;AACA,wBACI,6BAAC,UAAD;AAAY,MAAA,SAAS,EAAC,eAAtB;AAAsC,MAAA,UAAU,EAAE,KAAKP,KAAL,CAAWoD,UAA7D;AAAyE,MAAA,KAAK,EAAE,yBAAG,aAAH;AAAhF,oBACI,uDACI,uDAAe3B,MAAf,CADJ,eAEI,wDAAgByB,OAAhB,CAFJ,eAGI;AAAK,MAAA,SAAS,EAAC;AAAf,MAHJ,EAIKhD,SAAS,GAAG,KAAKoB,iBAAL,EAAH,GAA8B,KAAKhB,iBAAL,EAJ5C,CADJ,EAOK,CAACJ,SAAD,IAAciD,OAAd,iBACG;AAAK,MAAA,SAAS,EAAC;AAAf,oBACI;AAAQ,MAAA,OAAO,EAAE,MAAM,KAAK9C,MAAL;AAAvB,OAAuC,yBAAG,MAAH,CAAvC,CADJ,CARR,CADJ;AAeH;;AAhKmD,C,sDACjC;AACf+C,EAAAA,UAAU,EAAEC,mBAAUC,IAAV,CAAeC,UADZ;AAEfhD,EAAAA,OAAO,EAAE8C,mBAAUG,MAAV,CAAiBD,UAFX,CAEuB;;AAFvB,C","sourcesContent":["/*\nCopyright 2015, 2016 OpenMarket Ltd\nCopyright 2019 Michael Telatynski <7t3chguy@gmail.com>\nCopyright 2019 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 PropTypes from \"prop-types\";\nimport SyntaxHighlight from \"../views/elements/SyntaxHighlight\";\nimport { _t } from \"../../languageHandler\";\nimport * as sdk from \"../../index\";\nimport MatrixClientContext from \"../../contexts/MatrixClientContext\";\nimport { SendCustomEvent } from \"../views/dialogs/DevtoolsDialog\";\nimport { canEditContent } from \"../../utils/EventUtils\";\nimport { MatrixClientPeg } from '../../MatrixClientPeg';\nimport { replaceableComponent } from \"../../utils/replaceableComponent\";\n\n@replaceableComponent(\"structures.ViewSource\")\nexport default class ViewSource extends React.Component {\n    static propTypes = {\n        onFinished: PropTypes.func.isRequired,\n        mxEvent: PropTypes.object.isRequired, // the MatrixEvent associated with the context menu\n    };\n\n    constructor(props) {\n        super(props);\n\n        this.state = {\n            isEditing: false,\n        };\n    }\n\n    onBack() {\n        // TODO: refresh the \"Event ID:\" modal header\n        this.setState({ isEditing: false });\n    }\n\n    onEdit() {\n        this.setState({ isEditing: true });\n    }\n\n    // returns the dialog body for viewing the event source\n    viewSourceContent() {\n        const mxEvent = this.props.mxEvent.replacingEvent() || this.props.mxEvent; // show the replacing event, not the original, if it is an edit\n        const isEncrypted = mxEvent.isEncrypted();\n        const decryptedEventSource = mxEvent._clearEvent; // FIXME: _clearEvent is private\n        const originalEventSource = mxEvent.event;\n\n        if (isEncrypted) {\n            return (\n                <>\n                    <details open className=\"mx_ViewSource_details\">\n                        <summary>\n                            <span className=\"mx_ViewSource_heading\">{_t(\"Decrypted event source\")}</span>\n                        </summary>\n                        <SyntaxHighlight className=\"json\">{JSON.stringify(decryptedEventSource, null, 2)}</SyntaxHighlight>\n                    </details>\n                    <details className=\"mx_ViewSource_details\">\n                        <summary>\n                            <span className=\"mx_ViewSource_heading\">{_t(\"Original event source\")}</span>\n                        </summary>\n                        <SyntaxHighlight className=\"json\">{JSON.stringify(originalEventSource, null, 2)}</SyntaxHighlight>\n                    </details>\n                </>\n            );\n        } else {\n            return (\n                <>\n                    <div className=\"mx_ViewSource_heading\">{_t(\"Original event source\")}</div>\n                    <SyntaxHighlight className=\"json\">{JSON.stringify(originalEventSource, null, 2)}</SyntaxHighlight>\n                </>\n            );\n        }\n    }\n\n    // returns the id of the initial message, not the id of the previous edit\n    getBaseEventId() {\n        const mxEvent = this.props.mxEvent.replacingEvent() || this.props.mxEvent; // show the replacing event, not the original, if it is an edit\n        const isEncrypted = mxEvent.isEncrypted();\n        const baseMxEvent = this.props.mxEvent;\n\n        if (isEncrypted) {\n            // `relates_to` field is inside the encrypted event\n            return mxEvent.event.content[\"m.relates_to\"]?.event_id ?? baseMxEvent.getId();\n        } else {\n            return mxEvent.getContent()[\"m.relates_to\"]?.event_id ?? baseMxEvent.getId();\n        }\n    }\n\n    // returns the SendCustomEvent component prefilled with the correct details\n    editSourceContent() {\n        const mxEvent = this.props.mxEvent.replacingEvent() || this.props.mxEvent; // show the replacing event, not the original, if it is an edit\n\n        const isStateEvent = mxEvent.isState();\n        const roomId = mxEvent.getRoomId();\n        const originalContent = mxEvent.getContent();\n\n        if (isStateEvent) {\n            return (\n                <MatrixClientContext.Consumer>\n                    {(cli) => (\n                        <SendCustomEvent\n                            room={cli.getRoom(roomId)}\n                            forceStateEvent={true}\n                            onBack={() => this.onBack()}\n                            inputs={{\n                                eventType: mxEvent.getType(),\n                                evContent: JSON.stringify(originalContent, null, \"\\t\"),\n                                stateKey: mxEvent.getStateKey(),\n                            }}\n                        />\n                    )}\n                </MatrixClientContext.Consumer>\n            );\n        } else {\n            // prefill an edit-message event\n            // keep only the `body` and `msgtype` fields of originalContent\n            const bodyToStartFrom = originalContent[\"m.new_content\"]?.body ?? originalContent.body; // prefill the last edit body, to start editing from there\n            const newContent = {\n                \"body\": ` * ${bodyToStartFrom}`,\n                \"msgtype\": originalContent.msgtype,\n                \"m.new_content\": {\n                    body: bodyToStartFrom,\n                    msgtype: originalContent.msgtype,\n                },\n                \"m.relates_to\": {\n                    rel_type: \"m.replace\",\n                    event_id: this.getBaseEventId(),\n                },\n            };\n            return (\n                <MatrixClientContext.Consumer>\n                    {(cli) => (\n                        <SendCustomEvent\n                            room={cli.getRoom(roomId)}\n                            forceStateEvent={false}\n                            forceGeneralEvent={true}\n                            onBack={() => this.onBack()}\n                            inputs={{\n                                eventType: mxEvent.getType(),\n                                evContent: JSON.stringify(newContent, null, \"\\t\"),\n                            }}\n                        />\n                    )}\n                </MatrixClientContext.Consumer>\n            );\n        }\n    }\n\n    canSendStateEvent(mxEvent) {\n        const cli = MatrixClientPeg.get();\n        const room = cli.getRoom(mxEvent.getRoomId());\n        return room.currentState.mayClientSendStateEvent(mxEvent.getType(), cli);\n    }\n\n    render() {\n        const BaseDialog = sdk.getComponent(\"views.dialogs.BaseDialog\");\n        const mxEvent = this.props.mxEvent.replacingEvent() || this.props.mxEvent; // show the replacing event, not the original, if it is an edit\n\n        const isEditing = this.state.isEditing;\n        const roomId = mxEvent.getRoomId();\n        const eventId = mxEvent.getId();\n        const canEdit = mxEvent.isState() ? this.canSendStateEvent(mxEvent) : canEditContent(this.props.mxEvent);\n        return (\n            <BaseDialog className=\"mx_ViewSource\" onFinished={this.props.onFinished} title={_t(\"View Source\")}>\n                <div>\n                    <div>Room ID: {roomId}</div>\n                    <div>Event ID: {eventId}</div>\n                    <div className=\"mx_ViewSource_separator\" />\n                    {isEditing ? this.editSourceContent() : this.viewSourceContent()}\n                </div>\n                {!isEditing && canEdit && (\n                    <div className=\"mx_Dialog_buttons\">\n                        <button onClick={() => this.onEdit()}>{_t(\"Edit\")}</button>\n                    </div>\n                )}\n            </BaseDialog>\n        );\n    }\n}\n"]}