2016-02-08 18:04:54 +00:00
|
|
|
/*
|
2021-04-21 19:49:58 +00:00
|
|
|
Copyright 2015-2021 The Matrix.org Foundation C.I.C.
|
2016-02-08 18:04:54 +00:00
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2023-02-13 11:39:16 +00:00
|
|
|
import React, { ReactNode } from "react";
|
2021-10-22 22:23:32 +00:00
|
|
|
import { EventStatus, MatrixEvent } from "matrix-js-sdk/src/models/event";
|
2021-12-16 09:57:10 +00:00
|
|
|
import { SyncState, ISyncStateData } from "matrix-js-sdk/src/sync";
|
2021-10-22 22:23:32 +00:00
|
|
|
import { Room } from "matrix-js-sdk/src/models/room";
|
2023-02-13 11:39:16 +00:00
|
|
|
import { MatrixError } from "matrix-js-sdk/src/matrix";
|
2021-10-22 22:23:32 +00:00
|
|
|
|
2023-05-24 02:37:10 +00:00
|
|
|
import { Icon as WarningIcon } from "../../../res/img/feather-customised/warning-triangle.svg";
|
2018-08-16 12:31:17 +00:00
|
|
|
import { _t, _td } from "../../languageHandler";
|
2017-11-09 15:58:15 +00:00
|
|
|
import Resend from "../../Resend";
|
2020-05-14 02:41:41 +00:00
|
|
|
import dis from "../../dispatcher/dispatcher";
|
2021-06-29 12:11:58 +00:00
|
|
|
import { messageForResourceLimitError } from "../../utils/ErrorUtils";
|
|
|
|
import { Action } from "../../dispatcher/actions";
|
|
|
|
import { StaticNotificationState } from "../../stores/notifications/StaticNotificationState";
|
2021-04-21 19:49:58 +00:00
|
|
|
import AccessibleButton from "../views/elements/AccessibleButton";
|
|
|
|
import InlineSpinner from "../views/elements/InlineSpinner";
|
2021-09-13 16:28:52 +00:00
|
|
|
import MatrixClientContext from "../../contexts/MatrixClientContext";
|
2022-07-20 12:41:43 +00:00
|
|
|
import { RoomStatusBarUnsentMessages } from "./RoomStatusBarUnsentMessages";
|
2023-05-03 21:26:26 +00:00
|
|
|
import ExternalLink from "../views/elements/ExternalLink";
|
2017-01-20 16:51:35 +00:00
|
|
|
|
2017-01-23 15:01:39 +00:00
|
|
|
const STATUS_BAR_HIDDEN = 0;
|
|
|
|
const STATUS_BAR_EXPANDED = 1;
|
|
|
|
const STATUS_BAR_EXPANDED_LARGE = 2;
|
|
|
|
|
2022-10-24 06:50:21 +00:00
|
|
|
export function getUnsentMessages(room: Room, threadId?: string): MatrixEvent[] {
|
2017-11-09 15:58:15 +00:00
|
|
|
if (!room) {
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
return room.getPendingEvents().filter(function (ev) {
|
2022-10-24 06:50:21 +00:00
|
|
|
const isNotSent = ev.status === EventStatus.NOT_SENT;
|
|
|
|
const belongsToTheThread = threadId === ev.threadRootId;
|
|
|
|
return isNotSent && (!threadId || belongsToTheThread);
|
2017-11-09 15:58:15 +00:00
|
|
|
});
|
2018-10-27 03:50:35 +00:00
|
|
|
}
|
2017-11-09 15:58:15 +00:00
|
|
|
|
2021-09-12 08:01:14 +00:00
|
|
|
interface IProps {
|
|
|
|
// the room this statusbar is representing.
|
|
|
|
room: Room;
|
|
|
|
|
|
|
|
// true if the room is being peeked at. This affects components that shouldn't
|
|
|
|
// logically be shown when peeking, such as a prompt to invite people to a room.
|
|
|
|
isPeeking?: boolean;
|
|
|
|
// callback for when the user clicks on the 'resend all' button in the
|
|
|
|
// 'unsent messages' bar
|
|
|
|
onResendAllClick?: () => void;
|
|
|
|
|
|
|
|
// callback for when the user clicks on the 'cancel all' button in the
|
|
|
|
// 'unsent messages' bar
|
|
|
|
onCancelAllClick?: () => void;
|
|
|
|
|
|
|
|
// callback for when the user clicks on the 'invite others' button in the
|
|
|
|
// 'you are alone' bar
|
|
|
|
onInviteClick?: () => void;
|
|
|
|
|
|
|
|
// callback for when we do something that changes the size of the
|
|
|
|
// status bar. This is used to trigger a re-layout in the parent
|
|
|
|
// component.
|
|
|
|
onResize?: () => void;
|
|
|
|
|
|
|
|
// callback for when the status bar can be hidden from view, as it is
|
|
|
|
// not displaying anything
|
|
|
|
onHidden?: () => void;
|
|
|
|
|
|
|
|
// callback for when the status bar is displaying something and should
|
|
|
|
// be visible
|
|
|
|
onVisible?: () => void;
|
|
|
|
}
|
2016-02-08 18:04:54 +00:00
|
|
|
|
2021-09-12 08:01:14 +00:00
|
|
|
interface IState {
|
|
|
|
syncState: SyncState;
|
|
|
|
syncStateData: ISyncStateData;
|
|
|
|
unsentMessages: MatrixEvent[];
|
|
|
|
isResending: boolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
export default class RoomStatusBar extends React.PureComponent<IProps, IState> {
|
2022-01-05 15:14:44 +00:00
|
|
|
private unmounted = false;
|
2021-09-13 16:28:52 +00:00
|
|
|
public static contextType = MatrixClientContext;
|
|
|
|
|
2022-12-16 12:29:59 +00:00
|
|
|
public constructor(props: IProps, context: typeof MatrixClientContext) {
|
2021-09-13 16:28:52 +00:00
|
|
|
super(props, context);
|
2021-09-12 08:01:14 +00:00
|
|
|
|
|
|
|
this.state = {
|
2021-09-13 16:28:52 +00:00
|
|
|
syncState: this.context.getSyncState(),
|
|
|
|
syncStateData: this.context.getSyncStateData(),
|
2021-09-12 08:01:14 +00:00
|
|
|
unsentMessages: getUnsentMessages(this.props.room),
|
|
|
|
isResending: false,
|
|
|
|
};
|
|
|
|
}
|
2016-02-08 18:04:54 +00:00
|
|
|
|
2021-09-12 08:01:14 +00:00
|
|
|
public componentDidMount(): void {
|
2021-09-13 16:28:52 +00:00
|
|
|
const client = this.context;
|
|
|
|
client.on("sync", this.onSyncStateChange);
|
|
|
|
client.on("Room.localEchoUpdated", this.onRoomLocalEchoUpdated);
|
2016-02-08 18:04:54 +00:00
|
|
|
|
2021-09-12 08:01:14 +00:00
|
|
|
this.checkSize();
|
2020-08-29 11:14:16 +00:00
|
|
|
}
|
2017-01-23 15:01:39 +00:00
|
|
|
|
2021-09-12 08:01:14 +00:00
|
|
|
public componentDidUpdate(): void {
|
|
|
|
this.checkSize();
|
2020-08-29 11:14:16 +00:00
|
|
|
}
|
2016-02-23 11:06:16 +00:00
|
|
|
|
2021-09-12 08:01:14 +00:00
|
|
|
public componentWillUnmount(): void {
|
2022-01-05 15:14:44 +00:00
|
|
|
this.unmounted = true;
|
2016-02-15 21:50:39 +00:00
|
|
|
// we may have entirely lost our client as we're logging out before clicking login on the guest bar...
|
2021-09-13 16:28:52 +00:00
|
|
|
const client = this.context;
|
2016-02-23 16:17:50 +00:00
|
|
|
if (client) {
|
|
|
|
client.removeListener("sync", this.onSyncStateChange);
|
2021-09-12 08:01:14 +00:00
|
|
|
client.removeListener("Room.localEchoUpdated", this.onRoomLocalEchoUpdated);
|
2016-02-15 21:50:39 +00:00
|
|
|
}
|
2020-08-29 11:14:16 +00:00
|
|
|
}
|
2016-02-08 18:04:54 +00:00
|
|
|
|
2021-09-12 08:01:14 +00:00
|
|
|
private onSyncStateChange = (state: SyncState, prevState: SyncState, data: ISyncStateData): void => {
|
2016-02-08 18:04:54 +00:00
|
|
|
if (state === "SYNCING" && prevState === "SYNCING") {
|
|
|
|
return;
|
|
|
|
}
|
2022-01-05 15:14:44 +00:00
|
|
|
if (this.unmounted) return;
|
2016-02-08 18:04:54 +00:00
|
|
|
this.setState({
|
2017-10-11 16:56:17 +00:00
|
|
|
syncState: state,
|
2018-08-03 17:02:09 +00:00
|
|
|
syncStateData: data,
|
2016-02-08 18:04:54 +00:00
|
|
|
});
|
2020-08-29 11:14:16 +00:00
|
|
|
};
|
2016-02-08 18:04:54 +00:00
|
|
|
|
2021-09-12 08:01:14 +00:00
|
|
|
private onResendAllClick = (): void => {
|
2021-04-21 19:49:58 +00:00
|
|
|
Resend.resendUnsentEvents(this.props.room).then(() => {
|
2021-06-29 12:11:58 +00:00
|
|
|
this.setState({ isResending: false });
|
2021-04-21 19:49:58 +00:00
|
|
|
});
|
2021-06-29 12:11:58 +00:00
|
|
|
this.setState({ isResending: true });
|
2021-07-08 15:36:31 +00:00
|
|
|
dis.fire(Action.FocusSendMessageComposer);
|
2020-08-29 11:14:16 +00:00
|
|
|
};
|
2017-11-09 15:58:15 +00:00
|
|
|
|
2021-09-12 08:01:14 +00:00
|
|
|
private onCancelAllClick = (): void => {
|
2017-11-09 15:58:15 +00:00
|
|
|
Resend.cancelUnsentEvents(this.props.room);
|
2021-07-08 15:36:31 +00:00
|
|
|
dis.fire(Action.FocusSendMessageComposer);
|
2020-08-29 11:14:16 +00:00
|
|
|
};
|
2017-11-09 15:58:15 +00:00
|
|
|
|
2023-01-12 13:25:14 +00:00
|
|
|
private onRoomLocalEchoUpdated = (ev: MatrixEvent, room: Room): void => {
|
2017-11-09 15:58:15 +00:00
|
|
|
if (room.roomId !== this.props.room.roomId) return;
|
2021-04-27 14:58:18 +00:00
|
|
|
const messages = getUnsentMessages(this.props.room);
|
|
|
|
this.setState({
|
|
|
|
unsentMessages: messages,
|
|
|
|
isResending: messages.length > 0 && this.state.isResending,
|
|
|
|
});
|
2020-08-29 11:14:16 +00:00
|
|
|
};
|
2017-11-09 15:58:15 +00:00
|
|
|
|
2017-03-28 08:30:41 +00:00
|
|
|
// Check whether current size is greater than 0, if yes call props.onVisible
|
2021-09-12 08:01:14 +00:00
|
|
|
private checkSize(): void {
|
|
|
|
if (this.getSize()) {
|
2018-01-01 21:15:26 +00:00
|
|
|
if (this.props.onVisible) this.props.onVisible();
|
|
|
|
} else {
|
|
|
|
if (this.props.onHidden) this.props.onHidden();
|
2017-03-28 08:30:41 +00:00
|
|
|
}
|
2020-08-29 11:14:16 +00:00
|
|
|
}
|
2017-03-28 08:30:41 +00:00
|
|
|
|
2017-01-23 15:01:39 +00:00
|
|
|
// 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.
|
2021-09-12 08:01:14 +00:00
|
|
|
private getSize(): number {
|
|
|
|
if (this.shouldShowConnectionError()) {
|
2017-01-23 15:01:39 +00:00
|
|
|
return STATUS_BAR_EXPANDED;
|
2021-04-21 19:49:58 +00:00
|
|
|
} else if (this.state.unsentMessages.length > 0 || this.state.isResending) {
|
2017-01-23 15:01:39 +00:00
|
|
|
return STATUS_BAR_EXPANDED_LARGE;
|
2016-02-23 11:06:16 +00:00
|
|
|
}
|
2017-01-23 15:01:39 +00:00
|
|
|
return STATUS_BAR_HIDDEN;
|
2020-08-29 11:14:16 +00:00
|
|
|
}
|
2016-02-23 11:06:16 +00:00
|
|
|
|
2021-09-12 08:01:14 +00:00
|
|
|
private shouldShowConnectionError(): boolean {
|
2019-01-22 13:49:41 +00:00
|
|
|
// no conn bar trumps the "some not sent" msg since you can't resend without
|
2018-08-03 17:02:09 +00:00
|
|
|
// a connection!
|
|
|
|
// There's one situation in which we don't show this 'no connection' bar, and that's
|
2018-08-15 16:03:54 +00:00
|
|
|
// if it's a resource limit exceeded error: those are shown in the top bar.
|
2018-08-03 17:02:09 +00:00
|
|
|
const errorIsMauError = Boolean(
|
|
|
|
this.state.syncStateData &&
|
|
|
|
this.state.syncStateData.error &&
|
2021-09-12 08:01:14 +00:00
|
|
|
this.state.syncStateData.error.name === "M_RESOURCE_LIMIT_EXCEEDED",
|
2018-08-03 17:02:09 +00:00
|
|
|
);
|
|
|
|
return this.state.syncState === "ERROR" && !errorIsMauError;
|
2020-08-29 11:14:16 +00:00
|
|
|
}
|
2018-08-03 17:02:09 +00:00
|
|
|
|
2021-09-12 08:01:14 +00:00
|
|
|
private getUnsentMessageContent(): JSX.Element {
|
2017-11-09 15:58:15 +00:00
|
|
|
const unsentMessages = this.state.unsentMessages;
|
|
|
|
|
2023-02-13 11:39:16 +00:00
|
|
|
let title: ReactNode;
|
2017-11-09 15:58:15 +00:00
|
|
|
|
2023-02-13 11:39:16 +00:00
|
|
|
let consentError: MatrixError | null = null;
|
|
|
|
let resourceLimitError: MatrixError | null = null;
|
2020-05-28 15:59:27 +00:00
|
|
|
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 = _t(
|
|
|
|
"You can't send any messages until you review and agree to " +
|
|
|
|
"<consentLink>our terms and conditions</consentLink>.",
|
2017-11-16 15:59:16 +00:00
|
|
|
{},
|
|
|
|
{
|
2020-05-28 15:59:27 +00:00
|
|
|
consentLink: (sub) => (
|
2023-05-03 21:26:26 +00:00
|
|
|
<ExternalLink href={consentError!.data?.consent_uri} target="_blank" rel="noreferrer noopener">
|
2020-05-28 15:59:27 +00:00
|
|
|
{sub}
|
2023-05-03 21:26:26 +00:00
|
|
|
</ExternalLink>
|
2020-05-28 15:59:27 +00:00
|
|
|
),
|
2017-11-16 15:59:16 +00:00
|
|
|
},
|
2017-11-09 15:58:15 +00:00
|
|
|
);
|
2020-05-28 15:59:27 +00:00
|
|
|
} else if (resourceLimitError) {
|
|
|
|
title = messageForResourceLimitError(
|
|
|
|
resourceLimitError.data.limit_type,
|
2021-04-27 15:23:27 +00:00
|
|
|
resourceLimitError.data.admin_contact,
|
|
|
|
{
|
|
|
|
"monthly_active_user": _td(
|
|
|
|
"Your message wasn't sent because this homeserver has hit its Monthly Active User Limit. " +
|
|
|
|
"Please <a>contact your service administrator</a> to continue using the service.",
|
|
|
|
),
|
|
|
|
"hs_disabled": _td(
|
2022-05-05 03:50:30 +00:00
|
|
|
"Your message wasn't sent because this homeserver has been blocked by its administrator. " +
|
2021-04-27 15:23:27 +00:00
|
|
|
"Please <a>contact your service administrator</a> to continue using the service.",
|
|
|
|
),
|
|
|
|
"": _td(
|
|
|
|
"Your message wasn't sent because this homeserver has exceeded a resource limit. " +
|
|
|
|
"Please <a>contact your service administrator</a> to continue using the service.",
|
|
|
|
),
|
|
|
|
},
|
|
|
|
);
|
2017-11-09 15:58:15 +00:00
|
|
|
} else {
|
2021-04-21 19:49:58 +00:00
|
|
|
title = _t("Some of your messages have not been sent");
|
2017-11-09 15:58:15 +00:00
|
|
|
}
|
|
|
|
|
2021-04-21 19:49:58 +00:00
|
|
|
let buttonRow = (
|
|
|
|
<>
|
2021-09-12 08:01:14 +00:00
|
|
|
<AccessibleButton onClick={this.onCancelAllClick} className="mx_RoomStatusBar_unsentCancelAllBtn">
|
2021-07-19 21:43:11 +00:00
|
|
|
{_t("Delete all")}
|
2021-04-21 19:49:58 +00:00
|
|
|
</AccessibleButton>
|
2022-07-20 12:41:43 +00:00
|
|
|
<AccessibleButton onClick={this.onResendAllClick} className="mx_RoomStatusBar_unsentRetry">
|
2021-07-19 21:43:11 +00:00
|
|
|
{_t("Retry all")}
|
2021-04-21 19:49:58 +00:00
|
|
|
</AccessibleButton>
|
|
|
|
</>
|
|
|
|
);
|
|
|
|
if (this.state.isResending) {
|
|
|
|
buttonRow = (
|
|
|
|
<>
|
|
|
|
<InlineSpinner w={20} h={20} />
|
2021-07-19 21:43:11 +00:00
|
|
|
{/* span for css */}
|
|
|
|
<span>{_t("Sending")}</span>
|
2021-04-21 19:49:58 +00:00
|
|
|
</>
|
|
|
|
);
|
|
|
|
}
|
2020-05-28 15:59:27 +00:00
|
|
|
|
2022-07-20 12:41:43 +00:00
|
|
|
return (
|
|
|
|
<RoomStatusBarUnsentMessages
|
|
|
|
title={title}
|
|
|
|
description={_t("You can select all or individual messages to retry or delete")}
|
|
|
|
notificationState={StaticNotificationState.RED_EXCLAMATION}
|
|
|
|
buttons={buttonRow}
|
|
|
|
/>
|
|
|
|
);
|
2020-08-29 11:14:16 +00:00
|
|
|
}
|
2017-11-09 15:58:15 +00:00
|
|
|
|
2023-02-13 17:01:43 +00:00
|
|
|
public render(): React.ReactNode {
|
2021-09-12 08:01:14 +00:00
|
|
|
if (this.shouldShowConnectionError()) {
|
2016-02-08 18:04:54 +00:00
|
|
|
return (
|
2021-04-21 19:49:58 +00:00
|
|
|
<div className="mx_RoomStatusBar">
|
|
|
|
<div role="alert">
|
|
|
|
<div className="mx_RoomStatusBar_connectionLostBar">
|
2023-05-24 02:37:10 +00:00
|
|
|
<WarningIcon width="24" height="24" />
|
2021-04-21 19:49:58 +00:00
|
|
|
<div>
|
|
|
|
<div className="mx_RoomStatusBar_connectionLostBar_title">
|
2021-07-19 21:43:11 +00:00
|
|
|
{_t("Connectivity to the server has been lost.")}
|
2021-04-21 19:49:58 +00:00
|
|
|
</div>
|
|
|
|
<div className="mx_RoomStatusBar_connectionLostBar_desc">
|
2021-07-19 21:43:11 +00:00
|
|
|
{_t("Sent messages will be stored until your connection has returned.")}
|
2021-04-21 19:49:58 +00:00
|
|
|
</div>
|
|
|
|
</div>
|
2018-06-29 08:46:01 +00:00
|
|
|
</div>
|
2016-02-08 18:04:54 +00:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2021-04-21 19:49:58 +00:00
|
|
|
if (this.state.unsentMessages.length > 0 || this.state.isResending) {
|
2021-09-12 08:01:14 +00:00
|
|
|
return this.getUnsentMessageContent();
|
2016-02-08 18:04:54 +00:00
|
|
|
}
|
|
|
|
|
2016-02-09 14:42:32 +00:00
|
|
|
return null;
|
2020-08-29 11:14:16 +00:00
|
|
|
}
|
|
|
|
}
|