f62fe2626c
* Implement new notification settings UI * Sort new keywords at the front * Make ts-strict happier * Make ts-strict happier * chore: fixed lint issues * update beta card * Fix issue with the user settings test * chore: fixed lint issues * Add tests for notification settings * chore: fixed lint issues * fix: spurious text failures * improve tests further * make ts-strict happier * improve tests further * Reduce uncovered conditions * Correct snapshot * even more test coverage * Fix an issue with inverted rules * Update res/css/views/settings/tabs/_SettingsIndent.pcss Co-authored-by: Šimon Brandner <simon.bra.ag@gmail.com> * Fix license headers * Improve i18n * make linters happier * Improve beta labels * improve i18n * chore: fixed lint issues * fix: more lint issues * Update snapshots to match changed text * Update text as requested * Remove labs image * Update snapshots * Correct an issue with one of the tests * fix: keyword reconcilation code * Determine mute status more accurately * Address review comments * Prevent duplicate updates * Fix missing license header * slight change to avoid ts-strict complaining * fix test issue caused by previous merge --------- Co-authored-by: Šimon Brandner <simon.bra.ag@gmail.com>
102 lines
4.2 KiB
TypeScript
102 lines
4.2 KiB
TypeScript
/*
|
|
Copyright 2023 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.
|
|
*/
|
|
|
|
import { IPushRules, MatrixClient } from "matrix-js-sdk/src/matrix";
|
|
import { useCallback, useEffect, useMemo, useRef, useState } from "react";
|
|
|
|
import { NotificationSettings } from "../models/notificationsettings/NotificationSettings";
|
|
import { PushRuleDiff } from "../models/notificationsettings/PushRuleDiff";
|
|
import { reconcileNotificationSettings } from "../models/notificationsettings/reconcileNotificationSettings";
|
|
import { toNotificationSettings } from "../models/notificationsettings/toNotificationSettings";
|
|
|
|
async function applyChanges(cli: MatrixClient, changes: PushRuleDiff): Promise<void> {
|
|
await Promise.all(changes.deleted.map((change) => cli.deletePushRule("global", change.kind, change.rule_id)));
|
|
await Promise.all(changes.added.map((change) => cli.addPushRule("global", change.kind, change.rule_id, change)));
|
|
await Promise.all(
|
|
changes.updated.map(async (change) => {
|
|
if (change.enabled !== undefined) {
|
|
await cli.setPushRuleEnabled("global", change.kind, change.rule_id, change.enabled);
|
|
}
|
|
if (change.actions !== undefined) {
|
|
await cli.setPushRuleActions("global", change.kind, change.rule_id, change.actions);
|
|
}
|
|
}),
|
|
);
|
|
}
|
|
|
|
type UseNotificationSettings = {
|
|
model: NotificationSettings | null;
|
|
hasPendingChanges: boolean;
|
|
reconcile: (model: NotificationSettings) => void;
|
|
};
|
|
|
|
export function useNotificationSettings(cli: MatrixClient): UseNotificationSettings {
|
|
const run = useLinearisedPromise<void>();
|
|
const supportsIntentionalMentions = useMemo(() => cli.supportsIntentionalMentions(), [cli]);
|
|
|
|
const pushRules = useRef<IPushRules | null>(null);
|
|
const [model, setModel] = useState<NotificationSettings | null>(null);
|
|
const [hasPendingChanges, setPendingChanges] = useState<boolean>(false);
|
|
const updatePushRules = useCallback(async () => {
|
|
const rules = await cli.getPushRules();
|
|
const model = toNotificationSettings(rules, supportsIntentionalMentions);
|
|
const pendingChanges = reconcileNotificationSettings(rules, model, supportsIntentionalMentions);
|
|
pushRules.current = rules;
|
|
setPendingChanges(
|
|
pendingChanges.updated.length > 0 || pendingChanges.added.length > 0 || pendingChanges.deleted.length > 0,
|
|
);
|
|
setModel(model);
|
|
}, [cli, supportsIntentionalMentions]);
|
|
|
|
useEffect(() => {
|
|
run(updatePushRules).catch((err) => console.error(err));
|
|
}, [cli, run, updatePushRules]);
|
|
|
|
const reconcile = useCallback(
|
|
(model: NotificationSettings) => {
|
|
setModel(model);
|
|
run(async () => {
|
|
if (pushRules.current !== null) {
|
|
const changes = reconcileNotificationSettings(
|
|
pushRules.current,
|
|
model,
|
|
supportsIntentionalMentions,
|
|
);
|
|
await applyChanges(cli, changes);
|
|
await updatePushRules();
|
|
}
|
|
}).catch((err) => console.error(err));
|
|
},
|
|
[run, supportsIntentionalMentions, cli, updatePushRules],
|
|
);
|
|
|
|
return { model, hasPendingChanges, reconcile };
|
|
}
|
|
|
|
function useLinearisedPromise<T>(): (fun: () => Promise<T>) => Promise<T> {
|
|
const lastPromise = useRef<Promise<T> | null>(null);
|
|
|
|
return useCallback((fun: () => Promise<T>): Promise<T> => {
|
|
let next: Promise<T>;
|
|
if (lastPromise.current === null) {
|
|
next = fun();
|
|
} else {
|
|
next = lastPromise.current.then(fun);
|
|
}
|
|
lastPromise.current = next;
|
|
return next;
|
|
}, []);
|
|
}
|