/* Copyright 2015, 2016 OpenMarket Ltd Copyright 2017, 2018 Vector Creations Ltd Copyright 2019 Michael Telatynski <7t3chguy@gmail.com> Copyright 2019, 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, { ReactNode, useCallback, useContext, useEffect, useMemo, useState } from "react"; import classNames from "classnames"; import { ClientEvent, MatrixClient, RoomMember, Room, RoomStateEvent, MatrixEvent, User, Device, EventType, } from "matrix-js-sdk/src/matrix"; import { KnownMembership } from "matrix-js-sdk/src/types"; import { UserVerificationStatus, VerificationRequest } from "matrix-js-sdk/src/crypto-api"; import { logger } from "matrix-js-sdk/src/logger"; import { CryptoEvent } from "matrix-js-sdk/src/crypto"; import { Heading, MenuItem, Text } from "@vector-im/compound-web"; import ChatIcon from "@vector-im/compound-design-tokens/assets/web/icons/chat"; import CheckIcon from "@vector-im/compound-design-tokens/assets/web/icons/check"; import ShareIcon from "@vector-im/compound-design-tokens/assets/web/icons/share"; import MentionIcon from "@vector-im/compound-design-tokens/assets/web/icons/mention"; import { Icon as InviteIcon } from "@vector-im/compound-design-tokens/icons/user-add.svg"; import BlockIcon from "@vector-im/compound-design-tokens/assets/web/icons/block"; import DeleteIcon from "@vector-im/compound-design-tokens/assets/web/icons/delete"; import CloseIcon from "@vector-im/compound-design-tokens/assets/web/icons/close"; import { Icon as ChatProblemIcon } from "@vector-im/compound-design-tokens/icons/chat-problem.svg"; import { Icon as VisibilityOffIcon } from "@vector-im/compound-design-tokens/icons/visibility-off.svg"; import LeaveIcon from "@vector-im/compound-design-tokens/assets/web/icons/leave"; import dis from "../../../dispatcher/dispatcher"; import Modal from "../../../Modal"; import { _t, UserFriendlyError } from "../../../languageHandler"; import DMRoomMap from "../../../utils/DMRoomMap"; import AccessibleButton, { ButtonEvent } from "../elements/AccessibleButton"; import SdkConfig from "../../../SdkConfig"; import MultiInviter from "../../../utils/MultiInviter"; import E2EIcon from "../rooms/E2EIcon"; import { useTypedEventEmitter } from "../../../hooks/useEventEmitter"; import { textualPowerLevel } from "../../../Roles"; import MatrixClientContext from "../../../contexts/MatrixClientContext"; import { RightPanelPhases } from "../../../stores/right-panel/RightPanelStorePhases"; import EncryptionPanel from "./EncryptionPanel"; import { useAsyncMemo } from "../../../hooks/useAsyncMemo"; import { legacyVerifyUser, verifyDevice, verifyUser } from "../../../verification"; import { Action } from "../../../dispatcher/actions"; import { useIsEncrypted } from "../../../hooks/useIsEncrypted"; import BaseCard from "./BaseCard"; import { E2EStatus } from "../../../utils/ShieldUtils"; import ImageView from "../elements/ImageView"; import Spinner from "../elements/Spinner"; import PowerSelector from "../elements/PowerSelector"; import MemberAvatar from "../avatars/MemberAvatar"; import PresenceLabel from "../rooms/PresenceLabel"; import BulkRedactDialog from "../dialogs/BulkRedactDialog"; import ShareDialog from "../dialogs/ShareDialog"; import ErrorDialog from "../dialogs/ErrorDialog"; import QuestionDialog from "../dialogs/QuestionDialog"; import ConfirmUserActionDialog from "../dialogs/ConfirmUserActionDialog"; import { mediaFromMxc } from "../../../customisations/Media"; import { ComposerInsertPayload } from "../../../dispatcher/payloads/ComposerInsertPayload"; import ConfirmSpaceUserActionDialog from "../dialogs/ConfirmSpaceUserActionDialog"; import { bulkSpaceBehaviour } from "../../../utils/space"; import { shouldShowComponent } from "../../../customisations/helpers/UIComponents"; import { UIComponent } from "../../../settings/UIFeature"; import { TimelineRenderingType } from "../../../contexts/RoomContext"; import RightPanelStore from "../../../stores/right-panel/RightPanelStore"; import { IRightPanelCardState } from "../../../stores/right-panel/RightPanelStoreIPanelState"; import UserIdentifierCustomisations from "../../../customisations/UserIdentifier"; import PosthogTrackers from "../../../PosthogTrackers"; import { ViewRoomPayload } from "../../../dispatcher/payloads/ViewRoomPayload"; import { DirectoryMember, startDmOnFirstMessage } from "../../../utils/direct-messages"; import { SdkContextClass } from "../../../contexts/SDKContext"; import { asyncSome } from "../../../utils/arrays"; import { Flex } from "../../utils/Flex"; import CopyableText from "../elements/CopyableText"; export interface IDevice extends Device { ambiguous?: boolean; } export const disambiguateDevices = (devices: IDevice[]): void => { const names = Object.create(null); for (let i = 0; i < devices.length; i++) { const name = devices[i].displayName ?? ""; const indexList = names[name] || []; indexList.push(i); names[name] = indexList; } for (const name in names) { if (names[name].length > 1) { names[name].forEach((j: number) => { devices[j].ambiguous = true; }); } } }; export const getE2EStatus = async ( cli: MatrixClient, userId: string, devices: IDevice[], ): Promise => { const crypto = cli.getCrypto(); if (!crypto) return undefined; const isMe = userId === cli.getUserId(); const userTrust = await crypto.getUserVerificationStatus(userId); if (!userTrust.isCrossSigningVerified()) { return userTrust.wasCrossSigningVerified() ? E2EStatus.Warning : E2EStatus.Normal; } const anyDeviceUnverified = await asyncSome(devices, async (device) => { const { deviceId } = device; // For your own devices, we use the stricter check of cross-signing // verification to encourage everyone to trust their own devices via // cross-signing so that other users can then safely trust you. // For other people's devices, the more general verified check that // includes locally verified devices can be used. const deviceTrust = await crypto.getDeviceVerificationStatus(userId, deviceId); return isMe ? !deviceTrust?.crossSigningVerified : !deviceTrust?.isVerified(); }); return anyDeviceUnverified ? E2EStatus.Warning : E2EStatus.Verified; }; /** * Converts the member to a DirectoryMember and starts a DM with them. */ async function openDmForUser(matrixClient: MatrixClient, user: Member): Promise { const avatarUrl = user instanceof User ? user.avatarUrl : user.getMxcAvatarUrl(); const startDmUser = new DirectoryMember({ user_id: user.userId, display_name: user.rawDisplayName, avatar_url: avatarUrl, }); await startDmOnFirstMessage(matrixClient, [startDmUser]); } type SetUpdating = (updating: boolean) => void; function useHasCrossSigningKeys( cli: MatrixClient, member: User, canVerify: boolean, setUpdating: SetUpdating, ): boolean | undefined { return useAsyncMemo(async () => { if (!canVerify) { return undefined; } setUpdating(true); try { return await cli.getCrypto()?.userHasCrossSigningKeys(member.userId, true); } finally { setUpdating(false); } }, [cli, member, canVerify]); } /** * Display one device and the related actions * @param userId current user id * @param device device to display * @param isUserVerified false when the user is not verified * @constructor */ export function DeviceItem({ userId, device, isUserVerified, }: { userId: string; device: IDevice; isUserVerified: boolean; }): JSX.Element { const cli = useContext(MatrixClientContext); const isMe = userId === cli.getUserId(); /** is the device verified? */ const isVerified = useAsyncMemo(async () => { const deviceTrust = await cli.getCrypto()?.getDeviceVerificationStatus(userId, device.deviceId); if (!deviceTrust) return false; // For your own devices, we use the stricter check of cross-signing // verification to encourage everyone to trust their own devices via // cross-signing so that other users can then safely trust you. // For other people's devices, the more general verified check that // includes locally verified devices can be used. return isMe ? deviceTrust.crossSigningVerified : deviceTrust.isVerified(); }, [cli, userId, device]); const classes = classNames("mx_UserInfo_device", { mx_UserInfo_device_verified: isVerified, mx_UserInfo_device_unverified: !isVerified, }); const iconClasses = classNames("mx_E2EIcon", { mx_E2EIcon_normal: !isUserVerified, mx_E2EIcon_verified: isVerified, mx_E2EIcon_warning: isUserVerified && !isVerified, }); const onDeviceClick = (): void => { const user = cli.getUser(userId); if (user) { verifyDevice(cli, user, device); } }; let deviceName; if (!device.displayName?.trim()) { deviceName = device.deviceId; } else { deviceName = device.ambiguous ? device.displayName + " (" + device.deviceId + ")" : device.displayName; } let trustedLabel: string | undefined; if (isUserVerified) trustedLabel = isVerified ? _t("common|trusted") : _t("common|not_trusted"); if (isVerified === undefined) { // we're still deciding if the device is verified return
; } else if (isVerified) { return (
{deviceName}
{trustedLabel}
); } else { return (
{deviceName}
{trustedLabel}
); } } /** * Display a list of devices * @param devices devices to display * @param userId current user id * @param loading displays a spinner instead of the device section * @param isUserVerified is false when * - the user is not verified, or * - `MatrixClient.getCrypto.getUserVerificationStatus` async call is in progress (in which case `loading` will also be `true`) * @constructor */ function DevicesSection({ devices, userId, loading, isUserVerified, }: { devices: IDevice[]; userId: string; loading: boolean; isUserVerified: boolean; }): JSX.Element { const cli = useContext(MatrixClientContext); const [isExpanded, setExpanded] = useState(false); const deviceTrusts = useAsyncMemo(() => { const cryptoApi = cli.getCrypto(); if (!cryptoApi) return Promise.resolve(undefined); return Promise.all(devices.map((d) => cryptoApi.getDeviceVerificationStatus(userId, d.deviceId))); }, [cli, userId, devices]); if (loading || deviceTrusts === undefined) { // still loading return ; } const isMe = userId === cli.getUserId(); let expandSectionDevices: IDevice[] = []; const unverifiedDevices: IDevice[] = []; let expandCountCaption; let expandHideCaption; let expandIconClasses = "mx_E2EIcon"; const dehydratedDeviceIds: string[] = []; for (const device of devices) { if (device.dehydrated) { dehydratedDeviceIds.push(device.deviceId); } } // If the user has exactly one device marked as dehydrated, we consider // that as the dehydrated device, and hide it as a normal device (but // indicate that the user is using a dehydrated device). If the user has // more than one, that is anomalous, and we show all the devices so that // nothing is hidden. const dehydratedDeviceId: string | undefined = dehydratedDeviceIds.length == 1 ? dehydratedDeviceIds[0] : undefined; let dehydratedDeviceInExpandSection = false; if (isUserVerified) { for (let i = 0; i < devices.length; ++i) { const device = devices[i]; const deviceTrust = deviceTrusts[i]; // For your own devices, we use the stricter check of cross-signing // verification to encourage everyone to trust their own devices via // cross-signing so that other users can then safely trust you. // For other people's devices, the more general verified check that // includes locally verified devices can be used. const isVerified = deviceTrust && (isMe ? deviceTrust.crossSigningVerified : deviceTrust.isVerified()); if (isVerified) { // don't show dehydrated device as a normal device, if it's // verified if (device.deviceId === dehydratedDeviceId) { dehydratedDeviceInExpandSection = true; } else { expandSectionDevices.push(device); } } else { unverifiedDevices.push(device); } } expandCountCaption = _t("user_info|count_of_verified_sessions", { count: expandSectionDevices.length }); expandHideCaption = _t("user_info|hide_verified_sessions"); expandIconClasses += " mx_E2EIcon_verified"; } else { if (dehydratedDeviceId) { devices = devices.filter((device) => device.deviceId !== dehydratedDeviceId); dehydratedDeviceInExpandSection = true; } expandSectionDevices = devices; expandCountCaption = _t("user_info|count_of_sessions", { count: devices.length }); expandHideCaption = _t("user_info|hide_sessions"); expandIconClasses += " mx_E2EIcon_normal"; } let expandButton; if (expandSectionDevices.length) { if (isExpanded) { expandButton = ( setExpanded(false)}>
{expandHideCaption}
); } else { expandButton = ( setExpanded(true)}>
{expandCountCaption}
); } } let deviceList = unverifiedDevices.map((device, i) => { return ; }); if (isExpanded) { const keyStart = unverifiedDevices.length; deviceList = deviceList.concat( expandSectionDevices.map((device, i) => { return ( ); }), ); if (dehydratedDeviceInExpandSection) { deviceList.push(
{_t("user_info|dehydrated_device_enabled")}
); } } return (
{deviceList}
{expandButton}
); } const MessageButton = ({ member }: { member: Member }): JSX.Element => { const cli = useContext(MatrixClientContext); const [busy, setBusy] = useState(false); return ( { ev.preventDefault(); if (busy) return; setBusy(true); await openDmForUser(cli, member); setBusy(false); }} disabled={busy} label={_t("user_info|send_message")} Icon={ChatIcon} /> ); }; export const UserOptionsSection: React.FC<{ member: Member; canInvite: boolean; isSpace?: boolean; }> = ({ member, canInvite, isSpace, children }) => { const cli = useContext(MatrixClientContext); let insertPillButton: JSX.Element | undefined; let inviteUserButton: JSX.Element | undefined; let readReceiptButton: JSX.Element | undefined; const isMe = member.userId === cli.getUserId(); const onShareUserClick = (): void => { Modal.createDialog(ShareDialog, { target: member, }); }; // Only allow the user to ignore the user if its not ourselves // same goes for jumping to read receipt if (!isMe) { if (member instanceof RoomMember && member.roomId && !isSpace) { const onReadReceiptButton = function (): void { const room = cli.getRoom(member.roomId); dis.dispatch({ action: Action.ViewRoom, highlighted: true, // this could return null, the default prevents a type error event_id: room?.getEventReadUpTo(member.userId) || undefined, room_id: member.roomId, metricsTrigger: undefined, // room doesn't change }); }; const onInsertPillButton = function (): void { dis.dispatch({ action: Action.ComposerInsert, userId: member.userId, timelineRenderingType: TimelineRenderingType.Room, }); }; const room = member instanceof RoomMember ? cli.getRoom(member.roomId) : undefined; if (room?.getEventReadUpTo(member.userId)) { readReceiptButton = ( { ev.preventDefault(); onReadReceiptButton(); }} label={_t("user_info|jump_to_rr_button")} Icon={CheckIcon} /> ); } insertPillButton = ( { ev.preventDefault(); onInsertPillButton(); }} label={_t("action|mention")} Icon={MentionIcon} /> ); } if ( member instanceof RoomMember && canInvite && (member?.membership ?? KnownMembership.Leave) === KnownMembership.Leave && shouldShowComponent(UIComponent.InviteUsers) ) { const roomId = member && member.roomId ? member.roomId : SdkContextClass.instance.roomViewStore.getRoomId(); const onInviteUserButton = async (ev: Event): Promise => { try { // We use a MultiInviter to re-use the invite logic, even though we're only inviting one user. const inviter = new MultiInviter(cli, roomId || ""); await inviter.invite([member.userId]).then(() => { if (inviter.getCompletionState(member.userId) !== "invited") { const errorStringFromInviterUtility = inviter.getErrorText(member.userId); if (errorStringFromInviterUtility) { throw new Error(errorStringFromInviterUtility); } else { throw new UserFriendlyError("slash_command|invite_failed", { user: member.userId, roomId, cause: undefined, }); } } }); } catch (err) { const description = err instanceof Error ? err.message : _t("invite|failed_generic"); Modal.createDialog(ErrorDialog, { title: _t("invite|failed_title"), description, }); } PosthogTrackers.trackInteraction("WebRightPanelRoomUserInfoInviteButton", ev); }; inviteUserButton = ( { ev.preventDefault(); onInviteUserButton(ev); }} label={_t("action|invite")} Icon={InviteIcon} /> ); } } const shareUserButton = ( { ev.preventDefault(); onShareUserClick(); }} label={_t("user_info|share_button")} Icon={ShareIcon} /> ); const directMessageButton = isMe || !shouldShowComponent(UIComponent.CreateRooms) ? null : ; return ( {children} {directMessageButton} {inviteUserButton} {readReceiptButton} {shareUserButton} {insertPillButton} ); }; export const warnSelfDemote = async (isSpace: boolean): Promise => { const { finished } = Modal.createDialog(QuestionDialog, { title: _t("user_info|demote_self_confirm_title"), description: (
{isSpace ? _t("user_info|demote_self_confirm_description_space") : _t("user_info|demote_self_confirm_room")}
), button: _t("user_info|demote_button"), }); const [confirmed] = await finished; return !!confirmed; }; const Container: React.FC<{ children: ReactNode; }> = ({ children }) => { return
{children}
; }; interface IPowerLevelsContent { events?: Record; // eslint-disable-next-line camelcase users_default?: number; // eslint-disable-next-line camelcase events_default?: number; // eslint-disable-next-line camelcase state_default?: number; ban?: number; kick?: number; redact?: number; } export const isMuted = (member: RoomMember, powerLevelContent: IPowerLevelsContent): boolean => { if (!powerLevelContent || !member) return false; const levelToSend = (powerLevelContent.events ? powerLevelContent.events["m.room.message"] : null) || powerLevelContent.events_default; // levelToSend could be undefined as .events_default is optional. Coercing in this case using // Number() would always return false, so this preserves behaviour // FIXME: per the spec, if `events_default` is unset, it defaults to zero. If // the member has a negative powerlevel, this will give an incorrect result. if (levelToSend === undefined) return false; return member.powerLevel < levelToSend; }; export const getPowerLevels = (room: Room): IPowerLevelsContent => room?.currentState?.getStateEvents(EventType.RoomPowerLevels, "")?.getContent() || {}; export const useRoomPowerLevels = (cli: MatrixClient, room: Room): IPowerLevelsContent => { const [powerLevels, setPowerLevels] = useState(getPowerLevels(room)); const update = useCallback( (ev?: MatrixEvent) => { if (!room) return; if (ev && ev.getType() !== EventType.RoomPowerLevels) return; setPowerLevels(getPowerLevels(room)); }, [room], ); useTypedEventEmitter(cli, RoomStateEvent.Events, update); useEffect(() => { update(); return () => { setPowerLevels({}); }; }, [update]); return powerLevels; }; interface IBaseProps { member: RoomMember; isUpdating: boolean; startUpdating(): void; stopUpdating(): void; } export const RoomKickButton = ({ room, member, isUpdating, startUpdating, stopUpdating, }: Omit): JSX.Element | null => { const cli = useContext(MatrixClientContext); // check if user can be kicked/disinvited if (member.membership !== KnownMembership.Invite && member.membership !== KnownMembership.Join) return <>; const onKick = async (): Promise => { if (isUpdating) return; // only allow one operation at a time startUpdating(); const commonProps = { member, action: room.isSpaceRoom() ? member.membership === KnownMembership.Invite ? _t("user_info|disinvite_button_space") : _t("user_info|kick_button_space") : member.membership === KnownMembership.Invite ? _t("user_info|disinvite_button_room") : _t("user_info|kick_button_room"), title: member.membership === KnownMembership.Invite ? _t("user_info|disinvite_button_room_name", { roomName: room.name }) : _t("user_info|kick_button_room_name", { roomName: room.name }), askReason: member.membership === KnownMembership.Join, danger: true, }; let finished: Promise<[success?: boolean, reason?: string, rooms?: Room[]]>; if (room.isSpaceRoom()) { ({ finished } = Modal.createDialog( ConfirmSpaceUserActionDialog, { ...commonProps, space: room, spaceChildFilter: (child: Room) => { // Return true if the target member is not banned and we have sufficient PL to ban them const myMember = child.getMember(cli.credentials.userId || ""); const theirMember = child.getMember(member.userId); return ( !!myMember && !!theirMember && theirMember.membership === member.membership && myMember.powerLevel > theirMember.powerLevel && child.currentState.hasSufficientPowerLevelFor("kick", myMember.powerLevel) ); }, allLabel: _t("user_info|kick_button_space_everything"), specificLabel: _t("user_info|kick_space_specific"), warningMessage: _t("user_info|kick_space_warning"), }, "mx_ConfirmSpaceUserActionDialog_wrapper", )); } else { ({ finished } = Modal.createDialog(ConfirmUserActionDialog, commonProps)); } const [proceed, reason, rooms = []] = await finished; if (!proceed) { stopUpdating(); return; } bulkSpaceBehaviour(room, rooms, (room) => cli.kick(room.roomId, member.userId, reason || undefined)) .then( () => { // NO-OP; rely on the m.room.member event coming down else we could // get out of sync if we force setState here! logger.log("Kick success"); }, function (err) { logger.error("Kick error: " + err); Modal.createDialog(ErrorDialog, { title: _t("user_info|error_kicking_user"), description: err && err.message ? err.message : "Operation failed", }); }, ) .finally(() => { stopUpdating(); }); }; const kickLabel = room.isSpaceRoom() ? member.membership === KnownMembership.Invite ? _t("user_info|disinvite_button_space") : _t("user_info|kick_button_space") : member.membership === KnownMembership.Invite ? _t("user_info|disinvite_button_room") : _t("user_info|kick_button_room"); return ( { ev.preventDefault(); onKick(); }} disabled={isUpdating} label={kickLabel} kind="critical" Icon={LeaveIcon} /> ); }; const RedactMessagesButton: React.FC = ({ member }) => { const cli = useContext(MatrixClientContext); const onRedactAllMessages = (): void => { const room = cli.getRoom(member.roomId); if (!room) return; Modal.createDialog(BulkRedactDialog, { matrixClient: cli, room, member, }); }; return ( { ev.preventDefault(); onRedactAllMessages(); }} label={_t("user_info|redact_button")} kind="critical" Icon={CloseIcon} /> ); }; export const BanToggleButton = ({ room, member, isUpdating, startUpdating, stopUpdating, }: Omit): JSX.Element => { const cli = useContext(MatrixClientContext); const isBanned = member.membership === KnownMembership.Ban; const onBanOrUnban = async (): Promise => { if (isUpdating) return; // only allow one operation at a time startUpdating(); const commonProps = { member, action: room.isSpaceRoom() ? isBanned ? _t("user_info|unban_button_space") : _t("user_info|ban_button_space") : isBanned ? _t("user_info|unban_button_room") : _t("user_info|ban_button_room"), title: isBanned ? _t("user_info|unban_room_confirm_title", { roomName: room.name }) : _t("user_info|ban_room_confirm_title", { roomName: room.name }), askReason: !isBanned, danger: !isBanned, }; let finished: Promise<[success?: boolean, reason?: string, rooms?: Room[]]>; if (room.isSpaceRoom()) { ({ finished } = Modal.createDialog( ConfirmSpaceUserActionDialog, { ...commonProps, space: room, spaceChildFilter: isBanned ? (child: Room) => { // Return true if the target member is banned and we have sufficient PL to unban const myMember = child.getMember(cli.credentials.userId || ""); const theirMember = child.getMember(member.userId); return ( !!myMember && !!theirMember && theirMember.membership === KnownMembership.Ban && myMember.powerLevel > theirMember.powerLevel && child.currentState.hasSufficientPowerLevelFor("ban", myMember.powerLevel) ); } : (child: Room) => { // Return true if the target member isn't banned and we have sufficient PL to ban const myMember = child.getMember(cli.credentials.userId || ""); const theirMember = child.getMember(member.userId); return ( !!myMember && !!theirMember && theirMember.membership !== KnownMembership.Ban && myMember.powerLevel > theirMember.powerLevel && child.currentState.hasSufficientPowerLevelFor("ban", myMember.powerLevel) ); }, allLabel: isBanned ? _t("user_info|unban_space_everything") : _t("user_info|ban_space_everything"), specificLabel: isBanned ? _t("user_info|unban_space_specific") : _t("user_info|ban_space_specific"), warningMessage: isBanned ? _t("user_info|unban_space_warning") : _t("user_info|kick_space_warning"), }, "mx_ConfirmSpaceUserActionDialog_wrapper", )); } else { ({ finished } = Modal.createDialog(ConfirmUserActionDialog, commonProps)); } const [proceed, reason, rooms = []] = await finished; if (!proceed) { stopUpdating(); return; } const fn = (roomId: string): Promise => { if (isBanned) { return cli.unban(roomId, member.userId); } else { return cli.ban(roomId, member.userId, reason || undefined); } }; bulkSpaceBehaviour(room, rooms, (room) => fn(room.roomId)) .then( () => { // NO-OP; rely on the m.room.member event coming down else we could // get out of sync if we force setState here! logger.log("Ban success"); }, function (err) { logger.error("Ban error: " + err); Modal.createDialog(ErrorDialog, { title: _t("common|error"), description: _t("user_info|error_ban_user"), }); }, ) .finally(() => { stopUpdating(); }); }; let label = room.isSpaceRoom() ? _t("user_info|ban_button_space") : _t("user_info|ban_button_room"); if (isBanned) { label = room.isSpaceRoom() ? _t("user_info|unban_button_space") : _t("user_info|unban_button_room"); } return ( { ev.preventDefault(); onBanOrUnban(); }} disabled={isUpdating} label={label} kind="critical" Icon={ChatProblemIcon} /> ); }; interface IBaseRoomProps extends IBaseProps { room: Room; powerLevels: IPowerLevelsContent; children?: ReactNode; } // We do not show a Mute button for ourselves so it doesn't need to handle warning self demotion const MuteToggleButton: React.FC = ({ member, room, powerLevels, isUpdating, startUpdating, stopUpdating, }) => { const cli = useContext(MatrixClientContext); // Don't show the mute/unmute option if the user is not in the room if (member.membership !== KnownMembership.Join) return null; const muted = isMuted(member, powerLevels); const onMuteToggle = async (): Promise => { if (isUpdating) return; // only allow one operation at a time startUpdating(); const roomId = member.roomId; const target = member.userId; const powerLevelEvent = room.currentState.getStateEvents("m.room.power_levels", ""); const powerLevels = powerLevelEvent?.getContent(); const levelToSend = powerLevels?.events?.["m.room.message"] ?? powerLevels?.events_default; let level; if (muted) { // unmute level = levelToSend; } else { // mute level = levelToSend - 1; } level = parseInt(level); if (isNaN(level)) { stopUpdating(); return; } cli.setPowerLevel(roomId, target, level) .then( () => { // NO-OP; rely on the m.room.member event coming down else we could // get out of sync if we force setState here! logger.log("Mute toggle success"); }, function (err) { logger.error("Mute error: " + err); Modal.createDialog(ErrorDialog, { title: _t("common|error"), description: _t("user_info|error_mute_user"), }); }, ) .finally(() => { stopUpdating(); }); }; const muteLabel = muted ? _t("common|unmute") : _t("common|mute"); return ( { ev.preventDefault(); onMuteToggle(); }} disabled={isUpdating} label={muteLabel} kind="critical" Icon={VisibilityOffIcon} /> ); }; const IgnoreToggleButton: React.FC<{ member: User | RoomMember; }> = ({ member }) => { const cli = useContext(MatrixClientContext); const unignore = useCallback(() => { const ignoredUsers = cli.getIgnoredUsers(); const index = ignoredUsers.indexOf(member.userId); if (index !== -1) ignoredUsers.splice(index, 1); cli.setIgnoredUsers(ignoredUsers); }, [cli, member]); const ignore = useCallback(async () => { const name = (member instanceof User ? member.displayName : member.name) || member.userId; const { finished } = Modal.createDialog(QuestionDialog, { title: _t("user_info|ignore_confirm_title", { user: name }), description:
{_t("user_info|ignore_confirm_description")}
, button: _t("action|ignore"), }); const [confirmed] = await finished; if (confirmed) { const ignoredUsers = cli.getIgnoredUsers(); ignoredUsers.push(member.userId); cli.setIgnoredUsers(ignoredUsers); } }, [cli, member]); // Check whether the user is ignored const [isIgnored, setIsIgnored] = useState(cli.isUserIgnored(member.userId)); // Recheck if the user or client changes useEffect(() => { setIsIgnored(cli.isUserIgnored(member.userId)); }, [cli, member.userId]); // Recheck also if we receive new accountData m.ignored_user_list const accountDataHandler = useCallback( (ev) => { if (ev.getType() === "m.ignored_user_list") { setIsIgnored(cli.isUserIgnored(member.userId)); } }, [cli, member.userId], ); useTypedEventEmitter(cli, ClientEvent.AccountData, accountDataHandler); return ( { ev.preventDefault(); if (isIgnored) { unignore(); } else { ignore(); } }} label={isIgnored ? _t("user_info|unignore_button") : _t("user_info|ignore_button")} kind="critical" Icon={BlockIcon} /> ); }; export const RoomAdminToolsContainer: React.FC = ({ room, children, member, isUpdating, startUpdating, stopUpdating, powerLevels, }) => { const cli = useContext(MatrixClientContext); let kickButton; let banButton; let muteButton; let redactButton; const editPowerLevel = (powerLevels.events ? powerLevels.events["m.room.power_levels"] : null) || powerLevels.state_default; // if these do not exist in the event then they should default to 50 as per the spec const { ban: banPowerLevel = 50, kick: kickPowerLevel = 50, redact: redactPowerLevel = 50 } = powerLevels; const me = room.getMember(cli.getUserId() || ""); if (!me) { // we aren't in the room, so return no admin tooling return
; } const isMe = me.userId === member.userId; const canAffectUser = member.powerLevel < me.powerLevel || isMe; if (!isMe && canAffectUser && me.powerLevel >= kickPowerLevel) { kickButton = ( ); } if (me.powerLevel >= redactPowerLevel && !room.isSpaceRoom()) { redactButton = ( ); } if (!isMe && canAffectUser && me.powerLevel >= banPowerLevel) { banButton = ( ); } if (!isMe && canAffectUser && me.powerLevel >= Number(editPowerLevel) && !room.isSpaceRoom()) { muteButton = ( ); } if (kickButton || banButton || muteButton || redactButton || children) { return ( {muteButton} {redactButton} {kickButton} {banButton} {children} ); } return
; }; const useIsSynapseAdmin = (cli?: MatrixClient): boolean => { return useAsyncMemo(async () => (cli ? cli.isSynapseAdministrator().catch(() => false) : false), [cli], false); }; const useHomeserverSupportsCrossSigning = (cli: MatrixClient): boolean => { return useAsyncMemo( async () => { return cli.doesServerSupportUnstableFeature("org.matrix.e2e_cross_signing"); }, [cli], false, ); }; interface IRoomPermissions { modifyLevelMax: number; canEdit: boolean; canInvite: boolean; } function useRoomPermissions(cli: MatrixClient, room: Room, user: RoomMember): IRoomPermissions { const [roomPermissions, setRoomPermissions] = useState({ // modifyLevelMax is the max PL we can set this user to, typically min(their PL, our PL) && canSetPL modifyLevelMax: -1, canEdit: false, canInvite: false, }); const updateRoomPermissions = useCallback(() => { const powerLevels = room?.currentState.getStateEvents(EventType.RoomPowerLevels, "")?.getContent(); if (!powerLevels) return; const me = room.getMember(cli.getUserId() || ""); if (!me) return; const them = user; const isMe = me.userId === them.userId; const canAffectUser = them.powerLevel < me.powerLevel || isMe; let modifyLevelMax = -1; if (canAffectUser) { const editPowerLevel = powerLevels.events?.[EventType.RoomPowerLevels] ?? powerLevels.state_default ?? 50; if (me.powerLevel >= editPowerLevel) { modifyLevelMax = me.powerLevel; } } setRoomPermissions({ canInvite: me.powerLevel >= (powerLevels.invite ?? 0), canEdit: modifyLevelMax >= 0, modifyLevelMax, }); }, [cli, user, room]); useTypedEventEmitter(cli, RoomStateEvent.Update, updateRoomPermissions); useEffect(() => { updateRoomPermissions(); return () => { setRoomPermissions({ modifyLevelMax: -1, canEdit: false, canInvite: false, }); }; }, [updateRoomPermissions]); return roomPermissions; } const PowerLevelSection: React.FC<{ user: RoomMember; room: Room; roomPermissions: IRoomPermissions; powerLevels: IPowerLevelsContent; }> = ({ user, room, roomPermissions, powerLevels }) => { if (roomPermissions.canEdit) { return ; } else { const powerLevelUsersDefault = powerLevels.users_default || 0; const powerLevel = user.powerLevel; const role = textualPowerLevel(powerLevel, powerLevelUsersDefault); return (
{role}
); } }; export const PowerLevelEditor: React.FC<{ user: RoomMember; room: Room; roomPermissions: IRoomPermissions; }> = ({ user, room, roomPermissions }) => { const cli = useContext(MatrixClientContext); const [selectedPowerLevel, setSelectedPowerLevel] = useState(user.powerLevel); useEffect(() => { setSelectedPowerLevel(user.powerLevel); }, [user]); const onPowerChange = useCallback( async (powerLevel: number) => { setSelectedPowerLevel(powerLevel); const applyPowerChange = (roomId: string, target: string, powerLevel: number): Promise => { return cli.setPowerLevel(roomId, target, powerLevel).then( function () { // NO-OP; rely on the m.room.member event coming down else we could // get out of sync if we force setState here! logger.log("Power change success"); }, function (err) { logger.error("Failed to change power level " + err); Modal.createDialog(ErrorDialog, { title: _t("common|error"), description: _t("error|update_power_level"), }); }, ); }; const roomId = user.roomId; const target = user.userId; const powerLevelEvent = room.currentState.getStateEvents("m.room.power_levels", ""); if (!powerLevelEvent) return; const myUserId = cli.getUserId(); const myPower = powerLevelEvent.getContent().users[myUserId || ""]; if (myPower && parseInt(myPower) <= powerLevel && myUserId !== target) { const { finished } = Modal.createDialog(QuestionDialog, { title: _t("common|warning"), description: (
{_t("user_info|promote_warning")}
{_t("common|are_you_sure")}
), button: _t("action|continue"), }); const [confirmed] = await finished; if (!confirmed) return; } else if (myUserId === target && myPower && parseInt(myPower) > powerLevel) { // If we are changing our own PL it can only ever be decreasing, which we cannot reverse. try { if (!(await warnSelfDemote(room?.isSpaceRoom()))) return; } catch (e) { logger.error("Failed to warn about self demotion: ", e); } } await applyPowerChange(roomId, target, powerLevel); }, [user.roomId, user.userId, cli, room], ); const powerLevelEvent = room.currentState.getStateEvents("m.room.power_levels", ""); const powerLevelUsersDefault = powerLevelEvent ? powerLevelEvent.getContent().users_default : 0; return (
); }; async function getUserDeviceInfo( userId: string, cli: MatrixClient, downloadUncached = false, ): Promise { const userDeviceMap = await cli.getCrypto()?.getUserDeviceInfo([userId], downloadUncached); const devicesMap = userDeviceMap?.get(userId); if (!devicesMap) return; return Array.from(devicesMap.values()); } export const useDevices = (userId: string): IDevice[] | undefined | null => { const cli = useContext(MatrixClientContext); // undefined means yet to be loaded, null means failed to load, otherwise list of devices const [devices, setDevices] = useState(undefined); // Download device lists useEffect(() => { setDevices(undefined); let cancelled = false; async function downloadDeviceList(): Promise { try { const devices = await getUserDeviceInfo(userId, cli, true); if (cancelled || !devices) { // we got cancelled - presumably a different user now return; } disambiguateDevices(devices); setDevices(devices); } catch (err) { setDevices(null); } } downloadDeviceList(); // Handle being unmounted return () => { cancelled = true; }; }, [cli, userId]); // Listen to changes useEffect(() => { let cancel = false; const updateDevices = async (): Promise => { const newDevices = await getUserDeviceInfo(userId, cli); if (cancel || !newDevices) return; setDevices(newDevices); }; const onDevicesUpdated = (users: string[]): void => { if (!users.includes(userId)) return; updateDevices(); }; const onDeviceVerificationChanged = (_userId: string, deviceId: string): void => { if (_userId !== userId) return; updateDevices(); }; const onUserTrustStatusChanged = (_userId: string, trustLevel: UserVerificationStatus): void => { if (_userId !== userId) return; updateDevices(); }; cli.on(CryptoEvent.DevicesUpdated, onDevicesUpdated); cli.on(CryptoEvent.DeviceVerificationChanged, onDeviceVerificationChanged); cli.on(CryptoEvent.UserTrustStatusChanged, onUserTrustStatusChanged); // Handle being unmounted return () => { cancel = true; cli.removeListener(CryptoEvent.DevicesUpdated, onDevicesUpdated); cli.removeListener(CryptoEvent.DeviceVerificationChanged, onDeviceVerificationChanged); cli.removeListener(CryptoEvent.UserTrustStatusChanged, onUserTrustStatusChanged); }; }, [cli, userId]); return devices; }; const BasicUserInfo: React.FC<{ room: Room; member: User | RoomMember; devices: IDevice[]; isRoomEncrypted: boolean; }> = ({ room, member, devices, isRoomEncrypted }) => { const cli = useContext(MatrixClientContext); const powerLevels = useRoomPowerLevels(cli, room); // Load whether or not we are a Synapse Admin const isSynapseAdmin = useIsSynapseAdmin(cli); // Count of how many operations are currently in progress, if > 0 then show a Spinner const [pendingUpdateCount, setPendingUpdateCount] = useState(0); const startUpdating = useCallback(() => { setPendingUpdateCount(pendingUpdateCount + 1); }, [pendingUpdateCount]); const stopUpdating = useCallback(() => { setPendingUpdateCount(pendingUpdateCount - 1); }, [pendingUpdateCount]); const roomPermissions = useRoomPermissions(cli, room, member as RoomMember); const onSynapseDeactivate = useCallback(async () => { const { finished } = Modal.createDialog(QuestionDialog, { title: _t("user_info|deactivate_confirm_title"), description:
{_t("user_info|deactivate_confirm_description")}
, button: _t("user_info|deactivate_confirm_action"), danger: true, }); const [accepted] = await finished; if (!accepted) return; try { await cli.deactivateSynapseUser(member.userId); } catch (err) { logger.error("Failed to deactivate user"); logger.error(err); const description = err instanceof Error ? err.message : _t("invite|failed_generic"); Modal.createDialog(ErrorDialog, { title: _t("user_info|error_deactivate"), description, }); } }, [cli, member.userId]); let synapseDeactivateButton; let spinner; // We don't need a perfect check here, just something to pass as "probably not our homeserver". If // someone does figure out how to bypass this check the worst that happens is an error. if (isSynapseAdmin && member.userId.endsWith(`:${cli.getDomain()}`)) { synapseDeactivateButton = ( { ev.preventDefault(); onSynapseDeactivate(); }} label={_t("user_info|deactivate_confirm_action")} kind="critical" Icon={DeleteIcon} /> ); } let memberDetails; let adminToolsContainer; if (room && (member as RoomMember).roomId) { // hide the Roles section for DMs as it doesn't make sense there if (!DMRoomMap.shared().getUserIdForRoomId((member as RoomMember).roomId)) { memberDetails = ( ); } adminToolsContainer = ( 0} startUpdating={startUpdating} stopUpdating={stopUpdating} > {synapseDeactivateButton} ); } else if (synapseDeactivateButton) { adminToolsContainer = {synapseDeactivateButton}; } if (pendingUpdateCount > 0) { spinner = ; } // only display the devices list if our client supports E2E const cryptoEnabled = Boolean(cli.getCrypto()); let text; if (!isRoomEncrypted) { if (!cryptoEnabled) { text = _t("encryption|unsupported"); } else if (room && !room.isSpaceRoom()) { text = _t("user_info|room_unencrypted"); } } else if (!room.isSpaceRoom()) { text = _t("user_info|room_encrypted"); } let verifyButton; const homeserverSupportsCrossSigning = useHomeserverSupportsCrossSigning(cli); const userTrust = useAsyncMemo( async () => cli.getCrypto()?.getUserVerificationStatus(member.userId), [member.userId], // the user verification status is not initialized undefined, ); const hasUserVerificationStatus = Boolean(userTrust); const isUserVerified = Boolean(userTrust?.isVerified()); const isMe = member.userId === cli.getUserId(); const canVerify = hasUserVerificationStatus && homeserverSupportsCrossSigning && !isUserVerified && !isMe && devices && devices.length > 0; const setUpdating: SetUpdating = (updating) => { setPendingUpdateCount((count) => count + (updating ? 1 : -1)); }; const hasCrossSigningKeys = useHasCrossSigningKeys(cli, member as User, canVerify, setUpdating); // Display the spinner only when // - the devices are not populated yet, or // - the crypto is available and we don't have the user verification status yet const showDeviceListSpinner = (cryptoEnabled && !hasUserVerificationStatus) || devices === undefined; if (canVerify) { if (hasCrossSigningKeys !== undefined) { // Note: mx_UserInfo_verifyButton is for the end-to-end tests verifyButton = (
{ if (hasCrossSigningKeys) { verifyUser(cli, member as User); } else { legacyVerifyUser(cli, member as User); } }} > {_t("action|verify")}
); } else if (!showDeviceListSpinner) { // HACK: only show a spinner if the device section spinner is not shown, // to avoid showing a double spinner // We should ask for a design that includes all the different loading states here verifyButton = ; } } let editDevices; if (member.userId == cli.getUserId()) { editDevices = (
{ dis.dispatch({ action: Action.ViewUserDeviceSettings, }); }} > {_t("user_info|edit_own_devices")}
); } const securitySection = (

{_t("common|security")}

{text}

{verifyButton} {cryptoEnabled && ( )} {editDevices}
); return ( {securitySection} {memberDetails} {adminToolsContainer} {!isMe && ( )} {spinner} ); }; export type Member = User | RoomMember; export const UserInfoHeader: React.FC<{ member: Member; e2eStatus?: E2EStatus; roomId?: string; }> = ({ member, e2eStatus, roomId }) => { const cli = useContext(MatrixClientContext); const onMemberAvatarClick = useCallback(() => { const avatarUrl = (member as RoomMember).getMxcAvatarUrl ? (member as RoomMember).getMxcAvatarUrl() : (member as User).avatarUrl; const httpUrl = mediaFromMxc(avatarUrl).srcHttp; if (!httpUrl) return; const params = { src: httpUrl, name: (member as RoomMember).name || (member as User).displayName, }; Modal.createDialog(ImageView, params, "mx_Dialog_lightbox", undefined, true); }, [member]); const avatarUrl = (member as User).avatarUrl; let presenceState: string | undefined; let presenceLastActiveAgo: number | undefined; let presenceCurrentlyActive: boolean | undefined; if (member instanceof RoomMember && member.user) { presenceState = member.user.presence; presenceLastActiveAgo = member.user.lastActiveAgo; presenceCurrentlyActive = member.user.currentlyActive; } const enablePresenceByHsUrl = SdkConfig.get("enable_presence_by_hs_url"); let showPresence = true; if (enablePresenceByHsUrl && enablePresenceByHsUrl[cli.baseUrl] !== undefined) { showPresence = enablePresenceByHsUrl[cli.baseUrl]; } let presenceLabel: JSX.Element | undefined; if (showPresence) { presenceLabel = ( ); } const e2eIcon = e2eStatus ? : null; const userIdentifier = UserIdentifierCustomisations.getDisplayUserIdentifier?.(member.userId, { roomId, withDisplayName: true, }); const displayName = (member as RoomMember).rawDisplayName; return (
{displayName} {e2eIcon} {presenceLabel} userIdentifier} border={false}> {userIdentifier}
); }; interface IProps { user: Member; room?: Room; phase: RightPanelPhases.RoomMemberInfo | RightPanelPhases.SpaceMemberInfo | RightPanelPhases.EncryptionPanel; onClose(): void; verificationRequest?: VerificationRequest; verificationRequestPromise?: Promise; } const UserInfo: React.FC = ({ user, room, onClose, phase = RightPanelPhases.RoomMemberInfo, ...props }) => { const cli = useContext(MatrixClientContext); // fetch latest room member if we have a room, so we don't show historical information, falling back to user const member = useMemo(() => (room ? room.getMember(user.userId) || user : user), [room, user]); const isRoomEncrypted = useIsEncrypted(cli, room); const devices = useDevices(user.userId) ?? []; const e2eStatus = useAsyncMemo(async () => { if (!isRoomEncrypted || !devices) { return undefined; } return await getE2EStatus(cli, user.userId, devices); }, [cli, isRoomEncrypted, user.userId, devices]); const classes = ["mx_UserInfo"]; let cardState: IRightPanelCardState = {}; // We have no previousPhase for when viewing a UserInfo without a Room at this time if (room && phase === RightPanelPhases.EncryptionPanel) { cardState = { member }; } else if (room?.isSpaceRoom()) { cardState = { spaceId: room.roomId }; } const onEncryptionPanelClose = (): void => { RightPanelStore.instance.popCard(); }; let content: JSX.Element | undefined; switch (phase) { case RightPanelPhases.RoomMemberInfo: case RightPanelPhases.SpaceMemberInfo: content = ( ); break; case RightPanelPhases.EncryptionPanel: classes.push("mx_UserInfo_smallAvatar"); content = ( )} member={member as User | RoomMember} onClose={onEncryptionPanelClose} isRoomEncrypted={Boolean(isRoomEncrypted)} /> ); break; } let closeLabel: string | undefined; if (phase === RightPanelPhases.EncryptionPanel) { const verificationRequest = (props as React.ComponentProps).verificationRequest; if (verificationRequest && verificationRequest.pending) { closeLabel = _t("action|cancel"); } } const header = ( <> ); return ( { if (RightPanelStore.instance.previousCard.phase === RightPanelPhases.RoomMemberList) { PosthogTrackers.trackInteraction("WebRightPanelRoomUserInfoBackButton", ev); } }} > {header} {content} ); }; export default UserInfo;