/* Copyright 2021 - 2022 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, { useContext, useEffect, useState } from "react"; import { Room, RoomEvent } from "matrix-js-sdk/src/models/room"; import { EventType, RoomType } from "matrix-js-sdk/src/@types/event"; import { ClientEvent } from "matrix-js-sdk/src/client"; import { _t } from "../../../languageHandler"; import { useEventEmitterState, useTypedEventEmitter, useTypedEventEmitterState } from "../../../hooks/useEventEmitter"; import { useFeatureEnabled } from "../../../hooks/useSettings"; import SpaceStore from "../../../stores/spaces/SpaceStore"; import { ChevronFace, ContextMenuTooltipButton, useContextMenu } from "../../structures/ContextMenu"; import SpaceContextMenu from "../context_menus/SpaceContextMenu"; import { HomeButtonContextMenu } from "../spaces/SpacePanel"; import IconizedContextMenu, { IconizedContextMenuOption, IconizedContextMenuOptionList, } from "../context_menus/IconizedContextMenu"; import defaultDispatcher from "../../../dispatcher/dispatcher"; import { shouldShowSpaceInvite, showAddExistingRooms, showCreateNewRoom, showCreateNewSubspace, showSpaceInvite, } from "../../../utils/space"; import { Action } from "../../../dispatcher/actions"; import { useDispatcher } from "../../../hooks/useDispatcher"; import InlineSpinner from "../elements/InlineSpinner"; import MatrixClientContext from "../../../contexts/MatrixClientContext"; import RoomListStore, { LISTS_UPDATE_EVENT } from "../../../stores/room-list/RoomListStore"; import { getMetaSpaceName, MetaSpace, SpaceKey, UPDATE_HOME_BEHAVIOUR, UPDATE_SELECTED_SPACE, } from "../../../stores/spaces"; import TooltipTarget from "../elements/TooltipTarget"; import { BetaPill } from "../beta/BetaCard"; import PosthogTrackers from "../../../PosthogTrackers"; import { ViewRoomPayload } from "../../../dispatcher/payloads/ViewRoomPayload"; import { useWebSearchMetrics } from "../dialogs/SpotlightDialog"; import { shouldShowComponent } from "../../../customisations/helpers/UIComponents"; import { UIComponent } from "../../../settings/UIFeature"; const contextMenuBelow = (elementRect: DOMRect) => { // align the context menu's icons with the icon which opened the context menu const left = elementRect.left + window.pageXOffset; const top = elementRect.bottom + window.pageYOffset + 12; const chevronFace = ChevronFace.None; return { left, top, chevronFace }; }; // Long-running actions that should trigger a spinner enum PendingActionType { JoinRoom, BulkRedact, } const usePendingActions = (): Map> => { const cli = useContext(MatrixClientContext); const [actions, setActions] = useState(new Map>()); const addAction = (type: PendingActionType, key: string) => { const keys = new Set(actions.get(type)); keys.add(key); setActions(new Map(actions).set(type, keys)); }; const removeAction = (type: PendingActionType, key: string) => { const keys = new Set(actions.get(type)); if (keys.delete(key)) { setActions(new Map(actions).set(type, keys)); } }; useDispatcher(defaultDispatcher, payload => { switch (payload.action) { case Action.JoinRoom: addAction(PendingActionType.JoinRoom, payload.roomId); break; case Action.JoinRoomReady: case Action.JoinRoomError: removeAction(PendingActionType.JoinRoom, payload.roomId); break; case Action.BulkRedactStart: addAction(PendingActionType.BulkRedact, payload.roomId); break; case Action.BulkRedactEnd: removeAction(PendingActionType.BulkRedact, payload.roomId); break; } }); useTypedEventEmitter(cli, ClientEvent.Room, (room: Room) => removeAction(PendingActionType.JoinRoom, room.roomId), ); return actions; }; interface IProps { onVisibilityChange?(): void; } const RoomListHeader = ({ onVisibilityChange }: IProps) => { const cli = useContext(MatrixClientContext); const [mainMenuDisplayed, mainMenuHandle, openMainMenu, closeMainMenu] = useContextMenu(); const [plusMenuDisplayed, plusMenuHandle, openPlusMenu, closePlusMenu] = useContextMenu(); const [spaceKey, activeSpace] = useEventEmitterState<[SpaceKey, Room | null]>( SpaceStore.instance, UPDATE_SELECTED_SPACE, () => [SpaceStore.instance.activeSpace, SpaceStore.instance.activeSpaceRoom], ); const allRoomsInHome = useEventEmitterState(SpaceStore.instance, UPDATE_HOME_BEHAVIOUR, () => { return SpaceStore.instance.allRoomsInHome; }); const videoRoomsEnabled = useFeatureEnabled("feature_video_rooms"); const pendingActions = usePendingActions(); const filterCondition = RoomListStore.instance.getFirstNameFilterCondition(); const count = useEventEmitterState(RoomListStore.instance, LISTS_UPDATE_EVENT, () => { if (filterCondition) { return Object.values(RoomListStore.instance.orderedLists).flat(1).length; } else { return null; } }); const canShowMainMenu = activeSpace || spaceKey === MetaSpace.Home; useEffect(() => { if (mainMenuDisplayed && !canShowMainMenu) { // Space changed under us and we no longer has a main menu to draw closeMainMenu(); } }, [closeMainMenu, canShowMainMenu, mainMenuDisplayed]); // we pass null for the queryLength to inhibit the metrics hook for when there is no filterCondition useWebSearchMetrics(count, filterCondition ? filterCondition.search.length : null, false); const spaceName = useTypedEventEmitterState(activeSpace, RoomEvent.Name, () => activeSpace?.name); useEffect(() => { if (onVisibilityChange) { onVisibilityChange(); } }, [count, onVisibilityChange]); if (typeof count === "number") { return
{ _t("%(count)s results", { count }) }
; } const canAddRooms = activeSpace?.currentState?.maySendStateEvent(EventType.SpaceChild, cli.getUserId()); const canCreateRooms = shouldShowComponent(UIComponent.CreateRooms); const canExploreRooms = shouldShowComponent(UIComponent.ExploreRooms); // If the user can't do anything on the plus menu, don't show it. This aims to target the // plus menu shown on the Home tab primarily: the user has options to use the menu for // communities and spaces, but is at risk of no options on the Home tab. const canShowPlusMenu = canCreateRooms || canExploreRooms || activeSpace; let contextMenu: JSX.Element; if (mainMenuDisplayed && mainMenuHandle.current) { let ContextMenuComponent; if (activeSpace) { ContextMenuComponent = SpaceContextMenu; } else { ContextMenuComponent = HomeButtonContextMenu; } contextMenu = ; } else if (plusMenuDisplayed && activeSpace) { let inviteOption: JSX.Element; if (shouldShowSpaceInvite(activeSpace)) { inviteOption = { e.preventDefault(); e.stopPropagation(); showSpaceInvite(activeSpace); closePlusMenu(); }} />; } let newRoomOptions: JSX.Element; if (activeSpace?.currentState.maySendStateEvent(EventType.RoomAvatar, cli.getUserId())) { newRoomOptions = <> { e.preventDefault(); e.stopPropagation(); showCreateNewRoom(activeSpace); PosthogTrackers.trackInteraction("WebRoomListHeaderPlusMenuCreateRoomItem", e); closePlusMenu(); }} /> { videoRoomsEnabled && { e.preventDefault(); e.stopPropagation(); showCreateNewRoom(activeSpace, RoomType.ElementVideo); closePlusMenu(); }} /> } ; } contextMenu = { inviteOption } { newRoomOptions } { e.preventDefault(); e.stopPropagation(); defaultDispatcher.dispatch({ action: Action.ViewRoom, room_id: activeSpace.roomId, metricsTrigger: undefined, // other }); closePlusMenu(); PosthogTrackers.trackInteraction("WebRoomListHeaderPlusMenuExploreRoomsItem", e); }} /> { e.preventDefault(); e.stopPropagation(); showAddExistingRooms(activeSpace); closePlusMenu(); }} disabled={!canAddRooms} tooltip={!canAddRooms && _t("You do not have permissions to add rooms to this space")} /> { e.preventDefault(); e.stopPropagation(); showCreateNewSubspace(activeSpace); closePlusMenu(); }} disabled={!canAddRooms} tooltip={!canAddRooms && _t("You do not have permissions to add spaces to this space")} > ; } else if (plusMenuDisplayed) { let newRoomOpts: JSX.Element; let joinRoomOpt: JSX.Element; if (canCreateRooms) { newRoomOpts = <> { e.preventDefault(); e.stopPropagation(); defaultDispatcher.dispatch({ action: "view_create_chat" }); closePlusMenu(); }} /> { e.preventDefault(); e.stopPropagation(); defaultDispatcher.dispatch({ action: "view_create_room" }); PosthogTrackers.trackInteraction("WebRoomListHeaderPlusMenuCreateRoomItem", e); closePlusMenu(); }} /> { videoRoomsEnabled && { e.preventDefault(); e.stopPropagation(); defaultDispatcher.dispatch({ action: "view_create_room", type: RoomType.ElementVideo, }); closePlusMenu(); }} /> } ; } if (canExploreRooms) { joinRoomOpt = ( { e.preventDefault(); e.stopPropagation(); defaultDispatcher.dispatch({ action: Action.ViewRoomDirectory }); closePlusMenu(); }} /> ); } contextMenu = { newRoomOpts } { joinRoomOpt } ; } let title: string; if (activeSpace) { title = spaceName; } else { title = getMetaSpaceName(spaceKey as MetaSpace, allRoomsInHome); } const pendingActionSummary = [...pendingActions.entries()] .filter(([type, keys]) => keys.size > 0) .map(([type, keys]) => { switch (type) { case PendingActionType.JoinRoom: return _t("Currently joining %(count)s rooms", { count: keys.size }); case PendingActionType.BulkRedact: return _t("Currently removing messages in %(count)s rooms", { count: keys.size }); } }) .join("\n"); let contextMenuButton: JSX.Element =
{ title }
; if (canShowMainMenu) { contextMenuButton = { title } ; } return
{ contextMenuButton } { pendingActionSummary ? : null } { canShowPlusMenu && } { contextMenu }
; }; export default RoomListHeader;