/* Copyright 2020 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, { useCallback, useContext, useEffect, useMemo, useState } from "react"; import classNames from "classnames"; import { Room } from "matrix-js-sdk/src/matrix"; import MatrixClientContext from "../../../contexts/MatrixClientContext"; import { useIsEncrypted } from "../../../hooks/useIsEncrypted"; import BaseCard, { Group } from "./BaseCard"; import { _t } from "../../../languageHandler"; import RoomAvatar from "../avatars/RoomAvatar"; import AccessibleButton, { ButtonEvent, IAccessibleButtonProps } from "../elements/AccessibleButton"; import defaultDispatcher from "../../../dispatcher/dispatcher"; import { RightPanelPhases } from "../../../stores/right-panel/RightPanelStorePhases"; import Modal from "../../../Modal"; import ShareDialog from "../dialogs/ShareDialog"; import { useEventEmitter } from "../../../hooks/useEventEmitter"; import WidgetUtils from "../../../utils/WidgetUtils"; import { IntegrationManagers } from "../../../integrations/IntegrationManagers"; import SettingsStore from "../../../settings/SettingsStore"; import TextWithTooltip from "../elements/TextWithTooltip"; import WidgetAvatar from "../avatars/WidgetAvatar"; import AccessibleTooltipButton from "../elements/AccessibleTooltipButton"; import WidgetStore, { IApp } from "../../../stores/WidgetStore"; import { E2EStatus } from "../../../utils/ShieldUtils"; import { RoomPermalinkCreator } from "../../../utils/permalinks/Permalinks"; import RoomContext from "../../../contexts/RoomContext"; import { UIComponent, UIFeature } from "../../../settings/UIFeature"; import { ChevronFace, ContextMenuTooltipButton, useContextMenu } from "../../structures/ContextMenu"; import { WidgetContextMenu } from "../context_menus/WidgetContextMenu"; import { useRoomMemberCount } from "../../../hooks/useRoomMembers"; import { useFeatureEnabled } from "../../../hooks/useSettings"; import { usePinnedEvents } from "./PinnedMessagesCard"; import { Container, MAX_PINNED, WidgetLayoutStore } from "../../../stores/widgets/WidgetLayoutStore"; import RoomName from "../elements/RoomName"; import UIStore from "../../../stores/UIStore"; import ExportDialog from "../dialogs/ExportDialog"; import RightPanelStore from "../../../stores/right-panel/RightPanelStore"; import PosthogTrackers from "../../../PosthogTrackers"; import { shouldShowComponent } from "../../../customisations/helpers/UIComponents"; import { PollHistoryDialog } from "../dialogs/PollHistoryDialog"; interface IProps { room: Room; permalinkCreator: RoomPermalinkCreator; onClose(): void; onSearchClick?: () => void; } interface IAppsSectionProps { room: Room; } interface IButtonProps extends IAccessibleButtonProps { className: string; onClick(ev: ButtonEvent): void; } const Button: React.FC = ({ children, className, onClick, ...props }) => { return ( {children} ); }; export const useWidgets = (room: Room): IApp[] => { const [apps, setApps] = useState(() => WidgetStore.instance.getApps(room.roomId)); const updateApps = useCallback(() => { // Copy the array so that we always trigger a re-render, as some updates mutate the array of apps/settings setApps([...WidgetStore.instance.getApps(room.roomId)]); }, [room]); useEffect(updateApps, [room, updateApps]); useEventEmitter(WidgetStore.instance, room.roomId, updateApps); useEventEmitter(WidgetLayoutStore.instance, WidgetLayoutStore.emissionForRoom(room), updateApps); return apps; }; interface IAppRowProps { app: IApp; room: Room; } const AppRow: React.FC = ({ app, room }) => { const name = WidgetUtils.getWidgetName(app); const dataTitle = WidgetUtils.getWidgetDataTitle(app); const subtitle = dataTitle && " - " + dataTitle; const [canModifyWidget, setCanModifyWidget] = useState(); useEffect(() => { setCanModifyWidget(WidgetUtils.canUserModifyWidgets(room.client, room.roomId)); }, [room.client, room.roomId]); const onOpenWidgetClick = (): void => { RightPanelStore.instance.pushCard({ phase: RightPanelPhases.Widget, state: { widgetId: app.id }, }); }; const isPinned = WidgetLayoutStore.instance.isInContainer(room, app, Container.Top); const togglePin = isPinned ? () => { WidgetLayoutStore.instance.moveToContainer(room, app, Container.Right); } : () => { WidgetLayoutStore.instance.moveToContainer(room, app, Container.Top); }; const [menuDisplayed, handle, openMenu, closeMenu] = useContextMenu(); let contextMenu; if (menuDisplayed) { const rect = handle.current?.getBoundingClientRect(); const rightMargin = rect?.right ?? 0; const topMargin = rect?.top ?? 0; contextMenu = ( ); } const cannotPin = !isPinned && !WidgetLayoutStore.instance.canAddToContainer(room, Container.Top); let pinTitle: string; if (cannotPin) { pinTitle = _t("You can only pin up to %(count)s widgets", { count: MAX_PINNED }); } else { pinTitle = isPinned ? _t("action|unpin") : _t("action|pin"); } const isMaximised = WidgetLayoutStore.instance.isInContainer(room, app, Container.Center); const toggleMaximised = isMaximised ? () => { WidgetLayoutStore.instance.moveToContainer(room, app, Container.Right); } : () => { WidgetLayoutStore.instance.moveToContainer(room, app, Container.Center); }; const maximiseTitle = isMaximised ? _t("action|close") : _t("action|maximise"); let openTitle = ""; if (isPinned) { openTitle = _t("Unpin this widget to view it in this panel"); } else if (isMaximised) { openTitle = _t("Close this widget to view it in this panel"); } const classes = classNames("mx_BaseCard_Button mx_RoomSummaryCard_Button", { mx_RoomSummaryCard_Button_pinned: isPinned, mx_RoomSummaryCard_Button_maximised: isMaximised, }); return (
{name} {subtitle} {canModifyWidget && ( )} {contextMenu}
); }; const AppsSection: React.FC = ({ room }) => { const apps = useWidgets(room); // Filter out virtual widgets const realApps = useMemo(() => apps.filter((app) => app.eventId !== undefined), [apps]); const onManageIntegrations = (): void => { const managers = IntegrationManagers.sharedInstance(); if (!managers.hasManager()) { managers.openNoManagerDialog(); } else { // noinspection JSIgnoredPromiseFromCall managers.getPrimaryManager()?.open(room); } }; let copyLayoutBtn: JSX.Element | null = null; if (realApps.length > 0 && WidgetLayoutStore.instance.canCopyLayoutToRoom(room)) { copyLayoutBtn = ( WidgetLayoutStore.instance.copyLayoutToRoom(room)}> {_t("Set my room layout for everyone")} ); } return ( {realApps.map((app) => ( ))} {copyLayoutBtn} {realApps.length > 0 ? _t("Edit widgets, bridges & bots") : _t("Add widgets, bridges & bots")} ); }; const onRoomMembersClick = (ev: ButtonEvent): void => { RightPanelStore.instance.pushCard({ phase: RightPanelPhases.RoomMemberList }, true); PosthogTrackers.trackInteraction("WebRightPanelRoomInfoPeopleButton", ev); }; const onRoomFilesClick = (): void => { RightPanelStore.instance.pushCard({ phase: RightPanelPhases.FilePanel }, true); }; const onRoomPinsClick = (): void => { RightPanelStore.instance.pushCard({ phase: RightPanelPhases.PinnedMessages }, true); }; const onRoomSettingsClick = (ev: ButtonEvent): void => { defaultDispatcher.dispatch({ action: "open_room_settings" }); PosthogTrackers.trackInteraction("WebRightPanelRoomInfoSettingsButton", ev); }; const RoomSummaryCard: React.FC = ({ room, permalinkCreator, onClose, onSearchClick }) => { const cli = useContext(MatrixClientContext); const onShareRoomClick = (): void => { Modal.createDialog(ShareDialog, { target: room, }); }; const onRoomExportClick = async (): Promise => { Modal.createDialog(ExportDialog, { room, }); }; const onRoomPollHistoryClick = (): void => { Modal.createDialog(PollHistoryDialog, { room, matrixClient: cli, permalinkCreator, }); }; const isRoomEncrypted = useIsEncrypted(cli, room); const roomContext = useContext(RoomContext); const e2eStatus = roomContext.e2eStatus; const videoRoomsEnabled = useFeatureEnabled("feature_video_rooms"); const elementCallVideoRoomsEnabled = useFeatureEnabled("feature_element_call_video_rooms"); const isVideoRoom = videoRoomsEnabled && (room.isElementVideoRoom() || (elementCallVideoRoomsEnabled && room.isCallRoom())); const alias = room.getCanonicalAlias() || room.getAltAliases()[0] || ""; const header = (
{(name) => (

{name}

)}
{alias}
); const memberCount = useRoomMemberCount(room); const pinningEnabled = useFeatureEnabled("feature_pinning"); const pinCount = usePinnedEvents(pinningEnabled ? room : undefined)?.length; return ( {!isVideoRoom && ( )} {!isVideoRoom && ( )} {pinningEnabled && !isVideoRoom && ( )} {!isVideoRoom && ( )} {SettingsStore.getValue(UIFeature.Widgets) && !isVideoRoom && shouldShowComponent(UIComponent.AddIntegrations) && } ); }; export default RoomSummaryCard;