2022-09-28 17:13:09 +00:00
|
|
|
/*
|
|
|
|
Copyright 2022 The Matrix.org Foundation C.I.C.
|
|
|
|
|
|
|
|
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-08-09 07:18:41 +00:00
|
|
|
import {
|
|
|
|
MatrixClient,
|
|
|
|
LOCAL_NOTIFICATION_SETTINGS_PREFIX,
|
|
|
|
NotificationCountType,
|
|
|
|
Room,
|
2023-08-23 09:04:25 +00:00
|
|
|
LocalNotificationSettings,
|
|
|
|
ReceiptType,
|
2023-08-09 07:18:41 +00:00
|
|
|
} from "matrix-js-sdk/src/matrix";
|
2024-01-25 16:53:41 +00:00
|
|
|
import { IndicatorIcon } from "@vector-im/compound-web";
|
2022-10-10 14:18:38 +00:00
|
|
|
|
|
|
|
import SettingsStore from "../settings/SettingsStore";
|
2024-01-25 16:53:41 +00:00
|
|
|
import { NotificationLevel } from "../stores/notifications/NotificationLevel";
|
2024-02-01 17:58:57 +00:00
|
|
|
import { doesRoomHaveUnreadMessages } from "../Unread";
|
2022-10-10 14:18:38 +00:00
|
|
|
|
|
|
|
export const deviceNotificationSettingsKeys = [
|
|
|
|
"notificationsEnabled",
|
|
|
|
"notificationBodyEnabled",
|
|
|
|
"audioNotificationsEnabled",
|
|
|
|
];
|
2022-09-28 17:13:09 +00:00
|
|
|
|
2023-04-21 10:50:42 +00:00
|
|
|
export function getLocalNotificationAccountDataEventType(deviceId: string | null): string {
|
2022-09-28 17:13:09 +00:00
|
|
|
return `${LOCAL_NOTIFICATION_SETTINGS_PREFIX.name}.${deviceId}`;
|
|
|
|
}
|
|
|
|
|
2022-10-10 14:18:38 +00:00
|
|
|
export async function createLocalNotificationSettingsIfNeeded(cli: MatrixClient): Promise<void> {
|
2022-10-13 12:15:34 +00:00
|
|
|
if (cli.isGuest()) {
|
|
|
|
return;
|
|
|
|
}
|
2023-02-13 17:01:43 +00:00
|
|
|
const eventType = getLocalNotificationAccountDataEventType(cli.deviceId!);
|
2022-10-10 14:18:38 +00:00
|
|
|
const event = cli.getAccountData(eventType);
|
|
|
|
// New sessions will create an account data event to signify they support
|
|
|
|
// remote toggling of push notifications on this device. Default `is_silenced=true`
|
|
|
|
// For backwards compat purposes, older sessions will need to check settings value
|
|
|
|
// to determine what the state of `is_silenced`
|
|
|
|
if (!event) {
|
|
|
|
// If any of the above is true, we fall in the "backwards compat" case,
|
|
|
|
// and `is_silenced` will be set to `false`
|
2022-12-12 11:24:14 +00:00
|
|
|
const isSilenced = !deviceNotificationSettingsKeys.some((key) => SettingsStore.getValue(key));
|
2022-10-10 14:18:38 +00:00
|
|
|
|
|
|
|
await cli.setAccountData(eventType, {
|
|
|
|
is_silenced: isSilenced,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-29 14:23:02 +00:00
|
|
|
export function localNotificationsAreSilenced(cli: MatrixClient): boolean {
|
2023-02-13 17:01:43 +00:00
|
|
|
const eventType = getLocalNotificationAccountDataEventType(cli.deviceId!);
|
2022-09-29 14:23:02 +00:00
|
|
|
const event = cli.getAccountData(eventType);
|
2022-10-11 16:11:11 +00:00
|
|
|
return event?.getContent<LocalNotificationSettings>()?.is_silenced ?? false;
|
2022-09-29 14:23:02 +00:00
|
|
|
}
|
2022-11-15 10:27:13 +00:00
|
|
|
|
2022-12-22 13:18:38 +00:00
|
|
|
/**
|
|
|
|
* Mark a room as read
|
|
|
|
* @param room
|
|
|
|
* @param client
|
|
|
|
* @returns a promise that resolves when the room has been marked as read
|
|
|
|
*/
|
|
|
|
export async function clearRoomNotification(room: Room, client: MatrixClient): Promise<{} | undefined> {
|
2023-04-28 06:07:25 +00:00
|
|
|
const lastEvent = room.getLastLiveEvent();
|
2022-12-22 13:18:38 +00:00
|
|
|
|
|
|
|
try {
|
|
|
|
if (lastEvent) {
|
|
|
|
const receiptType = SettingsStore.getValue("sendReadReceipts", room.roomId)
|
|
|
|
? ReceiptType.Read
|
|
|
|
: ReceiptType.ReadPrivate;
|
|
|
|
return await client.sendReadReceipt(lastEvent, receiptType, true);
|
|
|
|
} else {
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
} finally {
|
|
|
|
// We've had a lot of stuck unread notifications that in e2ee rooms
|
|
|
|
// They occur on event decryption when clients try to replicate the logic
|
|
|
|
//
|
|
|
|
// This resets the notification on a room, even though no read receipt
|
|
|
|
// has been sent, particularly useful when the clients has incorrectly
|
|
|
|
// notified a user.
|
|
|
|
room.setUnreadNotificationCount(NotificationCountType.Highlight, 0);
|
|
|
|
room.setUnreadNotificationCount(NotificationCountType.Total, 0);
|
|
|
|
for (const thread of room.getThreads()) {
|
|
|
|
room.setThreadUnreadNotificationCount(thread.id, NotificationCountType.Highlight, 0);
|
|
|
|
room.setThreadUnreadNotificationCount(thread.id, NotificationCountType.Total, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Marks all rooms with an unread counter as read
|
|
|
|
* @param client The matrix client
|
|
|
|
* @returns a promise that resolves when all rooms have been marked as read
|
|
|
|
*/
|
|
|
|
export function clearAllNotifications(client: MatrixClient): Promise<Array<{} | undefined>> {
|
|
|
|
const receiptPromises = client.getRooms().reduce((promises: Array<Promise<{} | undefined>>, room: Room) => {
|
2024-02-01 17:58:57 +00:00
|
|
|
if (doesRoomHaveUnreadMessages(room, true)) {
|
2022-12-22 13:18:38 +00:00
|
|
|
const promise = clearRoomNotification(room, client);
|
|
|
|
promises.push(promise);
|
2022-11-15 10:27:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return promises;
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
return Promise.all(receiptPromises);
|
|
|
|
}
|
2024-01-25 16:53:41 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* A helper to transform a notification color to the what the Compound Icon Button
|
|
|
|
* expects
|
|
|
|
*/
|
|
|
|
export function notificationLevelToIndicator(
|
|
|
|
level: NotificationLevel,
|
|
|
|
): React.ComponentPropsWithRef<typeof IndicatorIcon>["indicator"] {
|
|
|
|
if (level <= NotificationLevel.None) {
|
|
|
|
return undefined;
|
|
|
|
} else if (level <= NotificationLevel.Activity) {
|
|
|
|
return "default";
|
|
|
|
} else if (level <= NotificationLevel.Notification) {
|
|
|
|
return "success";
|
|
|
|
} else {
|
|
|
|
return "critical";
|
|
|
|
}
|
|
|
|
}
|
2024-02-07 13:49:40 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Return the thread notification level for a room
|
|
|
|
* @param room
|
|
|
|
* @returns {NotificationLevel}
|
|
|
|
*/
|
|
|
|
export function getThreadNotificationLevel(room: Room): NotificationLevel {
|
|
|
|
const notificationCountType = room.threadsAggregateNotificationType;
|
|
|
|
switch (notificationCountType) {
|
|
|
|
case NotificationCountType.Highlight:
|
|
|
|
return NotificationLevel.Highlight;
|
|
|
|
case NotificationCountType.Total:
|
|
|
|
return NotificationLevel.Notification;
|
|
|
|
default:
|
|
|
|
return NotificationLevel.Activity;
|
|
|
|
}
|
|
|
|
}
|