[gtk+/multitouch: 16/27] Add a gestures interpreter



commit 964398322a23e485409489c14e6fce1eae71d717
Author: Carlos Garnacho <carlosg gnome org>
Date:   Tue Oct 25 02:19:07 2011 +0200

    Add a gestures interpreter
    
    This object is currently shallow, the API is defined but no gesture
    interpretation is performed. Additional API is declared to create
    and define gestures, which may consist of several strokes with an
    offset.

 gtk/Makefile.am              |    2 +
 gtk/gtk.h                    |    1 +
 gtk/gtkenums.h               |   14 +
 gtk/gtkgesturesinterpreter.c |  604 ++++++++++++++++++++++++++++++++++++++++++
 gtk/gtkgesturesinterpreter.h |  119 +++++++++
 gtk/gtkmarshalers.list       |    1 +
 6 files changed, 741 insertions(+), 0 deletions(-)
---
diff --git a/gtk/Makefile.am b/gtk/Makefile.am
index 231902f..3809ea0 100644
--- a/gtk/Makefile.am
+++ b/gtk/Makefile.am
@@ -240,6 +240,7 @@ gtk_public_h_sources = 		\
 	gtkfontchooserdialog.h  \
 	gtkfontchooserwidget.h	\
 	gtkframe.h		\
+	gtkgesturesinterpreter.h \
 	gtkgradient.h		\
 	gtkgrid.h		\
 	gtkhsv.h		\
@@ -585,6 +586,7 @@ gtk_base_c_sources = 		\
 	gtkfontchooserutils.c	\
 	gtkfontchooserwidget.c	\
 	gtkframe.c		\
+	gtkgesturesinterpreter.c \
 	gtkgradient.c		\
 	gtkgrid.c		\
 	gtkhsv.c		\
diff --git a/gtk/gtk.h b/gtk/gtk.h
index 8ac61c9..6ef517c 100644
--- a/gtk/gtk.h
+++ b/gtk/gtk.h
@@ -105,6 +105,7 @@
 #include <gtk/gtkfontchooserdialog.h>
 #include <gtk/gtkfontchooserwidget.h>
 #include <gtk/gtkframe.h>
+#include <gtk/gtkgesturesinterpreter.h>
 #include <gtk/gtkgradient.h>
 #include <gtk/gtkgrid.h>
 #include <gtk/gtkhsv.h>
diff --git a/gtk/gtkenums.h b/gtk/gtkenums.h
index 9188208..f1493e3 100644
--- a/gtk/gtkenums.h
+++ b/gtk/gtkenums.h
@@ -926,6 +926,20 @@ typedef enum {
   GTK_CAPTURED_EVENT_STORE   = 1 << 1
 } GtkCapturedEventFlags;
 
+typedef enum {
+  GTK_GESTURE_SWIPE_RIGHT = 1,
+  GTK_GESTURE_SWIPE_LEFT,
+  GTK_GESTURE_SWIPE_UP,
+  GTK_GESTURE_SWIPE_DOWN,
+  GTK_GESTURE_CIRCULAR_CLOCKWISE,
+  GTK_GESTURE_CIRCULAR_COUNTERCLOCKWISE
+} GtkGestureType;
+
+typedef enum {
+  GTK_GESTURE_FLAG_NONE                       = 0,
+  GTK_GESTURE_FLAG_IGNORE_INITIAL_ORIENTATION = 1 << 1
+} GtkGestureFlags;
+
 G_END_DECLS
 
 
diff --git a/gtk/gtkgesturesinterpreter.c b/gtk/gtkgesturesinterpreter.c
new file mode 100644
index 0000000..0df8c61
--- /dev/null
+++ b/gtk/gtkgesturesinterpreter.c
@@ -0,0 +1,604 @@
+/* GTK - The GIMP Toolkit
+ * Copyright (C) 2011 Carlos Garnacho <carlos lanedo com>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#include "config.h"
+#include "gtkgesturesinterpreter.h"
+#include "gtkmarshalers.h"
+#include "gtkintl.h"
+#include <gdk/gdk.h>
+#include <math.h>
+
+#define N_CIRCULAR_SIDES 12
+#define VECTORIZATION_ANGLE_THRESHOLD (G_PI_2 / 10)
+#define MINIMUM_CONFIDENCE_ALLOWED 0.8
+
+G_DEFINE_TYPE (GtkGesturesInterpreter, gtk_gestures_interpreter, G_TYPE_OBJECT)
+G_DEFINE_BOXED_TYPE (GtkGestureStroke, gtk_gesture_stroke,
+                     gtk_gesture_stroke_copy, gtk_gesture_stroke_free)
+G_DEFINE_BOXED_TYPE (GtkGesture, gtk_gesture,
+                     gtk_gesture_copy, gtk_gesture_free)
+
+typedef struct _GtkGesturesInterpreterPrivate GtkGesturesInterpreterPrivate;
+typedef struct _StrokeVector StrokeVector;
+typedef struct _RecordedGesture RecordedGesture;
+
+struct _GtkGesturesInterpreterPrivate
+{
+  GHashTable *events;
+  GArray *handled_gestures;
+};
+
+struct _StrokeVector
+{
+  gdouble angle;
+  guint16 length;
+};
+
+struct _GtkGestureStroke
+{
+  GArray *gesture_data;
+  guint total_length;
+  gint dx;
+  gint dy;
+};
+
+struct _GtkGesture
+{
+  GPtrArray *strokes;
+  GtkGestureFlags flags;
+};
+
+struct _RecordedGesture
+{
+  GArray *coordinates;
+  gint min_x;
+  gint max_x;
+  gint min_y;
+  gint max_y;
+};
+
+enum {
+  EVENTS_VECTORIZED,
+  GESTURE_DETECTED,
+  LAST_SIGNAL
+};
+
+static GHashTable *registered_gestures = NULL;
+static guint signals[LAST_SIGNAL] = { 0 };
+
+static void
+initialize_gestures_ht (void)
+{
+  GtkGestureStroke *stroke;
+  GtkGesture *gesture;
+  gdouble angle;
+  guint id, n;
+
+  if (registered_gestures)
+    return;
+
+  registered_gestures = g_hash_table_new (NULL, NULL);
+
+  /* Swipe right */
+  stroke = gtk_gesture_stroke_new ();
+  gtk_gesture_stroke_append_vector (stroke, G_PI / 2, 100);
+  gesture = gtk_gesture_new (stroke, 0);
+  id = gtk_gesture_register_static (gesture);
+  g_assert (id == GTK_GESTURE_SWIPE_RIGHT);
+  gtk_gesture_stroke_free (stroke);
+
+  /* Swipe left */
+  stroke = gtk_gesture_stroke_new ();
+  gtk_gesture_stroke_append_vector (stroke, 3 * G_PI / 2, 100);
+  gesture = gtk_gesture_new (stroke, 0);
+  id = gtk_gesture_register_static (gesture);
+  g_assert (id == GTK_GESTURE_SWIPE_LEFT);
+  gtk_gesture_stroke_free (stroke);
+
+  /* Swipe up */
+  stroke = gtk_gesture_stroke_new ();
+  gtk_gesture_stroke_append_vector (stroke, 0, 100);
+  gesture = gtk_gesture_new (stroke, 0);
+  id = gtk_gesture_register_static (gesture);
+  g_assert (id == GTK_GESTURE_SWIPE_UP);
+  gtk_gesture_stroke_free (stroke);
+
+  /* Swipe down */
+  stroke = gtk_gesture_stroke_new ();
+  gtk_gesture_stroke_append_vector (stroke, G_PI, 100);
+  gesture = gtk_gesture_new (stroke, 0);
+  id = gtk_gesture_register_static (gesture);
+  g_assert (id == GTK_GESTURE_SWIPE_DOWN);
+  gtk_gesture_stroke_free (stroke);
+
+  /* Circular clockwise */
+  stroke = gtk_gesture_stroke_new ();
+
+  for (n = 0; n < N_CIRCULAR_SIDES; n++)
+    {
+      angle = 2 * G_PI * ((gdouble) n / N_CIRCULAR_SIDES);
+      gtk_gesture_stroke_append_vector (stroke, angle, 50);
+    }
+
+  gesture = gtk_gesture_new (stroke,
+                             GTK_GESTURE_FLAG_IGNORE_INITIAL_ORIENTATION);
+  id = gtk_gesture_register_static (gesture);
+  g_assert (id == GTK_GESTURE_CIRCULAR_CLOCKWISE);
+  gtk_gesture_stroke_free (stroke);
+
+  /* Circular counterclockwise */
+  stroke = gtk_gesture_stroke_new ();
+
+  for (n = 0; n < N_CIRCULAR_SIDES; n++)
+    {
+      angle = 2 * G_PI * ((gdouble) (N_CIRCULAR_SIDES - n) / N_CIRCULAR_SIDES);
+      gtk_gesture_stroke_append_vector (stroke, angle, 50);
+    }
+
+  gesture = gtk_gesture_new (stroke,
+                             GTK_GESTURE_FLAG_IGNORE_INITIAL_ORIENTATION);
+  id = gtk_gesture_register_static (gesture);
+  g_assert (id == GTK_GESTURE_CIRCULAR_COUNTERCLOCKWISE);
+  gtk_gesture_stroke_free (stroke);
+}
+
+static void
+gtk_gestures_interpreter_finalize (GObject *object)
+{
+  GtkGesturesInterpreterPrivate *priv;
+
+  priv = GTK_GESTURES_INTERPRETER (object)->priv;
+
+  g_hash_table_destroy (priv->events);
+  g_array_free (priv->handled_gestures, TRUE);
+
+  G_OBJECT_CLASS (gtk_gestures_interpreter_parent_class)->finalize (object);
+}
+
+static void
+gtk_gestures_interpreter_class_init (GtkGesturesInterpreterClass *klass)
+{
+  GObjectClass *object_class = G_OBJECT_CLASS (klass);
+
+  object_class->finalize = gtk_gestures_interpreter_finalize;
+
+  initialize_gestures_ht ();
+
+  signals[EVENTS_VECTORIZED] =
+    g_signal_new (I_("events-vectorized"),
+		  G_TYPE_FROM_CLASS (object_class),
+		  G_SIGNAL_RUN_FIRST,
+                  G_STRUCT_OFFSET (GtkGesturesInterpreterClass,
+				   events_vectorized),
+                  NULL, NULL,
+                  _gtk_marshal_VOID__BOXED,
+                  G_TYPE_NONE, 1, GTK_TYPE_GESTURE);
+  signals[GESTURE_DETECTED] =
+    g_signal_new (I_("gesture-detected"),
+		  G_TYPE_FROM_CLASS (object_class),
+		  G_SIGNAL_RUN_FIRST,
+                  G_STRUCT_OFFSET (GtkGesturesInterpreterClass,
+				   gesture_detected),
+                  NULL, NULL,
+                  _gtk_marshal_VOID__UINT_DOUBLE,
+                  G_TYPE_NONE, 2,
+                  G_TYPE_UINT, G_TYPE_DOUBLE);
+
+  g_type_class_add_private (object_class, sizeof (GtkGesturesInterpreterPrivate));
+}
+
+static RecordedGesture *
+recorded_gesture_new (void)
+{
+  RecordedGesture *recorded;
+
+  recorded = g_slice_new0 (RecordedGesture);
+  recorded->coordinates = g_array_new (FALSE, FALSE, sizeof (GdkPoint));
+  recorded->min_x = G_MAXINT;
+  recorded->max_x = G_MININT;
+  recorded->min_y = G_MAXINT;
+  recorded->max_y = G_MININT;
+
+  return recorded;
+}
+
+static void
+recorded_gesture_free (RecordedGesture *recorded)
+{
+  g_array_free (recorded->coordinates, TRUE);
+  g_slice_free (RecordedGesture, recorded);
+}
+
+static void
+recorded_gesture_append_coordinate (RecordedGesture *recorded,
+                                    gint             x,
+                                    gint             y)
+{
+  GdkPoint point, *prev = NULL;
+
+  if (recorded->coordinates->len > 0)
+    prev = &g_array_index (recorded->coordinates, GdkPoint,
+			   recorded->coordinates->len - 1);
+
+  if (prev &&
+      prev->x == x &&
+      prev->y == y)
+    return;
+
+  point.x = x;
+  point.y = y;
+
+  g_array_append_val (recorded->coordinates, point);
+
+  recorded->min_x = MIN (recorded->min_x, x);
+  recorded->max_x = MAX (recorded->max_x, x);
+  recorded->min_y = MIN (recorded->min_y, y);
+  recorded->max_y = MAX (recorded->max_y, y);
+}
+
+static void
+gtk_gestures_interpreter_init (GtkGesturesInterpreter *interpreter)
+{
+  GtkGesturesInterpreterPrivate *priv;
+
+  priv = interpreter->priv = G_TYPE_INSTANCE_GET_PRIVATE (interpreter,
+                                                          GTK_TYPE_GESTURES_INTERPRETER,
+                                                          GtkGesturesInterpreterPrivate);
+  priv->events = g_hash_table_new_full (NULL, NULL, NULL,
+                                        (GDestroyNotify) recorded_gesture_free);
+  priv->handled_gestures = g_array_new (FALSE, FALSE, sizeof (guint));
+}
+
+/* Gesture stroke */
+GtkGestureStroke *
+gtk_gesture_stroke_new (void)
+{
+  GtkGestureStroke *stroke;
+
+  stroke = g_slice_new0 (GtkGestureStroke);
+  stroke->gesture_data = g_array_new (FALSE, FALSE,
+                                      sizeof (StrokeVector));
+  return stroke;
+}
+
+GtkGestureStroke *
+gtk_gesture_stroke_copy (const GtkGestureStroke *stroke)
+{
+  GtkGestureStroke *copy;
+
+  copy = gtk_gesture_stroke_new ();
+  g_array_append_vals (copy->gesture_data,
+                       stroke->gesture_data->data,
+                       stroke->gesture_data->len);
+  copy->total_length = stroke->total_length;
+  copy->dx = stroke->dx;
+  copy->dy = stroke->dy;
+
+  return copy;
+}
+
+void
+gtk_gesture_stroke_free (GtkGestureStroke *stroke)
+{
+  g_return_if_fail (stroke != NULL);
+
+  g_array_free (stroke->gesture_data, TRUE);
+  g_slice_free (GtkGestureStroke, stroke);
+}
+
+void
+gtk_gesture_stroke_append_vector (GtkGestureStroke *stroke,
+                                  gdouble           angle,
+                                  guint             length)
+{
+  StrokeVector vector;
+
+  g_return_if_fail (stroke != NULL);
+
+  vector.angle = fmod (angle, 2 * G_PI);
+  vector.length = length;
+  g_array_append_val (stroke->gesture_data, vector);
+
+  stroke->total_length += length;
+}
+
+guint
+gtk_gesture_stroke_get_n_vectors (const GtkGestureStroke *stroke)
+{
+  g_return_val_if_fail (stroke != NULL, 0);
+
+  return stroke->gesture_data->len;
+}
+
+gboolean
+gtk_gesture_stroke_get_vector (const GtkGestureStroke *stroke,
+                               guint                   n_vector,
+                               gdouble                *angle,
+                               guint                  *length,
+                               gdouble                *relative_length)
+{
+  StrokeVector *vector;
+
+  g_return_val_if_fail (stroke != NULL, FALSE);
+  g_return_val_if_fail (n_vector < stroke->gesture_data->len, FALSE);
+
+  vector = &g_array_index (stroke->gesture_data, StrokeVector, n_vector);
+
+  if (angle)
+    *angle = vector->angle;
+
+  if (length)
+    *length = vector->length;
+
+  if (relative_length)
+    *relative_length = (gdouble) vector->length / stroke->total_length;
+
+  return TRUE;
+}
+
+/* Gesture */
+GtkGesture *
+gtk_gesture_new (const GtkGestureStroke *stroke,
+                 GtkGestureFlags         flags)
+{
+  GtkGesture *gesture;
+
+  g_return_val_if_fail (stroke != NULL, NULL);
+  g_return_val_if_fail (stroke->gesture_data->len > 0, NULL);
+
+  gesture = g_slice_new0 (GtkGesture);
+  gesture->flags = flags;
+  gesture->strokes =
+    g_ptr_array_new_with_free_func ((GDestroyNotify) gtk_gesture_stroke_free);
+
+  g_ptr_array_add (gesture->strokes,
+                   gtk_gesture_stroke_copy (stroke));
+  return gesture;
+}
+
+void
+gtk_gesture_add_stroke (GtkGesture             *gesture,
+                        const GtkGestureStroke *stroke,
+                        gint                    dx,
+                        gint                    dy)
+{
+  GtkGestureStroke *copy;
+
+  g_return_if_fail (gesture != NULL);
+  g_return_if_fail (stroke != NULL);
+  g_return_if_fail (stroke->gesture_data->len > 0);
+
+  copy = gtk_gesture_stroke_copy (stroke);
+  copy->dx = dx;
+  copy->dy = dy;
+
+  g_ptr_array_add (gesture->strokes, copy);
+}
+
+GtkGesture *
+gtk_gesture_copy (const GtkGesture *gesture)
+{
+  GtkGesture *copy;
+  guint i;
+
+  g_return_val_if_fail (gesture != NULL, NULL);
+
+  copy = g_slice_new0 (GtkGesture);
+  copy->flags = gesture->flags;
+  copy->strokes =
+    g_ptr_array_new_with_free_func ((GDestroyNotify) gtk_gesture_stroke_free);
+
+  for (i = 0; i < gesture->strokes->len; i++)
+    {
+      GtkGestureStroke *stroke;
+
+      stroke = g_ptr_array_index (gesture->strokes, i);
+      g_ptr_array_add (copy->strokes,
+                       gtk_gesture_stroke_copy (stroke));
+    }
+
+  return copy;
+}
+
+void
+gtk_gesture_free (GtkGesture *gesture)
+{
+  g_return_if_fail (gesture != NULL);
+
+  g_ptr_array_free (gesture->strokes, TRUE);
+  g_slice_free (GtkGesture, gesture);
+}
+
+GtkGestureFlags
+gtk_gesture_get_flags (const GtkGesture *gesture)
+{
+  g_return_val_if_fail (gesture != NULL, 0);
+
+  return gesture->flags;
+}
+
+guint
+gtk_gesture_get_n_strokes (const GtkGesture *gesture)
+{
+  g_return_val_if_fail (gesture != NULL, 0);
+
+  return gesture->strokes->len;
+}
+
+const GtkGestureStroke *
+gtk_gesture_get_stroke (const GtkGesture *gesture,
+                        guint             n_stroke,
+                        gint             *dx,
+                        gint             *dy)
+{
+  GtkGestureStroke *stroke;
+
+  g_return_val_if_fail (gesture != NULL, NULL);
+  g_return_val_if_fail (n_stroke < gesture->strokes->len, NULL);
+
+  stroke = g_ptr_array_index (gesture->strokes, n_stroke);
+
+  if (stroke)
+    {
+      if (dx)
+        *dx = stroke->dx;
+
+      if (dy)
+        *dy = stroke->dy;
+    }
+
+  return stroke;
+}
+
+guint
+gtk_gesture_register (const GtkGesture *gesture)
+{
+  g_return_val_if_fail (gesture != NULL, 0);
+
+  return gtk_gesture_register_static (gtk_gesture_copy (gesture));
+}
+
+guint
+gtk_gesture_register_static (const GtkGesture *gesture)
+{
+  static guint gesture_id = 0;
+
+  g_return_val_if_fail (gesture != NULL, 0);
+
+  initialize_gestures_ht ();
+  gesture_id++;
+
+  g_hash_table_insert (registered_gestures,
+                       GUINT_TO_POINTER (gesture_id),
+                       (gpointer) gesture);
+  return gesture_id;
+}
+
+const GtkGesture *
+gtk_gesture_lookup (guint gesture_id)
+{
+  if (!registered_gestures)
+    return NULL;
+
+  return g_hash_table_lookup (registered_gestures,
+                              GUINT_TO_POINTER (gesture_id));
+}
+
+/* Gesture interpreter */
+GtkGesturesInterpreter *
+gtk_gestures_interpreter_new (void)
+{
+  return g_object_new (GTK_TYPE_GESTURES_INTERPRETER, NULL);
+}
+
+gboolean
+gtk_gestures_interpreter_add_gesture (GtkGesturesInterpreter *interpreter,
+                                      guint                   gesture_id)
+{
+  GtkGesturesInterpreterPrivate *priv;
+  guint i;
+
+  g_return_val_if_fail (GTK_IS_GESTURES_INTERPRETER (interpreter), FALSE);
+  g_return_val_if_fail (gtk_gesture_lookup (gesture_id) != NULL, FALSE);
+
+  priv = interpreter->priv;
+
+  for (i = 0; i < priv->handled_gestures->len; i++)
+    {
+      if (gesture_id == g_array_index (priv->handled_gestures, guint, i))
+        return FALSE;
+    }
+
+  g_array_append_val (priv->handled_gestures, gesture_id);
+
+  return TRUE;
+}
+
+void
+gtk_gestures_interpreter_remove_gesture (GtkGesturesInterpreter *interpreter,
+                                         guint                   gesture_id)
+{
+  GtkGesturesInterpreterPrivate *priv;
+  guint i;
+
+  g_return_if_fail (GTK_IS_GESTURES_INTERPRETER (interpreter));
+  g_return_if_fail (gtk_gesture_lookup (gesture_id) != NULL);
+
+  priv = interpreter->priv;
+
+  for (i = 0; i < priv->handled_gestures->len; i++)
+    {
+      guint handled_gesture_id;
+
+      handled_gesture_id = g_array_index (priv->handled_gestures, guint, i);
+
+      if (handled_gesture_id == gesture_id)
+        {
+          g_array_remove_index_fast (priv->handled_gestures, i);
+          break;
+        }
+    }
+}
+
+gboolean
+gtk_gestures_interpreter_feed_event (GtkGesturesInterpreter *interpreter,
+				     GdkEvent               *event)
+{
+  GtkGesturesInterpreterPrivate *priv;
+  RecordedGesture *recorded;
+  guint touch_id;
+  gdouble x, y;
+
+  g_return_val_if_fail (GTK_IS_GESTURES_INTERPRETER (interpreter), FALSE);
+  g_return_val_if_fail (event != NULL, FALSE);
+
+  priv = interpreter->priv;
+
+  if (!gdk_event_get_source_device (event))
+    return FALSE;
+
+  if (!gdk_event_get_coords (event, &x, &y))
+    return FALSE;
+
+  if (!gdk_event_get_touch_id (event, &touch_id))
+    touch_id = 0;
+
+  recorded = g_hash_table_lookup (priv->events,
+                                  GUINT_TO_POINTER (touch_id));
+  if (!recorded)
+    {
+      recorded = recorded_gesture_new ();
+      g_hash_table_insert (priv->events,
+                           GUINT_TO_POINTER (touch_id),
+                           recorded);
+    }
+
+  recorded_gesture_append_coordinate (recorded, x, y);
+
+  return TRUE;
+}
+
+gboolean
+gtk_gestures_interpreter_finish (GtkGesturesInterpreter *interpreter,
+                                 guint                  *gesture_id)
+{
+  g_return_val_if_fail (GTK_IS_GESTURES_INTERPRETER (interpreter), FALSE);
+
+  return TRUE;
+}
diff --git a/gtk/gtkgesturesinterpreter.h b/gtk/gtkgesturesinterpreter.h
new file mode 100644
index 0000000..72a53c4
--- /dev/null
+++ b/gtk/gtkgesturesinterpreter.h
@@ -0,0 +1,119 @@
+/* GTK - The GIMP Toolkit
+ * Copyright (C) 2011 Carlos Garnacho <carlosg gnome org>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#if !defined (__GTK_H_INSIDE__) && !defined (GTK_COMPILATION)
+#error "Only <gtk/gtk.h> can be included directly."
+#endif
+
+#ifndef __GTK_GESTURES_INTERPRETER_H__
+#define __GTK_GESTURES_INTERPRETER_H__
+
+#include <gtk/gtk.h>
+
+G_BEGIN_DECLS
+
+#define GTK_TYPE_GESTURE (gtk_gesture_get_type ())
+#define GTK_TYPE_GESTURE_STROKE (gtk_gesture_stroke_get_type ())
+
+#define GTK_TYPE_GESTURES_INTERPRETER         (gtk_gestures_interpreter_get_type ())
+#define GTK_GESTURES_INTERPRETER(o)           (G_TYPE_CHECK_INSTANCE_CAST ((o), GTK_TYPE_GESTURES_INTERPRETER, GtkGesturesInterpreter))
+#define GTK_GESTURES_INTERPRETER_CLASS(c)     (G_TYPE_CHECK_CLASS_CAST    ((c), GTK_TYPE_GESTURES_INTERPRETER, GtkGesturesInterpreterClass))
+#define GTK_IS_GESTURES_INTERPRETER(o)        (G_TYPE_CHECK_INSTANCE_TYPE ((o), GTK_TYPE_GESTURES_INTERPRETER))
+#define GTK_IS_GESTURES_INTERPRETER_CLASS(c)  (G_TYPE_CHECK_CLASS_TYPE    ((c), GTK_TYPE_GESTURES_INTERPRETER))
+#define GTK_GESTURES_INTERPRETER_GET_CLASS(o) (G_TYPE_INSTANCE_GET_CLASS  ((o), GTK_TYPE_GESTURES_INTERPRETER, GtkGesturesInterpreterClass))
+
+typedef struct _GtkGestureStroke GtkGestureStroke;
+typedef struct _GtkGesture GtkGesture;
+
+typedef struct _GtkGesturesInterpreter GtkGesturesInterpreter;
+typedef struct _GtkGesturesInterpreterClass GtkGesturesInterpreterClass;
+
+struct _GtkGesturesInterpreter
+{
+  GObject parent_object;
+  gpointer priv;
+};
+
+struct _GtkGesturesInterpreterClass
+{
+  GObjectClass parent_class;
+
+  void (* events_vectorized) (GtkGesturesInterpreter *interpreter,
+                              GtkGesture             *gesture);
+
+  void (* gesture_detected)  (GtkGesturesInterpreter *interpreter,
+                              guint                   gesture_id,
+                              gdouble                 confidence);
+};
+
+/* Gesture stroke */
+GType              gtk_gesture_stroke_get_type      (void) G_GNUC_CONST;
+GtkGestureStroke * gtk_gesture_stroke_new           (void);
+GtkGestureStroke * gtk_gesture_stroke_copy          (const GtkGestureStroke *stroke);
+void               gtk_gesture_stroke_free          (GtkGestureStroke *stroke);
+void               gtk_gesture_stroke_append_vector (GtkGestureStroke *stroke,
+						     gdouble           angle,
+						     guint             length);
+
+guint              gtk_gesture_stroke_get_n_vectors (const GtkGestureStroke *stroke);
+gboolean           gtk_gesture_stroke_get_vector    (const GtkGestureStroke *stroke,
+                                                     guint                   n_vector,
+                                                     gdouble                *angle,
+                                                     guint                  *length,
+                                                     gdouble                *relative_length);
+/* Gesture */
+GType        gtk_gesture_get_type   (void) G_GNUC_CONST;
+GtkGesture * gtk_gesture_new        (const GtkGestureStroke *stroke,
+                                     GtkGestureFlags         flags);
+void         gtk_gesture_add_stroke (GtkGesture             *gesture,
+				     const GtkGestureStroke *stroke,
+				     gint                    dx,
+				     gint                    dy);
+GtkGesture * gtk_gesture_copy       (const GtkGesture       *gesture);
+void         gtk_gesture_free       (GtkGesture             *gesture);
+
+GtkGestureFlags          gtk_gesture_get_flags     (const GtkGesture *gesture);
+guint                    gtk_gesture_get_n_strokes (const GtkGesture *gesture);
+const GtkGestureStroke * gtk_gesture_get_stroke    (const GtkGesture *gesture,
+                                                    guint             n_stroke,
+                                                    gint             *dx,
+                                                    gint             *dy);
+
+guint              gtk_gesture_register        (const GtkGesture *gesture);
+guint              gtk_gesture_register_static (const GtkGesture *gesture);
+const GtkGesture * gtk_gesture_lookup          (guint             gesture_id);
+
+/* Gestures interpreter */
+GType gtk_gestures_interpreter_get_type (void) G_GNUC_CONST;
+
+GtkGesturesInterpreter * gtk_gestures_interpreter_new (void);
+
+gboolean gtk_gestures_interpreter_add_gesture    (GtkGesturesInterpreter *interpreter,
+                                                  guint                   gesture_id);
+void     gtk_gestures_interpreter_remove_gesture (GtkGesturesInterpreter *interpreter,
+                                                  guint                   gesture_id);
+
+gboolean gtk_gestures_interpreter_feed_event (GtkGesturesInterpreter *interpreter,
+					      GdkEvent               *event);
+gboolean gtk_gestures_interpreter_finish     (GtkGesturesInterpreter *interpreter,
+                                              guint                  *gesture_id);
+
+G_END_DECLS
+
+#endif /* __GTK_GESTURES_INTERPRETER_H__ */
diff --git a/gtk/gtkmarshalers.list b/gtk/gtkmarshalers.list
index b75651f..b478e50 100644
--- a/gtk/gtkmarshalers.list
+++ b/gtk/gtkmarshalers.list
@@ -111,6 +111,7 @@ VOID:STRING,STRING,STRING
 VOID:STRING,INT,POINTER
 VOID:STRING,UINT,FLAGS
 VOID:STRING,UINT,FLAGS,UINT
+VOID:UINT,DOUBLE
 VOID:UINT,FLAGS,BOXED
 VOID:UINT,UINT
 VOID:UINT,STRING



[Date Prev][Date Next]   [Thread Prev][Thread Next]   [Thread Index] [Date Index] [Author Index]