UNPKG

matrix-react-sdk

Version:
390 lines (384 loc) 76.8 kB
"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                <