[mutter/wip/barriers-try-2: 7/8] barrier: Add signals for when a barrier gets hit



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]