forked from AuroraMiddleware/gtk
b151b1ba9e
Events that get to gtk_main_do_event() have the toplevel GdkWindow as event->any.window. Also, ensure that coordinates fall within sensible places of the windows, since those might have shadows, headerbars and whatnot on wayland.
1225 lines
35 KiB
C
1225 lines
35 KiB
C
#include <gtk/gtk.h>
|
|
|
|
typedef struct {
|
|
GtkWidget *widget;
|
|
gint x;
|
|
gint y;
|
|
guint state;
|
|
guint pressed : 1;
|
|
} PointState;
|
|
|
|
static PointState mouse_state;
|
|
static PointState touch_state[10]; /* touchpoint 0 gets pointer emulation,
|
|
* use it first in tests for consistency.
|
|
*/
|
|
|
|
#define EVENT_SEQUENCE(point) (GdkEventSequence*) ((point) - touch_state + 1)
|
|
|
|
static void
|
|
point_press (PointState *point,
|
|
GtkWidget *widget,
|
|
guint button)
|
|
{
|
|
GdkDisplay *display;
|
|
GdkDevice *device;
|
|
GdkSeat *seat;
|
|
GdkEvent *ev;
|
|
|
|
display = gtk_widget_get_display (widget);
|
|
seat = gdk_display_get_default_seat (display);
|
|
device = gdk_seat_get_pointer (seat);
|
|
|
|
if (point == &mouse_state)
|
|
{
|
|
ev = gdk_event_new (GDK_BUTTON_PRESS);
|
|
ev->any.window = g_object_ref (gtk_widget_get_window (widget));
|
|
ev->button.time = GDK_CURRENT_TIME;
|
|
ev->button.x = point->x;
|
|
ev->button.y = point->y;
|
|
ev->button.button = button;
|
|
ev->button.state = point->state;
|
|
|
|
point->state |= GDK_BUTTON1_MASK << (button - 1);
|
|
}
|
|
else
|
|
{
|
|
ev = gdk_event_new (GDK_TOUCH_BEGIN);
|
|
ev->any.window = g_object_ref (gtk_widget_get_window (widget));
|
|
ev->touch.time = GDK_CURRENT_TIME;
|
|
ev->touch.x = point->x;
|
|
ev->touch.y = point->y;
|
|
ev->touch.sequence = EVENT_SEQUENCE (point);
|
|
|
|
if (point == &touch_state[0])
|
|
ev->touch.emulating_pointer = TRUE;
|
|
}
|
|
|
|
gdk_event_set_device (ev, device);
|
|
|
|
gtk_main_do_event (ev);
|
|
|
|
gdk_event_free (ev);
|
|
|
|
point->widget = widget;
|
|
}
|
|
|
|
static void
|
|
point_update (PointState *point,
|
|
GtkWidget *widget,
|
|
gdouble x,
|
|
gdouble y)
|
|
{
|
|
GdkDisplay *display;
|
|
GdkDevice *device;
|
|
GdkSeat *seat;
|
|
GdkEvent *ev;
|
|
|
|
display = gtk_widget_get_display (widget);
|
|
seat = gdk_display_get_default_seat (display);
|
|
device = gdk_seat_get_pointer (seat);
|
|
|
|
point->x = x;
|
|
point->y = y;
|
|
|
|
if (point == &mouse_state)
|
|
{
|
|
ev = gdk_event_new (GDK_MOTION_NOTIFY);
|
|
ev->any.window = g_object_ref (gtk_widget_get_window (widget));
|
|
ev->button.time = GDK_CURRENT_TIME;
|
|
ev->motion.x = x;
|
|
ev->motion.y = y;
|
|
ev->motion.state = point->state;
|
|
}
|
|
else
|
|
{
|
|
if (!point->widget || widget != point->widget)
|
|
return;
|
|
|
|
ev = gdk_event_new (GDK_TOUCH_UPDATE);
|
|
ev->any.window = g_object_ref (gtk_widget_get_window (widget));
|
|
ev->touch.time = GDK_CURRENT_TIME;
|
|
ev->touch.x = x;
|
|
ev->touch.y = y;
|
|
ev->touch.sequence = EVENT_SEQUENCE (point);
|
|
ev->touch.state = 0;
|
|
|
|
if (point == &touch_state[0])
|
|
ev->touch.emulating_pointer = TRUE;
|
|
}
|
|
|
|
gdk_event_set_device (ev, device);
|
|
|
|
gtk_main_do_event (ev);
|
|
|
|
gdk_event_free (ev);
|
|
}
|
|
|
|
static void
|
|
point_release (PointState *point,
|
|
guint button)
|
|
{
|
|
GdkDisplay *display;
|
|
GdkDevice *device;
|
|
GdkSeat *seat;
|
|
GdkEvent *ev;
|
|
|
|
if (point->widget == NULL)
|
|
return;
|
|
|
|
display = gtk_widget_get_display (point->widget);
|
|
seat = gdk_display_get_default_seat (display);
|
|
device = gdk_seat_get_pointer (seat);
|
|
|
|
if (!point->widget)
|
|
return;
|
|
|
|
if (point == &mouse_state)
|
|
{
|
|
if ((point->state & (GDK_BUTTON1_MASK << (button - 1))) == 0)
|
|
return;
|
|
|
|
ev = gdk_event_new (GDK_BUTTON_RELEASE);
|
|
ev->any.window = g_object_ref (gtk_widget_get_window (point->widget));
|
|
ev->button.time = GDK_CURRENT_TIME;
|
|
ev->button.x = point->x;
|
|
ev->button.y = point->y;
|
|
ev->button.state = point->state;
|
|
|
|
point->state &= ~(GDK_BUTTON1_MASK << (button - 1));
|
|
}
|
|
else
|
|
{
|
|
ev = gdk_event_new (GDK_TOUCH_END);
|
|
ev->any.window = g_object_ref (gtk_widget_get_window (point->widget));
|
|
ev->touch.time = GDK_CURRENT_TIME;
|
|
ev->touch.x = point->x;
|
|
ev->touch.y = point->y;
|
|
ev->touch.sequence = EVENT_SEQUENCE (point);
|
|
ev->touch.state = point->state;
|
|
|
|
if (point == &touch_state[0])
|
|
ev->touch.emulating_pointer = TRUE;
|
|
}
|
|
|
|
gdk_event_set_device (ev, device);
|
|
|
|
gtk_main_do_event (ev);
|
|
|
|
gdk_event_free (ev);
|
|
}
|
|
|
|
static const gchar *
|
|
phase_nick (GtkPropagationPhase phase)
|
|
{
|
|
GTypeClass *class;
|
|
GEnumValue *value;
|
|
|
|
class = g_type_class_ref (GTK_TYPE_PROPAGATION_PHASE);
|
|
value = g_enum_get_value ((GEnumClass*)class, phase);
|
|
g_type_class_unref (class);
|
|
|
|
return value->value_nick;
|
|
}
|
|
|
|
static const gchar *
|
|
state_nick (GtkEventSequenceState state)
|
|
{
|
|
GTypeClass *class;
|
|
GEnumValue *value;
|
|
|
|
class = g_type_class_ref (GTK_TYPE_EVENT_SEQUENCE_STATE);
|
|
value = g_enum_get_value ((GEnumClass*)class, state);
|
|
g_type_class_unref (class);
|
|
|
|
return value->value_nick;
|
|
}
|
|
|
|
typedef struct {
|
|
GString *str;
|
|
gboolean exit;
|
|
} LegacyData;
|
|
|
|
static gboolean
|
|
legacy_cb (GtkWidget *w, GdkEventButton *button, gpointer data)
|
|
{
|
|
LegacyData *ld = data;
|
|
|
|
if (ld->str->len > 0)
|
|
g_string_append (ld->str, ", ");
|
|
g_string_append_printf (ld->str, "legacy %s", gtk_widget_get_name (w));
|
|
|
|
return ld->exit;
|
|
}
|
|
|
|
typedef struct {
|
|
GString *str;
|
|
GtkEventSequenceState state;
|
|
} GestureData;
|
|
|
|
static void
|
|
press_cb (GtkGesture *g, gint n_press, gdouble x, gdouble y, gpointer data)
|
|
{
|
|
GtkEventController *c = GTK_EVENT_CONTROLLER (g);
|
|
GdkEventSequence *sequence;
|
|
GtkPropagationPhase phase;
|
|
GestureData *gd = data;
|
|
const gchar *name;
|
|
|
|
name = g_object_get_data (G_OBJECT (g), "name");
|
|
phase = gtk_event_controller_get_propagation_phase (c);
|
|
|
|
if (gd->str->len > 0)
|
|
g_string_append (gd->str, ", ");
|
|
g_string_append_printf (gd->str, "%s %s", phase_nick (phase), name);
|
|
|
|
sequence = gtk_gesture_get_last_updated_sequence (g);
|
|
|
|
if (sequence)
|
|
g_string_append_printf (gd->str, " (%x)", GPOINTER_TO_UINT (sequence));
|
|
|
|
if (gd->state != GTK_EVENT_SEQUENCE_NONE)
|
|
gtk_gesture_set_state (g, gd->state);
|
|
}
|
|
|
|
static void
|
|
cancel_cb (GtkGesture *g, GdkEventSequence *sequence, gpointer data)
|
|
{
|
|
GestureData *gd = data;
|
|
const gchar *name;
|
|
|
|
name = g_object_get_data (G_OBJECT (g), "name");
|
|
|
|
if (gd->str->len > 0)
|
|
g_string_append (gd->str, ", ");
|
|
g_string_append_printf (gd->str, "%s cancelled", name);
|
|
}
|
|
|
|
static void
|
|
begin_cb (GtkGesture *g, GdkEventSequence *sequence, gpointer data)
|
|
{
|
|
GestureData *gd = data;
|
|
const gchar *name;
|
|
|
|
name = g_object_get_data (G_OBJECT (g), "name");
|
|
|
|
if (gd->str->len > 0)
|
|
g_string_append (gd->str, ", ");
|
|
g_string_append_printf (gd->str, "%s began", name);
|
|
|
|
if (gd->state != GTK_EVENT_SEQUENCE_NONE)
|
|
gtk_gesture_set_state (g, gd->state);
|
|
}
|
|
|
|
static void
|
|
end_cb (GtkGesture *g, GdkEventSequence *sequence, gpointer data)
|
|
{
|
|
GestureData *gd = data;
|
|
const gchar *name;
|
|
|
|
name = g_object_get_data (G_OBJECT (g), "name");
|
|
|
|
if (gd->str->len > 0)
|
|
g_string_append (gd->str, ", ");
|
|
g_string_append_printf (gd->str, "%s ended", name);
|
|
}
|
|
|
|
static void
|
|
update_cb (GtkGesture *g, GdkEventSequence *sequence, gpointer data)
|
|
{
|
|
GestureData *gd = data;
|
|
const gchar *name;
|
|
|
|
name = g_object_get_data (G_OBJECT (g), "name");
|
|
|
|
if (gd->str->len > 0)
|
|
g_string_append (gd->str, ", ");
|
|
g_string_append_printf (gd->str, "%s updated", name);
|
|
}
|
|
|
|
static void
|
|
state_changed_cb (GtkGesture *g, GdkEventSequence *sequence, GtkEventSequenceState state, gpointer data)
|
|
{
|
|
GestureData *gd = data;
|
|
const gchar *name;
|
|
|
|
name = g_object_get_data (G_OBJECT (g), "name");
|
|
|
|
if (gd->str->len > 0)
|
|
g_string_append (gd->str, ", ");
|
|
g_string_append_printf (gd->str, "%s state %s", name, state_nick (state));
|
|
|
|
if (sequence != NULL)
|
|
g_string_append_printf (gd->str, " (%x)", GPOINTER_TO_UINT (sequence));
|
|
}
|
|
|
|
|
|
static GtkGesture *
|
|
add_gesture (GtkWidget *w, const gchar *name, GtkPropagationPhase phase, GString *str, GtkEventSequenceState state)
|
|
{
|
|
GtkGesture *g;
|
|
GestureData *data;
|
|
|
|
data = g_new (GestureData, 1);
|
|
data->str = str;
|
|
data->state = state;
|
|
|
|
g = gtk_gesture_multi_press_new (w);
|
|
gtk_gesture_single_set_touch_only (GTK_GESTURE_SINGLE (g), FALSE);
|
|
gtk_gesture_single_set_button (GTK_GESTURE_SINGLE (g), 1);
|
|
gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (g), phase);
|
|
|
|
g_object_set_data (G_OBJECT (g), "name", (gpointer)name);
|
|
|
|
g_signal_connect (g, "pressed", G_CALLBACK (press_cb), data);
|
|
g_signal_connect (g, "cancel", G_CALLBACK (cancel_cb), data);
|
|
g_signal_connect (g, "update", G_CALLBACK (update_cb), data);
|
|
g_signal_connect (g, "sequence-state-changed", G_CALLBACK (state_changed_cb), data);
|
|
|
|
return g;
|
|
}
|
|
|
|
static GtkGesture *
|
|
add_mt_gesture (GtkWidget *w, const gchar *name, GtkPropagationPhase phase, GString *str, GtkEventSequenceState state)
|
|
{
|
|
GtkGesture *g;
|
|
GestureData *data;
|
|
|
|
data = g_new (GestureData, 1);
|
|
data->str = str;
|
|
data->state = state;
|
|
|
|
g = gtk_gesture_rotate_new (w);
|
|
gtk_event_controller_set_propagation_phase (GTK_EVENT_CONTROLLER (g), phase);
|
|
|
|
g_object_set_data (G_OBJECT (g), "name", (gpointer)name);
|
|
|
|
g_signal_connect (g, "begin", G_CALLBACK (begin_cb), data);
|
|
g_signal_connect (g, "update", G_CALLBACK (update_cb), data);
|
|
g_signal_connect (g, "end", G_CALLBACK (end_cb), data);
|
|
g_signal_connect (g, "sequence-state-changed", G_CALLBACK (state_changed_cb), data);
|
|
|
|
return g;
|
|
}
|
|
|
|
static void
|
|
add_legacy (GtkWidget *w, GString *str, gboolean exit)
|
|
{
|
|
LegacyData *data;
|
|
|
|
data = g_new (LegacyData, 1);
|
|
data->str = str;
|
|
data->exit = exit;
|
|
g_signal_connect (w, "button-press-event", G_CALLBACK (legacy_cb), data);
|
|
}
|
|
|
|
static void
|
|
test_phases (void)
|
|
{
|
|
GtkWidget *A, *B, *C;
|
|
GString *str;
|
|
GtkAllocation allocation;
|
|
|
|
A = gtk_window_new (GTK_WINDOW_TOPLEVEL);
|
|
gtk_widget_set_name (A, "A");
|
|
B = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
|
|
gtk_widget_set_name (B, "B");
|
|
C = gtk_event_box_new ();
|
|
gtk_widget_set_hexpand (C, TRUE);
|
|
gtk_widget_set_vexpand (C, TRUE);
|
|
gtk_widget_set_name (C, "C");
|
|
|
|
gtk_container_add (GTK_CONTAINER (A), B);
|
|
gtk_container_add (GTK_CONTAINER (B), C);
|
|
|
|
gtk_widget_show (A);
|
|
|
|
str = g_string_new ("");
|
|
|
|
add_gesture (A, "a1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (B, "b1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (C, "c1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (A, "a2", GTK_PHASE_TARGET, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (B, "b2", GTK_PHASE_TARGET, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (C, "c2", GTK_PHASE_TARGET, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (A, "a3", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (B, "b3", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (C, "c3", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
|
|
gtk_widget_get_allocation (A, &allocation);
|
|
|
|
point_update (&mouse_state, A, allocation.width / 2, allocation.height / 2);
|
|
point_press (&mouse_state, A, 1);
|
|
|
|
g_assert_cmpstr (str->str, ==,
|
|
"capture a1, "
|
|
"capture b1, "
|
|
"capture c1, "
|
|
"target c2, "
|
|
"bubble c3, "
|
|
"bubble b3, "
|
|
"bubble a3");
|
|
|
|
g_string_free (str, TRUE);
|
|
|
|
gtk_widget_destroy (A);
|
|
}
|
|
|
|
static void
|
|
test_mixed (void)
|
|
{
|
|
GtkWidget *A, *B, *C;
|
|
GString *str;
|
|
GtkAllocation allocation;
|
|
|
|
A = gtk_window_new (GTK_WINDOW_TOPLEVEL);
|
|
gtk_widget_set_name (A, "A");
|
|
B = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
|
|
gtk_widget_set_name (B, "B");
|
|
C = gtk_event_box_new ();
|
|
gtk_widget_set_hexpand (C, TRUE);
|
|
gtk_widget_set_vexpand (C, TRUE);
|
|
gtk_widget_set_name (C, "C");
|
|
|
|
gtk_container_add (GTK_CONTAINER (A), B);
|
|
gtk_container_add (GTK_CONTAINER (B), C);
|
|
|
|
gtk_widget_show (A);
|
|
|
|
str = g_string_new ("");
|
|
|
|
add_gesture (A, "a1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (B, "b1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (C, "c1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (A, "a2", GTK_PHASE_TARGET, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (B, "b2", GTK_PHASE_TARGET, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (C, "c2", GTK_PHASE_TARGET, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (A, "a3", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (B, "b3", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (C, "c3", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
|
|
add_legacy (A, str, GDK_EVENT_PROPAGATE);
|
|
add_legacy (B, str, GDK_EVENT_PROPAGATE);
|
|
add_legacy (C, str, GDK_EVENT_PROPAGATE);
|
|
|
|
gtk_widget_get_allocation (A, &allocation);
|
|
|
|
point_update (&mouse_state, A, allocation.width / 2, allocation.height / 2);
|
|
point_press (&mouse_state, A, 1);
|
|
|
|
g_assert_cmpstr (str->str, ==,
|
|
"capture a1, "
|
|
"capture b1, "
|
|
"capture c1, "
|
|
"target c2, "
|
|
"legacy C, "
|
|
"bubble c3, "
|
|
"legacy B, "
|
|
"bubble b3, "
|
|
"legacy A, "
|
|
"bubble a3");
|
|
|
|
g_string_free (str, TRUE);
|
|
|
|
gtk_widget_destroy (A);
|
|
}
|
|
|
|
static void
|
|
test_early_exit (void)
|
|
{
|
|
GtkWidget *A, *B, *C;
|
|
GString *str;
|
|
GtkAllocation allocation;
|
|
|
|
A = gtk_window_new (GTK_WINDOW_TOPLEVEL);
|
|
gtk_widget_set_name (A, "A");
|
|
B = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
|
|
gtk_widget_set_name (B, "B");
|
|
C = gtk_event_box_new ();
|
|
gtk_widget_set_hexpand (C, TRUE);
|
|
gtk_widget_set_vexpand (C, TRUE);
|
|
gtk_widget_set_name (C, "C");
|
|
|
|
gtk_container_add (GTK_CONTAINER (A), B);
|
|
gtk_container_add (GTK_CONTAINER (B), C);
|
|
|
|
gtk_widget_show (A);
|
|
|
|
str = g_string_new ("");
|
|
|
|
add_gesture (A, "a1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (B, "b1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (C, "c1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (C, "c2", GTK_PHASE_TARGET, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (A, "a3", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (B, "b3", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (C, "c3", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
|
|
add_legacy (A, str, GDK_EVENT_PROPAGATE);
|
|
add_legacy (B, str, GDK_EVENT_STOP);
|
|
add_legacy (C, str, GDK_EVENT_PROPAGATE);
|
|
|
|
gtk_widget_get_allocation (A, &allocation);
|
|
|
|
point_update (&mouse_state, A, allocation.width / 2, allocation.height / 2);
|
|
point_press (&mouse_state, A, 1);
|
|
|
|
g_assert_cmpstr (str->str, ==,
|
|
"capture a1, "
|
|
"capture b1, "
|
|
"capture c1, "
|
|
"target c2, "
|
|
"legacy C, "
|
|
"bubble c3, "
|
|
"legacy B");
|
|
|
|
g_string_free (str, TRUE);
|
|
|
|
gtk_widget_destroy (A);
|
|
}
|
|
|
|
static void
|
|
test_claim_capture (void)
|
|
{
|
|
GtkWidget *A, *B, *C;
|
|
GString *str;
|
|
GtkAllocation allocation;
|
|
|
|
A = gtk_window_new (GTK_WINDOW_TOPLEVEL);
|
|
gtk_widget_set_name (A, "A");
|
|
B = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
|
|
gtk_widget_set_name (B, "B");
|
|
C = gtk_event_box_new ();
|
|
gtk_widget_set_hexpand (C, TRUE);
|
|
gtk_widget_set_vexpand (C, TRUE);
|
|
gtk_widget_set_name (C, "C");
|
|
|
|
gtk_container_add (GTK_CONTAINER (A), B);
|
|
gtk_container_add (GTK_CONTAINER (B), C);
|
|
|
|
gtk_widget_show (A);
|
|
|
|
str = g_string_new ("");
|
|
|
|
add_gesture (A, "a1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (B, "b1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (C, "c1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_CLAIMED);
|
|
add_gesture (C, "c2", GTK_PHASE_TARGET, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (A, "a3", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (B, "b3", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (C, "c3", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
|
|
gtk_widget_get_allocation (A, &allocation);
|
|
|
|
point_update (&mouse_state, A, allocation.width / 2, allocation.height / 2);
|
|
point_press (&mouse_state, A, 1);
|
|
|
|
g_assert_cmpstr (str->str, ==,
|
|
"capture a1, "
|
|
"capture b1, "
|
|
"capture c1, "
|
|
"c1 state claimed");
|
|
|
|
g_string_free (str, TRUE);
|
|
|
|
gtk_widget_destroy (A);
|
|
}
|
|
|
|
static void
|
|
test_claim_target (void)
|
|
{
|
|
GtkWidget *A, *B, *C;
|
|
GString *str;
|
|
GtkAllocation allocation;
|
|
|
|
A = gtk_window_new (GTK_WINDOW_TOPLEVEL);
|
|
gtk_widget_set_name (A, "A");
|
|
B = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
|
|
gtk_widget_set_name (B, "B");
|
|
C = gtk_event_box_new ();
|
|
gtk_widget_set_hexpand (C, TRUE);
|
|
gtk_widget_set_vexpand (C, TRUE);
|
|
gtk_widget_set_name (C, "C");
|
|
|
|
gtk_container_add (GTK_CONTAINER (A), B);
|
|
gtk_container_add (GTK_CONTAINER (B), C);
|
|
|
|
gtk_widget_show (A);
|
|
|
|
str = g_string_new ("");
|
|
|
|
add_gesture (A, "a1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (B, "b1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (C, "c1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (C, "c2", GTK_PHASE_TARGET, str, GTK_EVENT_SEQUENCE_CLAIMED);
|
|
add_gesture (A, "a3", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (B, "b3", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (C, "c3", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
|
|
gtk_widget_get_allocation (A, &allocation);
|
|
point_update (&mouse_state, A, allocation.width / 2, allocation.height / 2);
|
|
point_press (&mouse_state, A, 1);
|
|
|
|
g_assert_cmpstr (str->str, ==,
|
|
"capture a1, "
|
|
"capture b1, "
|
|
"capture c1, "
|
|
"target c2, "
|
|
"c2 state claimed");
|
|
|
|
g_string_free (str, TRUE);
|
|
|
|
gtk_widget_destroy (A);
|
|
}
|
|
|
|
static void
|
|
test_claim_bubble (void)
|
|
{
|
|
GtkWidget *A, *B, *C;
|
|
GString *str;
|
|
GtkAllocation allocation;
|
|
|
|
A = gtk_window_new (GTK_WINDOW_TOPLEVEL);
|
|
gtk_widget_set_name (A, "A");
|
|
B = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
|
|
gtk_widget_set_name (B, "B");
|
|
C = gtk_event_box_new ();
|
|
gtk_widget_set_hexpand (C, TRUE);
|
|
gtk_widget_set_vexpand (C, TRUE);
|
|
gtk_widget_set_name (C, "C");
|
|
|
|
gtk_container_add (GTK_CONTAINER (A), B);
|
|
gtk_container_add (GTK_CONTAINER (B), C);
|
|
|
|
gtk_widget_show (A);
|
|
|
|
str = g_string_new ("");
|
|
|
|
add_gesture (A, "a1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (B, "b1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (C, "c1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (C, "c2", GTK_PHASE_TARGET, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (A, "a3", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (B, "b3", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_CLAIMED);
|
|
add_gesture (C, "c3", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
|
|
gtk_widget_get_allocation (A, &allocation);
|
|
point_update (&mouse_state, A, allocation.width / 2, allocation.height / 2);
|
|
point_press (&mouse_state, A, 1);
|
|
|
|
g_assert_cmpstr (str->str, ==,
|
|
"capture a1, "
|
|
"capture b1, "
|
|
"capture c1, "
|
|
"target c2, "
|
|
"bubble c3, "
|
|
"bubble b3, "
|
|
"c3 cancelled, "
|
|
"c2 cancelled, "
|
|
"c1 cancelled, "
|
|
"b3 state claimed"
|
|
);
|
|
|
|
g_string_free (str, TRUE);
|
|
|
|
gtk_widget_destroy (A);
|
|
}
|
|
|
|
static void
|
|
test_early_claim_capture (void)
|
|
{
|
|
GtkWidget *A, *B, *C;
|
|
GtkGesture *g;
|
|
GString *str;
|
|
GtkAllocation allocation;
|
|
|
|
A = gtk_window_new (GTK_WINDOW_TOPLEVEL);
|
|
gtk_widget_set_name (A, "A");
|
|
B = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
|
|
gtk_widget_set_name (B, "B");
|
|
C = gtk_event_box_new ();
|
|
gtk_widget_set_hexpand (C, TRUE);
|
|
gtk_widget_set_vexpand (C, TRUE);
|
|
gtk_widget_set_name (C, "C");
|
|
|
|
gtk_container_add (GTK_CONTAINER (A), B);
|
|
gtk_container_add (GTK_CONTAINER (B), C);
|
|
|
|
gtk_widget_show (A);
|
|
|
|
str = g_string_new ("");
|
|
|
|
add_gesture (A, "a1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
g = add_gesture (B, "b1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_CLAIMED);
|
|
add_gesture (C, "c1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_CLAIMED);
|
|
add_gesture (C, "c2", GTK_PHASE_TARGET, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (A, "a3", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (B, "b3", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (C, "c3", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
|
|
gtk_widget_get_allocation (A, &allocation);
|
|
point_update (&mouse_state, A, allocation.width / 2, allocation.height / 2);
|
|
point_press (&mouse_state, A, 1);
|
|
|
|
g_assert_cmpstr (str->str, ==,
|
|
"capture a1, "
|
|
"capture b1, "
|
|
"b1 state claimed");
|
|
|
|
/* Reset the string */
|
|
g_string_erase (str, 0, str->len);
|
|
|
|
gtk_gesture_set_state (g, GTK_EVENT_SEQUENCE_DENIED);
|
|
|
|
g_assert_cmpstr (str->str, ==,
|
|
"capture c1, "
|
|
"c1 state claimed, "
|
|
"b1 state denied");
|
|
|
|
point_release (&mouse_state, 1);
|
|
|
|
g_string_free (str, TRUE);
|
|
gtk_widget_destroy (A);
|
|
}
|
|
|
|
static void
|
|
test_late_claim_capture (void)
|
|
{
|
|
GtkWidget *A, *B, *C;
|
|
GtkGesture *g;
|
|
GString *str;
|
|
GtkAllocation allocation;
|
|
|
|
A = gtk_window_new (GTK_WINDOW_TOPLEVEL);
|
|
gtk_widget_set_name (A, "A");
|
|
B = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
|
|
gtk_widget_set_name (B, "B");
|
|
C = gtk_event_box_new ();
|
|
gtk_widget_set_hexpand (C, TRUE);
|
|
gtk_widget_set_vexpand (C, TRUE);
|
|
gtk_widget_set_name (C, "C");
|
|
|
|
gtk_container_add (GTK_CONTAINER (A), B);
|
|
gtk_container_add (GTK_CONTAINER (B), C);
|
|
|
|
gtk_widget_show (A);
|
|
|
|
str = g_string_new ("");
|
|
|
|
add_gesture (A, "a1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
g = add_gesture (B, "b1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (C, "c1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (C, "c2", GTK_PHASE_TARGET, str, GTK_EVENT_SEQUENCE_CLAIMED);
|
|
add_gesture (A, "a3", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (B, "b3", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (C, "c3", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
|
|
gtk_widget_get_allocation (A, &allocation);
|
|
point_update (&mouse_state, A, allocation.width / 2, allocation.height / 2);
|
|
point_press (&mouse_state, A, 1);
|
|
|
|
g_assert_cmpstr (str->str, ==,
|
|
"capture a1, "
|
|
"capture b1, "
|
|
"capture c1, "
|
|
"target c2, "
|
|
"c2 state claimed");
|
|
|
|
/* Reset the string */
|
|
g_string_erase (str, 0, str->len);
|
|
|
|
gtk_gesture_set_state (g, GTK_EVENT_SEQUENCE_CLAIMED);
|
|
|
|
g_assert_cmpstr (str->str, ==,
|
|
"c2 cancelled, "
|
|
"c1 cancelled, "
|
|
"b1 state claimed");
|
|
|
|
point_release (&mouse_state, 1);
|
|
|
|
g_string_free (str, TRUE);
|
|
gtk_widget_destroy (A);
|
|
}
|
|
|
|
static void
|
|
test_group (void)
|
|
{
|
|
GtkWidget *A, *B, *C;
|
|
GString *str;
|
|
GtkGesture *g1, *g2;
|
|
GtkAllocation allocation;
|
|
|
|
A = gtk_window_new (GTK_WINDOW_TOPLEVEL);
|
|
gtk_widget_set_name (A, "A");
|
|
B = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
|
|
gtk_widget_set_name (B, "B");
|
|
C = gtk_event_box_new ();
|
|
gtk_widget_set_hexpand (C, TRUE);
|
|
gtk_widget_set_vexpand (C, TRUE);
|
|
gtk_widget_set_name (C, "C");
|
|
|
|
gtk_container_add (GTK_CONTAINER (A), B);
|
|
gtk_container_add (GTK_CONTAINER (B), C);
|
|
|
|
gtk_widget_show (A);
|
|
|
|
str = g_string_new ("");
|
|
|
|
add_gesture (A, "a1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (B, "b1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (C, "c1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
g1 = add_gesture (C, "c2", GTK_PHASE_TARGET, str, GTK_EVENT_SEQUENCE_NONE);
|
|
g2 = add_gesture (C, "c3", GTK_PHASE_TARGET, str, GTK_EVENT_SEQUENCE_CLAIMED);
|
|
gtk_gesture_group (g1, g2);
|
|
add_gesture (A, "a3", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (B, "b3", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (C, "c4", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
|
|
gtk_widget_get_allocation (A, &allocation);
|
|
|
|
gtk_widget_get_allocation (A, &allocation);
|
|
point_update (&mouse_state, A, allocation.width / 2, allocation.height / 2);
|
|
point_press (&mouse_state, A, 1);
|
|
|
|
g_assert_cmpstr (str->str, ==,
|
|
"capture a1, "
|
|
"capture b1, "
|
|
"capture c1, "
|
|
"target c3, "
|
|
"c3 state claimed, "
|
|
"c2 state claimed, "
|
|
"target c2");
|
|
|
|
g_string_free (str, TRUE);
|
|
|
|
gtk_widget_destroy (A);
|
|
}
|
|
|
|
static void
|
|
test_gestures_outside_grab (void)
|
|
{
|
|
GtkWidget *A, *B, *C, *D;
|
|
GString *str;
|
|
GtkAllocation allocation;
|
|
|
|
A = gtk_window_new (GTK_WINDOW_TOPLEVEL);
|
|
gtk_widget_set_name (A, "A");
|
|
B = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
|
|
gtk_widget_set_name (B, "B");
|
|
C = gtk_event_box_new ();
|
|
gtk_widget_set_hexpand (C, TRUE);
|
|
gtk_widget_set_vexpand (C, TRUE);
|
|
gtk_widget_set_name (C, "C");
|
|
|
|
gtk_container_add (GTK_CONTAINER (A), B);
|
|
gtk_container_add (GTK_CONTAINER (B), C);
|
|
|
|
gtk_widget_show (A);
|
|
|
|
D = gtk_window_new (GTK_WINDOW_TOPLEVEL);
|
|
gtk_widget_show (D);
|
|
|
|
str = g_string_new ("");
|
|
|
|
add_gesture (A, "a1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (B, "b1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (C, "c1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (C, "c2", GTK_PHASE_TARGET, str, GTK_EVENT_SEQUENCE_CLAIMED);
|
|
add_gesture (B, "b2", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (A, "a2", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
|
|
gtk_widget_get_allocation (A, &allocation);
|
|
point_update (&mouse_state, A, allocation.width / 2, allocation.height / 2);
|
|
point_press (&mouse_state, A, 1);
|
|
|
|
g_assert_cmpstr (str->str, ==,
|
|
"capture a1, "
|
|
"capture b1, "
|
|
"capture c1, "
|
|
"target c2, "
|
|
"c2 state claimed");
|
|
|
|
/* Set a grab on another window */
|
|
g_string_erase (str, 0, str->len);
|
|
gtk_grab_add (D);
|
|
|
|
g_assert_cmpstr (str->str, ==,
|
|
"c1 cancelled, "
|
|
"c2 cancelled, "
|
|
"b1 cancelled, "
|
|
"a1 cancelled");
|
|
|
|
g_string_free (str, TRUE);
|
|
|
|
gtk_widget_destroy (A);
|
|
gtk_widget_destroy (D);
|
|
}
|
|
|
|
static void
|
|
test_gestures_inside_grab (void)
|
|
{
|
|
GtkWidget *A, *B, *C;
|
|
GString *str;
|
|
GtkAllocation allocation;
|
|
|
|
A = gtk_window_new (GTK_WINDOW_TOPLEVEL);
|
|
gtk_widget_set_name (A, "A");
|
|
B = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
|
|
gtk_widget_set_name (B, "B");
|
|
C = gtk_event_box_new ();
|
|
gtk_widget_set_hexpand (C, TRUE);
|
|
gtk_widget_set_vexpand (C, TRUE);
|
|
gtk_widget_set_name (C, "C");
|
|
|
|
gtk_container_add (GTK_CONTAINER (A), B);
|
|
gtk_container_add (GTK_CONTAINER (B), C);
|
|
|
|
gtk_widget_show (A);
|
|
|
|
str = g_string_new ("");
|
|
|
|
add_gesture (A, "a1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (B, "b1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (C, "c1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (C, "c2", GTK_PHASE_TARGET, str, GTK_EVENT_SEQUENCE_CLAIMED);
|
|
add_gesture (B, "b2", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (A, "a2", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
|
|
gtk_widget_get_allocation (A, &allocation);
|
|
point_update (&mouse_state, A, allocation.width / 2, allocation.height / 2);
|
|
point_press (&mouse_state, A, 1);
|
|
|
|
g_assert_cmpstr (str->str, ==,
|
|
"capture a1, "
|
|
"capture b1, "
|
|
"capture c1, "
|
|
"target c2, "
|
|
"c2 state claimed");
|
|
|
|
/* Set a grab on B */
|
|
g_string_erase (str, 0, str->len);
|
|
gtk_grab_add (B);
|
|
g_assert_cmpstr (str->str, ==,
|
|
"a1 cancelled");
|
|
|
|
/* Update with the grab under effect */
|
|
g_string_erase (str, 0, str->len);
|
|
point_update (&mouse_state, A, (allocation.width / 2) + 20,
|
|
(allocation.height / 2) + 20);
|
|
g_assert_cmpstr (str->str, ==,
|
|
"b1 updated, "
|
|
"c1 updated, "
|
|
"c2 updated");
|
|
|
|
g_string_free (str, TRUE);
|
|
|
|
gtk_widget_destroy (A);
|
|
}
|
|
|
|
static void
|
|
test_multitouch_on_single (void)
|
|
{
|
|
GtkWidget *A, *B, *C;
|
|
GString *str;
|
|
GtkAllocation allocation;
|
|
|
|
A = gtk_window_new (GTK_WINDOW_TOPLEVEL);
|
|
gtk_widget_set_name (A, "A");
|
|
B = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
|
|
gtk_widget_set_name (B, "B");
|
|
C = gtk_event_box_new ();
|
|
gtk_widget_set_hexpand (C, TRUE);
|
|
gtk_widget_set_vexpand (C, TRUE);
|
|
gtk_widget_set_name (C, "C");
|
|
|
|
gtk_container_add (GTK_CONTAINER (A), B);
|
|
gtk_container_add (GTK_CONTAINER (B), C);
|
|
|
|
gtk_widget_show (A);
|
|
|
|
str = g_string_new ("");
|
|
|
|
add_gesture (A, "a1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_NONE);
|
|
add_gesture (B, "b1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_CLAIMED);
|
|
|
|
gtk_widget_get_allocation (A, &allocation);
|
|
|
|
/* First touch down */
|
|
point_update (&touch_state[0], A, allocation.width / 2, allocation.height / 2);
|
|
point_press (&touch_state[0], A, 1);
|
|
|
|
g_assert_cmpstr (str->str, ==,
|
|
"capture a1 (1), "
|
|
"capture b1 (1), "
|
|
"b1 state claimed (1)");
|
|
|
|
/* Second touch down */
|
|
g_string_erase (str, 0, str->len);
|
|
point_update (&touch_state[1], A, (allocation.width / 2) + 20,
|
|
(allocation.height / 2) + 20);
|
|
point_press (&touch_state[1], A, 1);
|
|
|
|
g_assert_cmpstr (str->str, ==,
|
|
"a1 state denied (2), "
|
|
"b1 state denied (2)");
|
|
|
|
g_string_free (str, TRUE);
|
|
|
|
gtk_widget_destroy (A);
|
|
}
|
|
|
|
static void
|
|
test_multitouch_activation (void)
|
|
{
|
|
GtkWidget *A, *B, *C;
|
|
GString *str;
|
|
GtkAllocation allocation;
|
|
|
|
A = gtk_window_new (GTK_WINDOW_TOPLEVEL);
|
|
gtk_widget_set_name (A, "A");
|
|
B = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
|
|
gtk_widget_set_name (B, "B");
|
|
C = gtk_event_box_new ();
|
|
gtk_widget_set_hexpand (C, TRUE);
|
|
gtk_widget_set_vexpand (C, TRUE);
|
|
gtk_widget_set_name (C, "C");
|
|
|
|
gtk_container_add (GTK_CONTAINER (A), B);
|
|
gtk_container_add (GTK_CONTAINER (B), C);
|
|
|
|
gtk_widget_show (A);
|
|
|
|
str = g_string_new ("");
|
|
|
|
add_mt_gesture (C, "c1", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_CLAIMED);
|
|
gtk_widget_get_allocation (A, &allocation);
|
|
|
|
/* First touch down */
|
|
point_update (&touch_state[0], A, allocation.width / 2, allocation.height / 2);
|
|
point_press (&touch_state[0], A, 1);
|
|
|
|
g_assert_cmpstr (str->str, ==, "");
|
|
|
|
/* Second touch down */
|
|
point_update (&touch_state[1], A, (allocation.width / 2) + 20,
|
|
(allocation.height / 2) + 20);
|
|
point_press (&touch_state[1], A, 1);
|
|
|
|
g_assert_cmpstr (str->str, ==,
|
|
"c1 began, "
|
|
"c1 state claimed (2), "
|
|
"c1 state claimed");
|
|
|
|
/* First touch up */
|
|
g_string_erase (str, 0, str->len);
|
|
point_release (&touch_state[0], 1);
|
|
|
|
g_assert_cmpstr (str->str, ==,
|
|
"c1 ended");
|
|
|
|
/* A third touch down triggering again action */
|
|
g_string_erase (str, 0, str->len);
|
|
point_update (&touch_state[2], A, (allocation.width / 2) + 20,
|
|
(allocation.height / 2) + 20);
|
|
point_press (&touch_state[2], A, 1);
|
|
|
|
g_assert_cmpstr (str->str, ==,
|
|
"c1 began, "
|
|
"c1 state claimed (3)");
|
|
|
|
/* One touch up, gesture is finished again */
|
|
g_string_erase (str, 0, str->len);
|
|
point_release (&touch_state[2], 1);
|
|
|
|
g_assert_cmpstr (str->str, ==,
|
|
"c1 ended");
|
|
|
|
/* Another touch up, gesture remains inactive */
|
|
g_string_erase (str, 0, str->len);
|
|
point_release (&touch_state[1], 1);
|
|
|
|
g_assert_cmpstr (str->str, ==, "");
|
|
|
|
g_string_free (str, TRUE);
|
|
|
|
gtk_widget_destroy (A);
|
|
}
|
|
|
|
static void
|
|
test_multitouch_interaction (void)
|
|
{
|
|
GtkWidget *A, *B, *C;
|
|
GtkGesture *g;
|
|
GString *str;
|
|
GtkAllocation allocation;
|
|
|
|
A = gtk_window_new (GTK_WINDOW_TOPLEVEL);
|
|
gtk_widget_set_name (A, "A");
|
|
B = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
|
|
gtk_widget_set_name (B, "B");
|
|
C = gtk_event_box_new ();
|
|
gtk_widget_set_hexpand (C, TRUE);
|
|
gtk_widget_set_vexpand (C, TRUE);
|
|
gtk_widget_set_name (C, "C");
|
|
|
|
gtk_container_add (GTK_CONTAINER (A), B);
|
|
gtk_container_add (GTK_CONTAINER (B), C);
|
|
|
|
gtk_widget_show (A);
|
|
|
|
str = g_string_new ("");
|
|
|
|
g = add_gesture (A, "a1", GTK_PHASE_CAPTURE, str, GTK_EVENT_SEQUENCE_CLAIMED);
|
|
add_mt_gesture (C, "c1", GTK_PHASE_BUBBLE, str, GTK_EVENT_SEQUENCE_CLAIMED);
|
|
gtk_widget_get_allocation (A, &allocation);
|
|
|
|
/* First touch down, a1 claims the sequence */
|
|
point_update (&touch_state[0], A, allocation.width / 2, allocation.height / 2);
|
|
point_press (&touch_state[0], A, 1);
|
|
|
|
g_assert_cmpstr (str->str, ==,
|
|
"capture a1 (1), "
|
|
"a1 state claimed (1)");
|
|
|
|
/* Second touch down, a1 denies and c1 takes over */
|
|
g_string_erase (str, 0, str->len);
|
|
point_update (&touch_state[1], A, (allocation.width / 2) + 20,
|
|
(allocation.height / 2) + 20);
|
|
point_press (&touch_state[1], A, 1);
|
|
|
|
/* Denying sequences in touch-excess situation is a responsibility of the caller */
|
|
gtk_gesture_set_state (g, GTK_EVENT_SEQUENCE_DENIED);
|
|
|
|
g_assert_cmpstr (str->str, ==,
|
|
"a1 state denied (2), "
|
|
"c1 began, "
|
|
"c1 state claimed, "
|
|
"c1 state claimed (2), "
|
|
"a1 state denied (1)");
|
|
|
|
/* Move first point, only c1 should update */
|
|
g_string_erase (str, 0, str->len);
|
|
point_update (&touch_state[0], A, (allocation.width / 2) + 30,
|
|
(allocation.height / 2) + 30);
|
|
|
|
g_assert_cmpstr (str->str, ==,
|
|
"c1 updated");
|
|
|
|
/* First touch up */
|
|
g_string_erase (str, 0, str->len);
|
|
point_release (&touch_state[0], 1);
|
|
|
|
g_assert_cmpstr (str->str, ==,
|
|
"c1 ended");
|
|
|
|
/* A third touch down triggering again action on c1 */
|
|
g_string_erase (str, 0, str->len);
|
|
point_update (&touch_state[2], A, (allocation.width / 2) + 20,
|
|
(allocation.height / 2) + 20);
|
|
point_press (&touch_state[2], A, 1);
|
|
|
|
g_assert_cmpstr (str->str, ==,
|
|
"a1 state denied (3), "
|
|
"c1 began, "
|
|
"c1 state claimed (3)");
|
|
|
|
/* One touch up, gesture is finished again */
|
|
g_string_erase (str, 0, str->len);
|
|
point_release (&touch_state[2], 1);
|
|
|
|
g_assert_cmpstr (str->str, ==,
|
|
"c1 ended");
|
|
|
|
/* Another touch up, gesture remains inactive */
|
|
g_string_erase (str, 0, str->len);
|
|
point_release (&touch_state[1], 1);
|
|
|
|
g_assert_cmpstr (str->str, ==, "");
|
|
|
|
g_string_free (str, TRUE);
|
|
|
|
gtk_widget_destroy (A);
|
|
}
|
|
|
|
int
|
|
main (int argc, char *argv[])
|
|
{
|
|
gtk_test_init (&argc, &argv);
|
|
|
|
g_test_add_func ("/gestures/propagation/phases", test_phases);
|
|
g_test_add_func ("/gestures/propagation/mixed", test_mixed);
|
|
g_test_add_func ("/gestures/propagation/early-exit", test_early_exit);
|
|
g_test_add_func ("/gestures/claim/capture", test_claim_capture);
|
|
g_test_add_func ("/gestures/claim/target", test_claim_target);
|
|
g_test_add_func ("/gestures/claim/bubble", test_claim_bubble);
|
|
g_test_add_func ("/gestures/claim/early-capture", test_early_claim_capture);
|
|
g_test_add_func ("/gestures/claim/late-capture", test_late_claim_capture);
|
|
g_test_add_func ("/gestures/group", test_group);
|
|
g_test_add_func ("/gestures/grabs/gestures-outside-grab", test_gestures_outside_grab);
|
|
g_test_add_func ("/gestures/grabs/gestures-inside-grab", test_gestures_inside_grab);
|
|
g_test_add_func ("/gestures/multitouch/gesture-single", test_multitouch_on_single);
|
|
g_test_add_func ("/gestures/multitouch/multitouch-activation", test_multitouch_activation);
|
|
g_test_add_func ("/gestures/multitouch/interaction", test_multitouch_interaction);
|
|
|
|
return g_test_run ();
|
|
}
|