guint ref_count;
};
+typedef enum {
+ GDK_EVENT_FILTER_REMOVED = 1 << 0
+} GdkEventFilterFlags;
+
struct _GdkEventFilter {
GdkFilterFunc function;
gpointer data;
+ GdkEventFilterFlags flags;
+ guint ref_count;
};
struct _GdkClientFilter {
{
filter = (GdkEventFilter *)tmp_list->data;
if ((filter->function == function) && (filter->data == data))
- return;
+ {
+ filter->ref_count++;
+ return;
+ }
tmp_list = tmp_list->next;
}
filter = g_new (GdkEventFilter, 1);
filter->function = function;
filter->data = data;
+ filter->ref_count = 1;
+ filter->flags = 0;
if (private)
private->filters = g_list_append (private->filters, filter);
if ((filter->function == function) && (filter->data == data))
{
+ filter->flags |= GDK_EVENT_FILTER_REMOVED;
+ filter->ref_count--;
+ if (filter->ref_count != 0)
+ return;
+
if (private)
private->filters = g_list_remove_link (private->filters, node);
else
static gint
gdk_event_apply_filters (NSEvent *nsevent,
GdkEvent *event,
- GList *filters)
+ GList **filters)
{
GList *tmp_list;
GdkFilterReturn result;
- tmp_list = filters;
+ tmp_list = *filters;
while (tmp_list)
{
GdkEventFilter *filter = (GdkEventFilter*) tmp_list->data;
-
- tmp_list = tmp_list->next;
+ GList *node;
+
+ if ((filter->flags & GDK_EVENT_FILTER_REMOVED) != 0)
+ {
+ tmp_list = tmp_list->next;
+ continue;
+ }
+
+ filter->ref_count++;
result = filter->function (nsevent, event, filter->data);
+
+ /* get the next node after running the function since the
+ function may add or remove a next node */
+ node = tmp_list;
+ tmp_list = tmp_list->next;
+
+ filter->ref_count--;
+ if (filter->ref_count == 0)
+ {
+ *filters = g_list_remove_link (*filters, node);
+ g_list_free_1 (node);
+ g_free (filter);
+ }
+
if (result != GDK_FILTER_CONTINUE)
return result;
}
/* Apply global filters */
GdkFilterReturn result;
- result = gdk_event_apply_filters (nsevent, event, _gdk_default_filters);
+ result = gdk_event_apply_filters (nsevent, event, &_gdk_default_filters);
if (result != GDK_FILTER_CONTINUE)
{
return_val = (result == GDK_FILTER_TRANSLATE) ? TRUE : FALSE;
{
g_object_ref (window);
- result = gdk_event_apply_filters (nsevent, event, filter_private->filters);
+ result = gdk_event_apply_filters (nsevent, event, &filter_private->filters);
g_object_unref (window);
static GdkFilterReturn
apply_event_filters (GdkWindow *window,
MSG *msg,
- GList *filters)
+ GList **filters)
{
GdkFilterReturn result = GDK_FILTER_CONTINUE;
GdkEvent *event;
*/
node = _gdk_event_queue_append (_gdk_display, event);
- tmp_list = filters;
+ tmp_list = *filters;
while (tmp_list)
{
GdkEventFilter *filter = (GdkEventFilter *) tmp_list->data;
-
- tmp_list = tmp_list->next;
+ GList *node;
+
+ if ((filter->flags & GDK_EVENT_FILTER_REMOVED) != 0)
+ {
+ tmp_list = tmp_list->next;
+ continue;
+ }
+
+ filter->ref_count++;
result = filter->function (msg, event, filter->data);
+
+ /* get the next node after running the function since the
+ function may add or remove a next node */
+ node = tmp_list;
+ tmp_list = tmp_list->next;
+
+ filter->ref_count--;
+ if (filter->ref_count == 0)
+ {
+ *filters = g_list_remove_link (*filters, node);
+ g_list_free_1 (node);
+ g_free (filter);
+ }
+
if (result != GDK_FILTER_CONTINUE)
break;
}
{
/* Apply global filters */
- GdkFilterReturn result = apply_event_filters (NULL, msg, _gdk_default_filters);
+ GdkFilterReturn result = apply_event_filters (NULL, msg, &_gdk_default_filters);
/* If result is GDK_FILTER_CONTINUE, we continue as if nothing
* happened. If it is GDK_FILTER_REMOVE or GDK_FILTER_TRANSLATE,
{
/* Apply per-window filters */
- GdkFilterReturn result = apply_event_filters (window, msg, ((GdkWindowObject *) window)->filters);
+ GdkFilterReturn result = apply_event_filters (window, msg, &((GdkWindowObject *) window)->filters);
if (result == GDK_FILTER_REMOVE || result == GDK_FILTER_TRANSLATE)
{
static gint
gdk_event_apply_filters (XEvent *xevent,
GdkEvent *event,
- GList *filters)
+ GList **filters)
{
GList *tmp_list;
GdkFilterReturn result;
- tmp_list = filters;
+ tmp_list = *filters;
while (tmp_list)
{
GdkEventFilter *filter = (GdkEventFilter*) tmp_list->data;
+ GList *node;
- tmp_list = tmp_list->next;
+ if ((filter->flags & GDK_EVENT_FILTER_REMOVED) != 0)
+ {
+ tmp_list = tmp_list->next;
+ continue;
+ }
+
+ filter->ref_count++;
result = filter->function (xevent, event, filter->data);
+ /* get the next node after running the function since the
+ function may add or remove a next node */
+ node = tmp_list;
+ tmp_list = tmp_list->next;
+
+ filter->ref_count--;
+ if (filter->ref_count == 0)
+ {
+ *filters = g_list_remove_link (*filters, node);
+ g_list_free_1 (node);
+ g_free (filter);
+ }
+
if (result != GDK_FILTER_CONTINUE)
return result;
}
/* Apply global filters */
result = gdk_event_apply_filters (xevent, event,
- _gdk_default_filters);
+ &_gdk_default_filters);
if (result == GDK_FILTER_REMOVE)
{
if (filter_private->filters)
{
result = gdk_event_apply_filters (xevent, event,
- filter_private->filters);
+ &filter_private->filters);
if (result == GDK_FILTER_REMOVE)
{