185 lines
5.7 KiB
JavaScript
185 lines
5.7 KiB
JavaScript
/* global Signal:false */
|
||
/* global Backbone: false */
|
||
|
||
/* global drawAttention: false */
|
||
/* global i18n: false */
|
||
/* global isFocused: false */
|
||
/* global Signal: false */
|
||
/* global storage: false */
|
||
/* global Whisper: false */
|
||
/* global _: false */
|
||
|
||
// eslint-disable-next-line func-names
|
||
(function() {
|
||
'use strict';
|
||
|
||
window.Whisper = window.Whisper || {};
|
||
const { Settings } = Signal.Types;
|
||
|
||
const SettingNames = {
|
||
COUNT: 'count',
|
||
NAME: 'name',
|
||
MESSAGE: 'message',
|
||
};
|
||
|
||
function filter(text) {
|
||
return (text || '')
|
||
.replace(/&/g, '&')
|
||
.replace(/"/g, '"')
|
||
.replace(/'/g, ''')
|
||
.replace(/</g, '<')
|
||
.replace(/>/g, '>');
|
||
}
|
||
|
||
Whisper.Notifications = new (Backbone.Collection.extend({
|
||
initialize() {
|
||
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);
|
||
},
|
||
update() {
|
||
if (this.lastNotification) {
|
||
this.lastNotification.close();
|
||
this.lastNotification = null;
|
||
}
|
||
|
||
const { isEnabled } = this;
|
||
const isAppFocused = isFocused();
|
||
const isAudioNotificationEnabled =
|
||
storage.get('audio-notification') || false;
|
||
const isAudioNotificationSupported = Settings.isAudioNotificationSupported();
|
||
const numNotifications = this.length;
|
||
const userSetting = this.getUserSetting();
|
||
|
||
const status = Signal.Notifications.getStatus({
|
||
isAppFocused,
|
||
isAudioNotificationEnabled,
|
||
isAudioNotificationSupported,
|
||
isEnabled,
|
||
numNotifications,
|
||
userSetting,
|
||
});
|
||
|
||
if (status.type !== 'ok') {
|
||
if (status.shouldClearNotifications) {
|
||
this.reset([]);
|
||
}
|
||
|
||
return;
|
||
}
|
||
|
||
let title;
|
||
let message;
|
||
let iconUrl;
|
||
|
||
// 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();
|
||
switch (userSetting) {
|
||
case SettingNames.COUNT:
|
||
title = 'Signal';
|
||
message = newMessageCountLabel;
|
||
break;
|
||
case SettingNames.NAME: {
|
||
const lastMessageTitle = last.title;
|
||
title = newMessageCountLabel;
|
||
// eslint-disable-next-line prefer-destructuring
|
||
iconUrl = last.iconUrl;
|
||
if (numNotifications === 1) {
|
||
message = `${i18n('notificationFrom')} ${lastMessageTitle}`;
|
||
} else {
|
||
message = `${i18n(
|
||
'notificationMostRecentFrom'
|
||
)} ${lastMessageTitle}`;
|
||
}
|
||
break;
|
||
}
|
||
case SettingNames.MESSAGE:
|
||
if (numNotifications === 1) {
|
||
// eslint-disable-next-line prefer-destructuring
|
||
title = last.title;
|
||
// eslint-disable-next-line prefer-destructuring
|
||
message = last.message;
|
||
} else {
|
||
title = newMessageCountLabel;
|
||
message = `${i18n('notificationMostRecent')} ${last.message}`;
|
||
}
|
||
// eslint-disable-next-line prefer-destructuring
|
||
iconUrl = last.iconUrl;
|
||
break;
|
||
default:
|
||
window.log.error(
|
||
`Error: Unknown user notification setting: '${userSetting}'`
|
||
);
|
||
break;
|
||
}
|
||
|
||
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 we’ll
|
||
// 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.
|
||
},
|
||
getUserSetting() {
|
||
return storage.get('notification-setting') || SettingNames.MESSAGE;
|
||
},
|
||
onRemove() {
|
||
window.log.info('Remove notification');
|
||
this.update();
|
||
},
|
||
clear() {
|
||
window.log.info('Remove all notifications');
|
||
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();
|
||
},
|
||
enable() {
|
||
const needUpdate = !this.isEnabled;
|
||
this.isEnabled = true;
|
||
if (needUpdate) {
|
||
this.update();
|
||
}
|
||
},
|
||
disable() {
|
||
this.isEnabled = false;
|
||
},
|
||
}))();
|
||
})();
|