forked from AuroraMiddleware/gtk
ec57c6c10b
Sometimes path nodes can survive longer than the style context that created them. Don't crash in those cases. Fixes startup of mutter. Testcase included. https://bugzilla.gnome.org/show_bug.cgi?id=746407
385 lines
12 KiB
C
385 lines
12 KiB
C
#include <gtk/gtk.h>
|
|
|
|
static void
|
|
test_parse_selectors (void)
|
|
{
|
|
GtkCssProvider *provider;
|
|
GError *error;
|
|
gboolean res;
|
|
gint i;
|
|
const gchar *valid[] = {
|
|
"* {}",
|
|
"E {}",
|
|
"E F {}",
|
|
"E > F {}",
|
|
"E#id {}",
|
|
"#id {}",
|
|
"tab:first-child {}",
|
|
"tab:last-child {}",
|
|
"tab:nth-child(first) {}",
|
|
"tab:nth-child(last) {}",
|
|
"tab:nth-child(even) {}",
|
|
"tab:nth-child(odd) {}",
|
|
"tab:sorted {}",
|
|
".some-class {}",
|
|
".some-class.another-class {}",
|
|
".some-class .another-class {}",
|
|
"E * {}",
|
|
"E .class {}",
|
|
"E > .foo {}",
|
|
"E > #id {}",
|
|
"E:active {}",
|
|
"E:prelight {}",
|
|
"E:hover {}",
|
|
"E:selected {}",
|
|
"E:insensitive {}",
|
|
"E:inconsistent {}",
|
|
"E:focused {}",
|
|
"E:active:prelight {}",
|
|
"* > .notebook tab:first-child .label:focused {}",
|
|
"E, F {}",
|
|
"E, F /* comment here */ {}",
|
|
"E,/* comment here */ F {}",
|
|
"E1.e1_2 #T3_4 {}",
|
|
"E:first-child {}",
|
|
"E:last-child {}",
|
|
"E:nth-child(first) {}",
|
|
"E:nth-child(last) {}",
|
|
"E:nth-child(even) {}",
|
|
"E:nth-child(odd) {}",
|
|
"E:sorted {}",
|
|
"E:focused tab {}",
|
|
NULL
|
|
};
|
|
|
|
error = NULL;
|
|
for (i = 0; valid[i]; i++)
|
|
{
|
|
provider = gtk_css_provider_new ();
|
|
res = gtk_css_provider_load_from_data (provider, valid[i], -1, &error);
|
|
if (error)
|
|
g_print ("parsing '%s': got unexpected error: %s\n", valid[i], error->message);
|
|
g_assert_no_error (error);
|
|
g_assert (res);
|
|
|
|
g_object_unref (provider);
|
|
}
|
|
}
|
|
|
|
static void
|
|
test_path (void)
|
|
{
|
|
GtkWidgetPath *path;
|
|
GtkWidgetPath *path2;
|
|
gint pos;
|
|
GtkRegionFlags flags;
|
|
|
|
path = gtk_widget_path_new ();
|
|
g_assert_cmpint (gtk_widget_path_length (path), ==, 0);
|
|
|
|
pos = gtk_widget_path_append_type (path, GTK_TYPE_WINDOW);
|
|
g_assert_cmpint (pos, ==, 0);
|
|
g_assert_cmpint (gtk_widget_path_length (path), ==, 1);
|
|
g_assert (gtk_widget_path_iter_get_object_type (path, 0) == GTK_TYPE_WINDOW);
|
|
g_assert (gtk_widget_path_is_type (path, GTK_TYPE_WIDGET));
|
|
g_assert (gtk_widget_path_iter_get_name (path, 0) == NULL);
|
|
|
|
pos = gtk_widget_path_append_type (path, GTK_TYPE_WIDGET);
|
|
g_assert_cmpint (pos, ==, 1);
|
|
g_assert_cmpint (gtk_widget_path_length (path), ==, 2);
|
|
gtk_widget_path_iter_set_object_type (path, pos, GTK_TYPE_BUTTON);
|
|
g_assert (gtk_widget_path_is_type (path, GTK_TYPE_BUTTON));
|
|
g_assert (gtk_widget_path_has_parent (path, GTK_TYPE_WIDGET));
|
|
g_assert (gtk_widget_path_has_parent (path, GTK_TYPE_WINDOW));
|
|
g_assert (!gtk_widget_path_has_parent (path, GTK_TYPE_DIALOG));
|
|
g_assert (gtk_widget_path_iter_get_name (path, 1) == NULL);
|
|
|
|
gtk_widget_path_iter_set_name (path, 1, "name");
|
|
g_assert (gtk_widget_path_iter_has_name (path, 1, "name"));
|
|
|
|
gtk_widget_path_iter_add_class (path, 1, "class1");
|
|
gtk_widget_path_iter_add_class (path, 1, "class2");
|
|
g_assert (gtk_widget_path_iter_has_class (path, 1, "class1"));
|
|
g_assert (gtk_widget_path_iter_has_class (path, 1, "class2"));
|
|
g_assert (!gtk_widget_path_iter_has_class (path, 1, "class3"));
|
|
|
|
path2 = gtk_widget_path_copy (path);
|
|
g_assert (gtk_widget_path_iter_has_class (path2, 1, "class1"));
|
|
g_assert (gtk_widget_path_iter_has_class (path2, 1, "class2"));
|
|
g_assert (!gtk_widget_path_iter_has_class (path2, 1, "class3"));
|
|
gtk_widget_path_free (path2);
|
|
|
|
gtk_widget_path_iter_remove_class (path, 1, "class2");
|
|
g_assert (gtk_widget_path_iter_has_class (path, 1, "class1"));
|
|
g_assert (!gtk_widget_path_iter_has_class (path, 1, "class2"));
|
|
gtk_widget_path_iter_clear_classes (path, 1);
|
|
g_assert (!gtk_widget_path_iter_has_class (path, 1, "class1"));
|
|
|
|
G_GNUC_BEGIN_IGNORE_DEPRECATIONS
|
|
gtk_widget_path_iter_add_region (path, 1, "tab", 0);
|
|
gtk_widget_path_iter_add_region (path, 1, "title", GTK_REGION_EVEN | GTK_REGION_FIRST);
|
|
|
|
g_assert (gtk_widget_path_iter_has_region (path, 1, "tab", &flags) &&
|
|
flags == 0);
|
|
g_assert (gtk_widget_path_iter_has_region (path, 1, "title", &flags) &&
|
|
flags == (GTK_REGION_EVEN | GTK_REGION_FIRST));
|
|
g_assert (!gtk_widget_path_iter_has_region (path, 1, "extension", NULL));
|
|
|
|
path2 = gtk_widget_path_copy (path);
|
|
g_assert (gtk_widget_path_iter_has_region (path2, 1, "tab", &flags) &&
|
|
flags == 0);
|
|
g_assert (gtk_widget_path_iter_has_region (path2, 1, "title", &flags) &&
|
|
flags == (GTK_REGION_EVEN | GTK_REGION_FIRST));
|
|
g_assert (!gtk_widget_path_iter_has_region (path2, 1, "extension", NULL));
|
|
G_GNUC_END_IGNORE_DEPRECATIONS
|
|
|
|
gtk_widget_path_free (path2);
|
|
|
|
gtk_widget_path_free (path);
|
|
}
|
|
|
|
static void
|
|
test_match (void)
|
|
{
|
|
GtkStyleContext *context;
|
|
GtkWidgetPath *path;
|
|
GtkCssProvider *provider;
|
|
GError *error;
|
|
const gchar *data;
|
|
GdkRGBA color;
|
|
GdkRGBA expected;
|
|
|
|
error = NULL;
|
|
provider = gtk_css_provider_new ();
|
|
|
|
gdk_rgba_parse (&expected, "#fff");
|
|
|
|
context = gtk_style_context_new ();
|
|
|
|
path = gtk_widget_path_new ();
|
|
gtk_widget_path_append_type (path, GTK_TYPE_WINDOW);
|
|
gtk_widget_path_append_type (path, GTK_TYPE_BOX);
|
|
gtk_widget_path_append_type (path, GTK_TYPE_BUTTON);
|
|
gtk_widget_path_iter_set_name (path, 0, "mywindow");
|
|
gtk_widget_path_iter_add_class (path, 2, "button");
|
|
gtk_widget_path_iter_set_state (path, 0, GTK_STATE_FLAG_ACTIVE);
|
|
gtk_style_context_set_path (context, path);
|
|
gtk_widget_path_free (path);
|
|
|
|
gtk_style_context_add_provider (context,
|
|
GTK_STYLE_PROVIDER (provider),
|
|
GTK_STYLE_PROVIDER_PRIORITY_USER);
|
|
|
|
data = "* { color: #fff }";
|
|
gtk_css_provider_load_from_data (provider, data, -1, &error);
|
|
g_assert_no_error (error);
|
|
gtk_style_context_get_color (context, GTK_STATE_FLAG_NORMAL, &color);
|
|
g_assert (gdk_rgba_equal (&color, &expected));
|
|
|
|
data = "* { color: #f00 }\n"
|
|
"GtkButton { color: #fff }";
|
|
gtk_css_provider_load_from_data (provider, data, -1, &error);
|
|
g_assert_no_error (error);
|
|
gtk_style_context_get_color (context, GTK_STATE_FLAG_NORMAL, &color);
|
|
g_assert (gdk_rgba_equal (&color, &expected));
|
|
|
|
data = "* { color: #f00 }\n"
|
|
"GtkButton { color: #fff }\n"
|
|
"GtkWindow > GtkButton { color: #000 }";
|
|
gtk_css_provider_load_from_data (provider, data, -1, &error);
|
|
g_assert_no_error (error);
|
|
gtk_style_context_get_color (context, GTK_STATE_FLAG_NORMAL, &color);
|
|
g_assert (gdk_rgba_equal (&color, &expected));
|
|
|
|
data = "* { color: #f00 }\n"
|
|
".button { color: #fff }";
|
|
gtk_css_provider_load_from_data (provider, data, -1, &error);
|
|
g_assert_no_error (error);
|
|
gtk_style_context_get_color (context, GTK_STATE_FLAG_NORMAL, &color);
|
|
g_assert (gdk_rgba_equal (&color, &expected));
|
|
|
|
data = "* { color: #f00 }\n"
|
|
"GtkButton { color: #000 }\n"
|
|
".button { color: #fff }";
|
|
gtk_css_provider_load_from_data (provider, data, -1, &error);
|
|
g_assert_no_error (error);
|
|
gtk_style_context_get_color (context, GTK_STATE_FLAG_NORMAL, &color);
|
|
g_assert (gdk_rgba_equal (&color, &expected));
|
|
|
|
data = "* { color: #f00 }\n"
|
|
"GtkButton { color: #000 }\n"
|
|
"GtkWindow GtkButton { color: #fff }";
|
|
gtk_css_provider_load_from_data (provider, data, -1, &error);
|
|
g_assert_no_error (error);
|
|
gtk_style_context_get_color (context, GTK_STATE_FLAG_NORMAL, &color);
|
|
g_assert (gdk_rgba_equal (&color, &expected));
|
|
|
|
data = "* { color: #f00 }\n"
|
|
".button { color: #000 }\n"
|
|
"GtkWindow .button { color: #fff }";
|
|
gtk_css_provider_load_from_data (provider, data, -1, &error);
|
|
g_assert_no_error (error);
|
|
gtk_style_context_get_color (context, GTK_STATE_FLAG_NORMAL, &color);
|
|
g_assert (gdk_rgba_equal (&color, &expected));
|
|
|
|
data = "* { color: #f00 }\n"
|
|
"* .button { color: #000 }\n"
|
|
"#mywindow .button { color: #fff }";
|
|
gtk_css_provider_load_from_data (provider, data, -1, &error);
|
|
g_assert_no_error (error);
|
|
gtk_style_context_get_color (context, GTK_STATE_FLAG_NORMAL, &color);
|
|
g_assert (gdk_rgba_equal (&color, &expected));
|
|
|
|
data = "* { color: #f00 }\n"
|
|
"GtkWindow .button { color: #000 }\n"
|
|
"GtkWindow#mywindow .button { color: #fff }";
|
|
gtk_css_provider_load_from_data (provider, data, -1, &error);
|
|
g_assert_no_error (error);
|
|
gtk_style_context_get_color (context, GTK_STATE_FLAG_NORMAL, &color);
|
|
g_assert (gdk_rgba_equal (&color, &expected));
|
|
|
|
data = "* { color: #f00 }\n"
|
|
"GtkWindow .button { color: #000 }\n"
|
|
"GObject .button { color: #fff }";
|
|
gtk_css_provider_load_from_data (provider, data, -1, &error);
|
|
g_assert_no_error (error);
|
|
gtk_style_context_get_color (context, GTK_STATE_FLAG_NORMAL, &color);
|
|
g_assert (gdk_rgba_equal (&color, &expected));
|
|
|
|
data = "* { color: #f00 }\n"
|
|
"GtkWindow:backdrop .button { color: #000 }\n"
|
|
"GtkWindow .button { color: #111 }\n"
|
|
"GtkWindow:active .button { color: #fff }";
|
|
gtk_css_provider_load_from_data (provider, data, -1, &error);
|
|
g_assert_no_error (error);
|
|
gtk_style_context_get_color (context, GTK_STATE_FLAG_NORMAL, &color);
|
|
g_assert (gdk_rgba_equal (&color, &expected));
|
|
|
|
g_object_unref (provider);
|
|
g_object_unref (context);
|
|
}
|
|
|
|
static void
|
|
test_basic_properties (void)
|
|
{
|
|
GtkStyleContext *context;
|
|
GtkWidgetPath *path;
|
|
GdkRGBA *color;
|
|
GdkRGBA *bg_color;
|
|
PangoFontDescription *font;
|
|
|
|
context = gtk_style_context_new ();
|
|
path = gtk_widget_path_new ();
|
|
gtk_style_context_set_path (context, path);
|
|
gtk_widget_path_free (path);
|
|
|
|
gtk_style_context_get (context, 0,
|
|
"color", &color,
|
|
"background-color", &bg_color,
|
|
"font", &font,
|
|
NULL);
|
|
g_assert (color != NULL);
|
|
g_assert (bg_color != NULL);
|
|
g_assert (font != NULL);
|
|
|
|
gdk_rgba_free (color);
|
|
gdk_rgba_free (bg_color);
|
|
pango_font_description_free (font);
|
|
|
|
g_object_unref (context);
|
|
}
|
|
|
|
void
|
|
test_invalidate_saved (void)
|
|
{
|
|
GtkWidget *window;
|
|
GtkStyleContext *context;
|
|
|
|
window = gtk_window_new (GTK_WINDOW_TOPLEVEL);
|
|
|
|
context = gtk_widget_get_style_context (window);
|
|
gtk_style_context_save (context);
|
|
G_GNUC_BEGIN_IGNORE_DEPRECATIONS
|
|
gtk_style_context_invalidate (context);
|
|
G_GNUC_END_IGNORE_DEPRECATIONS
|
|
gtk_style_context_restore (context);
|
|
|
|
gtk_widget_destroy (window);
|
|
}
|
|
|
|
void
|
|
test_set_widget_path_saved (void)
|
|
{
|
|
GtkWidgetPath *path;
|
|
GtkCssProvider *provider;
|
|
GtkStyleContext *context;
|
|
GtkBorder padding;
|
|
|
|
context = gtk_style_context_new ();
|
|
|
|
provider = gtk_css_provider_new ();
|
|
gtk_style_context_add_provider (context,
|
|
GTK_STYLE_PROVIDER (provider),
|
|
GTK_STYLE_PROVIDER_PRIORITY_USER);
|
|
gtk_css_provider_load_from_data (provider,
|
|
"GtkWindow * { padding: 1px; }\n"
|
|
".foo * { padding: 2px; }\n",
|
|
-1,
|
|
NULL);
|
|
|
|
path = gtk_widget_path_new ();
|
|
gtk_widget_path_append_type (path, GTK_TYPE_WINDOW);
|
|
gtk_style_context_set_path (context, path);
|
|
|
|
gtk_style_context_get_padding (context, gtk_style_context_get_state (context), &padding);
|
|
g_assert_cmpint (padding.top, ==, 0);
|
|
|
|
gtk_style_context_save (context);
|
|
gtk_style_context_get_padding (context, gtk_style_context_get_state (context), &padding);
|
|
g_assert_cmpint (padding.top, ==, 1);
|
|
|
|
gtk_widget_path_iter_add_class (path, -1, "foo");
|
|
gtk_style_context_set_path (context, path);
|
|
gtk_style_context_get_padding (context, gtk_style_context_get_state (context), &padding);
|
|
g_assert_cmpint (padding.top, ==, 2);
|
|
|
|
gtk_style_context_restore (context);
|
|
gtk_style_context_get_padding (context, gtk_style_context_get_state (context), &padding);
|
|
g_assert_cmpint (padding.top, ==, 0);
|
|
|
|
gtk_widget_path_free (path);
|
|
g_object_unref (provider);
|
|
g_object_unref (context);
|
|
}
|
|
|
|
void
|
|
test_widget_path_parent (void)
|
|
{
|
|
GtkStyleContext *parent, *context;
|
|
|
|
parent = gtk_style_context_new ();
|
|
context = gtk_style_context_new ();
|
|
|
|
gtk_style_context_set_parent (context, parent);
|
|
|
|
g_object_unref (parent);
|
|
g_object_unref (context);
|
|
}
|
|
|
|
int
|
|
main (int argc, char *argv[])
|
|
{
|
|
gtk_init (NULL, NULL);
|
|
g_test_init (&argc, &argv, NULL);
|
|
|
|
g_test_add_func ("/style/parse/selectors", test_parse_selectors);
|
|
g_test_add_func ("/style/path", test_path);
|
|
g_test_add_func ("/style/match", test_match);
|
|
g_test_add_func ("/style/basic", test_basic_properties);
|
|
g_test_add_func ("/style/invalidate-saved", test_invalidate_saved);
|
|
g_test_add_func ("/style/set-widget-path-saved", test_set_widget_path_saved);
|
|
g_test_add_func ("/style/widget-path-parent", test_widget_path_parent);
|
|
|
|
return g_test_run ();
|
|
}
|