Skip to content
This repository was archived by the owner on Sep 11, 2024. It is now read-only.

Commit 3f3005a

Browse files
author
Germain
authored
Always use current profile on thread events (#9524)
1 parent 5fb0f5c commit 3f3005a

File tree

7 files changed

+160
-145
lines changed

7 files changed

+160
-145
lines changed

src/components/views/avatars/MemberAvatar.tsx

Lines changed: 50 additions & 94 deletions
Original file line numberDiff line numberDiff line change
@@ -15,19 +15,17 @@ See the License for the specific language governing permissions and
1515
limitations under the License.
1616
*/
1717

18-
import React from 'react';
18+
import React, { useContext } from 'react';
1919
import { RoomMember } from "matrix-js-sdk/src/models/room-member";
2020
import { ResizeMethod } from 'matrix-js-sdk/src/@types/partials';
21-
import { logger } from "matrix-js-sdk/src/logger";
2221

2322
import dis from "../../../dispatcher/dispatcher";
2423
import { Action } from "../../../dispatcher/actions";
2524
import BaseAvatar from "./BaseAvatar";
2625
import { mediaFromMxc } from "../../../customisations/Media";
2726
import { CardContext } from '../right_panel/context';
2827
import UserIdentifierCustomisations from '../../../customisations/UserIdentifier';
29-
import SettingsStore from "../../../settings/SettingsStore";
30-
import { MatrixClientPeg } from "../../../MatrixClientPeg";
28+
import { useRoomMemberProfile } from '../../../hooks/room/useRoomMemberProfile';
3129

3230
interface IProps extends Omit<React.ComponentProps<typeof BaseAvatar>, "name" | "idName" | "url"> {
3331
member: RoomMember | null;
@@ -46,100 +44,58 @@ interface IProps extends Omit<React.ComponentProps<typeof BaseAvatar>, "name" |
4644
hideTitle?: boolean;
4745
}
4846

49-
interface IState {
50-
name: string;
51-
title: string;
52-
imageUrl?: string;
53-
}
54-
55-
export default class MemberAvatar extends React.PureComponent<IProps, IState> {
56-
public static defaultProps = {
57-
width: 40,
58-
height: 40,
59-
resizeMethod: 'crop',
60-
viewUserOnClick: false,
61-
};
62-
63-
constructor(props: IProps) {
64-
super(props);
65-
66-
this.state = MemberAvatar.getState(props);
67-
}
68-
69-
public static getDerivedStateFromProps(nextProps: IProps): IState {
70-
return MemberAvatar.getState(nextProps);
71-
}
72-
73-
private static getState(props: IProps): IState {
74-
let member = props.member;
75-
if (member && !props.forceHistorical && SettingsStore.getValue("useOnlyCurrentProfiles")) {
76-
const room = MatrixClientPeg.get().getRoom(member.roomId);
77-
if (room) {
78-
member = room.getMember(member.userId);
79-
}
80-
}
81-
if (member?.name) {
82-
let imageUrl = null;
83-
const userTitle = UserIdentifierCustomisations.getDisplayUserIdentifier(
84-
member.userId, { roomId: member?.roomId },
47+
export default function MemberAvatar({
48+
width,
49+
height,
50+
resizeMethod = 'crop',
51+
viewUserOnClick,
52+
...props
53+
}: IProps) {
54+
const card = useContext(CardContext);
55+
56+
const member = useRoomMemberProfile({
57+
userId: props.member?.userId,
58+
member: props.member,
59+
forceHistorical: props.forceHistorical,
60+
});
61+
62+
const name = member?.name ?? props.fallbackUserId;
63+
let title: string | undefined = props.title;
64+
let imageUrl: string | undefined;
65+
if (member?.name) {
66+
if (member.getMxcAvatarUrl()) {
67+
imageUrl = mediaFromMxc(member.getMxcAvatarUrl() ?? "").getThumbnailOfSourceHttp(
68+
width,
69+
height,
70+
resizeMethod,
8571
);
86-
if (member.getMxcAvatarUrl()) {
87-
imageUrl = mediaFromMxc(member.getMxcAvatarUrl()).getThumbnailOfSourceHttp(
88-
props.width,
89-
props.height,
90-
props.resizeMethod,
91-
);
92-
}
93-
return {
94-
name: member.name,
95-
title: props.title || userTitle,
96-
imageUrl: imageUrl,
97-
};
98-
} else if (props.fallbackUserId) {
99-
return {
100-
name: props.fallbackUserId,
101-
title: props.fallbackUserId,
102-
};
103-
} else {
104-
logger.error("MemberAvatar called somehow with null member or fallbackUserId");
105-
return {} as IState; // prevent an explosion
10672
}
107-
}
108-
109-
render() {
110-
let {
111-
member,
112-
fallbackUserId,
113-
onClick,
114-
viewUserOnClick,
115-
// eslint-disable-next-line @typescript-eslint/no-unused-vars
116-
forceHistorical,
117-
hideTitle,
118-
...otherProps
119-
} = this.props;
120-
const userId = member ? member.userId : fallbackUserId;
12173

122-
if (viewUserOnClick) {
123-
onClick = () => {
74+
if (!title) {
75+
title = UserIdentifierCustomisations.getDisplayUserIdentifier(
76+
member?.userId ?? "", { roomId: member?.roomId ?? "" },
77+
) ?? props.fallbackUserId;
78+
}
79+
}
80+
const userId = member?.userId ?? props.fallbackUserId;
81+
82+
return (
83+
<BaseAvatar
84+
{...props}
85+
width={width}
86+
height={height}
87+
resizeMethod={resizeMethod}
88+
name={name ?? ""}
89+
title={props.hideTitle ? undefined : title}
90+
idName={userId}
91+
url={imageUrl}
92+
onClick={viewUserOnClick ? () => {
12493
dis.dispatch({
12594
action: Action.ViewUser,
126-
member: this.props.member,
127-
push: this.context.isCard,
95+
member: props.member,
96+
push: card.isCard,
12897
});
129-
};
130-
}
131-
132-
return (
133-
<BaseAvatar
134-
{...otherProps}
135-
name={this.state.name}
136-
title={hideTitle ? undefined : this.state.title}
137-
idName={userId}
138-
url={this.state.imageUrl}
139-
onClick={onClick}
140-
/>
141-
);
142-
}
98+
} : props.onClick}
99+
/>
100+
);
143101
}
144-
145-
MemberAvatar.contextType = CardContext;

src/components/views/messages/DisambiguatedProfile.tsx

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -23,7 +23,7 @@ import { getUserNameColorClass } from '../../../utils/FormattingUtils';
2323
import UserIdentifier from "../../../customisations/UserIdentifier";
2424

2525
interface IProps {
26-
member?: RoomMember;
26+
member?: RoomMember | null;
2727
fallbackName: string;
2828
onClick?(): void;
2929
colored?: boolean;

src/components/views/messages/SenderProfile.tsx

Lines changed: 16 additions & 40 deletions
Original file line numberDiff line numberDiff line change
@@ -18,51 +18,27 @@ import React from 'react';
1818
import { MatrixEvent } from "matrix-js-sdk/src/models/event";
1919
import { MsgType } from "matrix-js-sdk/src/@types/event";
2020

21-
import MatrixClientContext from "../../../contexts/MatrixClientContext";
2221
import DisambiguatedProfile from "./DisambiguatedProfile";
23-
import RoomContext, { TimelineRenderingType } from '../../../contexts/RoomContext';
24-
import SettingsStore from "../../../settings/SettingsStore";
25-
import { MatrixClientPeg } from "../../../MatrixClientPeg";
22+
import { useRoomMemberProfile } from '../../../hooks/room/useRoomMemberProfile';
2623

2724
interface IProps {
2825
mxEvent: MatrixEvent;
2926
onClick?(): void;
3027
}
3128

32-
export default class SenderProfile extends React.PureComponent<IProps> {
33-
public static contextType = MatrixClientContext;
34-
public context!: React.ContextType<typeof MatrixClientContext>;
35-
36-
render() {
37-
const { mxEvent, onClick } = this.props;
38-
const msgtype = mxEvent.getContent().msgtype;
39-
40-
let member = mxEvent.sender;
41-
if (SettingsStore.getValue("useOnlyCurrentProfiles")) {
42-
const room = MatrixClientPeg.get().getRoom(mxEvent.getRoomId());
43-
if (room) {
44-
member = room.getMember(mxEvent.getSender());
45-
}
46-
}
47-
48-
return <RoomContext.Consumer>
49-
{ roomContext => {
50-
if (msgtype === MsgType.Emote &&
51-
roomContext.timelineRenderingType !== TimelineRenderingType.ThreadsList
52-
) {
53-
return null; // emote message must include the name so don't duplicate it
54-
}
55-
56-
return (
57-
<DisambiguatedProfile
58-
fallbackName={mxEvent.getSender() || ""}
59-
onClick={onClick}
60-
member={member}
61-
colored={true}
62-
emphasizeDisplayName={true}
63-
/>
64-
);
65-
} }
66-
</RoomContext.Consumer>;
67-
}
29+
export default function SenderProfile({ mxEvent, onClick }: IProps) {
30+
const member = useRoomMemberProfile({
31+
userId: mxEvent.getSender(),
32+
member: mxEvent.sender,
33+
});
34+
35+
return mxEvent.getContent().msgtype !== MsgType.Emote
36+
? <DisambiguatedProfile
37+
fallbackName={mxEvent.getSender() ?? ""}
38+
onClick={onClick}
39+
member={member}
40+
colored={true}
41+
emphasizeDisplayName={true}
42+
/>
43+
: null;
6844
}

src/components/views/rooms/EventTile.tsx

Lines changed: 0 additions & 7 deletions
Original file line numberDiff line numberDiff line change
@@ -1311,7 +1311,6 @@ export class UnwrappedEventTile extends React.Component<EventTileProps, IState>
13111311
]);
13121312
}
13131313
case TimelineRenderingType.Thread: {
1314-
const room = MatrixClientPeg.get().getRoom(this.props.mxEvent.getRoomId());
13151314
return React.createElement(this.props.as || "li", {
13161315
"ref": this.ref,
13171316
"className": classes,
@@ -1325,12 +1324,6 @@ export class UnwrappedEventTile extends React.Component<EventTileProps, IState>
13251324
"onMouseEnter": () => this.setState({ hover: true }),
13261325
"onMouseLeave": () => this.setState({ hover: false }),
13271326
}, [
1328-
<div className="mx_EventTile_roomName" key="mx_EventTile_roomName">
1329-
<RoomAvatar room={room} width={28} height={28} />
1330-
<a href={permalink} onClick={this.onPermalinkClicked}>
1331-
{ room ? room.name : '' }
1332-
</a>
1333-
</div>,
13341327
<div className="mx_EventTile_senderDetails" key="mx_EventTile_senderDetails">
13351328
{ avatar }
13361329
{ sender }
Lines changed: 46 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,46 @@
1+
/*
2+
Copyright 2022 The Matrix.org Foundation C.I.C.
3+
4+
Licensed under the Apache License, Version 2.0 (the "License");
5+
you may not use this file except in compliance with the License.
6+
You may obtain a copy of the License at
7+
8+
http://www.apache.org/licenses/LICENSE-2.0
9+
10+
Unless required by applicable law or agreed to in writing, software
11+
distributed under the License is distributed on an "AS IS" BASIS,
12+
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
See the License for the specific language governing permissions and
14+
limitations under the License.
15+
*/
16+
17+
import { RoomMember } from "matrix-js-sdk/src/models/room-member";
18+
import { useContext, useEffect, useState } from "react";
19+
20+
import RoomContext, { TimelineRenderingType } from "../../contexts/RoomContext";
21+
import { useSettingValue } from "../useSettings";
22+
23+
export function useRoomMemberProfile({
24+
userId = "",
25+
member: propMember,
26+
forceHistorical = false,
27+
}: {
28+
userId: string | undefined;
29+
member?: RoomMember | null;
30+
forceHistorical?: boolean;
31+
}): RoomMember | undefined | null {
32+
const [member, setMember] = useState<RoomMember | undefined | null>(propMember);
33+
34+
const context = useContext(RoomContext);
35+
const useOnlyCurrentProfiles = useSettingValue("useOnlyCurrentProfiles");
36+
37+
useEffect(() => {
38+
const threadContexts = [TimelineRenderingType.ThreadsList, TimelineRenderingType.Thread];
39+
if ((propMember && !forceHistorical && useOnlyCurrentProfiles)
40+
|| threadContexts.includes(context?.timelineRenderingType)) {
41+
setMember(context?.room?.getMember(userId));
42+
}
43+
}, [forceHistorical, propMember, context.room, context?.timelineRenderingType, useOnlyCurrentProfiles, userId]);
44+
45+
return member;
46+
}

test/components/views/beacon/__snapshots__/BeaconMarker-test.tsx.snap

Lines changed: 46 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -184,21 +184,65 @@ exports[`<BeaconMarker /> renders marker when beacon has location 1`] = `
184184
Symbol(kCapture): false,
185185
}
186186
}
187-
resizeMethod="crop"
188187
viewUserOnClick={false}
189188
width={36}
190189
>
191190
<BaseAvatar
192191
height={36}
192+
hideTitle={false}
193193
idName="@alice:server"
194+
member={
195+
RoomMember {
196+
"_events": {},
197+
"_eventsCount": 0,
198+
"_isOutOfBand": false,
199+
"_maxListeners": undefined,
200+
"disambiguate": false,
201+
"events": {},
202+
"membership": undefined,
203+
"modified": 1647270879403,
204+
"name": "@alice:server",
205+
"powerLevel": 0,
206+
"powerLevelNorm": 0,
207+
"rawDisplayName": "@alice:server",
208+
"requestedProfileInfo": false,
209+
"roomId": "!room:server",
210+
"typing": false,
211+
"user": undefined,
212+
"userId": "@alice:server",
213+
Symbol(kCapture): false,
214+
}
215+
}
194216
name="@alice:server"
195217
resizeMethod="crop"
196218
title="@alice:server"
197-
url={null}
198219
width={36}
199220
>
200221
<span
201222
className="mx_BaseAvatar"
223+
hideTitle={false}
224+
member={
225+
RoomMember {
226+
"_events": {},
227+
"_eventsCount": 0,
228+
"_isOutOfBand": false,
229+
"_maxListeners": undefined,
230+
"disambiguate": false,
231+
"events": {},
232+
"membership": undefined,
233+
"modified": 1647270879403,
234+
"name": "@alice:server",
235+
"powerLevel": 0,
236+
"powerLevelNorm": 0,
237+
"rawDisplayName": "@alice:server",
238+
"requestedProfileInfo": false,
239+
"roomId": "!room:server",
240+
"typing": false,
241+
"user": undefined,
242+
"userId": "@alice:server",
243+
Symbol(kCapture): false,
244+
}
245+
}
202246
role="presentation"
203247
>
204248
<span

test/components/views/messages/__snapshots__/TextualBody-test.tsx.snap

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -14,4 +14,4 @@ exports[`<TextualBody /> renders formatted m.text correctly pills do not appear
1414
</span>"
1515
`;
1616

17-
exports[`<TextualBody /> renders formatted m.text correctly pills get injected correctly into the DOM 1`] = `"<span class="mx_EventTile_body markdown-body" dir="auto">Hey <span><bdi><a class="mx_Pill mx_UserPill"><img class="mx_BaseAvatar mx_BaseAvatar_image" src="mxc://avatar.url/image.png" style="width: 16px; height: 16px;" alt="" aria-hidden="true"><span class="mx_Pill_linkText">Member</span></a></bdi></span></span>"`;
17+
exports[`<TextualBody /> renders formatted m.text correctly pills get injected correctly into the DOM 1`] = `"<span class="mx_EventTile_body markdown-body" dir="auto">Hey <span><bdi><a class="mx_Pill mx_UserPill"><img class="mx_BaseAvatar mx_BaseAvatar_image" src="mxc://avatar.url/image.png" style="width: 16px; height: 16px;" alt="" member="[object Object]" aria-hidden="true"><span class="mx_Pill_linkText">Member</span></a></bdi></span></span>"`;

0 commit comments

Comments
 (0)