mirror of
https://github.com/vector-im/element-web.git
synced 2025-08-23 15:31:22 +02:00
* Add another null guard for member To hopefully fix https://github.com/vector-im/element-web/issues/21319 * Use getSender() Co-authored-by: Michael Telatynski <7t3chguy@gmail.com> Co-authored-by: Michael Telatynski <7t3chguy@gmail.com>
166 lines
5.9 KiB
TypeScript
166 lines
5.9 KiB
TypeScript
/*
|
|
Copyright 2015, 2016 OpenMarket Ltd
|
|
|
|
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 from 'react';
|
|
import { MatrixEvent } from "matrix-js-sdk/src/models/event";
|
|
import { MsgType } from "matrix-js-sdk/src/@types/event";
|
|
import { RoomStateEvent } from "matrix-js-sdk/src/models/room-state";
|
|
|
|
import Flair from '../elements/Flair';
|
|
import FlairStore from '../../../stores/FlairStore';
|
|
import { getUserNameColorClass } from '../../../utils/FormattingUtils';
|
|
import MatrixClientContext from "../../../contexts/MatrixClientContext";
|
|
import { replaceableComponent } from "../../../utils/replaceableComponent";
|
|
import UserIdentifier from '../../../customisations/UserIdentifier';
|
|
import RoomContext, { TimelineRenderingType } from '../../../contexts/RoomContext';
|
|
import SettingsStore from "../../../settings/SettingsStore";
|
|
import { MatrixClientPeg } from "../../../MatrixClientPeg";
|
|
|
|
interface IProps {
|
|
mxEvent: MatrixEvent;
|
|
onClick?(): void;
|
|
enableFlair: boolean;
|
|
}
|
|
|
|
interface IState {
|
|
userGroups: string[];
|
|
relatedGroups: string[];
|
|
}
|
|
|
|
@replaceableComponent("views.messages.SenderProfile")
|
|
export default class SenderProfile extends React.Component<IProps, IState> {
|
|
static contextType = MatrixClientContext;
|
|
public context!: React.ContextType<typeof MatrixClientContext>;
|
|
private unmounted = false;
|
|
|
|
constructor(props: IProps) {
|
|
super(props);
|
|
const senderId = this.props.mxEvent.getSender();
|
|
|
|
this.state = {
|
|
userGroups: FlairStore.cachedPublicisedGroups(senderId) || [],
|
|
relatedGroups: [],
|
|
};
|
|
}
|
|
|
|
componentDidMount() {
|
|
this.updateRelatedGroups();
|
|
|
|
if (this.state.userGroups.length === 0) {
|
|
this.getPublicisedGroups();
|
|
}
|
|
|
|
this.context.on(RoomStateEvent.Events, this.onRoomStateEvents);
|
|
}
|
|
|
|
componentWillUnmount() {
|
|
this.unmounted = true;
|
|
this.context.removeListener(RoomStateEvent.Events, this.onRoomStateEvents);
|
|
}
|
|
|
|
private async getPublicisedGroups() {
|
|
const userGroups = await FlairStore.getPublicisedGroupsCached(this.context, this.props.mxEvent.getSender());
|
|
if (this.unmounted) return;
|
|
this.setState({ userGroups });
|
|
}
|
|
|
|
private onRoomStateEvents = (event: MatrixEvent) => {
|
|
if (event.getType() === 'm.room.related_groups' && event.getRoomId() === this.props.mxEvent.getRoomId()) {
|
|
this.updateRelatedGroups();
|
|
}
|
|
};
|
|
|
|
private updateRelatedGroups() {
|
|
const room = this.context.getRoom(this.props.mxEvent.getRoomId());
|
|
if (!room) return;
|
|
|
|
const relatedGroupsEvent = room.currentState.getStateEvents('m.room.related_groups', '');
|
|
this.setState({
|
|
relatedGroups: relatedGroupsEvent?.getContent().groups || [],
|
|
});
|
|
}
|
|
|
|
private getDisplayedGroups(userGroups?: string[], relatedGroups?: string[]) {
|
|
let displayedGroups = userGroups || [];
|
|
if (relatedGroups && relatedGroups.length > 0) {
|
|
displayedGroups = relatedGroups.filter((groupId) => {
|
|
return displayedGroups.includes(groupId);
|
|
});
|
|
} else {
|
|
displayedGroups = [];
|
|
}
|
|
return displayedGroups;
|
|
}
|
|
|
|
render() {
|
|
const { mxEvent } = this.props;
|
|
const colorClass = getUserNameColorClass(mxEvent.getSender());
|
|
const { msgtype } = mxEvent.getContent();
|
|
|
|
let member = mxEvent.sender;
|
|
if (SettingsStore.getValue("feature_use_only_current_profiles")) {
|
|
const room = MatrixClientPeg.get().getRoom(mxEvent.getRoomId());
|
|
if (room) {
|
|
member = room.getMember(mxEvent.getSender());
|
|
}
|
|
}
|
|
|
|
const disambiguate = member?.disambiguate || mxEvent.sender?.disambiguate;
|
|
const displayName = member?.rawDisplayName || mxEvent.getSender() || "";
|
|
const mxid = member?.userId || mxEvent.getSender() || "";
|
|
|
|
return <RoomContext.Consumer>
|
|
{ roomContext => {
|
|
if (msgtype === MsgType.Emote &&
|
|
roomContext.timelineRenderingType !== TimelineRenderingType.ThreadsList
|
|
) {
|
|
return null; // emote message must include the name so don't duplicate it
|
|
}
|
|
|
|
let mxidElement;
|
|
if (disambiguate) {
|
|
mxidElement = (
|
|
<span className="mx_SenderProfile_mxid">
|
|
{ UserIdentifier.getDisplayUserIdentifier(
|
|
mxid, { withDisplayName: true, roomId: mxEvent.getRoomId() },
|
|
) }
|
|
</span>
|
|
);
|
|
}
|
|
|
|
let flair;
|
|
if (this.props.enableFlair) {
|
|
const displayedGroups = this.getDisplayedGroups(
|
|
this.state.userGroups, this.state.relatedGroups,
|
|
);
|
|
|
|
flair = <Flair key='flair' userId={mxEvent.getSender()} groups={displayedGroups} />;
|
|
}
|
|
|
|
return (
|
|
<div className="mx_SenderProfile" dir="auto" onClick={this.props.onClick}>
|
|
<span className={`mx_SenderProfile_displayName ${colorClass}`}>
|
|
{ displayName }
|
|
</span>
|
|
{ mxidElement }
|
|
{ flair }
|
|
</div>
|
|
);
|
|
} }
|
|
</RoomContext.Consumer>;
|
|
}
|
|
}
|