matrix-react-sdk
Version:
SDK for matrix.org using React
663 lines (659 loc) • 98.2 kB
JavaScript
"use strict";
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = void 0;
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _react = _interopRequireDefault(require("react"));
var _matrix = require("matrix-js-sdk/src/matrix");
var _types = require("matrix-js-sdk/src/types");
var _classnames = _interopRequireDefault(require("classnames"));
var _RoomViewLifecycle = require("@matrix-org/react-sdk-module-api/lib/lifecycles/RoomViewLifecycle");
var _MatrixClientPeg = require("../../../MatrixClientPeg");
var _dispatcher = _interopRequireDefault(require("../../../dispatcher/dispatcher"));
var _languageHandler = require("../../../languageHandler");
var _SdkConfig = _interopRequireDefault(require("../../../SdkConfig"));
var _IdentityAuthClient = _interopRequireDefault(require("../../../IdentityAuthClient"));
var _InviteReason = _interopRequireDefault(require("../elements/InviteReason"));
var _Spinner = _interopRequireDefault(require("../elements/Spinner"));
var _AccessibleButton = _interopRequireDefault(require("../elements/AccessibleButton"));
var _RoomAvatar = _interopRequireDefault(require("../avatars/RoomAvatar"));
var _SettingsStore = _interopRequireDefault(require("../../../settings/SettingsStore"));
var _UIFeature = require("../../../settings/UIFeature");
var _ModuleRunner = require("../../../modules/ModuleRunner");
var _askToJoin = require("../../../../res/img/element-icons/ask-to-join.svg");
var _Field = _interopRequireDefault(require("../elements/Field"));
/*
Copyright 2024 New Vector Ltd.
Copyright 2015-2021 The Matrix.org Foundation C.I.C.
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
Please see LICENSE files in the repository root for full details.
*/
const MemberEventHtmlReasonField = "io.element.html_reason";
var MessageCase = /*#__PURE__*/function (MessageCase) {
MessageCase["NotLoggedIn"] = "NotLoggedIn";
MessageCase["Joining"] = "Joining";
MessageCase["Loading"] = "Loading";
MessageCase["Rejecting"] = "Rejecting";
MessageCase["Kicked"] = "Kicked";
MessageCase["Banned"] = "Banned";
MessageCase["OtherThreePIDError"] = "OtherThreePIDError";
MessageCase["InvitedEmailNotFoundInAccount"] = "InvitedEmailNotFoundInAccount";
MessageCase["InvitedEmailNoIdentityServer"] = "InvitedEmailNoIdentityServer";
MessageCase["InvitedEmailMismatch"] = "InvitedEmailMismatch";
MessageCase["Invite"] = "Invite";
MessageCase["ViewingRoom"] = "ViewingRoom";
MessageCase["RoomNotFound"] = "RoomNotFound";
MessageCase["OtherError"] = "OtherError";
MessageCase["PromptAskToJoin"] = "PromptAskToJoin";
MessageCase["Knocked"] = "Knocked";
MessageCase["RequestDenied"] = "requestDenied";
return MessageCase;
}(MessageCase || {});
class RoomPreviewBar extends _react.default.Component {
constructor(props) {
super(props);
(0, _defineProperty2.default)(this, "onLoginClick", () => {
_dispatcher.default.dispatch({
action: "start_login",
screenAfterLogin: this.makeScreenAfterLogin()
});
});
(0, _defineProperty2.default)(this, "onRegisterClick", () => {
_dispatcher.default.dispatch({
action: "start_registration",
screenAfterLogin: this.makeScreenAfterLogin()
});
});
(0, _defineProperty2.default)(this, "onChangeReason", event => {
this.setState({
reason: event.target.value
});
});
this.state = {
busy: false
};
}
componentDidMount() {
this.checkInvitedEmail();
}
componentDidUpdate(prevProps, prevState) {
if (this.props.invitedEmail !== prevProps.invitedEmail || this.props.inviterName !== prevProps.inviterName) {
this.checkInvitedEmail();
}
}
async checkInvitedEmail() {
// If this is an invite and we've been told what email address was
// invited, fetch the user's account emails and discovery bindings so we
// can check them against the email that was invited.
if (this.props.inviterName && this.props.invitedEmail) {
this.setState({
busy: true
});
try {
// Gather the account 3PIDs
const account3pids = await _MatrixClientPeg.MatrixClientPeg.safeGet().getThreePids();
this.setState({
accountEmails: account3pids.threepids.filter(b => b.medium === "email").map(b => b.address)
});
// If we have an IS connected, use that to lookup the email and
// check the bound MXID.
if (!_MatrixClientPeg.MatrixClientPeg.safeGet().getIdentityServerUrl()) {
this.setState({
busy: false
});
return;
}
const authClient = new _IdentityAuthClient.default();
const identityAccessToken = await authClient.getAccessToken();
const result = await _MatrixClientPeg.MatrixClientPeg.safeGet().lookupThreePid("email", this.props.invitedEmail, identityAccessToken);
if (!("mxid" in result)) {
throw new _languageHandler.UserFriendlyError("room|error_3pid_invite_email_lookup");
}
this.setState({
invitedEmailMxid: result.mxid
});
} catch (err) {
this.setState({
threePidFetchError: err
});
}
this.setState({
busy: false
});
}
}
getMessageCase() {
const isGuest = _MatrixClientPeg.MatrixClientPeg.safeGet().isGuest();
if (isGuest) {
return MessageCase.NotLoggedIn;
}
const myMember = this.getMyMember();
if (myMember) {
const previousMembership = myMember.events.member?.getPrevContent().membership;
if (myMember.isKicked()) {
if (previousMembership === _types.KnownMembership.Knock) {
return MessageCase.RequestDenied;
} else if (this.props.promptAskToJoin) {
return MessageCase.PromptAskToJoin;
}
return MessageCase.Kicked;
} else if (myMember.membership === _types.KnownMembership.Ban) {
return MessageCase.Banned;
}
}
if (this.props.joining) {
return MessageCase.Joining;
} else if (this.props.rejecting) {
return MessageCase.Rejecting;
} else if (this.props.loading || this.state.busy) {
return MessageCase.Loading;
} else if (this.props.knocked) {
return MessageCase.Knocked;
} else if (this.props.canAskToJoinAndMembershipIsLeave || this.props.promptAskToJoin) {
return MessageCase.PromptAskToJoin;
}
if (this.props.inviterName) {
if (this.props.invitedEmail) {
if (this.state.threePidFetchError) {
return MessageCase.OtherThreePIDError;
} else if (this.state.accountEmails && !this.state.accountEmails.includes(this.props.invitedEmail)) {
return MessageCase.InvitedEmailNotFoundInAccount;
} else if (!_MatrixClientPeg.MatrixClientPeg.safeGet().getIdentityServerUrl()) {
return MessageCase.InvitedEmailNoIdentityServer;
} else if (this.state.invitedEmailMxid != _MatrixClientPeg.MatrixClientPeg.safeGet().getUserId()) {
return MessageCase.InvitedEmailMismatch;
}
}
return MessageCase.Invite;
} else if (this.props.error) {
if (this.props.error.errcode == "M_NOT_FOUND") {
return MessageCase.RoomNotFound;
} else {
return MessageCase.OtherError;
}
} else {
return MessageCase.ViewingRoom;
}
}
getKickOrBanInfo() {
const myMember = this.getMyMember();
if (!myMember) {
return {};
}
const kickerUserId = myMember.events.member?.getSender();
const kickerMember = kickerUserId ? this.props.room?.currentState.getMember(kickerUserId) : undefined;
const memberName = kickerMember?.name ?? kickerUserId;
const reason = myMember.events.member?.getContent().reason;
return {
memberName,
reason
};
}
joinRule() {
return this.props.room?.currentState.getStateEvents(_matrix.EventType.RoomJoinRules, "")?.getContent().join_rule ?? null;
}
getMyMember() {
return this.props.room?.getMember(_MatrixClientPeg.MatrixClientPeg.safeGet().getSafeUserId()) ?? null;
}
getInviteMember() {
const {
room
} = this.props;
if (!room) {
return null;
}
const myUserId = _MatrixClientPeg.MatrixClientPeg.safeGet().getSafeUserId();
const inviteEvent = room.currentState.getMember(myUserId);
if (!inviteEvent) {
return null;
}
const inviterUserId = inviteEvent.events.member?.getSender();
return inviterUserId ? room.currentState.getMember(inviterUserId) : null;
}
isDMInvite() {
const myMember = this.getMyMember();
if (!myMember) {
return false;
}
const memberContent = myMember.events.member?.getContent();
return memberContent?.membership === _types.KnownMembership.Invite && memberContent.is_direct;
}
makeScreenAfterLogin() {
return {
screen: "room",
params: {
email: this.props.invitedEmail,
signurl: this.props.signUrl,
room_name: this.props.oobData?.name ?? null,
room_avatar_url: this.props.oobData?.avatarUrl ?? null,
inviter_name: this.props.oobData?.inviterName ?? null
}
};
}
render() {
const brand = _SdkConfig.default.get().brand;
const roomName = this.props.room?.name ?? this.props.roomAlias ?? "";
const isSpace = this.props.room?.isSpaceRoom() ?? this.props.oobData?.roomType === _matrix.RoomType.Space;
let showSpinner = false;
let title;
let subTitle;
let reasonElement;
let primaryActionHandler;
let primaryActionLabel;
let secondaryActionHandler;
let secondaryActionLabel;
let footer;
const extraComponents = [];
const messageCase = this.getMessageCase();
switch (messageCase) {
case MessageCase.Joining:
{
if (this.props.oobData?.roomType || isSpace) {
title = isSpace ? (0, _languageHandler._t)("room|joining_space") : (0, _languageHandler._t)("room|joining_room");
} else {
title = (0, _languageHandler._t)("room|joining");
}
showSpinner = true;
break;
}
case MessageCase.Loading:
{
title = (0, _languageHandler._t)("common|loading");
showSpinner = true;
break;
}
case MessageCase.Rejecting:
{
title = (0, _languageHandler._t)("room|rejecting");
showSpinner = true;
break;
}
case MessageCase.NotLoggedIn:
{
const opts = {
canJoin: false
};
if (this.props.roomId) {
_ModuleRunner.ModuleRunner.instance.invoke(_RoomViewLifecycle.RoomViewLifecycle.PreviewRoomNotLoggedIn, opts, this.props.roomId);
}
if (opts.canJoin) {
title = (0, _languageHandler._t)("room|join_title");
primaryActionLabel = (0, _languageHandler._t)("action|join");
primaryActionHandler = () => {
_ModuleRunner.ModuleRunner.instance.invoke(_RoomViewLifecycle.RoomViewLifecycle.JoinFromRoomPreview, this.props.roomId);
};
} else {
title = (0, _languageHandler._t)("room|join_title_account");
if (_SettingsStore.default.getValue(_UIFeature.UIFeature.Registration)) {
primaryActionLabel = (0, _languageHandler._t)("room|join_button_account");
primaryActionHandler = this.onRegisterClick;
}
secondaryActionLabel = (0, _languageHandler._t)("action|sign_in");
secondaryActionHandler = this.onLoginClick;
}
if (this.props.previewLoading) {
footer = /*#__PURE__*/_react.default.createElement("div", null, /*#__PURE__*/_react.default.createElement(_Spinner.default, {
w: 20,
h: 20
}), (0, _languageHandler._t)("room|loading_preview"));
}
break;
}
case MessageCase.Kicked:
{
const {
memberName,
reason
} = this.getKickOrBanInfo();
if (roomName) {
title = (0, _languageHandler._t)("room|kicked_from_room_by", {
memberName,
roomName
});
} else {
title = (0, _languageHandler._t)("room|kicked_by", {
memberName
});
}
subTitle = reason ? (0, _languageHandler._t)("room|kick_reason", {
reason
}) : undefined;
if (isSpace) {
primaryActionLabel = (0, _languageHandler._t)("room|forget_space");
} else {
primaryActionLabel = (0, _languageHandler._t)("room|forget_room");
}
primaryActionHandler = this.props.onForgetClick;
if (this.joinRule() !== _matrix.JoinRule.Invite) {
secondaryActionLabel = primaryActionLabel;
secondaryActionHandler = primaryActionHandler;
primaryActionLabel = (0, _languageHandler._t)("room|rejoin_button");
primaryActionHandler = this.props.onJoinClick;
}
break;
}
case MessageCase.RequestDenied:
{
title = (0, _languageHandler._t)("room|knock_denied_title");
subTitle = (0, _languageHandler._t)("room|knock_denied_subtitle");
if (isSpace) {
primaryActionLabel = (0, _languageHandler._t)("room|forget_space");
} else {
primaryActionLabel = (0, _languageHandler._t)("room|forget_room");
}
primaryActionHandler = this.props.onForgetClick;
break;
}
case MessageCase.Banned:
{
const {
memberName,
reason
} = this.getKickOrBanInfo();
if (roomName) {
title = (0, _languageHandler._t)("room|banned_from_room_by", {
memberName,
roomName
});
} else {
title = (0, _languageHandler._t)("room|banned_by", {
memberName
});
}
subTitle = reason ? (0, _languageHandler._t)("room|kick_reason", {
reason
}) : undefined;
if (isSpace) {
primaryActionLabel = (0, _languageHandler._t)("room|forget_space");
} else {
primaryActionLabel = (0, _languageHandler._t)("room|forget_room");
}
primaryActionHandler = this.props.onForgetClick;
break;
}
case MessageCase.OtherThreePIDError:
{
if (roomName) {
title = (0, _languageHandler._t)("room|3pid_invite_error_title_room", {
roomName
});
} else {
title = (0, _languageHandler._t)("room|3pid_invite_error_title");
}
const joinRule = this.joinRule();
const errCodeMessage = (0, _languageHandler._t)("room|3pid_invite_error_description", {
errcode: this.state.threePidFetchError?.errcode || (0, _languageHandler._t)("error|unknown_error_code")
});
switch (joinRule) {
case "invite":
subTitle = [(0, _languageHandler._t)("room|3pid_invite_error_invite_subtitle"), errCodeMessage];
primaryActionLabel = (0, _languageHandler._t)("room|3pid_invite_error_invite_action");
primaryActionHandler = this.props.onJoinClick;
break;
case "public":
subTitle = (0, _languageHandler._t)("room|3pid_invite_error_public_subtitle");
primaryActionLabel = (0, _languageHandler._t)("room|join_the_discussion");
primaryActionHandler = this.props.onJoinClick;
break;
default:
subTitle = errCodeMessage;
primaryActionLabel = (0, _languageHandler._t)("room|3pid_invite_error_invite_action");
primaryActionHandler = this.props.onJoinClick;
break;
}
break;
}
case MessageCase.InvitedEmailNotFoundInAccount:
{
if (roomName) {
title = (0, _languageHandler._t)("room|3pid_invite_email_not_found_account_room", {
roomName,
email: this.props.invitedEmail
});
} else {
title = (0, _languageHandler._t)("room|3pid_invite_email_not_found_account", {
email: this.props.invitedEmail
});
}
subTitle = (0, _languageHandler._t)("room|link_email_to_receive_3pid_invite", {
brand
});
primaryActionLabel = (0, _languageHandler._t)("room|join_the_discussion");
primaryActionHandler = this.props.onJoinClick;
break;
}
case MessageCase.InvitedEmailNoIdentityServer:
{
if (roomName) {
title = (0, _languageHandler._t)("room|invite_sent_to_email_room", {
roomName,
email: this.props.invitedEmail
});
} else {
title = (0, _languageHandler._t)("room|invite_sent_to_email", {
email: this.props.invitedEmail
});
}
subTitle = (0, _languageHandler._t)("room|3pid_invite_no_is_subtitle", {
brand
});
primaryActionLabel = (0, _languageHandler._t)("room|join_the_discussion");
primaryActionHandler = this.props.onJoinClick;
break;
}
case MessageCase.InvitedEmailMismatch:
{
if (roomName) {
title = (0, _languageHandler._t)("room|invite_sent_to_email_room", {
roomName,
email: this.props.invitedEmail
});
} else {
title = (0, _languageHandler._t)("room|invite_sent_to_email", {
email: this.props.invitedEmail
});
}
subTitle = (0, _languageHandler._t)("room|invite_email_mismatch_suggestion", {
brand
});
primaryActionLabel = (0, _languageHandler._t)("room|join_the_discussion");
primaryActionHandler = this.props.onJoinClick;
break;
}
case MessageCase.Invite:
{
const isDM = this.isDMInvite();
const avatar = /*#__PURE__*/_react.default.createElement(_RoomAvatar.default, {
room: this.props.room,
oobData: this.props.oobData
});
const inviteMember = this.getInviteMember();
const userName = /*#__PURE__*/_react.default.createElement("span", {
className: "mx_RoomPreviewBar_inviter"
}, inviteMember?.rawDisplayName ?? this.props.inviterName);
const inviterElement = /*#__PURE__*/_react.default.createElement(_react.default.Fragment, null, isDM ? (0, _languageHandler._t)("room|dm_invite_subtitle", {}, {
userName
}) : (0, _languageHandler._t)("room|invite_subtitle", {}, {
userName
}), inviteMember && /*#__PURE__*/_react.default.createElement(_react.default.Fragment, null, /*#__PURE__*/_react.default.createElement("br", null), /*#__PURE__*/_react.default.createElement("span", {
className: "mx_RoomPreviewBar_inviter_mxid"
}, inviteMember.userId)));
if (isDM) {
title = (0, _languageHandler._t)("room|dm_invite_title", {
user: inviteMember?.name ?? this.props.inviterName
});
primaryActionLabel = (0, _languageHandler._t)("room|dm_invite_action");
} else {
title = (0, _languageHandler._t)("room|invite_title", {
roomName
});
primaryActionLabel = (0, _languageHandler._t)("action|accept");
}
subTitle = [avatar, inviterElement];
const myUserId = _MatrixClientPeg.MatrixClientPeg.safeGet().getSafeUserId();
const member = this.props.room?.currentState.getMember(myUserId);
const memberEventContent = member?.events.member?.getContent();
if (memberEventContent?.reason) {
reasonElement = /*#__PURE__*/_react.default.createElement(_InviteReason.default, {
reason: memberEventContent.reason,
htmlReason: memberEventContent[MemberEventHtmlReasonField]
});
}
primaryActionHandler = this.props.onJoinClick;
secondaryActionLabel = (0, _languageHandler._t)("action|reject");
secondaryActionHandler = this.props.onRejectClick;
if (this.props.onRejectAndIgnoreClick) {
extraComponents.push( /*#__PURE__*/_react.default.createElement(_AccessibleButton.default, {
kind: "secondary",
onClick: this.props.onRejectAndIgnoreClick,
key: "ignore"
}, (0, _languageHandler._t)("room|invite_reject_ignore")));
}
break;
}
case MessageCase.ViewingRoom:
{
if (this.props.canPreview) {
title = (0, _languageHandler._t)("room|peek_join_prompt", {
roomName
});
} else if (roomName) {
title = (0, _languageHandler._t)("room|no_peek_join_prompt", {
roomName
});
} else {
title = (0, _languageHandler._t)("room|no_peek_no_name_join_prompt");
}
primaryActionLabel = (0, _languageHandler._t)("room|join_the_discussion");
primaryActionHandler = this.props.onJoinClick;
break;
}
case MessageCase.RoomNotFound:
{
if (roomName) {
title = (0, _languageHandler._t)("room|not_found_title_name", {
roomName
});
} else {
title = (0, _languageHandler._t)("room|not_found_title");
}
subTitle = (0, _languageHandler._t)("room|not_found_subtitle");
break;
}
case MessageCase.OtherError:
{
if (roomName) {
title = (0, _languageHandler._t)("room|inaccessible_name", {
roomName
});
} else {
title = (0, _languageHandler._t)("room|inaccessible");
}
subTitle = [(0, _languageHandler._t)("room|inaccessible_subtitle_1"), (0, _languageHandler._t)("room|inaccessible_subtitle_2", {
errcode: String(this.props.error?.errcode)
}, {
issueLink: label => /*#__PURE__*/_react.default.createElement("a", {
href: _SdkConfig.default.get().feedback.new_issue_url,
target: "_blank",
rel: "noreferrer noopener"
}, label)
})];
break;
}
case MessageCase.PromptAskToJoin:
{
if (roomName) {
title = (0, _languageHandler._t)("room|knock_prompt_name", {
roomName
});
} else {
title = (0, _languageHandler._t)("room|knock_prompt");
}
const avatar = /*#__PURE__*/_react.default.createElement(_RoomAvatar.default, {
room: this.props.room,
oobData: this.props.oobData
});
subTitle = [avatar, (0, _languageHandler._t)("room|knock_subtitle")];
reasonElement = /*#__PURE__*/_react.default.createElement(_Field.default, {
autoFocus: true,
className: "mx_RoomPreviewBar_fullWidth",
element: "textarea",
onChange: this.onChangeReason,
placeholder: (0, _languageHandler._t)("room|knock_message_field_placeholder"),
type: "text",
value: this.state.reason ?? ""
});
primaryActionHandler = () => this.props.onSubmitAskToJoin && this.props.onSubmitAskToJoin(this.state.reason);
primaryActionLabel = (0, _languageHandler._t)("room|knock_send_action");
break;
}
case MessageCase.Knocked:
{
title = (0, _languageHandler._t)("room|knock_sent");
subTitle = [/*#__PURE__*/_react.default.createElement(_react.default.Fragment, null, /*#__PURE__*/_react.default.createElement(_askToJoin.Icon, {
className: "mx_Icon mx_Icon_16 mx_RoomPreviewBar_icon"
}), (0, _languageHandler._t)("room|knock_sent_subtitle"))];
secondaryActionHandler = this.props.onCancelAskToJoin;
secondaryActionLabel = (0, _languageHandler._t)("room|knock_cancel_action");
break;
}
}
let subTitleElements;
if (subTitle) {
if (!Array.isArray(subTitle)) {
subTitle = [subTitle];
}
subTitleElements = subTitle.map((t, i) => /*#__PURE__*/_react.default.createElement("p", {
key: `subTitle${i}`
}, t));
}
let titleElement;
if (showSpinner) {
titleElement = /*#__PURE__*/_react.default.createElement("h3", {
className: "mx_RoomPreviewBar_spinnerTitle"
}, /*#__PURE__*/_react.default.createElement(_Spinner.default, null), title);
} else {
titleElement = /*#__PURE__*/_react.default.createElement("h3", null, title);
}
let primaryButton;
if (primaryActionHandler) {
primaryButton = /*#__PURE__*/_react.default.createElement(_AccessibleButton.default, {
kind: "primary",
onClick: primaryActionHandler
}, primaryActionLabel);
}
let secondaryButton;
if (secondaryActionHandler) {
secondaryButton = /*#__PURE__*/_react.default.createElement(_AccessibleButton.default, {
kind: "secondary",
onClick: secondaryActionHandler
}, secondaryActionLabel);
}
const isPanel = this.props.canPreview;
const classes = (0, _classnames.default)("mx_RoomPreviewBar", `mx_RoomPreviewBar_${messageCase}`, {
mx_RoomPreviewBar_panel: isPanel,
mx_RoomPreviewBar_dialog: !isPanel
});
// ensure correct tab order for both views
const actions = isPanel ? /*#__PURE__*/_react.default.createElement(_react.default.Fragment, null, secondaryButton, extraComponents, primaryButton) : /*#__PURE__*/_react.default.createElement(_react.default.Fragment, null, primaryButton, extraComponents, secondaryButton);
return /*#__PURE__*/_react.default.createElement("div", {
role: "complementary",
className: classes
}, /*#__PURE__*/_react.default.createElement("div", {
className: "mx_RoomPreviewBar_message"
}, titleElement, subTitleElements), reasonElement, /*#__PURE__*/_react.default.createElement("div", {
className: (0, _classnames.default)("mx_RoomPreviewBar_actions", {
mx_RoomPreviewBar_fullWidth: messageCase === MessageCase.PromptAskToJoin
})
}, actions), /*#__PURE__*/_react.default.createElement("div", {
className: "mx_RoomPreviewBar_footer"
}, footer));
}
}
exports.default = RoomPreviewBar;
(0, _defineProperty2.default)(RoomPreviewBar, "defaultProps", {
onJoinClick() {}
});
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireDefault","require","_matrix","_types","_classnames","_RoomViewLifecycle","_MatrixClientPeg","_dispatcher","_languageHandler","_SdkConfig","_IdentityAuthClient","_InviteReason","_Spinner","_AccessibleButton","_RoomAvatar","_SettingsStore","_UIFeature","_ModuleRunner","_askToJoin","_Field","MemberEventHtmlReasonField","MessageCase","RoomPreviewBar","React","Component","constructor","props","_defineProperty2","default","dis","dispatch","action","screenAfterLogin","makeScreenAfterLogin","event","setState","reason","target","value","state","busy","componentDidMount","checkInvitedEmail","componentDidUpdate","prevProps","prevState","invitedEmail","inviterName","account3pids","MatrixClientPeg","safeGet","getThreePids","accountEmails","threepids","filter","b","medium","map","address","getIdentityServerUrl","authClient","IdentityAuthClient","identityAccessToken","getAccessToken","result","lookupThreePid","UserFriendlyError","invitedEmailMxid","mxid","err","threePidFetchError","getMessageCase","isGuest","NotLoggedIn","myMember","getMyMember","previousMembership","events","member","getPrevContent","membership","isKicked","KnownMembership","Knock","RequestDenied","promptAskToJoin","PromptAskToJoin","Kicked","Ban","Banned","joining","Joining","rejecting","Rejecting","loading","Loading","knocked","Knocked","canAskToJoinAndMembershipIsLeave","OtherThreePIDError","includes","InvitedEmailNotFoundInAccount","InvitedEmailNoIdentityServer","getUserId","InvitedEmailMismatch","Invite","error","errcode","RoomNotFound","OtherError","ViewingRoom","getKickOrBanInfo","kickerUserId","getSender","kickerMember","room","currentState","getMember","undefined","memberName","name","getContent","joinRule","getStateEvents","EventType","RoomJoinRules","join_rule","getSafeUserId","getInviteMember","myUserId","inviteEvent","inviterUserId","isDMInvite","memberContent","is_direct","screen","params","email","signurl","signUrl","room_name","oobData","room_avatar_url","avatarUrl","inviter_name","render","brand","SdkConfig","get","roomName","roomAlias","isSpace","isSpaceRoom","roomType","RoomType","Space","showSpinner","title","subTitle","reasonElement","primaryActionHandler","primaryActionLabel","secondaryActionHandler","secondaryActionLabel","footer","extraComponents","messageCase","_t","opts","canJoin","roomId","ModuleRunner","instance","invoke","RoomViewLifecycle","PreviewRoomNotLoggedIn","JoinFromRoomPreview","SettingsStore","getValue","UIFeature","Registration","onRegisterClick","onLoginClick","previewLoading","createElement","w","h","onForgetClick","JoinRule","onJoinClick","errCodeMessage","isDM","avatar","inviteMember","userName","className","rawDisplayName","inviterElement","Fragment","userId","user","memberEventContent","htmlReason","onRejectClick","onRejectAndIgnoreClick","push","kind","onClick","key","canPreview","String","issueLink","label","href","feedback","new_issue_url","rel","autoFocus","element","onChange","onChangeReason","placeholder","type","onSubmitAskToJoin","Icon","onCancelAskToJoin","subTitleElements","Array","isArray","t","i","titleElement","primaryButton","secondaryButton","isPanel","classes","classNames","mx_RoomPreviewBar_panel","mx_RoomPreviewBar_dialog","actions","role","mx_RoomPreviewBar_fullWidth","exports"],"sources":["../../../../src/components/views/rooms/RoomPreviewBar.tsx"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2015-2021 The Matrix.org Foundation C.I.C.\n\nSPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only\nPlease see LICENSE files in the repository root for full details.\n*/\n\nimport React, { ChangeEvent, ReactNode } from \"react\";\nimport { Room, RoomMember, EventType, RoomType, JoinRule, MatrixError } from \"matrix-js-sdk/src/matrix\";\nimport { KnownMembership, RoomJoinRulesEventContent } from \"matrix-js-sdk/src/types\";\nimport classNames from \"classnames\";\nimport { RoomPreviewOpts, RoomViewLifecycle } from \"@matrix-org/react-sdk-module-api/lib/lifecycles/RoomViewLifecycle\";\n\nimport { MatrixClientPeg } from \"../../../MatrixClientPeg\";\nimport dis from \"../../../dispatcher/dispatcher\";\nimport { _t, UserFriendlyError } from \"../../../languageHandler\";\nimport SdkConfig from \"../../../SdkConfig\";\nimport IdentityAuthClient from \"../../../IdentityAuthClient\";\nimport InviteReason from \"../elements/InviteReason\";\nimport { IOOBData } from \"../../../stores/ThreepidInviteStore\";\nimport Spinner from \"../elements/Spinner\";\nimport AccessibleButton from \"../elements/AccessibleButton\";\nimport RoomAvatar from \"../avatars/RoomAvatar\";\nimport SettingsStore from \"../../../settings/SettingsStore\";\nimport { UIFeature } from \"../../../settings/UIFeature\";\nimport { ModuleRunner } from \"../../../modules/ModuleRunner\";\nimport { Icon as AskToJoinIcon } from \"../../../../res/img/element-icons/ask-to-join.svg\";\nimport Field from \"../elements/Field\";\n\nconst MemberEventHtmlReasonField = \"io.element.html_reason\";\n\nenum MessageCase {\n    NotLoggedIn = \"NotLoggedIn\",\n    Joining = \"Joining\",\n    Loading = \"Loading\",\n    Rejecting = \"Rejecting\",\n    Kicked = \"Kicked\",\n    Banned = \"Banned\",\n    OtherThreePIDError = \"OtherThreePIDError\",\n    InvitedEmailNotFoundInAccount = \"InvitedEmailNotFoundInAccount\",\n    InvitedEmailNoIdentityServer = \"InvitedEmailNoIdentityServer\",\n    InvitedEmailMismatch = \"InvitedEmailMismatch\",\n    Invite = \"Invite\",\n    ViewingRoom = \"ViewingRoom\",\n    RoomNotFound = \"RoomNotFound\",\n    OtherError = \"OtherError\",\n    PromptAskToJoin = \"PromptAskToJoin\",\n    Knocked = \"Knocked\",\n    RequestDenied = \"requestDenied\",\n}\n\ninterface IProps {\n    // if inviterName is specified, the preview bar will shown an invite to the room.\n    // You should also specify onRejectClick if specifying inviterName\n    inviterName?: string;\n\n    // If invited by 3rd party invite, the email address the invite was sent to\n    invitedEmail?: string;\n\n    // For third party invites, information passed about the room out-of-band\n    oobData?: IOOBData;\n\n    // For third party invites, a URL for a 3pid invite signing service\n    signUrl?: string;\n\n    // A standard client/server API error object. If supplied, indicates that the\n    // caller was unable to fetch details about the room for the given reason.\n    error?: MatrixError;\n\n    canPreview?: boolean;\n    previewLoading?: boolean;\n\n    // The id of the room to be previewed, if it is known.\n    // (It may be unknown if we are waiting for an alias to be resolved.)\n    roomId?: string;\n\n    // A `Room` object for the room to be previewed, if we have one.\n    room?: Room;\n\n    loading?: boolean;\n    joining?: boolean;\n    rejecting?: boolean;\n    // The alias that was used to access this room, if appropriate\n    // If given, this will be how the room is referred to (eg.\n    // in error messages).\n    roomAlias?: string;\n\n    onJoinClick?(): void;\n    onRejectClick?(): void;\n    onRejectAndIgnoreClick?(): void;\n    onForgetClick?(): void;\n\n    canAskToJoinAndMembershipIsLeave?: boolean;\n    promptAskToJoin?: boolean;\n    knocked?: boolean;\n    onSubmitAskToJoin?(reason?: string): void;\n    onCancelAskToJoin?(): void;\n}\n\ninterface IState {\n    busy: boolean;\n    accountEmails?: string[];\n    invitedEmailMxid?: string;\n    threePidFetchError?: MatrixError;\n    reason?: string;\n}\n\nexport default class RoomPreviewBar extends React.Component<IProps, IState> {\n    public static defaultProps = {\n        onJoinClick() {},\n    };\n\n    public constructor(props: IProps) {\n        super(props);\n\n        this.state = {\n            busy: false,\n        };\n    }\n\n    public componentDidMount(): void {\n        this.checkInvitedEmail();\n    }\n\n    public componentDidUpdate(prevProps: IProps, prevState: IState): void {\n        if (this.props.invitedEmail !== prevProps.invitedEmail || this.props.inviterName !== prevProps.inviterName) {\n            this.checkInvitedEmail();\n        }\n    }\n\n    private async checkInvitedEmail(): Promise<void> {\n        // If this is an invite and we've been told what email address was\n        // invited, fetch the user's account emails and discovery bindings so we\n        // can check them against the email that was invited.\n        if (this.props.inviterName && this.props.invitedEmail) {\n            this.setState({ busy: true });\n            try {\n                // Gather the account 3PIDs\n                const account3pids = await MatrixClientPeg.safeGet().getThreePids();\n                this.setState({\n                    accountEmails: account3pids.threepids.filter((b) => b.medium === \"email\").map((b) => b.address),\n                });\n                // If we have an IS connected, use that to lookup the email and\n                // check the bound MXID.\n                if (!MatrixClientPeg.safeGet().getIdentityServerUrl()) {\n                    this.setState({ busy: false });\n                    return;\n                }\n                const authClient = new IdentityAuthClient();\n                const identityAccessToken = await authClient.getAccessToken();\n                const result = await MatrixClientPeg.safeGet().lookupThreePid(\n                    \"email\",\n                    this.props.invitedEmail,\n                    identityAccessToken!,\n                );\n                if (!(\"mxid\" in result)) {\n                    throw new UserFriendlyError(\"room|error_3pid_invite_email_lookup\");\n                }\n                this.setState({ invitedEmailMxid: result.mxid });\n            } catch (err) {\n                this.setState({ threePidFetchError: err as MatrixError });\n            }\n            this.setState({ busy: false });\n        }\n    }\n\n    private getMessageCase(): MessageCase {\n        const isGuest = MatrixClientPeg.safeGet().isGuest();\n\n        if (isGuest) {\n            return MessageCase.NotLoggedIn;\n        }\n\n        const myMember = this.getMyMember();\n\n        if (myMember) {\n            const previousMembership = myMember.events.member?.getPrevContent().membership;\n            if (myMember.isKicked()) {\n                if (previousMembership === KnownMembership.Knock) {\n                    return MessageCase.RequestDenied;\n                } else if (this.props.promptAskToJoin) {\n                    return MessageCase.PromptAskToJoin;\n                }\n                return MessageCase.Kicked;\n            } else if (myMember.membership === KnownMembership.Ban) {\n                return MessageCase.Banned;\n            }\n        }\n\n        if (this.props.joining) {\n            return MessageCase.Joining;\n        } else if (this.props.rejecting) {\n            return MessageCase.Rejecting;\n        } else if (this.props.loading || this.state.busy) {\n            return MessageCase.Loading;\n        } else if (this.props.knocked) {\n            return MessageCase.Knocked;\n        } else if (this.props.canAskToJoinAndMembershipIsLeave || this.props.promptAskToJoin) {\n            return MessageCase.PromptAskToJoin;\n        }\n\n        if (this.props.inviterName) {\n            if (this.props.invitedEmail) {\n                if (this.state.threePidFetchError) {\n                    return MessageCase.OtherThreePIDError;\n                } else if (this.state.accountEmails && !this.state.accountEmails.includes(this.props.invitedEmail)) {\n                    return MessageCase.InvitedEmailNotFoundInAccount;\n                } else if (!MatrixClientPeg.safeGet().getIdentityServerUrl()) {\n                    return MessageCase.InvitedEmailNoIdentityServer;\n                } else if (this.state.invitedEmailMxid != MatrixClientPeg.safeGet().getUserId()) {\n                    return MessageCase.InvitedEmailMismatch;\n                }\n            }\n            return MessageCase.Invite;\n        } else if (this.props.error) {\n            if ((this.props.error as MatrixError).errcode == \"M_NOT_FOUND\") {\n                return MessageCase.RoomNotFound;\n            } else {\n                return MessageCase.OtherError;\n            }\n        } else {\n            return MessageCase.ViewingRoom;\n        }\n    }\n\n    private getKickOrBanInfo(): { memberName?: string; reason?: string } {\n        const myMember = this.getMyMember();\n        if (!myMember) {\n            return {};\n        }\n\n        const kickerUserId = myMember.events.member?.getSender();\n        const kickerMember = kickerUserId ? this.props.room?.currentState.getMember(kickerUserId) : undefined;\n        const memberName = kickerMember?.name ?? kickerUserId;\n        const reason = myMember.events.member?.getContent().reason;\n        return { memberName, reason };\n    }\n\n    private joinRule(): JoinRule | null {\n        return (\n            this.props.room?.currentState\n                .getStateEvents(EventType.RoomJoinRules, \"\")\n                ?.getContent<RoomJoinRulesEventContent>().join_rule ?? null\n        );\n    }\n\n    private getMyMember(): RoomMember | null {\n        return this.props.room?.getMember(MatrixClientPeg.safeGet().getSafeUserId()) ?? null;\n    }\n\n    private getInviteMember(): RoomMember | null {\n        const { room } = this.props;\n        if (!room) {\n            return null;\n        }\n        const myUserId = MatrixClientPeg.safeGet().getSafeUserId();\n        const inviteEvent = room.currentState.getMember(myUserId);\n        if (!inviteEvent) {\n            return null;\n        }\n        const inviterUserId = inviteEvent.events.member?.getSender();\n        return inviterUserId ? room.currentState.getMember(inviterUserId) : null;\n    }\n\n    private isDMInvite(): boolean {\n        const myMember = this.getMyMember();\n        if (!myMember) {\n            return false;\n        }\n        const memberContent = myMember.events.member?.getContent();\n        return memberContent?.membership === KnownMembership.Invite && memberContent.is_direct;\n    }\n\n    private makeScreenAfterLogin(): { screen: string; params: Record<string, any> } {\n        return {\n            screen: \"room\",\n            params: {\n                email: this.props.invitedEmail,\n                signurl: this.props.signUrl,\n                room_name: this.props.oobData?.name ?? null,\n                room_avatar_url: this.props.oobData?.avatarUrl ?? null,\n                inviter_name: this.props.oobData?.inviterName ?? null,\n            },\n        };\n    }\n\n    private onLoginClick = (): void => {\n        dis.dispatch({ action: \"start_login\", screenAfterLogin: this.makeScreenAfterLogin() });\n    };\n\n    private onRegisterClick = (): void => {\n        dis.dispatch({ action: \"start_registration\", screenAfterLogin: this.makeScreenAfterLogin() });\n    };\n\n    private onChangeReason = (event: ChangeEvent<HTMLTextAreaElement>): void => {\n        this.setState({ reason: event.target.value });\n    };\n\n    public render(): React.ReactNode {\n        const brand = SdkConfig.get().brand;\n        const roomName = this.props.room?.name ?? this.props.roomAlias ?? \"\";\n        const isSpace = this.props.room?.isSpaceRoom() ?? this.props.oobData?.roomType === RoomType.Space;\n\n        let showSpinner = false;\n        let title: string | undefined;\n        let subTitle: string | ReactNode[] | undefined;\n        let reasonElement: JSX.Element | undefined;\n        let primaryActionHandler: (() => void) | undefined;\n        let primaryActionLabel: string | undefined;\n        let secondaryActionHandler: (() => void) | undefined;\n        let secondaryActionLabel: string | undefined;\n        let footer: JSX.Element | undefined;\n        const extraComponents: JSX.Element[] = [];\n\n        const messageCase = this.getMessageCase();\n        switch (messageCase) {\n            case MessageCase.Joining: {\n                if (this.props.oobData?.roomType || isSpace) {\n                    title = isSpace ? _t(\"room|joining_space\") : _t(\"room|joining_room\");\n                } else {\n                    title = _t(\"room|joining\");\n                }\n\n                showSpinner = true;\n                break;\n            }\n            case MessageCase.Loading: {\n                title = _t(\"common|loading\");\n                showSpinner = true;\n                break;\n            }\n            case MessageCase.Rejecting: {\n                title = _t(\"room|rejecting\");\n                showSpinner = true;\n                break;\n            }\n            case MessageCase.NotLoggedIn: {\n                const opts: RoomPreviewOpts = { canJoin: false };\n                if (this.props.roomId) {\n                    ModuleRunner.instance.invoke(RoomViewLifecycle.PreviewRoomNotLoggedIn, opts, this.props.roomId);\n                }\n                if (opts.canJoin) {\n                    title = _t(\"room|join_title\");\n                    primaryActionLabel = _t(\"action|join\");\n                    primaryActionHandler = () => {\n                        Mo