mirror of
https://github.com/brl/mutter.git
synced 2024-11-24 00:50:42 -05:00
3a79b7b066
Since we want these accessed from bindings this must be a boxed type. This has the side effect of making ClutterGrab a refcounted object, since we want to avoid JS from pointing to freed memory and maybe causing crashes if misusing the object after dismiss. Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/2099>
555 lines
14 KiB
C
555 lines
14 KiB
C
#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);
|
||
clutter_grab_unref (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);
|
||
clutter_grab_unref (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);
|
||
clutter_grab_unref (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);
|
||
clutter_grab_unref (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);
|
||
clutter_grab_unref (grab2);
|
||
event_log_compare ((EventLog *) &ungrab2_log, data.events);
|
||
|
||
clutter_grab_dismiss (grab1);
|
||
clutter_grab_unref (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);
|
||
clutter_grab_unref (grab2);
|
||
event_log_compare ((EventLog *) &ungrab2_log, data.events);
|
||
|
||
clutter_grab_dismiss (grab1);
|
||
clutter_grab_unref (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);
|
||
clutter_grab_unref (grab1);
|
||
event_log_compare ((EventLog *) &ungrab1_log, data.events);
|
||
|
||
clutter_grab_dismiss (grab2);
|
||
clutter_grab_unref (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);
|
||
clutter_grab_unref (grab1);
|
||
event_log_compare ((EventLog *) &ungrab1_log, data.events);
|
||
|
||
clutter_grab_dismiss (grab2);
|
||
clutter_grab_unref (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);
|
||
clutter_grab_unref (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);
|
||
clutter_grab_unref (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);
|
||
)
|