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