[mutter/wip/barriers-try-2: 7/8] barrier: Add signals for when a barrier gets hit
- From: Jasper St. Pierre <jstpierre src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [mutter/wip/barriers-try-2: 7/8] barrier: Add signals for when a barrier gets hit
- Date: Mon, 26 Nov 2012 01:40:34 +0000 (UTC)
commit 4bb17cb14106ac0b9cd947a0491a0ab31e2a138e
Author: Jasper St. Pierre <jstpierre mecheye net>
Date: Mon Jul 30 15:57:53 2012 -0300
barrier: Add signals for when a barrier gets hit
We want to know when a barrier gets hit so that we can implement
message tray pressure.
https://bugzilla.gnome.org/show_bug.cgi?id=677215
src/core/barrier.c | 93 ++++++++++++++++++++++++++++++++++++++++++++
src/core/display-private.h | 7 +++
src/core/display.c | 18 ++++++++
src/core/screen.c | 1 +
src/meta/barrier.h | 16 ++++++++
5 files changed, 135 insertions(+), 0 deletions(-)
---
diff --git a/src/core/barrier.c b/src/core/barrier.c
index ba232d0..b1c0c3d 100644
--- a/src/core/barrier.c
+++ b/src/core/barrier.c
@@ -4,6 +4,7 @@
#include <glib-object.h>
+#include <X11/extensions/XInput2.h>
#include <X11/extensions/Xfixes.h>
#include <meta/util.h>
#include <meta/barrier.h>
@@ -30,6 +31,14 @@ enum {
static GParamSpec *obj_props[PROP_LAST];
+enum {
+ HIT,
+
+ LAST_SIGNAL,
+};
+
+static guint obj_signals[LAST_SIGNAL];
+
struct _MetaBarrierPrivate
{
MetaDisplay *display;
@@ -157,6 +166,7 @@ meta_barrier_activate (MetaBarrier *barrier)
priv->x1, priv->y1,
priv->x2, priv->y2,
priv->directions, 0, NULL);
+ g_hash_table_insert (priv->display->barriers, GINT_TO_POINTER (priv->barrier), barrier);
}
static void
@@ -216,6 +226,24 @@ meta_barrier_class_init (MetaBarrierClass *klass)
g_object_class_install_properties (object_class, PROP_LAST, obj_props);
+ /**
+ * MetaBarrier::hit:
+ * @barrier: The #MetaBarrier that was hit
+ * @event: A #MetaBarrierEvent that has the details of how
+ * things were hit.
+ *
+ * When a pointer barrier is hit, this will trigger. This
+ * requires an XI2-enabled server.
+ */
+ obj_signals[HIT] =
+ g_signal_new ("hit",
+ G_TYPE_FROM_CLASS (object_class),
+ G_SIGNAL_RUN_FIRST,
+ 0,
+ NULL, NULL, NULL,
+ G_TYPE_NONE, 1,
+ META_TYPE_BARRIER_HIT_EVENT);
+
g_type_class_add_private (object_class, sizeof(MetaBarrierPrivate));
}
@@ -237,6 +265,7 @@ meta_barrier_destroy (MetaBarrier *barrier)
return;
XFixesDestroyPointerBarrier (dpy, priv->barrier);
+ g_hash_table_remove (priv->display->barriers, GINT_TO_POINTER (priv->barrier));
priv->barrier = 0;
}
@@ -245,3 +274,67 @@ meta_barrier_init (MetaBarrier *barrier)
{
barrier->priv = GET_PRIVATE (barrier);
}
+
+static void
+meta_barrier_hit (MetaBarrier *barrier,
+ XIBarrierNotifyEvent *xevent)
+{
+ MetaBarrierHitEvent *event = g_slice_new0 (MetaBarrierHitEvent);
+
+ event->ref_count = 1;
+ event->event_id = xevent->event_id;
+ event->dt = xevent->dt;
+
+ event->x = xevent->x;
+ event->y = xevent->y;
+ event->dx = xevent->dx;
+ event->dy = xevent->dy;
+ event->raw_dx = xevent->raw_dx;
+ event->raw_dy = xevent->raw_dy;
+
+ g_signal_emit (barrier, obj_signals[HIT], 0, event);
+}
+
+gboolean
+meta_display_process_barrier_event (MetaDisplay *display,
+ XIBarrierNotifyEvent *xev)
+{
+ MetaBarrier *barrier;
+
+ if (xev->evtype != XI_BarrierHitNotify)
+ return FALSE;
+
+ barrier = g_hash_table_lookup (display->barriers, GINT_TO_POINTER (xev->barrier));
+ if (barrier != NULL)
+ {
+ meta_barrier_hit (barrier, xev);
+ return TRUE;
+ }
+
+ return FALSE;
+}
+
+static MetaBarrierHitEvent *
+meta_barrier_hit_event_ref (MetaBarrierHitEvent *event)
+{
+ g_return_val_if_fail (event != NULL, NULL);
+ g_return_val_if_fail (event->ref_count > 0, NULL);
+
+ g_atomic_int_inc ((volatile int *)&event->ref_count);
+ return event;
+}
+
+static void
+meta_barrier_hit_event_unref (MetaBarrierHitEvent *event)
+{
+ g_return_if_fail (event != NULL);
+ g_return_if_fail (event->ref_count > 0);
+
+ if (g_atomic_int_dec_and_test ((volatile int *)&event->ref_count))
+ g_slice_free (MetaBarrierHitEvent, event);
+}
+
+G_DEFINE_BOXED_TYPE (MetaBarrierHitEvent,
+ meta_barrier_hit_event,
+ meta_barrier_hit_event_ref,
+ meta_barrier_hit_event_unref)
diff --git a/src/core/display-private.h b/src/core/display-private.h
index b4c1b71..1f685d2 100644
--- a/src/core/display-private.h
+++ b/src/core/display-private.h
@@ -39,6 +39,7 @@
#include <meta/display.h>
#include "keybindings-private.h"
#include <meta/prefs.h>
+#include <meta/barrier.h>
#ifdef HAVE_STARTUP_NOTIFICATION
#include <libsn/sn.h>
@@ -244,6 +245,9 @@ struct _MetaDisplay
unsigned int meta_mask;
MetaKeyCombo overlay_key_combo;
gboolean overlay_key_only_pressed;
+
+ /* barriers */
+ GHashTable *barriers;
/* Monitor cache */
unsigned int monitor_cache_invalidated : 1;
@@ -448,4 +452,7 @@ void meta_display_overlay_key_activate (MetaDisplay *display);
/* In above-tab-keycode.c */
guint meta_display_get_above_tab_keycode (MetaDisplay *display);
+gboolean meta_display_process_barrier_event (MetaDisplay *display,
+ XIBarrierNotifyEvent *event);
+
#endif
diff --git a/src/core/display.c b/src/core/display.c
index ebf7c06..ee00408 100644
--- a/src/core/display.c
+++ b/src/core/display.c
@@ -809,6 +809,10 @@ meta_display_open (void)
meta_verbose ("Not compiled with Xcursor support\n");
#endif /* !HAVE_XCURSOR */
+ the_display->barriers = g_hash_table_new_full (g_direct_hash,
+ g_direct_equal,
+ NULL, g_object_unref);
+
/* Create the leader window here. Set its properties and
* use the timestamp from one of the PropertyNotify events
* that will follow.
@@ -1832,6 +1836,10 @@ get_input_event (MetaDisplay *display,
case XI_Leave:
return xev;
break;
+ case XI_BarrierHitNotify:
+ case XI_BarrierPointerReleasedNotify:
+ return xev;
+ break;
default:
break;
}
@@ -2396,6 +2404,10 @@ event_callback (XEvent *event,
}
break;
+ case XI_BarrierHitNotify:
+ if (meta_display_process_barrier_event (display, (XIBarrierNotifyEvent *) xev))
+ filter_out_event = bypass_compositor = TRUE;
+ break;
}
}
else
@@ -2927,6 +2939,9 @@ event_get_modified_window (MetaDisplay *display,
case XI_Enter:
case XI_Leave:
return ((XIEnterEvent *) xev)->event;
+ case XI_BarrierHitNotify:
+ case XI_BarrierPointerReleasedNotify:
+ return ((XIBarrierNotifyEvent *) xev)->window;
}
}
@@ -3223,6 +3238,9 @@ meta_spew_xi2event (MetaDisplay *display,
case XI_Leave:
name = "XI_Leave";
break;
+ case XI_BarrierHitNotify:
+ name = "XI_BarrierHitNotify";
+ break;
}
/* TODO: extra */
diff --git a/src/core/screen.c b/src/core/screen.c
index afc567f..6608a31 100644
--- a/src/core/screen.c
+++ b/src/core/screen.c
@@ -751,6 +751,7 @@ meta_screen_new (MetaDisplay *display,
XISetMask (mask.mask, XI_FocusIn);
XISetMask (mask.mask, XI_FocusOut);
XISetMask (mask.mask, XI_Motion);
+ XISetMask (mask.mask, XI_BarrierHitNotify);
XISelectEvents (xdisplay, xroot, &mask, 1);
event_mask = (SubstructureRedirectMask | SubstructureNotifyMask |
diff --git a/src/meta/barrier.h b/src/meta/barrier.h
index 42460fc..97aae20 100644
--- a/src/meta/barrier.h
+++ b/src/meta/barrier.h
@@ -45,6 +45,22 @@ typedef enum {
META_BARRIER_DIRECTION_NEGATIVE_Y = (1L << 3),
} MetaBarrierDirection;
+typedef struct {
+ int ref_count;
+
+ int event_id;
+ int dt;
+ double x;
+ double y;
+ double dx;
+ double dy;
+ double raw_dx;
+ double raw_dy;
+} MetaBarrierHitEvent;
+
+#define META_TYPE_BARRIER_HIT_EVENT (meta_barrier_hit_event_get_type ())
+GType meta_barrier_hit_event_get_type (void) G_GNUC_CONST;
+
G_END_DECLS
#endif /* __META_BARRIER_H__ */
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]