UNPKG

matrix-react-sdk

Version:
663 lines (659 loc) 98.2 kB
"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