2017-03-21 11:15:18 -05:00
|
|
|
'use strict';
|
|
|
|
|
|
|
|
const EventEmitter = require('events');
|
|
|
|
const { notify } = require('./electron-notify.js');
|
2017-05-31 23:39:08 -05:00
|
|
|
const log = require('../log.js');
|
|
|
|
const logLevels = require('../enums/logLevels.js');
|
2017-03-21 11:15:18 -05:00
|
|
|
/**
|
|
|
|
* implementation for notifications interface,
|
|
|
|
* wrapper around electron-notify.
|
|
|
|
*/
|
|
|
|
class Notify {
|
2017-04-03 18:50:35 -05:00
|
|
|
/**
|
|
|
|
* Dislays a notifications
|
|
|
|
*
|
|
|
|
* @param {String} title Title of notification
|
|
|
|
* @param {Object} options {
|
|
|
|
* body {string} main text to display in notifications
|
|
|
|
* image {string} url of image to show in notification
|
|
|
|
* icon {string} url of image to show in notification
|
|
|
|
* flash {bool} true if notification should flash (default false)
|
|
|
|
* color {string} background color for notification
|
|
|
|
* tag {string} non-empty string to unique identify notf, if another
|
|
|
|
* notification arrives with same tag then it's content will
|
|
|
|
* replace existing notification.
|
|
|
|
* sticky {bool} if true notification will stay until user closes. default
|
|
|
|
* is false.
|
|
|
|
* data {object} arbitrary object to be stored with notification
|
|
|
|
* }
|
|
|
|
*/
|
2017-03-21 11:15:18 -05:00
|
|
|
constructor(title, options) {
|
2017-05-31 23:39:08 -05:00
|
|
|
log.send(logLevels.INFO, 'creating notification, text=' + options.body);
|
|
|
|
|
2017-04-06 12:07:58 -05:00
|
|
|
let emitter = new EventEmitter();
|
|
|
|
this.emitter = Queue(emitter);
|
2017-03-21 11:15:18 -05:00
|
|
|
|
|
|
|
this._id = notify({
|
|
|
|
title: title,
|
|
|
|
text: options.body,
|
2017-03-28 19:04:21 -05:00
|
|
|
image: options.image || options.icon,
|
2017-03-21 11:15:18 -05:00
|
|
|
flash: options.flash,
|
|
|
|
color: options.color,
|
2017-03-29 22:11:08 -05:00
|
|
|
tag: options.tag,
|
|
|
|
sticky: options.sticky || false,
|
2017-03-21 11:15:18 -05:00
|
|
|
onShowFunc: onShow.bind(this),
|
|
|
|
onClickFunc: onClick.bind(this),
|
2017-03-28 20:09:51 -05:00
|
|
|
onCloseFunc: onClose.bind(this),
|
|
|
|
onErrorFunc: onError.bind(this)
|
2017-03-21 11:15:18 -05:00
|
|
|
});
|
|
|
|
|
2017-05-31 23:39:08 -05:00
|
|
|
log.send(logLevels.INFO, 'created notification, id=' + this._id + ', text=' + options.body);
|
|
|
|
|
2017-03-28 19:04:21 -05:00
|
|
|
this._data = options.data || null;
|
|
|
|
|
2017-08-24 05:48:32 -05:00
|
|
|
/**
|
|
|
|
* Handles on show event
|
|
|
|
* @param arg
|
|
|
|
*/
|
2017-03-21 11:15:18 -05:00
|
|
|
function onShow(arg) {
|
|
|
|
if (arg.id === this._id) {
|
2017-05-31 23:39:08 -05:00
|
|
|
log.send(logLevels.INFO, 'showing notification, id=' + this._id);
|
2017-04-06 12:07:58 -05:00
|
|
|
this.emitter.queue('show', {
|
|
|
|
target: this
|
|
|
|
});
|
2017-03-21 11:15:18 -05:00
|
|
|
this._closeNotification = arg.closeNotification;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-24 05:48:32 -05:00
|
|
|
/**
|
|
|
|
* Handles on click event
|
|
|
|
* @param arg
|
|
|
|
*/
|
2017-03-21 11:15:18 -05:00
|
|
|
function onClick(arg) {
|
|
|
|
if (arg.id === this._id) {
|
2017-05-31 23:39:08 -05:00
|
|
|
log.send(logLevels.INFO, 'clicking notification, id=' + this._id);
|
2017-04-06 12:07:58 -05:00
|
|
|
this.emitter.queue('click', {
|
2017-04-03 18:50:35 -05:00
|
|
|
target: this
|
|
|
|
});
|
2017-03-21 11:15:18 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-24 05:48:32 -05:00
|
|
|
/**
|
|
|
|
* Handles on close event
|
|
|
|
* @param arg
|
|
|
|
*/
|
2017-03-21 11:15:18 -05:00
|
|
|
function onClose(arg) {
|
|
|
|
if (arg.id === this._id || arg.event === 'close-all') {
|
2017-05-31 23:39:08 -05:00
|
|
|
log.send(logLevels.INFO, 'closing notification, id=' + this._id);
|
2017-04-06 12:07:58 -05:00
|
|
|
this.emitter.queue('close', {
|
2017-04-03 18:50:35 -05:00
|
|
|
target: this
|
|
|
|
});
|
2017-03-21 11:15:18 -05:00
|
|
|
this.destroy();
|
|
|
|
}
|
|
|
|
}
|
2017-03-28 20:09:51 -05:00
|
|
|
|
2017-08-24 05:48:32 -05:00
|
|
|
/**
|
|
|
|
* Handles on error event
|
|
|
|
* @param arg
|
|
|
|
*/
|
2017-03-28 20:09:51 -05:00
|
|
|
function onError(arg) {
|
|
|
|
if (arg.id === this._id) {
|
|
|
|
// don't raise error event if handler doesn't exist, node
|
|
|
|
// will throw an exception
|
2017-05-31 23:39:08 -05:00
|
|
|
log.send(logLevels.ERROR, 'error for notification, id=' + this._id +
|
|
|
|
' error=' + (arg && arg.error));
|
2017-03-28 20:09:51 -05:00
|
|
|
if (this.emitter.eventNames().includes('error')) {
|
2017-04-06 12:07:58 -05:00
|
|
|
this.emitter.queue('error', arg.error || 'notification error');
|
2017-03-28 20:09:51 -05:00
|
|
|
}
|
|
|
|
this.destroy();
|
|
|
|
}
|
|
|
|
}
|
2017-03-21 11:15:18 -05:00
|
|
|
}
|
|
|
|
|
2017-04-03 18:50:35 -05:00
|
|
|
/**
|
2017-08-24 05:48:32 -05:00
|
|
|
* Closes notification
|
2017-04-03 18:50:35 -05:00
|
|
|
*/
|
2017-03-21 11:15:18 -05:00
|
|
|
close() {
|
|
|
|
if (typeof this._closeNotification === 'function') {
|
|
|
|
this._closeNotification('close');
|
|
|
|
}
|
|
|
|
this.destroy();
|
|
|
|
}
|
|
|
|
|
2017-04-03 18:50:35 -05:00
|
|
|
/**
|
2017-08-24 05:48:32 -05:00
|
|
|
* Always allow showing notifications.
|
2017-04-03 18:50:35 -05:00
|
|
|
* @return {string} 'granted'
|
|
|
|
*/
|
2017-03-21 11:15:18 -05:00
|
|
|
static get permission() {
|
|
|
|
return 'granted';
|
|
|
|
}
|
|
|
|
|
2017-04-03 18:50:35 -05:00
|
|
|
/**
|
2017-08-24 05:48:32 -05:00
|
|
|
* Returns data object passed in via constructor options
|
2017-04-03 18:50:35 -05:00
|
|
|
*/
|
2017-03-28 19:04:21 -05:00
|
|
|
get data() {
|
|
|
|
return this._data;
|
|
|
|
}
|
|
|
|
|
2017-04-03 18:50:35 -05:00
|
|
|
/**
|
2017-08-24 05:48:32 -05:00
|
|
|
* Adds event listeners for 'click', 'close', 'show', 'error' events
|
2017-04-03 18:50:35 -05:00
|
|
|
*
|
|
|
|
* @param {String} event event to listen for
|
|
|
|
* @param {func} cb callback invoked when event occurs
|
|
|
|
*/
|
2017-03-21 11:15:18 -05:00
|
|
|
addEventListener(event, cb) {
|
|
|
|
if (event && typeof cb === 'function') {
|
|
|
|
this.emitter.on(event, cb);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-03 18:50:35 -05:00
|
|
|
/**
|
2017-08-24 05:48:32 -05:00
|
|
|
* Removes event listeners for 'click', 'close', 'show', 'error' events
|
2017-04-03 18:50:35 -05:00
|
|
|
*
|
|
|
|
* @param {String} event event to stop listening for.
|
|
|
|
* @param {func} cb callback associated with original addEventListener
|
|
|
|
*/
|
2017-03-21 11:15:18 -05:00
|
|
|
removeEventListener(event, cb) {
|
|
|
|
if (event && typeof cb === 'function') {
|
2017-03-29 19:42:15 -05:00
|
|
|
this.emitter.removeListener(event, cb);
|
2017-03-21 11:15:18 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-06 12:07:58 -05:00
|
|
|
/**
|
2017-08-24 05:48:32 -05:00
|
|
|
* Removes all event listeners
|
2017-04-06 12:07:58 -05:00
|
|
|
*/
|
|
|
|
removeAllEvents() {
|
|
|
|
this.destroy();
|
|
|
|
}
|
|
|
|
|
2017-03-21 11:15:18 -05:00
|
|
|
//
|
|
|
|
// private stuff below here
|
|
|
|
//
|
2017-03-23 18:19:59 -05:00
|
|
|
destroy() {
|
2017-03-30 18:20:51 -05:00
|
|
|
this.emitter.removeAllListeners();
|
2017-03-23 18:19:59 -05:00
|
|
|
}
|
|
|
|
|
2017-03-21 11:15:18 -05:00
|
|
|
}
|
|
|
|
|
2017-04-06 12:07:58 -05:00
|
|
|
/**
|
|
|
|
* Allow emitter events to be queued before addEventListener called.
|
|
|
|
* Code adapted from: https://github.com/bredele/emitter-queue
|
|
|
|
*
|
|
|
|
* @param {Object} emitter Instance of node emitter that will get augmented.
|
|
|
|
* @return {Object} Modified emitter
|
|
|
|
*/
|
|
|
|
function Queue(emitter) {
|
|
|
|
/**
|
2017-08-24 02:51:02 -05:00
|
|
|
* Cache emitter on.
|
|
|
|
* @api private
|
|
|
|
*/
|
|
|
|
const cache = emitter.on;
|
2017-04-06 12:07:58 -05:00
|
|
|
let modifiedEmitter = emitter;
|
|
|
|
/**
|
|
|
|
* Emit event and store it if no
|
|
|
|
* defined callbacks.
|
|
|
|
* example:
|
|
|
|
*
|
|
|
|
* .queue('message', 'hi');
|
|
|
|
*
|
2017-08-24 02:51:02 -05:00
|
|
|
* @param {String} topic
|
2017-04-06 12:07:58 -05:00
|
|
|
*/
|
|
|
|
modifiedEmitter.queue = function(topic) {
|
|
|
|
this._queue = this._queue || {};
|
|
|
|
this._callbacks = this._callbacks || {};
|
|
|
|
if (this._callbacks[topic]) {
|
|
|
|
this.emit.apply(this, arguments);
|
|
|
|
} else {
|
|
|
|
(this._queue[topic] = this._queue[topic] || [])
|
|
|
|
.push([].slice.call(arguments, 1));
|
|
|
|
}
|
2017-08-24 02:51:02 -05:00
|
|
|
};
|
2017-04-06 12:07:58 -05:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Listen on the given `event` with `fn`.
|
|
|
|
*
|
|
|
|
* @param {String} event
|
|
|
|
* @param {Function} fn
|
2017-08-24 02:51:02 -05:00
|
|
|
* @return {Event}
|
2017-04-06 12:07:58 -05:00
|
|
|
*/
|
|
|
|
modifiedEmitter.on = modifiedEmitter.addEventListener = function(topic, fn) {
|
|
|
|
this._queue = this._queue || {};
|
2017-08-24 02:51:02 -05:00
|
|
|
const topics = this._queue[topic];
|
2017-04-06 12:07:58 -05:00
|
|
|
cache.apply(this, arguments);
|
|
|
|
|
|
|
|
if (!this._callbacks) {
|
|
|
|
this._callbacks = {};
|
|
|
|
}
|
|
|
|
this._callbacks[topic] = true;
|
|
|
|
|
|
|
|
if (topics) {
|
2017-08-24 02:51:02 -05:00
|
|
|
let i = 0;
|
|
|
|
const l = topics.length;
|
|
|
|
for(; i < l; i++) {
|
2017-04-06 12:07:58 -05:00
|
|
|
fn.apply(this, topics[i]);
|
|
|
|
}
|
|
|
|
delete this._queue[topic];
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
return modifiedEmitter;
|
|
|
|
}
|
|
|
|
|
2017-03-21 11:15:18 -05:00
|
|
|
module.exports = Notify;
|