matrix-react-sdk
Version:
SDK for matrix.org using React
641 lines (624 loc) • 120 kB
JavaScript
"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