Implemented a way to unregister desktop messages

This commit is contained in:
Pierre Pronchery 2013-12-19 04:59:42 +01:00
parent 142b23725d
commit 98f7f2620c
2 changed files with 82 additions and 20 deletions

View File

@ -30,6 +30,9 @@ typedef int (*DesktopMessageCallback)(void * data, uint32_t value1,
/* functions */ /* functions */
int desktop_message_register(GtkWidget * window, char const * destination, int desktop_message_register(GtkWidget * window, char const * destination,
DesktopMessageCallback callback, void * data); DesktopMessageCallback callback, void * data);
void desktop_message_unregister(GtkWidget * window,
DesktopMessageCallback callback, void * data);
int desktop_message_send(char const * destination, uint32_t value1, int desktop_message_send(char const * destination, uint32_t value1,
uint32_t value2, uint32_t value3); uint32_t value2, uint32_t value3);

View File

@ -17,6 +17,10 @@
#include <stdlib.h>
#ifdef DEBUG
# include <stdio.h>
#endif
#include <string.h> #include <string.h>
#include <errno.h> #include <errno.h>
#include <gtk/gtk.h> #include <gtk/gtk.h>
@ -28,16 +32,22 @@
/* Message */ /* Message */
#if !GTK_CHECK_VERSION(3, 0, 0)
/* private */ /* private */
/* types */ /* types */
#if !GTK_CHECK_VERSION(3, 0, 0)
typedef struct _MessageCallback typedef struct _MessageCallback
{ {
GtkWidget * window;
GtkWidget * widget; GtkWidget * widget;
Window window; Window xwindow;
DesktopMessageCallback callback; DesktopMessageCallback callback;
void * data; void * data;
} MessageCallback; } MessageCallback;
/* variables */
static MessageCallback ** _callbacks = NULL;
static size_t _callbacks_cnt = 0;
#endif #endif
@ -56,26 +66,34 @@ int desktop_message_register(GtkWidget * window, char const * destination,
DesktopMessageCallback callback, void * data) DesktopMessageCallback callback, void * data)
{ {
#if !GTK_CHECK_VERSION(3, 0, 0) #if !GTK_CHECK_VERSION(3, 0, 0)
MessageCallback ** p;
MessageCallback * mc; MessageCallback * mc;
GdkWindow * w; GdkWindow * gwindow;
GdkAtom atom;
/* XXX memory leak */ # ifdef DEBUG
fprintf(stderr, "DEBUG: %s(%p, \"%s\", %p, %p)\n", __func__, window,
destination, callback, data);
# endif
if((p = realloc(_callbacks, sizeof(*p) * (_callbacks_cnt + 1))) == NULL)
return -1;
_callbacks = p;
if((mc = object_new(sizeof(*mc))) == NULL) if((mc = object_new(sizeof(*mc))) == NULL)
return -1; return -1;
if(window == NULL) _callbacks[_callbacks_cnt++] = mc;
if((mc->window = window) == NULL)
{ {
window = gtk_window_new(GTK_WINDOW_TOPLEVEL); mc->widget = gtk_window_new(GTK_WINDOW_TOPLEVEL);
mc->widget = window; gtk_widget_realize(mc->widget);
gtk_widget_realize(window);
} }
else else
mc->widget = NULL; mc->widget = window;
w = gtk_widget_get_window(window); gwindow = gtk_widget_get_window(mc->widget);
mc->window = GDK_WINDOW_XWINDOW(w); mc->xwindow = GDK_WINDOW_XWINDOW(gwindow);
mc->callback = callback; mc->callback = callback;
mc->data = data; mc->data = data;
gdk_add_client_message_filter(gdk_atom_intern(destination, FALSE), atom = gdk_atom_intern(destination, FALSE);
_desktop_message_on_callback, mc); gdk_add_client_message_filter(atom, _desktop_message_on_callback, mc);
#endif #endif
return 0; return 0;
} }
@ -86,14 +104,16 @@ int desktop_message_send(char const * destination, uint32_t value1,
uint32_t value2, uint32_t value3) uint32_t value2, uint32_t value3)
{ {
#if !GTK_CHECK_VERSION(3, 0, 0) #if !GTK_CHECK_VERSION(3, 0, 0)
GdkAtom atom;
GdkEvent event; GdkEvent event;
GdkEventClient * client = &event.client; GdkEventClient * client = &event.client;
atom = gdk_atom_intern(destination, FALSE);
memset(&event, 0, sizeof(event)); memset(&event, 0, sizeof(event));
client->type = GDK_CLIENT_EVENT; client->type = GDK_CLIENT_EVENT;
client->window = NULL; client->window = NULL;
client->send_event = TRUE; client->send_event = TRUE;
client->message_type = gdk_atom_intern(destination, FALSE); client->message_type = atom;
client->data_format = 32; client->data_format = 32;
client->data.l[0] = value1; client->data.l[0] = value1;
client->data.l[1] = value2; client->data.l[1] = value2;
@ -104,6 +124,43 @@ int desktop_message_send(char const * destination, uint32_t value1,
} }
/* desktop_message_unregister */
void desktop_message_unregister(GtkWidget * window,
DesktopMessageCallback callback, void * data)
{
#if !GTK_CHECK_VERSION(3, 0, 0)
size_t i;
MessageCallback ** p;
MessageCallback * mc;
GdkWindow * w;
# ifdef DEBUG
fprintf(stderr, "DEBUG: %s(%p, %p)\n", __func__, callback, data);
# endif
for(i = 0; i < _callbacks_cnt; i++)
{
mc = _callbacks[i];
if(mc->window == window
&& mc->callback == callback
&& mc->data == data)
break;
}
if(i == _callbacks_cnt)
return;
w = gtk_widget_get_window(mc->widget);
gdk_window_remove_filter(w, _desktop_message_on_callback, mc);
if(mc->window == NULL)
gtk_widget_destroy(mc->widget);
object_delete(mc);
p = &_callbacks[i];
memmove(p, p + 1, sizeof(*p) * (_callbacks_cnt - i - 1));
if((p = realloc(_callbacks, sizeof(*p) * (--_callbacks_cnt))) != NULL
|| _callbacks_cnt == 0)
_callbacks = p;
#endif
}
/* private */ /* private */
/* callbacks */ /* callbacks */
#if !GTK_CHECK_VERSION(3, 0, 0) #if !GTK_CHECK_VERSION(3, 0, 0)
@ -122,19 +179,21 @@ static GdkFilterReturn _desktop_message_on_callback(GdkXEvent * xevent,
return GDK_FILTER_CONTINUE; return GDK_FILTER_CONTINUE;
xcme = &xev->xclient; xcme = &xev->xclient;
# ifdef DEBUG # ifdef DEBUG
fprintf(stderr, "DEBUG: %s(%lu %lu) %lu\n", __func__, xcme->serial, fprintf(stderr, "DEBUG: %s(%lu %lu %p) %lu\n", __func__, xcme->serial,
xcme->window, mc->window); xcme->window, mc, mc->xwindow);
# endif # endif
if(mc->window != xcme->window) if(mc->xwindow != xcme->window)
return GDK_FILTER_CONTINUE; return GDK_FILTER_CONTINUE;
# ifdef DEBUG
fprintf(stderr, "DEBUG: %s() %p(%p)\n", __func__, mc->callback,
mc->data);
# endif
value1 = xcme->data.l[0]; value1 = xcme->data.l[0];
value2 = xcme->data.l[1]; value2 = xcme->data.l[1];
value3 = xcme->data.l[2]; value3 = xcme->data.l[2];
if(mc->callback(mc->data, value1, value2, value3) == 0) if(mc->callback(mc->data, value1, value2, value3) == 0)
return GDK_FILTER_CONTINUE; return GDK_FILTER_CONTINUE;
if(mc->widget != NULL) desktop_message_unregister(mc->window, mc->callback, mc->data);
gtk_widget_destroy(mc->widget);
object_delete(mc);
return GDK_FILTER_REMOVE; return GDK_FILTER_REMOVE;
} }
#endif #endif