mirror of
https://github.com/moparisthebest/FireTray
synced 2025-01-09 12:38:04 -05:00
use nested timers to mimic sleep()
thread.processNextEvent() is prohibited in Firefox (though used in mozmill/util.js). Nested timers are a bit twisted and ugly. Need to check |yield| and generators: http://unixpapa.com/js/sleep.html, dactyl/base.jsm (K. Maglione)
This commit is contained in:
parent
c4c5a3425b
commit
5882e3b9c5
@ -15,7 +15,7 @@ let log = firetray.Logging.getLogger("firetray.Chat");
|
|||||||
firetray.Chat = {
|
firetray.Chat = {
|
||||||
initialized: false,
|
initialized: false,
|
||||||
observedTopics: {},
|
observedTopics: {},
|
||||||
shouldAcknowledgeConvs: {
|
shouldAcknowledgeConvs: { // TODO: FOUDIL: rename to convsToAcknoledge
|
||||||
ids: {},
|
ids: {},
|
||||||
length: function(){return Object.keys(this.ids).length;}
|
length: function(){return Object.keys(this.ids).length;}
|
||||||
},
|
},
|
||||||
@ -136,7 +136,8 @@ firetray.Chat = {
|
|||||||
if (this.shouldAcknowledgeConvs.length() > 1) return; // already calling attention
|
if (this.shouldAcknowledgeConvs.length() > 1) return; // already calling attention
|
||||||
|
|
||||||
this.setUrgencyMaybe(conv);
|
this.setUrgencyMaybe(conv);
|
||||||
firetray.ChatStatusIcon.startIconBlinking();
|
// firetray.ChatStatusIcon.startIconBlinking();
|
||||||
|
firetray.ChatStatusIcon.startCrossFade();
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -162,7 +163,8 @@ firetray.Chat = {
|
|||||||
if(this.shouldAcknowledgeConvs.length() === 0) {
|
if(this.shouldAcknowledgeConvs.length() === 0) {
|
||||||
log.debug("do stop icon blinking !!!");
|
log.debug("do stop icon blinking !!!");
|
||||||
firetray.ChatStatusIcon.setUrgency(xid, false);
|
firetray.ChatStatusIcon.setUrgency(xid, false);
|
||||||
firetray.ChatStatusIcon.stopIconBlinking();
|
// firetray.ChatStatusIcon.stopIconBlinking();
|
||||||
|
firetray.ChatStatusIcon.stopCrossFade();
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -40,6 +40,7 @@ firetray.ChatStatusIcon = {
|
|||||||
themedIconNameCurrent: null,
|
themedIconNameCurrent: null,
|
||||||
signals: {'focus-in': {callback: {}, handler: {}}},
|
signals: {'focus-in': {callback: {}, handler: {}}},
|
||||||
timers: {},
|
timers: {},
|
||||||
|
events: {},
|
||||||
|
|
||||||
init: function() {
|
init: function() {
|
||||||
if (!firetray.Handler.inMailApp) throw "ChatStatusIcon for mail app only";
|
if (!firetray.Handler.inMailApp) throw "ChatStatusIcon for mail app only";
|
||||||
@ -73,6 +74,11 @@ firetray.ChatStatusIcon = {
|
|||||||
},
|
},
|
||||||
|
|
||||||
setIconImageFromGIcon: function(gicon) {
|
setIconImageFromGIcon: function(gicon) {
|
||||||
|
if (firetray.Chat.shouldAcknowledgeConvs.length()) {
|
||||||
|
this.events['icon-changed'] = true;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (!firetray.ChatStatusIcon.trayIcon || !gicon)
|
if (!firetray.ChatStatusIcon.trayIcon || !gicon)
|
||||||
log.error("Icon missing");
|
log.error("Icon missing");
|
||||||
gtk.gtk_status_icon_set_from_gicon(firetray.ChatStatusIcon.trayIcon, gicon);
|
gtk.gtk_status_icon_set_from_gicon(firetray.ChatStatusIcon.trayIcon, gicon);
|
||||||
@ -91,89 +97,110 @@ firetray.ChatStatusIcon = {
|
|||||||
* EXPERIMENTAL fancy blinking.
|
* EXPERIMENTAL fancy blinking.
|
||||||
* TODO: how to wait for last fade in to restore themedIconNameCurrent
|
* TODO: how to wait for last fade in to restore themedIconNameCurrent
|
||||||
*/
|
*/
|
||||||
crossFade: function() {
|
startCrossFade: function() {
|
||||||
|
this.crossFade(this.buildPixBuf());
|
||||||
|
},
|
||||||
|
|
||||||
/* borrowed from mozmill utils.js*/
|
buildPixBuf: function() {
|
||||||
function sleep(milliseconds) {
|
let icon_theme = gtk.gtk_icon_theme_get_for_screen(gdk.gdk_screen_get_default());
|
||||||
var timeup = false;
|
|
||||||
function wait() { timeup = true; }
|
|
||||||
let timer = Components.classes["@mozilla.org/timer;1"]
|
|
||||||
.createInstance(Components.interfaces.nsITimer);
|
|
||||||
timer.initWithCallback(wait, milliseconds, Components.interfaces.nsITimer.TYPE_ONE_SHOT);
|
|
||||||
|
|
||||||
var thread = Components.classes["@mozilla.org/thread-manager;1"].
|
// get pixbuf
|
||||||
getService().currentThread;
|
let arry = gobject.gchar.ptr.array()(2);
|
||||||
while(!timeup) {
|
arry[0] = gobject.gchar.array()(firetray.ChatStatusIcon.themedIconNameCurrent);
|
||||||
thread.processNextEvent(true);
|
arry[1] = null;
|
||||||
|
log.debug("icon name="+firetray.ChatStatusIcon.themedIconNameCurrent+", theme="+icon_theme+", arry="+arry);
|
||||||
|
let icon_info = gtk.gtk_icon_theme_choose_icon(icon_theme, arry, 22, gtk.GTK_ICON_LOOKUP_FORCE_SIZE);
|
||||||
|
|
||||||
|
// create pixbuf
|
||||||
|
let pixbuf = gdk.gdk_pixbuf_copy(gtk.gtk_icon_info_load_icon(icon_info, null));
|
||||||
|
gtk.gtk_icon_info_free(icon_info); // gobject.g_object_unref(icon_info) in 3.8
|
||||||
|
|
||||||
|
// checks
|
||||||
|
if (gdk.gdk_pixbuf_get_colorspace(pixbuf) != gdk.GDK_COLORSPACE_RGB)
|
||||||
|
log.error("wrong colorspace for pixbuf");
|
||||||
|
if (gdk.gdk_pixbuf_get_bits_per_sample(pixbuf) != 8)
|
||||||
|
log.error("wrong bits_per_sample for pixbuf");
|
||||||
|
if (!gdk.gdk_pixbuf_get_has_alpha(pixbuf))
|
||||||
|
log.error("pixbuf doesn't have alpha");
|
||||||
|
let n_channels = gdk.gdk_pixbuf_get_n_channels(pixbuf);
|
||||||
|
if (n_channels != 4)
|
||||||
|
log.error("wrong nb of channels for pixbuf");
|
||||||
|
|
||||||
|
return pixbuf; // TO BE UNREFED WITH to g_object_unref() !!
|
||||||
|
},
|
||||||
|
|
||||||
|
crossFade: function(pixbuf) {
|
||||||
|
// init transform
|
||||||
|
let width = gdk.gdk_pixbuf_get_width(pixbuf);
|
||||||
|
let height = gdk.gdk_pixbuf_get_height(pixbuf);
|
||||||
|
log.debug("width="+width+", height="+height);
|
||||||
|
let n_channels = gdk.gdk_pixbuf_get_n_channels(pixbuf);
|
||||||
|
let length = width*height*n_channels;
|
||||||
|
let pixels = ctypes.cast(gdk.gdk_pixbuf_get_pixels(pixbuf),
|
||||||
|
gobject.guchar.array(length).ptr);
|
||||||
|
log.debug("pixels="+pixels);
|
||||||
|
|
||||||
|
// backup alpha for later fade-in
|
||||||
|
let buffer = new ArrayBuffer(width*height);
|
||||||
|
let alpha_bak = new Uint8Array(buffer);
|
||||||
|
for (let i=3; i<length; i+=n_channels)
|
||||||
|
alpha_bak[(i-3)/n_channels] = pixels.contents[i];
|
||||||
|
|
||||||
|
const ALPHA_STEP = 5;
|
||||||
|
let timers = [];
|
||||||
|
|
||||||
|
function fadeIn(alpha) {
|
||||||
|
for(let i=3; i<length; i+=n_channels)
|
||||||
|
if (pixels.contents[i]+ALPHA_STEP<=alpha_bak[(i-3)/n_channels]) {
|
||||||
|
pixels.contents[i] += ALPHA_STEP;
|
||||||
|
}
|
||||||
|
log.info("gtk_status_icon_set_from_pixbuf="+pixbuf);
|
||||||
|
gtk.gtk_status_icon_set_from_pixbuf(firetray.ChatStatusIcon.trayIcon, pixbuf);
|
||||||
|
|
||||||
|
if (alpha < 255) {
|
||||||
|
alpha += ALPHA_STEP;
|
||||||
|
timers.push(firetray.Utils.timer(10, Ci.nsITimer.TYPE_ONE_SHOT,
|
||||||
|
function(){fadeIn(alpha);}));
|
||||||
|
|
||||||
|
} else {
|
||||||
|
if (firetray.ChatStatusIcon.events['stop-cross-fade']) {
|
||||||
|
delete firetray.ChatStatusIcon.events['stop-cross-fade'];
|
||||||
|
firetray.ChatStatusIcon.setIconImage(firetray.ChatStatusIcon.themedIconNameCurrent);
|
||||||
|
return;
|
||||||
|
|
||||||
|
} else if (firetray.ChatStatusIcon.events['icon-changed']) {
|
||||||
|
delete firetray.ChatStatusIcon.events['icon-changed'];
|
||||||
|
firetray.ChatStatusIcon.crossFade(firetray.ChatStatusIcon.buildPixBuf());
|
||||||
|
return;
|
||||||
|
|
||||||
|
} else {
|
||||||
|
timers.push(firetray.Utils.timer(500, Ci.nsITimer.TYPE_ONE_SHOT,
|
||||||
|
function(){fadeOut(255);}));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let icon_theme = gtk.gtk_icon_theme_get_for_screen(gdk.gdk_screen_get_default());
|
function fadeOut(alpha) {
|
||||||
firetray.ChatStatusIcon.timers['cross-fade'] = firetray.Utils.timer(
|
for(let i=3; i<length; i+=n_channels)
|
||||||
500, Ci.nsITimer.TYPE_REPEATING_SLACK, function() {
|
if (pixels.contents[i]-ALPHA_STEP>0)
|
||||||
|
pixels.contents[i] -= ALPHA_STEP;
|
||||||
|
log.info("gtk_status_icon_set_from_pixbuf="+pixbuf);
|
||||||
|
gtk.gtk_status_icon_set_from_pixbuf(firetray.ChatStatusIcon.trayIcon, pixbuf);
|
||||||
|
|
||||||
// get pixbuf
|
if (alpha > 0) {
|
||||||
let arry = gobject.gchar.ptr.array()(2);
|
alpha -= ALPHA_STEP;
|
||||||
arry[0] = gobject.gchar.array()(firetray.ChatStatusIcon.themedIconNameCurrent);
|
timers.push(firetray.Utils.timer(10, Ci.nsITimer.TYPE_ONE_SHOT,
|
||||||
arry[1] = null;
|
function(){fadeOut(alpha);}));
|
||||||
log.debug("theme="+icon_theme+", arry="+arry);
|
|
||||||
let icon_info = gtk.gtk_icon_theme_choose_icon(icon_theme, arry, 22, gtk.GTK_ICON_LOOKUP_FORCE_SIZE);
|
|
||||||
|
|
||||||
// create pixbuf
|
} else {
|
||||||
let pixbuf = gdk.gdk_pixbuf_copy(gtk.gtk_icon_info_load_icon(icon_info, null));
|
fadeIn(0);
|
||||||
gtk.gtk_icon_info_free(icon_info); // gobject.g_object_unref(icon_info) in 3.8
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// checks
|
fadeOut(255);
|
||||||
if (gdk.gdk_pixbuf_get_colorspace(pixbuf) != gdk.GDK_COLORSPACE_RGB)
|
|
||||||
log.error("wrong colorspace for pixbuf");
|
|
||||||
if (gdk.gdk_pixbuf_get_bits_per_sample(pixbuf) != 8)
|
|
||||||
log.error("wrong bits_per_sample for pixbuf");
|
|
||||||
if (!gdk.gdk_pixbuf_get_has_alpha(pixbuf))
|
|
||||||
log.error("pixbuf doesn't have alpha");
|
|
||||||
let n_channels = gdk.gdk_pixbuf_get_n_channels(pixbuf);
|
|
||||||
if (n_channels != 4)
|
|
||||||
log.error("wrong nb of channels for pixbuf");
|
|
||||||
|
|
||||||
// init transform
|
gobject.g_object_unref(pixbuf); // FIXME: not sure if this shouldn't be done at 'stop-cross-fade'
|
||||||
let width = gdk.gdk_pixbuf_get_width(pixbuf);
|
log.info("pixbuf unref'd");
|
||||||
let height = gdk.gdk_pixbuf_get_height(pixbuf);
|
|
||||||
log.warn("width="+width+", height="+height);
|
|
||||||
let rowstride = gdk.gdk_pixbuf_get_rowstride(pixbuf);
|
|
||||||
log.warn("rowstride="+rowstride);
|
|
||||||
let length = width*height*n_channels;
|
|
||||||
let pixels = ctypes.cast(gdk.gdk_pixbuf_get_pixels(pixbuf),
|
|
||||||
gobject.guchar.array(length).ptr);
|
|
||||||
log.warn("pixels="+pixels);
|
|
||||||
|
|
||||||
// backup alpha for later fade-in
|
|
||||||
let buffer = new ArrayBuffer(width*height);
|
|
||||||
let alpha_bak = new Uint8Array(buffer);
|
|
||||||
for (let i=3; i<length; i+=n_channels)
|
|
||||||
alpha_bak[(i-3)/n_channels] = pixels.contents[i];
|
|
||||||
|
|
||||||
const ALPHA_STEP = 5;
|
|
||||||
|
|
||||||
// fade out
|
|
||||||
for (let a=255; a>0; a-=ALPHA_STEP) {
|
|
||||||
for(let i=3; i<length; i+=n_channels)
|
|
||||||
if (pixels.contents[i]-ALPHA_STEP>0)
|
|
||||||
pixels.contents[i] -= ALPHA_STEP;
|
|
||||||
gtk.gtk_status_icon_set_from_pixbuf(firetray.ChatStatusIcon.trayIcon, pixbuf);
|
|
||||||
sleep(10);
|
|
||||||
}
|
|
||||||
|
|
||||||
// fade in
|
|
||||||
for (let a=255; a>0; a-=ALPHA_STEP) {
|
|
||||||
for(let i=3; i<length; i+=n_channels)
|
|
||||||
if (pixels.contents[i]+ALPHA_STEP<=alpha_bak[(i-3)/n_channels]) {
|
|
||||||
pixels.contents[i] += ALPHA_STEP;
|
|
||||||
}
|
|
||||||
gtk.gtk_status_icon_set_from_pixbuf(firetray.ChatStatusIcon.trayIcon, pixbuf);
|
|
||||||
sleep(10);
|
|
||||||
}
|
|
||||||
|
|
||||||
gobject.g_object_unref(pixbuf);
|
|
||||||
});
|
|
||||||
},
|
},
|
||||||
|
|
||||||
startIconBlinking: function() { // gtk_status_icon_set_blinking() deprecated
|
startIconBlinking: function() { // gtk_status_icon_set_blinking() deprecated
|
||||||
@ -194,6 +221,10 @@ firetray.ChatStatusIcon = {
|
|||||||
this.on = false;
|
this.on = false;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
stopCrossFade: function() {
|
||||||
|
this.events['stop-cross-fade'] = true;
|
||||||
|
},
|
||||||
|
|
||||||
setUrgency: function(xid, urgent) {
|
setUrgency: function(xid, urgent) {
|
||||||
gtk.gtk_window_set_urgency_hint(firetray.Handler.gtkWindows.get(xid), urgent);
|
gtk.gtk_window_set_urgency_hint(firetray.Handler.gtkWindows.get(xid), urgent);
|
||||||
},
|
},
|
||||||
|
@ -331,6 +331,7 @@ firetray.Handler.setIconText = function(text, color) { // FIXME: function too lo
|
|||||||
gdk.gdk_pixbuf_composite(bufAlpha,dest,0,0,w,h,0,0,1,1,gdk.GDK_INTERP_NEAREST,255);
|
gdk.gdk_pixbuf_composite(bufAlpha,dest,0,0,w,h,0,0,1,1,gdk.GDK_INTERP_NEAREST,255);
|
||||||
gobject.g_object_unref(bufAlpha);
|
gobject.g_object_unref(bufAlpha);
|
||||||
|
|
||||||
|
log.info("gtk_status_icon_set_from_pixbuf="+dest);
|
||||||
gtk.gtk_status_icon_set_from_pixbuf(firetray.StatusIcon.trayIcon, dest);
|
gtk.gtk_status_icon_set_from_pixbuf(firetray.StatusIcon.trayIcon, dest);
|
||||||
} catch (x) {
|
} catch (x) {
|
||||||
log.error(x);
|
log.error(x);
|
||||||
|
Loading…
Reference in New Issue
Block a user