matrix-react-sdk
Version:
SDK for matrix.org using React
204 lines (166 loc) • 28.8 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 = _interopRequireWildcard(require("react"));
var _BaseDialog = _interopRequireDefault(require("./BaseDialog"));
var _languageHandler = require("../../../languageHandler");
var _AccessibleButton = _interopRequireDefault(require("../elements/AccessibleButton"));
var _matrixWidgetApi = require("matrix-widget-api");
var _StopGapWidgetDriver = require("../../../stores/widgets/StopGapWidgetDriver");
var _MatrixClientPeg = require("../../../MatrixClientPeg");
var _OwnProfileStore = require("../../../stores/OwnProfileStore");
var _arrays = require("../../../utils/arrays");
var _StopGapWidget = require("../../../stores/widgets/StopGapWidget");
var _replaceableComponent = require("../../../utils/replaceableComponent");
var _identifiers = require("../../../identifiers");
var _SettingsStore = _interopRequireDefault(require("../../../settings/SettingsStore"));
var _dec, _class, _temp;
function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); keys.push.apply(keys, symbols); } return keys; }
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; if (i % 2) { ownKeys(Object(source), true).forEach(function (key) { (0, _defineProperty2.default)(target, key, source[key]); }); } else if (Object.getOwnPropertyDescriptors) { Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)); } else { ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } } return target; }
const MAX_BUTTONS = 3;
let ModalWidgetDialog = (_dec = (0, _replaceableComponent.replaceableComponent)("views.dialogs.ModalWidgetDialog"), _dec(_class = (_temp = class ModalWidgetDialog extends React.PureComponent
/*:: <IProps, IState>*/
{
constructor(props) {
super(props);
(0, _defineProperty2.default)(this, "widget", void 0);
(0, _defineProperty2.default)(this, "possibleButtons", void 0);
(0, _defineProperty2.default)(this, "appFrame", /*#__PURE__*/React.createRef());
(0, _defineProperty2.default)(this, "state", {
disabledButtonIds: []
});
(0, _defineProperty2.default)(this, "onReady", () => {
this.state.messaging.sendWidgetConfig(this.props.widgetDefinition);
});
(0, _defineProperty2.default)(this, "onLoad", () => {
this.state.messaging.once("ready", this.onReady);
this.state.messaging.on(`action:${_matrixWidgetApi.WidgetApiFromWidgetAction.CloseModalWidget}`, this.onWidgetClose);
this.state.messaging.on(`action:${_matrixWidgetApi.WidgetApiFromWidgetAction.SetModalButtonEnabled}`, this.onButtonEnableToggle);
});
(0, _defineProperty2.default)(this, "onWidgetClose", (ev
/*: CustomEvent<IModalWidgetCloseRequest>*/
) => {
this.props.onFinished(true, ev.detail.data);
});
(0, _defineProperty2.default)(this, "onButtonEnableToggle", (ev
/*: CustomEvent<ISetModalButtonEnabledActionRequest>*/
) => {
ev.preventDefault();
const isClose = ev.detail.data.button === _matrixWidgetApi.BuiltInModalButtonID.Close;
if (isClose || !this.possibleButtons.includes(ev.detail.data.button)) {
return this.state.messaging.transport.reply(ev.detail, {
error: {
message: "Invalid button"
}
});
}
let buttonIds
/*: ModalButtonID[]*/
;
if (ev.detail.data.enabled) {
buttonIds = (0, _arrays.arrayFastClone)(this.state.disabledButtonIds).filter(i => i !== ev.detail.data.button);
} else {
// use a set to swap the operation to avoid memory leaky arrays.
const tempSet = new Set(this.state.disabledButtonIds);
tempSet.add(ev.detail.data.button);
buttonIds = Array.from(tempSet);
}
this.setState({
disabledButtonIds: buttonIds
});
this.state.messaging.transport.reply(ev.detail, {});
});
this.widget = new _StopGapWidget.ElementWidget(_objectSpread(_objectSpread({}, this.props.widgetDefinition), {}, {
creatorUserId: _MatrixClientPeg.MatrixClientPeg.get().getUserId(),
id: `modal_${this.props.sourceWidgetId}`
}));
this.possibleButtons = (this.props.widgetDefinition.buttons || []).map(b => b.id);
}
componentDidMount() {
const driver = new _StopGapWidgetDriver.StopGapWidgetDriver([], this.widget, _matrixWidgetApi.WidgetKind.Modal);
const messaging = new _matrixWidgetApi.ClientWidgetApi(this.widget, this.appFrame.current, driver);
this.setState({
messaging
});
}
componentWillUnmount() {
this.state.messaging.off("ready", this.onReady);
this.state.messaging.off(`action:${_matrixWidgetApi.WidgetApiFromWidgetAction.CloseModalWidget}`, this.onWidgetClose);
this.state.messaging.stop();
}
render() {
const templated = this.widget.getCompleteUrl({
widgetRoomId: this.props.widgetRoomId,
currentUserId: _MatrixClientPeg.MatrixClientPeg.get().getUserId(),
userDisplayName: _OwnProfileStore.OwnProfileStore.instance.displayName,
userHttpAvatarUrl: _OwnProfileStore.OwnProfileStore.instance.getHttpAvatarUrl(),
clientId: _identifiers.ELEMENT_CLIENT_ID,
clientTheme: _SettingsStore.default.getValue("theme"),
clientLanguage: (0, _languageHandler.getUserLanguage)()
});
const parsed = new URL(templated); // Add in some legacy support sprinkles (for non-popout widgets)
// TODO: Replace these with proper widget params
// See https://github.com/matrix-org/matrix-doc/pull/1958/files#r405714833
parsed.searchParams.set('widgetId', this.widget.id);
parsed.searchParams.set('parentUrl', window.location.href.split('#', 2)[0]); // Replace the encoded dollar signs back to dollar signs. They have no special meaning
// in HTTP, but URL parsers encode them anyways.
const widgetUrl = parsed.toString().replace(/%24/g, '$');
let buttons;
if (this.props.widgetDefinition.buttons) {
// show first button rightmost for a more natural specification
buttons = this.props.widgetDefinition.buttons.slice(0, MAX_BUTTONS).reverse().map(def => {
let kind = "secondary";
switch (def.kind) {
case _matrixWidgetApi.ModalButtonKind.Primary:
kind = "primary";
break;
case _matrixWidgetApi.ModalButtonKind.Secondary:
kind = "primary_outline";
break;
case _matrixWidgetApi.ModalButtonKind.Danger:
kind = "danger";
break;
}
const onClick = () => {
this.state.messaging.notifyModalWidgetButtonClicked(def.id);
};
const isDisabled = this.state.disabledButtonIds.includes(def.id);
return /*#__PURE__*/React.createElement(_AccessibleButton.default, {
key: def.id,
kind: kind,
onClick: onClick,
disabled: isDisabled
}, def.label);
});
}
return /*#__PURE__*/React.createElement(_BaseDialog.default, {
title: this.props.widgetDefinition.name || (0, _languageHandler._t)("Modal Widget"),
className: "mx_ModalWidgetDialog",
contentId: "mx_Dialog_content",
onFinished: this.props.onFinished
}, /*#__PURE__*/React.createElement("div", {
className: "mx_ModalWidgetDialog_warning"
}, /*#__PURE__*/React.createElement("img", {
src: require("../../../../res/img/element-icons/warning-badge.svg"),
height: "16",
width: "16",
alt: ""
}), (0, _languageHandler._t)("Data on this screen is shared with %(widgetDomain)s", {
widgetDomain: parsed.hostname
})), /*#__PURE__*/React.createElement("div", null, /*#__PURE__*/React.createElement("iframe", {
ref: this.appFrame,
sandbox: "allow-forms allow-scripts allow-same-origin",
src: widgetUrl,
onLoad: this.onLoad
})), /*#__PURE__*/React.createElement("div", {
className: "mx_ModalWidgetDialog_buttons"
}, buttons));
}
}, _temp)) || _class);
exports.default = ModalWidgetDialog;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/views/dialogs/ModalWidgetDialog.tsx"],"names":["MAX_BUTTONS","ModalWidgetDialog","React","PureComponent","constructor","props","createRef","disabledButtonIds","state","messaging","sendWidgetConfig","widgetDefinition","once","onReady","on","WidgetApiFromWidgetAction","CloseModalWidget","onWidgetClose","SetModalButtonEnabled","onButtonEnableToggle","ev","onFinished","detail","data","preventDefault","isClose","button","BuiltInModalButtonID","Close","possibleButtons","includes","transport","reply","error","message","buttonIds","enabled","filter","i","tempSet","Set","add","Array","from","setState","widget","ElementWidget","creatorUserId","MatrixClientPeg","get","getUserId","id","sourceWidgetId","buttons","map","b","componentDidMount","driver","StopGapWidgetDriver","WidgetKind","Modal","ClientWidgetApi","appFrame","current","componentWillUnmount","off","stop","render","templated","getCompleteUrl","widgetRoomId","currentUserId","userDisplayName","OwnProfileStore","instance","displayName","userHttpAvatarUrl","getHttpAvatarUrl","clientId","ELEMENT_CLIENT_ID","clientTheme","SettingsStore","getValue","clientLanguage","parsed","URL","searchParams","set","window","location","href","split","widgetUrl","toString","replace","slice","reverse","def","kind","ModalButtonKind","Primary","Secondary","Danger","onClick","notifyModalWidgetButtonClicked","isDisabled","label","name","require","widgetDomain","hostname","onLoad"],"mappings":";;;;;;;;;;;;;AAgBA;;AACA;;AACA;;AACA;;AACA;;AAeA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;;;;;;;AAcA,MAAMA,WAAW,GAAG,CAApB;IAGqBC,iB,WADpB,gDAAqB,iCAArB,C,yBAAD,MACqBA,iBADrB,SAC+CC,KAAK,CAACC;AADrD;AACmF;AAS/EC,EAAAA,WAAW,CAACC,KAAD,EAAQ;AACf,UAAMA,KAAN;AADe;AAAA;AAAA,iEANoCH,KAAK,CAACI,SAAN,EAMpC;AAAA,iDAJH;AACZC,MAAAA,iBAAiB,EAAE;AADP,KAIG;AAAA,mDAuBD,MAAM;AACpB,WAAKC,KAAL,CAAWC,SAAX,CAAqBC,gBAArB,CAAsC,KAAKL,KAAL,CAAWM,gBAAjD;AACH,KAzBkB;AAAA,kDA2BF,MAAM;AACnB,WAAKH,KAAL,CAAWC,SAAX,CAAqBG,IAArB,CAA0B,OAA1B,EAAmC,KAAKC,OAAxC;AACA,WAAKL,KAAL,CAAWC,SAAX,CAAqBK,EAArB,CAAyB,UAASC,2CAA0BC,gBAAiB,EAA7E,EAAgF,KAAKC,aAArF;AACA,WAAKT,KAAL,CAAWC,SAAX,CAAqBK,EAArB,CAAyB,UAASC,2CAA0BG,qBAAsB,EAAlF,EAAqF,KAAKC,oBAA1F;AACH,KA/BkB;AAAA,yDAiCK,CAACC;AAAD;AAAA,SAA+C;AACnE,WAAKf,KAAL,CAAWgB,UAAX,CAAsB,IAAtB,EAA4BD,EAAE,CAACE,MAAH,CAAUC,IAAtC;AACH,KAnCkB;AAAA,gEAqCY,CAACH;AAAD;AAAA,SAA0D;AACrFA,MAAAA,EAAE,CAACI,cAAH;AACA,YAAMC,OAAO,GAAGL,EAAE,CAACE,MAAH,CAAUC,IAAV,CAAeG,MAAf,KAA0BC,sCAAqBC,KAA/D;;AACA,UAAIH,OAAO,IAAI,CAAC,KAAKI,eAAL,CAAqBC,QAArB,CAA8BV,EAAE,CAACE,MAAH,CAAUC,IAAV,CAAeG,MAA7C,CAAhB,EAAsE;AAClE,eAAO,KAAKlB,KAAL,CAAWC,SAAX,CAAqBsB,SAArB,CAA+BC,KAA/B,CAAqCZ,EAAE,CAACE,MAAxC,EAAgD;AACnDW,UAAAA,KAAK,EAAE;AAACC,YAAAA,OAAO,EAAE;AAAV;AAD4C,SAAhD,CAAP;AAGH;;AAED,UAAIC;AAA0B;AAA9B;;AACA,UAAIf,EAAE,CAACE,MAAH,CAAUC,IAAV,CAAea,OAAnB,EAA4B;AACxBD,QAAAA,SAAS,GAAG,4BAAe,KAAK3B,KAAL,CAAWD,iBAA1B,EAA6C8B,MAA7C,CAAoDC,CAAC,IAAIA,CAAC,KAAKlB,EAAE,CAACE,MAAH,CAAUC,IAAV,CAAeG,MAA9E,CAAZ;AACH,OAFD,MAEO;AACH;AACA,cAAMa,OAAO,GAAG,IAAIC,GAAJ,CAAQ,KAAKhC,KAAL,CAAWD,iBAAnB,CAAhB;AACAgC,QAAAA,OAAO,CAACE,GAAR,CAAYrB,EAAE,CAACE,MAAH,CAAUC,IAAV,CAAeG,MAA3B;AACAS,QAAAA,SAAS,GAAGO,KAAK,CAACC,IAAN,CAAWJ,OAAX,CAAZ;AACH;;AACD,WAAKK,QAAL,CAAc;AAACrC,QAAAA,iBAAiB,EAAE4B;AAApB,OAAd;AACA,WAAK3B,KAAL,CAAWC,SAAX,CAAqBsB,SAArB,CAA+BC,KAA/B,CAAqCZ,EAAE,CAACE,MAAxC,EAAgD,EAAhD;AACH,KAzDkB;AAGf,SAAKuB,MAAL,GAAc,IAAIC,4BAAJ,iCACP,KAAKzC,KAAL,CAAWM,gBADJ;AAEVoC,MAAAA,aAAa,EAAEC,iCAAgBC,GAAhB,GAAsBC,SAAtB,EAFL;AAGVC,MAAAA,EAAE,EAAG,SAAQ,KAAK9C,KAAL,CAAW+C,cAAe;AAH7B,OAAd;AAKA,SAAKvB,eAAL,GAAuB,CAAC,KAAKxB,KAAL,CAAWM,gBAAX,CAA4B0C,OAA5B,IAAuC,EAAxC,EAA4CC,GAA5C,CAAgDC,CAAC,IAAIA,CAAC,CAACJ,EAAvD,CAAvB;AACH;;AAEMK,EAAAA,iBAAP,GAA2B;AACvB,UAAMC,MAAM,GAAG,IAAIC,wCAAJ,CAAyB,EAAzB,EAA6B,KAAKb,MAAlC,EAA0Cc,4BAAWC,KAArD,CAAf;AACA,UAAMnD,SAAS,GAAG,IAAIoD,gCAAJ,CAAoB,KAAKhB,MAAzB,EAAiC,KAAKiB,QAAL,CAAcC,OAA/C,EAAwDN,MAAxD,CAAlB;AACA,SAAKb,QAAL,CAAc;AAACnC,MAAAA;AAAD,KAAd;AACH;;AAEMuD,EAAAA,oBAAP,GAA8B;AAC1B,SAAKxD,KAAL,CAAWC,SAAX,CAAqBwD,GAArB,CAAyB,OAAzB,EAAkC,KAAKpD,OAAvC;AACA,SAAKL,KAAL,CAAWC,SAAX,CAAqBwD,GAArB,CAA0B,UAASlD,2CAA0BC,gBAAiB,EAA9E,EAAiF,KAAKC,aAAtF;AACA,SAAKT,KAAL,CAAWC,SAAX,CAAqByD,IAArB;AACH;;AAsCMC,EAAAA,MAAP,GAAgB;AACZ,UAAMC,SAAS,GAAG,KAAKvB,MAAL,CAAYwB,cAAZ,CAA2B;AACzCC,MAAAA,YAAY,EAAE,KAAKjE,KAAL,CAAWiE,YADgB;AAEzCC,MAAAA,aAAa,EAAEvB,iCAAgBC,GAAhB,GAAsBC,SAAtB,EAF0B;AAGzCsB,MAAAA,eAAe,EAAEC,iCAAgBC,QAAhB,CAAyBC,WAHD;AAIzCC,MAAAA,iBAAiB,EAAEH,iCAAgBC,QAAhB,CAAyBG,gBAAzB,EAJsB;AAKzCC,MAAAA,QAAQ,EAAEC,8BAL+B;AAMzCC,MAAAA,WAAW,EAAEC,uBAAcC,QAAd,CAAuB,OAAvB,CAN4B;AAOzCC,MAAAA,cAAc,EAAE;AAPyB,KAA3B,CAAlB;AAUA,UAAMC,MAAM,GAAG,IAAIC,GAAJ,CAAQjB,SAAR,CAAf,CAXY,CAaZ;AACA;AACA;;AACAgB,IAAAA,MAAM,CAACE,YAAP,CAAoBC,GAApB,CAAwB,UAAxB,EAAoC,KAAK1C,MAAL,CAAYM,EAAhD;AACAiC,IAAAA,MAAM,CAACE,YAAP,CAAoBC,GAApB,CAAwB,WAAxB,EAAqCC,MAAM,CAACC,QAAP,CAAgBC,IAAhB,CAAqBC,KAArB,CAA2B,GAA3B,EAAgC,CAAhC,EAAmC,CAAnC,CAArC,EAjBY,CAmBZ;AACA;;AACA,UAAMC,SAAS,GAAGR,MAAM,CAACS,QAAP,GAAkBC,OAAlB,CAA0B,MAA1B,EAAkC,GAAlC,CAAlB;AAEA,QAAIzC,OAAJ;;AACA,QAAI,KAAKhD,KAAL,CAAWM,gBAAX,CAA4B0C,OAAhC,EAAyC;AACrC;AACAA,MAAAA,OAAO,GAAG,KAAKhD,KAAL,CAAWM,gBAAX,CAA4B0C,OAA5B,CAAoC0C,KAApC,CAA0C,CAA1C,EAA6C/F,WAA7C,EAA0DgG,OAA1D,GAAoE1C,GAApE,CAAwE2C,GAAG,IAAI;AACrF,YAAIC,IAAI,GAAG,WAAX;;AACA,gBAAQD,GAAG,CAACC,IAAZ;AACI,eAAKC,iCAAgBC,OAArB;AACIF,YAAAA,IAAI,GAAG,SAAP;AACA;;AACJ,eAAKC,iCAAgBE,SAArB;AACIH,YAAAA,IAAI,GAAG,iBAAP;AACA;;AACJ,eAAKC,iCAAgBG,MAArB;AACIJ,YAAAA,IAAI,GAAG,QAAP;AACA;AATR;;AAYA,cAAMK,OAAO,GAAG,MAAM;AAClB,eAAK/F,KAAL,CAAWC,SAAX,CAAqB+F,8BAArB,CAAoDP,GAAG,CAAC9C,EAAxD;AACH,SAFD;;AAIA,cAAMsD,UAAU,GAAG,KAAKjG,KAAL,CAAWD,iBAAX,CAA6BuB,QAA7B,CAAsCmE,GAAG,CAAC9C,EAA1C,CAAnB;AAEA,4BAAO,oBAAC,yBAAD;AAAkB,UAAA,GAAG,EAAE8C,GAAG,CAAC9C,EAA3B;AAA+B,UAAA,IAAI,EAAE+C,IAArC;AAA2C,UAAA,OAAO,EAAEK,OAApD;AAA6D,UAAA,QAAQ,EAAEE;AAAvE,WACDR,GAAG,CAACS,KADH,CAAP;AAGH,OAvBS,CAAV;AAwBH;;AAED,wBAAO,oBAAC,mBAAD;AACH,MAAA,KAAK,EAAE,KAAKrG,KAAL,CAAWM,gBAAX,CAA4BgG,IAA5B,IAAoC,yBAAG,cAAH,CADxC;AAEH,MAAA,SAAS,EAAC,sBAFP;AAGH,MAAA,SAAS,EAAC,mBAHP;AAIH,MAAA,UAAU,EAAE,KAAKtG,KAAL,CAAWgB;AAJpB,oBAMH;AAAK,MAAA,SAAS,EAAC;AAAf,oBACI;AACI,MAAA,GAAG,EAAEuF,OAAO,CAAC,qDAAD,CADhB;AAEI,MAAA,MAAM,EAAC,IAFX;AAGI,MAAA,KAAK,EAAC,IAHV;AAII,MAAA,GAAG,EAAC;AAJR,MADJ,EAOK,yBAAG,qDAAH,EAA0D;AACvDC,MAAAA,YAAY,EAAEzB,MAAM,CAAC0B;AADkC,KAA1D,CAPL,CANG,eAiBH,8CACI;AACI,MAAA,GAAG,EAAE,KAAKhD,QADd;AAEI,MAAA,OAAO,EAAC,6CAFZ;AAGI,MAAA,GAAG,EAAE8B,SAHT;AAII,MAAA,MAAM,EAAE,KAAKmB;AAJjB,MADJ,CAjBG,eAyBH;AAAK,MAAA,SAAS,EAAC;AAAf,OACM1D,OADN,CAzBG,CAAP;AA6BH;;AArJ8E,C","sourcesContent":["/*\nCopyright 2020, 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 * as React from 'react';\nimport BaseDialog from './BaseDialog';\nimport { _t, getUserLanguage } from '../../../languageHandler';\nimport AccessibleButton from \"../elements/AccessibleButton\";\nimport {\n    ClientWidgetApi,\n    IModalWidgetCloseRequest,\n    IModalWidgetOpenRequestData,\n    IModalWidgetReturnData,\n    ISetModalButtonEnabledActionRequest,\n    IWidgetApiAcknowledgeResponseData,\n    IWidgetApiErrorResponseData,\n    BuiltInModalButtonID,\n    ModalButtonID,\n    ModalButtonKind,\n    Widget,\n    WidgetApiFromWidgetAction,\n    WidgetKind,\n} from \"matrix-widget-api\";\nimport {StopGapWidgetDriver} from \"../../../stores/widgets/StopGapWidgetDriver\";\nimport {MatrixClientPeg} from \"../../../MatrixClientPeg\";\nimport {OwnProfileStore} from \"../../../stores/OwnProfileStore\";\nimport { arrayFastClone } from \"../../../utils/arrays\";\nimport { ElementWidget } from \"../../../stores/widgets/StopGapWidget\";\nimport {replaceableComponent} from \"../../../utils/replaceableComponent\";\nimport {ELEMENT_CLIENT_ID} from \"../../../identifiers\";\nimport SettingsStore from \"../../../settings/SettingsStore\";\n\ninterface IProps {\n    widgetDefinition: IModalWidgetOpenRequestData;\n    widgetRoomId?: string;\n    sourceWidgetId: string;\n    onFinished(success: boolean, data?: IModalWidgetReturnData): void;\n}\n\ninterface IState {\n    messaging?: ClientWidgetApi;\n    disabledButtonIds: ModalButtonID[];\n}\n\nconst MAX_BUTTONS = 3;\n\n@replaceableComponent(\"views.dialogs.ModalWidgetDialog\")\nexport default class ModalWidgetDialog extends React.PureComponent<IProps, IState> {\n    private readonly widget: Widget;\n    private readonly possibleButtons: ModalButtonID[];\n    private appFrame: React.RefObject<HTMLIFrameElement> = React.createRef();\n\n    state: IState = {\n        disabledButtonIds: [],\n    };\n\n    constructor(props) {\n        super(props);\n\n        this.widget = new ElementWidget({\n            ...this.props.widgetDefinition,\n            creatorUserId: MatrixClientPeg.get().getUserId(),\n            id: `modal_${this.props.sourceWidgetId}`,\n        });\n        this.possibleButtons = (this.props.widgetDefinition.buttons || []).map(b => b.id);\n    }\n\n    public componentDidMount() {\n        const driver = new StopGapWidgetDriver( [], this.widget, WidgetKind.Modal);\n        const messaging = new ClientWidgetApi(this.widget, this.appFrame.current, driver);\n        this.setState({messaging});\n    }\n\n    public componentWillUnmount() {\n        this.state.messaging.off(\"ready\", this.onReady);\n        this.state.messaging.off(`action:${WidgetApiFromWidgetAction.CloseModalWidget}`, this.onWidgetClose);\n        this.state.messaging.stop();\n    }\n\n    private onReady = () => {\n        this.state.messaging.sendWidgetConfig(this.props.widgetDefinition);\n    };\n\n    private onLoad = () => {\n        this.state.messaging.once(\"ready\", this.onReady);\n        this.state.messaging.on(`action:${WidgetApiFromWidgetAction.CloseModalWidget}`, this.onWidgetClose);\n        this.state.messaging.on(`action:${WidgetApiFromWidgetAction.SetModalButtonEnabled}`, this.onButtonEnableToggle);\n    };\n\n    private onWidgetClose = (ev: CustomEvent<IModalWidgetCloseRequest>) => {\n        this.props.onFinished(true, ev.detail.data);\n    }\n\n    private onButtonEnableToggle = (ev: CustomEvent<ISetModalButtonEnabledActionRequest>) => {\n        ev.preventDefault();\n        const isClose = ev.detail.data.button === BuiltInModalButtonID.Close;\n        if (isClose || !this.possibleButtons.includes(ev.detail.data.button)) {\n            return this.state.messaging.transport.reply(ev.detail, {\n                error: {message: \"Invalid button\"},\n            } as IWidgetApiErrorResponseData);\n        }\n\n        let buttonIds: ModalButtonID[];\n        if (ev.detail.data.enabled) {\n            buttonIds = arrayFastClone(this.state.disabledButtonIds).filter(i => i !== ev.detail.data.button);\n        } else {\n            // use a set to swap the operation to avoid memory leaky arrays.\n            const tempSet = new Set(this.state.disabledButtonIds);\n            tempSet.add(ev.detail.data.button);\n            buttonIds = Array.from(tempSet);\n        }\n        this.setState({disabledButtonIds: buttonIds});\n        this.state.messaging.transport.reply(ev.detail, {} as IWidgetApiAcknowledgeResponseData);\n    };\n\n    public render() {\n        const templated = this.widget.getCompleteUrl({\n            widgetRoomId: this.props.widgetRoomId,\n            currentUserId: MatrixClientPeg.get().getUserId(),\n            userDisplayName: OwnProfileStore.instance.displayName,\n            userHttpAvatarUrl: OwnProfileStore.instance.getHttpAvatarUrl(),\n            clientId: ELEMENT_CLIENT_ID,\n            clientTheme: SettingsStore.getValue(\"theme\"),\n            clientLanguage: getUserLanguage(),\n        });\n\n        const parsed = new URL(templated);\n\n        // Add in some legacy support sprinkles (for non-popout widgets)\n        // TODO: Replace these with proper widget params\n        // See https://github.com/matrix-org/matrix-doc/pull/1958/files#r405714833\n        parsed.searchParams.set('widgetId', this.widget.id);\n        parsed.searchParams.set('parentUrl', window.location.href.split('#', 2)[0]);\n\n        // Replace the encoded dollar signs back to dollar signs. They have no special meaning\n        // in HTTP, but URL parsers encode them anyways.\n        const widgetUrl = parsed.toString().replace(/%24/g, '$');\n\n        let buttons;\n        if (this.props.widgetDefinition.buttons) {\n            // show first button rightmost for a more natural specification\n            buttons = this.props.widgetDefinition.buttons.slice(0, MAX_BUTTONS).reverse().map(def => {\n                let kind = \"secondary\";\n                switch (def.kind) {\n                    case ModalButtonKind.Primary:\n                        kind = \"primary\";\n                        break;\n                    case ModalButtonKind.Secondary:\n                        kind = \"primary_outline\";\n                        break\n                    case ModalButtonKind.Danger:\n                        kind = \"danger\";\n                        break;\n                }\n\n                const onClick = () => {\n                    this.state.messaging.notifyModalWidgetButtonClicked(def.id);\n                };\n\n                const isDisabled = this.state.disabledButtonIds.includes(def.id);\n\n                return <AccessibleButton key={def.id} kind={kind} onClick={onClick} disabled={isDisabled}>\n                    { def.label }\n                </AccessibleButton>;\n            });\n        }\n\n        return <BaseDialog\n            title={this.props.widgetDefinition.name || _t(\"Modal Widget\")}\n            className=\"mx_ModalWidgetDialog\"\n            contentId=\"mx_Dialog_content\"\n            onFinished={this.props.onFinished}\n        >\n            <div className=\"mx_ModalWidgetDialog_warning\">\n                <img\n                    src={require(\"../../../../res/img/element-icons/warning-badge.svg\")}\n                    height=\"16\"\n                    width=\"16\"\n                    alt=\"\"\n                />\n                {_t(\"Data on this screen is shared with %(widgetDomain)s\", {\n                    widgetDomain: parsed.hostname,\n                })}\n            </div>\n            <div>\n                <iframe\n                    ref={this.appFrame}\n                    sandbox=\"allow-forms allow-scripts allow-same-origin\"\n                    src={widgetUrl}\n                    onLoad={this.onLoad}\n                />\n            </div>\n            <div className=\"mx_ModalWidgetDialog_buttons\">\n                { buttons }\n            </div>\n        </BaseDialog>;\n    }\n}\n"]}