UNPKG

matrix-react-sdk

Version:
963 lines (955 loc) 155 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); Object.defineProperty(exports, "__esModule", { value: true }); Object.defineProperty(exports, "Command", { enumerable: true, get: function () { return _command.Command; } }); Object.defineProperty(exports, "CommandCategories", { enumerable: true, get: function () { return _interface.CommandCategories; } }); exports.Commands = exports.CommandMap = void 0; exports.getCommand = getCommand; exports.parseCommandString = parseCommandString; var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty")); var React = _interopRequireWildcard(require("react")); var _matrix = require("matrix-js-sdk/src/matrix"); var _logger = require("matrix-js-sdk/src/logger"); var _types = require("matrix-js-sdk/src/types"); var _dispatcher = _interopRequireDefault(require("./dispatcher/dispatcher")); var _languageHandler = require("./languageHandler"); var _Modal = _interopRequireDefault(require("./Modal")); var _MultiInviter = _interopRequireDefault(require("./utils/MultiInviter")); var _HtmlUtils = require("./HtmlUtils"); var _QuestionDialog = _interopRequireDefault(require("./components/views/dialogs/QuestionDialog")); var _WidgetUtils = _interopRequireDefault(require("./utils/WidgetUtils")); var _colour = require("./utils/colour"); var _UserAddress = require("./UserAddress"); var _UrlUtils = require("./utils/UrlUtils"); var _IdentityServerUtils = require("./utils/IdentityServerUtils"); var _WidgetType = require("./widgets/WidgetType"); var _Jitsi = require("./widgets/Jitsi"); var _BugReportDialog = _interopRequireDefault(require("./components/views/dialogs/BugReportDialog")); var _createRoom = require("./createRoom"); var _actions = require("./dispatcher/actions"); var _SdkConfig = _interopRequireDefault(require("./SdkConfig")); var _SettingsStore = _interopRequireDefault(require("./settings/SettingsStore")); var _UIFeature = require("./settings/UIFeature"); var _effects = require("./effects"); var _LegacyCallHandler = _interopRequireDefault(require("./LegacyCallHandler")); var _Rooms = require("./Rooms"); var _RoomUpgrade = require("./utils/RoomUpgrade"); var _DevtoolsDialog = _interopRequireDefault(require("./components/views/dialogs/DevtoolsDialog")); var _RoomUpgradeWarningDialog = _interopRequireDefault(require("./components/views/dialogs/RoomUpgradeWarningDialog")); var _InfoDialog = _interopRequireDefault(require("./components/views/dialogs/InfoDialog")); var _SlashCommandHelpDialog = _interopRequireDefault(require("./components/views/dialogs/SlashCommandHelpDialog")); var _UIComponents = require("./customisations/helpers/UIComponents"); var _RoomContext = require("./contexts/RoomContext"); var _VoipUserMapper = _interopRequireDefault(require("./VoipUserMapper")); var _serialize = require("./editor/serialize"); var _leaveBehaviour = require("./utils/leave-behaviour"); var _MatrixClientPeg = require("./MatrixClientPeg"); var _deviceInfo = require("./utils/crypto/deviceInfo"); var _utils = require("./slash-commands/utils"); var _op = require("./slash-commands/op"); var _interface = require("./slash-commands/interface"); var _command = require("./slash-commands/command"); var _join = require("./slash-commands/join"); 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 The Matrix.org Foundation C.I.C. Copyright 2019 Michael Telatynski <7t3chguy@gmail.com> Copyright 2018 New Vector Ltd Copyright 2015, 2016 OpenMarket Ltd SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only Please see LICENSE files in the repository root for full details. */ const Commands = exports.Commands = [new _command.Command({ command: "spoiler", args: "<message>", description: (0, _languageHandler._td)("slash_command|spoiler"), runFn: function (cli, roomId, threadId, message = "") { return (0, _utils.successSync)(_matrix.ContentHelpers.makeHtmlMessage(message, `<span data-mx-spoiler>${message}</span>`)); }, category: _interface.CommandCategories.messages }), new _command.Command({ command: "shrug", args: "<message>", description: (0, _languageHandler._td)("slash_command|shrug"), runFn: function (cli, roomId, threadId, args) { let message = "¯\\_(ツ)_/¯"; if (args) { message = message + " " + args; } return (0, _utils.successSync)(_matrix.ContentHelpers.makeTextMessage(message)); }, category: _interface.CommandCategories.messages }), new _command.Command({ command: "tableflip", args: "<message>", description: (0, _languageHandler._td)("slash_command|tableflip"), runFn: function (cli, roomId, threadId, args) { let message = "(╯°□°)╯︵ ┻━┻"; if (args) { message = message + " " + args; } return (0, _utils.successSync)(_matrix.ContentHelpers.makeTextMessage(message)); }, category: _interface.CommandCategories.messages }), new _command.Command({ command: "unflip", args: "<message>", description: (0, _languageHandler._td)("slash_command|unflip"), runFn: function (cli, roomId, threadId, args) { let message = "┬──┬ ノ( ゜-゜ノ)"; if (args) { message = message + " " + args; } return (0, _utils.successSync)(_matrix.ContentHelpers.makeTextMessage(message)); }, category: _interface.CommandCategories.messages }), new _command.Command({ command: "lenny", args: "<message>", description: (0, _languageHandler._td)("slash_command|lenny"), runFn: function (cli, roomId, threadId, args) { let message = "( ͡° ͜ʖ ͡°)"; if (args) { message = message + " " + args; } return (0, _utils.successSync)(_matrix.ContentHelpers.makeTextMessage(message)); }, category: _interface.CommandCategories.messages }), new _command.Command({ command: "plain", args: "<message>", description: (0, _languageHandler._td)("slash_command|plain"), runFn: function (cli, roomId, threadId, messages = "") { return (0, _utils.successSync)(_matrix.ContentHelpers.makeTextMessage(messages)); }, category: _interface.CommandCategories.messages }), new _command.Command({ command: "html", args: "<message>", description: (0, _languageHandler._td)("slash_command|html"), runFn: function (cli, roomId, threadId, messages = "") { return (0, _utils.successSync)(_matrix.ContentHelpers.makeHtmlMessage(messages, messages)); }, category: _interface.CommandCategories.messages }), new _command.Command({ command: "upgraderoom", args: "<new_version>", description: (0, _languageHandler._td)("slash_command|upgraderoom"), isEnabled: cli => !(0, _utils.isCurrentLocalRoom)(cli) && _SettingsStore.default.getValue("developerMode"), runFn: function (cli, roomId, threadId, args) { if (args) { const room = cli.getRoom(roomId); if (!room?.currentState.mayClientSendStateEvent("m.room.tombstone", cli)) { return (0, _utils.reject)(new _languageHandler.UserFriendlyError("slash_command|upgraderoom_permission_error")); } const { finished } = _Modal.default.createDialog(_RoomUpgradeWarningDialog.default, { roomId: roomId, targetVersion: args }, /*className=*/undefined, /*isPriority=*/false, /*isStatic=*/true); return (0, _utils.success)(finished.then(async ([resp]) => { if (!resp?.continue) return; await (0, _RoomUpgrade.upgradeRoom)(room, args, resp.invite); })); } return (0, _utils.reject)(this.getUsage()); }, category: _interface.CommandCategories.admin, renderingTypes: [_RoomContext.TimelineRenderingType.Room] }), new _command.Command({ command: "jumptodate", args: "<YYYY-MM-DD>", description: (0, _languageHandler._td)("slash_command|jumptodate"), isEnabled: () => _SettingsStore.default.getValue("feature_jump_to_date"), runFn: function (cli, roomId, threadId, args) { if (args) { return (0, _utils.success)((async () => { const unixTimestamp = Date.parse(args); if (!unixTimestamp) { throw new _languageHandler.UserFriendlyError("slash_command|jumptodate_invalid_input", { inputDate: args, cause: undefined }); } const { event_id: eventId, origin_server_ts: originServerTs } = await cli.timestampToEvent(roomId, unixTimestamp, _matrix.Direction.Forward); _logger.logger.log(`/timestamp_to_event: found ${eventId} (${originServerTs}) for timestamp=${unixTimestamp}`); _dispatcher.default.dispatch({ action: _actions.Action.ViewRoom, event_id: eventId, highlighted: true, room_id: roomId, metricsTrigger: "SlashCommand", metricsViaKeyboard: true }); })()); } return (0, _utils.reject)(this.getUsage()); }, category: _interface.CommandCategories.actions }), new _command.Command({ command: "nick", args: "<display_name>", description: (0, _languageHandler._td)("slash_command|nick"), runFn: function (cli, roomId, threadId, args) { if (args) { return (0, _utils.success)(cli.setDisplayName(args)); } return (0, _utils.reject)(this.getUsage()); }, category: _interface.CommandCategories.actions, renderingTypes: [_RoomContext.TimelineRenderingType.Room] }), new _command.Command({ command: "myroomnick", aliases: ["roomnick"], args: "<display_name>", description: (0, _languageHandler._td)("slash_command|myroomnick"), isEnabled: cli => !(0, _utils.isCurrentLocalRoom)(cli), runFn: function (cli, roomId, threadId, args) { if (args) { const ev = cli.getRoom(roomId)?.currentState.getStateEvents(_matrix.EventType.RoomMember, cli.getSafeUserId()); const content = _objectSpread(_objectSpread({}, ev ? ev.getContent() : { membership: _types.KnownMembership.Join }), {}, { displayname: args }); return (0, _utils.success)(cli.sendStateEvent(roomId, _matrix.EventType.RoomMember, content, cli.getSafeUserId())); } return (0, _utils.reject)(this.getUsage()); }, category: _interface.CommandCategories.actions, renderingTypes: [_RoomContext.TimelineRenderingType.Room] }), new _command.Command({ command: "roomavatar", args: "[<mxc_url>]", description: (0, _languageHandler._td)("slash_command|roomavatar"), isEnabled: cli => !(0, _utils.isCurrentLocalRoom)(cli), runFn: function (cli, roomId, threadId, args) { let promise = Promise.resolve(args ?? null); if (!args) { promise = (0, _utils.singleMxcUpload)(cli); } return (0, _utils.success)(promise.then(url => { if (!url) return; return cli.sendStateEvent(roomId, _matrix.EventType.RoomAvatar, { url }, ""); })); }, category: _interface.CommandCategories.actions, renderingTypes: [_RoomContext.TimelineRenderingType.Room] }), new _command.Command({ command: "myroomavatar", args: "[<mxc_url>]", description: (0, _languageHandler._td)("slash_command|myroomavatar"), isEnabled: cli => !(0, _utils.isCurrentLocalRoom)(cli), runFn: function (cli, roomId, threadId, args) { const room = cli.getRoom(roomId); const userId = cli.getSafeUserId(); let promise = Promise.resolve(args ?? null); if (!args) { promise = (0, _utils.singleMxcUpload)(cli); } return (0, _utils.success)(promise.then(url => { if (!url) return; const ev = room?.currentState.getStateEvents(_matrix.EventType.RoomMember, userId); const content = _objectSpread(_objectSpread({}, ev ? ev.getContent() : { membership: _types.KnownMembership.Join }), {}, { avatar_url: url }); return cli.sendStateEvent(roomId, _matrix.EventType.RoomMember, content, userId); })); }, category: _interface.CommandCategories.actions, renderingTypes: [_RoomContext.TimelineRenderingType.Room] }), new _command.Command({ command: "myavatar", args: "[<mxc_url>]", description: (0, _languageHandler._td)("slash_command|myavatar"), runFn: function (cli, roomId, threadId, args) { let promise = Promise.resolve(args ?? null); if (!args) { promise = (0, _utils.singleMxcUpload)(cli); } return (0, _utils.success)(promise.then(url => { if (!url) return; return cli.setAvatarUrl(url); })); }, category: _interface.CommandCategories.actions, renderingTypes: [_RoomContext.TimelineRenderingType.Room] }), new _command.Command({ command: "topic", args: "[<topic>]", description: (0, _languageHandler._td)("slash_command|topic"), isEnabled: cli => !(0, _utils.isCurrentLocalRoom)(cli), runFn: function (cli, roomId, threadId, args) { if (args) { const html = (0, _serialize.htmlSerializeFromMdIfNeeded)(args, { forceHTML: false }); return (0, _utils.success)(cli.setRoomTopic(roomId, args, html)); } const room = cli.getRoom(roomId); if (!room) { return (0, _utils.reject)(new _languageHandler.UserFriendlyError("slash_command|topic_room_error", { roomId, cause: undefined })); } const content = room.currentState.getStateEvents("m.room.topic", "")?.getContent(); const topic = !!content ? _matrix.ContentHelpers.parseTopicContent(content) : { text: (0, _languageHandler._t)("slash_command|topic_none") }; const body = (0, _HtmlUtils.topicToHtml)(topic.text, topic.html, undefined, true); _Modal.default.createDialog(_InfoDialog.default, { title: room.name, description: /*#__PURE__*/React.createElement(_HtmlUtils.Linkify, null, body), hasCloseButton: true, className: "markdown-body" }); return (0, _utils.success)(); }, category: _interface.CommandCategories.admin, renderingTypes: [_RoomContext.TimelineRenderingType.Room] }), new _command.Command({ command: "roomname", args: "<name>", description: (0, _languageHandler._td)("slash_command|roomname"), isEnabled: cli => !(0, _utils.isCurrentLocalRoom)(cli), runFn: function (cli, roomId, threadId, args) { if (args) { return (0, _utils.success)(cli.setRoomName(roomId, args)); } return (0, _utils.reject)(this.getUsage()); }, category: _interface.CommandCategories.admin, renderingTypes: [_RoomContext.TimelineRenderingType.Room] }), new _command.Command({ command: "invite", args: "<user-id> [<reason>]", description: (0, _languageHandler._td)("slash_command|invite"), analyticsName: "Invite", isEnabled: cli => !(0, _utils.isCurrentLocalRoom)(cli) && (0, _UIComponents.shouldShowComponent)(_UIFeature.UIComponent.InviteUsers), runFn: function (cli, roomId, threadId, args) { if (args) { const [address, reason] = args.split(/\s+(.+)/); if (address) { // We use a MultiInviter to re-use the invite logic, even though // we're only inviting one user. // If we need an identity server but don't have one, things // get a bit more complex here, but we try to show something // meaningful. let prom = Promise.resolve(); if ((0, _UserAddress.getAddressType)(address) === _UserAddress.AddressType.Email && !cli.getIdentityServerUrl()) { const defaultIdentityServerUrl = (0, _IdentityServerUtils.getDefaultIdentityServerUrl)(); if (defaultIdentityServerUrl) { const { finished } = _Modal.default.createDialog(_QuestionDialog.default, { title: (0, _languageHandler._t)("slash_command|invite_3pid_use_default_is_title"), description: /*#__PURE__*/React.createElement("p", null, (0, _languageHandler._t)("slash_command|invite_3pid_use_default_is_title_description", { defaultIdentityServerName: (0, _UrlUtils.abbreviateUrl)(defaultIdentityServerUrl) })), button: (0, _languageHandler._t)("action|continue") }); prom = finished.then(([useDefault]) => { if (useDefault) { (0, _IdentityServerUtils.setToDefaultIdentityServer)(cli); return; } throw new _languageHandler.UserFriendlyError("slash_command|invite_3pid_needs_is_error"); }); } else { return (0, _utils.reject)(new _languageHandler.UserFriendlyError("slash_command|invite_3pid_needs_is_error")); } } const inviter = new _MultiInviter.default(cli, roomId); return (0, _utils.success)(prom.then(() => { return inviter.invite([address], reason); }).then(() => { if (inviter.getCompletionState(address) !== "invited") { const errorStringFromInviterUtility = inviter.getErrorText(address); if (errorStringFromInviterUtility) { throw new Error(errorStringFromInviterUtility); } else { throw new _languageHandler.UserFriendlyError("slash_command|invite_failed", { user: address, roomId, cause: undefined }); } } })); } } return (0, _utils.reject)(this.getUsage()); }, category: _interface.CommandCategories.actions, renderingTypes: [_RoomContext.TimelineRenderingType.Room] }), _join.goto, _join.join, new _command.Command({ command: "part", args: "[<room-address>]", description: (0, _languageHandler._td)("action|leave_room"), analyticsName: "Part", isEnabled: cli => !(0, _utils.isCurrentLocalRoom)(cli), runFn: function (cli, roomId, threadId, args) { let targetRoomId; if (args) { const matches = args.match(/^(\S+)$/); if (matches) { let roomAlias = matches[1]; if (roomAlias[0] !== "#") return (0, _utils.reject)(this.getUsage()); if (!roomAlias.includes(":")) { roomAlias += ":" + cli.getDomain(); } // Try to find a room with this alias const rooms = cli.getRooms(); targetRoomId = rooms.find(room => { return room.getCanonicalAlias() === roomAlias || room.getAltAliases().includes(roomAlias); })?.roomId; if (!targetRoomId) { return (0, _utils.reject)(new _languageHandler.UserFriendlyError("slash_command|part_unknown_alias", { roomAlias, cause: undefined })); } } } if (!targetRoomId) targetRoomId = roomId; return (0, _utils.success)((0, _leaveBehaviour.leaveRoomBehaviour)(cli, targetRoomId)); }, category: _interface.CommandCategories.actions, renderingTypes: [_RoomContext.TimelineRenderingType.Room] }), new _command.Command({ command: "remove", aliases: ["kick"], args: "<user-id> [reason]", description: (0, _languageHandler._td)("slash_command|remove"), isEnabled: cli => !(0, _utils.isCurrentLocalRoom)(cli), runFn: function (cli, roomId, threadId, args) { if (args) { const matches = args.match(/^(\S+?)( +(.*))?$/); if (matches) { return (0, _utils.success)(cli.kick(roomId, matches[1], matches[3])); } } return (0, _utils.reject)(this.getUsage()); }, category: _interface.CommandCategories.admin, renderingTypes: [_RoomContext.TimelineRenderingType.Room] }), new _command.Command({ command: "ban", args: "<user-id> [reason]", description: (0, _languageHandler._td)("slash_command|ban"), isEnabled: cli => !(0, _utils.isCurrentLocalRoom)(cli), runFn: function (cli, roomId, threadId, args) { if (args) { const matches = args.match(/^(\S+?)( +(.*))?$/); if (matches) { return (0, _utils.success)(cli.ban(roomId, matches[1], matches[3])); } } return (0, _utils.reject)(this.getUsage()); }, category: _interface.CommandCategories.admin, renderingTypes: [_RoomContext.TimelineRenderingType.Room] }), new _command.Command({ command: "unban", args: "<user-id>", description: (0, _languageHandler._td)("slash_command|unban"), isEnabled: cli => !(0, _utils.isCurrentLocalRoom)(cli), runFn: function (cli, roomId, threadId, args) { if (args) { const matches = args.match(/^(\S+)$/); if (matches) { // Reset the user membership to "leave" to unban him return (0, _utils.success)(cli.unban(roomId, matches[1])); } } return (0, _utils.reject)(this.getUsage()); }, category: _interface.CommandCategories.admin, renderingTypes: [_RoomContext.TimelineRenderingType.Room] }), new _command.Command({ command: "ignore", args: "<user-id>", description: (0, _languageHandler._td)("slash_command|ignore"), runFn: function (cli, roomId, threadId, args) { if (args) { const matches = args.match(/^(@[^:]+:\S+)$/); if (matches) { const userId = matches[1]; const ignoredUsers = cli.getIgnoredUsers(); ignoredUsers.push(userId); // de-duped internally in the js-sdk return (0, _utils.success)(cli.setIgnoredUsers(ignoredUsers).then(() => { _Modal.default.createDialog(_InfoDialog.default, { title: (0, _languageHandler._t)("slash_command|ignore_dialog_title"), description: /*#__PURE__*/React.createElement("div", null, /*#__PURE__*/React.createElement("p", null, (0, _languageHandler._t)("slash_command|ignore_dialog_description", { userId }))) }); })); } } return (0, _utils.reject)(this.getUsage()); }, category: _interface.CommandCategories.actions }), new _command.Command({ command: "unignore", args: "<user-id>", description: (0, _languageHandler._td)("slash_command|unignore"), runFn: function (cli, roomId, threadId, args) { if (args) { const matches = args.match(/(^@[^:]+:\S+$)/); if (matches) { const userId = matches[1]; const ignoredUsers = cli.getIgnoredUsers(); const index = ignoredUsers.indexOf(userId); if (index !== -1) ignoredUsers.splice(index, 1); return (0, _utils.success)(cli.setIgnoredUsers(ignoredUsers).then(() => { _Modal.default.createDialog(_InfoDialog.default, { title: (0, _languageHandler._t)("slash_command|unignore_dialog_title"), description: /*#__PURE__*/React.createElement("div", null, /*#__PURE__*/React.createElement("p", null, (0, _languageHandler._t)("slash_command|unignore_dialog_description", { userId }))) }); })); } } return (0, _utils.reject)(this.getUsage()); }, category: _interface.CommandCategories.actions }), _op.op, _op.deop, new _command.Command({ command: "devtools", description: (0, _languageHandler._td)("slash_command|devtools"), runFn: function (cli, roomId, threadRootId) { _Modal.default.createDialog(_DevtoolsDialog.default, { roomId, threadRootId }, "mx_DevtoolsDialog_wrapper"); return (0, _utils.success)(); }, category: _interface.CommandCategories.advanced }), new _command.Command({ command: "addwidget", args: "<url | embed code | Jitsi url>", description: (0, _languageHandler._td)("slash_command|addwidget"), isEnabled: cli => _SettingsStore.default.getValue(_UIFeature.UIFeature.Widgets) && (0, _UIComponents.shouldShowComponent)(_UIFeature.UIComponent.AddIntegrations) && !(0, _utils.isCurrentLocalRoom)(cli), runFn: function (cli, roomId, threadId, widgetUrl) { if (!widgetUrl) { return (0, _utils.reject)(new _languageHandler.UserFriendlyError("slash_command|addwidget_missing_url")); } // Try and parse out a widget URL from iframes if (widgetUrl.toLowerCase().startsWith("<iframe ")) { const embed = new DOMParser().parseFromString(widgetUrl, "text/html").body; if (embed?.childNodes?.length === 1) { const iframe = embed.firstElementChild; if (iframe?.tagName.toLowerCase() === "iframe") { _logger.logger.log("Pulling URL out of iframe (embed code)"); if (!iframe.hasAttribute("src")) { return (0, _utils.reject)(new _languageHandler.UserFriendlyError("slash_command|addwidget_iframe_missing_src")); } widgetUrl = iframe.getAttribute("src"); } } } if (!widgetUrl.startsWith("https://") && !widgetUrl.startsWith("http://")) { return (0, _utils.reject)(new _languageHandler.UserFriendlyError("slash_command|addwidget_invalid_protocol")); } if (_WidgetUtils.default.canUserModifyWidgets(cli, roomId)) { const userId = cli.getUserId(); const nowMs = new Date().getTime(); const widgetId = encodeURIComponent(`${roomId}_${userId}_${nowMs}`); let type = _WidgetType.WidgetType.CUSTOM; let name = "Custom"; let data = {}; // Make the widget a Jitsi widget if it looks like a Jitsi widget const jitsiData = _Jitsi.Jitsi.getInstance().parsePreferredConferenceUrl(widgetUrl); if (jitsiData) { _logger.logger.log("Making /addwidget widget a Jitsi conference"); type = _WidgetType.WidgetType.JITSI; name = "Jitsi"; data = jitsiData; widgetUrl = _WidgetUtils.default.getLocalJitsiWrapperUrl(); } return (0, _utils.success)(_WidgetUtils.default.setRoomWidget(cli, roomId, widgetId, type, widgetUrl, name, data)); } else { return (0, _utils.reject)(new _languageHandler.UserFriendlyError("slash_command|addwidget_no_permissions")); } }, category: _interface.CommandCategories.admin, renderingTypes: [_RoomContext.TimelineRenderingType.Room] }), new _command.Command({ command: "verify", args: "<user-id> <device-id> <device-signing-key>", description: (0, _languageHandler._td)("slash_command|verify"), runFn: function (cli, roomId, threadId, args) { if (args) { const matches = args.match(/^(\S+) +(\S+) +(\S+)$/); if (matches) { const userId = matches[1]; const deviceId = matches[2]; const fingerprint = matches[3]; return (0, _utils.success)((async () => { const device = await (0, _deviceInfo.getDeviceCryptoInfo)(cli, userId, deviceId); if (!device) { throw new _languageHandler.UserFriendlyError("slash_command|verify_unknown_pair", { userId, deviceId, cause: undefined }); } const deviceTrust = await cli.getCrypto()?.getDeviceVerificationStatus(userId, deviceId); if (deviceTrust?.isVerified()) { if (device.getFingerprint() === fingerprint) { throw new _languageHandler.UserFriendlyError("slash_command|verify_nop"); } else { throw new _languageHandler.UserFriendlyError("slash_command|verify_nop_warning_mismatch"); } } if (device.getFingerprint() !== fingerprint) { const fprint = device.getFingerprint(); throw new _languageHandler.UserFriendlyError("slash_command|verify_mismatch", { fprint, userId, deviceId, fingerprint, cause: undefined }); } await cli.setDeviceVerified(userId, deviceId, true); // Tell the user we verified everything _Modal.default.createDialog(_InfoDialog.default, { title: (0, _languageHandler._t)("slash_command|verify_success_title"), description: /*#__PURE__*/React.createElement("div", null, /*#__PURE__*/React.createElement("p", null, (0, _languageHandler._t)("slash_command|verify_success_description", { userId, deviceId }))) }); })()); } } return (0, _utils.reject)(this.getUsage()); }, category: _interface.CommandCategories.advanced, renderingTypes: [_RoomContext.TimelineRenderingType.Room] }), new _command.Command({ command: "discardsession", description: (0, _languageHandler._td)("slash_command|discardsession"), isEnabled: cli => !(0, _utils.isCurrentLocalRoom)(cli), runFn: function (cli, roomId) { try { cli.forceDiscardSession(roomId); } catch (e) { return (0, _utils.reject)(e instanceof Error ? e.message : e); } return (0, _utils.success)(); }, category: _interface.CommandCategories.advanced, renderingTypes: [_RoomContext.TimelineRenderingType.Room] }), new _command.Command({ command: "remakeolm", description: (0, _languageHandler._td)("slash_command|remakeolm"), isEnabled: cli => { return _SettingsStore.default.getValue("developerMode") && !(0, _utils.isCurrentLocalRoom)(cli); }, runFn: (cli, roomId) => { try { const room = cli.getRoom(roomId); cli.forceDiscardSession(roomId); return (0, _utils.success)(room?.getEncryptionTargetMembers().then(members => { // noinspection JSIgnoredPromiseFromCall cli.crypto?.ensureOlmSessionsForUsers(members.map(m => m.userId), true); })); } catch (e) { return (0, _utils.reject)(e instanceof Error ? e.message : e); } }, category: _interface.CommandCategories.advanced, renderingTypes: [_RoomContext.TimelineRenderingType.Room] }), new _command.Command({ command: "rainbow", description: (0, _languageHandler._td)("slash_command|rainbow"), args: "<message>", runFn: function (cli, roomId, threadId, args) { if (!args) return (0, _utils.reject)(this.getUsage()); return (0, _utils.successSync)(_matrix.ContentHelpers.makeHtmlMessage(args, (0, _colour.textToHtmlRainbow)(args))); }, category: _interface.CommandCategories.messages }), new _command.Command({ command: "rainbowme", description: (0, _languageHandler._td)("slash_command|rainbowme"), args: "<message>", runFn: function (cli, roomId, threadId, args) { if (!args) return (0, _utils.reject)(this.getUsage()); return (0, _utils.successSync)(_matrix.ContentHelpers.makeHtmlEmote(args, (0, _colour.textToHtmlRainbow)(args))); }, category: _interface.CommandCategories.messages }), new _command.Command({ command: "help", description: (0, _languageHandler._td)("slash_command|help"), runFn: function () { _Modal.default.createDialog(_SlashCommandHelpDialog.default); return (0, _utils.success)(); }, category: _interface.CommandCategories.advanced }), new _command.Command({ command: "whois", description: (0, _languageHandler._td)("slash_command|whois"), args: "<user-id>", isEnabled: cli => !(0, _utils.isCurrentLocalRoom)(cli), runFn: function (cli, roomId, threadId, userId) { if (!userId || !userId.startsWith("@") || !userId.includes(":")) { return (0, _utils.reject)(this.getUsage()); } const member = cli.getRoom(roomId)?.getMember(userId); _dispatcher.default.dispatch({ action: _actions.Action.ViewUser, // XXX: We should be using a real member object and not assuming what the receiver wants. member: member || { userId } }); return (0, _utils.success)(); }, category: _interface.CommandCategories.advanced }), new _command.Command({ command: "rageshake", aliases: ["bugreport"], description: (0, _languageHandler._td)("slash_command|rageshake"), isEnabled: () => !!_SdkConfig.default.get().bug_report_endpoint_url, args: "<description>", runFn: function (cli, roomId, threadId, args) { return (0, _utils.success)(_Modal.default.createDialog(_BugReportDialog.default, { initialText: args }).finished); }, category: _interface.CommandCategories.advanced }), new _command.Command({ command: "tovirtual", description: (0, _languageHandler._td)("slash_command|tovirtual"), category: _interface.CommandCategories.advanced, isEnabled(cli) { return !!_LegacyCallHandler.default.instance.getSupportsVirtualRooms() && !(0, _utils.isCurrentLocalRoom)(cli); }, runFn: (cli, roomId) => { return (0, _utils.success)((async () => { const room = await _VoipUserMapper.default.sharedInstance().getVirtualRoomForRoom(roomId); if (!room) throw new _languageHandler.UserFriendlyError("slash_command|tovirtual_not_found"); _dispatcher.default.dispatch({ action: _actions.Action.ViewRoom, room_id: room.roomId, metricsTrigger: "SlashCommand", metricsViaKeyboard: true }); })()); } }), new _command.Command({ command: "query", description: (0, _languageHandler._td)("slash_command|query"), args: "<user-id>", runFn: function (cli, roomId, threadId, userId) { // easter-egg for now: look up phone numbers through the thirdparty API // (very dumb phone number detection...) const isPhoneNumber = userId && /^\+?[0123456789]+$/.test(userId); if (!userId || (!userId.startsWith("@") || !userId.includes(":")) && !isPhoneNumber) { return (0, _utils.reject)(this.getUsage()); } return (0, _utils.success)((async () => { if (isPhoneNumber) { const results = await _LegacyCallHandler.default.instance.pstnLookup(userId); if (!results || results.length === 0 || !results[0].userid) { throw new _languageHandler.UserFriendlyError("slash_command|query_not_found_phone_number"); } userId = results[0].userid; } const roomId = await (0, _createRoom.ensureDMExists)(cli, userId); if (!roomId) throw new Error("Failed to ensure DM exists"); _dispatcher.default.dispatch({ action: _actions.Action.ViewRoom, room_id: roomId, metricsTrigger: "SlashCommand", metricsViaKeyboard: true }); })()); }, category: _interface.CommandCategories.actions }), new _command.Command({ command: "msg", description: (0, _languageHandler._td)("slash_command|msg"), args: "<user-id> [<message>]", runFn: function (cli, roomId, threadId, args) { if (args) { // matches the first whitespace delimited group and then the rest of the string const matches = args.match(/^(\S+?)(?: +(.*))?$/s); if (matches) { const [userId, msg] = matches.slice(1); if (userId && userId.startsWith("@") && userId.includes(":")) { return (0, _utils.success)((async () => { const roomId = await (0, _createRoom.ensureDMExists)(cli, userId); if (!roomId) throw new Error("Failed to ensure DM exists"); _dispatcher.default.dispatch({ action: _actions.Action.ViewRoom, room_id: roomId, metricsTrigger: "SlashCommand", metricsViaKeyboard: true }); if (msg) { cli.sendTextMessage(roomId, msg); } })()); } } } return (0, _utils.reject)(this.getUsage()); }, category: _interface.CommandCategories.actions }), new _command.Command({ command: "holdcall", description: (0, _languageHandler._td)("slash_command|holdcall"), category: _interface.CommandCategories.other, isEnabled: cli => !(0, _utils.isCurrentLocalRoom)(cli), runFn: function (cli, roomId, threadId, args) { const call = _LegacyCallHandler.default.instance.getCallForRoom(roomId); if (!call) { return (0, _utils.reject)(new _languageHandler.UserFriendlyError("slash_command|no_active_call")); } call.setRemoteOnHold(true); return (0, _utils.success)(); }, renderingTypes: [_RoomContext.TimelineRenderingType.Room] }), new _command.Command({ command: "unholdcall", description: (0, _languageHandler._td)("slash_command|unholdcall"), category: _interface.CommandCategories.other, isEnabled: cli => !(0, _utils.isCurrentLocalRoom)(cli), runFn: function (cli, roomId, threadId, args) { const call = _LegacyCallHandler.default.instance.getCallForRoom(roomId); if (!call) { return (0, _utils.reject)(new _languageHandler.UserFriendlyError("slash_command|no_active_call")); } call.setRemoteOnHold(false); return (0, _utils.success)(); }, renderingTypes: [_RoomContext.TimelineRenderingType.Room] }), new _command.Command({ command: "converttodm", description: (0, _languageHandler._td)("slash_command|converttodm"), category: _interface.CommandCategories.other, isEnabled: cli => !(0, _utils.isCurrentLocalRoom)(cli), runFn: function (cli, roomId, threadId, args) { const room = cli.getRoom(roomId); if (!room) return (0, _utils.reject)(new _languageHandler.UserFriendlyError("slash_command|could_not_find_room")); return (0, _utils.success)((0, _Rooms.guessAndSetDMRoom)(room, true)); }, renderingTypes: [_RoomContext.TimelineRenderingType.Room] }), new _command.Command({ command: "converttoroom", description: (0, _languageHandler._td)("slash_command|converttoroom"), category: _interface.CommandCategories.other, isEnabled: cli => !(0, _utils.isCurrentLocalRoom)(cli), runFn: function (cli, roomId, threadId, args) { const room = cli.getRoom(roomId); if (!room) return (0, _utils.reject)(new _languageHandler.UserFriendlyError("slash_command|could_not_find_room")); return (0, _utils.success)((0, _Rooms.guessAndSetDMRoom)(room, false)); }, renderingTypes: [_RoomContext.TimelineRenderingType.Room] }), // Command definitions for autocompletion ONLY: // /me is special because its not handled by SlashCommands.js and is instead done inside the Composer classes new _command.Command({ command: "me", args: "<message>", description: (0, _languageHandler._td)("slash_command|me"), category: _interface.CommandCategories.messages, hideCompletionAfterSpace: true }), ..._effects.CHAT_EFFECTS.map(effect => { return new _command.Command({ command: effect.command, description: effect.description(), args: "<message>", runFn: function (cli, roomId, threadId, args) { let content; if (!args) { content = _matrix.ContentHelpers.makeEmoteMessage(effect.fallbackMessage()); } else { content = { msgtype: effect.msgType, body: args }; } _dispatcher.default.dispatch({ action: `effects.${effect.command}` }); return (0, _utils.successSync)(content); }, category: _interface.CommandCategories.effects, renderingTypes: [_RoomContext.TimelineRenderingType.Room] }); })]; // build a map from names and aliases to the Command objects. const CommandMap = exports.CommandMap = new Map(); Commands.forEach(cmd => { CommandMap.set(cmd.command, cmd); cmd.aliases.forEach(alias => { CommandMap.set(alias, cmd); }); }); function parseCommandString(input) { // trim any trailing whitespace, as it can confuse the parser for IRC-style commands input = input.trimEnd(); if (input[0] !== "/") return {}; // not a command const bits = input.match(/^(\S+?)(?:[ \n]+((.|\n)*))?$/); let cmd; let args; if (bits) { cmd = bits[1].substring(1).toLowerCase(); args = bits[2]; } else { cmd = input; } return { cmd, args }; } /** * Process the given text for /commands and returns a parsed command that can be used for running the operation. * @param {string} input The raw text input by the user. * @return {ICmd} The parsed command object. * Returns an empty object if the input didn't match a command. */ function getCommand(input) { const { cmd, args } = parseCommandString(input); if (cmd && CommandMap.has(cmd) && CommandMap.get(cmd).isEnabled(_MatrixClientPeg.MatrixClientPeg.get())) { return { cmd: CommandMap.get(cmd), args }; } return {}; } //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["React","_interopRequireWildcard","require","_matrix","_logger","_types","_dispatcher","_interopRequireDefault","_languageHandler","_Modal","_MultiInviter","_HtmlUtils","_QuestionDialog","_WidgetUtils","_colour","_UserAddress","_UrlUtils","_IdentityServerUtils","_WidgetType","_Jitsi","_BugReportDialog","_createRoom","_actions","_SdkConfig","_SettingsStore","_UIFeature","_effects","_LegacyCallHandler","_Rooms","_RoomUpgrade","_DevtoolsDialog","_RoomUpgradeWarningDialog","_InfoDialog","_SlashCommandHelpDialog","_UIComponents","_RoomContext","_VoipUserMapper","_serialize","_leaveBehaviour","_MatrixClientPeg","_deviceInfo","_utils","_op","_interface","_command","_join","_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","Commands","exports","Command","command","args","description","_td","runFn","cli","roomId","threadId","message","successSync","ContentHelpers","makeHtmlMessage","category","CommandCategories","messages","makeTextMessage","isEnabled","isCurrentLocalRoom","SettingsStore","getValue","room","getRoom","currentState","mayClientSendStateEvent","reject","UserFriendlyError","finished","Modal","createDialog","RoomUpgradeWarningDialog","targetVersion","undefined","success","then","resp","continue","upgradeRoom","invite","getUsage","admin","renderingTypes","TimelineRenderingType","Room","unixTimestamp","Date","parse","inputDate","cause","event_id","eventId","origin_server_ts","originServerTs","timestampToEvent","Direction","Forward","logger","log","dis","dispatch","action","Action","ViewRoom","highlighted","room_id","metricsTrigger","metricsViaKeyboard","actions","setDisplayName","aliases","ev","getStateEvents","EventType","RoomMember","getSafeUserId","content","getContent","membership","KnownMembership","Join","displayname","sendStateEvent","promise","Promise","resolve","singleMxcUpload","url","RoomAvatar","userId","avatar_url","setAvatarUrl","html","htmlSerializeFromMdIfNeeded","forceHTML","setRoomTopic","topic","parseTopicContent","text","_t","body","topicToHtml","InfoDialog","title","name","createElement","Linkify","hasCloseButton","className","setRoomName","analyticsName","shouldShowComponent","UIComponent","InviteUsers","address","reason","split","prom","getAddressType","AddressType","Email","getIdentityServerUrl","defaultIdentityServerUrl","getDefaultIdentityServerUrl","QuestionDialog","defaultIdentityServerName","abbreviateUrl","button","useDefault","setToDefaultIdentityServer","inviter","MultiInviter","getCompletionState","errorStringFromInviterUtility","getErrorText","Error","user","goto","join","targetRoomId","matches","match","roomAlias","includes","getDomain","rooms","getRooms","find","getCanonicalAlias","getAltAliases","leaveRoomBehaviour","kick","ban","unban","ignoredUsers","getIgnoredUsers","setIgnoredUsers","index","indexOf","splice","op","deop","threadRootId","DevtoolsDialog","advanced","UIFeature","Widgets","AddIntegrations","widgetUrl","toLowerCase","startsWith","embed","DOMParser","parseFromString","childNodes","iframe","firstElementChild","tagName","hasAttribute","getAttribute","WidgetUtils","canUserModifyWidgets","getUserId","nowMs","getTime","widgetId","encodeURIComponent","type","WidgetType","CUSTOM","data","jitsiData","Jitsi","getInstance","parsePreferredConferenceUrl","JITSI","getLocalJitsiWrapperUrl","setRoomWidget","deviceId","fingerprint","device","getDeviceCryptoInfo","deviceTrust","getCrypto","getDeviceVerificationStatus","isVerified","getFingerprint","fprint","setDeviceVerified","forceDiscardSession","getEncryptionTargetMembers","members","crypto","ensureOlmSessionsForUsers","map","m","textToHtmlRainbow","makeHtmlEmote","SlashCommandHelpDialog","member","getMember","ViewUser","SdkConfig","bug_report_endpoint_url","BugReportDialog","initialText","LegacyCallHandler","instance","getSupportsVirtualRooms","VoipUserMapper","sharedInstance","getVirtualRoomForRoom","isPhoneNumber","test","results","pstnLookup","userid","ensureDMExists","msg","slice","sendTextMessage","other","getCallForRoom","setRemoteOnHold","guessAndSetDMRoom","hideCompletionAfterSpace","CHAT_EFFECTS","effect","makeEmoteMessage","fallbackMessage","msgtype","msgType","effects","CommandMap","Map","cmd","alias","parseCommandString","input","trimEnd","bits","substring","getCommand","MatrixClientPeg"],"sources":["../src/SlashCommands.tsx"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2020 The Matrix.org Foundation C.I.C.\nCopyright 2019 Michael Telatynski <7t3chguy@gmail.com>\nCopyright 2018 New Vector Ltd\nCopyright 2015, 2016 OpenMarket 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 * as React from \"react\";\nimport { ContentHelpers, Direction, EventType, IContent, MRoomTopicEventContent, User } from \"matrix-js-sdk/src/matrix\";\nimport { logger } from \"matrix-js-sdk/src/logger\";\nimport { KnownMembership, RoomMemberEventContent } from \"matrix-js-sdk/src/types\";\n\nimport dis from \"./dispatcher/dispatcher\";\nimport { _t, _td, UserFriendlyError } from \"./languageHandler\";\nimport Modal from \"./Modal\";\nimport MultiInviter from \"./utils/MultiInviter\";\nimport { Linkify, topicToHtml } from \"./HtmlUtils\";\nimport QuestionDialog from \"./components/views/dialogs/QuestionDialog\";\nimport WidgetUtils from \"./utils/WidgetUtils\";\nimport { textToHtmlRainbow } from \"./utils/colour\";\nimport { AddressType, getAddressType } from \"./UserAddress\";\nimport { abbreviateUrl } from \"./utils/UrlUtils\";\nimport { getDefaultIdentityServerUrl, setToDefaultIdentityServer } from \"./utils/IdentityServerUtils\";\nimport { WidgetType } from \"./widgets/WidgetType\";\nimport { Jitsi } from \"./widgets/Jitsi\";\nimport BugReportDialog from \"./components/views/dialogs/BugReportDialog\";\nimport { ensureDMExists } from \"./createRoom\";\nimport { ViewUserPayload } from \"./dispatcher/payloads/ViewUserPayload\";\nimport { Action } from \"./dispatcher/actions\";\nimport SdkConfig from \"./SdkConfig\";\nimport SettingsStore from \"./settings/SettingsStore\";\nimport { UIComponent, UIFeature } from \"./settings/UIFeature\";\nimport { CHAT_EFFECTS } from \"./effects\";\nimport LegacyCallHandler from \"./LegacyCallHandler\";\nimport { guessAndSetDMRoom } from \"./Rooms\";\nimport { upgradeRoom } from \"./utils/RoomUpgrade\";\nimport DevtoolsDialog from \"./components/views/dialogs/DevtoolsDialog\";\nimport RoomUpg