forked from AuroraMiddleware/gtk
629 lines
15 KiB
C
629 lines
15 KiB
C
/*
|
|
* crossingevents.c: A test for crossing events
|
|
*
|
|
* Copyright (C) 2008 Cody Russell
|
|
*
|
|
* 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, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <gtk/gtk.h>
|
|
#include <string.h>
|
|
|
|
typedef struct {
|
|
GtkWidget *window;
|
|
GtkWidget *eventbox;
|
|
GtkWidget *frame;
|
|
GtkWidget *button;
|
|
GtkWidget *check;
|
|
gboolean events_connected;
|
|
GQueue *queue;
|
|
} CrossingTest;
|
|
|
|
typedef struct {
|
|
gboolean entered;
|
|
gchar *name;
|
|
gboolean synthesized;
|
|
GdkCrossingMode mode;
|
|
GdkNotifyType detail;
|
|
} CrossingEventData;
|
|
|
|
#define SLEEP_DURATION 100
|
|
|
|
void start_events (CrossingTest *test);
|
|
void stop_events (CrossingTest *test);
|
|
|
|
static gboolean
|
|
sleep_timeout_cb (gpointer data)
|
|
{
|
|
gtk_main_quit ();
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
sleep_in_main_loop (double fraction)
|
|
{
|
|
/* process all pending idles and events */
|
|
while (g_main_context_pending (NULL))
|
|
g_main_context_iteration (NULL, FALSE);
|
|
/* sleeping probably isn't strictly necessary here */
|
|
gdk_threads_add_timeout_full (G_MAXINT, fraction * SLEEP_DURATION, sleep_timeout_cb, NULL, NULL);
|
|
gtk_main ();
|
|
/* process any pending idles or events that arrived during sleep */
|
|
while (g_main_context_pending (NULL))
|
|
g_main_context_iteration (NULL, FALSE);
|
|
}
|
|
|
|
void
|
|
set_cursor (GtkWidget *widget)
|
|
{
|
|
int x, y, w, h;
|
|
|
|
gdk_window_get_origin (widget->window, &x, &y);
|
|
|
|
x += widget->allocation.x;
|
|
y += widget->allocation.y;
|
|
w = widget->allocation.width;
|
|
h = widget->allocation.height;
|
|
|
|
gdk_display_warp_pointer (gtk_widget_get_display (widget),
|
|
gtk_widget_get_screen (widget),
|
|
x + w / 2,
|
|
y + h / 2);
|
|
|
|
sleep_in_main_loop (0.5);
|
|
}
|
|
|
|
static gboolean
|
|
on_enter (GtkWidget *widget, GdkEventCrossing *event, gpointer user_data)
|
|
{
|
|
CrossingTest *test = (CrossingTest*)user_data;
|
|
|
|
CrossingEventData *evt = g_slice_new0 (CrossingEventData);
|
|
evt->entered = TRUE;
|
|
evt->name = g_strdup (gtk_widget_get_name (widget));
|
|
evt->synthesized = event->send_event;
|
|
evt->mode = event->mode;
|
|
evt->detail = event->detail;
|
|
|
|
if (!test->queue)
|
|
test->queue = g_queue_new ();
|
|
|
|
g_queue_push_tail (test->queue, evt);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
on_leave (GtkWidget *widget, GdkEventCrossing *event, gpointer user_data)
|
|
{
|
|
CrossingTest *test = (CrossingTest*)user_data;
|
|
|
|
CrossingEventData *evt = g_slice_new0 (CrossingEventData);
|
|
evt->entered = FALSE;
|
|
evt->name = g_strdup (gtk_widget_get_name (widget));
|
|
evt->synthesized = event->send_event;
|
|
evt->mode = event->mode;
|
|
evt->detail = event->detail;
|
|
|
|
if (!test->queue)
|
|
test->queue = g_queue_new ();
|
|
|
|
g_queue_push_tail (test->queue, evt);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
on_check_toggled (GtkWidget *toggle, GtkWidget *button)
|
|
{
|
|
gtk_widget_set_sensitive (button, gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (toggle)));
|
|
}
|
|
|
|
static void
|
|
sensitivity_setup (CrossingTest *test,
|
|
gconstpointer user_data)
|
|
{
|
|
GtkWidget *frame;
|
|
|
|
test->window = gtk_window_new (GTK_WINDOW_TOPLEVEL);
|
|
gtk_widget_set_name (test->window, "W");
|
|
frame = gtk_frame_new ("Crossing Events");
|
|
test->eventbox = gtk_event_box_new ();
|
|
gtk_widget_set_name (test->eventbox, "E");
|
|
|
|
GtkWidget *vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, FALSE, 10);
|
|
gtk_container_add (GTK_CONTAINER (test->window), frame);
|
|
gtk_container_add (GTK_CONTAINER (frame), test->eventbox);
|
|
gtk_container_add (GTK_CONTAINER (test->eventbox), vbox);
|
|
|
|
test->button = gtk_button_new_with_label ("Click me!");
|
|
gtk_widget_set_name (test->button, "B");
|
|
gtk_box_pack_start (GTK_BOX (vbox), test->button, FALSE, TRUE, 0);
|
|
|
|
test->check = gtk_check_button_new_with_label ("Sensitive?");
|
|
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (test->check), TRUE);
|
|
g_signal_connect (G_OBJECT (test->check),
|
|
"toggled", G_CALLBACK (on_check_toggled), test->button);
|
|
gtk_widget_set_name (test->check, "C");
|
|
gtk_box_pack_start (GTK_BOX (vbox), test->check, FALSE, TRUE, 0);
|
|
|
|
gtk_widget_show_all (test->window);
|
|
|
|
gtk_window_move (GTK_WINDOW (test->window), 0, 0);
|
|
|
|
sleep_in_main_loop (0.5);
|
|
}
|
|
|
|
static void
|
|
sensitivity_teardown (CrossingTest *test,
|
|
gconstpointer user_data)
|
|
{
|
|
stop_events (test);
|
|
gtk_widget_destroy (test->window);
|
|
|
|
if (test->queue != NULL)
|
|
{
|
|
g_queue_clear (test->queue);
|
|
test->queue = NULL;
|
|
}
|
|
}
|
|
|
|
void
|
|
start_events (CrossingTest *test)
|
|
{
|
|
if (!test->events_connected)
|
|
{
|
|
g_object_connect (G_OBJECT (test->window),
|
|
"signal::destroy", gtk_main_quit, NULL,
|
|
"signal::enter-notify-event", on_enter, test,
|
|
"signal::leave-notify-event", on_leave, test,
|
|
NULL);
|
|
g_object_connect (G_OBJECT (test->eventbox),
|
|
"signal::enter-notify-event", on_enter, test,
|
|
"signal::leave-notify-event", on_leave, test,
|
|
NULL);
|
|
g_object_connect (G_OBJECT (test->button),
|
|
"signal::enter-notify-event", on_enter, test,
|
|
"signal::leave-notify-event", on_leave, test,
|
|
NULL);
|
|
g_object_connect (G_OBJECT (test->check),
|
|
"signal::enter-notify-event", on_enter, test,
|
|
"signal::leave-notify-event", on_leave, test,
|
|
NULL);
|
|
test->events_connected = TRUE;
|
|
}
|
|
|
|
sleep_in_main_loop (0.5);
|
|
}
|
|
|
|
void
|
|
stop_events (CrossingTest *test)
|
|
{
|
|
if (test->events_connected)
|
|
{
|
|
g_object_disconnect (G_OBJECT (test->window),
|
|
"any_signal", gtk_main_quit, NULL,
|
|
"any_signal", on_enter, test,
|
|
"any_signal", on_leave, test,
|
|
NULL);
|
|
g_object_disconnect (G_OBJECT (test->eventbox),
|
|
"any_signal", on_enter, test,
|
|
"any_signal", on_leave, test,
|
|
NULL);
|
|
g_object_disconnect (G_OBJECT (test->button),
|
|
"any_signal", on_enter, test,
|
|
"any_signal", on_leave, test,
|
|
NULL);
|
|
g_object_disconnect (G_OBJECT (test->check),
|
|
"any_signal", G_CALLBACK (on_check_toggled), test->button,
|
|
"any_signal", on_enter, test,
|
|
"any_signal", on_leave, test,
|
|
NULL);
|
|
test->events_connected = FALSE;
|
|
}
|
|
}
|
|
|
|
void
|
|
move_cursor_away (CrossingTest *test)
|
|
{
|
|
gdk_display_warp_pointer (gtk_widget_get_display (test->window),
|
|
gtk_widget_get_screen (test->window),
|
|
1000, -1000);
|
|
|
|
sleep_in_main_loop (0.5);
|
|
}
|
|
|
|
void
|
|
check_event (CrossingTest *test,
|
|
const gchar *name,
|
|
gboolean entered,
|
|
gboolean synthesized,
|
|
GdkCrossingMode mode,
|
|
GdkNotifyType detail)
|
|
{
|
|
CrossingEventData *evt;
|
|
|
|
g_assert (test->queue != NULL);
|
|
|
|
evt = g_queue_pop_head (test->queue);
|
|
|
|
g_assert (evt->entered == entered);
|
|
g_assert (strcmp (evt->name, name) == 0);
|
|
g_assert (evt->synthesized == synthesized);
|
|
g_assert (evt->mode == mode);
|
|
|
|
if (evt->detail != detail)
|
|
g_print ("%s %s event, detail %d, expected detail %d\n",
|
|
synthesized ? "synthesized" : "native",
|
|
entered ? "enter" : "leave",
|
|
evt->detail, detail);
|
|
|
|
g_assert (evt->detail == detail);
|
|
}
|
|
|
|
/* Verify crossing events when moving into and out of a sensitive widget */
|
|
static void
|
|
cursor_on_sensitive (CrossingTest *test,
|
|
gconstpointer user_data)
|
|
{
|
|
move_cursor_away (test);
|
|
|
|
start_events (test);
|
|
|
|
set_cursor (test->button);
|
|
|
|
check_event (test,
|
|
"W",
|
|
TRUE,
|
|
FALSE, /* native */
|
|
GDK_CROSSING_NORMAL,
|
|
GDK_NOTIFY_NONLINEAR_VIRTUAL);
|
|
|
|
check_event (test,
|
|
"E",
|
|
TRUE,
|
|
FALSE, /* native */
|
|
GDK_CROSSING_NORMAL,
|
|
GDK_NOTIFY_NONLINEAR_VIRTUAL);
|
|
|
|
check_event (test,
|
|
"B",
|
|
TRUE,
|
|
FALSE, /* native */
|
|
GDK_CROSSING_NORMAL,
|
|
GDK_NOTIFY_NONLINEAR);
|
|
|
|
g_assert (g_queue_is_empty (test->queue));
|
|
|
|
move_cursor_away (test);
|
|
|
|
check_event (test,
|
|
"B",
|
|
FALSE,
|
|
FALSE, /* native */
|
|
GDK_CROSSING_NORMAL,
|
|
GDK_NOTIFY_NONLINEAR);
|
|
|
|
check_event (test,
|
|
"E",
|
|
FALSE,
|
|
FALSE, /* native */
|
|
GDK_CROSSING_NORMAL,
|
|
GDK_NOTIFY_NONLINEAR_VIRTUAL);
|
|
|
|
check_event (test,
|
|
"W",
|
|
FALSE,
|
|
FALSE, /* native */
|
|
GDK_CROSSING_NORMAL,
|
|
GDK_NOTIFY_NONLINEAR_VIRTUAL);
|
|
|
|
g_assert (g_queue_is_empty (test->queue));
|
|
|
|
stop_events (test);
|
|
}
|
|
|
|
static void
|
|
change_sensitive_to_insensitive (CrossingTest *test,
|
|
gconstpointer user_data)
|
|
{
|
|
move_cursor_away (test);
|
|
set_cursor (test->button);
|
|
|
|
start_events (test);
|
|
|
|
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (test->check), FALSE);
|
|
|
|
check_event (test,
|
|
"B",
|
|
FALSE,
|
|
TRUE, /* synthesized */
|
|
GDK_CROSSING_STATE_CHANGED,
|
|
GDK_NOTIFY_ANCESTOR);
|
|
|
|
check_event (test,
|
|
"E",
|
|
FALSE,
|
|
TRUE, /* synthesized */
|
|
GDK_CROSSING_STATE_CHANGED,
|
|
GDK_NOTIFY_VIRTUAL);
|
|
|
|
check_event (test,
|
|
"W",
|
|
FALSE,
|
|
TRUE, /* synthesized */
|
|
GDK_CROSSING_STATE_CHANGED,
|
|
GDK_NOTIFY_VIRTUAL);
|
|
|
|
g_assert (g_queue_is_empty (test->queue));
|
|
|
|
stop_events (test);
|
|
}
|
|
|
|
static void
|
|
change_insensitive_to_sensitive (CrossingTest *test,
|
|
gconstpointer user_data)
|
|
{
|
|
move_cursor_away (test);
|
|
set_cursor (test->button);
|
|
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (test->check), FALSE);
|
|
|
|
start_events (test);
|
|
|
|
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (test->check), TRUE);
|
|
|
|
check_event (test,
|
|
"W",
|
|
TRUE,
|
|
TRUE, /* synthesized */
|
|
GDK_CROSSING_STATE_CHANGED,
|
|
GDK_NOTIFY_VIRTUAL);
|
|
|
|
check_event (test,
|
|
"E",
|
|
TRUE,
|
|
TRUE, /* synthesized */
|
|
GDK_CROSSING_STATE_CHANGED,
|
|
GDK_NOTIFY_VIRTUAL);
|
|
|
|
check_event (test,
|
|
"B",
|
|
TRUE,
|
|
TRUE, /* synthesized */
|
|
GDK_CROSSING_STATE_CHANGED,
|
|
GDK_NOTIFY_ANCESTOR);
|
|
|
|
g_assert (g_queue_is_empty (test->queue));
|
|
|
|
stop_events (test);
|
|
}
|
|
|
|
static void
|
|
cursor_from_insensitive_to_sensitive (CrossingTest *test,
|
|
gconstpointer user_data)
|
|
{
|
|
set_cursor (test->button);
|
|
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (test->check), FALSE);
|
|
|
|
start_events (test);
|
|
|
|
set_cursor (test->check);
|
|
|
|
check_event (test,
|
|
"C",
|
|
TRUE,
|
|
FALSE, /* native */
|
|
GDK_CROSSING_NORMAL,
|
|
GDK_NOTIFY_NONLINEAR);
|
|
|
|
g_assert (g_queue_is_empty (test->queue));
|
|
|
|
stop_events (test);
|
|
}
|
|
|
|
static void
|
|
cursor_from_sensitive_to_insensitive (CrossingTest *test,
|
|
gconstpointer user_data)
|
|
{
|
|
set_cursor (test->check);
|
|
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (test->check), FALSE);
|
|
|
|
start_events (test);
|
|
|
|
set_cursor (test->button);
|
|
|
|
check_event (test,
|
|
"C",
|
|
FALSE,
|
|
FALSE, /* native */
|
|
GDK_CROSSING_NORMAL,
|
|
GDK_NOTIFY_NONLINEAR);
|
|
|
|
g_assert (g_queue_is_empty (test->queue));
|
|
|
|
stop_events (test);
|
|
}
|
|
|
|
static void
|
|
add_gtk_grab (CrossingTest *test,
|
|
gconstpointer user_data)
|
|
{
|
|
set_cursor (test->button);
|
|
|
|
start_events (test);
|
|
|
|
gtk_grab_add (test->check);
|
|
|
|
check_event (test,
|
|
"B",
|
|
FALSE,
|
|
TRUE, /* synthesized */
|
|
GDK_CROSSING_GTK_GRAB,
|
|
GDK_NOTIFY_ANCESTOR);
|
|
|
|
check_event (test,
|
|
"E",
|
|
FALSE,
|
|
TRUE, /* synthesized */
|
|
GDK_CROSSING_GTK_GRAB,
|
|
GDK_NOTIFY_ANCESTOR);
|
|
|
|
check_event (test,
|
|
"W",
|
|
FALSE,
|
|
TRUE, /* synthesized */
|
|
GDK_CROSSING_GTK_GRAB,
|
|
GDK_NOTIFY_ANCESTOR);
|
|
|
|
g_assert (g_queue_is_empty (test->queue));
|
|
|
|
stop_events (test);
|
|
}
|
|
|
|
static void
|
|
remove_gtk_grab (CrossingTest *test,
|
|
gconstpointer user_data)
|
|
{
|
|
set_cursor (test->button);
|
|
|
|
gtk_grab_add (test->check);
|
|
|
|
start_events (test);
|
|
|
|
gtk_grab_remove (test->check);
|
|
|
|
check_event (test,
|
|
"B",
|
|
TRUE,
|
|
TRUE, /* synthesized */
|
|
GDK_CROSSING_GTK_UNGRAB,
|
|
GDK_NOTIFY_ANCESTOR);
|
|
|
|
check_event (test,
|
|
"E",
|
|
TRUE,
|
|
TRUE, /* synthesized */
|
|
GDK_CROSSING_GTK_UNGRAB,
|
|
GDK_NOTIFY_ANCESTOR);
|
|
|
|
check_event (test,
|
|
"W",
|
|
TRUE,
|
|
TRUE, /* synthesized */
|
|
GDK_CROSSING_GTK_UNGRAB,
|
|
GDK_NOTIFY_ANCESTOR);
|
|
|
|
g_assert (g_queue_is_empty (test->queue));
|
|
|
|
stop_events (test);
|
|
}
|
|
|
|
static void
|
|
cursor_from_shadowed_to_unshadowed (CrossingTest *test,
|
|
gconstpointer user_data)
|
|
{
|
|
set_cursor (test->button);
|
|
|
|
gtk_grab_add (test->check);
|
|
|
|
start_events (test);
|
|
|
|
set_cursor (test->check);
|
|
|
|
check_event (test,
|
|
"C",
|
|
FALSE,
|
|
FALSE, /* native */
|
|
GDK_CROSSING_NORMAL,
|
|
GDK_NOTIFY_NONLINEAR);
|
|
|
|
check_event (test,
|
|
"C",
|
|
TRUE,
|
|
FALSE, /* native */
|
|
GDK_CROSSING_NORMAL,
|
|
GDK_NOTIFY_NONLINEAR);
|
|
|
|
g_assert (g_queue_is_empty (test->queue));
|
|
|
|
stop_events (test);
|
|
}
|
|
|
|
static void
|
|
cursor_from_unshadowed_to_shadowed (CrossingTest *test,
|
|
gconstpointer user_data)
|
|
{
|
|
set_cursor (test->check);
|
|
|
|
gtk_grab_add (test->check);
|
|
|
|
start_events (test);
|
|
|
|
set_cursor (test->button);
|
|
|
|
check_event (test,
|
|
"C",
|
|
FALSE,
|
|
FALSE, /* native */
|
|
GDK_CROSSING_NORMAL,
|
|
GDK_NOTIFY_NONLINEAR);
|
|
|
|
check_event (test,
|
|
"C",
|
|
TRUE,
|
|
FALSE, /* native */
|
|
GDK_CROSSING_NORMAL,
|
|
GDK_NOTIFY_NONLINEAR);
|
|
|
|
g_assert (g_queue_is_empty (test->queue));
|
|
|
|
stop_events (test);
|
|
}
|
|
|
|
int
|
|
main (int argc,
|
|
char **argv)
|
|
{
|
|
gtk_test_init (&argc, &argv, NULL);
|
|
|
|
g_test_add ("/crossings/cursor-on-sensitive", CrossingTest, NULL,
|
|
sensitivity_setup, cursor_on_sensitive, sensitivity_teardown);
|
|
|
|
g_test_add ("/crossings/change-sensitive-to-insensitive", CrossingTest, NULL,
|
|
sensitivity_setup, change_sensitive_to_insensitive, sensitivity_teardown);
|
|
|
|
g_test_add ("/crossings/cursor-from-insensitive-to-sensitive", CrossingTest, NULL,
|
|
sensitivity_setup, cursor_from_insensitive_to_sensitive, sensitivity_teardown);
|
|
|
|
g_test_add ("/crossings/cursor-from-sensitive-to-insensitive", CrossingTest, NULL,
|
|
sensitivity_setup, cursor_from_sensitive_to_insensitive, sensitivity_teardown);
|
|
|
|
g_test_add ("/crossings/change-insensitive-to-sensitive", CrossingTest, NULL,
|
|
sensitivity_setup, change_insensitive_to_sensitive, sensitivity_teardown);
|
|
|
|
g_test_add ("/crossings/add-gtk-grab", CrossingTest, NULL,
|
|
sensitivity_setup, add_gtk_grab, sensitivity_teardown);
|
|
|
|
g_test_add ("/crossings/remove-gtk-grab", CrossingTest, NULL,
|
|
sensitivity_setup, remove_gtk_grab, sensitivity_teardown);
|
|
|
|
g_test_add ("/crossings/cursor-from-shadowed-to-unshadowed", CrossingTest, NULL,
|
|
sensitivity_setup, cursor_from_shadowed_to_unshadowed, sensitivity_teardown);
|
|
|
|
g_test_add ("/crossings/cursor-from-unshadowed-to-shadowed", CrossingTest, NULL,
|
|
sensitivity_setup, cursor_from_unshadowed_to_shadowed, sensitivity_teardown);
|
|
|
|
return g_test_run ();
|
|
}
|