/* Copyright 2023 The Matrix.org Foundation C.I.C. Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ import React, { useEffect, useMemo, useState } from "react"; import { Body as BodyText, IconButton, Tooltip } from "@vector-im/compound-web"; import { Icon as VideoCallIcon } from "@vector-im/compound-design-tokens/icons/video-call-solid.svg"; import { Icon as VoiceCallIcon } from "@vector-im/compound-design-tokens/icons/voice-call.svg"; import { Icon as ThreadsIcon } from "@vector-im/compound-design-tokens/icons/threads-solid.svg"; import { Icon as NotificationsIcon } from "@vector-im/compound-design-tokens/icons/notifications-solid.svg"; import { Icon as VerifiedIcon } from "@vector-im/compound-design-tokens/icons/verified.svg"; import { Icon as ErrorIcon } from "@vector-im/compound-design-tokens/icons/error.svg"; import { Icon as PublicIcon } from "@vector-im/compound-design-tokens/icons/public.svg"; import { EventType, JoinRule, type Room } from "matrix-js-sdk/src/matrix"; import { ViewRoomOpts } from "@matrix-org/react-sdk-module-api/lib/lifecycles/RoomViewLifecycle"; import { useRoomName } from "../../../hooks/useRoomName"; import { RightPanelPhases } from "../../../stores/right-panel/RightPanelStorePhases"; import { useTopic } from "../../../hooks/room/useTopic"; import { useAccountData } from "../../../hooks/useAccountData"; import { useMatrixClientContext } from "../../../contexts/MatrixClientContext"; import { useRoomMemberCount, useRoomMembers } from "../../../hooks/useRoomMembers"; import { _t } from "../../../languageHandler"; import { Flex } from "../../utils/Flex"; import { Box } from "../../utils/Box"; import { useRoomCall } from "../../../hooks/room/useRoomCall"; import { useRoomThreadNotifications } from "../../../hooks/room/useRoomThreadNotifications"; import { NotificationColor } from "../../../stores/notifications/NotificationColor"; import { useGlobalNotificationState } from "../../../hooks/useGlobalNotificationState"; import SdkConfig from "../../../SdkConfig"; import { useFeatureEnabled } from "../../../hooks/useSettings"; import { useEncryptionStatus } from "../../../hooks/useEncryptionStatus"; import { E2EStatus } from "../../../utils/ShieldUtils"; import FacePile from "../elements/FacePile"; import { useRoomState } from "../../../hooks/useRoomState"; import RoomAvatar from "../avatars/RoomAvatar"; import { formatCount } from "../../../utils/FormattingUtils"; import RightPanelStore from "../../../stores/right-panel/RightPanelStore"; import { Linkify, topicToHtml } from "../../../HtmlUtils"; import PosthogTrackers from "../../../PosthogTrackers"; import { VideoRoomChatButton } from "./RoomHeader/VideoRoomChatButton"; import { RoomKnocksBar } from "./RoomKnocksBar"; /** * A helper to transform a notification color to the what the Compound Icon Button * expects */ function notificationColorToIndicator(color: NotificationColor): React.ComponentProps["indicator"] { if (color <= NotificationColor.None) { return undefined; } else if (color <= NotificationColor.Grey) { return "default"; } else { return "highlight"; } } export default function RoomHeader({ room, additionalButtons, }: { room: Room; additionalButtons?: ViewRoomOpts["buttons"]; }): JSX.Element { const client = useMatrixClientContext(); const roomName = useRoomName(room); const roomTopic = useTopic(room); const roomState = useRoomState(room); const members = useRoomMembers(room, 2500); const memberCount = useRoomMemberCount(room, { throttleWait: 2500 }); const { voiceCallDisabledReason, voiceCallClick, videoCallDisabledReason, videoCallClick } = useRoomCall(room); const groupCallsEnabled = useFeatureEnabled("feature_group_calls"); /** * A special mode where only Element Call is used. In this case we want to * hide the voice call button */ const useElementCallExclusively = useMemo(() => { return SdkConfig.get("element_call").use_exclusively && groupCallsEnabled; }, [groupCallsEnabled]); const threadNotifications = useRoomThreadNotifications(room); const globalNotificationState = useGlobalNotificationState(); const directRoomsList = useAccountData>(client, EventType.Direct); const [isDirectMessage, setDirectMessage] = useState(false); useEffect(() => { for (const [, dmRoomList] of Object.entries(directRoomsList)) { if (dmRoomList.includes(room?.roomId ?? "")) { setDirectMessage(true); break; } } }, [room, directRoomsList]); const e2eStatus = useEncryptionStatus(client, room); const notificationsEnabled = useFeatureEnabled("feature_notifications"); const roomTopicBody = useMemo( () => topicToHtml(roomTopic?.text, roomTopic?.html), [roomTopic?.html, roomTopic?.text], ); const askToJoinEnabled = useFeatureEnabled("feature_ask_to_join"); return ( <> {additionalButtons?.map((props) => { const label = props.label(); return ( { event.stopPropagation(); props.onClick(); }} > {typeof props.icon === "function" ? props.icon() : props.icon} ); })} {!useElementCallExclusively && ( )} {/* Renders nothing when room is not a video room */} { evt.stopPropagation(); RightPanelStore.instance.showOrHidePanel(RightPanelPhases.ThreadPanel); PosthogTrackers.trackInteraction("WebRoomHeaderButtonsThreadsButton", evt); }} aria-label={_t("common|threads")} > {notificationsEnabled && ( { evt.stopPropagation(); RightPanelStore.instance.showOrHidePanel(RightPanelPhases.NotificationPanel); }} aria-label={_t("notifications|enable_prompt_toast_title")} > )} {!isDirectMessage && ( { RightPanelStore.instance.showOrHidePanel(RightPanelPhases.RoomMemberList); e.stopPropagation(); }} > {formatCount(memberCount)} )} {askToJoinEnabled && } ); }