2016-03-05 05:30:18 +03:00
|
|
|
/*
|
|
|
|
Copyright 2015, 2016 OpenMarket Ltd
|
|
|
|
|
|
|
|
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.
|
|
|
|
*/
|
2021-06-24 11:40:30 +03:00
|
|
|
import React from 'react';
|
2021-06-17 14:35:40 +03:00
|
|
|
import { MatrixClientPeg } from './MatrixClientPeg';
|
2017-05-25 13:39:08 +03:00
|
|
|
import { _t } from './languageHandler';
|
2017-04-10 12:09:26 +03:00
|
|
|
import * as Roles from './Roles';
|
2021-06-17 14:35:40 +03:00
|
|
|
import { isValid3pidInvite } from "./RoomInvite";
|
2019-06-26 22:22:01 +03:00
|
|
|
import SettingsStore from "./settings/SettingsStore";
|
2021-06-17 14:35:40 +03:00
|
|
|
import { ALL_RULE_TYPES, ROOM_RULE_TYPES, SERVER_RULE_TYPES, USER_RULE_TYPES } from "./mjolnir/BanList";
|
|
|
|
import { WIDGET_LAYOUT_EVENT_TYPE } from "./stores/widgets/WidgetLayoutStore";
|
2021-06-24 11:40:30 +03:00
|
|
|
import { RightPanelPhases } from './stores/RightPanelStorePhases';
|
|
|
|
import { Action } from './dispatcher/actions';
|
|
|
|
import defaultDispatcher from './dispatcher/dispatcher';
|
|
|
|
import { SetRightPanelPhasePayload } from './dispatcher/payloads/SetRightPanelPhasePayload';
|
2021-06-24 13:24:42 +03:00
|
|
|
import { MatrixEvent } from "matrix-js-sdk/src/models/event";
|
2017-04-06 19:02:35 +03:00
|
|
|
|
2021-06-07 06:06:56 +03:00
|
|
|
// These functions are frequently used just to check whether an event has
|
|
|
|
// any text to display at all. For this reason they return deferred values
|
|
|
|
// to avoid the expense of looking up translations when they're not needed.
|
|
|
|
|
2021-06-30 05:13:18 +03:00
|
|
|
function textForMemberEvent(ev: MatrixEvent, allowJSX: boolean, showHiddenEvents?: boolean): () => string | null {
|
2015-10-30 05:07:04 +03:00
|
|
|
// XXX: SYJS-16 "sender is sometimes null for join messages"
|
2017-09-25 11:48:00 +03:00
|
|
|
const senderName = ev.sender ? ev.sender.name : ev.getSender();
|
|
|
|
const targetName = ev.target ? ev.target.name : ev.getStateKey();
|
|
|
|
const prevContent = ev.getPrevContent();
|
|
|
|
const content = ev.getContent();
|
2021-06-23 19:31:08 +03:00
|
|
|
const reason = content.reason;
|
2017-09-25 11:48:00 +03:00
|
|
|
|
|
|
|
switch (content.membership) {
|
|
|
|
case 'invite': {
|
|
|
|
const threePidContent = content.third_party_invite;
|
2015-12-17 18:48:14 +03:00
|
|
|
if (threePidContent) {
|
2016-03-02 19:04:24 +03:00
|
|
|
if (threePidContent.display_name) {
|
2021-06-24 04:42:47 +03:00
|
|
|
return () => _t('%(targetName)s accepted the invitation for %(displayName)s', {
|
2018-07-03 12:30:08 +03:00
|
|
|
targetName,
|
2017-09-25 11:48:00 +03:00
|
|
|
displayName: threePidContent.display_name,
|
|
|
|
});
|
2016-03-02 19:04:24 +03:00
|
|
|
} else {
|
2021-06-24 04:42:47 +03:00
|
|
|
return () => _t('%(targetName)s accepted an invitation', { targetName });
|
2016-03-02 19:04:24 +03:00
|
|
|
}
|
2017-09-25 11:48:00 +03:00
|
|
|
} else {
|
2021-06-24 04:42:47 +03:00
|
|
|
return () => _t('%(senderName)s invited %(targetName)s', { senderName, targetName });
|
2015-12-17 18:48:14 +03:00
|
|
|
}
|
2017-09-25 11:48:00 +03:00
|
|
|
}
|
2015-09-16 16:48:49 +03:00
|
|
|
case 'ban':
|
2021-06-23 19:31:08 +03:00
|
|
|
return () => reason
|
2021-06-24 04:42:47 +03:00
|
|
|
? _t('%(senderName)s banned %(targetName)s: %(reason)s', { senderName, targetName, reason })
|
|
|
|
: _t('%(senderName)s banned %(targetName)s', { senderName, targetName });
|
2015-09-16 16:48:49 +03:00
|
|
|
case 'join':
|
2017-09-25 11:48:00 +03:00
|
|
|
if (prevContent && prevContent.membership === 'join') {
|
|
|
|
if (prevContent.displayname && content.displayname && prevContent.displayname !== content.displayname) {
|
2021-06-24 04:42:47 +03:00
|
|
|
return () => _t('%(oldDisplayName)s changed their display name to %(displayName)s', {
|
2018-07-03 12:30:08 +03:00
|
|
|
oldDisplayName: prevContent.displayname,
|
|
|
|
displayName: content.displayname,
|
2017-09-25 11:48:00 +03:00
|
|
|
});
|
|
|
|
} else if (!prevContent.displayname && content.displayname) {
|
2021-06-24 04:42:47 +03:00
|
|
|
return () => _t('%(senderName)s set their display name to %(displayName)s', {
|
2018-07-03 12:30:08 +03:00
|
|
|
senderName: ev.getSender(),
|
|
|
|
displayName: content.displayname,
|
2017-09-25 11:48:00 +03:00
|
|
|
});
|
|
|
|
} else if (prevContent.displayname && !content.displayname) {
|
2021-06-24 04:42:47 +03:00
|
|
|
return () => _t('%(senderName)s removed their display name (%(oldDisplayName)s)', {
|
2018-07-03 12:30:08 +03:00
|
|
|
senderName,
|
|
|
|
oldDisplayName: prevContent.displayname,
|
2017-09-25 11:48:00 +03:00
|
|
|
});
|
2017-09-25 17:49:48 +03:00
|
|
|
} else if (prevContent.avatar_url && !content.avatar_url) {
|
2021-06-24 04:42:47 +03:00
|
|
|
return () => _t('%(senderName)s removed their profile picture', { senderName });
|
2017-09-25 11:48:00 +03:00
|
|
|
} else if (prevContent.avatar_url && content.avatar_url &&
|
|
|
|
prevContent.avatar_url !== content.avatar_url) {
|
2021-06-24 04:42:47 +03:00
|
|
|
return () => _t('%(senderName)s changed their profile picture', { senderName });
|
2017-09-25 11:48:00 +03:00
|
|
|
} else if (!prevContent.avatar_url && content.avatar_url) {
|
2021-06-24 04:42:47 +03:00
|
|
|
return () => _t('%(senderName)s set a profile picture', { senderName });
|
2021-06-16 01:59:42 +03:00
|
|
|
} else if (showHiddenEvents ?? SettingsStore.getValue("showHiddenEventsInTimeline")) {
|
2021-06-24 04:42:47 +03:00
|
|
|
// This is a null rejoin, it will only be visible if using 'show hidden events' (labs)
|
|
|
|
return () => _t("%(senderName)s made no change", { senderName });
|
2019-06-26 22:22:01 +03:00
|
|
|
} else {
|
2021-06-07 06:06:56 +03:00
|
|
|
return null;
|
2015-09-16 16:48:49 +03:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (!ev.target) console.warn("Join message has no target! -- " + ev.getContent().state_key);
|
2021-06-24 04:42:47 +03:00
|
|
|
return () => _t('%(targetName)s joined the room', { targetName });
|
2015-09-16 16:48:49 +03:00
|
|
|
}
|
|
|
|
case 'leave':
|
|
|
|
if (ev.getSender() === ev.getStateKey()) {
|
2020-09-24 18:16:20 +03:00
|
|
|
if (prevContent.membership === "invite") {
|
2021-06-24 04:42:47 +03:00
|
|
|
return () => _t('%(targetName)s rejected the invitation', { targetName });
|
2017-09-25 11:48:00 +03:00
|
|
|
} else {
|
2021-06-23 19:31:08 +03:00
|
|
|
return () => reason
|
2021-06-24 04:42:47 +03:00
|
|
|
? _t('%(targetName)s left the room: %(reason)s', { targetName, reason })
|
|
|
|
: _t('%(targetName)s left the room', { targetName });
|
2016-03-05 05:30:18 +03:00
|
|
|
}
|
2017-09-25 11:48:00 +03:00
|
|
|
} else if (prevContent.membership === "ban") {
|
2021-06-24 04:42:47 +03:00
|
|
|
return () => _t('%(senderName)s unbanned %(targetName)s', { senderName, targetName });
|
2017-09-25 11:48:00 +03:00
|
|
|
} else if (prevContent.membership === "invite") {
|
2021-06-23 19:31:08 +03:00
|
|
|
return () => reason
|
2021-06-24 04:42:47 +03:00
|
|
|
? _t('%(senderName)s withdrew %(targetName)s\'s invitation: %(reason)s', {
|
2021-06-23 19:31:08 +03:00
|
|
|
senderName,
|
|
|
|
targetName,
|
|
|
|
reason,
|
|
|
|
})
|
2021-06-29 15:11:58 +03:00
|
|
|
: _t('%(senderName)s withdrew %(targetName)s\'s invitation', { senderName, targetName });
|
2021-04-12 14:13:28 +03:00
|
|
|
} else if (prevContent.membership === "join") {
|
2021-06-23 19:31:08 +03:00
|
|
|
return () => reason
|
2021-06-24 04:42:47 +03:00
|
|
|
? _t('%(senderName)s kicked %(targetName)s: %(reason)s', {
|
2021-06-23 19:31:08 +03:00
|
|
|
senderName,
|
|
|
|
targetName,
|
|
|
|
reason,
|
|
|
|
})
|
2021-06-24 04:42:47 +03:00
|
|
|
: _t('%(senderName)s kicked %(targetName)s', { senderName, targetName });
|
2021-04-12 14:13:28 +03:00
|
|
|
} else {
|
2021-06-07 06:06:56 +03:00
|
|
|
return null;
|
2015-09-16 16:48:49 +03:00
|
|
|
}
|
|
|
|
}
|
2016-09-15 19:01:02 +03:00
|
|
|
}
|
2015-09-16 16:48:49 +03:00
|
|
|
|
2021-06-16 01:52:40 +03:00
|
|
|
function textForTopicEvent(ev: MatrixEvent): () => string | null {
|
2017-09-25 11:48:00 +03:00
|
|
|
const senderDisplayName = ev.sender && ev.sender.name ? ev.sender.name : ev.getSender();
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t('%(senderDisplayName)s changed the topic to "%(topic)s".', {
|
2018-07-03 12:30:08 +03:00
|
|
|
senderDisplayName,
|
2017-09-25 11:48:00 +03:00
|
|
|
topic: ev.getContent().topic,
|
|
|
|
});
|
2016-09-15 19:01:02 +03:00
|
|
|
}
|
2015-09-16 16:48:49 +03:00
|
|
|
|
2021-06-16 01:52:40 +03:00
|
|
|
function textForRoomNameEvent(ev: MatrixEvent): () => string | null {
|
2017-09-25 11:48:00 +03:00
|
|
|
const senderDisplayName = ev.sender && ev.sender.name ? ev.sender.name : ev.getSender();
|
2017-06-10 16:26:27 +03:00
|
|
|
|
2017-05-30 08:21:14 +03:00
|
|
|
if (!ev.getContent().name || ev.getContent().name.trim().length === 0) {
|
2021-06-29 15:11:58 +03:00
|
|
|
return () => _t('%(senderDisplayName)s removed the room name.', { senderDisplayName });
|
2017-05-30 08:21:14 +03:00
|
|
|
}
|
2020-03-07 21:36:24 +03:00
|
|
|
if (ev.getPrevContent().name) {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t('%(senderDisplayName)s changed the room name from %(oldRoomName)s to %(newRoomName)s.', {
|
2020-03-07 21:36:24 +03:00
|
|
|
senderDisplayName,
|
|
|
|
oldRoomName: ev.getPrevContent().name,
|
|
|
|
newRoomName: ev.getContent().name,
|
|
|
|
});
|
|
|
|
}
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t('%(senderDisplayName)s changed the room name to %(roomName)s.', {
|
2018-07-03 12:30:08 +03:00
|
|
|
senderDisplayName,
|
2017-09-25 11:48:00 +03:00
|
|
|
roomName: ev.getContent().name,
|
|
|
|
});
|
2016-09-15 19:01:02 +03:00
|
|
|
}
|
2015-10-30 05:07:04 +03:00
|
|
|
|
2021-06-16 01:52:40 +03:00
|
|
|
function textForTombstoneEvent(ev: MatrixEvent): () => string | null {
|
2019-01-11 01:15:45 +03:00
|
|
|
const senderDisplayName = ev.sender && ev.sender.name ? ev.sender.name : ev.getSender();
|
2021-06-29 15:11:58 +03:00
|
|
|
return () => _t('%(senderDisplayName)s upgraded this room.', { senderDisplayName });
|
2019-01-11 01:15:45 +03:00
|
|
|
}
|
|
|
|
|
2021-06-16 01:52:40 +03:00
|
|
|
function textForJoinRulesEvent(ev: MatrixEvent): () => string | null {
|
2019-02-07 22:14:58 +03:00
|
|
|
const senderDisplayName = ev.sender && ev.sender.name ? ev.sender.name : ev.getSender();
|
|
|
|
switch (ev.getContent().join_rule) {
|
|
|
|
case "public":
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t('%(senderDisplayName)s made the room public to whoever knows the link.', {
|
|
|
|
senderDisplayName,
|
|
|
|
});
|
2019-02-07 22:14:58 +03:00
|
|
|
case "invite":
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t('%(senderDisplayName)s made the room invite only.', {
|
|
|
|
senderDisplayName,
|
|
|
|
});
|
2019-02-07 22:14:58 +03:00
|
|
|
default:
|
|
|
|
// The spec supports "knock" and "private", however nothing implements these.
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t('%(senderDisplayName)s changed the join rule to %(rule)s', {
|
2019-02-07 22:14:58 +03:00
|
|
|
senderDisplayName,
|
|
|
|
rule: ev.getContent().join_rule,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-16 01:52:40 +03:00
|
|
|
function textForGuestAccessEvent(ev: MatrixEvent): () => string | null {
|
2019-02-07 22:14:58 +03:00
|
|
|
const senderDisplayName = ev.sender && ev.sender.name ? ev.sender.name : ev.getSender();
|
|
|
|
switch (ev.getContent().guest_access) {
|
|
|
|
case "can_join":
|
2021-06-29 15:11:58 +03:00
|
|
|
return () => _t('%(senderDisplayName)s has allowed guests to join the room.', { senderDisplayName });
|
2019-02-07 22:14:58 +03:00
|
|
|
case "forbidden":
|
2021-06-29 15:11:58 +03:00
|
|
|
return () => _t('%(senderDisplayName)s has prevented guests from joining the room.', { senderDisplayName });
|
2019-02-07 22:14:58 +03:00
|
|
|
default:
|
|
|
|
// There's no other options we can expect, however just for safety's sake we'll do this.
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t('%(senderDisplayName)s changed guest access to %(rule)s', {
|
2019-02-07 22:14:58 +03:00
|
|
|
senderDisplayName,
|
|
|
|
rule: ev.getContent().guest_access,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-16 01:52:40 +03:00
|
|
|
function textForRelatedGroupsEvent(ev: MatrixEvent): () => string | null {
|
2019-02-22 03:03:15 +03:00
|
|
|
const senderDisplayName = ev.sender && ev.sender.name ? ev.sender.name : ev.getSender();
|
|
|
|
const groups = ev.getContent().groups || [];
|
|
|
|
const prevGroups = ev.getPrevContent().groups || [];
|
|
|
|
const added = groups.filter((g) => !prevGroups.includes(g));
|
|
|
|
const removed = prevGroups.filter((g) => !groups.includes(g));
|
|
|
|
|
|
|
|
if (added.length && !removed.length) {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t('%(senderDisplayName)s enabled flair for %(groups)s in this room.', {
|
2019-02-22 03:03:15 +03:00
|
|
|
senderDisplayName,
|
|
|
|
groups: added.join(', '),
|
|
|
|
});
|
|
|
|
} else if (!added.length && removed.length) {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t('%(senderDisplayName)s disabled flair for %(groups)s in this room.', {
|
2019-02-22 03:03:15 +03:00
|
|
|
senderDisplayName,
|
|
|
|
groups: removed.join(', '),
|
|
|
|
});
|
|
|
|
} else if (added.length && removed.length) {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t('%(senderDisplayName)s enabled flair for %(newGroups)s and disabled flair for ' +
|
2019-02-22 03:03:15 +03:00
|
|
|
'%(oldGroups)s in this room.', {
|
|
|
|
senderDisplayName,
|
|
|
|
newGroups: added.join(', '),
|
|
|
|
oldGroups: removed.join(', '),
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
// Don't bother rendering this change (because there were no changes)
|
2021-06-07 06:06:56 +03:00
|
|
|
return null;
|
2019-02-22 03:03:15 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-16 01:52:40 +03:00
|
|
|
function textForServerACLEvent(ev: MatrixEvent): () => string | null {
|
2018-07-06 18:36:26 +03:00
|
|
|
const senderDisplayName = ev.sender && ev.sender.name ? ev.sender.name : ev.getSender();
|
2018-07-06 18:54:28 +03:00
|
|
|
const prevContent = ev.getPrevContent();
|
|
|
|
const current = ev.getContent();
|
|
|
|
const prev = {
|
|
|
|
deny: Array.isArray(prevContent.deny) ? prevContent.deny : [],
|
|
|
|
allow: Array.isArray(prevContent.allow) ? prevContent.allow : [],
|
|
|
|
allow_ip_literals: !(prevContent.allow_ip_literals === false),
|
|
|
|
};
|
2020-10-14 00:08:04 +03:00
|
|
|
|
2021-06-07 06:06:56 +03:00
|
|
|
let getText = null;
|
2018-07-06 17:31:21 +03:00
|
|
|
if (prev.deny.length === 0 && prev.allow.length === 0) {
|
2021-06-29 15:11:58 +03:00
|
|
|
getText = () => _t("%(senderDisplayName)s set the server ACLs for this room.", { senderDisplayName });
|
2018-07-06 12:18:31 +03:00
|
|
|
} else {
|
2021-06-29 15:11:58 +03:00
|
|
|
getText = () => _t("%(senderDisplayName)s changed the server ACLs for this room.", { senderDisplayName });
|
2018-07-06 12:18:31 +03:00
|
|
|
}
|
|
|
|
|
2018-07-06 22:22:37 +03:00
|
|
|
if (!Array.isArray(current.allow)) {
|
|
|
|
current.allow = [];
|
|
|
|
}
|
2020-10-14 00:08:04 +03:00
|
|
|
|
|
|
|
// If we know for sure everyone is banned, mark the room as obliterated
|
2018-07-06 18:54:28 +03:00
|
|
|
if (current.allow.length === 0) {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => getText() + " " +
|
|
|
|
_t("🎉 All servers are banned from participating! This room can no longer be used.");
|
2018-07-06 18:36:44 +03:00
|
|
|
}
|
|
|
|
|
2021-06-07 06:06:56 +03:00
|
|
|
return getText;
|
2016-09-15 19:01:02 +03:00
|
|
|
}
|
2015-10-30 05:07:04 +03:00
|
|
|
|
2021-06-16 01:52:40 +03:00
|
|
|
function textForMessageEvent(ev: MatrixEvent): () => string | null {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => {
|
|
|
|
const senderDisplayName = ev.sender && ev.sender.name ? ev.sender.name : ev.getSender();
|
|
|
|
let message = senderDisplayName + ': ' + ev.getContent().body;
|
|
|
|
if (ev.getContent().msgtype === "m.emote") {
|
|
|
|
message = "* " + senderDisplayName + " " + message;
|
|
|
|
} else if (ev.getContent().msgtype === "m.image") {
|
2021-06-29 15:11:58 +03:00
|
|
|
message = _t('%(senderDisplayName)s sent an image.', { senderDisplayName });
|
2021-06-07 06:06:56 +03:00
|
|
|
}
|
|
|
|
return message;
|
|
|
|
};
|
2016-09-15 19:01:02 +03:00
|
|
|
}
|
2015-09-16 16:48:49 +03:00
|
|
|
|
2021-06-16 01:52:40 +03:00
|
|
|
function textForCanonicalAliasEvent(ev: MatrixEvent): () => string | null {
|
2018-09-20 03:07:01 +03:00
|
|
|
const senderName = ev.sender && ev.sender.name ? ev.sender.name : ev.getSender();
|
|
|
|
const oldAlias = ev.getPrevContent().alias;
|
2020-03-09 20:24:48 +03:00
|
|
|
const oldAltAliases = ev.getPrevContent().alt_aliases || [];
|
2018-09-20 03:07:01 +03:00
|
|
|
const newAlias = ev.getContent().alias;
|
2020-03-09 20:24:48 +03:00
|
|
|
const newAltAliases = ev.getContent().alt_aliases || [];
|
|
|
|
const removedAltAliases = oldAltAliases.filter(alias => !newAltAliases.includes(alias));
|
|
|
|
const addedAltAliases = newAltAliases.filter(alias => !oldAltAliases.includes(alias));
|
2018-09-20 03:07:01 +03:00
|
|
|
|
2020-03-09 20:24:48 +03:00
|
|
|
if (!removedAltAliases.length && !addedAltAliases.length) {
|
|
|
|
if (newAlias) {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t('%(senderName)s set the main address for this room to %(address)s.', {
|
2020-03-09 20:24:48 +03:00
|
|
|
senderName: senderName,
|
|
|
|
address: ev.getContent().alias,
|
|
|
|
});
|
|
|
|
} else if (oldAlias) {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t('%(senderName)s removed the main address for this room.', {
|
2020-03-09 20:24:48 +03:00
|
|
|
senderName: senderName,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
} else if (newAlias === oldAlias) {
|
|
|
|
if (addedAltAliases.length && !removedAltAliases.length) {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t('%(senderName)s added the alternative addresses %(addresses)s for this room.', {
|
2020-03-09 20:24:48 +03:00
|
|
|
senderName: senderName,
|
|
|
|
addresses: addedAltAliases.join(", "),
|
|
|
|
count: addedAltAliases.length,
|
|
|
|
});
|
|
|
|
} if (removedAltAliases.length && !addedAltAliases.length) {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t('%(senderName)s removed the alternative addresses %(addresses)s for this room.', {
|
2020-03-09 20:24:48 +03:00
|
|
|
senderName: senderName,
|
|
|
|
addresses: removedAltAliases.join(", "),
|
|
|
|
count: removedAltAliases.length,
|
|
|
|
});
|
|
|
|
} if (removedAltAliases.length && addedAltAliases.length) {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t('%(senderName)s changed the alternative addresses for this room.', {
|
2020-03-09 20:24:48 +03:00
|
|
|
senderName: senderName,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
} else {
|
2020-03-10 20:45:51 +03:00
|
|
|
// both alias and alt_aliases where modified
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t('%(senderName)s changed the main and alternative addresses for this room.', {
|
2018-09-20 03:07:01 +03:00
|
|
|
senderName: senderName,
|
|
|
|
});
|
|
|
|
}
|
2020-03-10 20:45:51 +03:00
|
|
|
// in case there is no difference between the two events,
|
|
|
|
// say something as we can't simply hide the tile from here
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t('%(senderName)s changed the addresses for this room.', {
|
2020-03-10 20:45:51 +03:00
|
|
|
senderName: senderName,
|
|
|
|
});
|
2018-09-20 03:07:01 +03:00
|
|
|
}
|
|
|
|
|
2021-06-16 01:52:40 +03:00
|
|
|
function textForCallAnswerEvent(event: MatrixEvent): () => string | null {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => {
|
|
|
|
const senderName = event.sender ? event.sender.name : _t('Someone');
|
|
|
|
const supported = MatrixClientPeg.get().supportsVoip() ? '' : _t('(not supported by this browser)');
|
2021-06-29 15:11:58 +03:00
|
|
|
return _t('%(senderName)s answered the call.', { senderName }) + ' ' + supported;
|
2021-06-07 06:06:56 +03:00
|
|
|
};
|
2016-09-15 19:01:02 +03:00
|
|
|
}
|
2015-09-16 16:48:49 +03:00
|
|
|
|
2021-06-16 01:52:40 +03:00
|
|
|
function textForCallHangupEvent(event: MatrixEvent): () => string | null {
|
2021-06-07 06:06:56 +03:00
|
|
|
const getSenderName = () => event.sender ? event.sender.name : _t('Someone');
|
2017-06-10 16:26:27 +03:00
|
|
|
const eventContent = event.getContent();
|
2021-06-07 06:06:56 +03:00
|
|
|
let getReason = () => "";
|
2017-11-16 16:19:36 +03:00
|
|
|
if (!MatrixClientPeg.get().supportsVoip()) {
|
2021-06-07 06:06:56 +03:00
|
|
|
getReason = () => _t('(not supported by this browser)');
|
2017-11-16 16:19:36 +03:00
|
|
|
} else if (eventContent.reason) {
|
2017-06-11 09:19:19 +03:00
|
|
|
if (eventContent.reason === "ice_failed") {
|
2020-10-15 16:54:03 +03:00
|
|
|
// We couldn't establish a connection at all
|
2021-06-07 06:06:56 +03:00
|
|
|
getReason = () => _t('(could not connect media)');
|
2020-10-15 16:54:03 +03:00
|
|
|
} else if (eventContent.reason === "ice_timeout") {
|
|
|
|
// We established a connection but it died
|
2021-06-07 06:06:56 +03:00
|
|
|
getReason = () => _t('(connection failed)');
|
2020-10-15 16:54:03 +03:00
|
|
|
} else if (eventContent.reason === "user_media_failed") {
|
|
|
|
// The other side couldn't open capture devices
|
2021-06-07 06:06:56 +03:00
|
|
|
getReason = () => _t("(their device couldn't start the camera / microphone)");
|
2020-10-15 16:54:03 +03:00
|
|
|
} else if (eventContent.reason === "unknown_error") {
|
|
|
|
// An error code the other side doesn't have a way to express
|
|
|
|
// (as opposed to an error code they gave but we don't know about,
|
|
|
|
// in which case we show the error code)
|
2021-06-07 06:06:56 +03:00
|
|
|
getReason = () => _t("(an error occurred)");
|
2017-06-11 09:19:19 +03:00
|
|
|
} else if (eventContent.reason === "invite_timeout") {
|
2021-06-07 06:06:56 +03:00
|
|
|
getReason = () => _t('(no answer)');
|
2020-10-15 16:54:03 +03:00
|
|
|
} else if (eventContent.reason === "user hangup" || eventContent.reason === "user_hangup") {
|
2020-08-03 18:02:26 +03:00
|
|
|
// workaround for https://github.com/vector-im/element-web/issues/5178
|
2018-09-28 01:51:03 +03:00
|
|
|
// it seems Android randomly sets a reason of "user hangup" which is
|
|
|
|
// interpreted as an error code :(
|
|
|
|
// https://github.com/vector-im/riot-android/issues/2623
|
2020-10-15 16:54:03 +03:00
|
|
|
// Also the correct hangup code as of VoIP v1 (with underscore)
|
2021-06-07 06:06:56 +03:00
|
|
|
getReason = () => '';
|
2017-06-11 09:19:19 +03:00
|
|
|
} else {
|
2021-06-29 15:11:58 +03:00
|
|
|
getReason = () => _t('(unknown failure: %(reason)s)', { reason: eventContent.reason });
|
2017-06-11 09:19:19 +03:00
|
|
|
}
|
2017-06-10 16:26:27 +03:00
|
|
|
}
|
2021-06-29 15:11:58 +03:00
|
|
|
return () => _t('%(senderName)s ended the call.', { senderName: getSenderName() }) + ' ' + getReason();
|
2016-09-15 19:01:02 +03:00
|
|
|
}
|
2015-09-16 16:48:49 +03:00
|
|
|
|
2021-06-16 01:52:40 +03:00
|
|
|
function textForCallRejectEvent(event: MatrixEvent): () => string | null {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => {
|
|
|
|
const senderName = event.sender ? event.sender.name : _t('Someone');
|
2021-06-29 15:11:58 +03:00
|
|
|
return _t('%(senderName)s declined the call.', { senderName });
|
2021-06-07 06:06:56 +03:00
|
|
|
};
|
2020-10-15 16:54:03 +03:00
|
|
|
}
|
|
|
|
|
2021-06-16 01:52:40 +03:00
|
|
|
function textForCallInviteEvent(event: MatrixEvent): () => string | null {
|
2021-06-07 06:06:56 +03:00
|
|
|
const getSenderName = () => event.sender ? event.sender.name : _t('Someone');
|
2015-09-16 16:48:49 +03:00
|
|
|
// FIXME: Find a better way to determine this from the event?
|
2019-11-27 23:03:51 +03:00
|
|
|
let isVoice = true;
|
2015-09-16 16:48:49 +03:00
|
|
|
if (event.getContent().offer && event.getContent().offer.sdp &&
|
|
|
|
event.getContent().offer.sdp.indexOf('m=video') !== -1) {
|
2019-11-27 23:03:51 +03:00
|
|
|
isVoice = false;
|
|
|
|
}
|
|
|
|
const isSupported = MatrixClientPeg.get().supportsVoip();
|
|
|
|
|
|
|
|
// This ladder could be reduced down to a couple string variables, however other languages
|
|
|
|
// can have a hard time translating those strings. In an effort to make translations easier
|
2019-11-28 19:08:24 +03:00
|
|
|
// and more accurate, we break out the string-based variables to a couple booleans.
|
2019-11-27 23:03:51 +03:00
|
|
|
if (isVoice && isSupported) {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t("%(senderName)s placed a voice call.", {
|
|
|
|
senderName: getSenderName(),
|
|
|
|
});
|
2019-11-27 23:03:51 +03:00
|
|
|
} else if (isVoice && !isSupported) {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t("%(senderName)s placed a voice call. (not supported by this browser)", {
|
|
|
|
senderName: getSenderName(),
|
|
|
|
});
|
2019-11-27 23:03:51 +03:00
|
|
|
} else if (!isVoice && isSupported) {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t("%(senderName)s placed a video call.", {
|
|
|
|
senderName: getSenderName(),
|
|
|
|
});
|
2019-11-27 23:03:51 +03:00
|
|
|
} else if (!isVoice && !isSupported) {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t("%(senderName)s placed a video call. (not supported by this browser)", {
|
|
|
|
senderName: getSenderName(),
|
|
|
|
});
|
2015-09-16 16:48:49 +03:00
|
|
|
}
|
2016-09-15 19:01:02 +03:00
|
|
|
}
|
2015-09-16 16:48:49 +03:00
|
|
|
|
2021-06-16 01:52:40 +03:00
|
|
|
function textForThreePidInviteEvent(event: MatrixEvent): () => string | null {
|
2017-09-25 11:48:00 +03:00
|
|
|
const senderName = event.sender ? event.sender.name : event.getSender();
|
2019-03-29 05:38:15 +03:00
|
|
|
|
2019-03-29 20:45:07 +03:00
|
|
|
if (!isValid3pidInvite(event)) {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t('%(senderName)s revoked the invitation for %(targetDisplayName)s to join the room.', {
|
2019-03-29 05:38:15 +03:00
|
|
|
senderName,
|
2021-06-07 06:06:56 +03:00
|
|
|
targetDisplayName: event.getPrevContent().display_name || _t("Someone"),
|
2019-03-29 05:38:15 +03:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t('%(senderName)s sent an invitation to %(targetDisplayName)s to join the room.', {
|
2018-07-03 12:30:08 +03:00
|
|
|
senderName,
|
2017-09-25 11:48:00 +03:00
|
|
|
targetDisplayName: event.getContent().display_name,
|
|
|
|
});
|
2016-09-15 19:01:02 +03:00
|
|
|
}
|
2015-12-17 18:48:14 +03:00
|
|
|
|
2021-06-16 01:52:40 +03:00
|
|
|
function textForHistoryVisibilityEvent(event: MatrixEvent): () => string | null {
|
2017-09-17 15:28:17 +03:00
|
|
|
const senderName = event.sender ? event.sender.name : event.getSender();
|
|
|
|
switch (event.getContent().history_visibility) {
|
|
|
|
case 'invited':
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t('%(senderName)s made future room history visible to all room members, '
|
2021-06-29 15:11:58 +03:00
|
|
|
+ 'from the point they are invited.', { senderName });
|
2017-09-17 15:28:17 +03:00
|
|
|
case 'joined':
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t('%(senderName)s made future room history visible to all room members, '
|
2021-06-29 15:11:58 +03:00
|
|
|
+ 'from the point they joined.', { senderName });
|
2017-09-17 15:28:17 +03:00
|
|
|
case 'shared':
|
2021-06-29 15:11:58 +03:00
|
|
|
return () => _t('%(senderName)s made future room history visible to all room members.', { senderName });
|
2017-09-17 15:28:17 +03:00
|
|
|
case 'world_readable':
|
2021-06-29 15:11:58 +03:00
|
|
|
return () => _t('%(senderName)s made future room history visible to anyone.', { senderName });
|
2017-09-17 15:28:17 +03:00
|
|
|
default:
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t('%(senderName)s made future room history visible to unknown (%(visibility)s).', {
|
2018-07-03 12:30:08 +03:00
|
|
|
senderName,
|
2017-09-17 15:28:17 +03:00
|
|
|
visibility: event.getContent().history_visibility,
|
|
|
|
});
|
2016-03-16 02:47:40 +03:00
|
|
|
}
|
2016-09-15 19:01:02 +03:00
|
|
|
}
|
|
|
|
|
2017-04-06 19:02:35 +03:00
|
|
|
// Currently will only display a change if a user's power level is changed
|
2021-06-16 01:52:40 +03:00
|
|
|
function textForPowerEvent(event: MatrixEvent): () => string | null {
|
2017-04-06 19:02:35 +03:00
|
|
|
const senderName = event.sender ? event.sender.name : event.getSender();
|
2019-07-12 19:50:13 +03:00
|
|
|
if (!event.getPrevContent() || !event.getPrevContent().users ||
|
|
|
|
!event.getContent() || !event.getContent().users) {
|
2021-06-07 06:06:56 +03:00
|
|
|
return null;
|
2017-04-06 19:02:35 +03:00
|
|
|
}
|
|
|
|
const userDefault = event.getContent().users_default || 0;
|
|
|
|
// Construct set of userIds
|
2017-09-25 11:48:00 +03:00
|
|
|
const users = [];
|
2017-04-06 19:02:35 +03:00
|
|
|
Object.keys(event.getContent().users).forEach(
|
|
|
|
(userId) => {
|
|
|
|
if (users.indexOf(userId) === -1) users.push(userId);
|
2017-09-25 11:48:00 +03:00
|
|
|
},
|
2017-04-06 19:02:35 +03:00
|
|
|
);
|
|
|
|
Object.keys(event.getPrevContent().users).forEach(
|
|
|
|
(userId) => {
|
|
|
|
if (users.indexOf(userId) === -1) users.push(userId);
|
2017-09-25 11:48:00 +03:00
|
|
|
},
|
2017-04-06 19:02:35 +03:00
|
|
|
);
|
2021-06-07 06:06:56 +03:00
|
|
|
const diffs = [];
|
2017-04-06 19:02:35 +03:00
|
|
|
users.forEach((userId) => {
|
|
|
|
// Previous power level
|
|
|
|
const from = event.getPrevContent().users[userId];
|
|
|
|
// Current power level
|
|
|
|
const to = event.getContent().users[userId];
|
|
|
|
if (to !== from) {
|
2021-06-07 06:06:56 +03:00
|
|
|
diffs.push({ userId, from, to });
|
2017-04-06 19:02:35 +03:00
|
|
|
}
|
|
|
|
});
|
2021-06-07 06:06:56 +03:00
|
|
|
if (!diffs.length) {
|
|
|
|
return null;
|
2017-04-06 19:02:35 +03:00
|
|
|
}
|
2021-06-07 06:06:56 +03:00
|
|
|
// XXX: This is also surely broken for i18n
|
|
|
|
return () => _t('%(senderName)s changed the power level of %(powerLevelDiffText)s.', {
|
2018-07-03 12:30:08 +03:00
|
|
|
senderName,
|
2021-06-07 06:06:56 +03:00
|
|
|
powerLevelDiffText: diffs.map(diff =>
|
|
|
|
_t('%(userId)s from %(fromPowerLevel)s to %(toPowerLevel)s', {
|
|
|
|
userId: diff.userId,
|
|
|
|
fromPowerLevel: Roles.textualPowerLevel(diff.from, userDefault),
|
|
|
|
toPowerLevel: Roles.textualPowerLevel(diff.to, userDefault),
|
|
|
|
}),
|
|
|
|
).join(", "),
|
2017-05-27 17:52:24 +03:00
|
|
|
});
|
2017-04-06 19:02:35 +03:00
|
|
|
}
|
|
|
|
|
2021-06-24 17:58:56 +03:00
|
|
|
const onPinnedMessagesClick = (): void => {
|
|
|
|
defaultDispatcher.dispatch<SetRightPanelPhasePayload>({
|
|
|
|
action: Action.SetRightPanelPhase,
|
|
|
|
phase: RightPanelPhases.PinnedMessages,
|
|
|
|
allowClose: false,
|
|
|
|
});
|
2021-06-29 15:11:58 +03:00
|
|
|
};
|
2021-06-24 17:58:56 +03:00
|
|
|
|
2021-06-24 15:02:06 +03:00
|
|
|
function textForPinnedEvent(event: MatrixEvent, allowJSX: boolean): () => string | JSX.Element | null {
|
2021-06-24 11:40:30 +03:00
|
|
|
if (!SettingsStore.getValue("feature_pinning")) return null;
|
2020-01-06 08:04:42 +03:00
|
|
|
const senderName = event.sender ? event.sender.name : event.getSender();
|
2021-06-24 11:40:30 +03:00
|
|
|
|
2021-06-24 15:02:06 +03:00
|
|
|
if (allowJSX) {
|
|
|
|
return () => (
|
|
|
|
<span>
|
|
|
|
{
|
|
|
|
_t(
|
|
|
|
"%(senderName)s changed the <a>pinned messages</a> for the room.",
|
|
|
|
{ senderName },
|
|
|
|
{ "a": (sub) => <a onClick={onPinnedMessagesClick}> { sub } </a> },
|
|
|
|
)
|
|
|
|
}
|
|
|
|
</span>
|
|
|
|
);
|
|
|
|
}
|
2021-06-24 18:27:53 +03:00
|
|
|
|
2021-06-24 18:23:42 +03:00
|
|
|
return () => _t("%(senderName)s changed the pinned messages for the room.", { senderName });
|
2017-09-28 20:03:13 +03:00
|
|
|
}
|
|
|
|
|
2021-06-16 01:52:40 +03:00
|
|
|
function textForWidgetEvent(event: MatrixEvent): () => string | null {
|
2017-08-28 10:19:39 +03:00
|
|
|
const senderName = event.getSender();
|
2021-06-29 15:11:58 +03:00
|
|
|
const { name: prevName, type: prevType, url: prevUrl } = event.getPrevContent();
|
|
|
|
const { name, type, url } = event.getContent() || {};
|
2017-08-28 10:19:39 +03:00
|
|
|
|
|
|
|
let widgetName = name || prevName || type || prevType || '';
|
2017-08-18 20:02:50 +03:00
|
|
|
// Apply sentence case to widget name
|
|
|
|
if (widgetName && widgetName.length > 0) {
|
2020-11-24 14:27:02 +03:00
|
|
|
widgetName = widgetName[0].toUpperCase() + widgetName.slice(1);
|
2017-08-18 20:02:50 +03:00
|
|
|
}
|
2017-08-16 19:46:20 +03:00
|
|
|
|
2017-08-18 14:04:34 +03:00
|
|
|
// If the widget was removed, its content should be {}, but this is sufficiently
|
|
|
|
// equivalent to that condition.
|
2017-08-18 12:45:43 +03:00
|
|
|
if (url) {
|
2017-08-28 10:19:39 +03:00
|
|
|
if (prevUrl) {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t('%(widgetName)s widget modified by %(senderName)s', {
|
2017-08-28 10:19:39 +03:00
|
|
|
widgetName, senderName,
|
|
|
|
});
|
|
|
|
} else {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t('%(widgetName)s widget added by %(senderName)s', {
|
2017-08-28 10:19:39 +03:00
|
|
|
widgetName, senderName,
|
|
|
|
});
|
|
|
|
}
|
2017-08-16 19:46:20 +03:00
|
|
|
} else {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t('%(widgetName)s widget removed by %(senderName)s', {
|
2017-08-18 20:02:50 +03:00
|
|
|
widgetName, senderName,
|
2017-08-16 19:46:20 +03:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-16 01:52:40 +03:00
|
|
|
function textForWidgetLayoutEvent(event: MatrixEvent): () => string | null {
|
2021-01-19 05:31:11 +03:00
|
|
|
const senderName = event.sender?.name || event.getSender();
|
2021-06-29 15:11:58 +03:00
|
|
|
return () => _t("%(senderName)s has updated the widget layout", { senderName });
|
2021-01-19 05:31:11 +03:00
|
|
|
}
|
|
|
|
|
2021-06-16 01:52:40 +03:00
|
|
|
function textForMjolnirEvent(event: MatrixEvent): () => string | null {
|
2019-12-10 06:36:00 +03:00
|
|
|
const senderName = event.getSender();
|
2021-06-29 15:11:58 +03:00
|
|
|
const { entity: prevEntity } = event.getPrevContent();
|
|
|
|
const { entity, recommendation, reason } = event.getContent();
|
2019-12-10 06:36:00 +03:00
|
|
|
|
|
|
|
// Rule removed
|
|
|
|
if (!entity) {
|
|
|
|
if (USER_RULE_TYPES.includes(event.getType())) {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t("%(senderName)s removed the rule banning users matching %(glob)s",
|
2021-06-29 15:11:58 +03:00
|
|
|
{ senderName, glob: prevEntity });
|
2019-12-10 06:36:00 +03:00
|
|
|
} else if (ROOM_RULE_TYPES.includes(event.getType())) {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t("%(senderName)s removed the rule banning rooms matching %(glob)s",
|
2021-06-29 15:11:58 +03:00
|
|
|
{ senderName, glob: prevEntity });
|
2019-12-10 06:36:00 +03:00
|
|
|
} else if (SERVER_RULE_TYPES.includes(event.getType())) {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t("%(senderName)s removed the rule banning servers matching %(glob)s",
|
2021-06-29 15:11:58 +03:00
|
|
|
{ senderName, glob: prevEntity });
|
2019-12-10 06:36:00 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Unknown type. We'll say something, but we shouldn't end up here.
|
2021-06-29 15:11:58 +03:00
|
|
|
return () => _t("%(senderName)s removed a ban rule matching %(glob)s", { senderName, glob: prevEntity });
|
2019-12-10 06:36:00 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Invalid rule
|
2021-06-29 15:11:58 +03:00
|
|
|
if (!recommendation || !reason) return () => _t(`%(senderName)s updated an invalid ban rule`, { senderName });
|
2019-12-10 06:36:00 +03:00
|
|
|
|
|
|
|
// Rule updated
|
|
|
|
if (entity === prevEntity) {
|
|
|
|
if (USER_RULE_TYPES.includes(event.getType())) {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t("%(senderName)s updated the rule banning users matching %(glob)s for %(reason)s",
|
2021-06-29 15:11:58 +03:00
|
|
|
{ senderName, glob: entity, reason });
|
2019-12-10 06:36:00 +03:00
|
|
|
} else if (ROOM_RULE_TYPES.includes(event.getType())) {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t("%(senderName)s updated the rule banning rooms matching %(glob)s for %(reason)s",
|
2021-06-29 15:11:58 +03:00
|
|
|
{ senderName, glob: entity, reason });
|
2019-12-10 06:36:00 +03:00
|
|
|
} else if (SERVER_RULE_TYPES.includes(event.getType())) {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t("%(senderName)s updated the rule banning servers matching %(glob)s for %(reason)s",
|
2021-06-29 15:11:58 +03:00
|
|
|
{ senderName, glob: entity, reason });
|
2019-12-10 06:36:00 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Unknown type. We'll say something but we shouldn't end up here.
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t("%(senderName)s updated a ban rule matching %(glob)s for %(reason)s",
|
2021-06-29 15:11:58 +03:00
|
|
|
{ senderName, glob: entity, reason });
|
2019-12-10 06:36:00 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// New rule
|
|
|
|
if (!prevEntity) {
|
|
|
|
if (USER_RULE_TYPES.includes(event.getType())) {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t("%(senderName)s created a rule banning users matching %(glob)s for %(reason)s",
|
2021-06-29 15:11:58 +03:00
|
|
|
{ senderName, glob: entity, reason });
|
2019-12-10 06:36:00 +03:00
|
|
|
} else if (ROOM_RULE_TYPES.includes(event.getType())) {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t("%(senderName)s created a rule banning rooms matching %(glob)s for %(reason)s",
|
2021-06-29 15:11:58 +03:00
|
|
|
{ senderName, glob: entity, reason });
|
2019-12-10 06:36:00 +03:00
|
|
|
} else if (SERVER_RULE_TYPES.includes(event.getType())) {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t("%(senderName)s created a rule banning servers matching %(glob)s for %(reason)s",
|
2021-06-29 15:11:58 +03:00
|
|
|
{ senderName, glob: entity, reason });
|
2019-12-10 06:36:00 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Unknown type. We'll say something but we shouldn't end up here.
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t("%(senderName)s created a ban rule matching %(glob)s for %(reason)s",
|
2021-06-29 15:11:58 +03:00
|
|
|
{ senderName, glob: entity, reason });
|
2019-12-10 06:36:00 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// else the entity !== prevEntity - count as a removal & add
|
|
|
|
if (USER_RULE_TYPES.includes(event.getType())) {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t(
|
2021-04-29 20:57:02 +03:00
|
|
|
"%(senderName)s changed a rule that was banning users matching %(oldGlob)s to matching " +
|
2019-12-10 06:36:00 +03:00
|
|
|
"%(newGlob)s for %(reason)s",
|
2021-06-29 15:11:58 +03:00
|
|
|
{ senderName, oldGlob: prevEntity, newGlob: entity, reason },
|
2021-04-29 20:57:02 +03:00
|
|
|
);
|
2019-12-10 06:36:00 +03:00
|
|
|
} else if (ROOM_RULE_TYPES.includes(event.getType())) {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t(
|
2021-04-29 20:57:02 +03:00
|
|
|
"%(senderName)s changed a rule that was banning rooms matching %(oldGlob)s to matching " +
|
2019-12-10 06:36:00 +03:00
|
|
|
"%(newGlob)s for %(reason)s",
|
2021-06-29 15:11:58 +03:00
|
|
|
{ senderName, oldGlob: prevEntity, newGlob: entity, reason },
|
2021-04-29 20:57:02 +03:00
|
|
|
);
|
2019-12-10 06:36:00 +03:00
|
|
|
} else if (SERVER_RULE_TYPES.includes(event.getType())) {
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t(
|
2021-04-29 20:57:02 +03:00
|
|
|
"%(senderName)s changed a rule that was banning servers matching %(oldGlob)s to matching " +
|
2019-12-10 06:36:00 +03:00
|
|
|
"%(newGlob)s for %(reason)s",
|
2021-06-29 15:11:58 +03:00
|
|
|
{ senderName, oldGlob: prevEntity, newGlob: entity, reason },
|
2021-04-29 20:57:02 +03:00
|
|
|
);
|
2019-12-10 06:36:00 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Unknown type. We'll say something but we shouldn't end up here.
|
2021-06-07 06:06:56 +03:00
|
|
|
return () => _t("%(senderName)s updated a ban rule that was matching %(oldGlob)s to matching %(newGlob)s " +
|
2021-06-29 15:11:58 +03:00
|
|
|
"for %(reason)s", { senderName, oldGlob: prevEntity, newGlob: entity, reason });
|
2019-12-10 06:36:00 +03:00
|
|
|
}
|
|
|
|
|
2021-06-09 04:58:48 +03:00
|
|
|
interface IHandlers {
|
2021-06-30 05:13:18 +03:00
|
|
|
[type: string]:
|
|
|
|
(ev: MatrixEvent, allowJSX: boolean, showHiddenEvents?: boolean) =>
|
|
|
|
(() => string | JSX.Element | null);
|
2021-06-09 04:58:48 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
const handlers: IHandlers = {
|
2015-09-16 16:48:49 +03:00
|
|
|
'm.room.message': textForMessageEvent,
|
2017-09-25 11:48:00 +03:00
|
|
|
'm.call.invite': textForCallInviteEvent,
|
|
|
|
'm.call.answer': textForCallAnswerEvent,
|
|
|
|
'm.call.hangup': textForCallHangupEvent,
|
2020-10-15 16:54:03 +03:00
|
|
|
'm.call.reject': textForCallRejectEvent,
|
2017-10-06 14:07:38 +03:00
|
|
|
};
|
|
|
|
|
2021-06-09 04:58:48 +03:00
|
|
|
const stateHandlers: IHandlers = {
|
2018-09-20 03:07:01 +03:00
|
|
|
'm.room.canonical_alias': textForCanonicalAliasEvent,
|
2017-10-06 14:07:38 +03:00
|
|
|
'm.room.name': textForRoomNameEvent,
|
|
|
|
'm.room.topic': textForTopicEvent,
|
|
|
|
'm.room.member': textForMemberEvent,
|
2016-03-16 02:47:40 +03:00
|
|
|
'm.room.third_party_invite': textForThreePidInviteEvent,
|
|
|
|
'm.room.history_visibility': textForHistoryVisibilityEvent,
|
2017-04-06 19:02:35 +03:00
|
|
|
'm.room.power_levels': textForPowerEvent,
|
2017-09-28 20:03:13 +03:00
|
|
|
'm.room.pinned_events': textForPinnedEvent,
|
2018-07-06 12:18:31 +03:00
|
|
|
'm.room.server_acl': textForServerACLEvent,
|
2019-01-11 01:15:45 +03:00
|
|
|
'm.room.tombstone': textForTombstoneEvent,
|
2019-02-07 22:14:58 +03:00
|
|
|
'm.room.join_rules': textForJoinRulesEvent,
|
|
|
|
'm.room.guest_access': textForGuestAccessEvent,
|
2019-02-22 03:03:15 +03:00
|
|
|
'm.room.related_groups': textForRelatedGroupsEvent,
|
2017-08-16 19:46:20 +03:00
|
|
|
|
2020-08-03 18:02:26 +03:00
|
|
|
// TODO: Enable support for m.widget event type (https://github.com/vector-im/element-web/issues/13111)
|
2017-08-16 19:46:20 +03:00
|
|
|
'im.vector.modular.widgets': textForWidgetEvent,
|
2021-01-19 05:31:11 +03:00
|
|
|
[WIDGET_LAYOUT_EVENT_TYPE]: textForWidgetLayoutEvent,
|
2015-09-16 16:48:49 +03:00
|
|
|
};
|
|
|
|
|
2019-12-10 06:36:00 +03:00
|
|
|
// Add all the Mjolnir stuff to the renderer
|
|
|
|
for (const evType of ALL_RULE_TYPES) {
|
|
|
|
stateHandlers[evType] = textForMjolnirEvent;
|
|
|
|
}
|
|
|
|
|
2021-06-16 01:59:42 +03:00
|
|
|
/**
|
|
|
|
* Determines whether the given event has text to display.
|
|
|
|
* @param ev The event
|
|
|
|
* @param showHiddenEvents An optional cached setting value for showHiddenEventsInTimeline
|
|
|
|
* to avoid hitting the settings store
|
|
|
|
*/
|
|
|
|
export function hasText(ev: MatrixEvent, showHiddenEvents?: boolean): boolean {
|
2021-06-07 06:06:56 +03:00
|
|
|
const handler = (ev.isState() ? stateHandlers : handlers)[ev.getType()];
|
2021-06-30 05:13:18 +03:00
|
|
|
return Boolean(handler?.(ev, false, showHiddenEvents));
|
2021-06-07 06:06:56 +03:00
|
|
|
}
|
|
|
|
|
2021-06-16 01:59:42 +03:00
|
|
|
/**
|
|
|
|
* Gets the textual content of the given event.
|
|
|
|
* @param ev The event
|
2021-06-30 05:13:18 +03:00
|
|
|
* @param allowJSX Whether to output rich JSX content
|
2021-06-16 01:59:42 +03:00
|
|
|
* @param showHiddenEvents An optional cached setting value for showHiddenEventsInTimeline
|
|
|
|
* to avoid hitting the settings store
|
|
|
|
*/
|
2021-06-30 05:13:18 +03:00
|
|
|
export function textForEvent(
|
|
|
|
ev: MatrixEvent, allowJSX: boolean = false, showHiddenEvents?: boolean
|
|
|
|
): string | JSX.Element {
|
2019-12-20 03:45:24 +03:00
|
|
|
const handler = (ev.isState() ? stateHandlers : handlers)[ev.getType()];
|
2021-06-30 05:13:18 +03:00
|
|
|
return handler?.(ev, allowJSX, showHiddenEvents)?.() ?? '';
|
2019-12-20 03:45:24 +03:00
|
|
|
}
|