feat: show presence status badges in member list panels

Add online/offline/idle presence dots next to verification shields in
both the room members drawer and the common-settings members list.

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
2026-05-23 20:13:08 -04:00
parent a3b3ca90c9
commit 2e1e61c963
2 changed files with 15 additions and 0 deletions
@@ -58,6 +58,14 @@ import { useRoomCreators } from '../../../hooks/useRoomCreators';
import { getMouseEventCords } from '../../../utils/dom'; import { getMouseEventCords } from '../../../utils/dom';
import { useCrossSigningActive } from '../../../hooks/useCrossSigning'; import { useCrossSigningActive } from '../../../hooks/useCrossSigning';
import { MemberVerificationBadge } from '../../../components/MemberVerificationBadge'; import { MemberVerificationBadge } from '../../../components/MemberVerificationBadge';
import { useUserPresence } from '../../../hooks/useUserPresence';
import { PresenceBadge } from '../../../components/presence';
function MemberPresenceBadge({ userId }: { userId: string }) {
const presence = useUserPresence(userId);
if (!presence) return null;
return <PresenceBadge presence={presence.presence} status={presence.status} size="300" />;
}
const SEARCH_OPTIONS: UseAsyncSearchOptions = { const SEARCH_OPTIONS: UseAsyncSearchOptions = {
limit: 1000, limit: 1000,
@@ -333,6 +341,7 @@ export function Members({ requestClose }: MembersProps) {
useAuthentication={useAuthentication} useAuthentication={useAuthentication}
after={ after={
<> <>
<MemberPresenceBadge userId={tagOrMember.userId} />
{showEncryption && ( {showEncryption && (
<MemberVerificationBadge userId={tagOrMember.userId} /> <MemberVerificationBadge userId={tagOrMember.userId} />
)} )}
+6
View File
@@ -61,6 +61,8 @@ import { useFlattenPowerTagMembers, useGetMemberPowerTag } from '../../hooks/use
import { useRoomCreators } from '../../hooks/useRoomCreators'; import { useRoomCreators } from '../../hooks/useRoomCreators';
import { useCrossSigningActive } from '../../hooks/useCrossSigning'; import { useCrossSigningActive } from '../../hooks/useCrossSigning';
import { MemberVerificationBadge } from '../../components/MemberVerificationBadge'; import { MemberVerificationBadge } from '../../components/MemberVerificationBadge';
import { useUserPresence } from '../../hooks/useUserPresence';
import { PresenceBadge } from '../../components/presence';
type MemberDrawerHeaderProps = { type MemberDrawerHeaderProps = {
room: Room; room: Room;
@@ -131,6 +133,7 @@ function MemberItem({
const avatarUrl = avatarMxcUrl const avatarUrl = avatarMxcUrl
? mx.mxcUrlToHttp(avatarMxcUrl, 100, 100, 'crop', undefined, false, useAuthentication) ? mx.mxcUrlToHttp(avatarMxcUrl, 100, 100, 'crop', undefined, false, useAuthentication)
: undefined; : undefined;
const presence = useUserPresence(member.userId);
return ( return (
<MenuItem <MenuItem
@@ -152,6 +155,9 @@ function MemberItem({
} }
after={ after={
<> <>
{presence && (
<PresenceBadge presence={presence.presence} status={presence.status} size="300" />
)}
{showEncryption && <MemberVerificationBadge userId={member.userId} />} {showEncryption && <MemberVerificationBadge userId={member.userId} />}
{typing && ( {typing && (
<Badge size="300" variant="Secondary" fill="Soft" radii="Pill" outlined> <Badge size="300" variant="Secondary" fill="Soft" radii="Pill" outlined>