matrix-react-sdk
Version:
SDK for matrix.org using React
390 lines (384 loc) • 76.8 kB
JavaScript
"use strict";
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = exports.contextMenuBelow = exports.RoomTile = void 0;
var _extends2 = _interopRequireDefault(require("@babel/runtime/helpers/extends"));
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _react = _interopRequireWildcard(require("react"));
var _matrix = require("matrix-js-sdk/src/matrix");
var _types = require("matrix-js-sdk/src/types");
var _classnames = _interopRequireDefault(require("classnames"));
var _RovingTabIndex = require("../../../accessibility/RovingTabIndex");
var _AccessibleButton = _interopRequireDefault(require("../../views/elements/AccessibleButton"));
var _dispatcher = _interopRequireDefault(require("../../../dispatcher/dispatcher"));
var _actions = require("../../../dispatcher/actions");
var _languageHandler = require("../../../languageHandler");
var _ContextMenu = require("../../structures/ContextMenu");
var _models = require("../../../stores/room-list/models");
var _MessagePreviewStore = require("../../../stores/room-list/MessagePreviewStore");
var _DecoratedRoomAvatar = _interopRequireDefault(require("../avatars/DecoratedRoomAvatar"));
var _RoomNotifs = require("../../../RoomNotifs");
var _MatrixClientPeg = require("../../../MatrixClientPeg");
var _RoomNotificationContextMenu = require("../context_menus/RoomNotificationContextMenu");
var _NotificationBadge = _interopRequireDefault(require("./NotificationBadge"));
var _RoomNotificationStateStore = require("../../../stores/notifications/RoomNotificationStateStore");
var _NotificationState = require("../../../stores/notifications/NotificationState");
var _EchoChamber = require("../../../stores/local-echo/EchoChamber");
var _RoomEchoChamber = require("../../../stores/local-echo/RoomEchoChamber");
var _GenericEchoChamber = require("../../../stores/local-echo/GenericEchoChamber");
var _PosthogTrackers = _interopRequireDefault(require("../../../PosthogTrackers"));
var _KeyboardShortcuts = require("../../../accessibility/KeyboardShortcuts");
var _KeyBindingsManager = require("../../../KeyBindingsManager");
var _RoomGeneralContextMenu = require("../context_menus/RoomGeneralContextMenu");
var _CallStore = require("../../../stores/CallStore");
var _SDKContext = require("../../../contexts/SDKContext");
var _voiceBroadcast = require("../../../voice-broadcast");
var _RoomTileSubtitle = require("./RoomTileSubtitle");
var _UIComponents = require("../../../customisations/helpers/UIComponents");
var _UIFeature = require("../../../settings/UIFeature");
var _membership = require("../../../utils/membership");
var _SettingsStore = _interopRequireDefault(require("../../../settings/SettingsStore"));
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; }
/*
Copyright 2024 New Vector Ltd.
Copyright 2018 Michael Telatynski <7t3chguy@gmail.com>
Copyright 2015-2017 , 2019-2021 The Matrix.org Foundation C.I.C.
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
Please see LICENSE files in the repository root for full details.
*/
const messagePreviewId = roomId => `mx_RoomTile_messagePreview_${roomId}`;
const contextMenuBelow = elementRect => {
// align the context menu's icons with the icon which opened the context menu
const left = elementRect.left + window.scrollX - 9;
const top = elementRect.bottom + window.scrollY + 17;
const chevronFace = _ContextMenu.ChevronFace.None;
return {
left,
top,
chevronFace
};
};
exports.contextMenuBelow = contextMenuBelow;
class RoomTile extends _react.default.PureComponent {
constructor(props) {
super(props);
(0, _defineProperty2.default)(this, "dispatcherRef", void 0);
(0, _defineProperty2.default)(this, "roomTileRef", /*#__PURE__*/(0, _react.createRef)());
(0, _defineProperty2.default)(this, "notificationState", void 0);
(0, _defineProperty2.default)(this, "roomProps", void 0);
(0, _defineProperty2.default)(this, "onRoomNameUpdate", room => {
this.forceUpdate();
});
(0, _defineProperty2.default)(this, "onNotificationUpdate", () => {
this.forceUpdate(); // notification state changed - update
});
(0, _defineProperty2.default)(this, "onRoomPropertyUpdate", property => {
if (property === _RoomEchoChamber.CachedRoomKey.NotificationVolume) this.onNotificationUpdate();
// else ignore - not important for this tile
});
(0, _defineProperty2.default)(this, "onAction", payload => {
if (payload.action === _actions.Action.ViewRoom && payload.room_id === this.props.room.roomId && payload.show_room_tile) {
setTimeout(() => {
this.scrollIntoView();
});
}
});
(0, _defineProperty2.default)(this, "onRoomPreviewChanged", room => {
if (this.props.room && room.roomId === this.props.room.roomId) {
this.generatePreview();
}
});
(0, _defineProperty2.default)(this, "onCallChanged", (call, roomId) => {
if (roomId === this.props.room?.roomId) this.setState({
call
});
});
(0, _defineProperty2.default)(this, "scrollIntoView", () => {
if (!this.roomTileRef.current) return;
this.roomTileRef.current.scrollIntoView({
block: "nearest",
behavior: "auto"
});
});
(0, _defineProperty2.default)(this, "onTileClick", async ev => {
ev.preventDefault();
ev.stopPropagation();
const action = (0, _KeyBindingsManager.getKeyBindingsManager)().getAccessibilityAction(ev);
const clearSearch = [_KeyboardShortcuts.KeyBindingAction.Enter, _KeyboardShortcuts.KeyBindingAction.Space].includes(action);
_dispatcher.default.dispatch({
action: _actions.Action.ViewRoom,
show_room_tile: true,
// make sure the room is visible in the list
room_id: this.props.room.roomId,
clear_search: clearSearch,
metricsTrigger: "RoomList",
metricsViaKeyboard: ev.type !== "click"
});
});
(0, _defineProperty2.default)(this, "onActiveRoomUpdate", isActive => {
this.setState({
selected: isActive
});
});
(0, _defineProperty2.default)(this, "onNotificationsMenuOpenClick", ev => {
ev.preventDefault();
ev.stopPropagation();
const target = ev.target;
this.setState({
notificationsMenuPosition: target.getBoundingClientRect()
});
_PosthogTrackers.default.trackInteraction("WebRoomListRoomTileNotificationsMenu", ev);
});
(0, _defineProperty2.default)(this, "onCloseNotificationsMenu", () => {
this.setState({
notificationsMenuPosition: null
});
});
(0, _defineProperty2.default)(this, "onGeneralMenuOpenClick", ev => {
ev.preventDefault();
ev.stopPropagation();
const target = ev.target;
this.setState({
generalMenuPosition: target.getBoundingClientRect()
});
});
(0, _defineProperty2.default)(this, "onContextMenu", ev => {
// If we don't have a context menu to show, ignore the action.
if (!this.showContextMenu) return;
ev.preventDefault();
ev.stopPropagation();
this.setState({
generalMenuPosition: {
left: ev.clientX,
bottom: ev.clientY
}
});
});
(0, _defineProperty2.default)(this, "onCloseGeneralMenu", () => {
this.setState({
generalMenuPosition: null
});
});
this.state = {
selected: _SDKContext.SdkContextClass.instance.roomViewStore.getRoomId() === this.props.room.roomId,
notificationsMenuPosition: null,
generalMenuPosition: null,
call: _CallStore.CallStore.instance.getCall(this.props.room.roomId),
// generatePreview() will return nothing if the user has previews disabled
messagePreview: null
};
this.generatePreview();
this.notificationState = _RoomNotificationStateStore.RoomNotificationStateStore.instance.getRoomState(this.props.room);
this.roomProps = _EchoChamber.EchoChamber.forRoom(this.props.room);
}
get showContextMenu() {
return this.props.tag !== _models.DefaultTagID.Invite && this.props.room.getMyMembership() !== _types.KnownMembership.Knock && !(0, _membership.isKnockDenied)(this.props.room) && (0, _UIComponents.shouldShowComponent)(_UIFeature.UIComponent.RoomOptionsMenu);
}
get showMessagePreview() {
return !this.props.isMinimized && this.props.showMessagePreview;
}
componentDidUpdate(prevProps, prevState) {
const showMessageChanged = prevProps.showMessagePreview !== this.props.showMessagePreview;
const minimizedChanged = prevProps.isMinimized !== this.props.isMinimized;
if (showMessageChanged || minimizedChanged) {
this.generatePreview();
}
if (prevProps.room?.roomId !== this.props.room?.roomId) {
_MessagePreviewStore.MessagePreviewStore.instance.off(_MessagePreviewStore.MessagePreviewStore.getPreviewChangedEventName(prevProps.room), this.onRoomPreviewChanged);
_MessagePreviewStore.MessagePreviewStore.instance.on(_MessagePreviewStore.MessagePreviewStore.getPreviewChangedEventName(this.props.room), this.onRoomPreviewChanged);
prevProps.room?.off(_matrix.RoomEvent.Name, this.onRoomNameUpdate);
this.props.room?.on(_matrix.RoomEvent.Name, this.onRoomNameUpdate);
}
}
componentDidMount() {
// when we're first rendered (or our sublist is expanded) make sure we are visible if we're active
if (this.state.selected) {
this.scrollIntoView();
}
_SDKContext.SdkContextClass.instance.roomViewStore.addRoomListener(this.props.room.roomId, this.onActiveRoomUpdate);
this.dispatcherRef = _dispatcher.default.register(this.onAction);
_MessagePreviewStore.MessagePreviewStore.instance.on(_MessagePreviewStore.MessagePreviewStore.getPreviewChangedEventName(this.props.room), this.onRoomPreviewChanged);
this.notificationState.on(_NotificationState.NotificationStateEvents.Update, this.onNotificationUpdate);
this.roomProps.on(_GenericEchoChamber.PROPERTY_UPDATED, this.onRoomPropertyUpdate);
this.props.room.on(_matrix.RoomEvent.Name, this.onRoomNameUpdate);
_CallStore.CallStore.instance.on(_CallStore.CallStoreEvent.Call, this.onCallChanged);
// Recalculate the call for this room, since it could've changed between
// construction and mounting
this.setState({
call: _CallStore.CallStore.instance.getCall(this.props.room.roomId)
});
}
componentWillUnmount() {
_SDKContext.SdkContextClass.instance.roomViewStore.removeRoomListener(this.props.room.roomId, this.onActiveRoomUpdate);
_MessagePreviewStore.MessagePreviewStore.instance.off(_MessagePreviewStore.MessagePreviewStore.getPreviewChangedEventName(this.props.room), this.onRoomPreviewChanged);
this.props.room.off(_matrix.RoomEvent.Name, this.onRoomNameUpdate);
if (this.dispatcherRef) _dispatcher.default.unregister(this.dispatcherRef);
this.notificationState.off(_NotificationState.NotificationStateEvents.Update, this.onNotificationUpdate);
this.roomProps.off(_GenericEchoChamber.PROPERTY_UPDATED, this.onRoomPropertyUpdate);
_CallStore.CallStore.instance.off(_CallStore.CallStoreEvent.Call, this.onCallChanged);
}
async generatePreview() {
if (!this.showMessagePreview) {
return;
}
const messagePreview = (await _MessagePreviewStore.MessagePreviewStore.instance.getPreviewForRoom(this.props.room, this.props.tag)) ?? null;
this.setState({
messagePreview
});
}
renderNotificationsMenu(isActive) {
if (_MatrixClientPeg.MatrixClientPeg.safeGet().isGuest() || this.props.tag === _models.DefaultTagID.Archived || !this.showContextMenu || this.props.isMinimized) {
// the menu makes no sense in these cases so do not show one
return null;
}
const state = this.roomProps.notificationVolume;
const classes = (0, _classnames.default)("mx_RoomTile_notificationsButton", {
// Show bell icon for the default case too.
mx_RoomNotificationContextMenu_iconBell: state === _RoomNotifs.RoomNotifState.AllMessages,
mx_RoomNotificationContextMenu_iconBellDot: state === _RoomNotifs.RoomNotifState.AllMessagesLoud,
mx_RoomNotificationContextMenu_iconBellMentions: state === _RoomNotifs.RoomNotifState.MentionsOnly,
mx_RoomNotificationContextMenu_iconBellCrossed: state === _RoomNotifs.RoomNotifState.Mute,
// Only show the icon by default if the room is overridden to muted.
// TODO: [FTUE Notifications] Probably need to detect global mute state
mx_RoomTile_notificationsButton_show: state === _RoomNotifs.RoomNotifState.Mute
});
return /*#__PURE__*/_react.default.createElement(_react.default.Fragment, null, /*#__PURE__*/_react.default.createElement(_ContextMenu.ContextMenuTooltipButton, {
className: classes,
onClick: this.onNotificationsMenuOpenClick,
title: (0, _languageHandler._t)("room_list|notification_options"),
isExpanded: !!this.state.notificationsMenuPosition,
tabIndex: isActive ? 0 : -1
}), this.state.notificationsMenuPosition && /*#__PURE__*/_react.default.createElement(_RoomNotificationContextMenu.RoomNotificationContextMenu, (0, _extends2.default)({}, contextMenuBelow(this.state.notificationsMenuPosition), {
onFinished: this.onCloseNotificationsMenu,
room: this.props.room
})));
}
renderGeneralMenu() {
if (!this.showContextMenu) return null; // no menu to show
return /*#__PURE__*/_react.default.createElement(_react.default.Fragment, null, /*#__PURE__*/_react.default.createElement(_ContextMenu.ContextMenuTooltipButton, {
className: "mx_RoomTile_menuButton",
onClick: this.onGeneralMenuOpenClick,
title: (0, _languageHandler._t)("room|context_menu|title"),
isExpanded: !!this.state.generalMenuPosition
}), this.state.generalMenuPosition && /*#__PURE__*/_react.default.createElement(_RoomGeneralContextMenu.RoomGeneralContextMenu, (0, _extends2.default)({}, contextMenuBelow(this.state.generalMenuPosition), {
onFinished: this.onCloseGeneralMenu,
room: this.props.room,
onPostFavoriteClick: ev => _PosthogTrackers.default.trackInteraction("WebRoomListRoomTileContextMenuFavouriteToggle", ev),
onPostInviteClick: ev => _PosthogTrackers.default.trackInteraction("WebRoomListRoomTileContextMenuInviteItem", ev),
onPostSettingsClick: ev => _PosthogTrackers.default.trackInteraction("WebRoomListRoomTileContextMenuSettingsItem", ev),
onPostLeaveClick: ev => _PosthogTrackers.default.trackInteraction("WebRoomListRoomTileContextMenuLeaveItem", ev),
onPostMarkAsReadClick: ev => _PosthogTrackers.default.trackInteraction("WebRoomListRoomTileContextMenuMarkRead", ev),
onPostMarkAsUnreadClick: ev => _PosthogTrackers.default.trackInteraction("WebRoomListRoomTileContextMenuMarkUnread", ev)
})));
}
/**
* RoomTile has a subtile if one of the following applies:
* - there is a call
* - there is a live voice broadcast
* - message previews are enabled and there is a previewable message
*/
get shouldRenderSubtitle() {
return !!this.state.call || this.props.hasLiveVoiceBroadcast || this.props.showMessagePreview && !!this.state.messagePreview;
}
render() {
const classes = (0, _classnames.default)({
mx_RoomTile: true,
mx_RoomTile_sticky: _SettingsStore.default.getValue("feature_ask_to_join") && (this.props.room.getMyMembership() === _types.KnownMembership.Knock || (0, _membership.isKnockDenied)(this.props.room)),
mx_RoomTile_selected: this.state.selected,
mx_RoomTile_hasMenuOpen: !!(this.state.generalMenuPosition || this.state.notificationsMenuPosition),
mx_RoomTile_minimized: this.props.isMinimized
});
let name = this.props.room.name;
if (typeof name !== "string") name = "";
name = name.replace(":", ":\u200b"); // add a zero-width space to allow linewrapping after the colon
let badge;
if (!this.props.isMinimized && this.notificationState) {
// aria-hidden because we summarise the unread count/highlight status in a manual aria-label below
badge = /*#__PURE__*/_react.default.createElement("div", {
className: "mx_RoomTile_badgeContainer",
"aria-hidden": "true"
}, /*#__PURE__*/_react.default.createElement(_NotificationBadge.default, {
notification: this.notificationState,
roomId: this.props.room.roomId
}));
}
const subtitle = this.shouldRenderSubtitle ? /*#__PURE__*/_react.default.createElement(_RoomTileSubtitle.RoomTileSubtitle, {
call: this.state.call,
hasLiveVoiceBroadcast: this.props.hasLiveVoiceBroadcast,
messagePreview: this.state.messagePreview,
roomId: this.props.room.roomId,
showMessagePreview: this.props.showMessagePreview
}) : null;
const titleClasses = (0, _classnames.default)({
mx_RoomTile_title: true,
mx_RoomTile_titleWithSubtitle: !!subtitle,
mx_RoomTile_titleHasUnreadEvents: this.notificationState.isUnread
});
const titleContainer = this.props.isMinimized ? null : /*#__PURE__*/_react.default.createElement("div", {
className: "mx_RoomTile_titleContainer"
}, /*#__PURE__*/_react.default.createElement("div", {
title: name,
className: titleClasses,
tabIndex: -1
}, /*#__PURE__*/_react.default.createElement("span", {
dir: "auto"
}, name)), subtitle);
let ariaLabel = name;
// The following labels are written in such a fashion to increase screen reader efficiency (speed).
if (this.props.tag === _models.DefaultTagID.Invite) {
// append nothing
} else if (this.notificationState.hasMentions) {
ariaLabel += " " + (0, _languageHandler._t)("a11y|n_unread_messages_mentions", {
count: this.notificationState.count
});
} else if (this.notificationState.hasUnreadCount) {
ariaLabel += " " + (0, _languageHandler._t)("a11y|n_unread_messages", {
count: this.notificationState.count
});
} else if (this.notificationState.isUnread) {
ariaLabel += " " + (0, _languageHandler._t)("a11y|unread_messages");
}
let ariaDescribedBy;
if (this.showMessagePreview) {
ariaDescribedBy = messagePreviewId(this.props.room.roomId);
}
return /*#__PURE__*/_react.default.createElement(_react.default.Fragment, null, /*#__PURE__*/_react.default.createElement(_RovingTabIndex.RovingTabIndexWrapper, {
inputRef: this.roomTileRef
}, ({
onFocus,
isActive,
ref
}) => /*#__PURE__*/_react.default.createElement(_AccessibleButton.default, {
onFocus: onFocus,
tabIndex: isActive ? 0 : -1,
ref: ref,
className: classes,
onClick: this.onTileClick,
onContextMenu: this.onContextMenu,
role: "treeitem",
"aria-label": ariaLabel,
"aria-selected": this.state.selected,
"aria-describedby": ariaDescribedBy,
title: this.props.isMinimized && !this.state.generalMenuPosition ? name : undefined
}, /*#__PURE__*/_react.default.createElement(_DecoratedRoomAvatar.default, {
room: this.props.room,
size: "32px",
displayBadge: this.props.isMinimized,
tooltipProps: {
tabIndex: isActive ? 0 : -1
}
}), titleContainer, badge, this.renderGeneralMenu(), this.renderNotificationsMenu(isActive))));
}
}
exports.RoomTile = RoomTile;
const RoomTileHOC = props => {
const hasLiveVoiceBroadcast = (0, _voiceBroadcast.useHasRoomLiveVoiceBroadcast)(props.room);
return /*#__PURE__*/_react.default.createElement(RoomTile, (0, _extends2.default)({}, props, {
hasLiveVoiceBroadcast: hasLiveVoiceBroadcast
}));
};
var _default = exports.default = RoomTileHOC;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireWildcard","require","_matrix","_types","_classnames","_interopRequireDefault","_RovingTabIndex","_AccessibleButton","_dispatcher","_actions","_languageHandler","_ContextMenu","_models","_MessagePreviewStore","_DecoratedRoomAvatar","_RoomNotifs","_MatrixClientPeg","_RoomNotificationContextMenu","_NotificationBadge","_RoomNotificationStateStore","_NotificationState","_EchoChamber","_RoomEchoChamber","_GenericEchoChamber","_PosthogTrackers","_KeyboardShortcuts","_KeyBindingsManager","_RoomGeneralContextMenu","_CallStore","_SDKContext","_voiceBroadcast","_RoomTileSubtitle","_UIComponents","_UIFeature","_membership","_SettingsStore","_getRequireWildcardCache","e","WeakMap","r","t","__esModule","default","has","get","n","__proto__","a","Object","defineProperty","getOwnPropertyDescriptor","u","hasOwnProperty","call","i","set","messagePreviewId","roomId","contextMenuBelow","elementRect","left","window","scrollX","top","bottom","scrollY","chevronFace","ChevronFace","None","exports","RoomTile","React","PureComponent","constructor","props","_defineProperty2","createRef","room","forceUpdate","property","CachedRoomKey","NotificationVolume","onNotificationUpdate","payload","action","Action","ViewRoom","room_id","show_room_tile","setTimeout","scrollIntoView","generatePreview","setState","roomTileRef","current","block","behavior","ev","preventDefault","stopPropagation","getKeyBindingsManager","getAccessibilityAction","clearSearch","KeyBindingAction","Enter","Space","includes","defaultDispatcher","dispatch","clear_search","metricsTrigger","metricsViaKeyboard","type","isActive","selected","target","notificationsMenuPosition","getBoundingClientRect","PosthogTrackers","trackInteraction","generalMenuPosition","showContextMenu","clientX","clientY","state","SdkContextClass","instance","roomViewStore","getRoomId","CallStore","getCall","messagePreview","notificationState","RoomNotificationStateStore","getRoomState","roomProps","EchoChamber","forRoom","tag","DefaultTagID","Invite","getMyMembership","KnownMembership","Knock","isKnockDenied","shouldShowComponent","UIComponent","RoomOptionsMenu","showMessagePreview","isMinimized","componentDidUpdate","prevProps","prevState","showMessageChanged","minimizedChanged","MessagePreviewStore","off","getPreviewChangedEventName","onRoomPreviewChanged","on","RoomEvent","Name","onRoomNameUpdate","componentDidMount","addRoomListener","onActiveRoomUpdate","dispatcherRef","register","onAction","NotificationStateEvents","Update","PROPERTY_UPDATED","onRoomPropertyUpdate","CallStoreEvent","Call","onCallChanged","componentWillUnmount","removeRoomListener","unregister","getPreviewForRoom","renderNotificationsMenu","MatrixClientPeg","safeGet","isGuest","Archived","notificationVolume","classes","classNames","mx_RoomNotificationContextMenu_iconBell","RoomNotifState","AllMessages","mx_RoomNotificationContextMenu_iconBellDot","AllMessagesLoud","mx_RoomNotificationContextMenu_iconBellMentions","MentionsOnly","mx_RoomNotificationContextMenu_iconBellCrossed","Mute","mx_RoomTile_notificationsButton_show","createElement","Fragment","ContextMenuTooltipButton","className","onClick","onNotificationsMenuOpenClick","title","_t","isExpanded","tabIndex","RoomNotificationContextMenu","_extends2","onFinished","onCloseNotificationsMenu","renderGeneralMenu","onGeneralMenuOpenClick","RoomGeneralContextMenu","onCloseGeneralMenu","onPostFavoriteClick","onPostInviteClick","onPostSettingsClick","onPostLeaveClick","onPostMarkAsReadClick","onPostMarkAsUnreadClick","shouldRenderSubtitle","hasLiveVoiceBroadcast","render","mx_RoomTile","mx_RoomTile_sticky","SettingsStore","getValue","mx_RoomTile_selected","mx_RoomTile_hasMenuOpen","mx_RoomTile_minimized","name","replace","badge","notification","subtitle","RoomTileSubtitle","titleClasses","mx_RoomTile_title","mx_RoomTile_titleWithSubtitle","mx_RoomTile_titleHasUnreadEvents","isUnread","titleContainer","dir","ariaLabel","hasMentions","count","hasUnreadCount","ariaDescribedBy","RovingTabIndexWrapper","inputRef","onFocus","ref","onTileClick","onContextMenu","role","undefined","size","displayBadge","tooltipProps","RoomTileHOC","useHasRoomLiveVoiceBroadcast","_default"],"sources":["../../../../src/components/views/rooms/RoomTile.tsx"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2018 Michael Telatynski <7t3chguy@gmail.com>\nCopyright 2015-2017 , 2019-2021 The Matrix.org Foundation C.I.C.\n\nSPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only\nPlease see LICENSE files in the repository root for full details.\n*/\n\nimport React, { createRef } from \"react\";\nimport { Room, RoomEvent } from \"matrix-js-sdk/src/matrix\";\nimport { KnownMembership } from \"matrix-js-sdk/src/types\";\nimport classNames from \"classnames\";\n\nimport type { Call } from \"../../../models/Call\";\nimport { RovingTabIndexWrapper } from \"../../../accessibility/RovingTabIndex\";\nimport AccessibleButton, { ButtonEvent } from \"../../views/elements/AccessibleButton\";\nimport defaultDispatcher from \"../../../dispatcher/dispatcher\";\nimport { Action } from \"../../../dispatcher/actions\";\nimport { _t } from \"../../../languageHandler\";\nimport { ChevronFace, ContextMenuTooltipButton, MenuProps } from \"../../structures/ContextMenu\";\nimport { DefaultTagID, TagID } from \"../../../stores/room-list/models\";\nimport { MessagePreview, MessagePreviewStore } from \"../../../stores/room-list/MessagePreviewStore\";\nimport DecoratedRoomAvatar from \"../avatars/DecoratedRoomAvatar\";\nimport { RoomNotifState } from \"../../../RoomNotifs\";\nimport { MatrixClientPeg } from \"../../../MatrixClientPeg\";\nimport { RoomNotificationContextMenu } from \"../context_menus/RoomNotificationContextMenu\";\nimport NotificationBadge from \"./NotificationBadge\";\nimport { ActionPayload } from \"../../../dispatcher/payloads\";\nimport { RoomNotificationStateStore } from \"../../../stores/notifications/RoomNotificationStateStore\";\nimport { NotificationState, NotificationStateEvents } from \"../../../stores/notifications/NotificationState\";\nimport { EchoChamber } from \"../../../stores/local-echo/EchoChamber\";\nimport { CachedRoomKey, RoomEchoChamber } from \"../../../stores/local-echo/RoomEchoChamber\";\nimport { PROPERTY_UPDATED } from \"../../../stores/local-echo/GenericEchoChamber\";\nimport PosthogTrackers from \"../../../PosthogTrackers\";\nimport { ViewRoomPayload } from \"../../../dispatcher/payloads/ViewRoomPayload\";\nimport { KeyBindingAction } from \"../../../accessibility/KeyboardShortcuts\";\nimport { getKeyBindingsManager } from \"../../../KeyBindingsManager\";\nimport { RoomGeneralContextMenu } from \"../context_menus/RoomGeneralContextMenu\";\nimport { CallStore, CallStoreEvent } from \"../../../stores/CallStore\";\nimport { SdkContextClass } from \"../../../contexts/SDKContext\";\nimport { useHasRoomLiveVoiceBroadcast } from \"../../../voice-broadcast\";\nimport { RoomTileSubtitle } from \"./RoomTileSubtitle\";\nimport { shouldShowComponent } from \"../../../customisations/helpers/UIComponents\";\nimport { UIComponent } from \"../../../settings/UIFeature\";\nimport { isKnockDenied } from \"../../../utils/membership\";\nimport SettingsStore from \"../../../settings/SettingsStore\";\n\ninterface Props {\n    room: Room;\n    showMessagePreview: boolean;\n    isMinimized: boolean;\n    tag: TagID;\n}\n\ninterface ClassProps extends Props {\n    hasLiveVoiceBroadcast: boolean;\n}\n\ntype PartialDOMRect = Pick<DOMRect, \"left\" | \"bottom\">;\n\ninterface State {\n    selected: boolean;\n    notificationsMenuPosition: PartialDOMRect | null;\n    generalMenuPosition: PartialDOMRect | null;\n    call: Call | null;\n    messagePreview: MessagePreview | null;\n}\n\nconst messagePreviewId = (roomId: string): string => `mx_RoomTile_messagePreview_${roomId}`;\n\nexport const contextMenuBelow = (elementRect: PartialDOMRect): MenuProps => {\n    // align the context menu's icons with the icon which opened the context menu\n    const left = elementRect.left + window.scrollX - 9;\n    const top = elementRect.bottom + window.scrollY + 17;\n    const chevronFace = ChevronFace.None;\n    return { left, top, chevronFace };\n};\n\nexport class RoomTile extends React.PureComponent<ClassProps, State> {\n    private dispatcherRef?: string;\n    private roomTileRef = createRef<HTMLDivElement>();\n    private notificationState: NotificationState;\n    private roomProps: RoomEchoChamber;\n\n    public constructor(props: ClassProps) {\n        super(props);\n\n        this.state = {\n            selected: SdkContextClass.instance.roomViewStore.getRoomId() === this.props.room.roomId,\n            notificationsMenuPosition: null,\n            generalMenuPosition: null,\n            call: CallStore.instance.getCall(this.props.room.roomId),\n            // generatePreview() will return nothing if the user has previews disabled\n            messagePreview: null,\n        };\n        this.generatePreview();\n\n        this.notificationState = RoomNotificationStateStore.instance.getRoomState(this.props.room);\n        this.roomProps = EchoChamber.forRoom(this.props.room);\n    }\n\n    private onRoomNameUpdate = (room: Room): void => {\n        this.forceUpdate();\n    };\n\n    private onNotificationUpdate = (): void => {\n        this.forceUpdate(); // notification state changed - update\n    };\n\n    private onRoomPropertyUpdate = (property: CachedRoomKey): void => {\n        if (property === CachedRoomKey.NotificationVolume) this.onNotificationUpdate();\n        // else ignore - not important for this tile\n    };\n\n    private get showContextMenu(): boolean {\n        return (\n            this.props.tag !== DefaultTagID.Invite &&\n            this.props.room.getMyMembership() !== KnownMembership.Knock &&\n            !isKnockDenied(this.props.room) &&\n            shouldShowComponent(UIComponent.RoomOptionsMenu)\n        );\n    }\n\n    private get showMessagePreview(): boolean {\n        return !this.props.isMinimized && this.props.showMessagePreview;\n    }\n\n    public componentDidUpdate(prevProps: Readonly<Props>, prevState: Readonly<State>): void {\n        const showMessageChanged = prevProps.showMessagePreview !== this.props.showMessagePreview;\n        const minimizedChanged = prevProps.isMinimized !== this.props.isMinimized;\n        if (showMessageChanged || minimizedChanged) {\n            this.generatePreview();\n        }\n        if (prevProps.room?.roomId !== this.props.room?.roomId) {\n            MessagePreviewStore.instance.off(\n                MessagePreviewStore.getPreviewChangedEventName(prevProps.room),\n                this.onRoomPreviewChanged,\n            );\n            MessagePreviewStore.instance.on(\n                MessagePreviewStore.getPreviewChangedEventName(this.props.room),\n                this.onRoomPreviewChanged,\n            );\n            prevProps.room?.off(RoomEvent.Name, this.onRoomNameUpdate);\n            this.props.room?.on(RoomEvent.Name, this.onRoomNameUpdate);\n        }\n    }\n\n    public componentDidMount(): void {\n        // when we're first rendered (or our sublist is expanded) make sure we are visible if we're active\n        if (this.state.selected) {\n            this.scrollIntoView();\n        }\n\n        SdkContextClass.instance.roomViewStore.addRoomListener(this.props.room.roomId, this.onActiveRoomUpdate);\n        this.dispatcherRef = defaultDispatcher.register(this.onAction);\n        MessagePreviewStore.instance.on(\n            MessagePreviewStore.getPreviewChangedEventName(this.props.room),\n            this.onRoomPreviewChanged,\n        );\n        this.notificationState.on(NotificationStateEvents.Update, this.onNotificationUpdate);\n        this.roomProps.on(PROPERTY_UPDATED, this.onRoomPropertyUpdate);\n        this.props.room.on(RoomEvent.Name, this.onRoomNameUpdate);\n        CallStore.instance.on(CallStoreEvent.Call, this.onCallChanged);\n\n        // Recalculate the call for this room, since it could've changed between\n        // construction and mounting\n        this.setState({ call: CallStore.instance.getCall(this.props.room.roomId) });\n    }\n\n    public componentWillUnmount(): void {\n        SdkContextClass.instance.roomViewStore.removeRoomListener(this.props.room.roomId, this.onActiveRoomUpdate);\n        MessagePreviewStore.instance.off(\n            MessagePreviewStore.getPreviewChangedEventName(this.props.room),\n            this.onRoomPreviewChanged,\n        );\n        this.props.room.off(RoomEvent.Name, this.onRoomNameUpdate);\n        if (this.dispatcherRef) defaultDispatcher.unregister(this.dispatcherRef);\n        this.notificationState.off(NotificationStateEvents.Update, this.onNotificationUpdate);\n        this.roomProps.off(PROPERTY_UPDATED, this.onRoomPropertyUpdate);\n        CallStore.instance.off(CallStoreEvent.Call, this.onCallChanged);\n    }\n\n    private onAction = (payload: ActionPayload): void => {\n        if (\n            payload.action === Action.ViewRoom &&\n            payload.room_id === this.props.room.roomId &&\n            payload.show_room_tile\n        ) {\n            setTimeout(() => {\n                this.scrollIntoView();\n            });\n        }\n    };\n\n    private onRoomPreviewChanged = (room: Room): void => {\n        if (this.props.room && room.roomId === this.props.room.roomId) {\n            this.generatePreview();\n        }\n    };\n\n    private onCallChanged = (call: Call, roomId: string): void => {\n        if (roomId === this.props.room?.roomId) this.setState({ call });\n    };\n\n    private async generatePreview(): Promise<void> {\n        if (!this.showMessagePreview) {\n            return;\n        }\n\n        const messagePreview =\n            (await MessagePreviewStore.instance.getPreviewForRoom(this.props.room, this.props.tag)) ?? null;\n        this.setState({ messagePreview });\n    }\n\n    private scrollIntoView = (): void => {\n        if (!this.roomTileRef.current) return;\n        this.roomTileRef.current.scrollIntoView({\n            block: \"nearest\",\n            behavior: \"auto\",\n        });\n    };\n\n    private onTileClick = async (ev: ButtonEvent): Promise<void> => {\n        ev.preventDefault();\n        ev.stopPropagation();\n\n        const action = getKeyBindingsManager().getAccessibilityAction(ev as React.KeyboardEvent);\n        const clearSearch = ([KeyBindingAction.Enter, KeyBindingAction.Space] as Array<string | undefined>).includes(\n            action,\n        );\n\n        defaultDispatcher.dispatch<ViewRoomPayload>({\n            action: Action.ViewRoom,\n            show_room_tile: true, // make sure the room is visible in the list\n            room_id: this.props.room.roomId,\n            clear_search: clearSearch,\n            metricsTrigger: \"RoomList\",\n            metricsViaKeyboard: ev.type !== \"click\",\n        });\n    };\n\n    private onActiveRoomUpdate = (isActive: boolean): void => {\n        this.setState({ selected: isActive });\n    };\n\n    private onNotificationsMenuOpenClick = (ev: ButtonEvent): void => {\n        ev.preventDefault();\n        ev.stopPropagation();\n        const target = ev.target as HTMLButtonElement;\n        this.setState({ notificationsMenuPosition: target.getBoundingClientRect() });\n\n        PosthogTrackers.trackInteraction(\"WebRoomListRoomTileNotificationsMenu\", ev);\n    };\n\n    private onCloseNotificationsMenu = (): void => {\n        this.setState({ notificationsMenuPosition: null });\n    };\n\n    private onGeneralMenuOpenClick = (ev: ButtonEvent): void => {\n        ev.preventDefault();\n        ev.stopPropagation();\n        const target = ev.target as HTMLButtonElement;\n        this.setState({ generalMenuPosition: target.getBoundingClientRect() });\n    };\n\n    private onContextMenu = (ev: React.MouseEvent): void => {\n        // If we don't have a context menu to show, ignore the action.\n        if (!this.showContextMenu) return;\n\n        ev.preventDefault();\n        ev.stopPropagation();\n        this.setState({\n            generalMenuPosition: {\n                left: ev.clientX,\n                bottom: ev.clientY,\n            },\n        });\n    };\n\n    private onCloseGeneralMenu = (): void => {\n        this.setState({ generalMenuPosition: null });\n    };\n\n    private renderNotificationsMenu(isActive: boolean): React.ReactElement | null {\n        if (\n            MatrixClientPeg.safeGet().isGuest() ||\n            this.props.tag === DefaultTagID.Archived ||\n            !this.showContextMenu ||\n            this.props.isMinimized\n        ) {\n            // the menu makes no sense in these cases so do not show one\n            return null;\n        }\n\n        const state = this.roomProps.notificationVolume;\n\n        const classes = classNames(\"mx_RoomTile_notificationsButton\", {\n            // Show bell icon for the default case too.\n            mx_RoomNotificationContextMenu_iconBell: state === RoomNotifState.AllMessages,\n            mx_RoomNotificationContextMenu_iconBellDot: state === RoomNotifState.AllMessagesLoud,\n            mx_RoomNotificationContextMenu_iconBellMentions: state === RoomNotifState.MentionsOnly,\n            mx_RoomNotificationContextMenu_iconBellCrossed: state === RoomNotifState.Mute,\n\n            // Only show the icon by default if the room is overridden to muted.\n            // TODO: [FTUE Notifications] Probably need to detect global mute state\n            mx_RoomTile_notificationsButton_show: state === RoomNotifState.Mute,\n        });\n\n        return (\n            <React.Fragment>\n                <ContextMenuTooltipButton\n                    className={classes}\n                    onClick={this.onNotificationsMenuOpenClick}\n                    title={_t(\"room_list|notification_options\")}\n                    isExpanded={!!this.state.notificationsMenuPosition}\n                    tabIndex={isActive ? 0 : -1}\n                />\n                {this.state.notificationsMenuPosition && (\n                    <RoomNotificationContextMenu\n                        {...contextMenuBelow(this.state.notificationsMenuPosition)}\n                        onFinished={this.onCloseNotificationsMenu}\n                        room={this.props.room}\n                    />\n                )}\n            </React.Fragment>\n        );\n    }\n\n    private renderGeneralMenu(): React.ReactElement | null {\n        if (!this.showContextMenu) return null; // no menu to show\n        return (\n            <React.Fragment>\n                <ContextMenuTooltipButton\n                    className=\"mx_RoomTile_menuButton\"\n                    onClick={this.onGeneralMenuOpenClick}\n                    title={_t(\"room|context_menu|title\")}\n                    isExpanded={!!this.state.generalMenuPosition}\n                />\n                {this.state.generalMenuPosition && (\n                    <RoomGeneralContextMenu\n                        {...contextMenuBelow(this.state.generalMenuPosition)}\n                        onFinished={this.onCloseGeneralMenu}\n                        room={this.props.room}\n                        onPostFavoriteClick={(ev: ButtonEvent) =>\n                            PosthogTrackers.trackInteraction(\"WebRoomListRoomTileContextMenuFavouriteToggle\", ev)\n                        }\n                        onPostInviteClick={(ev: ButtonEvent) =>\n                            PosthogTrackers.trackInteraction(\"WebRoomListRoomTileContextMenuInviteItem\", ev)\n                        }\n                        onPostSettingsClick={(ev: ButtonEvent) =>\n                            PosthogTrackers.trackInteraction(\"WebRoomListRoomTileContextMenuSettingsItem\", ev)\n                        }\n                        onPostLeaveClick={(ev: ButtonEvent) =>\n                            PosthogTrackers.trackInteraction(\"WebRoomListRoomTileContextMenuLeaveItem\", ev)\n                        }\n                        onPostMarkAsReadClick={(ev: ButtonEvent) =>\n                            PosthogTrackers.trackInteraction(\"WebRoomListRoomTileContextMenuMarkRead\", ev)\n                        }\n                        onPostMarkAsUnreadClick={(ev: ButtonEvent) =>\n                            PosthogTrackers.trackInteraction(\"WebRoomListRoomTileContextMenuMarkUnread\", ev)\n                        }\n                    />\n                )}\n            </React.Fragment>\n        );\n    }\n\n    /**\n     * RoomTile has a subtile if one of the following applies:\n     * - there is a call\n     * - there is a live voice broadcast\n     * - message previews are enabled and there is a previewable message\n     */\n    private get shouldRenderSubtitle(): boolean {\n        return (\n            !!this.state.call ||\n            this.props.hasLiveVoiceBroadcast ||\n            (this.props.showMessagePreview && !!this.state.messagePreview)\n        );\n    }\n\n    public render(): React.ReactElement {\n        const classes = classNames({\n            mx_RoomTile: true,\n            mx_RoomTile_sticky:\n                SettingsStore.getValue(\"feature_ask_to_join\") &&\n                (this.props.room.getMyMembership() === KnownMembership.Knock || isKnockDenied(this.props.room)),\n            mx_RoomTile_selected: this.state.selected,\n            mx_RoomTile_hasMenuOpen: !!(this.state.generalMenuPosition || this.state.notificationsMenuPosition),\n            mx_RoomTile_minimized: this.props.isMinimized,\n        });\n\n        let name = this.props.room.name;\n        if (typeof name !== \"string\") name = \"\";\n        name = name.replace(\":\", \":\\u200b\"); // add a zero-width space to allow linewrapping after the colon\n\n        let badge: React.ReactNode;\n        if (!this.props.isMinimized && this.notificationState) {\n            // aria-hidden because we summarise the unread count/highlight status in a manual aria-label below\n            badge = (\n                <