2016-08-17 19:26:37 +02:00
|
|
|
/*
|
|
|
|
Copyright 2016 OpenMarket Ltd
|
2019-06-19 12:48:47 +02:00
|
|
|
Copyright 2019 The Matrix.org Foundation C.I.C.
|
2016-08-17 19:26:37 +02: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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
import MatrixClientPeg from './MatrixClientPeg';
|
|
|
|
import PushProcessor from 'matrix-js-sdk/lib/pushprocessor';
|
2017-07-12 14:58:14 +02:00
|
|
|
import Promise from 'bluebird';
|
2016-08-17 19:26:37 +02:00
|
|
|
|
2016-08-18 14:44:58 +02:00
|
|
|
export const ALL_MESSAGES_LOUD = 'all_messages_loud';
|
|
|
|
export const ALL_MESSAGES = 'all_messages';
|
|
|
|
export const MENTIONS_ONLY = 'mentions_only';
|
|
|
|
export const MUTE = 'mute';
|
|
|
|
|
2019-04-02 00:06:33 +02:00
|
|
|
export const BADGE_STATES = [ALL_MESSAGES, ALL_MESSAGES_LOUD];
|
|
|
|
export const MENTION_BADGE_STATES = [...BADGE_STATES, MENTIONS_ONLY];
|
2019-02-06 11:51:29 +01:00
|
|
|
|
2019-06-19 12:46:24 +02:00
|
|
|
export function shouldShowNotifBadge(roomNotifState) {
|
2019-06-19 13:12:19 +02:00
|
|
|
return BADGE_STATES.includes(roomNotifState);
|
2019-02-06 11:51:29 +01:00
|
|
|
}
|
|
|
|
|
2019-06-19 12:46:24 +02:00
|
|
|
export function shouldShowMentionBadge(roomNotifState) {
|
2019-06-19 13:12:19 +02:00
|
|
|
return MENTION_BADGE_STATES.includes(roomNotifState);
|
2019-02-06 11:51:29 +01:00
|
|
|
}
|
|
|
|
|
2019-06-19 13:06:32 +02:00
|
|
|
export function countRoomsWithNotif(rooms) {
|
|
|
|
return rooms.reduce((result, room, index) => {
|
|
|
|
const roomNotifState = getRoomNotifsState(room.roomId);
|
|
|
|
const highlight = room.getUnreadNotificationCount('highlight') > 0;
|
|
|
|
const notificationCount = room.getUnreadNotificationCount();
|
|
|
|
|
|
|
|
const notifBadges = notificationCount > 0 && _shouldShowNotifBadge(roomNotifState);
|
|
|
|
const mentionBadges = highlight && _shouldShowMentionBadge(roomNotifState);
|
|
|
|
const isInvite = room.hasMembershipState(MatrixClientPeg.get().credentials.userId, 'invite');
|
|
|
|
const badges = notifBadges || mentionBadges || isInvite;
|
|
|
|
|
|
|
|
if (badges) {
|
|
|
|
result.count++;
|
|
|
|
if (highlight) {
|
|
|
|
result.highlight = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}, {count: 0, highlight: false});
|
|
|
|
}
|
|
|
|
|
2019-02-06 11:51:29 +01:00
|
|
|
export function aggregateNotificationCount(rooms) {
|
|
|
|
return rooms.reduce((result, room, index) => {
|
|
|
|
const roomNotifState = getRoomNotifsState(room.roomId);
|
|
|
|
const highlight = room.getUnreadNotificationCount('highlight') > 0;
|
|
|
|
const notificationCount = room.getUnreadNotificationCount();
|
|
|
|
|
2019-06-19 12:46:24 +02:00
|
|
|
const notifBadges = notificationCount > 0 && shouldShowNotifBadge(roomNotifState);
|
|
|
|
const mentionBadges = highlight && shouldShowMentionBadge(roomNotifState);
|
2019-02-06 11:51:29 +01:00
|
|
|
const badges = notifBadges || mentionBadges;
|
|
|
|
|
|
|
|
if (badges) {
|
|
|
|
result.count += notificationCount;
|
|
|
|
if (highlight) {
|
|
|
|
result.highlight = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}, {count: 0, highlight: false});
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getRoomHasBadge(room) {
|
|
|
|
const roomNotifState = getRoomNotifsState(room.roomId);
|
|
|
|
const highlight = room.getUnreadNotificationCount('highlight') > 0;
|
|
|
|
const notificationCount = room.getUnreadNotificationCount();
|
|
|
|
|
2019-06-19 12:46:24 +02:00
|
|
|
const notifBadges = notificationCount > 0 && shouldShowNotifBadge(roomNotifState);
|
|
|
|
const mentionBadges = highlight && shouldShowMentionBadge(roomNotifState);
|
2019-02-06 11:51:29 +01:00
|
|
|
|
|
|
|
return notifBadges || mentionBadges;
|
|
|
|
}
|
|
|
|
|
2016-08-17 19:26:37 +02:00
|
|
|
export function getRoomNotifsState(roomId) {
|
2016-09-01 17:58:18 +02:00
|
|
|
if (MatrixClientPeg.get().isGuest()) return ALL_MESSAGES;
|
2016-08-18 15:00:14 +02:00
|
|
|
|
2016-08-17 19:26:37 +02:00
|
|
|
// look through the override rules for a rule affecting this room:
|
|
|
|
// if one exists, it will take precedence.
|
|
|
|
const muteRule = findOverrideMuteRule(roomId);
|
2016-08-18 15:00:14 +02:00
|
|
|
if (muteRule) {
|
2016-08-18 14:44:58 +02:00
|
|
|
return MUTE;
|
2016-08-17 19:26:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// for everything else, look at the room rule.
|
2018-02-06 18:50:53 +01:00
|
|
|
let roomRule = null;
|
|
|
|
try {
|
|
|
|
roomRule = MatrixClientPeg.get().getRoomPushRule('global', roomId);
|
|
|
|
} catch (err) {
|
|
|
|
// Possible that the client doesn't have pushRules yet. If so, it
|
|
|
|
// hasn't started eiher, so indicate that this room is not notifying.
|
|
|
|
return null;
|
|
|
|
}
|
2016-08-17 19:26:37 +02:00
|
|
|
|
|
|
|
// XXX: We have to assume the default is to notify for all messages
|
|
|
|
// (in particular this will be 'wrong' for one to one rooms because
|
|
|
|
// they will notify loudly for all messages)
|
2016-08-18 14:44:58 +02:00
|
|
|
if (!roomRule || !roomRule.enabled) return ALL_MESSAGES;
|
2016-08-17 19:26:37 +02:00
|
|
|
|
|
|
|
// a mute at the room level will still allow mentions
|
|
|
|
// to notify
|
2016-08-18 14:44:58 +02:00
|
|
|
if (isMuteRule(roomRule)) return MENTIONS_ONLY;
|
2016-08-17 19:26:37 +02:00
|
|
|
|
|
|
|
const actionsObject = PushProcessor.actionListToActionsObject(roomRule.actions);
|
2016-08-18 14:44:58 +02:00
|
|
|
if (actionsObject.tweaks.sound) return ALL_MESSAGES_LOUD;
|
2016-08-17 19:26:37 +02:00
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function setRoomNotifsState(roomId, newState) {
|
2017-07-01 15:28:12 +02:00
|
|
|
if (newState === MUTE) {
|
2016-08-18 15:00:14 +02:00
|
|
|
return setRoomNotifsStateMuted(roomId);
|
|
|
|
} else {
|
|
|
|
return setRoomNotifsStateUnmuted(roomId, newState);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-02 00:06:33 +02:00
|
|
|
export function getUnreadNotificationCount(room, type=null) {
|
|
|
|
let notificationCount = room.getUnreadNotificationCount(type);
|
|
|
|
|
|
|
|
// Check notification counts in the old room just in case there's some lost
|
|
|
|
// there. We only go one level down to avoid performance issues, and theory
|
|
|
|
// is that 1st generation rooms will have already been read by the 3rd generation.
|
|
|
|
const createEvent = room.currentState.getStateEvents("m.room.create", "");
|
|
|
|
if (createEvent && createEvent.getContent()['predecessor']) {
|
|
|
|
const oldRoomId = createEvent.getContent()['predecessor']['room_id'];
|
|
|
|
const oldRoom = MatrixClientPeg.get().getRoom(oldRoomId);
|
|
|
|
if (oldRoom) {
|
|
|
|
// We only ever care if there's highlights in the old room. No point in
|
|
|
|
// notifying the user for unread messages because they would have extreme
|
|
|
|
// difficulty changing their notification preferences away from "All Messages"
|
|
|
|
// and "Noisy".
|
|
|
|
notificationCount += oldRoom.getUnreadNotificationCount("highlight");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return notificationCount;
|
|
|
|
}
|
|
|
|
|
2016-08-18 15:00:14 +02:00
|
|
|
function setRoomNotifsStateMuted(roomId) {
|
2016-08-17 19:26:37 +02:00
|
|
|
const cli = MatrixClientPeg.get();
|
|
|
|
const promises = [];
|
|
|
|
|
2016-08-18 15:00:14 +02:00
|
|
|
// delete the room rule
|
|
|
|
const roomRule = cli.getRoomPushRule('global', roomId);
|
|
|
|
if (roomRule) {
|
|
|
|
promises.push(cli.deletePushRule('global', 'room', roomRule.rule_id));
|
|
|
|
}
|
2016-08-17 19:26:37 +02:00
|
|
|
|
2016-08-18 16:21:46 +02:00
|
|
|
// add/replace an override rule to squelch everything in this room
|
|
|
|
// NB. We use the room ID as the name of this rule too, although this
|
|
|
|
// is an override rule, not a room rule: it still pertains to this room
|
|
|
|
// though, so using the room ID as the rule ID is logical and prevents
|
|
|
|
// duplicate copies of the rule.
|
2016-08-18 15:00:14 +02:00
|
|
|
promises.push(cli.addPushRule('global', 'override', roomId, {
|
|
|
|
conditions: [
|
|
|
|
{
|
|
|
|
kind: 'event_match',
|
|
|
|
key: 'room_id',
|
|
|
|
pattern: roomId,
|
2017-07-01 15:28:12 +02:00
|
|
|
},
|
2016-08-18 15:00:14 +02:00
|
|
|
],
|
|
|
|
actions: [
|
|
|
|
'dont_notify',
|
2017-07-01 15:28:12 +02:00
|
|
|
],
|
2016-08-18 15:00:14 +02:00
|
|
|
}));
|
|
|
|
|
2017-07-12 15:04:20 +02:00
|
|
|
return Promise.all(promises);
|
2016-08-18 15:00:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
function setRoomNotifsStateUnmuted(roomId, newState) {
|
|
|
|
const cli = MatrixClientPeg.get();
|
|
|
|
const promises = [];
|
|
|
|
|
|
|
|
const overrideMuteRule = findOverrideMuteRule(roomId);
|
|
|
|
if (overrideMuteRule) {
|
|
|
|
promises.push(cli.deletePushRule('global', 'override', overrideMuteRule.rule_id));
|
|
|
|
}
|
|
|
|
|
2017-07-01 15:28:12 +02:00
|
|
|
if (newState === 'all_messages') {
|
2016-08-18 17:59:25 +02:00
|
|
|
const roomRule = cli.getRoomPushRule('global', roomId);
|
|
|
|
if (roomRule) {
|
|
|
|
promises.push(cli.deletePushRule('global', 'room', roomRule.rule_id));
|
|
|
|
}
|
2017-07-01 15:28:12 +02:00
|
|
|
} else if (newState === 'mentions_only') {
|
2016-08-18 15:00:14 +02:00
|
|
|
promises.push(cli.addPushRule('global', 'room', roomId, {
|
2016-08-17 19:26:37 +02:00
|
|
|
actions: [
|
|
|
|
'dont_notify',
|
2017-07-01 15:28:12 +02:00
|
|
|
],
|
2016-08-17 19:26:37 +02:00
|
|
|
}));
|
2016-08-18 15:00:14 +02:00
|
|
|
// https://matrix.org/jira/browse/SPEC-400
|
|
|
|
promises.push(cli.setPushRuleEnabled('global', 'room', roomId, true));
|
|
|
|
} else if ('all_messages_loud') {
|
|
|
|
promises.push(cli.addPushRule('global', 'room', roomId, {
|
|
|
|
actions: [
|
|
|
|
'notify',
|
|
|
|
{
|
|
|
|
set_tweak: 'sound',
|
|
|
|
value: 'default',
|
2017-07-01 15:28:12 +02:00
|
|
|
},
|
|
|
|
],
|
2016-08-18 15:00:14 +02:00
|
|
|
}));
|
|
|
|
// https://matrix.org/jira/browse/SPEC-400
|
|
|
|
promises.push(cli.setPushRuleEnabled('global', 'room', roomId, true));
|
2016-08-17 19:26:37 +02:00
|
|
|
}
|
|
|
|
|
2017-07-12 15:04:20 +02:00
|
|
|
return Promise.all(promises);
|
2016-08-17 19:26:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
function findOverrideMuteRule(roomId) {
|
2018-02-06 18:50:53 +01:00
|
|
|
if (!MatrixClientPeg.get().pushRules ||
|
|
|
|
!MatrixClientPeg.get().pushRules['global'] ||
|
|
|
|
!MatrixClientPeg.get().pushRules['global'].override) {
|
|
|
|
return null;
|
|
|
|
}
|
2016-08-17 19:26:37 +02:00
|
|
|
for (const rule of MatrixClientPeg.get().pushRules['global'].override) {
|
|
|
|
if (isRuleForRoom(roomId, rule)) {
|
2016-08-18 15:00:14 +02:00
|
|
|
if (isMuteRule(rule) && rule.enabled) {
|
2016-08-17 19:26:37 +02:00
|
|
|
return rule;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
function isRuleForRoom(roomId, rule) {
|
|
|
|
if (rule.conditions.length !== 1) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
const cond = rule.conditions[0];
|
2017-07-01 15:28:12 +02:00
|
|
|
return (cond.kind === 'event_match' && cond.key === 'room_id' && cond.pattern === roomId);
|
2016-08-17 19:26:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
function isMuteRule(rule) {
|
2017-07-01 15:28:12 +02:00
|
|
|
return (rule.actions.length === 1 && rule.actions[0] === 'dont_notify');
|
2016-08-17 19:26:37 +02:00
|
|
|
}
|