matrix-react-sdk
Version:
SDK for matrix.org using React
262 lines (190 loc) • 28.6 kB
JavaScript
;
var _interopRequireWildcard = require("@babel/runtime/helpers/interopRequireWildcard");
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 _objectWithoutProperties2 = _interopRequireDefault(require("@babel/runtime/helpers/objectWithoutProperties"));
var _react = _interopRequireWildcard(require("react"));
var _matrixWidgetApi = require("matrix-widget-api");
var _IconizedContextMenu = _interopRequireWildcard(require("./IconizedContextMenu"));
var _ContextMenu = require("../../structures/ContextMenu");
var _languageHandler = require("../../../languageHandler");
var _WidgetUtils = _interopRequireDefault(require("../../../utils/WidgetUtils"));
var _WidgetMessagingStore = require("../../../stores/widgets/WidgetMessagingStore");
var _RoomContext = _interopRequireDefault(require("../../../contexts/RoomContext"));
var _dispatcher = _interopRequireDefault(require("../../../dispatcher/dispatcher"));
var _SettingsStore = _interopRequireDefault(require("../../../settings/SettingsStore"));
var _Modal = _interopRequireDefault(require("../../../Modal"));
var _QuestionDialog = _interopRequireDefault(require("../dialogs/QuestionDialog"));
var _ErrorDialog = _interopRequireDefault(require("../dialogs/ErrorDialog"));
var _WidgetType = require("../../../widgets/WidgetType");
var _MatrixClientContext = _interopRequireDefault(require("../../../contexts/MatrixClientContext"));
var _WidgetLayoutStore = require("../../../stores/widgets/WidgetLayoutStore");
var _Livestream = require("../../../Livestream");
/*
Copyright 2020 The Matrix.org Foundation C.I.C.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
const WidgetContextMenu
/*: React.FC<IProps>*/
= (_ref) => {
let {
onFinished,
app,
userWidget,
onDeleteClick,
showUnpin
} = _ref,
props = (0, _objectWithoutProperties2.default)(_ref, ["onFinished", "app", "userWidget", "onDeleteClick", "showUnpin"]);
const cli = (0, _react.useContext)(_MatrixClientContext.default);
const {
room,
roomId
} = (0, _react.useContext)(_RoomContext.default);
const widgetMessaging = _WidgetMessagingStore.WidgetMessagingStore.instance.getMessagingForId(app.id);
const canModify = userWidget || _WidgetUtils.default.canUserModifyWidgets(roomId);
let streamAudioStreamButton;
if ((0, _Livestream.getConfigLivestreamUrl)() && _WidgetType.WidgetType.JITSI.matches(app.type)) {
const onStreamAudioClick = async () => {
try {
await (0, _Livestream.startJitsiAudioLivestream)(widgetMessaging, roomId);
} catch (err) {
console.error("Failed to start livestream", err); // XXX: won't i18n well, but looks like widget api only support 'message'?
const message = err.message || (0, _languageHandler._t)("Unable to start audio streaming.");
_Modal.default.createTrackedDialog('WidgetContext Menu', 'Livestream failed', _ErrorDialog.default, {
title: (0, _languageHandler._t)('Failed to start livestream'),
description: message
});
}
onFinished();
};
streamAudioStreamButton = /*#__PURE__*/_react.default.createElement(_IconizedContextMenu.IconizedContextMenuOption, {
onClick: onStreamAudioClick,
label: (0, _languageHandler._t)("Start audio stream")
});
}
let unpinButton;
if (showUnpin) {
const onUnpinClick = () => {
_WidgetLayoutStore.WidgetLayoutStore.instance.moveToContainer(room, app, _WidgetLayoutStore.Container.Right);
onFinished();
};
unpinButton = /*#__PURE__*/_react.default.createElement(_IconizedContextMenu.IconizedContextMenuOption, {
onClick: onUnpinClick,
label: (0, _languageHandler._t)("Unpin")
});
}
let editButton;
if (canModify && _WidgetUtils.default.isManagedByManager(app)) {
const onEditClick = () => {
_WidgetUtils.default.editWidget(room, app);
onFinished();
};
editButton = /*#__PURE__*/_react.default.createElement(_IconizedContextMenu.IconizedContextMenuOption, {
onClick: onEditClick,
label: (0, _languageHandler._t)("Edit")
});
}
let snapshotButton;
if (widgetMessaging?.hasCapability(_matrixWidgetApi.MatrixCapabilities.Screenshots)) {
const onSnapshotClick = () => {
widgetMessaging?.takeScreenshot().then(data => {
_dispatcher.default.dispatch({
action: 'picture_snapshot',
file: data.screenshot
});
}).catch(err => {
console.error("Failed to take screenshot: ", err);
});
onFinished();
};
snapshotButton = /*#__PURE__*/_react.default.createElement(_IconizedContextMenu.IconizedContextMenuOption, {
onClick: onSnapshotClick,
label: (0, _languageHandler._t)("Take a picture")
});
}
let deleteButton;
if (onDeleteClick || canModify) {
const onDeleteClickDefault = () => {
// Show delete confirmation dialog
_Modal.default.createTrackedDialog('Delete Widget', '', _QuestionDialog.default, {
title: (0, _languageHandler._t)("Delete Widget"),
description: (0, _languageHandler._t)("Deleting a widget removes it for all users in this room." + " Are you sure you want to delete this widget?"),
button: (0, _languageHandler._t)("Delete widget"),
onFinished: confirmed => {
if (!confirmed) return;
_WidgetUtils.default.setRoomWidget(roomId, app.id);
}
});
onFinished();
};
deleteButton = /*#__PURE__*/_react.default.createElement(_IconizedContextMenu.IconizedContextMenuOption, {
onClick: onDeleteClick || onDeleteClickDefault,
label: userWidget ? (0, _languageHandler._t)("Remove") : (0, _languageHandler._t)("Remove for everyone")
});
}
let isAllowedWidget = _SettingsStore.default.getValue("allowedWidgets", roomId)[app.eventId];
if (isAllowedWidget === undefined) {
isAllowedWidget = app.creatorUserId === cli.getUserId();
}
const isLocalWidget = _WidgetType.WidgetType.JITSI.matches(app.type);
let revokeButton;
if (!userWidget && !isLocalWidget && isAllowedWidget) {
const onRevokeClick = () => {
console.info("Revoking permission for widget to load: " + app.eventId);
const current = _SettingsStore.default.getValue("allowedWidgets", roomId);
current[app.eventId] = false;
const level = _SettingsStore.default.firstSupportedLevel("allowedWidgets");
_SettingsStore.default.setValue("allowedWidgets", roomId, level, current).catch(err => {
console.error(err); // We don't really need to do anything about this - the user will just hit the button again.
});
onFinished();
};
revokeButton = /*#__PURE__*/_react.default.createElement(_IconizedContextMenu.IconizedContextMenuOption, {
onClick: onRevokeClick,
label: (0, _languageHandler._t)("Revoke permissions")
});
}
const pinnedWidgets = _WidgetLayoutStore.WidgetLayoutStore.instance.getContainerWidgets(room, _WidgetLayoutStore.Container.Top);
const widgetIndex = pinnedWidgets.findIndex(widget => widget.id === app.id);
let moveLeftButton;
if (showUnpin && widgetIndex > 0) {
const onClick = () => {
_WidgetLayoutStore.WidgetLayoutStore.instance.moveWithinContainer(room, _WidgetLayoutStore.Container.Top, app, -1);
onFinished();
};
moveLeftButton = /*#__PURE__*/_react.default.createElement(_IconizedContextMenu.IconizedContextMenuOption, {
onClick: onClick,
label: (0, _languageHandler._t)("Move left")
});
}
let moveRightButton;
if (showUnpin && widgetIndex < pinnedWidgets.length - 1) {
const onClick = () => {
_WidgetLayoutStore.WidgetLayoutStore.instance.moveWithinContainer(room, _WidgetLayoutStore.Container.Top, app, 1);
onFinished();
};
moveRightButton = /*#__PURE__*/_react.default.createElement(_IconizedContextMenu.IconizedContextMenuOption, {
onClick: onClick,
label: (0, _languageHandler._t)("Move right")
});
}
return /*#__PURE__*/_react.default.createElement(_IconizedContextMenu.default, (0, _extends2.default)({}, props, {
chevronFace: _ContextMenu.ChevronFace.None,
onFinished: onFinished
}), /*#__PURE__*/_react.default.createElement(_IconizedContextMenu.IconizedContextMenuOptionList, null, streamAudioStreamButton, editButton, revokeButton, deleteButton, snapshotButton, moveLeftButton, moveRightButton, unpinButton));
};
var _default = WidgetContextMenu;
exports.default = _default;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/views/context_menus/WidgetContextMenu.tsx"],"names":["WidgetContextMenu","onFinished","app","userWidget","onDeleteClick","showUnpin","props","cli","MatrixClientContext","room","roomId","RoomContext","widgetMessaging","WidgetMessagingStore","instance","getMessagingForId","id","canModify","WidgetUtils","canUserModifyWidgets","streamAudioStreamButton","WidgetType","JITSI","matches","type","onStreamAudioClick","err","console","error","message","Modal","createTrackedDialog","ErrorDialog","title","description","unpinButton","onUnpinClick","WidgetLayoutStore","moveToContainer","Container","Right","editButton","isManagedByManager","onEditClick","editWidget","snapshotButton","hasCapability","MatrixCapabilities","Screenshots","onSnapshotClick","takeScreenshot","then","data","dis","dispatch","action","file","screenshot","catch","deleteButton","onDeleteClickDefault","QuestionDialog","button","confirmed","setRoomWidget","isAllowedWidget","SettingsStore","getValue","eventId","undefined","creatorUserId","getUserId","isLocalWidget","revokeButton","onRevokeClick","info","current","level","firstSupportedLevel","setValue","pinnedWidgets","getContainerWidgets","Top","widgetIndex","findIndex","widget","moveLeftButton","onClick","moveWithinContainer","moveRightButton","length","ChevronFace","None"],"mappings":"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","sourcesContent":["/*\nCopyright 2020 The Matrix.org Foundation C.I.C.\n\nLicensed under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License.\nYou may obtain a copy of the License at\n\n    http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software\ndistributed under the License is distributed on an \"AS IS\" BASIS,\nWITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\nSee the License for the specific language governing permissions and\nlimitations under the License.\n*/\n\nimport React, {useContext} from \"react\";\nimport {MatrixCapabilities} from \"matrix-widget-api\";\n\nimport IconizedContextMenu, {IconizedContextMenuOption, IconizedContextMenuOptionList} from \"./IconizedContextMenu\";\nimport {ChevronFace} from \"../../structures/ContextMenu\";\nimport {_t} from \"../../../languageHandler\";\nimport {IApp} from \"../../../stores/WidgetStore\";\nimport WidgetUtils from \"../../../utils/WidgetUtils\";\nimport {WidgetMessagingStore} from \"../../../stores/widgets/WidgetMessagingStore\";\nimport RoomContext from \"../../../contexts/RoomContext\";\nimport dis from \"../../../dispatcher/dispatcher\";\nimport SettingsStore from \"../../../settings/SettingsStore\";\nimport Modal from \"../../../Modal\";\nimport QuestionDialog from \"../dialogs/QuestionDialog\";\nimport ErrorDialog from \"../dialogs/ErrorDialog\";\nimport {WidgetType} from \"../../../widgets/WidgetType\";\nimport MatrixClientContext from \"../../../contexts/MatrixClientContext\";\nimport { Container, WidgetLayoutStore } from \"../../../stores/widgets/WidgetLayoutStore\";\nimport { getConfigLivestreamUrl, startJitsiAudioLivestream } from \"../../../Livestream\";\n\ninterface IProps extends React.ComponentProps<typeof IconizedContextMenu> {\n    app: IApp;\n    userWidget?: boolean;\n    showUnpin?: boolean;\n    // override delete handler\n    onDeleteClick?(): void;\n}\n\nconst WidgetContextMenu: React.FC<IProps> = ({\n    onFinished,\n    app,\n    userWidget,\n    onDeleteClick,\n    showUnpin,\n    ...props\n}) => {\n    const cli = useContext(MatrixClientContext);\n    const {room, roomId} = useContext(RoomContext);\n\n    const widgetMessaging = WidgetMessagingStore.instance.getMessagingForId(app.id);\n    const canModify = userWidget || WidgetUtils.canUserModifyWidgets(roomId);\n\n    let streamAudioStreamButton;\n    if (getConfigLivestreamUrl() && WidgetType.JITSI.matches(app.type)) {\n        const onStreamAudioClick = async () => {\n            try {\n                await startJitsiAudioLivestream(widgetMessaging, roomId);\n            } catch (err) {\n                console.error(\"Failed to start livestream\", err);\n                // XXX: won't i18n well, but looks like widget api only support 'message'?\n                const message = err.message || _t(\"Unable to start audio streaming.\");\n                Modal.createTrackedDialog('WidgetContext Menu', 'Livestream failed', ErrorDialog, {\n                    title: _t('Failed to start livestream'),\n                    description: message,\n                });\n            }\n            onFinished();\n        };\n        streamAudioStreamButton = <IconizedContextMenuOption\n            onClick={onStreamAudioClick} label={_t(\"Start audio stream\")}\n        />;\n    }\n\n    let unpinButton;\n    if (showUnpin) {\n        const onUnpinClick = () => {\n            WidgetLayoutStore.instance.moveToContainer(room, app, Container.Right);\n            onFinished();\n        };\n\n        unpinButton = <IconizedContextMenuOption onClick={onUnpinClick} label={_t(\"Unpin\")} />;\n    }\n\n    let editButton;\n    if (canModify && WidgetUtils.isManagedByManager(app)) {\n        const onEditClick = () => {\n            WidgetUtils.editWidget(room, app);\n            onFinished();\n        };\n\n        editButton = <IconizedContextMenuOption onClick={onEditClick} label={_t(\"Edit\")} />;\n    }\n\n    let snapshotButton;\n    if (widgetMessaging?.hasCapability(MatrixCapabilities.Screenshots)) {\n        const onSnapshotClick = () => {\n            widgetMessaging?.takeScreenshot().then(data => {\n                dis.dispatch({\n                    action: 'picture_snapshot',\n                    file: data.screenshot,\n                });\n            }).catch(err => {\n                console.error(\"Failed to take screenshot: \", err);\n            });\n            onFinished();\n        };\n\n        snapshotButton = <IconizedContextMenuOption onClick={onSnapshotClick} label={_t(\"Take a picture\")} />;\n    }\n\n    let deleteButton;\n    if (onDeleteClick || canModify) {\n        const onDeleteClickDefault = () => {\n            // Show delete confirmation dialog\n            Modal.createTrackedDialog('Delete Widget', '', QuestionDialog, {\n                title: _t(\"Delete Widget\"),\n                description: _t(\n                    \"Deleting a widget removes it for all users in this room.\" +\n                    \" Are you sure you want to delete this widget?\"),\n                button: _t(\"Delete widget\"),\n                onFinished: (confirmed) => {\n                    if (!confirmed) return;\n                    WidgetUtils.setRoomWidget(roomId, app.id);\n                },\n            });\n            onFinished();\n        };\n\n        deleteButton = <IconizedContextMenuOption\n            onClick={onDeleteClick || onDeleteClickDefault}\n            label={userWidget ? _t(\"Remove\") : _t(\"Remove for everyone\")}\n        />;\n    }\n\n    let isAllowedWidget = SettingsStore.getValue(\"allowedWidgets\", roomId)[app.eventId];\n    if (isAllowedWidget === undefined) {\n        isAllowedWidget = app.creatorUserId === cli.getUserId();\n    }\n\n    const isLocalWidget = WidgetType.JITSI.matches(app.type);\n    let revokeButton;\n    if (!userWidget && !isLocalWidget && isAllowedWidget) {\n        const onRevokeClick = () => {\n            console.info(\"Revoking permission for widget to load: \" + app.eventId);\n            const current = SettingsStore.getValue(\"allowedWidgets\", roomId);\n            current[app.eventId] = false;\n            const level = SettingsStore.firstSupportedLevel(\"allowedWidgets\");\n            SettingsStore.setValue(\"allowedWidgets\", roomId, level, current).catch(err => {\n                console.error(err);\n                // We don't really need to do anything about this - the user will just hit the button again.\n            });\n            onFinished();\n        };\n\n        revokeButton = <IconizedContextMenuOption onClick={onRevokeClick} label={_t(\"Revoke permissions\")} />;\n    }\n\n    const pinnedWidgets = WidgetLayoutStore.instance.getContainerWidgets(room, Container.Top);\n    const widgetIndex = pinnedWidgets.findIndex(widget => widget.id === app.id);\n\n    let moveLeftButton;\n    if (showUnpin && widgetIndex > 0) {\n        const onClick = () => {\n            WidgetLayoutStore.instance.moveWithinContainer(room, Container.Top, app, -1);\n            onFinished();\n        };\n\n        moveLeftButton = <IconizedContextMenuOption onClick={onClick} label={_t(\"Move left\")} />;\n    }\n\n    let moveRightButton;\n    if (showUnpin && widgetIndex < pinnedWidgets.length - 1) {\n        const onClick = () => {\n            WidgetLayoutStore.instance.moveWithinContainer(room, Container.Top, app, 1);\n            onFinished();\n        };\n\n        moveRightButton = <IconizedContextMenuOption onClick={onClick} label={_t(\"Move right\")} />;\n    }\n\n    return <IconizedContextMenu {...props} chevronFace={ChevronFace.None} onFinished={onFinished}>\n        <IconizedContextMenuOptionList>\n            { streamAudioStreamButton }\n            { editButton }\n            { revokeButton }\n            { deleteButton }\n            { snapshotButton }\n            { moveLeftButton }\n            { moveRightButton }\n            { unpinButton }\n        </IconizedContextMenuOptionList>\n    </IconizedContextMenu>;\n};\n\nexport default WidgetContextMenu;\n"]}