UNPKG

matrix-react-sdk

Version:
641 lines (624 loc) 120 kB
"use strict"; 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 _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty")); var _react = _interopRequireWildcard(require("react")); var _classnames = _interopRequireDefault(require("classnames")); var _matrixWidgetApi = require("matrix-widget-api"); var _matrix = require("matrix-js-sdk/src/matrix"); var _types = require("matrix-js-sdk/src/types"); var _logger = require("matrix-js-sdk/src/logger"); var _WidgetLifecycle = require("@matrix-org/react-sdk-module-api/lib/lifecycles/WidgetLifecycle"); var _overflowHorizontal = _interopRequireDefault(require("@vector-im/compound-design-tokens/assets/web/icons/overflow-horizontal")); var _AccessibleButton = _interopRequireDefault(require("./AccessibleButton")); var _languageHandler = require("../../../languageHandler"); var _AppPermission = _interopRequireDefault(require("./AppPermission")); var _AppWarning = _interopRequireDefault(require("./AppWarning")); var _Spinner = _interopRequireDefault(require("./Spinner")); var _dispatcher = _interopRequireDefault(require("../../../dispatcher/dispatcher")); var _ActiveWidgetStore = _interopRequireDefault(require("../../../stores/ActiveWidgetStore")); var _SettingsStore = _interopRequireDefault(require("../../../settings/SettingsStore")); var _ContextMenu = require("../../structures/ContextMenu"); var _PersistedElement = _interopRequireWildcard(require("./PersistedElement")); var _WidgetType = require("../../../widgets/WidgetType"); var _StopGapWidget = require("../../../stores/widgets/StopGapWidget"); var _WidgetContextMenu = require("../context_menus/WidgetContextMenu"); var _WidgetAvatar = _interopRequireDefault(require("../avatars/WidgetAvatar")); var _LegacyCallHandler = _interopRequireDefault(require("../../../LegacyCallHandler")); var _WidgetStore = require("../../../stores/WidgetStore"); var _minimiseCollapse = require("../../../../res/img/element-icons/minimise-collapse.svg"); var _maximiseExpand = require("../../../../res/img/element-icons/maximise-expand.svg"); var _minusButton = require("../../../../res/img/element-icons/minus-button.svg"); var _externalLink = require("../../../../res/img/feather-customised/widget/external-link.svg"); var _WidgetLayoutStore = require("../../../stores/widgets/WidgetLayoutStore"); var _OwnProfileStore = require("../../../stores/OwnProfileStore"); var _AsyncStore = require("../../../stores/AsyncStore"); var _WidgetUtils = _interopRequireDefault(require("../../../utils/WidgetUtils")); var _MatrixClientContext = _interopRequireDefault(require("../../../contexts/MatrixClientContext")); var _actions = require("../../../dispatcher/actions"); var _ElementWidgetCapabilities = require("../../../stores/widgets/ElementWidgetCapabilities"); var _WidgetMessagingStore = require("../../../stores/widgets/WidgetMessagingStore"); var _SDKContext = require("../../../contexts/SDKContext"); var _ModuleRunner = require("../../../modules/ModuleRunner"); var _UrlUtils = require("../../../utils/UrlUtils"); var _RightPanelStore = _interopRequireDefault(require("../../../stores/right-panel/RightPanelStore.ts")); var _RightPanelStorePhases = require("../../../stores/right-panel/RightPanelStorePhases.ts"); function _getRequireWildcardCache(e) { if ("function" != typeof WeakMap) return null; var r = new WeakMap(), t = new WeakMap(); return (_getRequireWildcardCache = function (e) { return e ? t : r; })(e); } function _interopRequireWildcard(e, r) { if (!r && e && e.__esModule) return e; if (null === e || "object" != typeof e && "function" != typeof e) return { default: e }; var t = _getRequireWildcardCache(r); if (t && t.has(e)) return t.get(e); var n = { __proto__: null }, a = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var u in e) if ("default" !== u && {}.hasOwnProperty.call(e, u)) { var i = a ? Object.getOwnPropertyDescriptor(e, u) : null; i && (i.get || i.set) ? Object.defineProperty(n, u, i) : n[u] = e[u]; } return n.default = e, t && t.set(e, n), n; } function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; } function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { (0, _defineProperty2.default)(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; } /* Copyright 2024 New Vector Ltd. Copyright 2020-2022 The Matrix.org Foundation C.I.C. Copyright 2019 Michael Telatynski <7t3chguy@gmail.com> Copyright 2018 New Vector Ltd Copyright 2017 Vector Creations Ltd SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only Please see LICENSE files in the repository root for full details. */ class AppTile extends _react.default.Component { constructor(_props, context) { super(_props, context); // Tiles in miniMode are floating, and therefore not docked (0, _defineProperty2.default)(this, "contextMenuButton", /*#__PURE__*/(0, _react.createRef)()); (0, _defineProperty2.default)(this, "iframe", void 0); // ref to the iframe (callback style) (0, _defineProperty2.default)(this, "allowedWidgetsWatchRef", void 0); (0, _defineProperty2.default)(this, "persistKey", void 0); (0, _defineProperty2.default)(this, "sgWidget", void 0); (0, _defineProperty2.default)(this, "dispatcherRef", void 0); (0, _defineProperty2.default)(this, "unmounted", false); (0, _defineProperty2.default)(this, "watchUserReady", () => { if (_OwnProfileStore.OwnProfileStore.instance.isProfileInfoFetched) { return; } _OwnProfileStore.OwnProfileStore.instance.once(_AsyncStore.UPDATE_EVENT, this.onUserReady); }); (0, _defineProperty2.default)(this, "onUserReady", () => { this.setState({ isUserProfileReady: true }); }); // This is a function to make the impact of calling SettingsStore slightly less (0, _defineProperty2.default)(this, "hasPermissionToLoad", props => { if (this.usingLocalWidget()) return true; if (!props.room) return true; // user widgets always have permissions const opts = { approved: undefined }; _ModuleRunner.ModuleRunner.instance.invoke(_WidgetLifecycle.WidgetLifecycle.PreLoadRequest, opts, new _StopGapWidget.ElementWidget(this.props.app)); if (opts.approved) return true; const currentlyAllowedWidgets = _SettingsStore.default.getValue("allowedWidgets", props.room.roomId); const allowed = (0, _WidgetStore.isAppWidget)(props.app) && props.app.eventId !== undefined && (currentlyAllowedWidgets[props.app.eventId] ?? false); return allowed || props.userId === props.creatorUserId; }); (0, _defineProperty2.default)(this, "onMyMembership", (room, membership) => { if ((membership === _types.KnownMembership.Leave || membership === _types.KnownMembership.Ban) && room.roomId === this.props.room?.roomId) { this.onUserLeftRoom(); } }); (0, _defineProperty2.default)(this, "onAllowedWidgetsChange", () => { const hasPermissionToLoad = this.hasPermissionToLoad(this.props); if (this.state.hasPermissionToLoad && !hasPermissionToLoad) { // Force the widget to be non-persistent (able to be deleted/forgotten) _ActiveWidgetStore.default.instance.destroyPersistentWidget(this.props.app.id, (0, _WidgetStore.isAppWidget)(this.props.app) ? this.props.app.roomId : null); _PersistedElement.default.destroyElement(this.persistKey); this.sgWidget?.stopMessaging(); } this.setState({ hasPermissionToLoad }); }); (0, _defineProperty2.default)(this, "iframeRefChange", ref => { this.iframe = ref; if (this.unmounted) return; if (ref) { this.startMessaging(); } else { this.resetWidget(this.props); } }); (0, _defineProperty2.default)(this, "onWidgetReady", () => { this.setState({ loading: false }); }); (0, _defineProperty2.default)(this, "updateRequiresClient", () => { this.setState({ requiresClient: !!this.sgWidget?.widgetApi?.hasCapability(_ElementWidgetCapabilities.ElementWidgetCapabilities.RequiresClient) }); }); (0, _defineProperty2.default)(this, "onAction", payload => { switch (payload.action) { case "m.sticker": if (payload.widgetId === this.props.app.id && this.sgWidget?.widgetApi?.hasCapability(_matrixWidgetApi.MatrixCapabilities.StickerSending)) { _dispatcher.default.dispatch({ action: "post_sticker_message", data: _objectSpread(_objectSpread({}, payload.data), {}, { threadId: this.props.threadId }) }); _dispatcher.default.dispatch({ action: "stickerpicker_close" }); } else { _logger.logger.warn("Ignoring sticker message. Invalid capability"); } break; case _actions.Action.AfterLeaveRoom: if (payload.room_id === this.props.room?.roomId) { // call this before we get it echoed down /sync, so it doesn't hang around as long and look jarring this.onUserLeftRoom(); } break; } }); (0, _defineProperty2.default)(this, "grantWidgetPermission", () => { const roomId = this.props.room?.roomId; const eventId = (0, _WidgetStore.isAppWidget)(this.props.app) ? this.props.app.eventId : undefined; _logger.logger.info("Granting permission for widget to load: " + eventId); const current = _SettingsStore.default.getValue("allowedWidgets", roomId); if (eventId !== undefined) current[eventId] = true; const level = _SettingsStore.default.firstSupportedLevel("allowedWidgets"); _SettingsStore.default.setValue("allowedWidgets", roomId ?? null, level, current).then(() => { this.setState({ hasPermissionToLoad: true }); // Fetch a token for the integration manager, now that we're allowed to this.startWidget(); }).catch(err => { _logger.logger.error(err); // We don't really need to do anything about this - the user will just hit the button again. }); }); // TODO replace with full screen interactions (0, _defineProperty2.default)(this, "onPopoutWidgetClick", () => { // Ensure Jitsi conferences are closed on pop-out, to not confuse the user to join them // twice from the same computer, which Jitsi can have problems with (audio echo/gain-loop). if (_WidgetType.WidgetType.JITSI.matches(this.props.app.type)) { this.reload(); } // Using Object.assign workaround as the following opens in a new window instead of a new tab. // window.open(this._getPopoutUrl(), '_blank', 'noopener=yes'); Object.assign(document.createElement("a"), { target: "_blank", href: this.sgWidget?.popoutUrl, rel: "noreferrer noopener" }).click(); }); (0, _defineProperty2.default)(this, "onToggleMaximisedClick", () => { if (!this.props.room) return; // ignore action - it shouldn't even be visible const targetContainer = _WidgetLayoutStore.WidgetLayoutStore.instance.isInContainer(this.props.room, this.props.app, _WidgetLayoutStore.Container.Center) ? _WidgetLayoutStore.Container.Top : _WidgetLayoutStore.Container.Center; _WidgetLayoutStore.WidgetLayoutStore.instance.moveToContainer(this.props.room, this.props.app, targetContainer); // If the right panel has a timeline, but we're about to show the timeline in the main view, pop the right panel if (targetContainer === _WidgetLayoutStore.Container.Top && _RightPanelStore.default.instance.currentCardForRoom(this.props.room.roomId).phase === _RightPanelStorePhases.RightPanelPhases.Timeline) { _RightPanelStore.default.instance.popCard(this.props.room.roomId); } }); (0, _defineProperty2.default)(this, "onMinimiseClicked", () => { if (!this.props.room) return; // ignore action - it shouldn't even be visible _WidgetLayoutStore.WidgetLayoutStore.instance.moveToContainer(this.props.room, this.props.app, _WidgetLayoutStore.Container.Right); }); (0, _defineProperty2.default)(this, "onContextMenuClick", () => { this.setState({ menuDisplayed: true }); }); (0, _defineProperty2.default)(this, "closeContextMenu", () => { this.setState({ menuDisplayed: false }); }); if (!this.props.miniMode) { _ActiveWidgetStore.default.instance.dockWidget(this.props.app.id, (0, _WidgetStore.isAppWidget)(this.props.app) ? this.props.app.roomId : null); } // The key used for PersistedElement this.persistKey = (0, _PersistedElement.getPersistKey)(_WidgetUtils.default.getWidgetUid(this.props.app)); try { this.sgWidget = new _StopGapWidget.StopGapWidget(this.props); this.setupSgListeners(); } catch (e) { _logger.logger.log("Failed to construct widget", e); this.sgWidget = null; } this.state = this.getNewState(_props); } onUserLeftRoom() { const isActiveWidget = _ActiveWidgetStore.default.instance.getWidgetPersistence(this.props.app.id, (0, _WidgetStore.isAppWidget)(this.props.app) ? this.props.app.roomId : null); if (isActiveWidget) { // We just left the room that the active widget was from. if (this.props.room && _SDKContext.SdkContextClass.instance.roomViewStore.getRoomId() !== this.props.room.roomId) { // If we are not actively looking at the room then destroy this widget entirely. this.endWidgetActions(); } else if (_WidgetType.WidgetType.JITSI.matches(this.props.app.type)) { // If this was a Jitsi then reload to end call. this.reload(); } else { // Otherwise just cancel its persistence. _ActiveWidgetStore.default.instance.destroyPersistentWidget(this.props.app.id, (0, _WidgetStore.isAppWidget)(this.props.app) ? this.props.app.roomId : null); } } } determineInitialRequiresClientState() { try { const mockWidget = new _StopGapWidget.ElementWidget(this.props.app); const widgetApi = _WidgetMessagingStore.WidgetMessagingStore.instance.getMessaging(mockWidget, this.props.room?.roomId); if (widgetApi) { // Load value from existing API to prevent resetting the requiresClient value on layout changes. return widgetApi.hasCapability(_ElementWidgetCapabilities.ElementWidgetCapabilities.RequiresClient); } } catch { // fallback to true } // requiresClient is initially set to true. This avoids the broken state of the popout // button being visible (for an instance) and then disappearing when the widget is loaded. // requiresClient <-> hide the popout button return true; } /** * Set initial component state when the App wUrl (widget URL) is being updated. * Component props *must* be passed (rather than relying on this.props). * @param {Object} newProps The new properties of the component * @return {Object} Updated component state to be set with setState */ getNewState(newProps) { return { initialising: true, // True while we are mangling the widget URL // Don't show loading at all if the widget is ready once the IFrame is loaded (waitForIframeLoad = true). // We only need the loading screen if the widget sends a contentLoaded event (waitForIframeLoad = false). loading: !this.props.waitForIframeLoad && !_PersistedElement.default.isMounted(this.persistKey), // Assume that widget has permission to load if we are the user who // added it to the room, or if explicitly granted by the user hasPermissionToLoad: this.hasPermissionToLoad(newProps), isUserProfileReady: _OwnProfileStore.OwnProfileStore.instance.isProfileInfoFetched, error: null, menuDisplayed: false, requiresClient: this.determineInitialRequiresClientState(), hasContextMenuOptions: (0, _WidgetContextMenu.showContextMenu)(this.context, this.props.room, newProps.app, newProps.userWidget, !newProps.userWidget, newProps.onDeleteClick) }; } isMixedContent() { const parentContentProtocol = window.location.protocol; const u = (0, _UrlUtils.parseUrl)(this.props.app.url); const childContentProtocol = u.protocol; if (parentContentProtocol === "https:" && childContentProtocol !== "https:") { _logger.logger.warn("Refusing to load mixed-content app:", parentContentProtocol, childContentProtocol, window.location, this.props.app.url); return true; } return false; } componentDidMount() { // Only fetch IM token on mount if we're showing and have permission to load if (this.sgWidget && this.state.hasPermissionToLoad) { this.startWidget(); } this.watchUserReady(); if (this.props.room) { this.context.on(_matrix.RoomEvent.MyMembership, this.onMyMembership); } this.allowedWidgetsWatchRef = _SettingsStore.default.watchSetting("allowedWidgets", null, this.onAllowedWidgetsChange); // Widget action listeners this.dispatcherRef = _dispatcher.default.register(this.onAction); } componentWillUnmount() { this.unmounted = true; if (!this.props.miniMode) { _ActiveWidgetStore.default.instance.undockWidget(this.props.app.id, (0, _WidgetStore.isAppWidget)(this.props.app) ? this.props.app.roomId : null); } // Only tear down the widget if no other component is keeping it alive, // because we support moving widgets between containers, in which case // another component will keep it loaded throughout the transition if (!_ActiveWidgetStore.default.instance.isLive(this.props.app.id, (0, _WidgetStore.isAppWidget)(this.props.app) ? this.props.app.roomId : null)) { this.endWidgetActions(); } // Widget action listeners if (this.dispatcherRef) _dispatcher.default.unregister(this.dispatcherRef); if (this.props.room) { this.context.off(_matrix.RoomEvent.MyMembership, this.onMyMembership); } if (this.allowedWidgetsWatchRef) _SettingsStore.default.unwatchSetting(this.allowedWidgetsWatchRef); _OwnProfileStore.OwnProfileStore.instance.removeListener(_AsyncStore.UPDATE_EVENT, this.onUserReady); } setupSgListeners() { this.sgWidget?.on("ready", this.onWidgetReady); this.sgWidget?.on("error:preparing", this.updateRequiresClient); // emits when the capabilities have been set up or changed this.sgWidget?.on("capabilitiesNotified", this.updateRequiresClient); } stopSgListeners() { if (!this.sgWidget) return; this.sgWidget?.off("ready", this.onWidgetReady); this.sgWidget.off("error:preparing", this.updateRequiresClient); this.sgWidget.off("capabilitiesNotified", this.updateRequiresClient); } resetWidget(newProps) { this.sgWidget?.stopMessaging(); this.stopSgListeners(); try { this.sgWidget = new _StopGapWidget.StopGapWidget(newProps); this.setupSgListeners(); this.startWidget(); } catch (e) { _logger.logger.error("Failed to construct widget", e); this.sgWidget = null; } } startWidget() { this.sgWidget?.prepare().then(() => { if (this.unmounted) return; this.setState({ initialising: false }); }); } startMessaging() { try { this.sgWidget?.startMessaging(this.iframe); } catch (e) { _logger.logger.error("Failed to start widget", e); } } componentDidUpdate(prevProps) { if (prevProps.app.url !== this.props.app.url) { this.getNewState(this.props); if (this.state.hasPermissionToLoad) { this.resetWidget(this.props); } } } /** * Ends all widget interaction, such as cancelling calls and disabling webcams. * @private * @returns {Promise<*>} Resolves when the widget is terminated, or timeout passed. */ async endWidgetActions() { // widget migration dev note: async to maintain signature // HACK: This is a really dirty way to ensure that Jitsi cleans up // its hold on the webcam. Without this, the widget holds a media // stream open, even after death. See https://github.com/vector-im/element-web/issues/7351 if (this.iframe) { // In practice we could just do `+= ''` to trick the browser // into thinking the URL changed, however I can foresee this // being optimized out by a browser. Instead, we'll just point // the iframe at a page that is reasonably safe to use in the // event the iframe doesn't wink away. // This is relative to where the Element instance is located. this.iframe.src = "about:blank"; } if (_WidgetType.WidgetType.JITSI.matches(this.props.app.type) && this.props.room) { _LegacyCallHandler.default.instance.hangupCallApp(this.props.room.roomId); } // Delete the widget from the persisted store for good measure. _PersistedElement.default.destroyElement(this.persistKey); _ActiveWidgetStore.default.instance.destroyPersistentWidget(this.props.app.id, (0, _WidgetStore.isAppWidget)(this.props.app) ? this.props.app.roomId : null); this.sgWidget?.stopMessaging({ forceDestroy: true }); } formatAppTileName() { let appTileName = "No name"; if (this.props.app.name && this.props.app.name.trim()) { appTileName = this.props.app.name.trim(); } return appTileName; } /** * Whether we're using a local version of the widget rather than loading the * actual widget URL * @returns {bool} true If using a local version of the widget */ usingLocalWidget() { return _WidgetType.WidgetType.JITSI.matches(this.props.app.type); } getTileTitle() { const name = this.formatAppTileName(); const titleSpacer = /*#__PURE__*/_react.default.createElement("span", null, "\xA0-\xA0"); let title = ""; if (this.props.widgetPageTitle && this.props.widgetPageTitle !== this.formatAppTileName()) { title = this.props.widgetPageTitle; } return /*#__PURE__*/_react.default.createElement("span", null, /*#__PURE__*/_react.default.createElement(_WidgetAvatar.default, { app: this.props.app, size: "20px" }), /*#__PURE__*/_react.default.createElement("h3", null, name), /*#__PURE__*/_react.default.createElement("span", null, title ? titleSpacer : "", title)); } reload() { this.endWidgetActions().then(() => { // reset messaging this.resetWidget(this.props); this.startMessaging(); if (this.iframe && this.sgWidget) { // Reload iframe this.iframe.src = this.sgWidget.embedUrl; } }); } render() { let appTileBody; // Note that there is advice saying allow-scripts shouldn't be used with allow-same-origin // because that would allow the iframe to programmatically remove the sandbox attribute, but // this would only be for content hosted on the same origin as the element client: anything // hosted on the same origin as the client will get the same access as if you clicked // a link to it. const sandboxFlags = "allow-forms allow-popups allow-popups-to-escape-sandbox " + "allow-same-origin allow-scripts allow-presentation allow-downloads"; // Additional iframe feature permissions // (see - https://sites.google.com/a/chromium.org/dev/Home/chromium-security/deprecating-permissions-in-cross-origin-iframes and https://wicg.github.io/feature-policy/) const iframeFeatures = "microphone; camera; encrypted-media; autoplay; display-capture; clipboard-write; " + "clipboard-read;"; const appTileBodyClass = (0, _classnames.default)({ "mx_AppTileBody": true, "mx_AppTileBody--large": !this.props.miniMode, "mx_AppTileBody--mini": this.props.miniMode, "mx_AppTileBody--loading": this.state.loading, // We don't want mx_AppTileBody (rounded corners) for call widgets "mx_AppTileBody--call": this.props.app.type === _WidgetType.WidgetType.CALL.preferred }); const appTileBodyStyles = {}; if (this.props.pointerEvents) { appTileBodyStyles.pointerEvents = this.props.pointerEvents; } const loadingElement = /*#__PURE__*/_react.default.createElement("div", { className: "mx_AppTileBody_fadeInSpinner" }, /*#__PURE__*/_react.default.createElement(_Spinner.default, { message: (0, _languageHandler._t)("common|loading") })); const widgetTitle = _WidgetUtils.default.getWidgetName(this.props.app); if (this.sgWidget === null) { appTileBody = /*#__PURE__*/_react.default.createElement("div", { className: appTileBodyClass, style: appTileBodyStyles }, /*#__PURE__*/_react.default.createElement(_AppWarning.default, { errorMsg: (0, _languageHandler._t)("widget|error_loading") })); } else if (!this.state.hasPermissionToLoad && this.props.room) { // only possible for room widgets, can assert this.props.room here const isEncrypted = this.context.isRoomEncrypted(this.props.room.roomId); appTileBody = /*#__PURE__*/_react.default.createElement("div", { className: appTileBodyClass, style: appTileBodyStyles }, /*#__PURE__*/_react.default.createElement(_AppPermission.default, { roomId: this.props.room.roomId, creatorUserId: this.props.creatorUserId, url: this.sgWidget.embedUrl, isRoomEncrypted: isEncrypted, onPermissionGranted: this.grantWidgetPermission })); } else if (this.state.initialising || !this.state.isUserProfileReady) { appTileBody = /*#__PURE__*/_react.default.createElement("div", { className: appTileBodyClass, style: appTileBodyStyles }, loadingElement); } else { if (this.isMixedContent()) { appTileBody = /*#__PURE__*/_react.default.createElement("div", { className: appTileBodyClass, style: appTileBodyStyles }, /*#__PURE__*/_react.default.createElement(_AppWarning.default, { errorMsg: (0, _languageHandler._t)("widget|error_mixed_content") })); } else { appTileBody = /*#__PURE__*/_react.default.createElement(_react.default.Fragment, null, /*#__PURE__*/_react.default.createElement("div", { className: appTileBodyClass, style: appTileBodyStyles }, this.state.loading && loadingElement, /*#__PURE__*/_react.default.createElement("iframe", { title: widgetTitle, allow: iframeFeatures, ref: this.iframeRefChange, src: this.sgWidget.embedUrl, allowFullScreen: true, sandbox: sandboxFlags })), this.props.overlay); if (!this.props.userWidget) { // All room widgets can theoretically be allowed to remain on screen, so we // wrap them all in a PersistedElement from the get-go. If we wait, the iframe // will be re-mounted later, which means the widget has to start over, which is // bad. // Also wrap the PersistedElement in a div to fix the height, otherwise // AppTile's border is in the wrong place // For persisted apps in PiP we want the zIndex to be higher then for other persisted apps (100) // otherwise there are issues that the PiP view is drawn UNDER another widget (Persistent app) when dragged around. const zIndexAboveOtherPersistentElements = 101; appTileBody = /*#__PURE__*/_react.default.createElement("div", { className: "mx_AppTile_persistedWrapper" }, /*#__PURE__*/_react.default.createElement(_PersistedElement.default, { zIndex: this.props.miniMode ? zIndexAboveOtherPersistentElements : 9, persistKey: this.persistKey, moveRef: this.props.movePersistedElement }, appTileBody)); } } } let appTileClasses; if (this.props.miniMode) { appTileClasses = { mx_AppTile_mini: true }; } else if (this.props.fullWidth) { appTileClasses = { mx_AppTileFullWidth: true }; } else { appTileClasses = { mx_AppTile: true }; } appTileClasses = (0, _classnames.default)(appTileClasses); let contextMenu; if (this.state.menuDisplayed) { contextMenu = /*#__PURE__*/_react.default.createElement(_WidgetContextMenu.WidgetContextMenu, (0, _extends2.default)({}, (0, _ContextMenu.aboveLeftOf)(this.contextMenuButton.current.getBoundingClientRect()), { app: this.props.app, onFinished: this.closeContextMenu, showUnpin: !this.props.userWidget, userWidget: this.props.userWidget, onEditClick: this.props.onEditClick, onDeleteClick: this.props.onDeleteClick })); } const layoutButtons = []; if (this.props.showLayoutButtons) { const isMaximised = this.props.room && _WidgetLayoutStore.WidgetLayoutStore.instance.isInContainer(this.props.room, this.props.app, _WidgetLayoutStore.Container.Center); layoutButtons.push( /*#__PURE__*/_react.default.createElement(_AccessibleButton.default, { key: "toggleMaximised", className: "mx_AppTileMenuBar_widgets_button", title: isMaximised ? (0, _languageHandler._t)("widget|unmaximise") : (0, _languageHandler._t)("action|maximise"), onClick: this.onToggleMaximisedClick }, isMaximised ? /*#__PURE__*/_react.default.createElement(_minimiseCollapse.Icon, { className: "mx_Icon mx_Icon_12" }) : /*#__PURE__*/_react.default.createElement(_maximiseExpand.Icon, { className: "mx_Icon mx_Icon_12" }))); layoutButtons.push( /*#__PURE__*/_react.default.createElement(_AccessibleButton.default, { key: "minimise", className: "mx_AppTileMenuBar_widgets_button", title: (0, _languageHandler._t)("action|minimise"), onClick: this.onMinimiseClicked }, /*#__PURE__*/_react.default.createElement(_minusButton.Icon, { className: "mx_Icon mx_Icon_12" }))); } return /*#__PURE__*/_react.default.createElement(_react.default.Fragment, null, /*#__PURE__*/_react.default.createElement("div", { className: appTileClasses, id: this.props.app.id }, this.props.showMenubar && /*#__PURE__*/_react.default.createElement("div", { className: "mx_AppTileMenuBar" }, /*#__PURE__*/_react.default.createElement("span", { className: "mx_AppTileMenuBar_title", style: { pointerEvents: this.props.handleMinimisePointerEvents ? "all" : "none" } }, this.props.showTitle && this.getTileTitle()), /*#__PURE__*/_react.default.createElement("span", { className: "mx_AppTileMenuBar_widgets" }, layoutButtons, this.props.showPopout && !this.state.requiresClient && /*#__PURE__*/_react.default.createElement(_AccessibleButton.default, { className: "mx_AppTileMenuBar_widgets_button", title: (0, _languageHandler._t)("widget|popout"), onClick: this.onPopoutWidgetClick }, /*#__PURE__*/_react.default.createElement(_externalLink.Icon, { className: "mx_Icon mx_Icon_12 mx_Icon--stroke" })), this.state.hasContextMenuOptions && /*#__PURE__*/_react.default.createElement(_ContextMenu.ContextMenuButton, { className: "mx_AppTileMenuBar_widgets_button", label: (0, _languageHandler._t)("common|options"), isExpanded: this.state.menuDisplayed, ref: this.contextMenuButton, onClick: this.onContextMenuClick }, /*#__PURE__*/_react.default.createElement(_overflowHorizontal.default, { className: "mx_Icon mx_Icon_12" })))), appTileBody), contextMenu); } } exports.default = AppTile; (0, _defineProperty2.default)(AppTile, "contextType", _MatrixClientContext.default); (0, _defineProperty2.default)(AppTile, "defaultProps", { waitForIframeLoad: true, showMenubar: true, showTitle: true, showPopout: true, handleMinimisePointerEvents: false, userWidget: false, miniMode: false, threadId: null, showLayoutButtons: true }); //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireWildcard","require","_classnames","_interopRequireDefault","_matrixWidgetApi","_matrix","_types","_logger","_WidgetLifecycle","_overflowHorizontal","_AccessibleButton","_languageHandler","_AppPermission","_AppWarning","_Spinner","_dispatcher","_ActiveWidgetStore","_SettingsStore","_ContextMenu","_PersistedElement","_WidgetType","_StopGapWidget","_WidgetContextMenu","_WidgetAvatar","_LegacyCallHandler","_WidgetStore","_minimiseCollapse","_maximiseExpand","_minusButton","_externalLink","_WidgetLayoutStore","_OwnProfileStore","_AsyncStore","_WidgetUtils","_MatrixClientContext","_actions","_ElementWidgetCapabilities","_WidgetMessagingStore","_SDKContext","_ModuleRunner","_UrlUtils","_RightPanelStore","_RightPanelStorePhases","_getRequireWildcardCache","e","WeakMap","r","t","__esModule","default","has","get","n","__proto__","a","Object","defineProperty","getOwnPropertyDescriptor","u","hasOwnProperty","call","i","set","ownKeys","keys","getOwnPropertySymbols","o","filter","enumerable","push","apply","_objectSpread","arguments","length","forEach","_defineProperty2","getOwnPropertyDescriptors","defineProperties","AppTile","React","Component","constructor","props","context","createRef","OwnProfileStore","instance","isProfileInfoFetched","once","UPDATE_EVENT","onUserReady","setState","isUserProfileReady","usingLocalWidget","room","opts","approved","undefined","ModuleRunner","invoke","WidgetLifecycle","PreLoadRequest","ElementWidget","app","currentlyAllowedWidgets","SettingsStore","getValue","roomId","allowed","isAppWidget","eventId","userId","creatorUserId","membership","KnownMembership","Leave","Ban","onUserLeftRoom","hasPermissionToLoad","state","ActiveWidgetStore","destroyPersistentWidget","id","PersistedElement","destroyElement","persistKey","sgWidget","stopMessaging","ref","iframe","unmounted","startMessaging","resetWidget","loading","requiresClient","widgetApi","hasCapability","ElementWidgetCapabilities","RequiresClient","payload","action","widgetId","MatrixCapabilities","StickerSending","dis","dispatch","data","threadId","logger","warn","Action","AfterLeaveRoom","room_id","info","current","level","firstSupportedLevel","setValue","then","startWidget","catch","err","error","WidgetType","JITSI","matches","type","reload","assign","document","createElement","target","href","popoutUrl","rel","click","targetContainer","WidgetLayoutStore","isInContainer","Container","Center","Top","moveToContainer","RightPanelStore","currentCardForRoom","phase","RightPanelPhases","Timeline","popCard","Right","menuDisplayed","miniMode","dockWidget","getPersistKey","WidgetUtils","getWidgetUid","StopGapWidget","setupSgListeners","log","getNewState","isActiveWidget","getWidgetPersistence","SdkContextClass","roomViewStore","getRoomId","endWidgetActions","determineInitialRequiresClientState","mockWidget","WidgetMessagingStore","getMessaging","newProps","initialising","waitForIframeLoad","isMounted","hasContextMenuOptions","showContextMenu","userWidget","onDeleteClick","isMixedContent","parentContentProtocol","window","location","protocol","parseUrl","url","childContentProtocol","componentDidMount","watchUserReady","on","RoomEvent","MyMembership","onMyMembership","allowedWidgetsWatchRef","watchSetting","onAllowedWidgetsChange","dispatcherRef","register","onAction","componentWillUnmount","undockWidget","isLive","unregister","off","unwatchSetting","removeListener","onWidgetReady","updateRequiresClient","stopSgListeners","prepare","componentDidUpdate","prevProps","src","LegacyCallHandler","hangupCallApp","forceDestroy","formatAppTileName","appTileName","name","trim","getTileTitle","titleSpacer","title","widgetPageTitle","size","embedUrl","render","appTileBody","sandboxFlags","iframeFeatures","appTileBodyClass","classNames","CALL","preferred","appTileBodyStyles","pointerEvents","loadingElement","className","message","_t","widgetTitle","getWidgetName","style","errorMsg","isEncrypted","isRoomEncrypted","onPermissionGranted","grantWidgetPermission","Fragment","allow","iframeRefChange","allowFullScreen","sandbox","overlay","zIndexAboveOtherPersistentElements","zIndex","moveRef","movePersistedElement","appTileClasses","mx_AppTile_mini","fullWidth","mx_AppTileFullWidth","mx_AppTile","contextMenu","WidgetContextMenu","_extends2","aboveLeftOf","contextMenuButton","getBoundingClientRect","onFinished","closeContextMenu","showUnpin","onEditClick","layoutButtons","showLayoutButtons","isMaximised","key","onClick","onToggleMaximisedClick","Icon","onMinimiseClicked","showMenubar","handleMinimisePointerEvents","showTitle","showPopout","onPopoutWidgetClick","ContextMenuButton","label","isExpanded","onContextMenuClick","exports","MatrixClientContext"],"sources":["../../../../src/components/views/elements/AppTile.tsx"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2020-2022 The Matrix.org Foundation C.I.C.\nCopyright 2019 Michael Telatynski <7t3chguy@gmail.com>\nCopyright 2018 New Vector Ltd\nCopyright 2017 Vector Creations Ltd\n\nSPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only\nPlease see LICENSE files in the repository root for full details.\n*/\n\nimport React, { ContextType, createRef, CSSProperties, MutableRefObject, ReactNode } from \"react\";\nimport classNames from \"classnames\";\nimport { IWidget, MatrixCapabilities } from \"matrix-widget-api\";\nimport { Room, RoomEvent } from \"matrix-js-sdk/src/matrix\";\nimport { KnownMembership } from \"matrix-js-sdk/src/types\";\nimport { logger } from \"matrix-js-sdk/src/logger\";\nimport { ApprovalOpts, WidgetLifecycle } from \"@matrix-org/react-sdk-module-api/lib/lifecycles/WidgetLifecycle\";\nimport EllipsisIcon from \"@vector-im/compound-design-tokens/assets/web/icons/overflow-horizontal\";\n\nimport AccessibleButton from \"./AccessibleButton\";\nimport { _t } from \"../../../languageHandler\";\nimport AppPermission from \"./AppPermission\";\nimport AppWarning from \"./AppWarning\";\nimport Spinner from \"./Spinner\";\nimport dis from \"../../../dispatcher/dispatcher\";\nimport ActiveWidgetStore from \"../../../stores/ActiveWidgetStore\";\nimport SettingsStore from \"../../../settings/SettingsStore\";\nimport { aboveLeftOf, ContextMenuButton } from \"../../structures/ContextMenu\";\nimport PersistedElement, { getPersistKey } from \"./PersistedElement\";\nimport { WidgetType } from \"../../../widgets/WidgetType\";\nimport { ElementWidget, StopGapWidget } from \"../../../stores/widgets/StopGapWidget\";\nimport { showContextMenu, WidgetContextMenu } from \"../context_menus/WidgetContextMenu\";\nimport WidgetAvatar from \"../avatars/WidgetAvatar\";\nimport LegacyCallHandler from \"../../../LegacyCallHandler\";\nimport { IApp, isAppWidget } from \"../../../stores/WidgetStore\";\nimport { Icon as CollapseIcon } from \"../../../../res/img/element-icons/minimise-collapse.svg\";\nimport { Icon as MaximiseIcon } from \"../../../../res/img/element-icons/maximise-expand.svg\";\nimport { Icon as MinimiseIcon } from \"../../../../res/img/element-icons/minus-button.svg\";\nimport { Icon as PopoutIcon } from \"../../../../res/img/feather-customised/widget/external-link.svg\";\nimport { Container, WidgetLayoutStore } from \"../../../stores/widgets/WidgetLayoutStore\";\nimport { OwnProfileStore } from \"../../../stores/OwnProfileStore\";\nimport { UPDATE_EVENT } from \"../../../stores/AsyncStore\";\nimport WidgetUtils from \"../../../utils/WidgetUtils\";\nimport MatrixClientContext from \"../../../contexts/MatrixClientContext\";\nimport { ActionPayload } from \"../../../dispatcher/payloads\";\nimport { Action } from \"../../../dispatcher/actions\";\nimport { ElementWidgetCapabilities } from \"../../../stores/widgets/ElementWidgetCapabilities\";\nimport { WidgetMessagingStore } from \"../../../stores/widgets/WidgetMessagingStore\";\nimport { SdkContextClass } from \"../../../contexts/SDKContext\";\nimport { ModuleRunner } from \"../../../modules/ModuleRunner\";\nimport { parseUrl } from \"../../../utils/UrlUtils\";\nimport RightPanelStore from \"../../../stores/right-panel/RightPanelStore.ts\";\nimport { RightPanelPhases } from \"../../../stores/right-panel/RightPanelStorePhases.ts\";\n\ninterface IProps {\n    app: IWidget | IApp;\n    // If room is not specified then it is an account level widget\n    // which bypasses permission prompts as it was added explicitly by that user\n    room?: Room;\n    threadId?: string | null;\n    // Specifying 'fullWidth' as true will render the app tile to fill the width of the app drawer container.\n    // This should be set to true when there is only one widget in the app drawer, otherwise it should be false.\n    fullWidth?: boolean;\n    // Optional. If set, renders a smaller view of the widget\n    miniMode?: boolean;\n    // UserId of the current user\n    userId: string;\n    // UserId of the entity that added / modified the widget\n    creatorUserId: string;\n    waitForIframeLoad: boolean;\n    showMenubar?: boolean;\n    // Optional onEditClickHandler (overrides default behaviour)\n    onEditClick?: () => void;\n    // Optional onDeleteClickHandler (overrides default behaviour)\n    onDeleteClick?: () => void;\n    // Optionally hide the tile title\n    showTitle?: boolean;\n    // Optionally handle minimise button pointer events (default false)\n    handleMinimisePointerEvents?: boolean;\n    // Optionally hide the popout widget icon\n    showPopout?: boolean;\n    // Is this an instance of a user widget\n    userWidget: boolean;\n    // sets the pointer-events property on the iframe\n    pointerEvents?: CSSProperties[\"pointerEvents\"];\n    widgetPageTitle?: string;\n    showLayoutButtons?: boolean;\n    // Handle to manually notify the PersistedElement that it needs to move\n    movePersistedElement?: MutableRefObject<(() => void) | undefined>;\n    // An element to render after the iframe as an overlay\n    overlay?: ReactNode;\n    // If defined this async method will be called when the widget requests to become sticky.\n    // It will only become sticky once the returned promise resolves.\n    // This is useful because: Widget B is sticky. Making widget A sticky will kill widget B immediately.\n    // This promise allows to do Widget B related cleanup before Widget A becomes sticky. (e.g. hangup a Voip call)\n    stickyPromise?: () => Promise<void>;\n}\n\ninterface IState {\n    initialising: boolean; // True while we are mangling the widget URL\n    // True while the iframe content is loading\n    loading: boolean;\n    // Assume that widget has permission to load if we are the user who\n    // added it to the room, or if explicitly granted by the user\n    hasPermissionToLoad: boolean;\n    // Wait for user profile load to display correct name\n    isUserProfileReady: boolean;\n    error: Error | null;\n    menuDisplayed: boolean;\n    requiresClient: boolean;\n    hasContextMenuOptions: boolean;\n}\n\nexport default class AppTile extends React.Component<IProps, IState> {\n    public static contextType = MatrixClientContext;\n    public declare context: ContextType<typeof MatrixClientContext>;\n\n    public static defaultProps: Partial<IProps> = {\n        waitForIframeLoad: true,\n        showMenubar: true,\n        showTitle: true,\n        showPopout: true,\n        handleMinimisePointerEvents: false,\n        userWidget: false,\n        miniMode: false,\n        threadId: null,\n        showLayoutButtons: true,\n    };\n\n    private contextMenuButton = createRef<any>();\n    private iframe?: HTMLIFrameElement; // ref to the iframe (callback style)\n    private allowedWidgetsWatchRef?: string;\n    private persistKey: string;\n    private sgWidget: StopGapWidget | null;\n    private dispatcherRef?: string;\n    private unmounted = false;\n\n    public constructor(props: IProps, context: ContextType<typeof MatrixClientContext>) {\n        super(props, context);\n\n        // Tiles in miniMode are floating, and therefore not docked\n        if (!this.props.miniMode) {\n            ActiveWidgetStore.instance.dockWidget(\n                this.props.app.id,\n                isAppWidget(this.props.app) ? this.props.app.roomId : null,\n            );\n        }\n\n        // The key used for PersistedElement\n        this.persistKey = getPersistKey(WidgetUtils.getWidgetUid(this.props.app));\n        try {\n            this.sgWidget = new StopGapWidget(this.props);\n            this.setupSgListeners();\n        } catch (e) {\n            logger.log(\"Failed to construct widget\", e);\n            this.sgWidget = null;\n        }\n\n        this.state = this.getNewState(props);\n    }\n\n    private watchUserReady = (): void => {\n        if (OwnProfileStore.instance.isProfileInfoFetched) {\n            return;\n        }\n        OwnProfileStore.instance.once(UPDATE_EVENT, this.onUserReady);\n    };\n\n    private onUserReady = (): void => {\n        this.setState({ isUserProfileReady: true });\n    };\n\n    // T