tests: Add tests for crossing events generated during ClutterGrab
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/2068>
This commit is contained in:
parent
7bfc472ad3
commit
75f263ddb8
540
src/tests/clutter/conform/grab.c
Normal file
540
src/tests/clutter/conform/grab.c
Normal file
@ -0,0 +1,540 @@
|
||||
#define CLUTTER_DISABLE_DEPRECATION_WARNINGS
|
||||
#include <clutter/clutter.h>
|
||||
|
||||
#include "tests/clutter-test-utils.h"
|
||||
|
||||
typedef struct
|
||||
{
|
||||
const char *name;
|
||||
ClutterEventType type;
|
||||
} EventLog;
|
||||
|
||||
typedef struct
|
||||
{
|
||||
ClutterActor *stage, *a, *b, *c;
|
||||
GArray *events;
|
||||
} TestData;
|
||||
|
||||
static void
|
||||
event_log_compare (EventLog *expected,
|
||||
GArray *obtained)
|
||||
{
|
||||
EventLog *elem;
|
||||
guint i;
|
||||
|
||||
for (i = 0; expected[i].name != NULL; i++)
|
||||
{
|
||||
g_assert_cmpuint (i, <, obtained->len);
|
||||
elem = &g_array_index (obtained, EventLog, i);
|
||||
g_assert_cmpuint (expected[i].type, ==, elem->type);
|
||||
g_assert_cmpstr (expected[i].name, ==, elem->name);
|
||||
}
|
||||
|
||||
if (i != obtained->len)
|
||||
{
|
||||
elem = &g_array_index (obtained, EventLog, i);
|
||||
g_critical ("Unexpected event %d on actor '%s'",
|
||||
elem->type, elem->name);
|
||||
}
|
||||
|
||||
g_assert_cmpuint (i, ==, obtained->len);
|
||||
|
||||
/* Clear the array for future comparisons */
|
||||
g_array_set_size (obtained, 0);
|
||||
}
|
||||
|
||||
static gboolean
|
||||
event_cb (ClutterActor *actor,
|
||||
ClutterEvent *event,
|
||||
gpointer user_data)
|
||||
{
|
||||
GArray *events = user_data;
|
||||
|
||||
if ((event->type == CLUTTER_ENTER ||
|
||||
event->type == CLUTTER_LEAVE) &&
|
||||
(event->any.flags & CLUTTER_EVENT_FLAG_GRAB_NOTIFY) != 0)
|
||||
{
|
||||
EventLog entry = { clutter_actor_get_name (actor), event->type };
|
||||
|
||||
g_debug ("Event '%s' on actor '%s'",
|
||||
entry.type == CLUTTER_ENTER ? "ENTER" : "LEAVE",
|
||||
entry.name);
|
||||
g_array_append_val (events, entry);
|
||||
}
|
||||
|
||||
return CLUTTER_EVENT_PROPAGATE;
|
||||
}
|
||||
|
||||
static void
|
||||
create_actors (ClutterActor *stage,
|
||||
ClutterActor **a,
|
||||
ClutterActor **b,
|
||||
ClutterActor **c)
|
||||
{
|
||||
/* This builds the following tree:
|
||||
*
|
||||
* stage
|
||||
* ╱ ╲
|
||||
* a c
|
||||
* ╱
|
||||
* b
|
||||
*/
|
||||
|
||||
*a = clutter_actor_new ();
|
||||
clutter_actor_set_name (*a, "a");
|
||||
clutter_actor_set_reactive (*a, TRUE);
|
||||
clutter_actor_set_width (*a, clutter_actor_get_width (stage) / 2);
|
||||
clutter_actor_set_height (*a, clutter_actor_get_height (stage));
|
||||
clutter_actor_add_child (stage, *a);
|
||||
|
||||
*b = clutter_actor_new ();
|
||||
clutter_actor_set_name (*b, "b");
|
||||
clutter_actor_set_reactive (*b, TRUE);
|
||||
clutter_actor_set_width (*b, clutter_actor_get_width (stage) / 2);
|
||||
clutter_actor_set_height (*b, clutter_actor_get_height (stage));
|
||||
clutter_actor_add_child (*a, *b);
|
||||
|
||||
*c = clutter_actor_new ();
|
||||
clutter_actor_set_name (*c, "c");
|
||||
clutter_actor_set_reactive (*c, TRUE);
|
||||
clutter_actor_set_x (*c, clutter_actor_get_width (stage) / 2);
|
||||
clutter_actor_set_width (*c, clutter_actor_get_width (stage) / 2);
|
||||
clutter_actor_set_height (*c, clutter_actor_get_height (stage));
|
||||
clutter_actor_add_child (stage, *c);
|
||||
}
|
||||
|
||||
static void
|
||||
has_pointer_cb (ClutterActor *actor)
|
||||
{
|
||||
if (clutter_actor_has_pointer (actor))
|
||||
clutter_test_quit ();
|
||||
}
|
||||
|
||||
static void
|
||||
create_pointer (ClutterActor *actor)
|
||||
{
|
||||
ClutterVirtualInputDevice *pointer;
|
||||
ClutterSeat *seat;
|
||||
guint notify_id;
|
||||
|
||||
seat = clutter_backend_get_default_seat (clutter_get_default_backend ());
|
||||
pointer = clutter_seat_create_virtual_device (seat, CLUTTER_POINTER_DEVICE);
|
||||
|
||||
clutter_virtual_input_device_notify_absolute_motion (pointer,
|
||||
0,
|
||||
clutter_actor_get_x (actor) +
|
||||
clutter_actor_get_width (actor) / 2,
|
||||
clutter_actor_get_y (actor) +
|
||||
clutter_actor_get_height (actor) / 2);
|
||||
|
||||
notify_id = g_signal_connect (actor, "notify::has-pointer",
|
||||
G_CALLBACK (has_pointer_cb), NULL);
|
||||
clutter_test_main ();
|
||||
|
||||
g_signal_handler_disconnect (actor, notify_id);
|
||||
|
||||
g_object_unref (pointer);
|
||||
}
|
||||
|
||||
static void
|
||||
connect_signals (ClutterActor *stage,
|
||||
ClutterActor *a,
|
||||
ClutterActor *b,
|
||||
ClutterActor *c,
|
||||
gpointer user_data)
|
||||
{
|
||||
g_signal_connect (stage, "event", G_CALLBACK (event_cb), user_data);
|
||||
g_signal_connect (a, "event", G_CALLBACK (event_cb), user_data);
|
||||
g_signal_connect (b, "event", G_CALLBACK (event_cb), user_data);
|
||||
g_signal_connect (c, "event", G_CALLBACK (event_cb), user_data);
|
||||
}
|
||||
|
||||
static void
|
||||
disconnect_signals (ClutterActor *stage,
|
||||
ClutterActor *a,
|
||||
ClutterActor *b,
|
||||
ClutterActor *c,
|
||||
gpointer user_data)
|
||||
{
|
||||
g_signal_handlers_disconnect_by_func (stage, event_cb, user_data);
|
||||
g_signal_handlers_disconnect_by_func (a, event_cb, user_data);
|
||||
g_signal_handlers_disconnect_by_func (b, event_cb, user_data);
|
||||
g_signal_handlers_disconnect_by_func (c, event_cb, user_data);
|
||||
}
|
||||
|
||||
static void
|
||||
test_data_init (TestData *data)
|
||||
{
|
||||
ClutterActor *stage;
|
||||
ClutterActor *a, *b, *c;
|
||||
GArray *events;
|
||||
|
||||
stage = clutter_test_get_stage ();
|
||||
clutter_actor_set_name (stage, "stage");
|
||||
create_actors (stage, &a, &b, &c);
|
||||
clutter_actor_show (stage);
|
||||
create_pointer (b);
|
||||
|
||||
events = g_array_new (TRUE, TRUE, sizeof (EventLog));
|
||||
|
||||
connect_signals (stage, a, b, c, events);
|
||||
|
||||
*data = (TestData) {
|
||||
stage, a, b, c, events,
|
||||
};
|
||||
}
|
||||
|
||||
static void
|
||||
test_data_shutdown (TestData *data)
|
||||
{
|
||||
disconnect_signals (data->stage, data->a, data->b, data->c, data->events);
|
||||
clutter_actor_destroy (data->c);
|
||||
clutter_actor_destroy (data->b);
|
||||
clutter_actor_destroy (data->a);
|
||||
g_array_unref (data->events);
|
||||
}
|
||||
|
||||
static void
|
||||
grab_under_pointer (void)
|
||||
{
|
||||
TestData data;
|
||||
ClutterGrab *grab;
|
||||
EventLog grab_log[] = {
|
||||
{ "a", CLUTTER_LEAVE },
|
||||
{ "stage", CLUTTER_LEAVE },
|
||||
{ NULL, 0 },
|
||||
};
|
||||
EventLog ungrab_log[] = {
|
||||
{ "a", CLUTTER_ENTER },
|
||||
{ "stage", CLUTTER_ENTER },
|
||||
{ NULL, 0 },
|
||||
};
|
||||
|
||||
test_data_init (&data);
|
||||
|
||||
/* Grab 'b', pointer is on 'b' */
|
||||
grab = clutter_stage_grab (CLUTTER_STAGE (data.stage), data.b);
|
||||
event_log_compare ((EventLog *) &grab_log, data.events);
|
||||
|
||||
clutter_grab_dismiss (grab);
|
||||
event_log_compare ((EventLog *) &ungrab_log, data.events);
|
||||
|
||||
test_data_shutdown (&data);
|
||||
}
|
||||
|
||||
static void
|
||||
grab_under_pointers_parent (void)
|
||||
{
|
||||
TestData data;
|
||||
ClutterGrab *grab;
|
||||
EventLog grab_log[] = {
|
||||
{ "stage", CLUTTER_LEAVE },
|
||||
{ NULL, 0 },
|
||||
};
|
||||
EventLog ungrab_log[] = {
|
||||
{ "stage", CLUTTER_ENTER },
|
||||
{ NULL, 0 },
|
||||
};
|
||||
|
||||
test_data_init (&data);
|
||||
|
||||
/* Grab 'a', pointer is on its child 'b' */
|
||||
grab = clutter_stage_grab (CLUTTER_STAGE (data.stage), data.a);
|
||||
event_log_compare ((EventLog *) &grab_log, data.events);
|
||||
|
||||
clutter_grab_dismiss (grab);
|
||||
event_log_compare ((EventLog *) &ungrab_log, data.events);
|
||||
|
||||
test_data_shutdown (&data);
|
||||
}
|
||||
|
||||
static void
|
||||
grab_outside_pointer (void)
|
||||
{
|
||||
TestData data;
|
||||
ClutterGrab *grab;
|
||||
EventLog grab_log[] = {
|
||||
{ "b", CLUTTER_LEAVE },
|
||||
{ "a", CLUTTER_LEAVE },
|
||||
{ "stage", CLUTTER_LEAVE },
|
||||
{ NULL, 0 },
|
||||
};
|
||||
EventLog ungrab_log[] = {
|
||||
{ "b", CLUTTER_ENTER },
|
||||
{ "a", CLUTTER_ENTER },
|
||||
{ "stage", CLUTTER_ENTER },
|
||||
{ NULL, 0 },
|
||||
};
|
||||
|
||||
test_data_init (&data);
|
||||
|
||||
/* Grab 'c', pointer is on 'b' */
|
||||
grab = clutter_stage_grab (CLUTTER_STAGE (data.stage), data.c);
|
||||
event_log_compare ((EventLog *) &grab_log, data.events);
|
||||
|
||||
clutter_grab_dismiss (grab);
|
||||
event_log_compare ((EventLog *) &ungrab_log, data.events);
|
||||
|
||||
test_data_shutdown (&data);
|
||||
}
|
||||
|
||||
static void
|
||||
grab_stage (void)
|
||||
{
|
||||
TestData data;
|
||||
ClutterGrab *grab;
|
||||
EventLog grab_log[] = {
|
||||
{ NULL, 0 },
|
||||
};
|
||||
EventLog ungrab_log[] = {
|
||||
{ NULL, 0 },
|
||||
};
|
||||
|
||||
test_data_init (&data);
|
||||
|
||||
/* Grab 'stage', pointer is on 'b' */
|
||||
grab = clutter_stage_grab (CLUTTER_STAGE (data.stage), data.stage);
|
||||
event_log_compare ((EventLog *) &grab_log, data.events);
|
||||
|
||||
clutter_grab_dismiss (grab);
|
||||
event_log_compare ((EventLog *) &ungrab_log, data.events);
|
||||
|
||||
test_data_shutdown (&data);
|
||||
}
|
||||
|
||||
static void
|
||||
grab_stack_1 (void)
|
||||
{
|
||||
TestData data;
|
||||
ClutterGrab *grab1, *grab2;
|
||||
EventLog grab1_log[] = {
|
||||
{ "a", CLUTTER_LEAVE },
|
||||
{ "stage", CLUTTER_LEAVE },
|
||||
{ NULL, 0 },
|
||||
};
|
||||
EventLog grab2_log[] = {
|
||||
{ "b", CLUTTER_LEAVE },
|
||||
{ "a", CLUTTER_LEAVE },
|
||||
{ "stage", CLUTTER_LEAVE },
|
||||
{ NULL, 0 },
|
||||
};
|
||||
EventLog ungrab2_log[] = {
|
||||
{ "b", CLUTTER_ENTER },
|
||||
{ NULL, 0 },
|
||||
};
|
||||
EventLog ungrab1_log[] = {
|
||||
{ "a", CLUTTER_ENTER },
|
||||
{ "stage", CLUTTER_ENTER },
|
||||
{ NULL, 0 },
|
||||
};
|
||||
|
||||
test_data_init (&data);
|
||||
|
||||
/* Grab 'b', pointer is on 'b' */
|
||||
grab1 = clutter_stage_grab (CLUTTER_STAGE (data.stage), data.b);
|
||||
event_log_compare ((EventLog *) &grab1_log, data.events);
|
||||
|
||||
/* Grab 'c', pointer and grab is on 'b' */
|
||||
grab2 = clutter_stage_grab (CLUTTER_STAGE (data.stage), data.c);
|
||||
event_log_compare ((EventLog *) &grab2_log, data.events);
|
||||
|
||||
/* Dismiss orderly */
|
||||
clutter_grab_dismiss (grab2);
|
||||
event_log_compare ((EventLog *) &ungrab2_log, data.events);
|
||||
|
||||
clutter_grab_dismiss (grab1);
|
||||
event_log_compare ((EventLog *) &ungrab1_log, data.events);
|
||||
|
||||
test_data_shutdown (&data);
|
||||
}
|
||||
|
||||
static void
|
||||
grab_stack_2 (void)
|
||||
{
|
||||
TestData data;
|
||||
ClutterGrab *grab1, *grab2;
|
||||
EventLog grab1_log[] = {
|
||||
{ "b", CLUTTER_LEAVE },
|
||||
{ "a", CLUTTER_LEAVE },
|
||||
{ "stage", CLUTTER_LEAVE },
|
||||
{ NULL, 0 },
|
||||
};
|
||||
EventLog grab2_log[] = {
|
||||
{ "b", CLUTTER_ENTER },
|
||||
{ NULL, 0 },
|
||||
};
|
||||
EventLog ungrab2_log[] = {
|
||||
{ "b", CLUTTER_LEAVE },
|
||||
{ "a", CLUTTER_LEAVE },
|
||||
{ "stage", CLUTTER_LEAVE },
|
||||
{ NULL, 0 },
|
||||
};
|
||||
EventLog ungrab1_log[] = {
|
||||
{ "b", CLUTTER_ENTER },
|
||||
{ "a", CLUTTER_ENTER },
|
||||
{ "stage", CLUTTER_ENTER },
|
||||
{ NULL, 0 },
|
||||
};
|
||||
|
||||
test_data_init (&data);
|
||||
|
||||
/* Grab 'c', pointer is on 'b' */
|
||||
grab1 = clutter_stage_grab (CLUTTER_STAGE (data.stage), data.c);
|
||||
event_log_compare ((EventLog *) &grab1_log, data.events);
|
||||
|
||||
/* Grab 'b', pointer is on b, prior grab is on 'c' */
|
||||
grab2 = clutter_stage_grab (CLUTTER_STAGE (data.stage), data.b);
|
||||
event_log_compare ((EventLog *) &grab2_log, data.events);
|
||||
|
||||
/* Dismiss orderly */
|
||||
clutter_grab_dismiss (grab2);
|
||||
event_log_compare ((EventLog *) &ungrab2_log, data.events);
|
||||
|
||||
clutter_grab_dismiss (grab1);
|
||||
event_log_compare ((EventLog *) &ungrab1_log, data.events);
|
||||
|
||||
test_data_shutdown (&data);
|
||||
}
|
||||
|
||||
static void
|
||||
grab_unordered_ungrab_1 (void)
|
||||
{
|
||||
TestData data;
|
||||
ClutterGrab *grab1, *grab2;
|
||||
EventLog grab1_log[] = {
|
||||
{ "a", CLUTTER_LEAVE },
|
||||
{ "stage", CLUTTER_LEAVE },
|
||||
{ NULL, 0 },
|
||||
};
|
||||
EventLog grab2_log[] = {
|
||||
{ "b", CLUTTER_LEAVE },
|
||||
{ "a", CLUTTER_LEAVE },
|
||||
{ "stage", CLUTTER_LEAVE },
|
||||
{ NULL, 0 },
|
||||
};
|
||||
EventLog ungrab1_log[] = {
|
||||
{ NULL, 0 },
|
||||
};
|
||||
EventLog ungrab2_log[] = {
|
||||
{ "b", CLUTTER_ENTER },
|
||||
{ "a", CLUTTER_ENTER },
|
||||
{ "stage", CLUTTER_ENTER },
|
||||
{ NULL, 0 },
|
||||
};
|
||||
|
||||
test_data_init (&data);
|
||||
|
||||
/* Grab 'b', pointer is on 'b' */
|
||||
grab1 = clutter_stage_grab (CLUTTER_STAGE (data.stage), data.b);
|
||||
event_log_compare ((EventLog *) &grab1_log, data.events);
|
||||
|
||||
/* Grab 'c', pointer and grab is on 'b' */
|
||||
grab2 = clutter_stage_grab (CLUTTER_STAGE (data.stage), data.c);
|
||||
event_log_compare ((EventLog *) &grab2_log, data.events);
|
||||
|
||||
/* Dismiss disorderly */
|
||||
clutter_grab_dismiss (grab1);
|
||||
event_log_compare ((EventLog *) &ungrab1_log, data.events);
|
||||
|
||||
clutter_grab_dismiss (grab2);
|
||||
event_log_compare ((EventLog *) &ungrab2_log, data.events);
|
||||
|
||||
test_data_shutdown (&data);
|
||||
}
|
||||
|
||||
static void
|
||||
grab_unordered_ungrab_2 (void)
|
||||
{
|
||||
TestData data;
|
||||
ClutterGrab *grab1, *grab2;
|
||||
EventLog grab1_log[] = {
|
||||
{ "b", CLUTTER_LEAVE },
|
||||
{ "a", CLUTTER_LEAVE },
|
||||
{ "stage", CLUTTER_LEAVE },
|
||||
{ NULL, 0 },
|
||||
};
|
||||
EventLog grab2_log[] = {
|
||||
{ "b", CLUTTER_ENTER },
|
||||
{ NULL, 0 },
|
||||
};
|
||||
EventLog ungrab1_log[] = {
|
||||
{ NULL, 0 },
|
||||
};
|
||||
EventLog ungrab2_log[] = {
|
||||
{ "a", CLUTTER_ENTER },
|
||||
{ "stage", CLUTTER_ENTER },
|
||||
{ NULL, 0 },
|
||||
};
|
||||
|
||||
test_data_init (&data);
|
||||
|
||||
/* Grab 'c', pointer is on 'b' */
|
||||
grab1 = clutter_stage_grab (CLUTTER_STAGE (data.stage), data.c);
|
||||
event_log_compare ((EventLog *) &grab1_log, data.events);
|
||||
|
||||
/* Grab 'b', pointer is on b, prior grab is on 'c' */
|
||||
grab2 = clutter_stage_grab (CLUTTER_STAGE (data.stage), data.b);
|
||||
event_log_compare ((EventLog *) &grab2_log, data.events);
|
||||
|
||||
/* Dismiss disorderly */
|
||||
clutter_grab_dismiss (grab1);
|
||||
event_log_compare ((EventLog *) &ungrab1_log, data.events);
|
||||
|
||||
clutter_grab_dismiss (grab2);
|
||||
event_log_compare ((EventLog *) &ungrab2_log, data.events);
|
||||
|
||||
test_data_shutdown (&data);
|
||||
}
|
||||
|
||||
static void
|
||||
grab_key_focus_in_grab (void)
|
||||
{
|
||||
TestData data;
|
||||
ClutterGrab *grab;
|
||||
|
||||
test_data_init (&data);
|
||||
|
||||
clutter_actor_grab_key_focus (data.b);
|
||||
g_assert_true (clutter_actor_has_key_focus (data.b));
|
||||
|
||||
grab = clutter_stage_grab (CLUTTER_STAGE (data.stage), data.b);
|
||||
g_assert_true (clutter_actor_has_key_focus (data.b));
|
||||
|
||||
clutter_grab_dismiss (grab);
|
||||
g_assert_true (clutter_actor_has_key_focus (data.b));
|
||||
|
||||
test_data_shutdown (&data);
|
||||
}
|
||||
|
||||
static void
|
||||
grab_key_focus_outside_grab (void)
|
||||
{
|
||||
TestData data;
|
||||
ClutterGrab *grab;
|
||||
|
||||
test_data_init (&data);
|
||||
|
||||
clutter_actor_grab_key_focus (data.b);
|
||||
g_assert_true (clutter_actor_has_key_focus (data.b));
|
||||
|
||||
grab = clutter_stage_grab (CLUTTER_STAGE (data.stage), data.c);
|
||||
g_assert_false (clutter_actor_has_key_focus (data.b));
|
||||
|
||||
clutter_grab_dismiss (grab);
|
||||
g_assert_true (clutter_actor_has_key_focus (data.b));
|
||||
|
||||
test_data_shutdown (&data);
|
||||
}
|
||||
|
||||
CLUTTER_TEST_SUITE (
|
||||
CLUTTER_TEST_UNIT ("/grab/grab-under-pointer", grab_under_pointer)
|
||||
CLUTTER_TEST_UNIT ("/grab/grab-under-pointers-parent", grab_under_pointers_parent)
|
||||
CLUTTER_TEST_UNIT ("/grab/grab-outside-pointer", grab_outside_pointer)
|
||||
CLUTTER_TEST_UNIT ("/grab/grab-stage", grab_stage)
|
||||
CLUTTER_TEST_UNIT ("/grab/grab-stack-1", grab_stack_1)
|
||||
CLUTTER_TEST_UNIT ("/grab/grab-stack-2", grab_stack_2)
|
||||
CLUTTER_TEST_UNIT ("/grab/grab-unordered-ungrab-1", grab_unordered_ungrab_1)
|
||||
CLUTTER_TEST_UNIT ("/grab/grab-unordered-ungrab-2", grab_unordered_ungrab_2)
|
||||
CLUTTER_TEST_UNIT ("/grab/key-focus-in-grab", grab_key_focus_in_grab);
|
||||
CLUTTER_TEST_UNIT ("/grab/key-focus-outside-grab", grab_key_focus_outside_grab);
|
||||
)
|
@ -35,6 +35,7 @@ clutter_conform_tests_general_tests = [
|
||||
'color',
|
||||
'frame-clock',
|
||||
'frame-clock-timeline',
|
||||
'grab',
|
||||
'interval',
|
||||
'script-parser',
|
||||
'timeline',
|
||||
|
Loading…
Reference in New Issue
Block a user