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 "gtk9slice.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 slice_value_parse (GtkCssParser *parser,
956 gdouble distance_top, distance_bottom;
957 gdouble distance_left, distance_right;
958 GtkSliceSideModifier mods[2];
962 GError *error = NULL;
966 /* Parse image url */
967 file = gtk_css_parse_url (parser, base);
971 if (!_gtk_css_parser_try_double (parser, &distance_top) ||
972 !_gtk_css_parser_try_double (parser, &distance_right) ||
973 !_gtk_css_parser_try_double (parser, &distance_bottom) ||
974 !_gtk_css_parser_try_double (parser, &distance_left))
976 _gtk_css_parser_error (parser, "Expected a number");
977 g_object_unref (file);
981 for (i = 0; i < 2; i++)
983 if (_gtk_css_parser_try (parser, "stretch", TRUE))
984 mods[i] = GTK_SLICE_STRETCH;
985 else if (_gtk_css_parser_try (parser, "repeat", TRUE))
986 mods[i] = GTK_SLICE_REPEAT;
989 mods[1] = mods[0] = GTK_SLICE_STRETCH;
996 path = g_file_get_path (file);
997 g_object_unref (file);
998 pixbuf = gdk_pixbuf_new_from_file (path, &error);
1002 _gtk_css_parser_take_error (parser, error);
1006 slice = _gtk_9slice_new (pixbuf,
1007 distance_top, distance_bottom,
1008 distance_left, distance_right,
1010 g_object_unref (pixbuf);
1012 g_value_take_boxed (value, slice);
1017 enum_value_parse (GtkCssParser *parser,
1021 GEnumClass *enum_class;
1022 GEnumValue *enum_value;
1025 str = _gtk_css_parser_try_ident (parser, TRUE);
1028 _gtk_css_parser_error (parser, "Expected an identifier");
1032 enum_class = g_type_class_ref (G_VALUE_TYPE (value));
1033 enum_value = g_enum_get_value_by_nick (enum_class, str);
1036 g_value_set_enum (value, enum_value->value);
1038 _gtk_css_parser_error (parser,
1039 "Unknown value '%s' for enum type '%s'",
1040 str, g_type_name (G_VALUE_TYPE (value)));
1042 g_type_class_unref (enum_class);
1045 return enum_value != NULL;
1049 enum_value_print (const GValue *value,
1052 GEnumClass *enum_class;
1053 GEnumValue *enum_value;
1055 enum_class = g_type_class_ref (G_VALUE_TYPE (value));
1056 enum_value = g_enum_get_value (enum_class, g_value_get_enum (value));
1058 g_string_append (string, enum_value->value_nick);
1060 g_type_class_unref (enum_class);
1064 flags_value_parse (GtkCssParser *parser,
1068 GFlagsClass *flags_class;
1069 GFlagsValue *flag_value;
1073 flags_class = g_type_class_ref (G_VALUE_TYPE (value));
1076 str = _gtk_css_parser_try_ident (parser, TRUE);
1079 _gtk_css_parser_error (parser, "Expected an identifier");
1080 g_type_class_unref (flags_class);
1084 flag_value = g_flags_get_value_by_nick (flags_class, str);
1087 _gtk_css_parser_error (parser,
1088 "Unknown flag value '%s' for type '%s'",
1089 str, g_type_name (G_VALUE_TYPE (value)));
1090 /* XXX Do we want to return FALSE here? We can get
1091 * forward-compatibility for new values this way
1094 g_type_class_unref (flags_class);
1100 while (_gtk_css_parser_try (parser, ",", FALSE));
1102 g_type_class_unref (flags_class);
1104 g_value_set_enum (value, flags);
1110 flags_value_print (const GValue *value,
1113 GFlagsClass *flags_class;
1116 flags_class = g_type_class_ref (G_VALUE_TYPE (value));
1117 flags = g_value_get_flags (value);
1119 for (i = 0; i < flags_class->n_values; i++)
1121 GFlagsValue *flags_value = &flags_class->values[i];
1123 if (flags & flags_value->value)
1125 if (string->len != 0)
1126 g_string_append (string, ", ");
1128 g_string_append (string, flags_value->value_nick);
1132 g_type_class_unref (flags_class);
1136 bindings_value_parse (GtkCssParser *parser,
1141 GtkBindingSet *binding_set;
1144 array = g_ptr_array_new ();
1147 name = _gtk_css_parser_try_ident (parser, TRUE);
1150 _gtk_css_parser_error (parser, "Not a valid binding name");
1151 g_ptr_array_free (array, TRUE);
1155 binding_set = gtk_binding_set_find (name);
1159 _gtk_css_parser_error (parser, "No binding set named '%s'", name);
1164 g_ptr_array_add (array, binding_set);
1167 while (_gtk_css_parser_try (parser, ",", TRUE));
1169 g_value_take_boxed (value, array);
1175 bindings_value_print (const GValue *value,
1181 array = g_value_get_boxed (value);
1183 for (i = 0; i < array->len; i++)
1185 GtkBindingSet *binding_set = g_ptr_array_index (array, i);
1188 g_string_append (string, ", ");
1189 g_string_append (string, binding_set->set_name);
1194 border_corner_radius_value_parse (GtkCssParser *parser,
1198 GtkCssBorderCornerRadius corner;
1200 if (!_gtk_css_parser_try_double (parser, &corner.horizontal))
1202 _gtk_css_parser_error (parser, "Expected a number");
1205 else if (corner.horizontal < 0)
1208 if (!_gtk_css_parser_try_double (parser, &corner.vertical))
1209 corner.vertical = corner.horizontal;
1210 else if (corner.vertical < 0)
1213 g_value_set_boxed (value, &corner);
1217 _gtk_css_parser_error (parser, "Border radius values cannot be negative");
1222 border_corner_radius_value_print (const GValue *value,
1225 GtkCssBorderCornerRadius *corner;
1227 corner = g_value_get_boxed (value);
1231 g_string_append (string, "none");
1235 string_append_double (string, corner->horizontal);
1236 if (corner->horizontal != corner->vertical)
1238 g_string_append_c (string, ' ');
1239 string_append_double (string, corner->vertical);
1244 border_radius_value_parse (GtkCssParser *parser,
1248 GtkCssBorderRadius border;
1250 if (!_gtk_css_parser_try_double (parser, &border.top_left.horizontal))
1252 _gtk_css_parser_error (parser, "Expected a number");
1255 else if (border.top_left.horizontal < 0)
1258 if (_gtk_css_parser_try_double (parser, &border.top_right.horizontal))
1260 if (border.top_right.horizontal < 0)
1262 if (_gtk_css_parser_try_double (parser, &border.bottom_right.horizontal))
1264 if (border.bottom_right.horizontal < 0)
1266 if (!_gtk_css_parser_try_double (parser, &border.bottom_left.horizontal))
1267 border.bottom_left.horizontal = border.top_right.horizontal;
1268 else if (border.bottom_left.horizontal < 0)
1273 border.bottom_right.horizontal = border.top_left.horizontal;
1274 border.bottom_left.horizontal = border.top_right.horizontal;
1279 border.top_right.horizontal = border.top_left.horizontal;
1280 border.bottom_right.horizontal = border.top_left.horizontal;
1281 border.bottom_left.horizontal = border.top_left.horizontal;
1284 if (_gtk_css_parser_try (parser, "/", TRUE))
1286 if (!_gtk_css_parser_try_double (parser, &border.top_left.vertical))
1288 _gtk_css_parser_error (parser, "Expected a number");
1291 else if (border.top_left.vertical < 0)
1294 if (_gtk_css_parser_try_double (parser, &border.top_right.vertical))
1296 if (border.top_right.vertical < 0)
1298 if (_gtk_css_parser_try_double (parser, &border.bottom_right.vertical))
1300 if (border.bottom_right.vertical < 0)
1302 if (!_gtk_css_parser_try_double (parser, &border.bottom_left.vertical))
1303 border.bottom_left.vertical = border.top_right.vertical;
1304 else if (border.bottom_left.vertical < 0)
1309 border.bottom_right.vertical = border.top_left.vertical;
1310 border.bottom_left.vertical = border.top_right.vertical;
1315 border.top_right.vertical = border.top_left.vertical;
1316 border.bottom_right.vertical = border.top_left.vertical;
1317 border.bottom_left.vertical = border.top_left.vertical;
1322 border.top_left.vertical = border.top_left.horizontal;
1323 border.top_right.vertical = border.top_right.horizontal;
1324 border.bottom_right.vertical = border.bottom_right.horizontal;
1325 border.bottom_left.vertical = border.bottom_left.horizontal;
1328 /* border-radius is an int property for backwards-compat reasons */
1329 g_value_unset (value);
1330 g_value_init (value, GTK_TYPE_CSS_BORDER_RADIUS);
1331 g_value_set_boxed (value, &border);
1336 _gtk_css_parser_error (parser, "Border radius values cannot be negative");
1341 border_radius_value_print (const GValue *value,
1344 GtkCssBorderRadius *border;
1346 border = g_value_get_boxed (value);
1350 g_string_append (string, "none");
1354 string_append_double (string, border->top_left.horizontal);
1355 if (border->top_left.horizontal != border->top_right.horizontal ||
1356 border->top_left.horizontal != border->bottom_right.horizontal ||
1357 border->top_left.horizontal != border->bottom_left.horizontal)
1359 g_string_append_c (string, ' ');
1360 string_append_double (string, border->top_right.horizontal);
1361 if (border->top_left.horizontal != border->bottom_right.horizontal ||
1362 border->top_right.horizontal != border->bottom_left.horizontal)
1364 g_string_append_c (string, ' ');
1365 string_append_double (string, border->bottom_right.horizontal);
1366 if (border->top_right.horizontal != border->bottom_left.horizontal)
1368 g_string_append_c (string, ' ');
1369 string_append_double (string, border->bottom_left.horizontal);
1374 if (border->top_left.horizontal != border->top_left.vertical ||
1375 border->top_right.horizontal != border->top_right.vertical ||
1376 border->bottom_right.horizontal != border->bottom_right.vertical ||
1377 border->bottom_left.horizontal != border->bottom_left.vertical)
1379 g_string_append (string, " / ");
1380 string_append_double (string, border->top_left.vertical);
1381 if (border->top_left.vertical != border->top_right.vertical ||
1382 border->top_left.vertical != border->bottom_right.vertical ||
1383 border->top_left.vertical != border->bottom_left.vertical)
1385 g_string_append_c (string, ' ');
1386 string_append_double (string, border->top_right.vertical);
1387 if (border->top_left.vertical != border->bottom_right.vertical ||
1388 border->top_right.vertical != border->bottom_left.vertical)
1390 g_string_append_c (string, ' ');
1391 string_append_double (string, border->bottom_right.vertical);
1392 if (border->top_right.vertical != border->bottom_left.vertical)
1394 g_string_append_c (string, ' ');
1395 string_append_double (string, border->bottom_left.vertical);
1406 unpack_border (const GValue *value,
1413 GParameter *parameter = g_new0 (GParameter, 4);
1414 GtkBorder *border = g_value_get_boxed (value);
1416 parameter[0].name = top;
1417 g_value_init (¶meter[0].value, G_TYPE_INT);
1418 g_value_set_int (¶meter[0].value, border->top);
1419 parameter[1].name = left;
1420 g_value_init (¶meter[1].value, G_TYPE_INT);
1421 g_value_set_int (¶meter[1].value, border->left);
1422 parameter[2].name = bottom;
1423 g_value_init (¶meter[2].value, G_TYPE_INT);
1424 g_value_set_int (¶meter[2].value, border->bottom);
1425 parameter[3].name = right;
1426 g_value_init (¶meter[3].value, G_TYPE_INT);
1427 g_value_set_int (¶meter[3].value, border->right);
1434 pack_border (GValue *value,
1435 GtkStyleProperties *props,
1436 GtkStateFlags state,
1445 gtk_style_properties_get (props,
1458 g_value_set_boxed (value, &border);
1462 unpack_border_width (const GValue *value,
1465 return unpack_border (value, n_params,
1466 "border-top-width", "border-left-width",
1467 "border-bottom-width", "border-right-width");
1471 pack_border_width (GValue *value,
1472 GtkStyleProperties *props,
1473 GtkStateFlags state)
1475 pack_border (value, props, state,
1476 "border-top-width", "border-left-width",
1477 "border-bottom-width", "border-right-width");
1481 unpack_padding (const GValue *value,
1484 return unpack_border (value, n_params,
1485 "padding-top", "padding-left",
1486 "padding-bottom", "padding-right");
1490 pack_padding (GValue *value,
1491 GtkStyleProperties *props,
1492 GtkStateFlags state)
1494 pack_border (value, props, state,
1495 "padding-top", "padding-left",
1496 "padding-bottom", "padding-right");
1500 unpack_margin (const GValue *value,
1503 return unpack_border (value, n_params,
1504 "margin-top", "margin-left",
1505 "margin-bottom", "margin-right");
1509 pack_margin (GValue *value,
1510 GtkStyleProperties *props,
1511 GtkStateFlags state)
1513 pack_border (value, props, state,
1514 "margin-top", "margin-left",
1515 "margin-bottom", "margin-right");
1519 unpack_border_radius (const GValue *value,
1522 GParameter *parameter = g_new0 (GParameter, 4);
1523 GtkCssBorderRadius *border;
1525 if (G_VALUE_HOLDS_BOXED (value))
1526 border = g_value_get_boxed (value);
1530 parameter[0].name = "border-top-left-radius";
1531 g_value_init (¶meter[0].value, GTK_TYPE_CSS_BORDER_CORNER_RADIUS);
1532 parameter[1].name = "border-top-right-radius";
1533 g_value_init (¶meter[1].value, GTK_TYPE_CSS_BORDER_CORNER_RADIUS);
1534 parameter[2].name = "border-bottom-right-radius";
1535 g_value_init (¶meter[2].value, GTK_TYPE_CSS_BORDER_CORNER_RADIUS);
1536 parameter[3].name = "border-bottom-left-radius";
1537 g_value_init (¶meter[3].value, GTK_TYPE_CSS_BORDER_CORNER_RADIUS);
1540 g_value_set_boxed (¶meter[0].value, &border->top_left);
1541 g_value_set_boxed (¶meter[1].value, &border->top_right);
1542 g_value_set_boxed (¶meter[2].value, &border->bottom_right);
1543 g_value_set_boxed (¶meter[3].value, &border->bottom_left);
1551 pack_border_radius (GValue *value,
1552 GtkStyleProperties *props,
1553 GtkStateFlags state)
1555 GtkCssBorderCornerRadius *top_left;
1557 /* NB: We are an int property, so we have to resolve to an int here.
1558 * So we just resolve to an int. We pick one and stick to it.
1559 * Lesson learned: Don't query border-radius shorthand, query the
1560 * real properties instead. */
1561 gtk_style_properties_get (props,
1563 "border-top-left-radius", &top_left,
1567 g_value_set_int (value, top_left->horizontal);
1575 css_string_funcs_init (void)
1577 if (G_LIKELY (parse_funcs != NULL))
1580 parse_funcs = g_hash_table_new (NULL, NULL);
1581 print_funcs = g_hash_table_new (NULL, NULL);
1583 register_conversion_function (GDK_TYPE_RGBA,
1586 register_conversion_function (GDK_TYPE_COLOR,
1589 register_conversion_function (GTK_TYPE_SYMBOLIC_COLOR,
1590 symbolic_color_value_parse,
1591 symbolic_color_value_print);
1592 register_conversion_function (PANGO_TYPE_FONT_DESCRIPTION,
1593 font_description_value_parse,
1594 font_description_value_print);
1595 register_conversion_function (G_TYPE_BOOLEAN,
1596 boolean_value_parse,
1597 boolean_value_print);
1598 register_conversion_function (G_TYPE_INT,
1601 register_conversion_function (G_TYPE_UINT,
1604 register_conversion_function (G_TYPE_DOUBLE,
1606 double_value_print);
1607 register_conversion_function (G_TYPE_FLOAT,
1610 register_conversion_function (G_TYPE_STRING,
1612 string_value_print);
1613 register_conversion_function (GTK_TYPE_THEMING_ENGINE,
1614 theming_engine_value_parse,
1615 theming_engine_value_print);
1616 register_conversion_function (GTK_TYPE_ANIMATION_DESCRIPTION,
1617 animation_description_value_parse,
1618 animation_description_value_print);
1619 register_conversion_function (GTK_TYPE_BORDER,
1621 border_value_print);
1622 register_conversion_function (GTK_TYPE_GRADIENT,
1623 gradient_value_parse,
1624 gradient_value_print);
1625 register_conversion_function (CAIRO_GOBJECT_TYPE_PATTERN,
1626 pattern_value_parse,
1628 register_conversion_function (GTK_TYPE_9SLICE,
1631 register_conversion_function (GTK_TYPE_SHADOW,
1633 shadow_value_print);
1634 register_conversion_function (G_TYPE_ENUM,
1637 register_conversion_function (G_TYPE_FLAGS,
1643 _gtk_style_property_parse_value (const GtkStyleProperty *property,
1645 GtkCssParser *parser,
1648 GtkStyleParseFunc func;
1650 g_return_val_if_fail (value != NULL, FALSE);
1651 g_return_val_if_fail (parser != NULL, FALSE);
1653 css_string_funcs_init ();
1657 if (_gtk_css_parser_try (parser, "none", TRUE))
1659 /* Insert the default value, so it has an opportunity
1660 * to override other style providers when merged
1662 g_param_value_set_default (property->pspec, value);
1665 else if (property->property_parse_func)
1667 GError *error = NULL;
1671 value_str = _gtk_css_parser_read_value (parser);
1672 if (value_str == NULL)
1675 success = (*property->property_parse_func) (value_str, value, &error);
1682 func = property->parse_func;
1688 func = g_hash_table_lookup (parse_funcs,
1689 GSIZE_TO_POINTER (G_VALUE_TYPE (value)));
1691 func = g_hash_table_lookup (parse_funcs,
1692 GSIZE_TO_POINTER (g_type_fundamental (G_VALUE_TYPE (value))));
1696 _gtk_css_parser_error (parser,
1697 "Cannot convert to type '%s'",
1698 g_type_name (G_VALUE_TYPE (value)));
1702 return (*func) (parser, base, value);
1706 _gtk_style_property_print_value (const GtkStyleProperty *property,
1707 const GValue *value,
1710 GtkStylePrintFunc func;
1712 css_string_funcs_init ();
1715 func = property->print_func;
1720 func = g_hash_table_lookup (print_funcs,
1721 GSIZE_TO_POINTER (G_VALUE_TYPE (value)));
1723 func = g_hash_table_lookup (print_funcs,
1724 GSIZE_TO_POINTER (g_type_fundamental (G_VALUE_TYPE (value))));
1728 char *s = g_strdup_value_contents (value);
1729 g_string_append (string, s);
1734 func (value, string);
1738 _gtk_style_property_default_value (const GtkStyleProperty *property,
1739 GtkStyleProperties *properties,
1742 if (property->default_value_func)
1743 property->default_value_func (properties, value);
1744 else if (property->pspec->value_type == GTK_TYPE_THEMING_ENGINE)
1745 g_value_set_object (value, gtk_theming_engine_load (NULL));
1746 else if (property->pspec->value_type == PANGO_TYPE_FONT_DESCRIPTION)
1747 g_value_take_boxed (value, pango_font_description_from_string ("Sans 10"));
1748 else if (property->pspec->value_type == GDK_TYPE_RGBA)
1751 gdk_rgba_parse (&color, "pink");
1752 g_value_set_boxed (value, &color);
1754 else if (property->pspec->value_type == GTK_TYPE_BORDER)
1756 g_value_take_boxed (value, gtk_border_new ());
1759 g_param_value_set_default (property->pspec, value);
1763 _gtk_style_property_is_shorthand (const GtkStyleProperty *property)
1765 g_return_val_if_fail (property != NULL, FALSE);
1767 return property->pack_func != NULL;
1771 _gtk_style_property_unpack (const GtkStyleProperty *property,
1772 const GValue *value,
1775 g_return_val_if_fail (property != NULL, NULL);
1776 g_return_val_if_fail (property->unpack_func != NULL, NULL);
1777 g_return_val_if_fail (value != NULL, NULL);
1778 g_return_val_if_fail (n_params != NULL, NULL);
1780 return property->unpack_func (value, n_params);
1784 _gtk_style_property_pack (const GtkStyleProperty *property,
1785 GtkStyleProperties *props,
1786 GtkStateFlags state,
1789 g_return_if_fail (property != NULL);
1790 g_return_if_fail (property->pack_func != NULL);
1791 g_return_if_fail (GTK_IS_STYLE_PROPERTIES (props));
1792 g_return_if_fail (G_IS_VALUE (value));
1794 property->pack_func (value, props, state);
1798 gtk_style_property_init (void)
1802 if (G_LIKELY (properties))
1805 /* stuff is never freed, so no need for free functions */
1806 properties = g_hash_table_new (g_str_hash, g_str_equal);
1808 /* note that gtk_style_properties_register_property() calls this function,
1809 * so make sure we're sanely inited to avoid infloops */
1811 pspec = g_param_spec_boxed ("color",
1815 gtk_style_param_set_inherit (pspec, TRUE);
1816 gtk_style_properties_register_property (NULL, pspec);
1818 gtk_style_properties_register_property (NULL,
1819 g_param_spec_boxed ("background-color",
1824 pspec = g_param_spec_boxed ("font",
1827 PANGO_TYPE_FONT_DESCRIPTION, 0);
1828 gtk_style_param_set_inherit (pspec, TRUE);
1829 gtk_style_properties_register_property (NULL, pspec);
1831 pspec = g_param_spec_boxed ("text-shadow",
1834 GTK_TYPE_SHADOW, 0);
1835 gtk_style_param_set_inherit (pspec, TRUE);
1836 gtk_style_properties_register_property (NULL, pspec);
1838 gtk_style_properties_register_property (NULL,
1839 g_param_spec_int ("margin-top",
1842 0, G_MAXINT, 0, 0));
1843 gtk_style_properties_register_property (NULL,
1844 g_param_spec_int ("margin-left",
1847 0, G_MAXINT, 0, 0));
1848 gtk_style_properties_register_property (NULL,
1849 g_param_spec_int ("margin-bottom",
1852 0, G_MAXINT, 0, 0));
1853 gtk_style_properties_register_property (NULL,
1854 g_param_spec_int ("margin-right",
1857 0, G_MAXINT, 0, 0));
1858 _gtk_style_property_register (g_param_spec_boxed ("margin",
1861 GTK_TYPE_BORDER, 0),
1868 gtk_style_properties_register_property (NULL,
1869 g_param_spec_int ("padding-top",
1872 0, G_MAXINT, 0, 0));
1873 gtk_style_properties_register_property (NULL,
1874 g_param_spec_int ("padding-left",
1877 0, G_MAXINT, 0, 0));
1878 gtk_style_properties_register_property (NULL,
1879 g_param_spec_int ("padding-bottom",
1881 "Padding at bottom",
1882 0, G_MAXINT, 0, 0));
1883 gtk_style_properties_register_property (NULL,
1884 g_param_spec_int ("padding-right",
1887 0, G_MAXINT, 0, 0));
1888 _gtk_style_property_register (g_param_spec_boxed ("padding",
1891 GTK_TYPE_BORDER, 0),
1898 gtk_style_properties_register_property (NULL,
1899 g_param_spec_int ("border-top-width",
1901 "Border width at top",
1902 0, G_MAXINT, 0, 0));
1903 gtk_style_properties_register_property (NULL,
1904 g_param_spec_int ("border-left-width",
1905 "border left width",
1906 "Border width at left",
1907 0, G_MAXINT, 0, 0));
1908 gtk_style_properties_register_property (NULL,
1909 g_param_spec_int ("border-bottom-width",
1910 "border bottom width",
1911 "Border width at bottom",
1912 0, G_MAXINT, 0, 0));
1913 gtk_style_properties_register_property (NULL,
1914 g_param_spec_int ("border-right-width",
1915 "border right width",
1916 "Border width at right",
1917 0, G_MAXINT, 0, 0));
1918 _gtk_style_property_register (g_param_spec_boxed ("border-width",
1920 "Border width, in pixels",
1921 GTK_TYPE_BORDER, 0),
1923 unpack_border_width,
1929 _gtk_style_property_register (g_param_spec_boxed ("border-top-left-radius",
1930 "Border top left radius",
1931 "Border radius of top left corner, in pixels",
1932 GTK_TYPE_CSS_BORDER_CORNER_RADIUS, 0),
1936 border_corner_radius_value_parse,
1937 border_corner_radius_value_print,
1939 _gtk_style_property_register (g_param_spec_boxed ("border-top-right-radius",
1940 "Border top right radius",
1941 "Border radius of top right corner, in pixels",
1942 GTK_TYPE_CSS_BORDER_CORNER_RADIUS, 0),
1946 border_corner_radius_value_parse,
1947 border_corner_radius_value_print,
1949 _gtk_style_property_register (g_param_spec_boxed ("border-bottom-right-radius",
1950 "Border bottom right radius",
1951 "Border radius of bottom right corner, in pixels",
1952 GTK_TYPE_CSS_BORDER_CORNER_RADIUS, 0),
1956 border_corner_radius_value_parse,
1957 border_corner_radius_value_print,
1959 _gtk_style_property_register (g_param_spec_boxed ("border-bottom-left-radius",
1960 "Border bottom left radius",
1961 "Border radius of bottom left corner, in pixels",
1962 GTK_TYPE_CSS_BORDER_CORNER_RADIUS, 0),
1966 border_corner_radius_value_parse,
1967 border_corner_radius_value_print,
1969 _gtk_style_property_register (g_param_spec_int ("border-radius",
1971 "Border radius, in pixels",
1974 unpack_border_radius,
1976 border_radius_value_parse,
1977 border_radius_value_print,
1980 gtk_style_properties_register_property (NULL,
1981 g_param_spec_enum ("border-style",
1984 GTK_TYPE_BORDER_STYLE,
1985 GTK_BORDER_STYLE_NONE, 0));
1986 gtk_style_properties_register_property (NULL,
1987 g_param_spec_boxed ("border-color",
1991 gtk_style_properties_register_property (NULL,
1992 g_param_spec_boxed ("background-image",
1995 CAIRO_GOBJECT_TYPE_PATTERN, 0));
1996 gtk_style_properties_register_property (NULL,
1997 g_param_spec_boxed ("border-image",
2000 GTK_TYPE_9SLICE, 0));
2001 gtk_style_properties_register_property (NULL,
2002 g_param_spec_object ("engine",
2005 GTK_TYPE_THEMING_ENGINE, 0));
2006 gtk_style_properties_register_property (NULL,
2007 g_param_spec_boxed ("transition",
2008 "Transition animation description",
2009 "Transition animation description",
2010 GTK_TYPE_ANIMATION_DESCRIPTION, 0));
2012 /* Private property holding the binding sets */
2013 _gtk_style_property_register (g_param_spec_boxed ("gtk-key-bindings",
2016 G_TYPE_PTR_ARRAY, 0),
2020 bindings_value_parse,
2021 bindings_value_print,
2025 const GtkStyleProperty *
2026 _gtk_style_property_lookup (const char *name)
2028 gtk_style_property_init ();
2030 return g_hash_table_lookup (properties, name);
2034 _gtk_style_property_register (GParamSpec *pspec,
2035 GtkStylePropertyParser property_parse_func,
2036 GtkStyleUnpackFunc unpack_func,
2037 GtkStylePackFunc pack_func,
2038 GtkStyleParseFunc parse_func,
2039 GtkStylePrintFunc print_func,
2040 GtkStyleDefaultValueFunc default_value_func)
2042 const GtkStyleProperty *existing;
2043 GtkStyleProperty *node;
2045 g_return_if_fail ((pack_func == NULL) == (unpack_func == NULL));
2047 gtk_style_property_init ();
2049 existing = _gtk_style_property_lookup (pspec->name);
2050 if (existing != NULL)
2052 g_warning ("Property \"%s\" was already registered with type %s",
2053 pspec->name, g_type_name (existing->pspec->value_type));
2057 node = g_slice_new0 (GtkStyleProperty);
2058 node->pspec = pspec;
2059 node->property_parse_func = property_parse_func;
2060 node->pack_func = pack_func;
2061 node->unpack_func = unpack_func;
2062 node->parse_func = parse_func;
2063 node->print_func = print_func;
2064 node->default_value_func = default_value_func;
2066 g_hash_table_insert (properties, pspec->name, node);