1 /* GTK - The GIMP Toolkit
2 * Copyright (C) 2010 Carlos Garnacho <carlosg@gnome.org>
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser General Public
6 * License as published by the Free Software Foundation; either
7 * version 2 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Lesser General Public License for more details.
14 * You should have received a copy of the GNU Lesser General Public
15 * License along with this library; if not, write to the
16 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
17 * Boston, MA 02111-1307, USA.
22 #include "gtkstylepropertyprivate.h"
28 #include <gdk-pixbuf/gdk-pixbuf.h>
29 #include <cairo-gobject.h>
31 #include "gtkcssprovider.h"
32 #include "gtkcssparserprivate.h"
33 #include "gtkcsstypesprivate.h"
35 /* the actual parsers we have */
36 #include "gtkanimationdescription.h"
37 #include "gtkbindings.h"
38 #include "gtkborderimageprivate.h"
39 #include "gtkgradient.h"
40 #include "gtkshadowprivate.h"
41 #include "gtkthemingengine.h"
42 #include "gtktypebuiltins.h"
44 static GHashTable *parse_funcs = NULL;
45 static GHashTable *print_funcs = NULL;
46 static GHashTable *properties = NULL;
49 register_conversion_function (GType type,
50 GtkStyleParseFunc parse,
51 GtkStylePrintFunc print)
54 g_hash_table_insert (parse_funcs, GSIZE_TO_POINTER (type), parse);
56 g_hash_table_insert (print_funcs, GSIZE_TO_POINTER (type), print);
60 string_append_double (GString *string,
63 char buf[G_ASCII_DTOSTR_BUF_SIZE];
65 g_ascii_dtostr (buf, sizeof (buf), d);
66 g_string_append (string, buf);
70 string_append_string (GString *str,
75 g_string_append_c (str, '"');
78 len = strcspn (string, "\"\n\r\f");
79 g_string_append (str, string);
86 g_string_append (str, "\\A ");
89 g_string_append (str, "\\D ");
92 g_string_append (str, "\\C ");
95 g_string_append (str, "\\\"");
98 g_assert_not_reached ();
103 g_string_append_c (str, '"');
106 /*** IMPLEMENTATIONS ***/
109 rgba_value_parse (GtkCssParser *parser,
113 GtkSymbolicColor *symbolic;
116 symbolic = _gtk_css_parser_read_symbolic_color (parser);
117 if (symbolic == NULL)
120 if (gtk_symbolic_color_resolve (symbolic, NULL, &rgba))
122 g_value_set_boxed (value, &rgba);
123 gtk_symbolic_color_unref (symbolic);
127 g_value_unset (value);
128 g_value_init (value, GTK_TYPE_SYMBOLIC_COLOR);
129 g_value_take_boxed (value, symbolic);
136 rgba_value_print (const GValue *value,
139 const GdkRGBA *rgba = g_value_get_boxed (value);
142 g_string_append (string, "none");
145 char *s = gdk_rgba_to_string (rgba);
146 g_string_append (string, s);
152 color_value_parse (GtkCssParser *parser,
156 GtkSymbolicColor *symbolic;
159 symbolic = _gtk_css_parser_read_symbolic_color (parser);
160 if (symbolic == NULL)
163 if (gtk_symbolic_color_resolve (symbolic, NULL, &rgba))
167 color.red = rgba.red * 65535. + 0.5;
168 color.green = rgba.green * 65535. + 0.5;
169 color.blue = rgba.blue * 65535. + 0.5;
171 g_value_set_boxed (value, &color);
175 g_value_unset (value);
176 g_value_init (value, GTK_TYPE_SYMBOLIC_COLOR);
177 g_value_take_boxed (value, symbolic);
184 color_value_print (const GValue *value,
187 const GdkColor *color = g_value_get_boxed (value);
190 g_string_append (string, "none");
193 char *s = gdk_color_to_string (color);
194 g_string_append (string, s);
200 symbolic_color_value_parse (GtkCssParser *parser,
204 GtkSymbolicColor *symbolic;
206 symbolic = _gtk_css_parser_read_symbolic_color (parser);
207 if (symbolic == NULL)
210 g_value_take_boxed (value, symbolic);
215 symbolic_color_value_print (const GValue *value,
218 GtkSymbolicColor *symbolic = g_value_get_boxed (value);
220 if (symbolic == NULL)
221 g_string_append (string, "none");
224 char *s = gtk_symbolic_color_to_string (symbolic);
225 g_string_append (string, s);
231 font_description_value_parse (GtkCssParser *parser,
235 PangoFontDescription *font_desc;
238 str = _gtk_css_parser_read_value (parser);
242 font_desc = pango_font_description_from_string (str);
244 g_value_take_boxed (value, font_desc);
249 font_description_value_print (const GValue *value,
252 const PangoFontDescription *desc = g_value_get_boxed (value);
255 g_string_append (string, "none");
258 char *s = pango_font_description_to_string (desc);
259 g_string_append (string, s);
265 boolean_value_parse (GtkCssParser *parser,
269 if (_gtk_css_parser_try (parser, "true", TRUE) ||
270 _gtk_css_parser_try (parser, "1", TRUE))
272 g_value_set_boolean (value, TRUE);
275 else if (_gtk_css_parser_try (parser, "false", TRUE) ||
276 _gtk_css_parser_try (parser, "0", TRUE))
278 g_value_set_boolean (value, FALSE);
283 _gtk_css_parser_error (parser, "Expected a boolean value");
289 boolean_value_print (const GValue *value,
292 if (g_value_get_boolean (value))
293 g_string_append (string, "true");
295 g_string_append (string, "false");
299 int_value_parse (GtkCssParser *parser,
305 if (!_gtk_css_parser_try_int (parser, &i))
307 _gtk_css_parser_error (parser, "Expected a valid integer value");
311 g_value_set_int (value, i);
316 int_value_print (const GValue *value,
319 g_string_append_printf (string, "%d", g_value_get_int (value));
323 uint_value_parse (GtkCssParser *parser,
329 if (!_gtk_css_parser_try_uint (parser, &u))
331 _gtk_css_parser_error (parser, "Expected a valid unsigned value");
335 g_value_set_uint (value, u);
340 uint_value_print (const GValue *value,
343 g_string_append_printf (string, "%u", g_value_get_uint (value));
347 double_value_parse (GtkCssParser *parser,
353 if (!_gtk_css_parser_try_double (parser, &d))
355 _gtk_css_parser_error (parser, "Expected a number");
359 g_value_set_double (value, d);
364 double_value_print (const GValue *value,
367 string_append_double (string, g_value_get_double (value));
371 float_value_parse (GtkCssParser *parser,
377 if (!_gtk_css_parser_try_double (parser, &d))
379 _gtk_css_parser_error (parser, "Expected a number");
383 g_value_set_float (value, d);
388 float_value_print (const GValue *value,
391 string_append_double (string, g_value_get_float (value));
395 string_value_parse (GtkCssParser *parser,
399 char *str = _gtk_css_parser_read_string (parser);
404 g_value_take_string (value, str);
409 string_value_print (const GValue *value,
412 string_append_string (str, g_value_get_string (value));
416 theming_engine_value_parse (GtkCssParser *parser,
420 GtkThemingEngine *engine;
423 str = _gtk_css_parser_try_ident (parser, TRUE);
426 _gtk_css_parser_error (parser, "Expected a valid theme name");
430 engine = gtk_theming_engine_load (str);
433 _gtk_css_parser_error (parser, "Themeing engine '%s' not found", str);
438 g_value_set_object (value, engine);
444 theming_engine_value_print (const GValue *value,
447 GtkThemingEngine *engine;
450 engine = g_value_get_object (value);
452 g_string_append (string, "none");
455 /* XXX: gtk_theming_engine_get_name()? */
456 g_object_get (engine, "name", &name, NULL);
457 g_string_append (string, name);
463 animation_description_value_parse (GtkCssParser *parser,
467 GtkAnimationDescription *desc;
470 str = _gtk_css_parser_read_value (parser);
474 desc = _gtk_animation_description_from_string (str);
479 _gtk_css_parser_error (parser, "Invalid animation description");
483 g_value_take_boxed (value, desc);
488 animation_description_value_print (const GValue *value,
491 GtkAnimationDescription *desc = g_value_get_boxed (value);
494 g_string_append (string, "none");
496 _gtk_animation_description_print (desc, string);
500 border_value_parse (GtkCssParser *parser,
504 GtkBorder border = { 0, };
507 for (i = 0; i < G_N_ELEMENTS (numbers); i++)
509 if (!_gtk_css_parser_try_uint (parser, &numbers[i]))
512 /* XXX: shouldn't allow spaces here? */
513 _gtk_css_parser_try (parser, "px", TRUE);
518 _gtk_css_parser_error (parser, "Expected valid border");
522 border.top = numbers[0];
524 border.right = numbers[1];
526 border.right = border.top;
528 border.bottom = numbers[2];
530 border.bottom = border.top;
532 border.left = numbers[3];
534 border.left = border.right;
536 g_value_set_boxed (value, &border);
541 border_value_print (const GValue *value, GString *string)
543 const GtkBorder *border = g_value_get_boxed (value);
546 g_string_append (string, "none");
547 else if (border->left != border->right)
548 g_string_append_printf (string, "%d %d %d %d", border->top, border->right, border->bottom, border->left);
549 else if (border->top != border->bottom)
550 g_string_append_printf (string, "%d %d %d", border->top, border->right, border->bottom);
551 else if (border->top != border->left)
552 g_string_append_printf (string, "%d %d", border->top, border->right);
554 g_string_append_printf (string, "%d", border->top);
558 gradient_value_parse (GtkCssParser *parser,
562 GtkGradient *gradient;
563 cairo_pattern_type_t type;
567 if (!_gtk_css_parser_try (parser, "-gtk-gradient", TRUE))
569 _gtk_css_parser_error (parser,
570 "Expected '-gtk-gradient'");
574 if (!_gtk_css_parser_try (parser, "(", TRUE))
576 _gtk_css_parser_error (parser,
577 "Expected '(' after '-gtk-gradient'");
581 /* Parse gradient type */
582 if (_gtk_css_parser_try (parser, "linear", TRUE))
583 type = CAIRO_PATTERN_TYPE_LINEAR;
584 else if (_gtk_css_parser_try (parser, "radial", TRUE))
585 type = CAIRO_PATTERN_TYPE_RADIAL;
588 _gtk_css_parser_error (parser,
589 "Gradient type must be 'radial' or 'linear'");
593 /* Parse start/stop position parameters */
594 for (i = 0; i < 2; i++)
596 if (! _gtk_css_parser_try (parser, ",", TRUE))
598 _gtk_css_parser_error (parser,
603 if (_gtk_css_parser_try (parser, "left", TRUE))
605 else if (_gtk_css_parser_try (parser, "right", TRUE))
607 else if (_gtk_css_parser_try (parser, "center", TRUE))
609 else if (!_gtk_css_parser_try_double (parser, &coords[i * 3]))
611 _gtk_css_parser_error (parser,
612 "Expected a valid X coordinate");
616 if (_gtk_css_parser_try (parser, "top", TRUE))
617 coords[i * 3 + 1] = 0;
618 else if (_gtk_css_parser_try (parser, "bottom", TRUE))
619 coords[i * 3 + 1] = 1;
620 else if (_gtk_css_parser_try (parser, "center", TRUE))
621 coords[i * 3 + 1] = 0.5;
622 else if (!_gtk_css_parser_try_double (parser, &coords[i * 3 + 1]))
624 _gtk_css_parser_error (parser,
625 "Expected a valid Y coordinate");
629 if (type == CAIRO_PATTERN_TYPE_RADIAL)
632 if (! _gtk_css_parser_try (parser, ",", TRUE))
634 _gtk_css_parser_error (parser,
639 if (! _gtk_css_parser_try_double (parser, &coords[(i * 3) + 2]))
641 _gtk_css_parser_error (parser,
642 "Expected a numer for the radius");
648 if (type == CAIRO_PATTERN_TYPE_LINEAR)
649 gradient = gtk_gradient_new_linear (coords[0], coords[1], coords[3], coords[4]);
651 gradient = gtk_gradient_new_radial (coords[0], coords[1], coords[2],
652 coords[3], coords[4], coords[5]);
654 while (_gtk_css_parser_try (parser, ",", TRUE))
656 GtkSymbolicColor *color;
659 if (_gtk_css_parser_try (parser, "from", TRUE))
663 if (!_gtk_css_parser_try (parser, "(", TRUE))
665 gtk_gradient_unref (gradient);
666 _gtk_css_parser_error (parser,
672 else if (_gtk_css_parser_try (parser, "to", TRUE))
676 if (!_gtk_css_parser_try (parser, "(", TRUE))
678 gtk_gradient_unref (gradient);
679 _gtk_css_parser_error (parser,
685 else if (_gtk_css_parser_try (parser, "color-stop", TRUE))
687 if (!_gtk_css_parser_try (parser, "(", TRUE))
689 gtk_gradient_unref (gradient);
690 _gtk_css_parser_error (parser,
695 if (!_gtk_css_parser_try_double (parser, &position))
697 gtk_gradient_unref (gradient);
698 _gtk_css_parser_error (parser,
699 "Expected a valid number");
703 if (!_gtk_css_parser_try (parser, ",", TRUE))
705 gtk_gradient_unref (gradient);
706 _gtk_css_parser_error (parser,
713 gtk_gradient_unref (gradient);
714 _gtk_css_parser_error (parser,
715 "Not a valid color-stop definition");
719 color = _gtk_css_parser_read_symbolic_color (parser);
722 gtk_gradient_unref (gradient);
726 gtk_gradient_add_color_stop (gradient, position, color);
727 gtk_symbolic_color_unref (color);
729 if (!_gtk_css_parser_try (parser, ")", TRUE))
731 gtk_gradient_unref (gradient);
732 _gtk_css_parser_error (parser,
738 if (!_gtk_css_parser_try (parser, ")", TRUE))
740 gtk_gradient_unref (gradient);
741 _gtk_css_parser_error (parser,
746 g_value_take_boxed (value, gradient);
751 gradient_value_print (const GValue *value,
754 GtkGradient *gradient = g_value_get_boxed (value);
756 if (gradient == NULL)
757 g_string_append (string, "none");
760 char *s = gtk_gradient_to_string (gradient);
761 g_string_append (string, s);
767 gtk_css_parse_url (GtkCssParser *parser,
773 if (_gtk_css_parser_try (parser, "url", FALSE))
775 if (!_gtk_css_parser_try (parser, "(", TRUE))
777 _gtk_css_parser_skip_whitespace (parser);
778 if (_gtk_css_parser_try (parser, "(", TRUE))
782 error = g_error_new_literal (GTK_CSS_PROVIDER_ERROR,
783 GTK_CSS_PROVIDER_ERROR_DEPRECATED,
784 "Whitespace between 'url' and '(' is not allowed");
786 _gtk_css_parser_take_error (parser, error);
790 _gtk_css_parser_error (parser, "Expected '(' after 'url'");
795 path = _gtk_css_parser_read_string (parser);
799 if (!_gtk_css_parser_try (parser, ")", TRUE))
801 _gtk_css_parser_error (parser, "No closing ')' found for 'url'");
808 path = _gtk_css_parser_try_name (parser, TRUE);
811 _gtk_css_parser_error (parser, "Not a valid url");
816 file = g_file_resolve_relative_path (base, path);
823 pattern_value_parse (GtkCssParser *parser,
827 if (_gtk_css_parser_begins_with (parser, '-'))
829 g_value_unset (value);
830 g_value_init (value, GTK_TYPE_GRADIENT);
831 return gradient_value_parse (parser, base, value);
835 GError *error = NULL;
839 cairo_surface_t *surface;
840 cairo_pattern_t *pattern;
842 cairo_matrix_t matrix;
844 file = gtk_css_parse_url (parser, base);
848 path = g_file_get_path (file);
849 g_object_unref (file);
851 pixbuf = gdk_pixbuf_new_from_file (path, &error);
855 _gtk_css_parser_take_error (parser, error);
859 surface = cairo_image_surface_create (CAIRO_FORMAT_ARGB32,
860 gdk_pixbuf_get_width (pixbuf),
861 gdk_pixbuf_get_height (pixbuf));
862 cr = cairo_create (surface);
863 gdk_cairo_set_source_pixbuf (cr, pixbuf, 0, 0);
865 pattern = cairo_pattern_create_for_surface (surface);
867 cairo_matrix_init_scale (&matrix,
868 gdk_pixbuf_get_width (pixbuf),
869 gdk_pixbuf_get_height (pixbuf));
870 cairo_pattern_set_matrix (pattern, &matrix);
872 cairo_surface_destroy (surface);
874 g_object_unref (pixbuf);
876 g_value_take_boxed (value, pattern);
883 shadow_value_parse (GtkCssParser *parser,
887 gboolean have_inset, have_color, have_lengths;
888 gdouble hoffset, voffset, blur, spread;
889 GtkSymbolicColor *color;
893 shadow = _gtk_shadow_new ();
897 have_inset = have_lengths = have_color = FALSE;
899 for (i = 0; i < 3; i++)
902 _gtk_css_parser_try (parser, "inset", TRUE))
909 _gtk_css_parser_try_double (parser, &hoffset))
913 if (!_gtk_css_parser_try_double (parser, &voffset))
915 _gtk_css_parser_error (parser, "Horizontal and vertical offsets are required");
916 _gtk_shadow_unref (shadow);
920 if (!_gtk_css_parser_try_double (parser, &blur))
923 if (!_gtk_css_parser_try_double (parser, &spread))
933 /* XXX: the color is optional and UA-defined if it's missing,
934 * but it doesn't really make sense for us...
936 color = _gtk_css_parser_read_symbolic_color (parser);
940 _gtk_shadow_unref (shadow);
946 if (!have_color || !have_lengths)
948 _gtk_css_parser_error (parser, "Must specify at least color and offsets");
949 _gtk_shadow_unref (shadow);
953 _gtk_shadow_append (shadow,
958 gtk_symbolic_color_unref (color);
961 while (_gtk_css_parser_try (parser, ",", TRUE));
963 g_value_take_boxed (value, shadow);
968 shadow_value_print (const GValue *value,
973 shadow = g_value_get_boxed (value);
976 g_string_append (string, "none");
978 _gtk_shadow_print (shadow, string);
982 border_image_repeat_value_parse (GtkCssParser *parser,
986 GtkCssBorderImageRepeat image_repeat;
987 GtkCssRepeatStyle styles[2];
990 for (i = 0; i < 2; i++)
992 if (_gtk_css_parser_try (parser, "stretch", TRUE))
993 styles[i] = GTK_CSS_REPEAT_STYLE_NONE;
994 else if (_gtk_css_parser_try (parser, "repeat", TRUE))
995 styles[i] = GTK_CSS_REPEAT_STYLE_REPEAT;
996 else if (_gtk_css_parser_try (parser, "round", TRUE))
997 styles[i] = GTK_CSS_REPEAT_STYLE_ROUND;
998 else if (_gtk_css_parser_try (parser, "space", TRUE))
999 styles[i] = GTK_CSS_REPEAT_STYLE_SPACE;
1002 styles[1] = styles[0] = GTK_CSS_REPEAT_STYLE_NONE;
1006 styles[i] = styles[0];
1009 image_repeat.hrepeat = styles[0];
1010 image_repeat.vrepeat = styles[1];
1012 g_value_set_boxed (value, &image_repeat);
1017 static const gchar *
1018 border_image_repeat_style_to_string (GtkCssRepeatStyle repeat)
1022 case GTK_CSS_REPEAT_STYLE_NONE:
1024 case GTK_CSS_REPEAT_STYLE_REPEAT:
1026 case GTK_CSS_REPEAT_STYLE_ROUND:
1028 case GTK_CSS_REPEAT_STYLE_SPACE:
1036 border_image_repeat_value_print (const GValue *value,
1039 GtkCssBorderImageRepeat *image_repeat;
1041 image_repeat = g_value_get_boxed (value);
1043 g_string_append_printf (string, "%s %s",
1044 border_image_repeat_style_to_string (image_repeat->hrepeat),
1045 border_image_repeat_style_to_string (image_repeat->vrepeat));
1049 border_image_value_parse (GtkCssParser *parser,
1053 GValue temp = { 0, };
1054 cairo_pattern_t *pattern = NULL;
1055 GtkGradient *gradient = NULL;
1056 GtkBorder slice, *width = NULL, *parsed_slice;
1057 GtkCssBorderImageRepeat repeat, *parsed_repeat;
1058 gboolean retval = FALSE;
1059 GtkBorderImage *image = NULL;
1061 g_value_init (&temp, CAIRO_GOBJECT_TYPE_PATTERN);
1063 if (!pattern_value_parse (parser, base, &temp))
1066 if (G_VALUE_TYPE (&temp) == GTK_TYPE_GRADIENT)
1067 gradient = g_value_dup_boxed (&temp);
1069 pattern = g_value_dup_boxed (&temp);
1071 g_value_unset (&temp);
1072 g_value_init (&temp, GTK_TYPE_BORDER);
1074 if (!border_value_parse (parser, base, &temp))
1077 parsed_slice = g_value_get_boxed (&temp);
1078 slice = *parsed_slice;
1080 if (_gtk_css_parser_try (parser, "/", TRUE))
1082 g_value_unset (&temp);
1083 g_value_init (&temp, GTK_TYPE_BORDER);
1085 if (!border_value_parse (parser, base, &temp))
1088 width = g_value_dup_boxed (&temp);
1091 g_value_unset (&temp);
1092 g_value_init (&temp, GTK_TYPE_CSS_BORDER_IMAGE_REPEAT);
1094 if (!border_image_repeat_value_parse (parser, base, &temp))
1097 parsed_repeat = g_value_get_boxed (&temp);
1098 repeat = *parsed_repeat;
1100 g_value_unset (&temp);
1102 if (gradient != NULL)
1103 image = _gtk_border_image_new_for_gradient (gradient, &slice, width, &repeat);
1104 else if (pattern != NULL)
1105 image = _gtk_border_image_new (pattern, &slice, width, &repeat);
1110 g_value_take_boxed (value, image);
1114 if (pattern != NULL)
1115 cairo_pattern_destroy (pattern);
1117 if (gradient != NULL)
1118 gtk_gradient_unref (gradient);
1121 gtk_border_free (width);
1127 enum_value_parse (GtkCssParser *parser,
1131 GEnumClass *enum_class;
1132 GEnumValue *enum_value;
1135 str = _gtk_css_parser_try_ident (parser, TRUE);
1138 _gtk_css_parser_error (parser, "Expected an identifier");
1142 enum_class = g_type_class_ref (G_VALUE_TYPE (value));
1143 enum_value = g_enum_get_value_by_nick (enum_class, str);
1146 g_value_set_enum (value, enum_value->value);
1148 _gtk_css_parser_error (parser,
1149 "Unknown value '%s' for enum type '%s'",
1150 str, g_type_name (G_VALUE_TYPE (value)));
1152 g_type_class_unref (enum_class);
1155 return enum_value != NULL;
1159 enum_value_print (const GValue *value,
1162 GEnumClass *enum_class;
1163 GEnumValue *enum_value;
1165 enum_class = g_type_class_ref (G_VALUE_TYPE (value));
1166 enum_value = g_enum_get_value (enum_class, g_value_get_enum (value));
1168 g_string_append (string, enum_value->value_nick);
1170 g_type_class_unref (enum_class);
1174 flags_value_parse (GtkCssParser *parser,
1178 GFlagsClass *flags_class;
1179 GFlagsValue *flag_value;
1183 flags_class = g_type_class_ref (G_VALUE_TYPE (value));
1186 str = _gtk_css_parser_try_ident (parser, TRUE);
1189 _gtk_css_parser_error (parser, "Expected an identifier");
1190 g_type_class_unref (flags_class);
1194 flag_value = g_flags_get_value_by_nick (flags_class, str);
1197 _gtk_css_parser_error (parser,
1198 "Unknown flag value '%s' for type '%s'",
1199 str, g_type_name (G_VALUE_TYPE (value)));
1200 /* XXX Do we want to return FALSE here? We can get
1201 * forward-compatibility for new values this way
1204 g_type_class_unref (flags_class);
1210 while (_gtk_css_parser_try (parser, ",", FALSE));
1212 g_type_class_unref (flags_class);
1214 g_value_set_enum (value, flags);
1220 flags_value_print (const GValue *value,
1223 GFlagsClass *flags_class;
1226 flags_class = g_type_class_ref (G_VALUE_TYPE (value));
1227 flags = g_value_get_flags (value);
1229 for (i = 0; i < flags_class->n_values; i++)
1231 GFlagsValue *flags_value = &flags_class->values[i];
1233 if (flags & flags_value->value)
1235 if (string->len != 0)
1236 g_string_append (string, ", ");
1238 g_string_append (string, flags_value->value_nick);
1242 g_type_class_unref (flags_class);
1246 bindings_value_parse (GtkCssParser *parser,
1251 GtkBindingSet *binding_set;
1254 array = g_ptr_array_new ();
1257 name = _gtk_css_parser_try_ident (parser, TRUE);
1260 _gtk_css_parser_error (parser, "Not a valid binding name");
1261 g_ptr_array_free (array, TRUE);
1265 binding_set = gtk_binding_set_find (name);
1269 _gtk_css_parser_error (parser, "No binding set named '%s'", name);
1274 g_ptr_array_add (array, binding_set);
1277 while (_gtk_css_parser_try (parser, ",", TRUE));
1279 g_value_take_boxed (value, array);
1285 bindings_value_print (const GValue *value,
1291 array = g_value_get_boxed (value);
1293 for (i = 0; i < array->len; i++)
1295 GtkBindingSet *binding_set = g_ptr_array_index (array, i);
1298 g_string_append (string, ", ");
1299 g_string_append (string, binding_set->set_name);
1304 border_corner_radius_value_parse (GtkCssParser *parser,
1308 GtkCssBorderCornerRadius corner;
1310 if (!_gtk_css_parser_try_double (parser, &corner.horizontal))
1312 _gtk_css_parser_error (parser, "Expected a number");
1315 else if (corner.horizontal < 0)
1318 if (!_gtk_css_parser_try_double (parser, &corner.vertical))
1319 corner.vertical = corner.horizontal;
1320 else if (corner.vertical < 0)
1323 g_value_set_boxed (value, &corner);
1327 _gtk_css_parser_error (parser, "Border radius values cannot be negative");
1332 border_corner_radius_value_print (const GValue *value,
1335 GtkCssBorderCornerRadius *corner;
1337 corner = g_value_get_boxed (value);
1341 g_string_append (string, "none");
1345 string_append_double (string, corner->horizontal);
1346 if (corner->horizontal != corner->vertical)
1348 g_string_append_c (string, ' ');
1349 string_append_double (string, corner->vertical);
1354 border_radius_value_parse (GtkCssParser *parser,
1358 GtkCssBorderRadius border;
1360 if (!_gtk_css_parser_try_double (parser, &border.top_left.horizontal))
1362 _gtk_css_parser_error (parser, "Expected a number");
1365 else if (border.top_left.horizontal < 0)
1368 if (_gtk_css_parser_try_double (parser, &border.top_right.horizontal))
1370 if (border.top_right.horizontal < 0)
1372 if (_gtk_css_parser_try_double (parser, &border.bottom_right.horizontal))
1374 if (border.bottom_right.horizontal < 0)
1376 if (!_gtk_css_parser_try_double (parser, &border.bottom_left.horizontal))
1377 border.bottom_left.horizontal = border.top_right.horizontal;
1378 else if (border.bottom_left.horizontal < 0)
1383 border.bottom_right.horizontal = border.top_left.horizontal;
1384 border.bottom_left.horizontal = border.top_right.horizontal;
1389 border.top_right.horizontal = border.top_left.horizontal;
1390 border.bottom_right.horizontal = border.top_left.horizontal;
1391 border.bottom_left.horizontal = border.top_left.horizontal;
1394 if (_gtk_css_parser_try (parser, "/", TRUE))
1396 if (!_gtk_css_parser_try_double (parser, &border.top_left.vertical))
1398 _gtk_css_parser_error (parser, "Expected a number");
1401 else if (border.top_left.vertical < 0)
1404 if (_gtk_css_parser_try_double (parser, &border.top_right.vertical))
1406 if (border.top_right.vertical < 0)
1408 if (_gtk_css_parser_try_double (parser, &border.bottom_right.vertical))
1410 if (border.bottom_right.vertical < 0)
1412 if (!_gtk_css_parser_try_double (parser, &border.bottom_left.vertical))
1413 border.bottom_left.vertical = border.top_right.vertical;
1414 else if (border.bottom_left.vertical < 0)
1419 border.bottom_right.vertical = border.top_left.vertical;
1420 border.bottom_left.vertical = border.top_right.vertical;
1425 border.top_right.vertical = border.top_left.vertical;
1426 border.bottom_right.vertical = border.top_left.vertical;
1427 border.bottom_left.vertical = border.top_left.vertical;
1432 border.top_left.vertical = border.top_left.horizontal;
1433 border.top_right.vertical = border.top_right.horizontal;
1434 border.bottom_right.vertical = border.bottom_right.horizontal;
1435 border.bottom_left.vertical = border.bottom_left.horizontal;
1438 /* border-radius is an int property for backwards-compat reasons */
1439 g_value_unset (value);
1440 g_value_init (value, GTK_TYPE_CSS_BORDER_RADIUS);
1441 g_value_set_boxed (value, &border);
1446 _gtk_css_parser_error (parser, "Border radius values cannot be negative");
1451 border_radius_value_print (const GValue *value,
1454 GtkCssBorderRadius *border;
1456 border = g_value_get_boxed (value);
1460 g_string_append (string, "none");
1464 string_append_double (string, border->top_left.horizontal);
1465 if (border->top_left.horizontal != border->top_right.horizontal ||
1466 border->top_left.horizontal != border->bottom_right.horizontal ||
1467 border->top_left.horizontal != border->bottom_left.horizontal)
1469 g_string_append_c (string, ' ');
1470 string_append_double (string, border->top_right.horizontal);
1471 if (border->top_left.horizontal != border->bottom_right.horizontal ||
1472 border->top_right.horizontal != border->bottom_left.horizontal)
1474 g_string_append_c (string, ' ');
1475 string_append_double (string, border->bottom_right.horizontal);
1476 if (border->top_right.horizontal != border->bottom_left.horizontal)
1478 g_string_append_c (string, ' ');
1479 string_append_double (string, border->bottom_left.horizontal);
1484 if (border->top_left.horizontal != border->top_left.vertical ||
1485 border->top_right.horizontal != border->top_right.vertical ||
1486 border->bottom_right.horizontal != border->bottom_right.vertical ||
1487 border->bottom_left.horizontal != border->bottom_left.vertical)
1489 g_string_append (string, " / ");
1490 string_append_double (string, border->top_left.vertical);
1491 if (border->top_left.vertical != border->top_right.vertical ||
1492 border->top_left.vertical != border->bottom_right.vertical ||
1493 border->top_left.vertical != border->bottom_left.vertical)
1495 g_string_append_c (string, ' ');
1496 string_append_double (string, border->top_right.vertical);
1497 if (border->top_left.vertical != border->bottom_right.vertical ||
1498 border->top_right.vertical != border->bottom_left.vertical)
1500 g_string_append_c (string, ' ');
1501 string_append_double (string, border->bottom_right.vertical);
1502 if (border->top_right.vertical != border->bottom_left.vertical)
1504 g_string_append_c (string, ' ');
1505 string_append_double (string, border->bottom_left.vertical);
1514 border_color_value_parse (GtkCssParser *parser,
1518 if (_gtk_css_parser_try (parser, "transparent", TRUE))
1520 GdkRGBA transparent = { 0, 0, 0, 0 };
1522 g_value_set_boxed (value, &transparent);
1527 return rgba_value_parse (parser, base, value);
1531 border_color_shorthand_value_parse (GtkCssParser *parser,
1535 GtkSymbolicColor *symbolic;
1538 array = g_ptr_array_new_with_free_func ((GDestroyNotify) gtk_symbolic_color_unref);
1542 if (_gtk_css_parser_try (parser, "transparent", TRUE))
1544 GdkRGBA transparent = { 0, 0, 0, 0 };
1546 symbolic = gtk_symbolic_color_new_literal (&transparent);
1550 symbolic = _gtk_css_parser_read_symbolic_color (parser);
1552 if (symbolic == NULL)
1556 g_ptr_array_add (array, symbolic);
1558 while (array->len < 4 &&
1559 !_gtk_css_parser_is_eof (parser) &&
1560 !_gtk_css_parser_begins_with (parser, ';') &&
1561 !_gtk_css_parser_begins_with (parser, '}'));
1566 g_assert_not_reached ();
1569 g_ptr_array_add (array, gtk_symbolic_color_ref (g_ptr_array_index (array, 0)));
1572 g_ptr_array_add (array, gtk_symbolic_color_ref (g_ptr_array_index (array, 0)));
1575 g_ptr_array_add (array, gtk_symbolic_color_ref (g_ptr_array_index (array, 1)));
1581 g_value_unset (value);
1582 g_value_init (value, G_TYPE_PTR_ARRAY);
1583 g_value_set_boxed (value, array);
1591 unpack_border (const GValue *value,
1598 GParameter *parameter = g_new0 (GParameter, 4);
1599 GtkBorder *border = g_value_get_boxed (value);
1601 parameter[0].name = top;
1602 g_value_init (¶meter[0].value, G_TYPE_INT);
1603 g_value_set_int (¶meter[0].value, border->top);
1604 parameter[1].name = left;
1605 g_value_init (¶meter[1].value, G_TYPE_INT);
1606 g_value_set_int (¶meter[1].value, border->left);
1607 parameter[2].name = bottom;
1608 g_value_init (¶meter[2].value, G_TYPE_INT);
1609 g_value_set_int (¶meter[2].value, border->bottom);
1610 parameter[3].name = right;
1611 g_value_init (¶meter[3].value, G_TYPE_INT);
1612 g_value_set_int (¶meter[3].value, border->right);
1619 pack_border (GValue *value,
1620 GtkStyleProperties *props,
1621 GtkStateFlags state,
1630 gtk_style_properties_get (props,
1643 g_value_set_boxed (value, &border);
1647 unpack_border_width (const GValue *value,
1650 return unpack_border (value, n_params,
1651 "border-top-width", "border-left-width",
1652 "border-bottom-width", "border-right-width");
1656 pack_border_width (GValue *value,
1657 GtkStyleProperties *props,
1658 GtkStateFlags state)
1660 pack_border (value, props, state,
1661 "border-top-width", "border-left-width",
1662 "border-bottom-width", "border-right-width");
1666 unpack_padding (const GValue *value,
1669 return unpack_border (value, n_params,
1670 "padding-top", "padding-left",
1671 "padding-bottom", "padding-right");
1675 pack_padding (GValue *value,
1676 GtkStyleProperties *props,
1677 GtkStateFlags state)
1679 pack_border (value, props, state,
1680 "padding-top", "padding-left",
1681 "padding-bottom", "padding-right");
1685 unpack_margin (const GValue *value,
1688 return unpack_border (value, n_params,
1689 "margin-top", "margin-left",
1690 "margin-bottom", "margin-right");
1694 pack_margin (GValue *value,
1695 GtkStyleProperties *props,
1696 GtkStateFlags state)
1698 pack_border (value, props, state,
1699 "margin-top", "margin-left",
1700 "margin-bottom", "margin-right");
1704 unpack_border_radius (const GValue *value,
1707 GParameter *parameter = g_new0 (GParameter, 4);
1708 GtkCssBorderRadius *border;
1710 if (G_VALUE_HOLDS_BOXED (value))
1711 border = g_value_get_boxed (value);
1715 parameter[0].name = "border-top-left-radius";
1716 g_value_init (¶meter[0].value, GTK_TYPE_CSS_BORDER_CORNER_RADIUS);
1717 parameter[1].name = "border-top-right-radius";
1718 g_value_init (¶meter[1].value, GTK_TYPE_CSS_BORDER_CORNER_RADIUS);
1719 parameter[2].name = "border-bottom-right-radius";
1720 g_value_init (¶meter[2].value, GTK_TYPE_CSS_BORDER_CORNER_RADIUS);
1721 parameter[3].name = "border-bottom-left-radius";
1722 g_value_init (¶meter[3].value, GTK_TYPE_CSS_BORDER_CORNER_RADIUS);
1725 g_value_set_boxed (¶meter[0].value, &border->top_left);
1726 g_value_set_boxed (¶meter[1].value, &border->top_right);
1727 g_value_set_boxed (¶meter[2].value, &border->bottom_right);
1728 g_value_set_boxed (¶meter[3].value, &border->bottom_left);
1736 pack_border_radius (GValue *value,
1737 GtkStyleProperties *props,
1738 GtkStateFlags state)
1740 GtkCssBorderCornerRadius *top_left;
1742 /* NB: We are an int property, so we have to resolve to an int here.
1743 * So we just resolve to an int. We pick one and stick to it.
1744 * Lesson learned: Don't query border-radius shorthand, query the
1745 * real properties instead. */
1746 gtk_style_properties_get (props,
1748 "border-top-left-radius", &top_left,
1752 g_value_set_int (value, top_left->horizontal);
1758 unpack_border_color (const GValue *value,
1761 GParameter *parameter = g_new0 (GParameter, 4);
1764 type = G_VALUE_TYPE (value);
1765 if (type == G_TYPE_PTR_ARRAY)
1766 type = GTK_TYPE_SYMBOLIC_COLOR;
1768 parameter[0].name = "border-top-color";
1769 g_value_init (¶meter[0].value, type);
1770 parameter[1].name = "border-right-color";
1771 g_value_init (¶meter[1].value, type);
1772 parameter[2].name = "border-bottom-color";
1773 g_value_init (¶meter[2].value, type);
1774 parameter[3].name = "border-left-color";
1775 g_value_init (¶meter[3].value, type);
1777 if (G_VALUE_TYPE (value) == G_TYPE_PTR_ARRAY)
1779 GPtrArray *array = g_value_get_boxed (value);
1782 for (i = 0; i < 4; i++)
1783 g_value_set_boxed (¶meter[i].value, g_ptr_array_index (array, i));
1787 /* can be RGBA or symbolic color */
1788 gpointer p = g_value_get_boxed (value);
1790 g_value_set_boxed (¶meter[0].value, p);
1791 g_value_set_boxed (¶meter[1].value, p);
1792 g_value_set_boxed (¶meter[2].value, p);
1793 g_value_set_boxed (¶meter[3].value, p);
1801 pack_border_color (GValue *value,
1802 GtkStyleProperties *props,
1803 GtkStateFlags state)
1805 /* NB: We are a color property, so we have to resolve to a color here.
1806 * So we just resolve to a color. We pick one and stick to it.
1807 * Lesson learned: Don't query border-color shorthand, query the
1808 * real properties instead. */
1809 g_value_unset (value);
1810 gtk_style_properties_get_property (props, "border-top-color", state, value);
1813 /*** default values ***/
1816 border_image_width_default_value (GtkStyleProperties *props,
1817 GtkStateFlags state,
1823 border_color_default_value (GtkStyleProperties *props,
1824 GtkStateFlags state,
1827 g_value_unset (value);
1828 gtk_style_properties_get_property (props, "color", state, value);
1834 css_string_funcs_init (void)
1836 if (G_LIKELY (parse_funcs != NULL))
1839 parse_funcs = g_hash_table_new (NULL, NULL);
1840 print_funcs = g_hash_table_new (NULL, NULL);
1842 register_conversion_function (GDK_TYPE_RGBA,
1845 register_conversion_function (GDK_TYPE_COLOR,
1848 register_conversion_function (GTK_TYPE_SYMBOLIC_COLOR,
1849 symbolic_color_value_parse,
1850 symbolic_color_value_print);
1851 register_conversion_function (PANGO_TYPE_FONT_DESCRIPTION,
1852 font_description_value_parse,
1853 font_description_value_print);
1854 register_conversion_function (G_TYPE_BOOLEAN,
1855 boolean_value_parse,
1856 boolean_value_print);
1857 register_conversion_function (G_TYPE_INT,
1860 register_conversion_function (G_TYPE_UINT,
1863 register_conversion_function (G_TYPE_DOUBLE,
1865 double_value_print);
1866 register_conversion_function (G_TYPE_FLOAT,
1869 register_conversion_function (G_TYPE_STRING,
1871 string_value_print);
1872 register_conversion_function (GTK_TYPE_THEMING_ENGINE,
1873 theming_engine_value_parse,
1874 theming_engine_value_print);
1875 register_conversion_function (GTK_TYPE_ANIMATION_DESCRIPTION,
1876 animation_description_value_parse,
1877 animation_description_value_print);
1878 register_conversion_function (GTK_TYPE_BORDER,
1880 border_value_print);
1881 register_conversion_function (GTK_TYPE_GRADIENT,
1882 gradient_value_parse,
1883 gradient_value_print);
1884 register_conversion_function (CAIRO_GOBJECT_TYPE_PATTERN,
1885 pattern_value_parse,
1887 register_conversion_function (GTK_TYPE_BORDER_IMAGE,
1888 border_image_value_parse,
1890 register_conversion_function (GTK_TYPE_CSS_BORDER_IMAGE_REPEAT,
1891 border_image_repeat_value_parse,
1892 border_image_repeat_value_print);
1893 register_conversion_function (GTK_TYPE_SHADOW,
1895 shadow_value_print);
1896 register_conversion_function (G_TYPE_ENUM,
1899 register_conversion_function (G_TYPE_FLAGS,
1905 _gtk_style_property_parse_value (const GtkStyleProperty *property,
1907 GtkCssParser *parser,
1910 GtkStyleParseFunc func;
1912 g_return_val_if_fail (value != NULL, FALSE);
1913 g_return_val_if_fail (parser != NULL, FALSE);
1915 css_string_funcs_init ();
1919 if (_gtk_css_parser_try (parser, "none", TRUE))
1921 /* Insert the default value, so it has an opportunity
1922 * to override other style providers when merged
1924 g_param_value_set_default (property->pspec, value);
1927 else if (property->property_parse_func)
1929 GError *error = NULL;
1933 value_str = _gtk_css_parser_read_value (parser);
1934 if (value_str == NULL)
1937 success = (*property->property_parse_func) (value_str, value, &error);
1944 func = property->parse_func;
1950 func = g_hash_table_lookup (parse_funcs,
1951 GSIZE_TO_POINTER (G_VALUE_TYPE (value)));
1953 func = g_hash_table_lookup (parse_funcs,
1954 GSIZE_TO_POINTER (g_type_fundamental (G_VALUE_TYPE (value))));
1958 _gtk_css_parser_error (parser,
1959 "Cannot convert to type '%s'",
1960 g_type_name (G_VALUE_TYPE (value)));
1964 return (*func) (parser, base, value);
1968 _gtk_style_property_print_value (const GtkStyleProperty *property,
1969 const GValue *value,
1972 GtkStylePrintFunc func;
1974 css_string_funcs_init ();
1977 func = property->print_func;
1982 func = g_hash_table_lookup (print_funcs,
1983 GSIZE_TO_POINTER (G_VALUE_TYPE (value)));
1985 func = g_hash_table_lookup (print_funcs,
1986 GSIZE_TO_POINTER (g_type_fundamental (G_VALUE_TYPE (value))));
1990 char *s = g_strdup_value_contents (value);
1991 g_string_append (string, s);
1996 func (value, string);
2000 _gtk_style_property_default_value (const GtkStyleProperty *property,
2001 GtkStyleProperties *properties,
2002 GtkStateFlags state,
2005 if (property->default_value_func)
2006 property->default_value_func (properties, state, value);
2007 else if (property->pspec->value_type == GTK_TYPE_THEMING_ENGINE)
2008 g_value_set_object (value, gtk_theming_engine_load (NULL));
2009 else if (property->pspec->value_type == PANGO_TYPE_FONT_DESCRIPTION)
2010 g_value_take_boxed (value, pango_font_description_from_string ("Sans 10"));
2011 else if (property->pspec->value_type == GDK_TYPE_RGBA)
2014 gdk_rgba_parse (&color, "pink");
2015 g_value_set_boxed (value, &color);
2017 else if (property->pspec->value_type == GTK_TYPE_BORDER)
2019 g_value_take_boxed (value, gtk_border_new ());
2022 g_param_value_set_default (property->pspec, value);
2026 _gtk_style_property_is_inherit (const GtkStyleProperty *property)
2028 g_return_val_if_fail (property != NULL, FALSE);
2030 return property->flags & GTK_STYLE_PROPERTY_INHERIT ? TRUE : FALSE;
2034 resolve_color (GtkStyleProperties *props,
2039 /* Resolve symbolic color to GdkRGBA */
2040 if (!gtk_symbolic_color_resolve (g_value_get_boxed (value), props, &color))
2043 /* Store it back, this is where GdkRGBA caching happens */
2044 g_value_unset (value);
2045 g_value_init (value, GDK_TYPE_RGBA);
2046 g_value_set_boxed (value, &color);
2052 resolve_color_rgb (GtkStyleProperties *props,
2055 GdkColor color = { 0 };
2058 if (!gtk_symbolic_color_resolve (g_value_get_boxed (value), props, &rgba))
2061 color.red = rgba.red * 65535. + 0.5;
2062 color.green = rgba.green * 65535. + 0.5;
2063 color.blue = rgba.blue * 65535. + 0.5;
2065 g_value_unset (value);
2066 g_value_init (value, GDK_TYPE_COLOR);
2067 g_value_set_boxed (value, &color);
2073 resolve_gradient (GtkStyleProperties *props,
2076 cairo_pattern_t *gradient;
2078 if (!gtk_gradient_resolve (g_value_get_boxed (value), props, &gradient))
2081 /* Store it back, this is where cairo_pattern_t caching happens */
2082 g_value_unset (value);
2083 g_value_init (value, CAIRO_GOBJECT_TYPE_PATTERN);
2084 g_value_take_boxed (value, gradient);
2090 resolve_shadow (GtkStyleProperties *props,
2093 GtkShadow *resolved, *base;
2095 base = g_value_get_boxed (value);
2100 if (_gtk_shadow_get_resolved (base))
2103 resolved = _gtk_shadow_resolve (base, props);
2104 if (resolved == NULL)
2107 g_value_take_boxed (value, resolved);
2113 _gtk_style_property_resolve (const GtkStyleProperty *property,
2114 GtkStyleProperties *props,
2115 GtkStateFlags state,
2118 if (G_VALUE_TYPE (val) == GTK_TYPE_SYMBOLIC_COLOR)
2120 if (property->pspec->value_type == GDK_TYPE_RGBA)
2122 if (resolve_color (props, val))
2125 else if (property->pspec->value_type == GDK_TYPE_COLOR)
2127 if (resolve_color_rgb (props, val))
2131 g_value_unset (val);
2132 g_value_init (val, property->pspec->value_type);
2133 _gtk_style_property_default_value (property, props, state, val);
2135 else if (G_VALUE_TYPE (val) == GDK_TYPE_RGBA)
2137 if (g_value_get_boxed (val) == NULL)
2138 _gtk_style_property_default_value (property, props, state, val);
2140 else if (G_VALUE_TYPE (val) == GTK_TYPE_GRADIENT)
2142 g_return_if_fail (property->pspec->value_type == CAIRO_GOBJECT_TYPE_PATTERN);
2144 if (!resolve_gradient (props, val))
2146 g_value_unset (val);
2147 g_value_init (val, CAIRO_GOBJECT_TYPE_PATTERN);
2148 _gtk_style_property_default_value (property, props, state, val);
2151 else if (G_VALUE_TYPE (val) == GTK_TYPE_SHADOW)
2153 if (!resolve_shadow (props, val))
2154 _gtk_style_property_default_value (property, props, state, val);
2159 _gtk_style_property_is_shorthand (const GtkStyleProperty *property)
2161 g_return_val_if_fail (property != NULL, FALSE);
2163 return property->pack_func != NULL;
2167 _gtk_style_property_unpack (const GtkStyleProperty *property,
2168 const GValue *value,
2171 g_return_val_if_fail (property != NULL, NULL);
2172 g_return_val_if_fail (property->unpack_func != NULL, NULL);
2173 g_return_val_if_fail (value != NULL, NULL);
2174 g_return_val_if_fail (n_params != NULL, NULL);
2176 return property->unpack_func (value, n_params);
2180 _gtk_style_property_pack (const GtkStyleProperty *property,
2181 GtkStyleProperties *props,
2182 GtkStateFlags state,
2185 g_return_if_fail (property != NULL);
2186 g_return_if_fail (property->pack_func != NULL);
2187 g_return_if_fail (GTK_IS_STYLE_PROPERTIES (props));
2188 g_return_if_fail (G_IS_VALUE (value));
2190 property->pack_func (value, props, state);
2194 gtk_style_property_init (void)
2196 if (G_LIKELY (properties))
2199 /* stuff is never freed, so no need for free functions */
2200 properties = g_hash_table_new (g_str_hash, g_str_equal);
2202 /* note that gtk_style_properties_register_property() calls this function,
2203 * so make sure we're sanely inited to avoid infloops */
2205 _gtk_style_property_register (g_param_spec_boxed ("color",
2209 GTK_STYLE_PROPERTY_INHERIT,
2217 gtk_style_properties_register_property (NULL,
2218 g_param_spec_boxed ("background-color",
2223 _gtk_style_property_register (g_param_spec_boxed ("font",
2226 PANGO_TYPE_FONT_DESCRIPTION, 0),
2227 GTK_STYLE_PROPERTY_INHERIT,
2231 font_description_value_parse,
2232 font_description_value_print,
2235 _gtk_style_property_register (g_param_spec_boxed ("text-shadow",
2238 GTK_TYPE_SHADOW, 0),
2239 GTK_STYLE_PROPERTY_INHERIT,
2247 _gtk_style_property_register (g_param_spec_boxed ("icon-shadow",
2250 GTK_TYPE_SHADOW, 0),
2251 GTK_STYLE_PROPERTY_INHERIT,
2259 gtk_style_properties_register_property (NULL,
2260 g_param_spec_boxed ("box-shadow",
2263 GTK_TYPE_SHADOW, 0));
2264 gtk_style_properties_register_property (NULL,
2265 g_param_spec_int ("margin-top",
2268 0, G_MAXINT, 0, 0));
2269 gtk_style_properties_register_property (NULL,
2270 g_param_spec_int ("margin-left",
2273 0, G_MAXINT, 0, 0));
2274 gtk_style_properties_register_property (NULL,
2275 g_param_spec_int ("margin-bottom",
2278 0, G_MAXINT, 0, 0));
2279 gtk_style_properties_register_property (NULL,
2280 g_param_spec_int ("margin-right",
2283 0, G_MAXINT, 0, 0));
2284 _gtk_style_property_register (g_param_spec_boxed ("margin",
2287 GTK_TYPE_BORDER, 0),
2295 gtk_style_properties_register_property (NULL,
2296 g_param_spec_int ("padding-top",
2299 0, G_MAXINT, 0, 0));
2300 gtk_style_properties_register_property (NULL,
2301 g_param_spec_int ("padding-left",
2304 0, G_MAXINT, 0, 0));
2305 gtk_style_properties_register_property (NULL,
2306 g_param_spec_int ("padding-bottom",
2308 "Padding at bottom",
2309 0, G_MAXINT, 0, 0));
2310 gtk_style_properties_register_property (NULL,
2311 g_param_spec_int ("padding-right",
2314 0, G_MAXINT, 0, 0));
2315 _gtk_style_property_register (g_param_spec_boxed ("padding",
2318 GTK_TYPE_BORDER, 0),
2326 gtk_style_properties_register_property (NULL,
2327 g_param_spec_int ("border-top-width",
2329 "Border width at top",
2330 0, G_MAXINT, 0, 0));
2331 gtk_style_properties_register_property (NULL,
2332 g_param_spec_int ("border-left-width",
2333 "border left width",
2334 "Border width at left",
2335 0, G_MAXINT, 0, 0));
2336 gtk_style_properties_register_property (NULL,
2337 g_param_spec_int ("border-bottom-width",
2338 "border bottom width",
2339 "Border width at bottom",
2340 0, G_MAXINT, 0, 0));
2341 gtk_style_properties_register_property (NULL,
2342 g_param_spec_int ("border-right-width",
2343 "border right width",
2344 "Border width at right",
2345 0, G_MAXINT, 0, 0));
2346 _gtk_style_property_register (g_param_spec_boxed ("border-width",
2348 "Border width, in pixels",
2349 GTK_TYPE_BORDER, 0),
2352 unpack_border_width,
2358 _gtk_style_property_register (g_param_spec_boxed ("border-top-left-radius",
2359 "Border top left radius",
2360 "Border radius of top left corner, in pixels",
2361 GTK_TYPE_CSS_BORDER_CORNER_RADIUS, 0),
2366 border_corner_radius_value_parse,
2367 border_corner_radius_value_print,
2369 _gtk_style_property_register (g_param_spec_boxed ("border-top-right-radius",
2370 "Border top right radius",
2371 "Border radius of top right corner, in pixels",
2372 GTK_TYPE_CSS_BORDER_CORNER_RADIUS, 0),
2377 border_corner_radius_value_parse,
2378 border_corner_radius_value_print,
2380 _gtk_style_property_register (g_param_spec_boxed ("border-bottom-right-radius",
2381 "Border bottom right radius",
2382 "Border radius of bottom right corner, in pixels",
2383 GTK_TYPE_CSS_BORDER_CORNER_RADIUS, 0),
2388 border_corner_radius_value_parse,
2389 border_corner_radius_value_print,
2391 _gtk_style_property_register (g_param_spec_boxed ("border-bottom-left-radius",
2392 "Border bottom left radius",
2393 "Border radius of bottom left corner, in pixels",
2394 GTK_TYPE_CSS_BORDER_CORNER_RADIUS, 0),
2399 border_corner_radius_value_parse,
2400 border_corner_radius_value_print,
2402 _gtk_style_property_register (g_param_spec_int ("border-radius",
2404 "Border radius, in pixels",
2408 unpack_border_radius,
2410 border_radius_value_parse,
2411 border_radius_value_print,
2414 gtk_style_properties_register_property (NULL,
2415 g_param_spec_enum ("border-style",
2418 GTK_TYPE_BORDER_STYLE,
2419 GTK_BORDER_STYLE_NONE, 0));
2420 _gtk_style_property_register (g_param_spec_boxed ("border-top-color",
2428 border_color_value_parse,
2430 border_color_default_value);
2431 _gtk_style_property_register (g_param_spec_boxed ("border-right-color",
2432 "Border right color",
2433 "Border right color",
2439 border_color_value_parse,
2441 border_color_default_value);
2442 _gtk_style_property_register (g_param_spec_boxed ("border-bottom-color",
2443 "Border bottom color",
2444 "Border bottom color",
2450 border_color_value_parse,
2452 border_color_default_value);
2453 _gtk_style_property_register (g_param_spec_boxed ("border-left-color",
2454 "Border left color",
2455 "Border left color",
2461 border_color_value_parse,
2463 border_color_default_value);
2464 _gtk_style_property_register (g_param_spec_boxed ("border-color",
2470 unpack_border_color,
2472 border_color_shorthand_value_parse,
2476 gtk_style_properties_register_property (NULL,
2477 g_param_spec_boxed ("background-image",
2480 CAIRO_GOBJECT_TYPE_PATTERN, 0));
2481 gtk_style_properties_register_property (NULL,
2482 g_param_spec_boxed ("border-image-source",
2483 "Border image source",
2484 "Border image source",
2485 CAIRO_GOBJECT_TYPE_PATTERN, 0));
2486 gtk_style_properties_register_property (NULL,
2487 g_param_spec_boxed ("border-image-repeat",
2488 "Border image repeat",
2489 "Border image repeat",
2490 GTK_TYPE_CSS_BORDER_IMAGE_REPEAT, 0));
2491 gtk_style_properties_register_property (NULL,
2492 g_param_spec_boxed ("border-image-slice",
2493 "Border image slice",
2494 "Border image slice",
2495 GTK_TYPE_BORDER, 0));
2496 _gtk_style_property_register (g_param_spec_boxed ("border-image-width",
2497 "Border image width",
2498 "Border image width",
2499 GTK_TYPE_BORDER, 0),
2506 border_image_width_default_value);
2507 _gtk_style_property_register (g_param_spec_boxed ("border-image",
2510 GTK_TYPE_BORDER_IMAGE, 0),
2513 _gtk_border_image_unpack,
2514 _gtk_border_image_pack,
2518 gtk_style_properties_register_property (NULL,
2519 g_param_spec_object ("engine",
2522 GTK_TYPE_THEMING_ENGINE, 0));
2523 gtk_style_properties_register_property (NULL,
2524 g_param_spec_boxed ("transition",
2525 "Transition animation description",
2526 "Transition animation description",
2527 GTK_TYPE_ANIMATION_DESCRIPTION, 0));
2529 /* Private property holding the binding sets */
2530 _gtk_style_property_register (g_param_spec_boxed ("gtk-key-bindings",
2533 G_TYPE_PTR_ARRAY, 0),
2538 bindings_value_parse,
2539 bindings_value_print,
2543 const GtkStyleProperty *
2544 _gtk_style_property_lookup (const char *name)
2546 gtk_style_property_init ();
2548 return g_hash_table_lookup (properties, name);
2552 _gtk_style_property_register (GParamSpec *pspec,
2553 GtkStylePropertyFlags flags,
2554 GtkStylePropertyParser property_parse_func,
2555 GtkStyleUnpackFunc unpack_func,
2556 GtkStylePackFunc pack_func,
2557 GtkStyleParseFunc parse_func,
2558 GtkStylePrintFunc print_func,
2559 GtkStyleDefaultValueFunc default_value_func)
2561 const GtkStyleProperty *existing;
2562 GtkStyleProperty *node;
2564 g_return_if_fail ((pack_func == NULL) == (unpack_func == NULL));
2566 gtk_style_property_init ();
2568 existing = _gtk_style_property_lookup (pspec->name);
2569 if (existing != NULL)
2571 g_warning ("Property \"%s\" was already registered with type %s",
2572 pspec->name, g_type_name (existing->pspec->value_type));
2576 node = g_slice_new0 (GtkStyleProperty);
2577 node->flags = flags;
2578 node->pspec = pspec;
2579 node->property_parse_func = property_parse_func;
2580 node->pack_func = pack_func;
2581 node->unpack_func = unpack_func;
2582 node->parse_func = parse_func;
2583 node->print_func = print_func;
2584 node->default_value_func = default_value_func;
2586 g_hash_table_insert (properties, pspec->name, node);