matrix-react-sdk
Version:
SDK for matrix.org using React
211 lines (207 loc) • 35.8 kB
JavaScript
"use strict";
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = void 0;
exports.getUnsentMessages = getUnsentMessages;
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _react = _interopRequireDefault(require("react"));
var _matrix = require("matrix-js-sdk/src/matrix");
var _warningTriangle = require("../../../res/img/feather-customised/warning-triangle.svg");
var _languageHandler = require("../../languageHandler");
var _Resend = _interopRequireDefault(require("../../Resend"));
var _dispatcher = _interopRequireDefault(require("../../dispatcher/dispatcher"));
var _ErrorUtils = require("../../utils/ErrorUtils");
var _actions = require("../../dispatcher/actions");
var _StaticNotificationState = require("../../stores/notifications/StaticNotificationState");
var _AccessibleButton = _interopRequireDefault(require("../views/elements/AccessibleButton"));
var _InlineSpinner = _interopRequireDefault(require("../views/elements/InlineSpinner"));
var _MatrixClientContext = _interopRequireDefault(require("../../contexts/MatrixClientContext"));
var _RoomStatusBarUnsentMessages = require("./RoomStatusBarUnsentMessages");
var _ExternalLink = _interopRequireDefault(require("../views/elements/ExternalLink"));
/*
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 STATUS_BAR_HIDDEN = 0;
const STATUS_BAR_EXPANDED = 1;
const STATUS_BAR_EXPANDED_LARGE = 2;
function getUnsentMessages(room, threadId) {
if (!room) {
return [];
}
return room.getPendingEvents().filter(function (ev) {
const isNotSent = ev.status === _matrix.EventStatus.NOT_SENT;
const belongsToTheThread = threadId === ev.threadRootId;
return isNotSent && (!threadId || belongsToTheThread);
});
}
class RoomStatusBar extends _react.default.PureComponent {
constructor(props, context) {
super(props, context);
(0, _defineProperty2.default)(this, "unmounted", false);
(0, _defineProperty2.default)(this, "onSyncStateChange", (state, prevState, data) => {
if (state === "SYNCING" && prevState === "SYNCING") {
return;
}
if (this.unmounted) return;
this.setState({
syncState: state,
syncStateData: data ?? null
});
});
(0, _defineProperty2.default)(this, "onResendAllClick", () => {
_Resend.default.resendUnsentEvents(this.props.room).then(() => {
this.setState({
isResending: false
});
});
this.setState({
isResending: true
});
_dispatcher.default.fire(_actions.Action.FocusSendMessageComposer);
});
(0, _defineProperty2.default)(this, "onCancelAllClick", () => {
_Resend.default.cancelUnsentEvents(this.props.room);
_dispatcher.default.fire(_actions.Action.FocusSendMessageComposer);
});
(0, _defineProperty2.default)(this, "onRoomLocalEchoUpdated", (ev, room) => {
if (room.roomId !== this.props.room.roomId) return;
const messages = getUnsentMessages(this.props.room);
this.setState({
unsentMessages: messages,
isResending: messages.length > 0 && this.state.isResending
});
});
this.state = {
syncState: this.context.getSyncState(),
syncStateData: this.context.getSyncStateData(),
unsentMessages: getUnsentMessages(this.props.room),
isResending: false
};
}
componentDidMount() {
const client = this.context;
client.on(_matrix.ClientEvent.Sync, this.onSyncStateChange);
client.on(_matrix.RoomEvent.LocalEchoUpdated, this.onRoomLocalEchoUpdated);
this.checkSize();
}
componentDidUpdate() {
this.checkSize();
}
componentWillUnmount() {
this.unmounted = true;
// we may have entirely lost our client as we're logging out before clicking login on the guest bar...
const client = this.context;
if (client) {
client.removeListener(_matrix.ClientEvent.Sync, this.onSyncStateChange);
client.removeListener(_matrix.RoomEvent.LocalEchoUpdated, this.onRoomLocalEchoUpdated);
}
}
// Check whether current size is greater than 0, if yes call props.onVisible
checkSize() {
if (this.getSize()) {
if (this.props.onVisible) this.props.onVisible();
} else {
if (this.props.onHidden) this.props.onHidden();
}
}
// We don't need the actual height - just whether it is likely to have
// changed - so we use '0' to indicate normal size, and other values to
// indicate other sizes.
getSize() {
if (this.shouldShowConnectionError()) {
return STATUS_BAR_EXPANDED;
} else if (this.state.unsentMessages.length > 0 || this.state.isResending) {
return STATUS_BAR_EXPANDED_LARGE;
}
return STATUS_BAR_HIDDEN;
}
shouldShowConnectionError() {
// no conn bar trumps the "some not sent" msg since you can't resend without
// a connection!
// There's one situation in which we don't show this 'no connection' bar, and that's
// if it's a resource limit exceeded error: those are shown in the top bar.
const errorIsMauError = Boolean(this.state.syncStateData && this.state.syncStateData.error && this.state.syncStateData.error.name === "M_RESOURCE_LIMIT_EXCEEDED");
return this.state.syncState === "ERROR" && !errorIsMauError;
}
getUnsentMessageContent() {
const unsentMessages = this.state.unsentMessages;
let title;
let consentError = null;
let resourceLimitError = null;
for (const m of unsentMessages) {
if (m.error && m.error.errcode === "M_CONSENT_NOT_GIVEN") {
consentError = m.error;
break;
} else if (m.error && m.error.errcode === "M_RESOURCE_LIMIT_EXCEEDED") {
resourceLimitError = m.error;
break;
}
}
if (consentError) {
title = (0, _languageHandler._t)("room|status_bar|requires_consent_agreement", {}, {
consentLink: sub => /*#__PURE__*/_react.default.createElement(_ExternalLink.default, {
href: consentError.data?.consent_uri,
target: "_blank",
rel: "noreferrer noopener"
}, sub)
});
} else if (resourceLimitError) {
title = (0, _ErrorUtils.messageForResourceLimitError)(resourceLimitError.data.limit_type, resourceLimitError.data.admin_contact, {
"monthly_active_user": (0, _languageHandler._td)("room|status_bar|monthly_user_limit_reached"),
"hs_disabled": (0, _languageHandler._td)("room|status_bar|homeserver_blocked"),
"": (0, _languageHandler._td)("room|status_bar|exceeded_resource_limit")
});
} else {
title = (0, _languageHandler._t)("room|status_bar|some_messages_not_sent");
}
let buttonRow = /*#__PURE__*/_react.default.createElement(_react.default.Fragment, null, /*#__PURE__*/_react.default.createElement(_AccessibleButton.default, {
onClick: this.onCancelAllClick,
className: "mx_RoomStatusBar_unsentCancelAllBtn"
}, (0, _languageHandler._t)("room|status_bar|delete_all")), /*#__PURE__*/_react.default.createElement(_AccessibleButton.default, {
onClick: this.onResendAllClick,
className: "mx_RoomStatusBar_unsentRetry"
}, (0, _languageHandler._t)("room|status_bar|retry_all")));
if (this.state.isResending) {
buttonRow = /*#__PURE__*/_react.default.createElement(_react.default.Fragment, null, /*#__PURE__*/_react.default.createElement(_InlineSpinner.default, {
w: 20,
h: 20
}), /*#__PURE__*/_react.default.createElement("span", null, (0, _languageHandler._t)("forward|sending")));
}
return /*#__PURE__*/_react.default.createElement(_RoomStatusBarUnsentMessages.RoomStatusBarUnsentMessages, {
title: title,
description: (0, _languageHandler._t)("room|status_bar|select_messages_to_retry"),
notificationState: _StaticNotificationState.StaticNotificationState.RED_EXCLAMATION,
buttons: buttonRow
});
}
render() {
if (this.shouldShowConnectionError()) {
return /*#__PURE__*/_react.default.createElement("div", {
className: "mx_RoomStatusBar"
}, /*#__PURE__*/_react.default.createElement("div", {
role: "alert"
}, /*#__PURE__*/_react.default.createElement("div", {
className: "mx_RoomStatusBar_connectionLostBar"
}, /*#__PURE__*/_react.default.createElement(_warningTriangle.Icon, {
width: "24px",
height: "24px"
}), /*#__PURE__*/_react.default.createElement("div", null, /*#__PURE__*/_react.default.createElement("div", {
className: "mx_RoomStatusBar_connectionLostBar_title"
}, (0, _languageHandler._t)("room|status_bar|server_connectivity_lost_title")), /*#__PURE__*/_react.default.createElement("div", {
className: "mx_RoomStatusBar_connectionLostBar_desc"
}, (0, _languageHandler._t)("room|status_bar|server_connectivity_lost_description"))))));
}
if (this.state.unsentMessages.length > 0 || this.state.isResending) {
return this.getUnsentMessageContent();
}
return null;
}
}
exports.default = RoomStatusBar;
(0, _defineProperty2.default)(RoomStatusBar, "contextType", _MatrixClientContext.default);
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireDefault","require","_matrix","_warningTriangle","_languageHandler","_Resend","_dispatcher","_ErrorUtils","_actions","_StaticNotificationState","_AccessibleButton","_InlineSpinner","_MatrixClientContext","_RoomStatusBarUnsentMessages","_ExternalLink","STATUS_BAR_HIDDEN","STATUS_BAR_EXPANDED","STATUS_BAR_EXPANDED_LARGE","getUnsentMessages","room","threadId","getPendingEvents","filter","ev","isNotSent","status","EventStatus","NOT_SENT","belongsToTheThread","threadRootId","RoomStatusBar","React","PureComponent","constructor","props","context","_defineProperty2","default","state","prevState","data","unmounted","setState","syncState","syncStateData","Resend","resendUnsentEvents","then","isResending","dis","fire","Action","FocusSendMessageComposer","cancelUnsentEvents","roomId","messages","unsentMessages","length","getSyncState","getSyncStateData","componentDidMount","client","on","ClientEvent","Sync","onSyncStateChange","RoomEvent","LocalEchoUpdated","onRoomLocalEchoUpdated","checkSize","componentDidUpdate","componentWillUnmount","removeListener","getSize","onVisible","onHidden","shouldShowConnectionError","errorIsMauError","Boolean","error","name","getUnsentMessageContent","title","consentError","resourceLimitError","m","errcode","_t","consentLink","sub","createElement","href","consent_uri","target","rel","messageForResourceLimitError","limit_type","admin_contact","_td","buttonRow","Fragment","onClick","onCancelAllClick","className","onResendAllClick","w","h","RoomStatusBarUnsentMessages","description","notificationState","StaticNotificationState","RED_EXCLAMATION","buttons","render","role","Icon","width","height","exports","MatrixClientContext"],"sources":["../../../src/components/structures/RoomStatusBar.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, { ReactNode } from \"react\";\nimport {\n    ClientEvent,\n    EventStatus,\n    MatrixError,\n    MatrixEvent,\n    Room,\n    RoomEvent,\n    SyncState,\n    SyncStateData,\n} from \"matrix-js-sdk/src/matrix\";\n\nimport { Icon as WarningIcon } from \"../../../res/img/feather-customised/warning-triangle.svg\";\nimport { _t, _td } from \"../../languageHandler\";\nimport Resend from \"../../Resend\";\nimport dis from \"../../dispatcher/dispatcher\";\nimport { messageForResourceLimitError } from \"../../utils/ErrorUtils\";\nimport { Action } from \"../../dispatcher/actions\";\nimport { StaticNotificationState } from \"../../stores/notifications/StaticNotificationState\";\nimport AccessibleButton from \"../views/elements/AccessibleButton\";\nimport InlineSpinner from \"../views/elements/InlineSpinner\";\nimport MatrixClientContext from \"../../contexts/MatrixClientContext\";\nimport { RoomStatusBarUnsentMessages } from \"./RoomStatusBarUnsentMessages\";\nimport ExternalLink from \"../views/elements/ExternalLink\";\n\nconst STATUS_BAR_HIDDEN = 0;\nconst STATUS_BAR_EXPANDED = 1;\nconst STATUS_BAR_EXPANDED_LARGE = 2;\n\nexport function getUnsentMessages(room: Room, threadId?: string): MatrixEvent[] {\n    if (!room) {\n        return [];\n    }\n    return room.getPendingEvents().filter(function (ev) {\n        const isNotSent = ev.status === EventStatus.NOT_SENT;\n        const belongsToTheThread = threadId === ev.threadRootId;\n        return isNotSent && (!threadId || belongsToTheThread);\n    });\n}\n\ninterface IProps {\n    // the room this statusbar is representing.\n    room: Room;\n\n    // true if the room is being peeked at. This affects components that shouldn't\n    // logically be shown when peeking, such as a prompt to invite people to a room.\n    isPeeking?: boolean;\n    // callback for when the user clicks on the 'resend all' button in the\n    // 'unsent messages' bar\n    onResendAllClick?: () => void;\n\n    // callback for when the user clicks on the 'cancel all' button in the\n    // 'unsent messages' bar\n    onCancelAllClick?: () => void;\n\n    // callback for when the user clicks on the 'invite others' button in the\n    // 'you are alone' bar\n    onInviteClick?: () => void;\n\n    // callback for when we do something that changes the size of the\n    // status bar. This is used to trigger a re-layout in the parent\n    // component.\n    onResize?: () => void;\n\n    // callback for when the status bar can be hidden from view, as it is\n    // not displaying anything\n    onHidden?: () => void;\n\n    // callback for when the status bar is displaying something and should\n    // be visible\n    onVisible?: () => void;\n}\n\ninterface IState {\n    syncState: SyncState | null;\n    syncStateData: SyncStateData | null;\n    unsentMessages: MatrixEvent[];\n    isResending: boolean;\n}\n\nexport default class RoomStatusBar extends React.PureComponent<IProps, IState> {\n    private unmounted = false;\n    public static contextType = MatrixClientContext;\n    public declare context: React.ContextType<typeof MatrixClientContext>;\n\n    public constructor(props: IProps, context: React.ContextType<typeof MatrixClientContext>) {\n        super(props, context);\n\n        this.state = {\n            syncState: this.context.getSyncState(),\n            syncStateData: this.context.getSyncStateData(),\n            unsentMessages: getUnsentMessages(this.props.room),\n            isResending: false,\n        };\n    }\n\n    public componentDidMount(): void {\n        const client = this.context;\n        client.on(ClientEvent.Sync, this.onSyncStateChange);\n        client.on(RoomEvent.LocalEchoUpdated, this.onRoomLocalEchoUpdated);\n\n        this.checkSize();\n    }\n\n    public componentDidUpdate(): void {\n        this.checkSize();\n    }\n\n    public componentWillUnmount(): void {\n        this.unmounted = true;\n        // we may have entirely lost our client as we're logging out before clicking login on the guest bar...\n        const client = this.context;\n        if (client) {\n            client.removeListener(ClientEvent.Sync, this.onSyncStateChange);\n            client.removeListener(RoomEvent.LocalEchoUpdated, this.onRoomLocalEchoUpdated);\n        }\n    }\n\n    private onSyncStateChange = (state: SyncState, prevState: SyncState | null, data?: SyncStateData): void => {\n        if (state === \"SYNCING\" && prevState === \"SYNCING\") {\n            return;\n        }\n        if (this.unmounted) return;\n        this.setState({\n            syncState: state,\n            syncStateData: data ?? null,\n        });\n    };\n\n    private onResendAllClick = (): void => {\n        Resend.resendUnsentEvents(this.props.room).then(() => {\n            this.setState({ isResending: false });\n        });\n        this.setState({ isResending: true });\n        dis.fire(Action.FocusSendMessageComposer);\n    };\n\n    private onCancelAllClick = (): void => {\n        Resend.cancelUnsentEvents(this.props.room);\n        dis.fire(Action.FocusSendMessageComposer);\n    };\n\n    private onRoomLocalEchoUpdated = (ev: MatrixEvent, room: Room): void => {\n        if (room.roomId !== this.props.room.roomId) return;\n        const messages = getUnsentMessages(this.props.room);\n        this.setState({\n            unsentMessages: messages,\n            isResending: messages.length > 0 && this.state.isResending,\n        });\n    };\n\n    // Check whether current size is greater than 0, if yes call props.onVisible\n    private checkSize(): void {\n        if (this.getSize()) {\n            if (this.props.onVisible) this.props.onVisible();\n        } else {\n            if (this.props.onHidden) this.props.onHidden();\n        }\n    }\n\n    // We don't need the actual height - just whether it is likely to have\n    // changed - so we use '0' to indicate normal size, and other values to\n    // indicate other sizes.\n    private getSize(): number {\n        if (this.shouldShowConnectionError()) {\n            return STATUS_BAR_EXPANDED;\n        } else if (this.state.unsentMessages.length > 0 || this.state.isResending) {\n            return STATUS_BAR_EXPANDED_LARGE;\n        }\n        return STATUS_BAR_HIDDEN;\n    }\n\n    private shouldShowConnectionError(): boolean {\n        // no conn bar trumps the \"some not sent\" msg since you can't resend without\n        // a connection!\n        // There's one situation in which we don't show this 'no connection' bar, and that's\n        // if it's a resource limit exceeded error: those are shown in the top bar.\n        const errorIsMauError = Boolean(\n            this.state.syncStateData &&\n                this.state.syncStateData.error &&\n                this.state.syncStateData.error.name === \"M_RESOURCE_LIMIT_EXCEEDED\",\n        );\n        return this.state.syncState === \"ERROR\" && !errorIsMauError;\n    }\n\n    private getUnsentMessageContent(): JSX.Element {\n        const unsentMessages = this.state.unsentMessages;\n\n        let title: ReactNode;\n\n        let consentError: MatrixError | null = null;\n        let resourceLimitError: MatrixError | null = null;\n        for (const m of unsentMessages) {\n            if (m.error && m.error.errcode === \"M_CONSENT_NOT_GIVEN\") {\n                consentError = m.error;\n                break;\n            } else if (m.error && m.error.errcode === \"M_RESOURCE_LIMIT_EXCEEDED\") {\n                resourceLimitError = m.error;\n                break;\n            }\n        }\n        if (consentError) {\n            title = _t(\n                \"room|status_bar|requires_consent_agreement\",\n                {},\n                {\n                    consentLink: (sub) => (\n                        <ExternalLink href={consentError!.data?.consent_uri} target=\"_blank\" rel=\"noreferrer noopener\">\n                            {sub}\n                        </ExternalLink>\n                    ),\n                },\n            );\n        } else if (resourceLimitError) {\n            title = messageForResourceLimitError(\n                resourceLimitError.data.limit_type,\n                resourceLimitError.data.admin_contact,\n                {\n                    \"monthly_active_user\": _td(\"room|status_bar|monthly_user_limit_reached\"),\n                    \"hs_disabled\": _td(\"room|status_bar|homeserver_blocked\"),\n                    \"\": _td(\"room|status_bar|exceeded_resource_limit\"),\n                },\n            );\n        } else {\n            title = _t(\"room|status_bar|some_messages_not_sent\");\n        }\n\n        let buttonRow = (\n            <>\n                <AccessibleButton onClick={this.onCancelAllClick} className=\"mx_RoomStatusBar_unsentCancelAllBtn\">\n                    {_t(\"room|status_bar|delete_all\")}\n                </AccessibleButton>\n                <AccessibleButton onClick={this.onResendAllClick} className=\"mx_RoomStatusBar_unsentRetry\">\n                    {_t(\"room|status_bar|retry_all\")}\n                </AccessibleButton>\n            </>\n        );\n        if (this.state.isResending) {\n            buttonRow = (\n                <>\n                    <InlineSpinner w={20} h={20} />\n                    {/* span for css */}\n                    <span>{_t(\"forward|sending\")}</span>\n                </>\n            );\n        }\n\n        return (\n            <RoomStatusBarUnsentMessages\n                title={title}\n                description={_t(\"room|status_bar|select_messages_to_retry\")}\n                notificationState={StaticNotificationState.RED_EXCLAMATION}\n                buttons={buttonRow}\n            />\n        );\n    }\n\n    public render(): React.ReactNode {\n        if (this.shouldShowConnectionError()) {\n            return (\n                <div className=\"mx_RoomStatusBar\">\n                    <div role=\"alert\">\n                        <div className=\"mx_RoomStatusBar_connectionLostBar\">\n                            <WarningIcon width=\"24px\" height=\"24px\" />\n                            <div>\n                                <div className=\"mx_RoomStatusBar_connectionLostBar_title\">\n                                    {_t(\"room|status_bar|server_connectivity_lost_title\")}\n                                </div>\n                                <div className=\"mx_RoomStatusBar_connectionLostBar_desc\">\n                                    {_t(\"room|status_bar|server_connectivity_lost_description\")}\n                                </div>\n                            </div>\n                        </div>\n                    </div>\n                </div>\n            );\n        }\n\n        if (this.state.unsentMessages.length > 0 || this.state.isResending) {\n            return this.getUnsentMessageContent();\n        }\n\n        return null;\n    }\n}\n"],"mappings":"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","ignoreList":[]}