window: Don't invalidate cssnode during get_preferred_width()

Getting the shadow width must not call gtk_style_context_set_state()
because that will invalidate the node and cause a style-updated emission
which can cause gtk_widget_queue_resize() calls.

And calling queue_resize() from get_preferred_size() essentially means
the size is permanently invalid because you invalidate it while
querying it.

This causes flickering of windows when going from/to backdrop state. To
avoid this we either need to fix the theme to not have different shadow
sizes in those cases or we need to ensure the window doesn't flicker in
the first place.
This commit is contained in:
Benjamin Otte 2015-11-14 18:28:34 +01:00
parent 38bfec8ea4
commit 55735cee2f

View File

@ -6717,9 +6717,8 @@ get_shadow_width (GtkWindow *window,
GtkBorder d = { 0 }; GtkBorder d = { 0 };
GtkBorder margin; GtkBorder margin;
GtkStyleContext *context; GtkStyleContext *context;
GtkStateFlags state, s; GtkStateFlags s;
GtkCssValue *shadows; GtkCssValue *shadows;
gint i;
*shadow_width = border; *shadow_width = border;
@ -6738,44 +6737,30 @@ get_shadow_width (GtkWindow *window,
if (!_gtk_widget_is_toplevel (GTK_WIDGET (window))) if (!_gtk_widget_is_toplevel (GTK_WIDGET (window)))
return; return;
state = _gtk_widget_get_state_flags (GTK_WIDGET (window));
context = _gtk_widget_get_style_context (GTK_WIDGET (window)); context = _gtk_widget_get_style_context (GTK_WIDGET (window));
gtk_style_context_save_to_node (context, priv->decoration_node); gtk_style_context_save_to_node (context, priv->decoration_node);
s = gtk_style_context_get_state (context);
/* We don't want windows to jump as they go to backdrop, /* Always sum border + padding */
* therefore we use the maximum of the decoration sizes gtk_style_context_get_border (context, s, &border);
* for focused and unfocused. gtk_style_context_get_padding (context, s, &d);
*/ sum_borders (&d, &border);
for (i = 0; i < 2; i++)
/* Calculate the size of the drop shadows ... */
shadows = _gtk_style_context_peek_property (context, GTK_CSS_PROPERTY_BOX_SHADOW);
_gtk_css_shadows_value_get_extents (shadows, &border);
if (priv->type != GTK_WINDOW_POPUP)
{ {
if (i == 0) /* ... and compare it to the margin size, which we use for resize grips */
s = state & ~GTK_STATE_FLAG_BACKDROP; gtk_style_context_get_margin (context, s, &margin);
else max_borders (&border, &margin);
s = state | GTK_STATE_FLAG_BACKDROP;
gtk_style_context_set_state (context, s);
/* Always sum border + padding */
gtk_style_context_get_border (context, s, &border);
gtk_style_context_get_padding (context, s, &d);
sum_borders (&d, &border);
/* Calculate the size of the drop shadows ... */
shadows = _gtk_style_context_peek_property (context, GTK_CSS_PROPERTY_BOX_SHADOW);
_gtk_css_shadows_value_get_extents (shadows, &border);
if (priv->type != GTK_WINDOW_POPUP)
{
/* ... and compare it to the margin size, which we use for resize grips */
gtk_style_context_get_margin (context, s, &margin);
max_borders (&border, &margin);
}
sum_borders (&d, &border);
max_borders (shadow_width, &d);
} }
sum_borders (&d, &border);
*shadow_width = d;
gtk_style_context_restore (context); gtk_style_context_restore (context);
} }