forked from AuroraMiddleware/gtk
cec7ff1ba7
On gdk_display_close(), the GtkSettings attached to the display are freed. Yet the gtk CSS code may still be called from the widget unparent, leading to a segfault. Check if the GtkSettings is not NULL and bail out nicely if not. Closes: https://gitlab.gnome.org/GNOME/gtk/-/issues/2780
330 lines
9.8 KiB
C
330 lines
9.8 KiB
C
/* GTK - The GIMP Toolkit
|
|
* Copyright (C) 2014 Benjamin Otte <otte@gnome.org>
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "gtkcsswidgetnodeprivate.h"
|
|
|
|
#include "gtkcontainerprivate.h"
|
|
#include "gtkcssanimatedstyleprivate.h"
|
|
#include "gtkprivate.h"
|
|
#include "gtksettingsprivate.h"
|
|
#include "gtkstylecontextprivate.h"
|
|
#include "gtkwidgetprivate.h"
|
|
/* widgets for special casing go here */
|
|
#include "gtkbox.h"
|
|
|
|
G_DEFINE_TYPE (GtkCssWidgetNode, gtk_css_widget_node, GTK_TYPE_CSS_NODE)
|
|
|
|
static void
|
|
gtk_css_widget_node_finalize (GObject *object)
|
|
{
|
|
GtkCssWidgetNode *node = GTK_CSS_WIDGET_NODE (object);
|
|
|
|
g_object_unref (node->last_updated_style);
|
|
|
|
G_OBJECT_CLASS (gtk_css_widget_node_parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
gtk_css_widget_node_style_changed (GtkCssNode *cssnode,
|
|
GtkCssStyleChange *change)
|
|
{
|
|
GtkCssWidgetNode *node;
|
|
|
|
node = GTK_CSS_WIDGET_NODE (cssnode);
|
|
|
|
if (node->widget)
|
|
gtk_widget_clear_path (node->widget);
|
|
|
|
GTK_CSS_NODE_CLASS (gtk_css_widget_node_parent_class)->style_changed (cssnode, change);
|
|
}
|
|
|
|
static gboolean
|
|
gtk_css_widget_node_queue_callback (GtkWidget *widget,
|
|
GdkFrameClock *frame_clock,
|
|
gpointer user_data)
|
|
{
|
|
GtkCssNode *node = user_data;
|
|
|
|
gtk_css_node_invalidate_frame_clock (node, TRUE);
|
|
_gtk_container_queue_restyle (GTK_CONTAINER (widget));
|
|
|
|
return G_SOURCE_CONTINUE;
|
|
}
|
|
|
|
static GtkCssStyle *
|
|
gtk_css_widget_node_update_style (GtkCssNode *cssnode,
|
|
GtkCssChange change,
|
|
gint64 timestamp,
|
|
GtkCssStyle *style)
|
|
{
|
|
GtkCssWidgetNode *widget_node = GTK_CSS_WIDGET_NODE (cssnode);
|
|
|
|
if (widget_node->widget != NULL)
|
|
{
|
|
GtkStyleContext *context = _gtk_widget_peek_style_context (widget_node->widget);
|
|
if (context)
|
|
gtk_style_context_clear_property_cache (context);
|
|
}
|
|
|
|
return GTK_CSS_NODE_CLASS (gtk_css_widget_node_parent_class)->update_style (cssnode, change, timestamp, style);
|
|
}
|
|
|
|
static void
|
|
gtk_css_widget_node_queue_validate (GtkCssNode *node)
|
|
{
|
|
GtkCssWidgetNode *widget_node = GTK_CSS_WIDGET_NODE (node);
|
|
|
|
G_GNUC_BEGIN_IGNORE_DEPRECATIONS
|
|
if (GTK_IS_RESIZE_CONTAINER (widget_node->widget))
|
|
widget_node->validate_cb_id = gtk_widget_add_tick_callback (widget_node->widget,
|
|
gtk_css_widget_node_queue_callback,
|
|
node,
|
|
NULL);
|
|
G_GNUC_END_IGNORE_DEPRECATIONS
|
|
}
|
|
|
|
static void
|
|
gtk_css_widget_node_dequeue_validate (GtkCssNode *node)
|
|
{
|
|
GtkCssWidgetNode *widget_node = GTK_CSS_WIDGET_NODE (node);
|
|
|
|
G_GNUC_BEGIN_IGNORE_DEPRECATIONS
|
|
if (GTK_IS_RESIZE_CONTAINER (widget_node->widget))
|
|
gtk_widget_remove_tick_callback (widget_node->widget,
|
|
widget_node->validate_cb_id);
|
|
G_GNUC_END_IGNORE_DEPRECATIONS
|
|
}
|
|
|
|
static void
|
|
gtk_css_widget_node_validate (GtkCssNode *node)
|
|
{
|
|
GtkCssWidgetNode *widget_node = GTK_CSS_WIDGET_NODE (node);
|
|
GtkCssStyleChange change;
|
|
GtkCssStyle *style;
|
|
|
|
if (widget_node->widget == NULL)
|
|
return;
|
|
|
|
style = gtk_css_node_get_style (node);
|
|
|
|
gtk_css_style_change_init (&change, widget_node->last_updated_style, style);
|
|
if (gtk_css_style_change_has_change (&change))
|
|
{
|
|
GtkStyleContext *context;
|
|
|
|
context = _gtk_widget_peek_style_context (widget_node->widget);
|
|
if (context)
|
|
gtk_style_context_validate (context, &change);
|
|
else
|
|
_gtk_widget_style_context_invalidated (widget_node->widget);
|
|
g_set_object (&widget_node->last_updated_style, style);
|
|
}
|
|
gtk_css_style_change_finish (&change);
|
|
}
|
|
|
|
typedef GtkWidgetPath * (* GetPathForChildFunc) (GtkContainer *, GtkWidget *);
|
|
|
|
static gboolean
|
|
widget_needs_widget_path (GtkWidget *widget)
|
|
{
|
|
static GetPathForChildFunc funcs[2];
|
|
GtkContainerClass *class;
|
|
GtkWidget *parent;
|
|
GetPathForChildFunc parent_func;
|
|
guint i;
|
|
|
|
if (G_UNLIKELY (funcs[0] == NULL))
|
|
{
|
|
i = 0;
|
|
|
|
class = (GtkContainerClass*)g_type_class_ref (GTK_TYPE_CONTAINER);
|
|
funcs[i++] = class->get_path_for_child;
|
|
g_type_class_unref (class);
|
|
|
|
class = (GtkContainerClass*)g_type_class_ref (GTK_TYPE_BOX);
|
|
funcs[i++] = class->get_path_for_child;
|
|
g_type_class_unref (class);
|
|
|
|
g_assert (i == G_N_ELEMENTS (funcs));
|
|
}
|
|
|
|
parent = _gtk_widget_get_parent (widget);
|
|
if (parent == NULL)
|
|
return FALSE;
|
|
|
|
parent_func = GTK_CONTAINER_GET_CLASS (GTK_CONTAINER (parent))->get_path_for_child;
|
|
for (i = 0; i < G_N_ELEMENTS (funcs); i++)
|
|
{
|
|
if (funcs[i] == parent_func)
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
gboolean
|
|
gtk_css_widget_node_init_matcher (GtkCssNode *node,
|
|
GtkCssMatcher *matcher)
|
|
{
|
|
GtkCssWidgetNode *widget_node = GTK_CSS_WIDGET_NODE (node);
|
|
|
|
if (widget_node->widget == NULL)
|
|
return FALSE;
|
|
|
|
if (!widget_needs_widget_path (widget_node->widget))
|
|
return GTK_CSS_NODE_CLASS (gtk_css_widget_node_parent_class)->init_matcher (node, matcher);
|
|
|
|
return _gtk_css_matcher_init (matcher,
|
|
gtk_widget_get_path (widget_node->widget),
|
|
gtk_css_node_get_declaration (node));
|
|
}
|
|
|
|
static GtkWidgetPath *
|
|
gtk_css_widget_node_create_widget_path (GtkCssNode *node)
|
|
{
|
|
GtkCssWidgetNode *widget_node = GTK_CSS_WIDGET_NODE (node);
|
|
GtkWidgetPath *path;
|
|
guint length;
|
|
|
|
if (widget_node->widget == NULL)
|
|
path = gtk_widget_path_new ();
|
|
else
|
|
path = _gtk_widget_create_path (widget_node->widget);
|
|
|
|
length = gtk_widget_path_length (path);
|
|
if (length > 0)
|
|
{
|
|
gtk_css_node_declaration_add_to_widget_path (gtk_css_node_get_declaration (node),
|
|
path,
|
|
length - 1);
|
|
}
|
|
|
|
return path;
|
|
}
|
|
|
|
static const GtkWidgetPath *
|
|
gtk_css_widget_node_get_widget_path (GtkCssNode *node)
|
|
{
|
|
GtkCssWidgetNode *widget_node = GTK_CSS_WIDGET_NODE (node);
|
|
|
|
if (widget_node->widget == NULL)
|
|
return NULL;
|
|
|
|
return gtk_widget_get_path (widget_node->widget);
|
|
}
|
|
|
|
static GtkStyleProviderPrivate *
|
|
gtk_css_widget_node_get_style_provider (GtkCssNode *node)
|
|
{
|
|
GtkCssWidgetNode *widget_node = GTK_CSS_WIDGET_NODE (node);
|
|
GtkStyleContext *context;
|
|
GtkStyleCascade *cascade;
|
|
GtkSettings *settings;
|
|
|
|
if (widget_node->widget == NULL)
|
|
return NULL;
|
|
|
|
context = _gtk_widget_peek_style_context (widget_node->widget);
|
|
if (context)
|
|
return gtk_style_context_get_style_provider (context);
|
|
|
|
settings = gtk_widget_get_settings (widget_node->widget);
|
|
if (!settings)
|
|
return NULL;
|
|
|
|
cascade = _gtk_settings_get_style_cascade (gtk_widget_get_settings (widget_node->widget),
|
|
gtk_widget_get_scale_factor (widget_node->widget));
|
|
return GTK_STYLE_PROVIDER_PRIVATE (cascade);
|
|
}
|
|
|
|
static GdkFrameClock *
|
|
gtk_css_widget_node_get_frame_clock (GtkCssNode *node)
|
|
{
|
|
GtkCssWidgetNode *widget_node = GTK_CSS_WIDGET_NODE (node);
|
|
|
|
if (widget_node->widget == NULL)
|
|
return NULL;
|
|
|
|
if (!gtk_settings_get_enable_animations (gtk_widget_get_settings (widget_node->widget)))
|
|
return NULL;
|
|
|
|
return gtk_widget_get_frame_clock (widget_node->widget);
|
|
}
|
|
|
|
static void
|
|
gtk_css_widget_node_class_init (GtkCssWidgetNodeClass *klass)
|
|
{
|
|
GtkCssNodeClass *node_class = GTK_CSS_NODE_CLASS (klass);
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
object_class->finalize = gtk_css_widget_node_finalize;
|
|
node_class->update_style = gtk_css_widget_node_update_style;
|
|
node_class->validate = gtk_css_widget_node_validate;
|
|
node_class->queue_validate = gtk_css_widget_node_queue_validate;
|
|
node_class->dequeue_validate = gtk_css_widget_node_dequeue_validate;
|
|
node_class->init_matcher = gtk_css_widget_node_init_matcher;
|
|
node_class->create_widget_path = gtk_css_widget_node_create_widget_path;
|
|
node_class->get_widget_path = gtk_css_widget_node_get_widget_path;
|
|
node_class->get_style_provider = gtk_css_widget_node_get_style_provider;
|
|
node_class->get_frame_clock = gtk_css_widget_node_get_frame_clock;
|
|
node_class->style_changed = gtk_css_widget_node_style_changed;
|
|
}
|
|
|
|
static void
|
|
gtk_css_widget_node_init (GtkCssWidgetNode *node)
|
|
{
|
|
node->last_updated_style = g_object_ref (gtk_css_static_style_get_default ());
|
|
}
|
|
|
|
GtkCssNode *
|
|
gtk_css_widget_node_new (GtkWidget *widget)
|
|
{
|
|
GtkCssWidgetNode *result;
|
|
|
|
gtk_internal_return_val_if_fail (GTK_IS_WIDGET (widget), NULL);
|
|
|
|
result = g_object_new (GTK_TYPE_CSS_WIDGET_NODE, NULL);
|
|
result->widget = widget;
|
|
gtk_css_node_set_visible (GTK_CSS_NODE (result),
|
|
_gtk_widget_get_visible (widget));
|
|
|
|
return GTK_CSS_NODE (result);
|
|
}
|
|
|
|
void
|
|
gtk_css_widget_node_widget_destroyed (GtkCssWidgetNode *node)
|
|
{
|
|
gtk_internal_return_if_fail (GTK_IS_CSS_WIDGET_NODE (node));
|
|
gtk_internal_return_if_fail (node->widget != NULL);
|
|
|
|
node->widget = NULL;
|
|
/* Contents of this node are now undefined.
|
|
* So we don't clear the style or anything.
|
|
*/
|
|
}
|
|
|
|
GtkWidget *
|
|
gtk_css_widget_node_get_widget (GtkCssWidgetNode *node)
|
|
{
|
|
gtk_internal_return_val_if_fail (GTK_IS_CSS_WIDGET_NODE (node), NULL);
|
|
|
|
return node->widget;
|
|
}
|
|
|