signal-desktop/js/notifications.js

212 lines
6.6 KiB
JavaScript
Raw Normal View History

/* global Signal:false */
2018-05-02 20:43:32 +00:00
/* global Backbone: false */
/* global drawAttention: false */
2018-05-02 20:43:32 +00:00
/* global i18n: false */
/* global Signal: false */
/* global storage: false */
/* global Whisper: false */
/* global _: false */
2018-05-02 20:43:32 +00:00
// eslint-disable-next-line func-names
2018-04-27 21:25:04 +00:00
(function() {
'use strict';
2018-05-02 20:43:32 +00:00
2018-04-27 21:25:04 +00:00
window.Whisper = window.Whisper || {};
2018-05-02 20:43:32 +00:00
const { Settings } = Signal.Types;
2018-05-02 22:02:49 +00:00
const SettingNames = {
2018-04-27 21:25:04 +00:00
COUNT: 'count',
NAME: 'name',
MESSAGE: 'message',
};
function filter(text) {
return (text || '')
.replace(/&/g, '&')
.replace(/"/g, '"')
.replace(/'/g, ''')
.replace(/</g, '&lt;')
.replace(/>/g, '&gt;');
}
2018-04-27 21:25:04 +00:00
Whisper.Notifications = new (Backbone.Collection.extend({
2018-05-02 20:43:32 +00:00
initialize() {
2018-04-27 21:25:04 +00:00
this.isEnabled = false;
this.on('add', this.update);
this.on('remove', this.onRemove);
this.lastNotification = null;
// Testing indicated that trying to create/destroy notifications too quickly
// resulted in notifications that stuck around forever, requiring the user
// to manually close them. This introduces a minimum amount of time between calls,
// and batches up the quick successive update() calls we get from an incoming
// read sync, which might have a number of messages referenced inside of it.
this.fastUpdate = this.update;
this.update = _.debounce(this.update, 1000);
2018-04-27 21:25:04 +00:00
},
2018-05-02 20:43:32 +00:00
update() {
if (this.lastNotification) {
this.lastNotification.close();
this.lastNotification = null;
}
2018-04-27 21:25:04 +00:00
const { isEnabled } = this;
const isAppFocused = window.isActive();
2018-04-27 21:25:04 +00:00
const isAudioNotificationEnabled =
storage.get('audio-notification') || false;
const isAudioNotificationSupported = Settings.isAudioNotificationSupported();
const numNotifications = this.length;
2018-05-02 22:04:00 +00:00
const userSetting = this.getUserSetting();
const status = Signal.Notifications.getStatus({
isAppFocused,
isAudioNotificationEnabled,
isAudioNotificationSupported,
isEnabled,
2018-04-27 21:25:04 +00:00
numNotifications,
2018-05-02 22:04:00 +00:00
userSetting,
2018-04-27 21:25:04 +00:00
});
2018-02-23 21:25:11 +00:00
2018-05-02 22:04:00 +00:00
if (status.type !== 'ok') {
if (status.shouldClearNotifications) {
this.reset([]);
2018-05-02 22:04:00 +00:00
}
2018-04-27 21:25:04 +00:00
return;
}
2018-05-02 20:43:32 +00:00
let title;
let message;
let iconUrl;
2018-04-27 21:25:04 +00:00
// NOTE: i18n has more complex rules for pluralization than just
// distinguishing between zero (0) and other (non-zero),
// e.g. Russian:
// http://docs.translatehouse.org/projects/localization-guide/en/latest/l10n/pluralforms.html
const newMessageCountLabel = `${numNotifications} ${
numNotifications === 1 ? i18n('newMessage') : i18n('newMessages')
}`;
const last = this.last().toJSON();
2018-05-02 22:04:00 +00:00
switch (userSetting) {
2018-05-02 22:02:49 +00:00
case SettingNames.COUNT:
2018-04-27 21:25:04 +00:00
title = 'Signal';
message = newMessageCountLabel;
2018-04-27 21:25:04 +00:00
break;
case SettingNames.NAME: {
const lastMessageTitle = last.title;
title = newMessageCountLabel;
// eslint-disable-next-line prefer-destructuring
iconUrl = last.iconUrl;
if (numNotifications === 1) {
2020-01-17 22:23:19 +00:00
if (last.reaction) {
message = i18n('notificationReaction', [
lastMessageTitle,
last.reaction.emoji,
]);
} else {
message = `${i18n('notificationFrom')} ${lastMessageTitle}`;
}
} else if (last.reaction) {
message = i18n('notificationReactionMostRecent', [
lastMessageTitle,
last.reaction.emoji,
]);
} else {
message = `${i18n(
'notificationMostRecentFrom'
)} ${lastMessageTitle}`;
}
2018-04-27 21:25:04 +00:00
break;
}
2018-05-02 22:02:49 +00:00
case SettingNames.MESSAGE:
2018-04-27 21:25:04 +00:00
if (numNotifications === 1) {
// eslint-disable-next-line prefer-destructuring
title = last.title;
2020-01-17 22:23:19 +00:00
if (last.reaction) {
message = i18n('notificationReactionMessage', [
2020-01-17 22:23:19 +00:00
last.title,
last.reaction.emoji,
last.message,
2020-01-17 22:23:19 +00:00
]);
} else {
// eslint-disable-next-line prefer-destructuring
message = last.message;
}
} else if (last.reaction) {
title = newMessageCountLabel;
message = i18n('notificationReactionMessageMostRecent', [
2020-01-17 22:23:19 +00:00
last.title,
last.reaction.emoji,
last.message,
2020-01-17 22:23:19 +00:00
]);
2018-04-27 21:25:04 +00:00
} else {
title = newMessageCountLabel;
message = `${i18n('notificationMostRecent')} ${last.message}`;
2018-04-27 21:25:04 +00:00
}
// eslint-disable-next-line prefer-destructuring
iconUrl = last.iconUrl;
2018-04-27 21:25:04 +00:00
break;
2018-05-02 20:43:32 +00:00
default:
window.log.error(
`Error: Unknown user notification setting: '${userSetting}'`
);
2018-05-02 20:43:32 +00:00
break;
2018-04-27 21:25:04 +00:00
}
const shouldHideExpiringMessageBody =
last.isExpiringMessage && Signal.OS.isMacOS();
if (shouldHideExpiringMessageBody) {
message = i18n('newMessage');
}
drawAttention();
this.lastNotification = new Notification(title, {
body: window.platform === 'linux' ? filter(message) : message,
icon: iconUrl,
silent: !status.shouldPlayNotificationSound,
});
this.lastNotification.onclick = () =>
this.trigger('click', last.conversationId, last.messageId);
// We continue to build up more and more messages for our notifications
// until the user comes back to our app or closes the app. Then well
// clear everything out. The good news is that we'll have a maximum of
// 1 notification in the Notification area (something like
// 10 new messages) assuming that `Notification::close` does its job.
2018-04-27 21:25:04 +00:00
},
2018-05-02 22:04:00 +00:00
getUserSetting() {
2018-05-02 22:02:49 +00:00
return storage.get('notification-setting') || SettingNames.MESSAGE;
2018-04-27 21:25:04 +00:00
},
2018-05-02 20:43:32 +00:00
onRemove() {
window.log.info('Remove notification');
this.update();
2018-04-27 21:25:04 +00:00
},
2018-05-02 20:43:32 +00:00
clear() {
window.log.info('Remove all notifications');
2018-04-27 21:25:04 +00:00
this.reset([]);
this.update();
},
// We don't usually call this, but when the process is shutting down, we should at
// least try to remove the notification immediately instead of waiting for the
// normal debounce.
fastClear() {
this.reset([]);
this.fastUpdate();
2018-04-27 21:25:04 +00:00
},
2018-05-02 20:43:32 +00:00
enable() {
2018-04-27 21:25:04 +00:00
const needUpdate = !this.isEnabled;
this.isEnabled = true;
if (needUpdate) {
this.update();
}
},
2018-05-02 20:43:32 +00:00
disable() {
2018-04-27 21:25:04 +00:00
this.isEnabled = false;
},
}))();
})();