UNPKG

matrix-react-sdk

Version:
373 lines (370 loc) 68 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); Object.defineProperty(exports, "__esModule", { value: true }); exports.default = void 0; var _react = _interopRequireWildcard(require("react")); var _classnames = _interopRequireDefault(require("classnames")); var _compoundWeb = require("@vector-im/compound-web"); var _favourite = _interopRequireDefault(require("@vector-im/compound-design-tokens/assets/web/icons/favourite")); var _userAdd = _interopRequireDefault(require("@vector-im/compound-design-tokens/assets/web/icons/user-add")); var _link = _interopRequireDefault(require("@vector-im/compound-design-tokens/assets/web/icons/link")); var _settings = _interopRequireDefault(require("@vector-im/compound-design-tokens/assets/web/icons/settings")); var _exportArchive = _interopRequireDefault(require("@vector-im/compound-design-tokens/assets/web/icons/export-archive")); var _leave = _interopRequireDefault(require("@vector-im/compound-design-tokens/assets/web/icons/leave")); var _files = _interopRequireDefault(require("@vector-im/compound-design-tokens/assets/web/icons/files")); var _extensions = _interopRequireDefault(require("@vector-im/compound-design-tokens/assets/web/icons/extensions")); var _userProfile = _interopRequireDefault(require("@vector-im/compound-design-tokens/assets/web/icons/user-profile")); var _threads = _interopRequireDefault(require("@vector-im/compound-design-tokens/assets/web/icons/threads")); var _polls = _interopRequireDefault(require("@vector-im/compound-design-tokens/assets/web/icons/polls")); var _pin = _interopRequireDefault(require("@vector-im/compound-design-tokens/assets/web/icons/pin")); var _lockSolid = _interopRequireDefault(require("@vector-im/compound-design-tokens/assets/web/icons/lock-solid")); var _lockOff = _interopRequireDefault(require("@vector-im/compound-design-tokens/assets/web/icons/lock-off")); var _public = _interopRequireDefault(require("@vector-im/compound-design-tokens/assets/web/icons/public")); var _error = _interopRequireDefault(require("@vector-im/compound-design-tokens/assets/web/icons/error")); var _chevronDown = _interopRequireDefault(require("@vector-im/compound-design-tokens/assets/web/icons/chevron-down")); var _matrix = require("matrix-js-sdk/src/matrix"); var _MatrixClientContext = _interopRequireDefault(require("../../../contexts/MatrixClientContext")); var _useIsEncrypted = require("../../../hooks/useIsEncrypted"); var _BaseCard = _interopRequireDefault(require("./BaseCard")); var _languageHandler = require("../../../languageHandler"); var _RoomAvatar = _interopRequireDefault(require("../avatars/RoomAvatar")); var _dispatcher = _interopRequireDefault(require("../../../dispatcher/dispatcher")); var _RightPanelStorePhases = require("../../../stores/right-panel/RightPanelStorePhases"); var _Modal = _interopRequireDefault(require("../../../Modal")); var _ShareDialog = _interopRequireDefault(require("../dialogs/ShareDialog")); var _useEventEmitter = require("../../../hooks/useEventEmitter"); var _ShieldUtils = require("../../../utils/ShieldUtils"); var _RoomContext = _interopRequireWildcard(require("../../../contexts/RoomContext")); var _RoomName = _interopRequireDefault(require("../elements/RoomName")); var _ExportDialog = _interopRequireDefault(require("../dialogs/ExportDialog")); var _RightPanelStore = _interopRequireDefault(require("../../../stores/right-panel/RightPanelStore")); var _PosthogTrackers = _interopRequireDefault(require("../../../PosthogTrackers")); var _PollHistoryDialog = require("../dialogs/PollHistoryDialog"); var _Flex = require("../../utils/Flex"); var _RoomListStore = _interopRequireWildcard(require("../../../stores/room-list/RoomListStore")); var _models = require("../../../stores/room-list/models"); var _tagRoom = require("../../../utils/room/tagRoom"); var _canInviteTo = require("../../../utils/room/canInviteTo"); var _inviteToRoom = require("../../../utils/room/inviteToRoom"); var _useAccountData = require("../../../hooks/useAccountData"); var _useRoomState = require("../../../hooks/useRoomState"); var _useTopic = require("../../../hooks/room/useTopic"); var _HtmlUtils = require("../../../HtmlUtils"); var _Box = require("../../utils/Box"); var _RoomTopic = require("../elements/RoomTopic"); var _useDispatcher = require("../../../hooks/useDispatcher"); var _actions = require("../../../dispatcher/actions"); var _Keyboard = require("../../../Keyboard"); var _useTransition = require("../../../hooks/useTransition"); var _videoRooms = require("../../../utils/video-rooms"); var _usePinnedEvents = require("../../../hooks/usePinnedEvents"); var _ReleaseAnnouncement = require("../../structures/ReleaseAnnouncement.tsx"); 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 2020 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 onRoomMembersClick = () => { _RightPanelStore.default.instance.pushCard({ phase: _RightPanelStorePhases.RightPanelPhases.RoomMemberList }, true); }; const onRoomThreadsClick = () => { _RightPanelStore.default.instance.pushCard({ phase: _RightPanelStorePhases.RightPanelPhases.ThreadPanel }, true); }; const onRoomFilesClick = () => { _RightPanelStore.default.instance.pushCard({ phase: _RightPanelStorePhases.RightPanelPhases.FilePanel }, true); }; const onRoomExtensionsClick = () => { _RightPanelStore.default.instance.pushCard({ phase: _RightPanelStorePhases.RightPanelPhases.Extensions }, true); }; const onRoomPinsClick = () => { _PosthogTrackers.default.trackInteraction("PinnedMessageRoomInfoButton"); _RightPanelStore.default.instance.pushCard({ phase: _RightPanelStorePhases.RightPanelPhases.PinnedMessages }, true); }; const onRoomSettingsClick = ev => { _dispatcher.default.dispatch({ action: "open_room_settings" }); _PosthogTrackers.default.trackInteraction("WebRightPanelRoomInfoSettingsButton", ev); }; const RoomTopic = ({ room }) => { const [expanded, setExpanded] = (0, _react.useState)(true); const topic = (0, _useTopic.useTopic)(room); const body = (0, _HtmlUtils.topicToHtml)(topic?.text, topic?.html); const canEditTopic = (0, _useRoomState.useRoomState)(room, state => state.maySendStateEvent(_matrix.EventType.RoomTopic, room.client.getSafeUserId())); const onEditClick = e => { e.preventDefault(); e.stopPropagation(); _dispatcher.default.dispatch({ action: "open_room_settings" }); }; if (!body && !canEditTopic) { return null; } if (!body) { return /*#__PURE__*/_react.default.createElement(_Flex.Flex, { as: "section", direction: "column", justify: "center", gap: "var(--cpd-space-2x)", className: "mx_RoomSummaryCard_topic" }, /*#__PURE__*/_react.default.createElement(_Box.Box, { flex: "1" }, /*#__PURE__*/_react.default.createElement(_compoundWeb.Link, { kind: "primary", onClick: onEditClick }, /*#__PURE__*/_react.default.createElement(_compoundWeb.Text, { size: "sm", weight: "regular" }, (0, _languageHandler._t)("right_panel|add_topic"))))); } const content = expanded ? /*#__PURE__*/_react.default.createElement(_HtmlUtils.Linkify, null, body) : body; return /*#__PURE__*/_react.default.createElement(_Flex.Flex, { as: "section", direction: "column", justify: "center", gap: "var(--cpd-space-2x)", className: (0, _classnames.default)("mx_RoomSummaryCard_topic", { mx_RoomSummaryCard_topic_collapsed: !expanded }) }, /*#__PURE__*/_react.default.createElement(_Box.Box, { flex: "1", className: "mx_RoomSummaryCard_topic_container" }, /*#__PURE__*/_react.default.createElement(_compoundWeb.Text, { size: "sm", weight: "regular", onClick: ev => { if (ev.target instanceof HTMLAnchorElement) { (0, _RoomTopic.onRoomTopicLinkClick)(ev); return; } } }, content), /*#__PURE__*/_react.default.createElement(_compoundWeb.IconButton, { className: "mx_RoomSummaryCard_topic_chevron", size: "24px", onClick: () => setExpanded(!expanded) }, /*#__PURE__*/_react.default.createElement(_chevronDown.default, null))), expanded && canEditTopic && /*#__PURE__*/_react.default.createElement(_Box.Box, { flex: "1", className: "mx_RoomSummaryCard_topic_edit" }, /*#__PURE__*/_react.default.createElement(_compoundWeb.Link, { kind: "primary", onClick: onEditClick }, /*#__PURE__*/_react.default.createElement(_compoundWeb.Text, { size: "sm", weight: "regular" }, (0, _languageHandler._t)("action|edit"))))); }; const RoomSummaryCard = ({ room, permalinkCreator, onSearchChange, onSearchCancel, focusRoomSearch }) => { const cli = (0, _react.useContext)(_MatrixClientContext.default); const onShareRoomClick = () => { _Modal.default.createDialog(_ShareDialog.default, { target: room }); }; const onRoomExportClick = async () => { _Modal.default.createDialog(_ExportDialog.default, { room }); }; const onRoomPollHistoryClick = () => { _Modal.default.createDialog(_PollHistoryDialog.PollHistoryDialog, { room, matrixClient: cli, permalinkCreator }); }; const onLeaveRoomClick = () => { _dispatcher.default.dispatch({ action: "leave_room", room_id: room.roomId }); }; const isRoomEncrypted = (0, _useIsEncrypted.useIsEncrypted)(cli, room); const roomContext = (0, _react.useContext)(_RoomContext.default); const e2eStatus = roomContext.e2eStatus; const isVideoRoom = (0, _videoRooms.isVideoRoom)(room); const roomState = (0, _useRoomState.useRoomState)(room); const directRoomsList = (0, _useAccountData.useAccountData)(room.client, _matrix.EventType.Direct); const [isDirectMessage, setDirectMessage] = (0, _react.useState)(false); (0, _react.useEffect)(() => { for (const [, dmRoomList] of Object.entries(directRoomsList)) { if (dmRoomList.includes(room?.roomId ?? "")) { setDirectMessage(true); break; } } }, [room, directRoomsList]); const searchInputRef = (0, _react.useRef)(null); (0, _useDispatcher.useDispatcher)(_dispatcher.default, payload => { if (payload.action === _actions.Action.FocusMessageSearch) { searchInputRef.current?.focus(); } }); // Clear the search field when the user leaves the search view (0, _useTransition.useTransition)(prevTimelineRenderingType => { if (prevTimelineRenderingType === _RoomContext.TimelineRenderingType.Search && roomContext.timelineRenderingType !== _RoomContext.TimelineRenderingType.Search && searchInputRef.current) { searchInputRef.current.value = ""; } }, [roomContext.timelineRenderingType]); const alias = room.getCanonicalAlias() || room.getAltAliases()[0] || ""; const roomInfo = /*#__PURE__*/_react.default.createElement("header", { className: "mx_RoomSummaryCard_container" }, /*#__PURE__*/_react.default.createElement(_RoomAvatar.default, { room: room, size: "80px", viewAvatarOnClick: true }), /*#__PURE__*/_react.default.createElement(_RoomName.default, { room: room }, name => /*#__PURE__*/_react.default.createElement(_compoundWeb.Heading, { as: "h1", size: "md", weight: "semibold", className: "mx_RoomSummaryCard_roomName text-primary", title: name }, name)), /*#__PURE__*/_react.default.createElement(_compoundWeb.Text, { as: "div", size: "sm", weight: "semibold", className: "mx_RoomSummaryCard_alias text-secondary", title: alias }, alias), /*#__PURE__*/_react.default.createElement(_Flex.Flex, { as: "section", justify: "center", gap: "var(--cpd-space-2x)", className: "mx_RoomSummaryCard_badges" }, !isDirectMessage && roomState.getJoinRule() === _matrix.JoinRule.Public && /*#__PURE__*/_react.default.createElement(_compoundWeb.Badge, { kind: "grey" }, /*#__PURE__*/_react.default.createElement(_public.default, { width: "1em" }), (0, _languageHandler._t)("common|public_room")), isRoomEncrypted && e2eStatus !== _ShieldUtils.E2EStatus.Warning && /*#__PURE__*/_react.default.createElement(_compoundWeb.Badge, { kind: "green" }, /*#__PURE__*/_react.default.createElement(_lockSolid.default, { width: "1em" }), (0, _languageHandler._t)("common|encrypted")), !e2eStatus && /*#__PURE__*/_react.default.createElement(_compoundWeb.Badge, { kind: "grey" }, /*#__PURE__*/_react.default.createElement(_lockOff.default, { width: "1em" }), (0, _languageHandler._t)("common|unencrypted")), e2eStatus === _ShieldUtils.E2EStatus.Warning && /*#__PURE__*/_react.default.createElement(_compoundWeb.Badge, { kind: "red" }, /*#__PURE__*/_react.default.createElement(_error.default, { width: "1em" }), (0, _languageHandler._t)("common|not_trusted"))), /*#__PURE__*/_react.default.createElement(RoomTopic, { room: room })); const pinCount = (0, _usePinnedEvents.usePinnedEvents)(room).length; const roomTags = (0, _useEventEmitter.useEventEmitterState)(_RoomListStore.default.instance, _RoomListStore.LISTS_UPDATE_EVENT, () => _RoomListStore.default.instance.getTagsForRoom(room)); const canInviteToState = (0, _useEventEmitter.useEventEmitterState)(room, _matrix.RoomStateEvent.Update, () => (0, _canInviteTo.canInviteTo)(room)); const isFavorite = roomTags.includes(_models.DefaultTagID.Favourite); const header = onSearchChange && /*#__PURE__*/_react.default.createElement(_compoundWeb.Form.Root, { className: "mx_RoomSummaryCard_search", onSubmit: e => e.preventDefault() }, /*#__PURE__*/_react.default.createElement(_compoundWeb.Search, { placeholder: (0, _languageHandler._t)("room|search|placeholder"), name: "room_message_search", onChange: onSearchChange, className: "mx_no_textinput", ref: searchInputRef, autoFocus: focusRoomSearch, onKeyDown: e => { if (searchInputRef.current && e.key === _Keyboard.Key.ESCAPE) { searchInputRef.current.value = ""; onSearchCancel?.(); } } })); return /*#__PURE__*/_react.default.createElement(_BaseCard.default, { id: "room-summary-panel", className: "mx_RoomSummaryCard", ariaLabelledBy: "room-summary-panel-tab", role: "tabpanel", header: header }, roomInfo, /*#__PURE__*/_react.default.createElement(_compoundWeb.Separator, null), /*#__PURE__*/_react.default.createElement("div", { role: "menubar", "aria-orientation": "vertical" }, /*#__PURE__*/_react.default.createElement(_compoundWeb.ToggleMenuItem, { Icon: _favourite.default, label: (0, _languageHandler._t)("room|context_menu|favourite"), checked: isFavorite, onSelect: () => (0, _tagRoom.tagRoom)(room, _models.DefaultTagID.Favourite) }), /*#__PURE__*/_react.default.createElement(_compoundWeb.MenuItem, { Icon: _userAdd.default, label: (0, _languageHandler._t)("action|invite"), disabled: !canInviteToState, onSelect: () => (0, _inviteToRoom.inviteToRoom)(room) }), /*#__PURE__*/_react.default.createElement(_compoundWeb.Separator, null), /*#__PURE__*/_react.default.createElement(_compoundWeb.MenuItem, { Icon: _userProfile.default, label: (0, _languageHandler._t)("common|people"), onSelect: onRoomMembersClick }), /*#__PURE__*/_react.default.createElement(_compoundWeb.MenuItem, { Icon: _threads.default, label: (0, _languageHandler._t)("common|threads"), onSelect: onRoomThreadsClick }), !isVideoRoom && /*#__PURE__*/_react.default.createElement(_react.default.Fragment, null, /*#__PURE__*/_react.default.createElement(_ReleaseAnnouncement.ReleaseAnnouncement, { feature: "pinningMessageList", header: (0, _languageHandler._t)("right_panel|pinned_messages|release_announcement|title"), description: (0, _languageHandler._t)("right_panel|pinned_messages|release_announcement|description"), closeLabel: (0, _languageHandler._t)("right_panel|pinned_messages|release_announcement|close"), placement: "top" }, /*#__PURE__*/_react.default.createElement("div", null, /*#__PURE__*/_react.default.createElement(_compoundWeb.MenuItem, { Icon: _pin.default, label: (0, _languageHandler._t)("right_panel|pinned_messages_button"), onSelect: onRoomPinsClick }, /*#__PURE__*/_react.default.createElement(_compoundWeb.Text, { as: "span", size: "sm" }, pinCount)))), /*#__PURE__*/_react.default.createElement(_compoundWeb.MenuItem, { Icon: _files.default, label: (0, _languageHandler._t)("right_panel|files_button"), onSelect: onRoomFilesClick }), /*#__PURE__*/_react.default.createElement(_compoundWeb.MenuItem, { Icon: _extensions.default, label: (0, _languageHandler._t)("right_panel|extensions_button"), onSelect: onRoomExtensionsClick })), /*#__PURE__*/_react.default.createElement(_compoundWeb.Separator, null), /*#__PURE__*/_react.default.createElement(_compoundWeb.MenuItem, { Icon: _link.default, label: (0, _languageHandler._t)("action|copy_link"), onSelect: onShareRoomClick }), !isVideoRoom && /*#__PURE__*/_react.default.createElement(_react.default.Fragment, null, /*#__PURE__*/_react.default.createElement(_compoundWeb.MenuItem, { Icon: _polls.default, label: (0, _languageHandler._t)("right_panel|polls_button"), onSelect: onRoomPollHistoryClick }), /*#__PURE__*/_react.default.createElement(_compoundWeb.MenuItem, { Icon: _exportArchive.default, label: (0, _languageHandler._t)("export_chat|title"), onSelect: onRoomExportClick })), /*#__PURE__*/_react.default.createElement(_compoundWeb.MenuItem, { Icon: _settings.default, label: (0, _languageHandler._t)("common|settings"), onSelect: onRoomSettingsClick }), /*#__PURE__*/_react.default.createElement(_compoundWeb.Separator, null), /*#__PURE__*/_react.default.createElement(_compoundWeb.MenuItem, { Icon: _leave.default, kind: "critical", label: (0, _languageHandler._t)("action|leave_room"), onSelect: onLeaveRoomClick }))); }; var _default = exports.default = RoomSummaryCard; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireWildcard","require","_classnames","_interopRequireDefault","_compoundWeb","_favourite","_userAdd","_link","_settings","_exportArchive","_leave","_files","_extensions","_userProfile","_threads","_polls","_pin","_lockSolid","_lockOff","_public","_error","_chevronDown","_matrix","_MatrixClientContext","_useIsEncrypted","_BaseCard","_languageHandler","_RoomAvatar","_dispatcher","_RightPanelStorePhases","_Modal","_ShareDialog","_useEventEmitter","_ShieldUtils","_RoomContext","_RoomName","_ExportDialog","_RightPanelStore","_PosthogTrackers","_PollHistoryDialog","_Flex","_RoomListStore","_models","_tagRoom","_canInviteTo","_inviteToRoom","_useAccountData","_useRoomState","_useTopic","_HtmlUtils","_Box","_RoomTopic","_useDispatcher","_actions","_Keyboard","_useTransition","_videoRooms","_usePinnedEvents","_ReleaseAnnouncement","_getRequireWildcardCache","e","WeakMap","r","t","__esModule","default","has","get","n","__proto__","a","Object","defineProperty","getOwnPropertyDescriptor","u","hasOwnProperty","call","i","set","onRoomMembersClick","RightPanelStore","instance","pushCard","phase","RightPanelPhases","RoomMemberList","onRoomThreadsClick","ThreadPanel","onRoomFilesClick","FilePanel","onRoomExtensionsClick","Extensions","onRoomPinsClick","PosthogTrackers","trackInteraction","PinnedMessages","onRoomSettingsClick","ev","defaultDispatcher","dispatch","action","RoomTopic","room","expanded","setExpanded","useState","topic","useTopic","body","topicToHtml","text","html","canEditTopic","useRoomState","state","maySendStateEvent","EventType","client","getSafeUserId","onEditClick","preventDefault","stopPropagation","createElement","Flex","as","direction","justify","gap","className","Box","flex","Link","kind","onClick","Text","size","weight","_t","content","Linkify","classNames","mx_RoomSummaryCard_topic_collapsed","target","HTMLAnchorElement","onRoomTopicLinkClick","IconButton","RoomSummaryCard","permalinkCreator","onSearchChange","onSearchCancel","focusRoomSearch","cli","useContext","MatrixClientContext","onShareRoomClick","Modal","createDialog","ShareDialog","onRoomExportClick","ExportDialog","onRoomPollHistoryClick","PollHistoryDialog","matrixClient","onLeaveRoomClick","room_id","roomId","isRoomEncrypted","useIsEncrypted","roomContext","RoomContext","e2eStatus","isVideoRoom","calcIsVideoRoom","roomState","directRoomsList","useAccountData","Direct","isDirectMessage","setDirectMessage","useEffect","dmRoomList","entries","includes","searchInputRef","useRef","useDispatcher","payload","Action","FocusMessageSearch","current","focus","useTransition","prevTimelineRenderingType","TimelineRenderingType","Search","timelineRenderingType","value","alias","getCanonicalAlias","getAltAliases","roomInfo","viewAvatarOnClick","name","Heading","title","getJoinRule","JoinRule","Public","Badge","width","E2EStatus","Warning","pinCount","usePinnedEvents","length","roomTags","useEventEmitterState","RoomListStore","LISTS_UPDATE_EVENT","getTagsForRoom","canInviteToState","RoomStateEvent","Update","canInviteTo","isFavorite","DefaultTagID","Favourite","header","Form","Root","onSubmit","placeholder","onChange","ref","autoFocus","onKeyDown","key","Key","ESCAPE","id","ariaLabelledBy","role","Separator","ToggleMenuItem","Icon","FavouriteIcon","label","checked","onSelect","tagRoom","MenuItem","UserAddIcon","disabled","inviteToRoom","UserProfileIcon","ThreadsIcon","Fragment","ReleaseAnnouncement","feature","description","closeLabel","placement","PinIcon","FilesIcon","ExtensionsIcon","LinkIcon","PollsIcon","ExportArchiveIcon","SettingsIcon","LeaveIcon","_default","exports"],"sources":["../../../../src/components/views/right_panel/RoomSummaryCard.tsx"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2020 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, { ChangeEvent, SyntheticEvent, useContext, useEffect, useRef, useState } from \"react\";\nimport classNames from \"classnames\";\nimport {\n    MenuItem,\n    Separator,\n    ToggleMenuItem,\n    Text,\n    Badge,\n    Heading,\n    IconButton,\n    Link,\n    Search,\n    Form,\n} from \"@vector-im/compound-web\";\nimport FavouriteIcon from \"@vector-im/compound-design-tokens/assets/web/icons/favourite\";\nimport UserAddIcon from \"@vector-im/compound-design-tokens/assets/web/icons/user-add\";\nimport LinkIcon from \"@vector-im/compound-design-tokens/assets/web/icons/link\";\nimport SettingsIcon from \"@vector-im/compound-design-tokens/assets/web/icons/settings\";\nimport ExportArchiveIcon from \"@vector-im/compound-design-tokens/assets/web/icons/export-archive\";\nimport LeaveIcon from \"@vector-im/compound-design-tokens/assets/web/icons/leave\";\nimport FilesIcon from \"@vector-im/compound-design-tokens/assets/web/icons/files\";\nimport ExtensionsIcon from \"@vector-im/compound-design-tokens/assets/web/icons/extensions\";\nimport UserProfileIcon from \"@vector-im/compound-design-tokens/assets/web/icons/user-profile\";\nimport ThreadsIcon from \"@vector-im/compound-design-tokens/assets/web/icons/threads\";\nimport PollsIcon from \"@vector-im/compound-design-tokens/assets/web/icons/polls\";\nimport PinIcon from \"@vector-im/compound-design-tokens/assets/web/icons/pin\";\nimport LockIcon from \"@vector-im/compound-design-tokens/assets/web/icons/lock-solid\";\nimport LockOffIcon from \"@vector-im/compound-design-tokens/assets/web/icons/lock-off\";\nimport PublicIcon from \"@vector-im/compound-design-tokens/assets/web/icons/public\";\nimport ErrorIcon from \"@vector-im/compound-design-tokens/assets/web/icons/error\";\nimport ChevronDownIcon from \"@vector-im/compound-design-tokens/assets/web/icons/chevron-down\";\nimport { EventType, JoinRule, Room, RoomStateEvent } from \"matrix-js-sdk/src/matrix\";\n\nimport MatrixClientContext from \"../../../contexts/MatrixClientContext\";\nimport { useIsEncrypted } from \"../../../hooks/useIsEncrypted\";\nimport BaseCard from \"./BaseCard\";\nimport { _t } from \"../../../languageHandler\";\nimport RoomAvatar from \"../avatars/RoomAvatar\";\nimport defaultDispatcher from \"../../../dispatcher/dispatcher\";\nimport { RightPanelPhases } from \"../../../stores/right-panel/RightPanelStorePhases\";\nimport Modal from \"../../../Modal\";\nimport ShareDialog from \"../dialogs/ShareDialog\";\nimport { useEventEmitterState } from \"../../../hooks/useEventEmitter\";\nimport { E2EStatus } from \"../../../utils/ShieldUtils\";\nimport { RoomPermalinkCreator } from \"../../../utils/permalinks/Permalinks\";\nimport RoomContext, { TimelineRenderingType } from \"../../../contexts/RoomContext\";\nimport RoomName from \"../elements/RoomName\";\nimport ExportDialog from \"../dialogs/ExportDialog\";\nimport RightPanelStore from \"../../../stores/right-panel/RightPanelStore\";\nimport PosthogTrackers from \"../../../PosthogTrackers\";\nimport { PollHistoryDialog } from \"../dialogs/PollHistoryDialog\";\nimport { Flex } from \"../../utils/Flex\";\nimport RoomListStore, { LISTS_UPDATE_EVENT } from \"../../../stores/room-list/RoomListStore\";\nimport { DefaultTagID } from \"../../../stores/room-list/models\";\nimport { tagRoom } from \"../../../utils/room/tagRoom\";\nimport { canInviteTo } from \"../../../utils/room/canInviteTo\";\nimport { inviteToRoom } from \"../../../utils/room/inviteToRoom\";\nimport { useAccountData } from \"../../../hooks/useAccountData\";\nimport { useRoomState } from \"../../../hooks/useRoomState\";\nimport { useTopic } from \"../../../hooks/room/useTopic\";\nimport { Linkify, topicToHtml } from \"../../../HtmlUtils\";\nimport { Box } from \"../../utils/Box\";\nimport { onRoomTopicLinkClick } from \"../elements/RoomTopic\";\nimport { useDispatcher } from \"../../../hooks/useDispatcher\";\nimport { Action } from \"../../../dispatcher/actions\";\nimport { Key } from \"../../../Keyboard\";\nimport { useTransition } from \"../../../hooks/useTransition\";\nimport { isVideoRoom as calcIsVideoRoom } from \"../../../utils/video-rooms\";\nimport { usePinnedEvents } from \"../../../hooks/usePinnedEvents\";\nimport { ReleaseAnnouncement } from \"../../structures/ReleaseAnnouncement.tsx\";\n\ninterface IProps {\n    room: Room;\n    permalinkCreator: RoomPermalinkCreator;\n    onSearchChange?: (e: ChangeEvent) => void;\n    onSearchCancel?: () => void;\n    focusRoomSearch?: boolean;\n}\n\nconst onRoomMembersClick = (): void => {\n    RightPanelStore.instance.pushCard({ phase: RightPanelPhases.RoomMemberList }, true);\n};\n\nconst onRoomThreadsClick = (): void => {\n    RightPanelStore.instance.pushCard({ phase: RightPanelPhases.ThreadPanel }, true);\n};\n\nconst onRoomFilesClick = (): void => {\n    RightPanelStore.instance.pushCard({ phase: RightPanelPhases.FilePanel }, true);\n};\n\nconst onRoomExtensionsClick = (): void => {\n    RightPanelStore.instance.pushCard({ phase: RightPanelPhases.Extensions }, true);\n};\n\nconst onRoomPinsClick = (): void => {\n    PosthogTrackers.trackInteraction(\"PinnedMessageRoomInfoButton\");\n    RightPanelStore.instance.pushCard({ phase: RightPanelPhases.PinnedMessages }, true);\n};\n\nconst onRoomSettingsClick = (ev: Event): void => {\n    defaultDispatcher.dispatch({ action: \"open_room_settings\" });\n    PosthogTrackers.trackInteraction(\"WebRightPanelRoomInfoSettingsButton\", ev);\n};\n\nconst RoomTopic: React.FC<Pick<IProps, \"room\">> = ({ room }): JSX.Element | null => {\n    const [expanded, setExpanded] = useState(true);\n\n    const topic = useTopic(room);\n    const body = topicToHtml(topic?.text, topic?.html);\n\n    const canEditTopic = useRoomState(room, (state) =>\n        state.maySendStateEvent(EventType.RoomTopic, room.client.getSafeUserId()),\n    );\n    const onEditClick = (e: SyntheticEvent): void => {\n        e.preventDefault();\n        e.stopPropagation();\n        defaultDispatcher.dispatch({ action: \"open_room_settings\" });\n    };\n\n    if (!body && !canEditTopic) {\n        return null;\n    }\n\n    if (!body) {\n        return (\n            <Flex\n                as=\"section\"\n                direction=\"column\"\n                justify=\"center\"\n                gap=\"var(--cpd-space-2x)\"\n                className=\"mx_RoomSummaryCard_topic\"\n            >\n                <Box flex=\"1\">\n                    <Link kind=\"primary\" onClick={onEditClick}>\n                        <Text size=\"sm\" weight=\"regular\">\n                            {_t(\"right_panel|add_topic\")}\n                        </Text>\n                    </Link>\n                </Box>\n            </Flex>\n        );\n    }\n\n    const content = expanded ? <Linkify>{body}</Linkify> : body;\n    return (\n        <Flex\n            as=\"section\"\n            direction=\"column\"\n            justify=\"center\"\n            gap=\"var(--cpd-space-2x)\"\n            className={classNames(\"mx_RoomSummaryCard_topic\", {\n                mx_RoomSummaryCard_topic_collapsed: !expanded,\n            })}\n        >\n            <Box flex=\"1\" className=\"mx_RoomSummaryCard_topic_container\">\n                <Text\n                    size=\"sm\"\n                    weight=\"regular\"\n                    onClick={(ev: React.MouseEvent): void => {\n                        if (ev.target instanceof HTMLAnchorElement) {\n                            onRoomTopicLinkClick(ev);\n                            return;\n                        }\n                    }}\n                >\n                    {content}\n                </Text>\n                <IconButton\n                    className=\"mx_RoomSummaryCard_topic_chevron\"\n                    size=\"24px\"\n                    onClick={() => setExpanded(!expanded)}\n                >\n                    <ChevronDownIcon />\n                </IconButton>\n            </Box>\n            {expanded && canEditTopic && (\n                <Box flex=\"1\" className=\"mx_RoomSummaryCard_topic_edit\">\n                    <Link kind=\"primary\" onClick={onEditClick}>\n                        <Text size=\"sm\" weight=\"regular\">\n                            {_t(\"action|edit\")}\n                        </Text>\n                    </Link>\n                </Box>\n            )}\n        </Flex>\n    );\n};\n\nconst RoomSummaryCard: React.FC<IProps> = ({\n    room,\n    permalinkCreator,\n    onSearchChange,\n    onSearchCancel,\n    focusRoomSearch,\n}) => {\n    const cli = useContext(MatrixClientContext);\n\n    const onShareRoomClick = (): void => {\n        Modal.createDialog(ShareDialog, {\n            target: room,\n        });\n    };\n\n    const onRoomExportClick = async (): Promise<void> => {\n        Modal.createDialog(ExportDialog, {\n            room,\n        });\n    };\n\n    const onRoomPollHistoryClick = (): void => {\n        Modal.createDialog(PollHistoryDialog, {\n            room,\n            matrixClient: cli,\n            permalinkCreator,\n        });\n    };\n\n    const onLeaveRoomClick = (): void => {\n        defaultDispatcher.dispatch({\n            action: \"leave_room\",\n            room_id: room.roomId,\n        });\n    };\n\n    const isRoomEncrypted = useIsEncrypted(cli, room);\n    const roomContext = useContext(RoomContext);\n    const e2eStatus = roomContext.e2eStatus;\n    const isVideoRoom = calcIsVideoRoom(room);\n\n    const roomState = useRoomState(room);\n    const directRoomsList = useAccountData<Record<string, string[]>>(room.client, EventType.Direct);\n    const [isDirectMessage, setDirectMessage] = useState(false);\n    useEffect(() => {\n        for (const [, dmRoomList] of Object.entries(directRoomsList)) {\n            if (dmRoomList.includes(room?.roomId ?? \"\")) {\n                setDirectMessage(true);\n                break;\n            }\n        }\n    }, [room, directRoomsList]);\n\n    const searchInputRef = useRef<HTMLInputElement>(null);\n    useDispatcher(defaultDispatcher, (payload) => {\n        if (payload.action === Action.FocusMessageSearch) {\n            searchInputRef.current?.focus();\n        }\n    });\n    // Clear the search field when the user leaves the search view\n    useTransition(\n        (prevTimelineRenderingType) => {\n            if (\n                prevTimelineRenderingType === TimelineRenderingType.Search &&\n                roomContext.timelineRenderingType !== TimelineRenderingType.Search &&\n                searchInputRef.current\n            ) {\n                searchInputRef.current.value = \"\";\n            }\n        },\n        [roomContext.timelineRenderingType],\n    );\n\n    const alias = room.getCanonicalAlias() || room.getAltAliases()[0] || \"\";\n    const roomInfo = (\n        <header className=\"mx_RoomSummaryCard_container\">\n            <RoomAvatar room={room} size=\"80px\" viewAvatarOnClick />\n            <RoomName room={room}>\n                {(name) => (\n                    <Heading\n                        as=\"h1\"\n                        size=\"md\"\n                        weight=\"semibold\"\n                        className=\"mx_RoomSummaryCard_roomName text-primary\"\n                        title={name}\n                    >\n                        {name}\n                    </Heading>\n                )}\n            </RoomName>\n            <Text\n                as=\"div\"\n                size=\"sm\"\n                weight=\"semibold\"\n                className=\"mx_RoomSummaryCard_alias text-secondary\"\n                title={alias}\n            >\n                {alias}\n            </Text>\n\n            <Flex as=\"section\" justify=\"center\" gap=\"var(--cpd-space-2x)\" className=\"mx_RoomSummaryCard_badges\">\n                {!isDirectMessage && roomState.getJoinRule() === JoinRule.Public && (\n                    <Badge kind=\"grey\">\n                        <PublicIcon width=\"1em\" />\n                        {_t(\"common|public_room\")}\n                    </Badge>\n                )}\n\n                {isRoomEncrypted && e2eStatus !== E2EStatus.Warning && (\n                    <Badge kind=\"green\">\n                        <LockIcon width=\"1em\" />\n                        {_t(\"common|encrypted\")}\n                    </Badge>\n                )}\n\n                {!e2eStatus && (\n                    <Badge kind=\"grey\">\n                        <LockOffIcon width=\"1em\" />\n                        {_t(\"common|unencrypted\")}\n                    </Badge>\n                )}\n\n                {e2eStatus === E2EStatus.Warning && (\n                    <Badge kind=\"red\">\n                        <ErrorIcon width=\"1em\" />\n                        {_t(\"common|not_trusted\")}\n                    </Badge>\n                )}\n            </Flex>\n\n            <RoomTopic room={room} />\n        </header>\n    );\n\n    const pinCount = usePinnedEvents(room).length;\n\n    const roomTags = useEventEmitterState(RoomListStore.instance, LISTS_UPDATE_EVENT, () =>\n        RoomListStore.instance.getTagsForRoom(room),\n    );\n    const canInviteToState = useEventEmitterState(room, RoomStateEvent.Update, () => canInviteTo(room));\n    const isFavorite = roomTags.includes(DefaultTagID.Favourite);\n\n    const header = onSearchChange && (\n        <Form.Root className=\"mx_RoomSummaryCard_search\" onSubmit={(e) => e.preventDefault()}>\n            <Search\n                placeholder={_t(\"room|search|placeholder\")}\n                name=\"room_message_search\"\n                onChange={onSearchChange}\n                className=\"mx_no_textinput\"\n                ref={searchInputRef}\n                autoFocus={focusRoomSearch}\n                onKeyDown={(e) => {\n                    if (searchInputRef.current && e.key === Key.ESCAPE) {\n                        searchInputRef.current.value = \"\";\n                        onSearchCancel?.();\n                    }\n                }}\n            />\n        </Form.Root>\n    );\n\n    return (\n        <BaseCard\n            id=\"room-summary-panel\"\n            className=\"mx_RoomSummaryCard\"\n            ariaLabelledBy=\"room-summary-panel-tab\"\n            role=\"tabpanel\"\n            header={header}\n        >\n            {roomInfo}\n\n            <Separator />\n\n            <div role=\"menubar\" aria-orientation=\"vertical\">\n                <ToggleMenuItem\n                    Icon={FavouriteIcon}\n                    label={_t(\"room|context_menu|favourite\")}\n                    checked={isFavorite}\n                    onSelect={() => tagRoom(room, DefaultTagID.Favourite)}\n                />\n                <MenuItem\n                    Icon={UserAddIcon}\n                    label={_t(\"action|invite\")}\n                    disabled={!canInviteToState}\n                    onSelect={() => inviteToRoom(room)}\n                />\n\n                <Separator />\n\n                <MenuItem Icon={UserProfileIcon} label={_t(\"common|people\")} onSelect={onRoomMembersClick} />\n                <MenuItem Icon={ThreadsIcon} label={_t(\"common|threads\")} onSelect={onRoomThreadsClick} />\n                {!isVideoRoom && (\n                    <>\n                        <ReleaseAnnouncement\n                            feature=\"pinningMessageList\"\n                            header={_t(\"right_panel|pinned_messages|release_announcement|title\")}\n                            description={_t(\"right_panel|pinned_messages|release_announcement|description\")}\n                            closeLabel={_t(\"right_panel|pinned_messages|release_announcement|close\")}\n                            placement=\"top\"\n                        >\n                            <div>\n                                <MenuItem\n                                    Icon={PinIcon}\n                                    label={_t(\"right_panel|pinned_messages_button\")}\n                                    onSelect={onRoomPinsClick}\n                                >\n                                    <Text as=\"span\" size=\"sm\">\n                                        {pinCount}\n                                    </Text>\n                                </MenuItem>\n                            </div>\n                        </ReleaseAnnouncement>\n                        <MenuItem Icon={FilesIcon} label={_t(\"right_panel|files_button\")} onSelect={onRoomFilesClick} />\n                        <MenuItem\n                            Icon={ExtensionsIcon}\n                            label={_t(\"right_panel|extensions_button\")}\n                            onSelect={onRoomExtensionsClick}\n                        />\n                    </>\n                )}\n\n                <Separator />\n\n                <MenuItem Icon={LinkIcon} label={_t(\"action|copy_link\")} onSelect={onShareRoomClick} />\n\n                {!isVideoRoom && (\n                    <>\n                        <MenuItem\n                            Icon={PollsIcon}\n                            label={_t(\"right_panel|polls_button\")}\n                            onSelect={onRoomPollHistoryClick}\n                        />\n                        <MenuItem\n                            Icon={ExportArchiveIcon}\n                            label={_t(\"export_chat|title\")}\n                            onSelect={onRoomExportClick}\n                        />\n                    </>\n                )}\n\n                <MenuItem Icon={SettingsIcon} label={_t(\"common|settings\")} onSelect={onRoomSettingsClick} />\n\n                <Separator />\n\n                <MenuItem\n                    Icon={LeaveIcon}\n                    kind=\"critical\"\n                    label={_t(\"action|leave_room\")}\n                    onSelect={onLeaveRoomClick}\n                />\n            </div>\n        </BaseCard>\n    );\n};\n\nexport default RoomSummaryCard;\n"],"mappings":";;;;;;;AAQA,IAAAA,MAAA,GAAAC,uBAAA,CAAAC,OAAA;AACA,IAAAC,WAAA,GAAAC,sBAAA,CAAAF,OAAA;AACA,IAAAG,YAAA,GAAAH,OAAA;AAYA,IAAAI,UAAA,GAAAF,sBAAA,CAAAF,OAAA;AACA,IAAAK,QAAA,GAAAH,sBAAA,CAAAF,OAAA;AACA,IAAAM,KAAA,GAAAJ,sBAAA,CAAAF,OAAA;AACA,IAAAO,SAAA,GAAAL,sBAAA,CAAAF,OAAA;AACA,IAAAQ,cAAA,GAAAN,sBAAA,CAAAF,OAAA;AACA,IAAAS,MAAA,GAAAP,sBAAA,CAAAF,OAAA;AACA,IAAAU,MAAA,GAAAR,sBAAA,CAAAF,OAAA;AACA,IAAAW,WAAA,GAAAT,sBAAA,CAAAF,OAAA;AACA,IAAAY,YAAA,GAAAV,sBAAA,CAAAF,OAAA;AACA,IAAAa,QAAA,GAAAX,sBAAA,CAAAF,OAAA;AACA,IAAAc,MAAA,GAAAZ,sBAAA,CAAAF,OAAA;AACA,IAAAe,IAAA,GAAAb,sBAAA,CAAAF,OAAA;AACA,IAAAgB,UAAA,GAAAd,sBAAA,CAAAF,OAAA;AACA,IAAAiB,QAAA,GAAAf,sBAAA,CAAAF,OAAA;AACA,IAAAkB,OAAA,GAAAhB,sBAAA,CAAAF,OAAA;AACA,IAAAmB,MAAA,GAAAjB,sBAAA,CAAAF,OAAA;AACA,IAAAoB,YAAA,GAAAlB,sBAAA,CAAAF,OAAA;AACA,IAAAqB,OAAA,GAAArB,OAAA;AAEA,IAAAsB,oBAAA,GAAApB,sBAAA,CAAAF,OAAA;AACA,IAAAuB,eAAA,GAAAvB,OAAA;AACA,IAAAwB,SAAA,GAAAtB,sBAAA,CAAAF,OAAA;AACA,IAAAyB,gBAAA,GAAAzB,OAAA;AACA,IAAA0B,WAAA,GAAAxB,sBAAA,CAAAF,OAAA;AACA,IAAA2B,WAAA,GAAAzB,sBAAA,CAAAF,OAAA;AACA,IAAA4B,sBAAA,GAAA5B,OAAA;AACA,IAAA6B,MAAA,GAAA3B,sBAAA,CAAAF,OAAA;AACA,IAAA8B,YAAA,GAAA5B,sBAAA,CAAAF,OAAA;AACA,IAAA+B,gBAAA,GAAA/B,OAAA;AACA,IAAAgC,YAAA,GAAAhC,OAAA;AAEA,IAAAiC,YAAA,GAAAlC,uBAAA,CAAAC,OAAA;AACA,IAAA