194 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
		
		
			
		
	
	
			194 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| 
								 | 
							
								var GNTP = require('./gntp.js');
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Interface for registering Growl applications and sending Growl notifications.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * @api private
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								function Growly() {
							 | 
						||
| 
								 | 
							
								    this.appname = 'Growly';
							 | 
						||
| 
								 | 
							
								    this.notifications = undefined;
							 | 
						||
| 
								 | 
							
								    this.labels = undefined;
							 | 
						||
| 
								 | 
							
								    this.count = 0;
							 | 
						||
| 
								 | 
							
								    this.registered = false;
							 | 
						||
| 
								 | 
							
								    this.host = undefined;
							 | 
						||
| 
								 | 
							
								    this.port = undefined;
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Returns an array of label strings extracted from each notification object in
							 | 
						||
| 
								 | 
							
								 * `Growly.notifications`.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * @param {Array} notifications
							 | 
						||
| 
								 | 
							
								 * @return {Array} notification labels
							 | 
						||
| 
								 | 
							
								 * @api private
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								Growly.prototype.getLabels = function() {
							 | 
						||
| 
								 | 
							
								    return this.notifications.map(function(notif) {
							 | 
						||
| 
								 | 
							
								        return notif.label;
							 | 
						||
| 
								 | 
							
								    });
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Set the host to be used by GNTP requests.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * @param {String} host
							 | 
						||
| 
								 | 
							
								 * @param {Number} port
							 | 
						||
| 
								 | 
							
								 * @api public
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								Growly.prototype.setHost = function(host, port) {
							 | 
						||
| 
								 | 
							
								    this.host = host;
							 | 
						||
| 
								 | 
							
								    this.port = port;
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Register an application with the name `appname` (required), icon `appicon`, and
							 | 
						||
| 
								 | 
							
								 * a list of notification types `notifications`. If provided, `callback` will be
							 | 
						||
| 
								 | 
							
								 * called when the request completes with the first argument being an `err` error
							 | 
						||
| 
								 | 
							
								 * object if the request failed.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Each object in the `notifications` array defines a type of notification the
							 | 
						||
| 
								 | 
							
								 * application will have with the following properties:
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 *  - `.label` name used to identify the type of notification being used (required)
							 | 
						||
| 
								 | 
							
								 *  - `.dispname` name users will see in Growl's preference panel (defaults to `.label`)
							 | 
						||
| 
								 | 
							
								 *  - `.enabled` whether or not notifications of this type are enabled (defaults to true)
							 | 
						||
| 
								 | 
							
								 *  - `.icon` default icon notifications of this type should use (url, file path, or Buffer object)
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 *  Example registration:
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 *      growl.register('My Application', 'path/to/icon.png', [
							 | 
						||
| 
								 | 
							
								 *          { label: 'success', dispname: 'Success', icon: 'path/to/success.png' },
							 | 
						||
| 
								 | 
							
								 *          { label: 'warning', dispname: 'Warning', icon: 'path/to/warning.png', enabled: false }
							 | 
						||
| 
								 | 
							
								 *      ], function(err) { console.log(err || 'Registration successful!'); });
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * @param {String} appname
							 | 
						||
| 
								 | 
							
								 * @param {String|Buffer} appicon
							 | 
						||
| 
								 | 
							
								 * @param {Array} notifications
							 | 
						||
| 
								 | 
							
								 * @param {Function} callback
							 | 
						||
| 
								 | 
							
								 * @api public
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								Growly.prototype.register = function(appname, appicon, notifications, callback) {
							 | 
						||
| 
								 | 
							
								    var gntp;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    if (typeof appicon === 'object') {
							 | 
						||
| 
								 | 
							
								        notifications = appicon;
							 | 
						||
| 
								 | 
							
								        appicon = undefined;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    if (notifications === undefined || !notifications.length) {
							 | 
						||
| 
								 | 
							
								        notifications = [{ label: 'default', dispname: 'Default Notification', enabled: true }];
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    if (typeof arguments[arguments.length - 1] === 'function') {
							 | 
						||
| 
								 | 
							
								        callback = arguments[arguments.length - 1];
							 | 
						||
| 
								 | 
							
								    } else {
							 | 
						||
| 
								 | 
							
								        callback = function() {};
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    this.appname = appname;
							 | 
						||
| 
								 | 
							
								    this.notifications = notifications;
							 | 
						||
| 
								 | 
							
								    this.labels = this.getLabels();
							 | 
						||
| 
								 | 
							
								    this.registered = true;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    gntp = new GNTP('REGISTER', { host: this.host, port: this.port });
							 | 
						||
| 
								 | 
							
								    gntp.add('Application-Name', appname);
							 | 
						||
| 
								 | 
							
								    gntp.add('Application-Icon', appicon);
							 | 
						||
| 
								 | 
							
								    gntp.add('Notifications-Count', notifications.length);
							 | 
						||
| 
								 | 
							
								    gntp.newline();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    notifications.forEach(function(notif) {
							 | 
						||
| 
								 | 
							
								        if (notif.enabled === undefined) notif.enabled = true;
							 | 
						||
| 
								 | 
							
								        gntp.add('Notification-Name', notif.label);
							 | 
						||
| 
								 | 
							
								        gntp.add('Notification-Display-Name', notif.dispname);
							 | 
						||
| 
								 | 
							
								        gntp.add('Notification-Enabled', notif.enabled ? 'True' : 'False');
							 | 
						||
| 
								 | 
							
								        gntp.add('Notification-Icon', notif.icon);
							 | 
						||
| 
								 | 
							
								        gntp.newline();
							 | 
						||
| 
								 | 
							
								    });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    gntp.send(callback);
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Send a notification with `text` content. Growly will lazily register itself
							 | 
						||
| 
								 | 
							
								 * if the user hasn't already before sending the notification.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * A notification can have the following `opts` options:
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 *  - `.label` type of notification to use (defaults to the first registered type)
							 | 
						||
| 
								 | 
							
								 *  - `.title` title of the notification
							 | 
						||
| 
								 | 
							
								 *  - `.icon` url, file path, or Buffer instance for the notification's icon.
							 | 
						||
| 
								 | 
							
								 *  - `.sticky` whether or not to sticky the notification (defaults to false)
							 | 
						||
| 
								 | 
							
								 *  - `.priority` the priority of the notification from lowest (-2) to highest (2)
							 | 
						||
| 
								 | 
							
								 *  - `.coalescingId` replace/update the matching previous notification. May be ignored.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * If provided, `callback` will be called when the user interacts with the notification.
							 | 
						||
| 
								 | 
							
								 * The first argument will be an `err` error object, and the second argument an `action`
							 | 
						||
| 
								 | 
							
								 * string equal to either 'clicked' or 'closed' (whichever action the user took.)
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Example notification:
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 *     growl.notify('Stuffs broken!', { label: 'warning' }, function(err, action) {
							 | 
						||
| 
								 | 
							
								 *         console.log('Action:', action);
							 | 
						||
| 
								 | 
							
								 *     });
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * @param {String} text
							 | 
						||
| 
								 | 
							
								 * @param {Object} opts
							 | 
						||
| 
								 | 
							
								 * @param {Function} callback
							 | 
						||
| 
								 | 
							
								 * @api public
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								Growly.prototype.notify = function(text, opts, callback) {
							 | 
						||
| 
								 | 
							
								    var self = this,
							 | 
						||
| 
								 | 
							
								        gntp;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /* Lazy registration. */
							 | 
						||
| 
								 | 
							
								    if (!this.registered) {
							 | 
						||
| 
								 | 
							
								        this.register(this.appname, function(err) {
							 | 
						||
| 
								 | 
							
								            if (err) console.log(err);
							 | 
						||
| 
								 | 
							
								            self.notify.call(self, text, opts, callback);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								        return;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    opts = opts || {};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    if (typeof opts === 'function') {
							 | 
						||
| 
								 | 
							
								        callback = opts;
							 | 
						||
| 
								 | 
							
								        opts = {};
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    gntp = new GNTP('NOTIFY', { host: this.host, port: this.port });
							 | 
						||
| 
								 | 
							
								    gntp.add('Application-Name', this.appname);
							 | 
						||
| 
								 | 
							
								    gntp.add('Notification-Name', opts.label || this.labels[0]);
							 | 
						||
| 
								 | 
							
								    gntp.add('Notification-ID', ++this.count);
							 | 
						||
| 
								 | 
							
								    gntp.add('Notification-Title', opts.title);
							 | 
						||
| 
								 | 
							
								    gntp.add('Notification-Text', text);
							 | 
						||
| 
								 | 
							
								    gntp.add('Notification-Sticky', opts.sticky ? 'True' : 'False');
							 | 
						||
| 
								 | 
							
								    gntp.add('Notification-Priority', opts.priority);
							 | 
						||
| 
								 | 
							
								    gntp.add('Notification-Icon', opts.icon);
							 | 
						||
| 
								 | 
							
								    gntp.add('Notification-Coalescing-ID', opts.coalescingId || undefined);
							 | 
						||
| 
								 | 
							
								    gntp.add('Notification-Callback-Context', callback ? 'context' : undefined);
							 | 
						||
| 
								 | 
							
								    gntp.add('Notification-Callback-Context-Type', callback ? 'string' : undefined);
							 | 
						||
| 
								 | 
							
								    gntp.add('Notification-Callback-Target', undefined);
							 | 
						||
| 
								 | 
							
								    gntp.newline();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    gntp.send(function(err, resp) {
							 | 
						||
| 
								 | 
							
								        if (callback && err) {
							 | 
						||
| 
								 | 
							
								            callback(err);
							 | 
						||
| 
								 | 
							
								        } else if (callback && resp.state === 'CALLBACK') {
							 | 
						||
| 
								 | 
							
								            callback(undefined, resp['Notification-Callback-Result'].toLowerCase());
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    });
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Expose an instance of the Growly object.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								module.exports = new Growly();
							 |