gtkmain: Add API to get the target of a GdkEvent

This can be used on grabbing situations to determine the target of
the event. This mainly matters for GtkMenu.
This commit is contained in:
Carlos Garnacho 2017-05-04 13:49:34 +02:00
parent da56fae5df
commit 33c5f3c193
2 changed files with 54 additions and 0 deletions

View File

@ -1329,6 +1329,7 @@ synth_crossing_for_motion (GtkWidget *widget,
gdouble x, y; gdouble x, y;
event = gdk_event_new (enter ? GDK_ENTER_NOTIFY : GDK_LEAVE_NOTIFY); event = gdk_event_new (enter ? GDK_ENTER_NOTIFY : GDK_LEAVE_NOTIFY);
gdk_event_set_user_data (event, G_OBJECT (widget));
gdk_event_set_device (event, gdk_event_get_device (source)); gdk_event_set_device (event, gdk_event_get_device (source));
gdk_event_set_source_device (event, gdk_event_get_source_device (source)); gdk_event_set_source_device (event, gdk_event_get_source_device (source));
@ -1678,6 +1679,8 @@ gtk_main_do_event (GdkEvent *event)
if (check_event_in_child_popover (event_widget, grab_widget)) if (check_event_in_child_popover (event_widget, grab_widget))
grab_widget = event_widget; grab_widget = event_widget;
gdk_event_set_user_data (event, G_OBJECT (event_widget));
/* If the widget receiving events is actually blocked by another /* If the widget receiving events is actually blocked by another
* device GTK+ grab * device GTK+ grab
*/ */
@ -2361,6 +2364,50 @@ gtk_get_event_widget (GdkEvent *event)
return widget; return widget;
} }
/**
* gtk_get_event_target:
* @event: a #GdkEvent
*
* If @event is %NULL or the event was not associated with any widget,
* returns %NULL, otherwise returns the widget that is the deepmost
* receiver of the event.
*
* Returns: (transfer none) (nullable): the target widget, or %NULL
*
* Since: 3.90
*/
GtkWidget *
gtk_get_event_target (GdkEvent *event)
{
return GTK_WIDGET (gdk_event_get_user_data (event));
}
/**
* gtk_get_event_target_with_type:
* @event: a #GdkEvent
*
* If @event is %NULL or the event was not associated with any widget,
* returns %NULL, otherwise returns first widget found from the event
* target to the toplevel that matches @type.
*
* Returns: (transfer none) (nullable): the widget in the target stack
* with the given type, or %NULL
*
* Since: 3.90
*/
GtkWidget *
gtk_get_event_target_with_type (GdkEvent *event,
GType type)
{
GtkWidget *target;
target = gtk_get_event_target (event);
while (target && !g_type_is_a (G_OBJECT_TYPE (target), type))
target = gtk_widget_get_parent (target);
return target;
}
static gboolean static gboolean
propagate_event_up (GtkWidget *widget, propagate_event_up (GtkWidget *widget,
GdkEvent *event, GdkEvent *event,

View File

@ -151,6 +151,13 @@ GdkDevice *gtk_get_current_event_device (void);
GDK_AVAILABLE_IN_ALL GDK_AVAILABLE_IN_ALL
GtkWidget *gtk_get_event_widget (GdkEvent *event); GtkWidget *gtk_get_event_widget (GdkEvent *event);
GDK_AVAILABLE_IN_3_90
GtkWidget *gtk_get_event_target (GdkEvent *event);
GDK_AVAILABLE_IN_3_90
GtkWidget *gtk_get_event_target_with_type (GdkEvent *event,
GType type);
GDK_AVAILABLE_IN_ALL GDK_AVAILABLE_IN_ALL
void gtk_propagate_event (GtkWidget *widget, void gtk_propagate_event (GtkWidget *widget,
GdkEvent *event); GdkEvent *event);