matrix-react-sdk
Version:
SDK for matrix.org using React
133 lines (123 loc) • 19.1 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.useUnreadThreadRooms = useUnreadThreadRooms;
var _react = require("react");
var _matrix = require("matrix-js-sdk/src/matrix");
var _lodash = require("lodash");
var _Unread = require("../../../../Unread");
var _NotificationLevel = require("../../../../stores/notifications/NotificationLevel");
var _notifications = require("../../../../utils/notifications");
var _useSettings = require("../../../../hooks/useSettings");
var _MatrixClientContext = require("../../../../contexts/MatrixClientContext");
var _useEventEmitter = require("../../../../hooks/useEventEmitter");
var _VisibilityProvider = require("../../../../stores/room-list/filters/VisibilityProvider");
/*
* Copyright 2024 New Vector Ltd.
* Copyright 2024 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 MIN_UPDATE_INTERVAL_MS = 500;
/**
* Return the greatest notification level of all thread, the list of rooms with unread threads, and their notification level.
* The result is computed when the client syncs, or when forceComputation is true
* @param forceComputation
* @returns {Result}
*/
function useUnreadThreadRooms(forceComputation) {
const msc3946ProcessDynamicPredecessor = (0, _useSettings.useSettingValue)("feature_dynamic_room_predecessors");
const settingTACOnlyNotifs = (0, _useSettings.useSettingValue)("Notifications.tac_only_notifications");
const mxClient = (0, _MatrixClientContext.useMatrixClientContext)();
const [result, setResult] = (0, _react.useState)({
greatestNotificationLevel: _NotificationLevel.NotificationLevel.None,
rooms: []
});
const doUpdate = (0, _react.useCallback)(() => {
setResult(computeUnreadThreadRooms(mxClient, msc3946ProcessDynamicPredecessor, settingTACOnlyNotifs));
}, [mxClient, msc3946ProcessDynamicPredecessor, settingTACOnlyNotifs]);
// The exhautive deps lint rule can't compute dependencies here since it's not a plain inline func.
// We make this as simple as possible so its only dep is doUpdate itself.
// eslint-disable-next-line react-hooks/exhaustive-deps
const scheduleUpdate = (0, _react.useCallback)((0, _lodash.throttle)(doUpdate, MIN_UPDATE_INTERVAL_MS, {
leading: false,
trailing: true
}), [doUpdate]);
// Listen to sync events to update the result
(0, _useEventEmitter.useEventEmitter)(mxClient, _matrix.ClientEvent.Sync, scheduleUpdate);
// and also when events get decrypted, since this will often happen after the sync
// event and may change notifications.
(0, _useEventEmitter.useEventEmitter)(mxClient, _matrix.MatrixEventEvent.Decrypted, scheduleUpdate);
// Force the list computation
(0, _react.useEffect)(() => {
if (forceComputation) {
doUpdate();
}
}, [doUpdate, forceComputation]);
return result;
}
/**
* Compute the greatest notification level of all thread, the list of rooms with unread threads, and their notification level.
* @param mxClient - MatrixClient
* @param msc3946ProcessDynamicPredecessor
*/
function computeUnreadThreadRooms(mxClient, msc3946ProcessDynamicPredecessor, settingTACOnlyNotifs) {
// Only count visible rooms to not torment the user with notification counts in rooms they can't see.
// This will include highlights from the previous version of the room internally
const visibleRooms = mxClient.getVisibleRooms(msc3946ProcessDynamicPredecessor);
let greatestNotificationLevel = _NotificationLevel.NotificationLevel.None;
const rooms = [];
for (const room of visibleRooms) {
// We only care about rooms with unread threads
if (_VisibilityProvider.VisibilityProvider.instance.isRoomVisible(room) && (0, _Unread.doesRoomHaveUnreadThreads)(room)) {
// Get the greatest notification level of all threads
const notificationLevel = (0, _notifications.getThreadNotificationLevel)(room);
// If the room has an activity notification or less, we ignore it
if (settingTACOnlyNotifs && notificationLevel <= _NotificationLevel.NotificationLevel.Activity) {
continue;
}
if (notificationLevel > greatestNotificationLevel) {
greatestNotificationLevel = notificationLevel;
}
rooms.push({
room,
notificationLevel
});
}
}
const sortedRooms = rooms.sort((a, b) => sortRoom(a, b));
return {
greatestNotificationLevel,
rooms: sortedRooms
};
}
/**
* Store the room and its thread notification level
*/
/**
* Sort notification level by the most important notification level to the least important
* Highlight > Notification > Activity
* If the notification level is the same, we sort by the most recent thread
* @param roomDataA - room and notification level of room A
* @param roomDataB - room and notification level of room B
* @returns {number}
*/
function sortRoom(roomDataA, roomDataB) {
const {
notificationLevel: notificationLevelA,
room: roomA
} = roomDataA;
const {
notificationLevel: notificationLevelB,
room: roomB
} = roomDataB;
const timestampA = roomA.getLastThread()?.events.at(-1)?.getTs();
const timestampB = roomB.getLastThread()?.events.at(-1)?.getTs();
// NotificationLevel is a numeric enum, so we can compare them directly
if (notificationLevelA > notificationLevelB) return -1;else if (notificationLevelB > notificationLevelA) return 1;
// Display most recent first
else if (!timestampA) return 1;else if (!timestampB) return -1;else return timestampB - timestampA;
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","require","_matrix","_lodash","_Unread","_NotificationLevel","_notifications","_useSettings","_MatrixClientContext","_useEventEmitter","_VisibilityProvider","MIN_UPDATE_INTERVAL_MS","useUnreadThreadRooms","forceComputation","msc3946ProcessDynamicPredecessor","useSettingValue","settingTACOnlyNotifs","mxClient","useMatrixClientContext","result","setResult","useState","greatestNotificationLevel","NotificationLevel","None","rooms","doUpdate","useCallback","computeUnreadThreadRooms","scheduleUpdate","throttle","leading","trailing","useEventEmitter","ClientEvent","Sync","MatrixEventEvent","Decrypted","useEffect","visibleRooms","getVisibleRooms","room","VisibilityProvider","instance","isRoomVisible","doesRoomHaveUnreadThreads","notificationLevel","getThreadNotificationLevel","Activity","push","sortedRooms","sort","a","b","sortRoom","roomDataA","roomDataB","notificationLevelA","roomA","notificationLevelB","roomB","timestampA","getLastThread","events","at","getTs","timestampB"],"sources":["../../../../../src/components/views/spaces/threads-activity-centre/useUnreadThreadRooms.ts"],"sourcesContent":["/*\n * Copyright 2024 New Vector Ltd.\n * Copyright 2024 The Matrix.org Foundation C.I.C.\n *\n * SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only\n * Please see LICENSE files in the repository root for full details.\n */\n\nimport { useCallback, useEffect, useState } from \"react\";\nimport { ClientEvent, MatrixClient, MatrixEventEvent, Room } from \"matrix-js-sdk/src/matrix\";\nimport { throttle } from \"lodash\";\n\nimport { doesRoomHaveUnreadThreads } from \"../../../../Unread\";\nimport { NotificationLevel } from \"../../../../stores/notifications/NotificationLevel\";\nimport { getThreadNotificationLevel } from \"../../../../utils/notifications\";\nimport { useSettingValue } from \"../../../../hooks/useSettings\";\nimport { useMatrixClientContext } from \"../../../../contexts/MatrixClientContext\";\nimport { useEventEmitter } from \"../../../../hooks/useEventEmitter\";\nimport { VisibilityProvider } from \"../../../../stores/room-list/filters/VisibilityProvider\";\n\nconst MIN_UPDATE_INTERVAL_MS = 500;\n\ntype Result = {\n    greatestNotificationLevel: NotificationLevel;\n    rooms: Array<{ room: Room; notificationLevel: NotificationLevel }>;\n};\n\n/**\n * Return the greatest notification level of all thread, the list of rooms with unread threads, and their notification level.\n * The result is computed when the client syncs, or when forceComputation is true\n * @param forceComputation\n * @returns {Result}\n */\nexport function useUnreadThreadRooms(forceComputation: boolean): Result {\n    const msc3946ProcessDynamicPredecessor = useSettingValue<boolean>(\"feature_dynamic_room_predecessors\");\n    const settingTACOnlyNotifs = useSettingValue<boolean>(\"Notifications.tac_only_notifications\");\n    const mxClient = useMatrixClientContext();\n\n    const [result, setResult] = useState<Result>({ greatestNotificationLevel: NotificationLevel.None, rooms: [] });\n\n    const doUpdate = useCallback(() => {\n        setResult(computeUnreadThreadRooms(mxClient, msc3946ProcessDynamicPredecessor, settingTACOnlyNotifs));\n    }, [mxClient, msc3946ProcessDynamicPredecessor, settingTACOnlyNotifs]);\n\n    // The exhautive deps lint rule can't compute dependencies here since it's not a plain inline func.\n    // We make this as simple as possible so its only dep is doUpdate itself.\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n    const scheduleUpdate = useCallback(\n        throttle(doUpdate, MIN_UPDATE_INTERVAL_MS, {\n            leading: false,\n            trailing: true,\n        }),\n        [doUpdate],\n    );\n\n    // Listen to sync events to update the result\n    useEventEmitter(mxClient, ClientEvent.Sync, scheduleUpdate);\n    // and also when events get decrypted, since this will often happen after the sync\n    // event and may change notifications.\n    useEventEmitter(mxClient, MatrixEventEvent.Decrypted, scheduleUpdate);\n\n    // Force the list computation\n    useEffect(() => {\n        if (forceComputation) {\n            doUpdate();\n        }\n    }, [doUpdate, forceComputation]);\n\n    return result;\n}\n\n/**\n * Compute the greatest notification level of all thread, the list of rooms with unread threads, and their notification level.\n * @param mxClient - MatrixClient\n * @param msc3946ProcessDynamicPredecessor\n */\nfunction computeUnreadThreadRooms(\n    mxClient: MatrixClient,\n    msc3946ProcessDynamicPredecessor: boolean,\n    settingTACOnlyNotifs: boolean,\n): Result {\n    // Only count visible rooms to not torment the user with notification counts in rooms they can't see.\n    // This will include highlights from the previous version of the room internally\n    const visibleRooms = mxClient.getVisibleRooms(msc3946ProcessDynamicPredecessor);\n\n    let greatestNotificationLevel = NotificationLevel.None;\n    const rooms: Result[\"rooms\"] = [];\n\n    for (const room of visibleRooms) {\n        // We only care about rooms with unread threads\n        if (VisibilityProvider.instance.isRoomVisible(room) && doesRoomHaveUnreadThreads(room)) {\n            // Get the greatest notification level of all threads\n            const notificationLevel = getThreadNotificationLevel(room);\n\n            // If the room has an activity notification or less, we ignore it\n            if (settingTACOnlyNotifs && notificationLevel <= NotificationLevel.Activity) {\n                continue;\n            }\n\n            if (notificationLevel > greatestNotificationLevel) {\n                greatestNotificationLevel = notificationLevel;\n            }\n\n            rooms.push({ room, notificationLevel });\n        }\n    }\n\n    const sortedRooms = rooms.sort((a, b) => sortRoom(a, b));\n    return { greatestNotificationLevel, rooms: sortedRooms };\n}\n\n/**\n * Store the room and its thread notification level\n */\ntype RoomData = Result[\"rooms\"][0];\n\n/**\n * Sort notification level by the most important notification level to the least important\n * Highlight > Notification > Activity\n * If the notification level is the same, we sort by the most recent thread\n * @param roomDataA - room and notification level of room A\n * @param roomDataB - room and notification level of room B\n * @returns {number}\n */\nfunction sortRoom(roomDataA: RoomData, roomDataB: RoomData): number {\n    const { notificationLevel: notificationLevelA, room: roomA } = roomDataA;\n    const { notificationLevel: notificationLevelB, room: roomB } = roomDataB;\n\n    const timestampA = roomA.getLastThread()?.events.at(-1)?.getTs();\n    const timestampB = roomB.getLastThread()?.events.at(-1)?.getTs();\n\n    // NotificationLevel is a numeric enum, so we can compare them directly\n    if (notificationLevelA > notificationLevelB) return -1;\n    else if (notificationLevelB > notificationLevelA) return 1;\n    // Display most recent first\n    else if (!timestampA) return 1;\n    else if (!timestampB) return -1;\n    else return timestampB - timestampA;\n}\n"],"mappings":"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","ignoreList":[]}