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,
888 gdouble hoffset, voffset, blur, spread;
889 GtkSymbolicColor *color;
892 shadow = _gtk_shadow_new ();
896 inset = _gtk_css_parser_try (parser, "inset", TRUE);
898 if (!_gtk_css_parser_try_double (parser, &hoffset) ||
899 !_gtk_css_parser_try_double (parser, &voffset))
901 _gtk_css_parser_error (parser, "Horizontal and vertical offsets are required");
902 _gtk_shadow_unref (shadow);
906 if (!_gtk_css_parser_try_double (parser, &blur))
909 if (!_gtk_css_parser_try_double (parser, &spread))
912 /* XXX: the color is optional and UA-defined if it's missing,
913 * but it doesn't really make sense for us...
915 color = _gtk_css_parser_read_symbolic_color (parser);
919 _gtk_shadow_unref (shadow);
923 _gtk_shadow_append (shadow,
928 gtk_symbolic_color_unref (color);
931 while (_gtk_css_parser_try (parser, ",", TRUE));
933 g_value_take_boxed (value, shadow);
938 shadow_value_print (const GValue *value,
943 shadow = g_value_get_boxed (value);
946 g_string_append (string, "none");
948 _gtk_shadow_print (shadow, string);
952 border_image_repeat_value_parse (GtkCssParser *parser,
956 GtkCssBorderImageRepeat image_repeat;
957 GtkCssRepeatStyle styles[2];
960 for (i = 0; i < 2; i++)
962 if (_gtk_css_parser_try (parser, "stretch", TRUE))
963 styles[i] = GTK_CSS_REPEAT_STYLE_NONE;
964 else if (_gtk_css_parser_try (parser, "repeat", TRUE))
965 styles[i] = GTK_CSS_REPEAT_STYLE_REPEAT;
966 else if (_gtk_css_parser_try (parser, "round", TRUE))
967 styles[i] = GTK_CSS_REPEAT_STYLE_ROUND;
968 else if (_gtk_css_parser_try (parser, "space", TRUE))
969 styles[i] = GTK_CSS_REPEAT_STYLE_SPACE;
972 styles[1] = styles[0] = GTK_CSS_REPEAT_STYLE_NONE;
976 styles[i] = styles[0];
979 image_repeat.hrepeat = styles[0];
980 image_repeat.vrepeat = styles[1];
982 g_value_set_boxed (value, &image_repeat);
988 border_image_repeat_style_to_string (GtkCssRepeatStyle repeat)
992 case GTK_CSS_REPEAT_STYLE_NONE:
994 case GTK_CSS_REPEAT_STYLE_REPEAT:
996 case GTK_CSS_REPEAT_STYLE_ROUND:
998 case GTK_CSS_REPEAT_STYLE_SPACE:
1006 border_image_repeat_value_print (const GValue *value,
1009 GtkCssBorderImageRepeat *image_repeat;
1011 image_repeat = g_value_get_boxed (value);
1013 g_string_append_printf (string, "%s %s",
1014 border_image_repeat_style_to_string (image_repeat->hrepeat),
1015 border_image_repeat_style_to_string (image_repeat->vrepeat));
1019 border_image_value_parse (GtkCssParser *parser,
1023 GValue temp = { 0, };
1024 cairo_pattern_t *pattern = NULL;
1025 GtkGradient *gradient = NULL;
1026 GtkBorder slice, *width = NULL, *parsed_slice;
1027 GtkCssBorderImageRepeat repeat, *parsed_repeat;
1028 gboolean retval = FALSE;
1029 GtkBorderImage *image = NULL;
1031 g_value_init (&temp, CAIRO_GOBJECT_TYPE_PATTERN);
1033 if (!pattern_value_parse (parser, base, &temp))
1036 if (G_VALUE_TYPE (&temp) == GTK_TYPE_GRADIENT)
1037 gradient = g_value_dup_boxed (&temp);
1039 pattern = g_value_dup_boxed (&temp);
1041 g_value_unset (&temp);
1042 g_value_init (&temp, GTK_TYPE_BORDER);
1044 if (!border_value_parse (parser, base, &temp))
1047 parsed_slice = g_value_get_boxed (&temp);
1048 slice = *parsed_slice;
1050 if (_gtk_css_parser_try (parser, "/", TRUE))
1052 g_value_unset (&temp);
1053 g_value_init (&temp, GTK_TYPE_BORDER);
1055 if (!border_value_parse (parser, base, &temp))
1058 width = g_value_dup_boxed (&temp);
1061 g_value_unset (&temp);
1062 g_value_init (&temp, GTK_TYPE_CSS_BORDER_IMAGE_REPEAT);
1064 if (!border_image_repeat_value_parse (parser, base, &temp))
1067 parsed_repeat = g_value_get_boxed (&temp);
1068 repeat = *parsed_repeat;
1070 g_value_unset (&temp);
1072 if (gradient != NULL)
1073 image = _gtk_border_image_new_for_gradient (gradient, &slice, width, &repeat);
1074 else if (pattern != NULL)
1075 image = _gtk_border_image_new (pattern, &slice, width, &repeat);
1080 g_value_take_boxed (value, image);
1084 if (pattern != NULL)
1085 cairo_pattern_destroy (pattern);
1087 if (gradient != NULL)
1088 gtk_gradient_unref (gradient);
1091 gtk_border_free (width);
1097 enum_value_parse (GtkCssParser *parser,
1101 GEnumClass *enum_class;
1102 GEnumValue *enum_value;
1105 str = _gtk_css_parser_try_ident (parser, TRUE);
1108 _gtk_css_parser_error (parser, "Expected an identifier");
1112 enum_class = g_type_class_ref (G_VALUE_TYPE (value));
1113 enum_value = g_enum_get_value_by_nick (enum_class, str);
1116 g_value_set_enum (value, enum_value->value);
1118 _gtk_css_parser_error (parser,
1119 "Unknown value '%s' for enum type '%s'",
1120 str, g_type_name (G_VALUE_TYPE (value)));
1122 g_type_class_unref (enum_class);
1125 return enum_value != NULL;
1129 enum_value_print (const GValue *value,
1132 GEnumClass *enum_class;
1133 GEnumValue *enum_value;
1135 enum_class = g_type_class_ref (G_VALUE_TYPE (value));
1136 enum_value = g_enum_get_value (enum_class, g_value_get_enum (value));
1138 g_string_append (string, enum_value->value_nick);
1140 g_type_class_unref (enum_class);
1144 flags_value_parse (GtkCssParser *parser,
1148 GFlagsClass *flags_class;
1149 GFlagsValue *flag_value;
1153 flags_class = g_type_class_ref (G_VALUE_TYPE (value));
1156 str = _gtk_css_parser_try_ident (parser, TRUE);
1159 _gtk_css_parser_error (parser, "Expected an identifier");
1160 g_type_class_unref (flags_class);
1164 flag_value = g_flags_get_value_by_nick (flags_class, str);
1167 _gtk_css_parser_error (parser,
1168 "Unknown flag value '%s' for type '%s'",
1169 str, g_type_name (G_VALUE_TYPE (value)));
1170 /* XXX Do we want to return FALSE here? We can get
1171 * forward-compatibility for new values this way
1174 g_type_class_unref (flags_class);
1180 while (_gtk_css_parser_try (parser, ",", FALSE));
1182 g_type_class_unref (flags_class);
1184 g_value_set_enum (value, flags);
1190 flags_value_print (const GValue *value,
1193 GFlagsClass *flags_class;
1196 flags_class = g_type_class_ref (G_VALUE_TYPE (value));
1197 flags = g_value_get_flags (value);
1199 for (i = 0; i < flags_class->n_values; i++)
1201 GFlagsValue *flags_value = &flags_class->values[i];
1203 if (flags & flags_value->value)
1205 if (string->len != 0)
1206 g_string_append (string, ", ");
1208 g_string_append (string, flags_value->value_nick);
1212 g_type_class_unref (flags_class);
1216 bindings_value_parse (GtkCssParser *parser,
1221 GtkBindingSet *binding_set;
1224 array = g_ptr_array_new ();
1227 name = _gtk_css_parser_try_ident (parser, TRUE);
1230 _gtk_css_parser_error (parser, "Not a valid binding name");
1231 g_ptr_array_free (array, TRUE);
1235 binding_set = gtk_binding_set_find (name);
1239 _gtk_css_parser_error (parser, "No binding set named '%s'", name);
1244 g_ptr_array_add (array, binding_set);
1247 while (_gtk_css_parser_try (parser, ",", TRUE));
1249 g_value_take_boxed (value, array);
1255 bindings_value_print (const GValue *value,
1261 array = g_value_get_boxed (value);
1263 for (i = 0; i < array->len; i++)
1265 GtkBindingSet *binding_set = g_ptr_array_index (array, i);
1268 g_string_append (string, ", ");
1269 g_string_append (string, binding_set->set_name);
1274 border_corner_radius_value_parse (GtkCssParser *parser,
1278 GtkCssBorderCornerRadius corner;
1280 if (!_gtk_css_parser_try_double (parser, &corner.horizontal))
1282 _gtk_css_parser_error (parser, "Expected a number");
1285 else if (corner.horizontal < 0)
1288 if (!_gtk_css_parser_try_double (parser, &corner.vertical))
1289 corner.vertical = corner.horizontal;
1290 else if (corner.vertical < 0)
1293 g_value_set_boxed (value, &corner);
1297 _gtk_css_parser_error (parser, "Border radius values cannot be negative");
1302 border_corner_radius_value_print (const GValue *value,
1305 GtkCssBorderCornerRadius *corner;
1307 corner = g_value_get_boxed (value);
1311 g_string_append (string, "none");
1315 string_append_double (string, corner->horizontal);
1316 if (corner->horizontal != corner->vertical)
1318 g_string_append_c (string, ' ');
1319 string_append_double (string, corner->vertical);
1324 border_radius_value_parse (GtkCssParser *parser,
1328 GtkCssBorderRadius border;
1330 if (!_gtk_css_parser_try_double (parser, &border.top_left.horizontal))
1332 _gtk_css_parser_error (parser, "Expected a number");
1335 else if (border.top_left.horizontal < 0)
1338 if (_gtk_css_parser_try_double (parser, &border.top_right.horizontal))
1340 if (border.top_right.horizontal < 0)
1342 if (_gtk_css_parser_try_double (parser, &border.bottom_right.horizontal))
1344 if (border.bottom_right.horizontal < 0)
1346 if (!_gtk_css_parser_try_double (parser, &border.bottom_left.horizontal))
1347 border.bottom_left.horizontal = border.top_right.horizontal;
1348 else if (border.bottom_left.horizontal < 0)
1353 border.bottom_right.horizontal = border.top_left.horizontal;
1354 border.bottom_left.horizontal = border.top_right.horizontal;
1359 border.top_right.horizontal = border.top_left.horizontal;
1360 border.bottom_right.horizontal = border.top_left.horizontal;
1361 border.bottom_left.horizontal = border.top_left.horizontal;
1364 if (_gtk_css_parser_try (parser, "/", TRUE))
1366 if (!_gtk_css_parser_try_double (parser, &border.top_left.vertical))
1368 _gtk_css_parser_error (parser, "Expected a number");
1371 else if (border.top_left.vertical < 0)
1374 if (_gtk_css_parser_try_double (parser, &border.top_right.vertical))
1376 if (border.top_right.vertical < 0)
1378 if (_gtk_css_parser_try_double (parser, &border.bottom_right.vertical))
1380 if (border.bottom_right.vertical < 0)
1382 if (!_gtk_css_parser_try_double (parser, &border.bottom_left.vertical))
1383 border.bottom_left.vertical = border.top_right.vertical;
1384 else if (border.bottom_left.vertical < 0)
1389 border.bottom_right.vertical = border.top_left.vertical;
1390 border.bottom_left.vertical = border.top_right.vertical;
1395 border.top_right.vertical = border.top_left.vertical;
1396 border.bottom_right.vertical = border.top_left.vertical;
1397 border.bottom_left.vertical = border.top_left.vertical;
1402 border.top_left.vertical = border.top_left.horizontal;
1403 border.top_right.vertical = border.top_right.horizontal;
1404 border.bottom_right.vertical = border.bottom_right.horizontal;
1405 border.bottom_left.vertical = border.bottom_left.horizontal;
1408 /* border-radius is an int property for backwards-compat reasons */
1409 g_value_unset (value);
1410 g_value_init (value, GTK_TYPE_CSS_BORDER_RADIUS);
1411 g_value_set_boxed (value, &border);
1416 _gtk_css_parser_error (parser, "Border radius values cannot be negative");
1421 border_radius_value_print (const GValue *value,
1424 GtkCssBorderRadius *border;
1426 border = g_value_get_boxed (value);
1430 g_string_append (string, "none");
1434 string_append_double (string, border->top_left.horizontal);
1435 if (border->top_left.horizontal != border->top_right.horizontal ||
1436 border->top_left.horizontal != border->bottom_right.horizontal ||
1437 border->top_left.horizontal != border->bottom_left.horizontal)
1439 g_string_append_c (string, ' ');
1440 string_append_double (string, border->top_right.horizontal);
1441 if (border->top_left.horizontal != border->bottom_right.horizontal ||
1442 border->top_right.horizontal != border->bottom_left.horizontal)
1444 g_string_append_c (string, ' ');
1445 string_append_double (string, border->bottom_right.horizontal);
1446 if (border->top_right.horizontal != border->bottom_left.horizontal)
1448 g_string_append_c (string, ' ');
1449 string_append_double (string, border->bottom_left.horizontal);
1454 if (border->top_left.horizontal != border->top_left.vertical ||
1455 border->top_right.horizontal != border->top_right.vertical ||
1456 border->bottom_right.horizontal != border->bottom_right.vertical ||
1457 border->bottom_left.horizontal != border->bottom_left.vertical)
1459 g_string_append (string, " / ");
1460 string_append_double (string, border->top_left.vertical);
1461 if (border->top_left.vertical != border->top_right.vertical ||
1462 border->top_left.vertical != border->bottom_right.vertical ||
1463 border->top_left.vertical != border->bottom_left.vertical)
1465 g_string_append_c (string, ' ');
1466 string_append_double (string, border->top_right.vertical);
1467 if (border->top_left.vertical != border->bottom_right.vertical ||
1468 border->top_right.vertical != border->bottom_left.vertical)
1470 g_string_append_c (string, ' ');
1471 string_append_double (string, border->bottom_right.vertical);
1472 if (border->top_right.vertical != border->bottom_left.vertical)
1474 g_string_append_c (string, ' ');
1475 string_append_double (string, border->bottom_left.vertical);
1486 unpack_border (const GValue *value,
1493 GParameter *parameter = g_new0 (GParameter, 4);
1494 GtkBorder *border = g_value_get_boxed (value);
1496 parameter[0].name = top;
1497 g_value_init (¶meter[0].value, G_TYPE_INT);
1498 g_value_set_int (¶meter[0].value, border->top);
1499 parameter[1].name = left;
1500 g_value_init (¶meter[1].value, G_TYPE_INT);
1501 g_value_set_int (¶meter[1].value, border->left);
1502 parameter[2].name = bottom;
1503 g_value_init (¶meter[2].value, G_TYPE_INT);
1504 g_value_set_int (¶meter[2].value, border->bottom);
1505 parameter[3].name = right;
1506 g_value_init (¶meter[3].value, G_TYPE_INT);
1507 g_value_set_int (¶meter[3].value, border->right);
1514 pack_border (GValue *value,
1515 GtkStyleProperties *props,
1516 GtkStateFlags state,
1525 gtk_style_properties_get (props,
1538 g_value_set_boxed (value, &border);
1542 unpack_border_width (const GValue *value,
1545 return unpack_border (value, n_params,
1546 "border-top-width", "border-left-width",
1547 "border-bottom-width", "border-right-width");
1551 pack_border_width (GValue *value,
1552 GtkStyleProperties *props,
1553 GtkStateFlags state)
1555 pack_border (value, props, state,
1556 "border-top-width", "border-left-width",
1557 "border-bottom-width", "border-right-width");
1561 unpack_padding (const GValue *value,
1564 return unpack_border (value, n_params,
1565 "padding-top", "padding-left",
1566 "padding-bottom", "padding-right");
1570 pack_padding (GValue *value,
1571 GtkStyleProperties *props,
1572 GtkStateFlags state)
1574 pack_border (value, props, state,
1575 "padding-top", "padding-left",
1576 "padding-bottom", "padding-right");
1580 unpack_margin (const GValue *value,
1583 return unpack_border (value, n_params,
1584 "margin-top", "margin-left",
1585 "margin-bottom", "margin-right");
1589 pack_margin (GValue *value,
1590 GtkStyleProperties *props,
1591 GtkStateFlags state)
1593 pack_border (value, props, state,
1594 "margin-top", "margin-left",
1595 "margin-bottom", "margin-right");
1599 unpack_border_radius (const GValue *value,
1602 GParameter *parameter = g_new0 (GParameter, 4);
1603 GtkCssBorderRadius *border;
1605 if (G_VALUE_HOLDS_BOXED (value))
1606 border = g_value_get_boxed (value);
1610 parameter[0].name = "border-top-left-radius";
1611 g_value_init (¶meter[0].value, GTK_TYPE_CSS_BORDER_CORNER_RADIUS);
1612 parameter[1].name = "border-top-right-radius";
1613 g_value_init (¶meter[1].value, GTK_TYPE_CSS_BORDER_CORNER_RADIUS);
1614 parameter[2].name = "border-bottom-right-radius";
1615 g_value_init (¶meter[2].value, GTK_TYPE_CSS_BORDER_CORNER_RADIUS);
1616 parameter[3].name = "border-bottom-left-radius";
1617 g_value_init (¶meter[3].value, GTK_TYPE_CSS_BORDER_CORNER_RADIUS);
1620 g_value_set_boxed (¶meter[0].value, &border->top_left);
1621 g_value_set_boxed (¶meter[1].value, &border->top_right);
1622 g_value_set_boxed (¶meter[2].value, &border->bottom_right);
1623 g_value_set_boxed (¶meter[3].value, &border->bottom_left);
1631 pack_border_radius (GValue *value,
1632 GtkStyleProperties *props,
1633 GtkStateFlags state)
1635 GtkCssBorderCornerRadius *top_left;
1637 /* NB: We are an int property, so we have to resolve to an int here.
1638 * So we just resolve to an int. We pick one and stick to it.
1639 * Lesson learned: Don't query border-radius shorthand, query the
1640 * real properties instead. */
1641 gtk_style_properties_get (props,
1643 "border-top-left-radius", &top_left,
1647 g_value_set_int (value, top_left->horizontal);
1655 css_string_funcs_init (void)
1657 if (G_LIKELY (parse_funcs != NULL))
1660 parse_funcs = g_hash_table_new (NULL, NULL);
1661 print_funcs = g_hash_table_new (NULL, NULL);
1663 register_conversion_function (GDK_TYPE_RGBA,
1666 register_conversion_function (GDK_TYPE_COLOR,
1669 register_conversion_function (GTK_TYPE_SYMBOLIC_COLOR,
1670 symbolic_color_value_parse,
1671 symbolic_color_value_print);
1672 register_conversion_function (PANGO_TYPE_FONT_DESCRIPTION,
1673 font_description_value_parse,
1674 font_description_value_print);
1675 register_conversion_function (G_TYPE_BOOLEAN,
1676 boolean_value_parse,
1677 boolean_value_print);
1678 register_conversion_function (G_TYPE_INT,
1681 register_conversion_function (G_TYPE_UINT,
1684 register_conversion_function (G_TYPE_DOUBLE,
1686 double_value_print);
1687 register_conversion_function (G_TYPE_FLOAT,
1690 register_conversion_function (G_TYPE_STRING,
1692 string_value_print);
1693 register_conversion_function (GTK_TYPE_THEMING_ENGINE,
1694 theming_engine_value_parse,
1695 theming_engine_value_print);
1696 register_conversion_function (GTK_TYPE_ANIMATION_DESCRIPTION,
1697 animation_description_value_parse,
1698 animation_description_value_print);
1699 register_conversion_function (GTK_TYPE_BORDER,
1701 border_value_print);
1702 register_conversion_function (GTK_TYPE_GRADIENT,
1703 gradient_value_parse,
1704 gradient_value_print);
1705 register_conversion_function (CAIRO_GOBJECT_TYPE_PATTERN,
1706 pattern_value_parse,
1708 register_conversion_function (GTK_TYPE_BORDER_IMAGE,
1709 border_image_value_parse,
1711 register_conversion_function (GTK_TYPE_CSS_BORDER_IMAGE_REPEAT,
1712 border_image_repeat_value_parse,
1713 border_image_repeat_value_print);
1714 register_conversion_function (GTK_TYPE_SHADOW,
1716 shadow_value_print);
1717 register_conversion_function (G_TYPE_ENUM,
1720 register_conversion_function (G_TYPE_FLAGS,
1726 _gtk_style_property_parse_value (const GtkStyleProperty *property,
1728 GtkCssParser *parser,
1731 GtkStyleParseFunc func;
1733 g_return_val_if_fail (value != NULL, FALSE);
1734 g_return_val_if_fail (parser != NULL, FALSE);
1736 css_string_funcs_init ();
1740 if (_gtk_css_parser_try (parser, "none", TRUE))
1742 /* Insert the default value, so it has an opportunity
1743 * to override other style providers when merged
1745 g_param_value_set_default (property->pspec, value);
1748 else if (property->property_parse_func)
1750 GError *error = NULL;
1754 value_str = _gtk_css_parser_read_value (parser);
1755 if (value_str == NULL)
1758 success = (*property->property_parse_func) (value_str, value, &error);
1765 func = property->parse_func;
1771 func = g_hash_table_lookup (parse_funcs,
1772 GSIZE_TO_POINTER (G_VALUE_TYPE (value)));
1774 func = g_hash_table_lookup (parse_funcs,
1775 GSIZE_TO_POINTER (g_type_fundamental (G_VALUE_TYPE (value))));
1779 _gtk_css_parser_error (parser,
1780 "Cannot convert to type '%s'",
1781 g_type_name (G_VALUE_TYPE (value)));
1785 return (*func) (parser, base, value);
1789 _gtk_style_property_print_value (const GtkStyleProperty *property,
1790 const GValue *value,
1793 GtkStylePrintFunc func;
1795 css_string_funcs_init ();
1798 func = property->print_func;
1803 func = g_hash_table_lookup (print_funcs,
1804 GSIZE_TO_POINTER (G_VALUE_TYPE (value)));
1806 func = g_hash_table_lookup (print_funcs,
1807 GSIZE_TO_POINTER (g_type_fundamental (G_VALUE_TYPE (value))));
1811 char *s = g_strdup_value_contents (value);
1812 g_string_append (string, s);
1817 func (value, string);
1821 _gtk_style_property_default_value (const GtkStyleProperty *property,
1822 GtkStyleProperties *properties,
1825 if (property->default_value_func)
1826 property->default_value_func (properties, value);
1827 else if (property->pspec->value_type == GTK_TYPE_THEMING_ENGINE)
1828 g_value_set_object (value, gtk_theming_engine_load (NULL));
1829 else if (property->pspec->value_type == PANGO_TYPE_FONT_DESCRIPTION)
1830 g_value_take_boxed (value, pango_font_description_from_string ("Sans 10"));
1831 else if (property->pspec->value_type == GDK_TYPE_RGBA)
1834 gdk_rgba_parse (&color, "pink");
1835 g_value_set_boxed (value, &color);
1837 else if (property->pspec->value_type == GTK_TYPE_BORDER)
1839 g_value_take_boxed (value, gtk_border_new ());
1842 g_param_value_set_default (property->pspec, value);
1846 _gtk_style_property_is_inherit (const GtkStyleProperty *property)
1848 g_return_val_if_fail (property != NULL, FALSE);
1850 return property->flags & GTK_STYLE_PROPERTY_INHERIT ? TRUE : FALSE;
1854 resolve_color (GtkStyleProperties *props,
1859 /* Resolve symbolic color to GdkRGBA */
1860 if (!gtk_symbolic_color_resolve (g_value_get_boxed (value), props, &color))
1863 /* Store it back, this is where GdkRGBA caching happens */
1864 g_value_unset (value);
1865 g_value_init (value, GDK_TYPE_RGBA);
1866 g_value_set_boxed (value, &color);
1872 resolve_color_rgb (GtkStyleProperties *props,
1875 GdkColor color = { 0 };
1878 if (!gtk_symbolic_color_resolve (g_value_get_boxed (value), props, &rgba))
1881 color.red = rgba.red * 65535. + 0.5;
1882 color.green = rgba.green * 65535. + 0.5;
1883 color.blue = rgba.blue * 65535. + 0.5;
1885 g_value_unset (value);
1886 g_value_init (value, GDK_TYPE_COLOR);
1887 g_value_set_boxed (value, &color);
1893 resolve_gradient (GtkStyleProperties *props,
1896 cairo_pattern_t *gradient;
1898 if (!gtk_gradient_resolve (g_value_get_boxed (value), props, &gradient))
1901 /* Store it back, this is where cairo_pattern_t caching happens */
1902 g_value_unset (value);
1903 g_value_init (value, CAIRO_GOBJECT_TYPE_PATTERN);
1904 g_value_take_boxed (value, gradient);
1910 resolve_shadow (GtkStyleProperties *props,
1913 GtkShadow *resolved, *base;
1915 base = g_value_get_boxed (value);
1920 if (_gtk_shadow_get_resolved (base))
1923 resolved = _gtk_shadow_resolve (base, props);
1924 if (resolved == NULL)
1927 g_value_take_boxed (value, resolved);
1933 _gtk_style_property_resolve (const GtkStyleProperty *property,
1934 GtkStyleProperties *props,
1937 if (G_VALUE_TYPE (val) == GTK_TYPE_SYMBOLIC_COLOR)
1939 if (property->pspec->value_type == GDK_TYPE_RGBA)
1941 if (!resolve_color (props, val))
1942 _gtk_style_property_resolve (property, props, val);
1944 else if (property->pspec->value_type == GDK_TYPE_COLOR)
1946 if (!resolve_color_rgb (props, val))
1947 _gtk_style_property_resolve (property, props, val);
1950 _gtk_style_property_resolve (property, props, val);
1952 else if (G_VALUE_TYPE (val) == GTK_TYPE_GRADIENT)
1954 g_return_if_fail (property->pspec->value_type == CAIRO_GOBJECT_TYPE_PATTERN);
1956 if (!resolve_gradient (props, val))
1957 _gtk_style_property_resolve (property, props, val);
1959 else if (G_VALUE_TYPE (val) == GTK_TYPE_SHADOW)
1961 if (!resolve_shadow (props, val))
1962 _gtk_style_property_resolve (property, props, val);
1967 _gtk_style_property_is_shorthand (const GtkStyleProperty *property)
1969 g_return_val_if_fail (property != NULL, FALSE);
1971 return property->pack_func != NULL;
1975 _gtk_style_property_unpack (const GtkStyleProperty *property,
1976 const GValue *value,
1979 g_return_val_if_fail (property != NULL, NULL);
1980 g_return_val_if_fail (property->unpack_func != NULL, NULL);
1981 g_return_val_if_fail (value != NULL, NULL);
1982 g_return_val_if_fail (n_params != NULL, NULL);
1984 return property->unpack_func (value, n_params);
1988 _gtk_style_property_pack (const GtkStyleProperty *property,
1989 GtkStyleProperties *props,
1990 GtkStateFlags state,
1993 g_return_if_fail (property != NULL);
1994 g_return_if_fail (property->pack_func != NULL);
1995 g_return_if_fail (GTK_IS_STYLE_PROPERTIES (props));
1996 g_return_if_fail (G_IS_VALUE (value));
1998 property->pack_func (value, props, state);
2002 gtk_style_property_init (void)
2004 if (G_LIKELY (properties))
2007 /* stuff is never freed, so no need for free functions */
2008 properties = g_hash_table_new (g_str_hash, g_str_equal);
2010 /* note that gtk_style_properties_register_property() calls this function,
2011 * so make sure we're sanely inited to avoid infloops */
2013 _gtk_style_property_register (g_param_spec_boxed ("color",
2017 GTK_STYLE_PROPERTY_INHERIT,
2025 gtk_style_properties_register_property (NULL,
2026 g_param_spec_boxed ("background-color",
2031 _gtk_style_property_register (g_param_spec_boxed ("font",
2034 PANGO_TYPE_FONT_DESCRIPTION, 0),
2035 GTK_STYLE_PROPERTY_INHERIT,
2039 font_description_value_parse,
2040 font_description_value_print,
2043 _gtk_style_property_register (g_param_spec_boxed ("text-shadow",
2046 GTK_TYPE_SHADOW, 0),
2047 GTK_STYLE_PROPERTY_INHERIT,
2055 _gtk_style_property_register (g_param_spec_boxed ("icon-shadow",
2058 GTK_TYPE_SHADOW, 0),
2059 GTK_STYLE_PROPERTY_INHERIT,
2067 gtk_style_properties_register_property (NULL,
2068 g_param_spec_int ("margin-top",
2071 0, G_MAXINT, 0, 0));
2072 gtk_style_properties_register_property (NULL,
2073 g_param_spec_int ("margin-left",
2076 0, G_MAXINT, 0, 0));
2077 gtk_style_properties_register_property (NULL,
2078 g_param_spec_int ("margin-bottom",
2081 0, G_MAXINT, 0, 0));
2082 gtk_style_properties_register_property (NULL,
2083 g_param_spec_int ("margin-right",
2086 0, G_MAXINT, 0, 0));
2087 _gtk_style_property_register (g_param_spec_boxed ("margin",
2090 GTK_TYPE_BORDER, 0),
2098 gtk_style_properties_register_property (NULL,
2099 g_param_spec_int ("padding-top",
2102 0, G_MAXINT, 0, 0));
2103 gtk_style_properties_register_property (NULL,
2104 g_param_spec_int ("padding-left",
2107 0, G_MAXINT, 0, 0));
2108 gtk_style_properties_register_property (NULL,
2109 g_param_spec_int ("padding-bottom",
2111 "Padding at bottom",
2112 0, G_MAXINT, 0, 0));
2113 gtk_style_properties_register_property (NULL,
2114 g_param_spec_int ("padding-right",
2117 0, G_MAXINT, 0, 0));
2118 _gtk_style_property_register (g_param_spec_boxed ("padding",
2121 GTK_TYPE_BORDER, 0),
2129 gtk_style_properties_register_property (NULL,
2130 g_param_spec_int ("border-top-width",
2132 "Border width at top",
2133 0, G_MAXINT, 0, 0));
2134 gtk_style_properties_register_property (NULL,
2135 g_param_spec_int ("border-left-width",
2136 "border left width",
2137 "Border width at left",
2138 0, G_MAXINT, 0, 0));
2139 gtk_style_properties_register_property (NULL,
2140 g_param_spec_int ("border-bottom-width",
2141 "border bottom width",
2142 "Border width at bottom",
2143 0, G_MAXINT, 0, 0));
2144 gtk_style_properties_register_property (NULL,
2145 g_param_spec_int ("border-right-width",
2146 "border right width",
2147 "Border width at right",
2148 0, G_MAXINT, 0, 0));
2149 _gtk_style_property_register (g_param_spec_boxed ("border-width",
2151 "Border width, in pixels",
2152 GTK_TYPE_BORDER, 0),
2155 unpack_border_width,
2161 _gtk_style_property_register (g_param_spec_boxed ("border-top-left-radius",
2162 "Border top left radius",
2163 "Border radius of top left corner, in pixels",
2164 GTK_TYPE_CSS_BORDER_CORNER_RADIUS, 0),
2169 border_corner_radius_value_parse,
2170 border_corner_radius_value_print,
2172 _gtk_style_property_register (g_param_spec_boxed ("border-top-right-radius",
2173 "Border top right radius",
2174 "Border radius of top right corner, in pixels",
2175 GTK_TYPE_CSS_BORDER_CORNER_RADIUS, 0),
2180 border_corner_radius_value_parse,
2181 border_corner_radius_value_print,
2183 _gtk_style_property_register (g_param_spec_boxed ("border-bottom-right-radius",
2184 "Border bottom right radius",
2185 "Border radius of bottom right corner, in pixels",
2186 GTK_TYPE_CSS_BORDER_CORNER_RADIUS, 0),
2191 border_corner_radius_value_parse,
2192 border_corner_radius_value_print,
2194 _gtk_style_property_register (g_param_spec_boxed ("border-bottom-left-radius",
2195 "Border bottom left radius",
2196 "Border radius of bottom left corner, in pixels",
2197 GTK_TYPE_CSS_BORDER_CORNER_RADIUS, 0),
2202 border_corner_radius_value_parse,
2203 border_corner_radius_value_print,
2205 _gtk_style_property_register (g_param_spec_int ("border-radius",
2207 "Border radius, in pixels",
2211 unpack_border_radius,
2213 border_radius_value_parse,
2214 border_radius_value_print,
2217 gtk_style_properties_register_property (NULL,
2218 g_param_spec_enum ("border-style",
2221 GTK_TYPE_BORDER_STYLE,
2222 GTK_BORDER_STYLE_NONE, 0));
2223 gtk_style_properties_register_property (NULL,
2224 g_param_spec_boxed ("border-color",
2228 gtk_style_properties_register_property (NULL,
2229 g_param_spec_boxed ("background-image",
2232 CAIRO_GOBJECT_TYPE_PATTERN, 0));
2233 gtk_style_properties_register_property (NULL,
2234 g_param_spec_boxed ("border-image-source",
2235 "Border image source",
2236 "Border image source",
2237 CAIRO_GOBJECT_TYPE_PATTERN, 0));
2238 gtk_style_properties_register_property (NULL,
2239 g_param_spec_boxed ("border-image-repeat",
2240 "Border image repeat",
2241 "Border image repeat",
2242 GTK_TYPE_CSS_BORDER_IMAGE_REPEAT, 0));
2243 gtk_style_properties_register_property (NULL,
2244 g_param_spec_boxed ("border-image-slice",
2245 "Border image slice",
2246 "Border image slice",
2247 GTK_TYPE_BORDER, 0));
2248 gtk_style_properties_register_property (NULL,
2249 g_param_spec_boxed ("border-image-width",
2250 "Border image width",
2251 "Border image width",
2252 GTK_TYPE_BORDER, 0));
2253 _gtk_style_property_register (g_param_spec_boxed ("border-image",
2256 GTK_TYPE_BORDER_IMAGE, 0),
2259 _gtk_border_image_unpack,
2260 _gtk_border_image_pack,
2264 gtk_style_properties_register_property (NULL,
2265 g_param_spec_object ("engine",
2268 GTK_TYPE_THEMING_ENGINE, 0));
2269 gtk_style_properties_register_property (NULL,
2270 g_param_spec_boxed ("transition",
2271 "Transition animation description",
2272 "Transition animation description",
2273 GTK_TYPE_ANIMATION_DESCRIPTION, 0));
2275 /* Private property holding the binding sets */
2276 _gtk_style_property_register (g_param_spec_boxed ("gtk-key-bindings",
2279 G_TYPE_PTR_ARRAY, 0),
2284 bindings_value_parse,
2285 bindings_value_print,
2289 const GtkStyleProperty *
2290 _gtk_style_property_lookup (const char *name)
2292 gtk_style_property_init ();
2294 return g_hash_table_lookup (properties, name);
2298 _gtk_style_property_register (GParamSpec *pspec,
2299 GtkStylePropertyFlags flags,
2300 GtkStylePropertyParser property_parse_func,
2301 GtkStyleUnpackFunc unpack_func,
2302 GtkStylePackFunc pack_func,
2303 GtkStyleParseFunc parse_func,
2304 GtkStylePrintFunc print_func,
2305 GtkStyleDefaultValueFunc default_value_func)
2307 const GtkStyleProperty *existing;
2308 GtkStyleProperty *node;
2310 g_return_if_fail ((pack_func == NULL) == (unpack_func == NULL));
2312 gtk_style_property_init ();
2314 existing = _gtk_style_property_lookup (pspec->name);
2315 if (existing != NULL)
2317 g_warning ("Property \"%s\" was already registered with type %s",
2318 pspec->name, g_type_name (existing->pspec->value_type));
2322 node = g_slice_new0 (GtkStyleProperty);
2323 node->flags = flags;
2324 node->pspec = pspec;
2325 node->property_parse_func = property_parse_func;
2326 node->pack_func = pack_func;
2327 node->unpack_func = unpack_func;
2328 node->parse_func = parse_func;
2329 node->print_func = print_func;
2330 node->default_value_func = default_value_func;
2332 g_hash_table_insert (properties, pspec->name, node);