gdk/surface: Handle clicks outside client surfaces

The owner_events=TRUE grab makes GDK on X11 see events happening
outside every client window as received on the grab window.
Additionally check that the pointer is inside the grab window
(i.e. it received GDK_CROSSING_NORMAL crossing events for the
core pointer) in order to handle clicks happening outside client
windows.

These new paths are expected to be a no-op on Wayland, and to
also work for touchscreen input on X11, due to emulated pointer
events.
This commit is contained in:
Carlos Garnacho 2022-12-23 16:04:02 +01:00
parent f6bd63c9ad
commit e7f798dcf5
2 changed files with 18 additions and 7 deletions

View File

@ -2789,9 +2789,10 @@ check_autohide (GdkEvent *event)
{ {
GdkDisplay *display; GdkDisplay *display;
GdkDevice *device; GdkDevice *device;
GdkSurface *grab_surface; GdkSurface *grab_surface, *event_surface;
GdkEventType evtype = gdk_event_get_event_type (event);
switch ((guint) gdk_event_get_event_type (event)) switch ((guint) evtype)
{ {
case GDK_BUTTON_PRESS: case GDK_BUTTON_PRESS:
#if 0 #if 0
@ -2810,13 +2811,15 @@ check_autohide (GdkEvent *event)
device = gdk_event_get_device (event); device = gdk_event_get_device (event);
if (gdk_device_grab_info (display, device, &grab_surface, NULL)) if (gdk_device_grab_info (display, device, &grab_surface, NULL))
{ {
GdkSurface *event_surface;
event_surface = gdk_event_get_surface (event); event_surface = gdk_event_get_surface (event);
if (event_surface->autohide &&
!event_surface->has_pointer)
event_surface = NULL;
if (grab_surface != event_surface && if (grab_surface->autohide &&
grab_surface != event_surface->parent && (!event_surface ||
grab_surface->autohide) (grab_surface != event_surface &&
grab_surface != event_surface->parent)))
{ {
GdkSurface *surface = grab_surface; GdkSurface *surface = grab_surface;
@ -2831,6 +2834,13 @@ check_autohide (GdkEvent *event)
} }
} }
break; break;
case GDK_ENTER_NOTIFY:
case GDK_LEAVE_NOTIFY:
event_surface = gdk_event_get_surface (event);
if (event_surface->autohide &&
gdk_crossing_event_get_mode (event) == GDK_CROSSING_NORMAL)
event_surface->has_pointer = evtype == GDK_ENTER_NOTIFY;
break;
default:; default:;
} }

View File

@ -77,6 +77,7 @@ struct _GdkSurface
guint autohide : 1; guint autohide : 1;
guint shortcuts_inhibited : 1; guint shortcuts_inhibited : 1;
guint request_motion : 1; guint request_motion : 1;
guint has_pointer : 1;
guint request_motion_id; guint request_motion_id;