compositor: Ignore offscreen windows when unredirecting

When determining whether we should unredirect a window or not, ignore
offscreen windows, and just check the top most visible window.

Previously this was not an issue, but since 'stack-tracker: Keep
override redirect windows on top' we started sorting the UI frames
window, which is an offscreen override redirect window, on top, causing
the unredirect checking code to always check whether to unredirect the
UI frames window. This effectively disabled the compositor bypass
functionality.

https://bugzilla.gnome.org/show_bug.cgi?id=788493
This commit is contained in:
Jonas Ådahl 2017-10-13 16:29:20 +08:00
parent 49ee46d924
commit d71b0d3d27
2 changed files with 43 additions and 6 deletions

View File

@ -31,6 +31,8 @@ struct _MetaCompositor
CoglContext *context; CoglContext *context;
MetaWindowActor *top_window_actor;
/* Used for unredirecting fullscreen windows */ /* Used for unredirecting fullscreen windows */
guint disable_unredirect_count; guint disable_unredirect_count;
MetaWindow *unredirected_window; MetaWindow *unredirected_window;

View File

@ -923,6 +923,32 @@ sync_actor_stacking (MetaCompositor *compositor)
g_list_free (backgrounds); g_list_free (backgrounds);
} }
/*
* Find the top most window that is visible on the screen. The intention of
* this is to avoid offscreen windows that isn't actually part of the visible
* desktop (such as the UI frames override redirect window).
*/
static MetaWindowActor *
get_top_visible_window_actor (MetaCompositor *compositor)
{
GList *l;
for (l = g_list_last (compositor->windows); l; l = l->prev)
{
MetaWindowActor *window_actor = l->data;
MetaWindow *window = meta_window_actor_get_meta_window (window_actor);
MetaRectangle buffer_rect;
meta_window_get_buffer_rect (window, &buffer_rect);
if (meta_rectangle_overlap (&compositor->display->screen->rect,
&buffer_rect))
return window_actor;
}
return NULL;
}
void void
meta_compositor_sync_stack (MetaCompositor *compositor, meta_compositor_sync_stack (MetaCompositor *compositor,
GList *stack) GList *stack)
@ -1009,6 +1035,8 @@ meta_compositor_sync_stack (MetaCompositor *compositor,
} }
sync_actor_stacking (compositor); sync_actor_stacking (compositor);
compositor->top_window_actor = get_top_visible_window_actor (compositor);
} }
void void
@ -1065,19 +1093,26 @@ static gboolean
meta_pre_paint_func (gpointer data) meta_pre_paint_func (gpointer data)
{ {
GList *l; GList *l;
MetaWindowActor *top_window; MetaWindowActor *top_window_actor;
MetaCompositor *compositor = data; MetaCompositor *compositor = data;
if (compositor->windows == NULL) if (compositor->windows == NULL)
return TRUE; return TRUE;
top_window = g_list_last (compositor->windows)->data; top_window_actor = compositor->top_window_actor;
if (top_window_actor &&
if (meta_window_actor_should_unredirect (top_window) && meta_window_actor_should_unredirect (top_window_actor) &&
compositor->disable_unredirect_count == 0) compositor->disable_unredirect_count == 0)
set_unredirected_window (compositor, meta_window_actor_get_meta_window (top_window)); {
MetaWindow *top_window;
top_window = meta_window_actor_get_meta_window (top_window_actor);
set_unredirected_window (compositor, top_window);
}
else else
{
set_unredirected_window (compositor, NULL); set_unredirected_window (compositor, NULL);
}
for (l = compositor->windows; l; l = l->next) for (l = compositor->windows; l; l = l->next)
meta_window_actor_pre_paint (l->data); meta_window_actor_pre_paint (l->data);