/* 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, { useCallback, useContext, useEffect, useMemo, useState } from 'react'; import classNames from 'classnames'; import { ClientEvent, MatrixClient } from 'matrix-js-sdk/src/client'; import { RoomMember } from 'matrix-js-sdk/src/models/room-member'; import { User } from 'matrix-js-sdk/src/models/user'; import { Room } from 'matrix-js-sdk/src/models/room'; import { MatrixEvent } from 'matrix-js-sdk/src/models/event'; import { VerificationRequest } from "matrix-js-sdk/src/crypto/verification/request/VerificationRequest"; import { EventType } from "matrix-js-sdk/src/@types/event"; import { logger } from "matrix-js-sdk/src/logger"; import { CryptoEvent } from "matrix-js-sdk/src/crypto"; import { RoomStateEvent } from "matrix-js-sdk/src/models/room-state"; import dis from '../../../dispatcher/dispatcher'; import Modal from '../../../Modal'; import { _t } from '../../../languageHandler'; import createRoom from '../../../createRoom'; import DMRoomMap from '../../../utils/DMRoomMap'; import AccessibleButton, { ButtonEvent } from '../elements/AccessibleButton'; import SdkConfig from '../../../SdkConfig'; import { RoomViewStore } from "../../../stores/RoomViewStore"; import MultiInviter from "../../../utils/MultiInviter"; import { MatrixClientPeg } from "../../../MatrixClientPeg"; 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 { UserTab } from "../dialogs/UserTab"; 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 RoomAvatar from "../avatars/RoomAvatar"; import RoomName from "../elements/RoomName"; import { mediaFromMxc } from "../../../customisations/Media"; import UIStore from "../../../stores/UIStore"; 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 { useUserStatusMessage } from "../../../hooks/useUserStatusMessage"; import UserIdentifierCustomisations from '../../../customisations/UserIdentifier'; import PosthogTrackers from "../../../PosthogTrackers"; import { ViewRoomPayload } from "../../../dispatcher/payloads/ViewRoomPayload"; import { findDMForUser } from "../../../utils/direct-messages"; import { privateShouldBeEncrypted } from "../../../utils/rooms"; export interface IDevice { deviceId: string; ambiguous?: boolean; getDisplayName(): string; } const disambiguateDevices = (devices: IDevice[]) => { const names = Object.create(null); for (let i = 0; i < devices.length; i++) { const name = devices[i].getDisplayName(); const indexList = names[name] || []; indexList.push(i); names[name] = indexList; } for (const name in names) { if (names[name].length > 1) { names[name].forEach((j) => { devices[j].ambiguous = true; }); } } }; export const getE2EStatus = (cli: MatrixClient, userId: string, devices: IDevice[]): E2EStatus => { const isMe = userId === cli.getUserId(); const userTrust = cli.checkUserTrust(userId); if (!userTrust.isCrossSigningVerified()) { return userTrust.wasCrossSigningVerified() ? E2EStatus.Warning : E2EStatus.Normal; } const anyDeviceUnverified = devices.some(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 = cli.checkDeviceTrust(userId, deviceId); return isMe ? !deviceTrust.isCrossSigningVerified() : !deviceTrust.isVerified(); }); return anyDeviceUnverified ? E2EStatus.Warning : E2EStatus.Verified; }; async function openDMForUser(matrixClient: MatrixClient, userId: string, viaKeyboard = false): Promise { const lastActiveRoom = findDMForUser(matrixClient, userId); if (lastActiveRoom) { dis.dispatch({ action: Action.ViewRoom, room_id: lastActiveRoom.roomId, metricsTrigger: "MessageUser", metricsViaKeyboard: viaKeyboard, }); return; } const createRoomOptions = { dmUserId: userId, encryption: undefined, }; if (privateShouldBeEncrypted()) { // Check whether all users have uploaded device keys before. // If so, enable encryption in the new room. const usersToDevicesMap = await matrixClient.downloadKeys([userId]); const allHaveDeviceKeys = Object.values(usersToDevicesMap).every(devices => { // `devices` is an object of the form { deviceId: deviceInfo, ... }. return Object.keys(devices).length > 0; }); if (allHaveDeviceKeys) { createRoomOptions.encryption = true; } } await createRoom(createRoomOptions); } type SetUpdating = (updating: boolean) => void; function useHasCrossSigningKeys(cli: MatrixClient, member: User, canVerify: boolean, setUpdating: SetUpdating) { return useAsyncMemo(async () => { if (!canVerify) { return undefined; } setUpdating(true); try { await cli.downloadKeys([member.userId]); const xsi = cli.getStoredCrossSigningForUser(member.userId); const key = xsi && xsi.getId(); return !!key; } finally { setUpdating(false); } }, [cli, member, canVerify], undefined); } function DeviceItem({ userId, device }: { userId: string, device: IDevice }) { const cli = useContext(MatrixClientContext); const isMe = userId === cli.getUserId(); const deviceTrust = cli.checkDeviceTrust(userId, device.deviceId); const userTrust = cli.checkUserTrust(userId); // 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 = isMe ? deviceTrust.isCrossSigningVerified() : deviceTrust.isVerified(); const classes = classNames("mx_UserInfo_device", { mx_UserInfo_device_verified: isVerified, mx_UserInfo_device_unverified: !isVerified, }); const iconClasses = classNames("mx_E2EIcon", { mx_E2EIcon_normal: !userTrust.isVerified(), mx_E2EIcon_verified: isVerified, mx_E2EIcon_warning: userTrust.isVerified() && !isVerified, }); const onDeviceClick = () => { verifyDevice(cli.getUser(userId), device); }; let deviceName; if (!device.getDisplayName()?.trim()) { deviceName = device.deviceId; } else { deviceName = device.ambiguous ? device.getDisplayName() + " (" + device.deviceId + ")" : device.getDisplayName(); } let trustedLabel = null; if (userTrust.isVerified()) trustedLabel = isVerified ? _t("Trusted") : _t("Not trusted"); if (isVerified) { return (
{ deviceName }
{ trustedLabel }
); } else { return (
{ deviceName }
{ trustedLabel }
); } } function DevicesSection({ devices, userId, loading }: { devices: IDevice[], userId: string, loading: boolean }) { const cli = useContext(MatrixClientContext); const userTrust = cli.checkUserTrust(userId); const [isExpanded, setExpanded] = useState(false); if (loading) { // still loading return ; } if (devices === null) { return <>{ _t("Unable to load session list") }; } const isMe = userId === cli.getUserId(); const deviceTrusts = devices.map(d => cli.checkDeviceTrust(userId, d.deviceId)); let expandSectionDevices = []; const unverifiedDevices = []; let expandCountCaption; let expandHideCaption; let expandIconClasses = "mx_E2EIcon"; if (userTrust.isVerified()) { 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 = isMe ? deviceTrust.isCrossSigningVerified() : deviceTrust.isVerified(); if (isVerified) { expandSectionDevices.push(device); } else { unverifiedDevices.push(device); } } expandCountCaption = _t("%(count)s verified sessions", { count: expandSectionDevices.length }); expandHideCaption = _t("Hide verified sessions"); expandIconClasses += " mx_E2EIcon_verified"; } else { expandSectionDevices = devices; expandCountCaption = _t("%(count)s sessions", { count: devices.length }); expandHideCaption = _t("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 (); })); } return (
{ deviceList }
{ expandButton }
); } const MessageButton = ({ userId }: { userId: string }) => { const cli = useContext(MatrixClientContext); const [busy, setBusy] = useState(false); return ( { if (busy) return; setBusy(true); await openDMForUser(cli, userId, ev.type !== "click"); setBusy(false); }} className="mx_UserInfo_field" disabled={busy} > { _t("Message") } ); }; const UserOptionsSection: React.FC<{ member: RoomMember; isIgnored: boolean; canInvite: boolean; isSpace?: boolean; }> = ({ member, isIgnored, canInvite, isSpace }) => { const cli = useContext(MatrixClientContext); let ignoreButton = null; let insertPillButton = null; let inviteUserButton = null; let readReceiptButton = null; const isMe = member.userId === cli.getUserId(); const onShareUserClick = () => { Modal.createTrackedDialog('share room member dialog', '', ShareDialog, { target: member, }); }; // Only allow the user to ignore the user if its not ourselves // same goes for jumping to read receipt if (!isMe) { const onIgnoreToggle = () => { const ignoredUsers = cli.getIgnoredUsers(); if (isIgnored) { const index = ignoredUsers.indexOf(member.userId); if (index !== -1) ignoredUsers.splice(index, 1); } else { ignoredUsers.push(member.userId); } cli.setIgnoredUsers(ignoredUsers); }; ignoreButton = ( { isIgnored ? _t("Unignore") : _t("Ignore") } ); if (member.roomId && !isSpace) { const onReadReceiptButton = function() { const room = cli.getRoom(member.roomId); dis.dispatch({ action: Action.ViewRoom, highlighted: true, event_id: room.getEventReadUpTo(member.userId), room_id: member.roomId, metricsTrigger: undefined, // room doesn't change }); }; const onInsertPillButton = function() { dis.dispatch({ action: Action.ComposerInsert, userId: member.userId, timelineRenderingType: TimelineRenderingType.Room, }); }; const room = cli.getRoom(member.roomId); if (room?.getEventReadUpTo(member.userId)) { readReceiptButton = ( { _t('Jump to read receipt') } ); } insertPillButton = ( { _t('Mention') } ); } if (canInvite && (member?.membership ?? 'leave') === 'leave' && shouldShowComponent(UIComponent.InviteUsers)) { const roomId = member && member.roomId ? member.roomId : RoomViewStore.instance.getRoomId(); const onInviteUserButton = async (ev: ButtonEvent) => { try { // We use a MultiInviter to re-use the invite logic, even though we're only inviting one user. const inviter = new MultiInviter(roomId); await inviter.invite([member.userId]).then(() => { if (inviter.getCompletionState(member.userId) !== "invited") { throw new Error(inviter.getErrorText(member.userId)); } }); } catch (err) { Modal.createTrackedDialog('Failed to invite', '', ErrorDialog, { title: _t('Failed to invite'), description: ((err && err.message) ? err.message : _t("Operation failed")), }); } PosthogTrackers.trackInteraction("WebRightPanelRoomUserInfoInviteButton", ev); }; inviteUserButton = ( { _t('Invite') } ); } } const shareUserButton = ( { _t('Share Link to User') } ); let directMessageButton: JSX.Element; if (!isMe) { directMessageButton = ; } return (

{ _t("Options") }

{ directMessageButton } { readReceiptButton } { shareUserButton } { insertPillButton } { inviteUserButton } { ignoreButton }
); }; const warnSelfDemote = async (isSpace: boolean) => { const { finished } = Modal.createTrackedDialog('Demoting Self', '', QuestionDialog, { title: _t("Demote yourself?"), description:
{ isSpace ? _t("You will not be able to undo this change as you are demoting yourself, " + "if you are the last privileged user in the space it will be impossible " + "to regain privileges.") : _t("You will not be able to undo this change as you are demoting yourself, " + "if you are the last privileged user in the room it will be impossible " + "to regain privileges.") }
, button: _t("Demote"), }); const [confirmed] = await finished; return confirmed; }; const GenericAdminToolsContainer: React.FC<{}> = ({ children }) => { return (

{ _t("Admin Tools") }

{ 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; } const isMuted = (member: RoomMember, powerLevelContent: IPowerLevelsContent) => { if (!powerLevelContent || !member) return false; const levelToSend = ( (powerLevelContent.events ? powerLevelContent.events["m.room.message"] : null) || powerLevelContent.events_default ); return member.powerLevel < levelToSend; }; const getPowerLevels = room => room?.currentState?.getStateEvents(EventType.RoomPowerLevels, "")?.getContent() || {}; export const useRoomPowerLevels = (cli: MatrixClient, room: Room) => { 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; startUpdating(): void; stopUpdating(): void; } const RoomKickButton = ({ room, member, startUpdating, stopUpdating }: Omit) => { const cli = useContext(MatrixClientContext); // check if user can be kicked/disinvited if (member.membership !== "invite" && member.membership !== "join") return null; const onKick = async () => { const { finished } = Modal.createTrackedDialog( 'Confirm User Action Dialog', 'onKick', room.isSpaceRoom() ? ConfirmSpaceUserActionDialog : ConfirmUserActionDialog, { member, action: member.membership === "invite" ? _t("Disinvite") : _t("Remove from chat"), title: member.membership === "invite" ? _t("Disinvite from %(roomName)s", { roomName: room.name }) : _t("Remove from %(roomName)s", { roomName: room.name }), askReason: member.membership === "join", danger: true, // space-specific props 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("Remove them from everything I'm able to"), specificLabel: _t("Remove them from specific things I'm able to"), warningMessage: _t("They'll still be able to access whatever you're not an admin of."), }, room.isSpaceRoom() ? "mx_ConfirmSpaceUserActionDialog_wrapper" : undefined, ); const [proceed, reason, rooms = []] = await finished; if (!proceed) return; startUpdating(); 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.createTrackedDialog('Failed to kick', '', ErrorDialog, { title: _t("Failed to remove user"), description: ((err && err.message) ? err.message : "Operation failed"), }); }).finally(() => { stopUpdating(); }); }; const kickLabel = member.membership === "invite" ? _t("Disinvite") : _t("Remove from room"); return { kickLabel } ; }; const RedactMessagesButton: React.FC = ({ member }) => { const cli = useContext(MatrixClientContext); const onRedactAllMessages = () => { const room = cli.getRoom(member.roomId); if (!room) return; Modal.createTrackedDialog("Bulk Redact Dialog", "", BulkRedactDialog, { matrixClient: cli, room, member, }); }; return { _t("Remove recent messages") } ; }; const BanToggleButton = ({ room, member, startUpdating, stopUpdating }: Omit) => { const cli = useContext(MatrixClientContext); const isBanned = member.membership === "ban"; const onBanOrUnban = async () => { const { finished } = Modal.createTrackedDialog( 'Confirm User Action Dialog', 'onBanOrUnban', room.isSpaceRoom() ? ConfirmSpaceUserActionDialog : ConfirmUserActionDialog, { member, action: room.isSpaceRoom() ? (isBanned ? _t("Unban from space") : _t("Ban from space")) : (isBanned ? _t("Unban from room") : _t("Ban from room")), title: isBanned ? _t("Unban from %(roomName)s", { roomName: room.name }) : _t("Ban from %(roomName)s", { roomName: room.name }), askReason: !isBanned, danger: !isBanned, // space-specific props 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 === "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 !== "ban" && myMember.powerLevel > theirMember.powerLevel && child.currentState.hasSufficientPowerLevelFor("ban", myMember.powerLevel); }, allLabel: isBanned ? _t("Unban them from everything I'm able to") : _t("Ban them from everything I'm able to"), specificLabel: isBanned ? _t("Unban them from specific things I'm able to") : _t("Ban them from specific things I'm able to"), warningMessage: isBanned ? _t("They won't be able to access whatever you're not an admin of.") : _t("They'll still be able to access whatever you're not an admin of."), }, room.isSpaceRoom() ? "mx_ConfirmSpaceUserActionDialog_wrapper" : undefined, ); const [proceed, reason, rooms = []] = await finished; if (!proceed) return; startUpdating(); const fn = (roomId: string) => { 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.createTrackedDialog('Failed to ban user', '', ErrorDialog, { title: _t("Error"), description: _t("Failed to ban user"), }); }).finally(() => { stopUpdating(); }); }; let label = room.isSpaceRoom() ? _t("Ban from space") : _t("Ban from room"); if (isBanned) { label = room.isSpaceRoom() ? _t("Unban from space") : _t("Unban from room"); } const classes = classNames("mx_UserInfo_field", { mx_UserInfo_destructive: !isBanned, }); return { label } ; }; interface IBaseRoomProps extends IBaseProps { room: Room; powerLevels: IPowerLevelsContent; } const MuteToggleButton: React.FC = ({ member, room, powerLevels, startUpdating, stopUpdating }) => { const cli = useContext(MatrixClientContext); // Don't show the mute/unmute option if the user is not in the room if (member.membership !== "join") return null; const muted = isMuted(member, powerLevels); const onMuteToggle = async () => { const roomId = member.roomId; const target = member.userId; // if muting self, warn as it may be irreversible if (target === cli.getUserId()) { try { if (!(await warnSelfDemote(room?.isSpaceRoom()))) return; } catch (e) { logger.error("Failed to warn about self demotion: ", e); return; } } const powerLevelEvent = room.currentState.getStateEvents("m.room.power_levels", ""); if (!powerLevelEvent) return; const powerLevels = powerLevelEvent.getContent(); const levelToSend = ( (powerLevels.events ? powerLevels.events["m.room.message"] : null) || powerLevels.events_default ); let level; if (muted) { // unmute level = levelToSend; } else { // mute level = levelToSend - 1; } level = parseInt(level); if (!isNaN(level)) { startUpdating(); cli.setPowerLevel(roomId, target, level, powerLevelEvent).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.createTrackedDialog('Failed to mute user', '', ErrorDialog, { title: _t("Error"), description: _t("Failed to mute user"), }); }).finally(() => { stopUpdating(); }); } }; const classes = classNames("mx_UserInfo_field", { mx_UserInfo_destructive: !muted, }); const muteLabel = muted ? _t("Unmute") : _t("Mute"); return { muteLabel } ; }; const RoomAdminToolsContainer: React.FC = ({ room, children, member, 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 >= editPowerLevel && !room.isSpaceRoom()) { muteButton = ( ); } if (kickButton || banButton || muteButton || redactButton || children) { return { muteButton } { kickButton } { banButton } { redactButton } { children } ; } return
; }; const useIsSynapseAdmin = (cli: MatrixClient) => { const [isAdmin, setIsAdmin] = useState(false); useEffect(() => { cli.isSynapseAdministrator().then((isAdmin) => { setIsAdmin(isAdmin); }, () => { setIsAdmin(false); }); }, [cli]); return isAdmin; }; const useHomeserverSupportsCrossSigning = (cli: MatrixClient) => { 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 ?? 50), 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 }
); } }; 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, target, powerLevel, powerLevelEvent) => { return cli.setPowerLevel(roomId, target, parseInt(powerLevel), powerLevelEvent).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.createTrackedDialog('Failed to change power level', '', ErrorDialog, { title: _t("Error"), description: _t("Failed to change 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.createTrackedDialog('Promote to PL100 Warning', '', QuestionDialog, { title: _t("Warning!"), description:
{ _t("You will not be able to undo this change as you are promoting the user " + "to have the same power level as yourself.") }
{ _t("Are you sure?") }
, button: _t("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, powerLevelEvent); }, [user.roomId, user.userId, cli, room]); const powerLevelEvent = room.currentState.getStateEvents("m.room.power_levels", ""); const powerLevelUsersDefault = powerLevelEvent ? powerLevelEvent.getContent().users_default : 0; return (
); }; export const useDevices = (userId: string) => { 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() { try { await cli.downloadKeys([userId], true); const devices = cli.getStoredDevicesForUser(userId); if (cancelled) { // 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 () => { const newDevices = cli.getStoredDevicesForUser(userId); if (cancel) return; setDevices(newDevices); }; const onDevicesUpdated = (users) => { if (!users.includes(userId)) return; updateDevices(); }; const onDeviceVerificationChanged = (_userId, device) => { if (_userId !== userId) return; updateDevices(); }; const onUserTrustStatusChanged = (_userId, trustStatus) => { 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); // 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); // 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.createTrackedDialog('Synapse User Deactivation', '', QuestionDialog, { title: _t("Deactivate user?"), description:
{ _t( "Deactivating this user will log them out and prevent them from logging back in. Additionally, " + "they will leave all the rooms they are in. This action cannot be reversed. Are you sure you " + "want to deactivate this user?", ) }
, button: _t("Deactivate user"), 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); Modal.createTrackedDialog('Failed to deactivate Synapse user', '', ErrorDialog, { title: _t('Failed to deactivate user'), description: ((err && err.message) ? err.message : _t("Operation failed")), }); } }, [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. // FIXME this should be using cli instead of MatrixClientPeg.matrixClient if (isSynapseAdmin && member.userId.endsWith(`:${MatrixClientPeg.getHomeserverName()}`)) { synapseDeactivateButton = ( { _t("Deactivate user") } ); } 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 =

{ _t("Role in ", {}, { RoomName: () => { room.name }, }) }

; } adminToolsContainer = ( { synapseDeactivateButton } ); } else if (synapseDeactivateButton) { adminToolsContainer = ( { synapseDeactivateButton } ); } if (pendingUpdateCount > 0) { spinner = ; } // only display the devices list if our client supports E2E const cryptoEnabled = cli.isCryptoEnabled(); let text; if (!isRoomEncrypted) { if (!cryptoEnabled) { text = _t("This client does not support end-to-end encryption."); } else if (room && !room.isSpaceRoom()) { text = _t("Messages in this room are not end-to-end encrypted."); } } else if (!room.isSpaceRoom()) { text = _t("Messages in this room are end-to-end encrypted."); } let verifyButton; const homeserverSupportsCrossSigning = useHomeserverSupportsCrossSigning(cli); const userTrust = cryptoEnabled && cli.checkUserTrust(member.userId); const userVerified = cryptoEnabled && userTrust.isCrossSigningVerified(); const isMe = member.userId === cli.getUserId(); const canVerify = cryptoEnabled && homeserverSupportsCrossSigning && !userVerified && !isMe && devices && devices.length > 0; const setUpdating = (updating) => { setPendingUpdateCount(count => count + (updating ? 1 : -1)); }; const hasCrossSigningKeys = useHasCrossSigningKeys(cli, member as User, canVerify, setUpdating); const showDeviceListSpinner = devices === undefined; if (canVerify) { if (hasCrossSigningKeys !== undefined) { // Note: mx_UserInfo_verifyButton is for the end-to-end tests verifyButton = ( { if (hasCrossSigningKeys) { verifyUser(member as User); } else { legacyVerifyUser(member as User); } }} > { _t("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.ViewUserSettings, initialTabId: UserTab.Security, }); }} > { _t("Edit devices") }
); } const securitySection = (

{ _t("Security") }

{ text }

{ verifyButton } { cryptoEnabled && } { editDevices }
); return { memberDetails } { securitySection } { adminToolsContainer } { spinner } ; }; export type Member = User | RoomMember; const UserInfoHeader: React.FC<{ member: Member; e2eStatus: E2EStatus; roomId?: string; }> = ({ member, e2eStatus, roomId }) => { const cli = useContext(MatrixClientContext); const statusMessage = useUserStatusMessage(member); const onMemberAvatarClick = useCallback(() => { const avatarUrl = (member as RoomMember).getMxcAvatarUrl ? (member as RoomMember).getMxcAvatarUrl() : (member as User).avatarUrl; if (!avatarUrl) return; const httpUrl = mediaFromMxc(avatarUrl).srcHttp; const params = { src: httpUrl, name: (member as RoomMember).name || (member as User).displayName, }; Modal.createDialog(ImageView, params, "mx_Dialog_lightbox", null, true); }, [member]); const avatarElement = (
); let presenceState; let presenceLastActiveAgo; let presenceCurrentlyActive; 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 = null; if (showPresence) { presenceLabel = ( ); } let statusLabel = null; if (statusMessage) { statusLabel = { statusMessage }; } let e2eIcon; if (e2eStatus) { e2eIcon = ; } const displayName = (member as RoomMember).rawDisplayName; return { avatarElement }

{ e2eIcon } { displayName }

{ UserIdentifierCustomisations.getDisplayUserIdentifier(member.userId, { roomId, withDisplayName: true }) }
{ presenceLabel } { statusLabel }
; }; 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); let e2eStatus; if (isRoomEncrypted && devices) { e2eStatus = getE2EStatus(cli, 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 = () => { RightPanelStore.instance.popCard(); }; let content; 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={isRoomEncrypted} /> ); break; } let closeLabel = undefined; if (phase === RightPanelPhases.EncryptionPanel) { const verificationRequest = (props as React.ComponentProps).verificationRequest; if (verificationRequest && verificationRequest.pending) { closeLabel = _t("Cancel"); } } let scopeHeader; if (room?.isSpaceRoom()) { scopeHeader =
; } const header = <> { scopeHeader } ; return { if (RightPanelStore.instance.previousCard.phase === RightPanelPhases.RoomMemberList) { PosthogTrackers.trackInteraction("WebRightPanelRoomUserInfoBackButton", ev); } }} > { content } ; }; export default UserInfo;