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, see <http://www.gnu.org/licenses/>.
20 #include "gtkstylepropertyprivate.h"
22 #include <gobject/gvaluecollector.h>
23 #include <gdk-pixbuf/gdk-pixbuf.h>
24 #include <cairo-gobject.h>
27 #include "gtkcssparserprivate.h"
28 #include "gtkcssstylefuncsprivate.h"
29 #include "gtkcssstylepropertyprivate.h"
30 #include "gtkcsstypesprivate.h"
32 #include "gtkprivatetypebuiltins.h"
33 #include "gtkstylepropertiesprivate.h"
35 /* this is in case round() is not provided by the compiler,
36 * such as in the case of C89 compilers, like MSVC
38 #include "fallback-c89.c"
40 /* the actual parsers we have */
41 #include "gtkanimationdescription.h"
42 #include "gtkbindings.h"
43 #include "gtkcssimageprivate.h"
44 #include "gtkgradient.h"
45 #include "gtkshadowprivate.h"
46 #include "gtksymboliccolorprivate.h"
47 #include "gtkthemingengine.h"
48 #include "gtktypebuiltins.h"
49 #include "gtkwin32themeprivate.h"
51 /*** REGISTRATION ***/
54 gtk_css_style_property_register (const char * name,
58 GtkStylePropertyFlags flags,
59 GtkCssStylePropertyParseFunc parse_value,
60 GtkCssStylePropertyPrintFunc print_value,
61 GtkCssStylePropertyComputeFunc compute_value,
62 GtkCssStylePropertyEqualFunc equal_func,
63 GtkCssValue * initial_value)
65 GtkCssStyleProperty *node;
67 g_assert (initial_value != NULL);
68 g_assert (parse_value != NULL);
70 node = g_object_new (GTK_TYPE_CSS_STYLE_PROPERTY,
71 "value-type", value_type,
72 "computed-type", computed_type,
73 "inherit", (flags & GTK_STYLE_PROPERTY_INHERIT) ? TRUE : FALSE,
74 "initial-value", initial_value,
78 node->parse_value = parse_value;
80 node->print_value = print_value;
82 node->compute_value = compute_value;
84 node->equal_func = equal_func;
86 _gtk_css_value_unref (initial_value);
92 string_append_string (GString *str,
97 g_string_append_c (str, '"');
100 len = strcspn (string, "\"\n\r\f");
101 g_string_append (str, string);
108 g_string_append (str, "\\A ");
111 g_string_append (str, "\\D ");
114 g_string_append (str, "\\C ");
117 g_string_append (str, "\\\"");
120 g_assert_not_reached ();
125 g_string_append_c (str, '"');
128 /*** IMPLEMENTATIONS ***/
131 color_parse (GtkCssStyleProperty *property,
132 GtkCssParser *parser,
135 GtkSymbolicColor *symbolic;
137 if (_gtk_css_parser_try (parser, "currentcolor", TRUE))
139 symbolic = gtk_symbolic_color_ref (_gtk_symbolic_color_get_current_color ());
143 symbolic = _gtk_css_parser_read_symbolic_color (parser);
144 if (symbolic == NULL)
148 return _gtk_css_value_new_take_symbolic_color (symbolic);
152 color_compute (GtkCssStyleProperty *property,
153 GtkStyleContext *context,
154 GtkCssValue *specified)
156 GtkSymbolicColor *symbolic = _gtk_css_value_get_symbolic_color (specified);
157 GtkCssValue *resolved;
159 if (symbolic == _gtk_symbolic_color_get_current_color ())
161 /* The computed value of the ‘currentColor’ keyword is the computed
162 * value of the ‘color’ property. If the ‘currentColor’ keyword is
163 * set on the ‘color’ property itself, it is treated as ‘color: inherit’.
165 if (g_str_equal (_gtk_style_property_get_name (GTK_STYLE_PROPERTY (property)), "color"))
167 GtkStyleContext *parent = gtk_style_context_get_parent (context);
170 return _gtk_css_value_ref (_gtk_style_context_peek_property (parent, "color"));
172 return _gtk_css_style_compute_value (context,
174 _gtk_css_style_property_get_initial_value (property));
178 return _gtk_css_value_ref (_gtk_style_context_peek_property (context, "color"));
181 else if ((resolved = _gtk_style_context_resolve_color_value (context,
188 return color_compute (property,
190 _gtk_css_style_property_get_initial_value (property));
195 font_family_parse (GtkCssStyleProperty *property,
196 GtkCssParser *parser,
202 /* We don't special case generic families. Pango should do
205 names = g_ptr_array_new ();
208 name = _gtk_css_parser_try_ident (parser, TRUE);
211 GString *string = g_string_new (name);
213 while ((name = _gtk_css_parser_try_ident (parser, TRUE)))
215 g_string_append_c (string, ' ');
216 g_string_append (string, name);
219 name = g_string_free (string, FALSE);
223 name = _gtk_css_parser_read_string (parser);
226 g_ptr_array_free (names, TRUE);
231 g_ptr_array_add (names, name);
232 } while (_gtk_css_parser_try (parser, ",", TRUE));
234 /* NULL-terminate array */
235 g_ptr_array_add (names, NULL);
236 return _gtk_css_value_new_take_strv ((char **) g_ptr_array_free (names, FALSE));
240 font_family_value_print (GtkCssStyleProperty *property,
241 const GtkCssValue *value,
244 const char **names = _gtk_css_value_get_strv (value);
246 if (names == NULL || *names == NULL)
248 g_string_append (string, "none");
252 string_append_string (string, *names);
256 g_string_append (string, ", ");
257 string_append_string (string, *names);
263 parse_pango_style (GtkCssStyleProperty *property,
264 GtkCssParser *parser,
269 if (!_gtk_css_parser_try_enum (parser, PANGO_TYPE_STYLE, &value))
271 _gtk_css_parser_error (parser, "unknown value for property");
275 return _gtk_css_value_new_from_enum (PANGO_TYPE_STYLE, value);
279 parse_pango_weight (GtkCssStyleProperty *property,
280 GtkCssParser *parser,
285 if (!_gtk_css_parser_try_enum (parser, PANGO_TYPE_WEIGHT, &value))
287 _gtk_css_parser_error (parser, "unknown value for property");
291 return _gtk_css_value_new_from_enum (PANGO_TYPE_WEIGHT, value);
295 parse_pango_variant (GtkCssStyleProperty *property,
296 GtkCssParser *parser,
301 if (!_gtk_css_parser_try_enum (parser, PANGO_TYPE_VARIANT, &value))
303 _gtk_css_parser_error (parser, "unknown value for property");
307 return _gtk_css_value_new_from_enum (PANGO_TYPE_VARIANT, value);
311 parse_border_style (GtkCssStyleProperty *property,
312 GtkCssParser *parser,
317 if (!_gtk_css_parser_try_enum (parser, GTK_TYPE_BORDER_STYLE, &value))
319 _gtk_css_parser_error (parser, "unknown value for property");
323 return _gtk_css_value_new_from_enum (GTK_TYPE_BORDER_STYLE, value);
327 parse_css_area (GtkCssStyleProperty *property,
328 GtkCssParser *parser,
333 if (!_gtk_css_parser_try_enum (parser, GTK_TYPE_CSS_AREA, &value))
335 _gtk_css_parser_error (parser, "unknown value for property");
339 return _gtk_css_value_new_from_enum (GTK_TYPE_CSS_AREA, value);
343 bindings_value_parse (GtkCssStyleProperty *property,
344 GtkCssParser *parser,
348 GtkBindingSet *binding_set;
351 array = g_ptr_array_new ();
354 name = _gtk_css_parser_try_ident (parser, TRUE);
357 _gtk_css_parser_error (parser, "Not a valid binding name");
358 g_ptr_array_free (array, TRUE);
362 binding_set = gtk_binding_set_find (name);
366 _gtk_css_parser_error (parser, "No binding set named '%s'", name);
371 g_ptr_array_add (array, binding_set);
374 while (_gtk_css_parser_try (parser, ",", TRUE));
376 return _gtk_css_value_new_take_binding_sets (array);
380 bindings_value_print (GtkCssStyleProperty *property,
381 const GtkCssValue *value,
387 array = _gtk_css_value_get_boxed (value);
389 for (i = 0; i < array->len; i++)
391 GtkBindingSet *binding_set = g_ptr_array_index (array, i);
394 g_string_append (string, ", ");
395 g_string_append (string, binding_set->set_name);
400 shadow_value_parse (GtkCssStyleProperty *property,
401 GtkCssParser *parser,
404 gboolean have_inset, have_color, have_lengths;
405 gdouble hoffset, voffset, blur, spread;
406 GtkSymbolicColor *color;
410 if (_gtk_css_parser_try (parser, "none", TRUE))
411 return _gtk_css_value_new_take_shadow (NULL);
413 shadow = _gtk_shadow_new ();
417 have_inset = have_lengths = have_color = FALSE;
419 for (i = 0; i < 3; i++)
422 _gtk_css_parser_try (parser, "inset", TRUE))
429 _gtk_css_parser_try_double (parser, &hoffset))
433 if (!_gtk_css_parser_try_double (parser, &voffset))
435 _gtk_css_parser_error (parser, "Horizontal and vertical offsets are required");
436 _gtk_shadow_unref (shadow);
440 if (!_gtk_css_parser_try_double (parser, &blur))
443 if (!_gtk_css_parser_try_double (parser, &spread))
453 /* XXX: the color is optional and UA-defined if it's missing,
454 * but it doesn't really make sense for us...
456 color = _gtk_css_parser_read_symbolic_color (parser);
460 _gtk_shadow_unref (shadow);
466 if (!have_color || !have_lengths)
468 _gtk_css_parser_error (parser, "Must specify at least color and offsets");
469 _gtk_shadow_unref (shadow);
473 _gtk_shadow_append (shadow,
478 gtk_symbolic_color_unref (color);
481 while (_gtk_css_parser_try (parser, ",", TRUE));
483 return _gtk_css_value_new_take_shadow (shadow);
487 shadow_value_print (GtkCssStyleProperty *property,
488 const GtkCssValue *value,
493 shadow = _gtk_css_value_get_shadow (value);
496 g_string_append (string, "none");
498 _gtk_shadow_print (shadow, string);
502 shadow_value_compute (GtkCssStyleProperty *property,
503 GtkStyleContext *context,
504 GtkCssValue *specified)
508 shadow = _gtk_css_value_get_shadow (specified);
510 shadow = _gtk_shadow_resolve (shadow, context);
512 return _gtk_css_value_new_take_shadow (shadow);
516 border_corner_radius_value_parse (GtkCssStyleProperty *property,
517 GtkCssParser *parser,
520 GtkCssBorderCornerRadius corner;
522 if (!_gtk_css_parser_read_number (parser,
524 GTK_CSS_POSITIVE_ONLY
525 | GTK_CSS_PARSE_PERCENT
526 | GTK_CSS_NUMBER_AS_PIXELS
527 | GTK_CSS_PARSE_LENGTH))
530 if (!_gtk_css_parser_has_number (parser))
531 corner.vertical = corner.horizontal;
532 else if (!_gtk_css_parser_read_number (parser,
534 GTK_CSS_POSITIVE_ONLY
535 | GTK_CSS_PARSE_PERCENT
536 | GTK_CSS_NUMBER_AS_PIXELS
537 | GTK_CSS_PARSE_LENGTH))
540 return _gtk_css_value_new_from_border_corner_radius (&corner);
544 border_corner_radius_value_print (GtkCssStyleProperty *property,
545 const GtkCssValue *value,
548 const GtkCssBorderCornerRadius *corner;
550 corner = _gtk_css_value_get_border_corner_radius (value);
552 _gtk_css_number_print (&corner->horizontal, string);
554 if (!_gtk_css_number_equal (&corner->horizontal, &corner->vertical))
556 g_string_append_c (string, ' ');
557 _gtk_css_number_print (&corner->vertical, string);
562 css_image_value_parse (GtkCssStyleProperty *property,
563 GtkCssParser *parser,
568 if (_gtk_css_parser_try (parser, "none", TRUE))
572 image = _gtk_css_image_new_parse (parser, base);
577 return _gtk_css_value_new_take_image (image);
581 css_image_value_print (GtkCssStyleProperty *property,
582 const GtkCssValue *value,
585 GtkCssImage *image = _gtk_css_value_get_image (value);
588 _gtk_css_image_print (image, string);
590 g_string_append (string, "none");
594 css_image_value_compute (GtkCssStyleProperty *property,
595 GtkStyleContext *context,
596 GtkCssValue *specified)
598 GtkCssImage *image, *computed;
600 image = _gtk_css_value_get_image (specified);
603 return _gtk_css_value_ref (specified);
605 computed = _gtk_css_image_compute (image, context);
607 if (computed == image)
609 g_object_unref (computed);
610 return _gtk_css_value_ref (specified);
613 return _gtk_css_value_new_take_image (computed);
617 font_size_parse (GtkCssStyleProperty *property,
618 GtkCssParser *parser,
623 if (!_gtk_css_parser_try_double (parser, &d))
625 _gtk_css_parser_error (parser, "Expected a number");
629 return _gtk_css_value_new_from_double (d);
633 outline_parse (GtkCssStyleProperty *property,
634 GtkCssParser *parser,
639 if (!_gtk_css_parser_try_int (parser, &i))
641 _gtk_css_parser_error (parser, "Expected an integer");
645 return _gtk_css_value_new_from_int (i);
649 border_image_repeat_parse (GtkCssStyleProperty *property,
650 GtkCssParser *parser,
653 GValue value = G_VALUE_INIT;
656 g_value_init (&value, GTK_TYPE_CSS_BORDER_IMAGE_REPEAT);
657 if (!_gtk_css_style_parse_value (&value, parser, base))
659 g_value_unset (&value);
663 result = _gtk_css_value_new_from_gvalue (&value);
664 g_value_unset (&value);
670 border_image_slice_parse (GtkCssStyleProperty *property,
671 GtkCssParser *parser,
674 GValue value = G_VALUE_INIT;
677 g_value_init (&value, GTK_TYPE_BORDER);
678 if (!_gtk_css_style_parse_value (&value, parser, base))
680 g_value_unset (&value);
684 result = _gtk_css_value_new_from_gvalue (&value);
685 g_value_unset (&value);
691 border_image_width_parse (GtkCssStyleProperty *property,
692 GtkCssParser *parser,
695 GValue value = G_VALUE_INIT;
698 g_value_init (&value, GTK_TYPE_BORDER);
699 if (!_gtk_css_style_parse_value (&value, parser, base))
701 g_value_unset (&value);
705 result = _gtk_css_value_new_from_gvalue (&value);
706 g_value_unset (&value);
712 engine_parse (GtkCssStyleProperty *property,
713 GtkCssParser *parser,
716 GtkThemingEngine *engine;
719 if (_gtk_css_parser_try (parser, "none", TRUE))
720 return _gtk_css_value_new_from_theming_engine (gtk_theming_engine_load (NULL));
722 str = _gtk_css_parser_try_ident (parser, TRUE);
725 _gtk_css_parser_error (parser, "Expected a valid theme name");
729 engine = gtk_theming_engine_load (str);
733 _gtk_css_parser_error (parser, "Theming engine '%s' not found", str);
740 return _gtk_css_value_new_from_theming_engine (engine);
744 transition_parse (GtkCssStyleProperty *property,
745 GtkCssParser *parser,
748 GValue value = G_VALUE_INIT;
751 g_value_init (&value, GTK_TYPE_ANIMATION_DESCRIPTION);
752 if (!_gtk_css_style_parse_value (&value, parser, base))
754 g_value_unset (&value);
758 result = _gtk_css_value_new_from_gvalue (&value);
759 g_value_unset (&value);
765 parse_margin (GtkCssStyleProperty *property,
766 GtkCssParser *parser,
771 if (!_gtk_css_parser_read_number (parser,
773 GTK_CSS_NUMBER_AS_PIXELS
774 | GTK_CSS_PARSE_LENGTH))
777 return _gtk_css_value_new_from_number (&number);
781 compute_margin (GtkCssStyleProperty *property,
782 GtkStyleContext *context,
783 GtkCssValue *specified)
787 if (_gtk_css_number_compute (&number,
788 _gtk_css_value_get_number (specified),
791 return _gtk_css_value_new_from_number (&number);
793 return _gtk_css_value_ref (specified);
797 parse_padding (GtkCssStyleProperty *property,
798 GtkCssParser *parser,
803 if (!_gtk_css_parser_read_number (parser,
805 GTK_CSS_POSITIVE_ONLY
806 | GTK_CSS_NUMBER_AS_PIXELS
807 | GTK_CSS_PARSE_LENGTH))
810 return _gtk_css_value_new_from_number (&number);
814 compute_padding (GtkCssStyleProperty *property,
815 GtkStyleContext *context,
816 GtkCssValue *specified)
820 if (_gtk_css_number_compute (&number,
821 _gtk_css_value_get_number (specified),
823 return _gtk_css_value_new_from_number (&number);
824 return _gtk_css_value_ref (specified);
828 parse_border_width (GtkCssStyleProperty *property,
829 GtkCssParser *parser,
834 if (!_gtk_css_parser_read_number (parser,
836 GTK_CSS_POSITIVE_ONLY
837 | GTK_CSS_NUMBER_AS_PIXELS
838 | GTK_CSS_PARSE_LENGTH))
841 return _gtk_css_value_new_from_number (&number);
845 compute_border_width (GtkCssStyleProperty *property,
846 GtkStyleContext *context,
847 GtkCssValue *specified)
849 GtkCssStyleProperty *style;
850 GtkBorderStyle border_style;
854 /* The -1 is magic that is only true because we register the style
855 * properties directly after the width properties.
857 style = _gtk_css_style_property_lookup_by_id (_gtk_css_style_property_get_id (property) - 1);
859 border_style = _gtk_css_value_get_border_style (_gtk_style_context_peek_property (context, _gtk_style_property_get_name (GTK_STYLE_PROPERTY (style))));
861 if (border_style == GTK_BORDER_STYLE_NONE ||
862 border_style == GTK_BORDER_STYLE_HIDDEN)
868 _gtk_css_number_compute (&number,
869 _gtk_css_value_get_number (specified),
871 value = round (number.value);
873 return _gtk_css_value_new_from_int (value);
877 background_repeat_value_parse (GtkCssStyleProperty *property,
878 GtkCssParser *parser,
881 int repeat, vertical;
883 if (!_gtk_css_parser_try_enum (parser, GTK_TYPE_CSS_BACKGROUND_REPEAT, &repeat))
885 _gtk_css_parser_error (parser, "Not a valid value");
889 if (repeat <= GTK_CSS_BACKGROUND_REPEAT_MASK)
891 if (_gtk_css_parser_try_enum (parser, GTK_TYPE_CSS_BACKGROUND_REPEAT, &vertical))
893 if (vertical >= GTK_CSS_BACKGROUND_REPEAT_MASK)
895 _gtk_css_parser_error (parser, "Not a valid 2nd value");
899 repeat |= vertical << GTK_CSS_BACKGROUND_REPEAT_SHIFT;
902 repeat |= repeat << GTK_CSS_BACKGROUND_REPEAT_SHIFT;
905 return _gtk_css_value_new_from_enum (GTK_TYPE_CSS_BACKGROUND_REPEAT, repeat);
909 background_repeat_value_print (GtkCssStyleProperty *property,
910 const GtkCssValue *value,
913 GEnumClass *enum_class;
914 GEnumValue *enum_value;
915 GtkCssBackgroundRepeat repeat;
917 repeat = _gtk_css_value_get_enum (value);
918 enum_class = g_type_class_ref (GTK_TYPE_CSS_BACKGROUND_REPEAT);
919 enum_value = g_enum_get_value (enum_class, repeat);
921 /* only triggers for 'repeat-x' and 'repeat-y' */
923 g_string_append (string, enum_value->value_nick);
926 enum_value = g_enum_get_value (enum_class, GTK_CSS_BACKGROUND_HORIZONTAL (repeat));
927 g_string_append (string, enum_value->value_nick);
929 if (GTK_CSS_BACKGROUND_HORIZONTAL (repeat) != GTK_CSS_BACKGROUND_VERTICAL (repeat))
931 enum_value = g_enum_get_value (enum_class, GTK_CSS_BACKGROUND_VERTICAL (repeat));
932 g_string_append (string, " ");
933 g_string_append (string, enum_value->value_nick);
937 g_type_class_unref (enum_class);
941 background_size_parse (GtkCssStyleProperty *property,
942 GtkCssParser *parser,
945 GtkCssBackgroundSize size = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), FALSE, FALSE};
947 if (_gtk_css_parser_try (parser, "cover", TRUE))
949 else if (_gtk_css_parser_try (parser, "contain", TRUE))
953 if (_gtk_css_parser_try (parser, "auto", TRUE))
954 _gtk_css_number_init (&size.width, 0, GTK_CSS_PX);
955 else if (!_gtk_css_parser_read_number (parser,
957 GTK_CSS_POSITIVE_ONLY
958 | GTK_CSS_PARSE_PERCENT
959 | GTK_CSS_PARSE_LENGTH))
962 if (_gtk_css_parser_try (parser, "auto", TRUE))
963 _gtk_css_number_init (&size.height, 0, GTK_CSS_PX);
964 else if (_gtk_css_parser_has_number (parser))
966 if (!_gtk_css_parser_read_number (parser,
968 GTK_CSS_POSITIVE_ONLY
969 | GTK_CSS_PARSE_PERCENT
970 | GTK_CSS_PARSE_LENGTH))
974 _gtk_css_number_init (&size.height, 0, GTK_CSS_PX);
977 return _gtk_css_value_new_from_background_size (&size);
981 background_size_print (GtkCssStyleProperty *property,
982 const GtkCssValue *value,
985 const GtkCssBackgroundSize *size = _gtk_css_value_get_background_size (value);
988 g_string_append (string, "cover");
989 else if (size->contain)
990 g_string_append (string, "contain");
993 if (size->width.value == 0)
994 g_string_append (string, "auto");
996 _gtk_css_number_print (&size->width, string);
998 if (size->height.value != 0)
1000 g_string_append (string, " ");
1001 _gtk_css_number_print (&size->height, string);
1006 static GtkCssValue *
1007 background_size_compute (GtkCssStyleProperty *property,
1008 GtkStyleContext *context,
1009 GtkCssValue *specified)
1011 const GtkCssBackgroundSize *ssize = _gtk_css_value_get_background_size (specified);
1012 GtkCssBackgroundSize csize;
1015 csize.cover = ssize->cover;
1016 csize.contain = ssize->contain;
1017 changed = _gtk_css_number_compute (&csize.width,
1020 changed |= _gtk_css_number_compute (&csize.height,
1024 return _gtk_css_value_new_from_background_size (&csize);
1025 return _gtk_css_value_ref (specified);
1028 static GtkCssValue *
1029 background_position_parse (GtkCssStyleProperty *property,
1030 GtkCssParser *parser,
1033 static const struct {
1036 gboolean horizontal;
1039 { "left", 0, TRUE, FALSE },
1040 { "right", 100, TRUE, FALSE },
1041 { "center", 50, TRUE, TRUE },
1042 { "top", 0, FALSE, TRUE },
1043 { "bottom", 100, FALSE, TRUE },
1044 { NULL , 0, TRUE, FALSE }, /* used for numbers */
1045 { NULL , 50, TRUE, TRUE } /* used for no value */
1047 GtkCssBackgroundPosition pos;
1048 GtkCssNumber *missing;
1049 guint first, second;
1051 for (first = 0; names[first].name != NULL; first++)
1053 if (_gtk_css_parser_try (parser, names[first].name, TRUE))
1055 if (names[first].horizontal)
1057 _gtk_css_number_init (&pos.x, names[first].percentage, GTK_CSS_PERCENT);
1062 _gtk_css_number_init (&pos.y, names[first].percentage, GTK_CSS_PERCENT);
1068 if (names[first].name == NULL)
1071 if (!_gtk_css_parser_read_number (parser,
1073 GTK_CSS_PARSE_PERCENT
1074 | GTK_CSS_PARSE_LENGTH))
1078 for (second = 0; names[second].name != NULL; second++)
1080 if (_gtk_css_parser_try (parser, names[second].name, TRUE))
1082 _gtk_css_number_init (missing, names[second].percentage, GTK_CSS_PERCENT);
1087 if (names[second].name == NULL)
1089 if (_gtk_css_parser_has_number (parser))
1091 if (missing != &pos.y)
1093 _gtk_css_parser_error (parser, "Invalid combination of values");
1096 if (!_gtk_css_parser_read_number (parser,
1098 GTK_CSS_PARSE_PERCENT
1099 | GTK_CSS_PARSE_LENGTH))
1105 _gtk_css_number_init (missing, 50, GTK_CSS_PERCENT);
1110 if ((names[first].horizontal && !names[second].vertical) ||
1111 (!names[first].horizontal && !names[second].horizontal))
1113 _gtk_css_parser_error (parser, "Invalid combination of values");
1118 return _gtk_css_value_new_from_background_position (&pos);
1122 background_position_print (GtkCssStyleProperty *property,
1123 const GtkCssValue *value,
1126 const GtkCssBackgroundPosition *pos = _gtk_css_value_get_background_position (value);
1127 static const GtkCssNumber center = GTK_CSS_NUMBER_INIT (50, GTK_CSS_PERCENT);
1128 static const struct {
1131 GtkCssNumber number;
1133 { "left", "top", GTK_CSS_NUMBER_INIT (0, GTK_CSS_PERCENT) },
1134 { "right", "bottom", GTK_CSS_NUMBER_INIT (100, GTK_CSS_PERCENT) }
1138 if (_gtk_css_number_equal (&pos->x, ¢er))
1140 if (_gtk_css_number_equal (&pos->y, ¢er))
1142 g_string_append (string, "center");
1148 for (i = 0; i < G_N_ELEMENTS (values); i++)
1150 if (_gtk_css_number_equal (&pos->x, &values[i].number))
1152 g_string_append (string, values[i].x_name);
1156 if (i == G_N_ELEMENTS (values))
1157 _gtk_css_number_print (&pos->x, string);
1159 if (_gtk_css_number_equal (&pos->y, ¢er))
1162 g_string_append_c (string, ' ');
1165 for (i = 0; i < G_N_ELEMENTS (values); i++)
1167 if (_gtk_css_number_equal (&pos->y, &values[i].number))
1169 g_string_append (string, values[i].y_name);
1173 if (i == G_N_ELEMENTS (values))
1175 if (_gtk_css_number_equal (&pos->x, ¢er))
1176 g_string_append (string, "center ");
1177 _gtk_css_number_print (&pos->y, string);
1181 static GtkCssValue *
1182 background_position_compute (GtkCssStyleProperty *property,
1183 GtkStyleContext *context,
1184 GtkCssValue *specified)
1186 const GtkCssBackgroundPosition *spos = _gtk_css_value_get_background_position (specified);
1187 GtkCssBackgroundPosition cpos;
1190 changed = _gtk_css_number_compute (&cpos.x,
1193 changed |= _gtk_css_number_compute (&cpos.y,
1197 return _gtk_css_value_new_from_background_position (&cpos);
1198 return _gtk_css_value_ref (specified);
1201 /*** REGISTRATION ***/
1203 static GtkSymbolicColor *
1204 gtk_symbolic_color_new_rgba (double red,
1209 GdkRGBA rgba = { red, green, blue, alpha };
1211 return gtk_symbolic_color_new_literal (&rgba);
1215 _gtk_css_style_property_init_properties (void)
1217 char *default_font_family[] = { "Sans", NULL };
1218 GtkCssNumber number;
1219 GtkCssBackgroundSize default_background_size = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), FALSE, FALSE };
1220 GtkCssBackgroundPosition default_background_position = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PERCENT), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PERCENT)};
1221 GtkCssBorderCornerRadius no_corner_radius = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX) };
1222 GtkBorder border_of_ones = { 1, 1, 1, 1 };
1223 GtkCssBorderImageRepeat border_image_repeat = { GTK_CSS_REPEAT_STYLE_STRETCH, GTK_CSS_REPEAT_STYLE_STRETCH };
1225 /* Initialize "color" and "font-size" first,
1226 * so that when computing values later they are
1227 * done first. That way, 'currentColor' and font
1228 * sizes in em can be looked up properly */
1229 gtk_css_style_property_register ("color",
1230 GTK_TYPE_SYMBOLIC_COLOR,
1233 GTK_STYLE_PROPERTY_INHERIT,
1238 _gtk_css_value_new_take_symbolic_color (
1239 gtk_symbolic_color_new_rgba (1, 1, 1, 1)));
1240 gtk_css_style_property_register ("font-size",
1244 GTK_STYLE_PROPERTY_INHERIT,
1249 _gtk_css_value_new_from_double (10.0));
1251 /* properties that aren't referenced when computing values
1253 gtk_css_style_property_register ("background-color",
1254 GTK_TYPE_SYMBOLIC_COLOR,
1262 _gtk_css_value_new_take_symbolic_color (
1263 gtk_symbolic_color_new_rgba (0, 0, 0, 0)));
1265 gtk_css_style_property_register ("font-family",
1269 GTK_STYLE_PROPERTY_INHERIT,
1271 font_family_value_print,
1274 _gtk_css_value_new_take_strv (g_strdupv (default_font_family)));
1275 gtk_css_style_property_register ("font-style",
1279 GTK_STYLE_PROPERTY_INHERIT,
1284 _gtk_css_value_new_from_enum (PANGO_TYPE_STYLE,
1285 PANGO_STYLE_NORMAL));
1286 gtk_css_style_property_register ("font-variant",
1290 GTK_STYLE_PROPERTY_INHERIT,
1291 parse_pango_variant,
1295 _gtk_css_value_new_from_enum (PANGO_TYPE_VARIANT,
1296 PANGO_VARIANT_NORMAL));
1297 /* xxx: need to parse this properly, ie parse the numbers */
1298 gtk_css_style_property_register ("font-weight",
1302 GTK_STYLE_PROPERTY_INHERIT,
1307 _gtk_css_value_new_from_enum (PANGO_TYPE_WEIGHT,
1308 PANGO_WEIGHT_NORMAL));
1310 gtk_css_style_property_register ("text-shadow",
1314 GTK_STYLE_PROPERTY_INHERIT,
1317 shadow_value_compute,
1319 _gtk_css_value_new_take_shadow (NULL));
1321 gtk_css_style_property_register ("icon-shadow",
1325 GTK_STYLE_PROPERTY_INHERIT,
1328 shadow_value_compute,
1330 _gtk_css_value_new_take_shadow (NULL));
1332 gtk_css_style_property_register ("box-shadow",
1339 shadow_value_compute,
1341 _gtk_css_value_new_take_shadow (NULL));
1343 _gtk_css_number_init (&number, 0, GTK_CSS_PX);
1344 gtk_css_style_property_register ("margin-top",
1345 GTK_TYPE_CSS_NUMBER,
1346 GTK_TYPE_CSS_NUMBER,
1353 _gtk_css_value_new_from_number (&number));
1354 gtk_css_style_property_register ("margin-left",
1355 GTK_TYPE_CSS_NUMBER,
1356 GTK_TYPE_CSS_NUMBER,
1363 _gtk_css_value_new_from_number (&number));
1364 gtk_css_style_property_register ("margin-bottom",
1365 GTK_TYPE_CSS_NUMBER,
1366 GTK_TYPE_CSS_NUMBER,
1373 _gtk_css_value_new_from_number (&number));
1374 gtk_css_style_property_register ("margin-right",
1375 GTK_TYPE_CSS_NUMBER,
1376 GTK_TYPE_CSS_NUMBER,
1383 _gtk_css_value_new_from_number (&number));
1384 gtk_css_style_property_register ("padding-top",
1385 GTK_TYPE_CSS_NUMBER,
1386 GTK_TYPE_CSS_NUMBER,
1393 _gtk_css_value_new_from_number (&number));
1394 gtk_css_style_property_register ("padding-left",
1395 GTK_TYPE_CSS_NUMBER,
1396 GTK_TYPE_CSS_NUMBER,
1403 _gtk_css_value_new_from_number (&number));
1404 gtk_css_style_property_register ("padding-bottom",
1405 GTK_TYPE_CSS_NUMBER,
1406 GTK_TYPE_CSS_NUMBER,
1413 _gtk_css_value_new_from_number (&number));
1414 gtk_css_style_property_register ("padding-right",
1415 GTK_TYPE_CSS_NUMBER,
1416 GTK_TYPE_CSS_NUMBER,
1423 _gtk_css_value_new_from_number (&number));
1424 /* IMPORTANT: compute_border_width() requires that the border-width
1425 * properties be immeditaly followed by the border-style properties
1427 gtk_css_style_property_register ("border-top-style",
1428 GTK_TYPE_BORDER_STYLE,
1429 GTK_TYPE_BORDER_STYLE,
1430 GTK_TYPE_BORDER_STYLE,
1436 _gtk_css_value_new_from_border_style (GTK_BORDER_STYLE_NONE));
1437 gtk_css_style_property_register ("border-top-width",
1438 GTK_TYPE_CSS_NUMBER,
1444 compute_border_width,
1446 _gtk_css_value_new_from_number (&number));
1447 gtk_css_style_property_register ("border-left-style",
1448 GTK_TYPE_BORDER_STYLE,
1449 GTK_TYPE_BORDER_STYLE,
1450 GTK_TYPE_BORDER_STYLE,
1456 _gtk_css_value_new_from_border_style (GTK_BORDER_STYLE_NONE));
1457 gtk_css_style_property_register ("border-left-width",
1458 GTK_TYPE_CSS_NUMBER,
1464 compute_border_width,
1466 _gtk_css_value_new_from_number (&number));
1467 gtk_css_style_property_register ("border-bottom-style",
1468 GTK_TYPE_BORDER_STYLE,
1469 GTK_TYPE_BORDER_STYLE,
1470 GTK_TYPE_BORDER_STYLE,
1476 _gtk_css_value_new_from_border_style (GTK_BORDER_STYLE_NONE));
1477 gtk_css_style_property_register ("border-bottom-width",
1478 GTK_TYPE_CSS_NUMBER,
1484 compute_border_width,
1486 _gtk_css_value_new_from_number (&number));
1487 gtk_css_style_property_register ("border-right-style",
1488 GTK_TYPE_BORDER_STYLE,
1489 GTK_TYPE_BORDER_STYLE,
1490 GTK_TYPE_BORDER_STYLE,
1496 _gtk_css_value_new_from_border_style (GTK_BORDER_STYLE_NONE));
1497 gtk_css_style_property_register ("border-right-width",
1498 GTK_TYPE_CSS_NUMBER,
1504 compute_border_width,
1506 _gtk_css_value_new_from_number (&number));
1508 gtk_css_style_property_register ("border-top-left-radius",
1509 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1510 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1511 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1513 border_corner_radius_value_parse,
1514 border_corner_radius_value_print,
1517 _gtk_css_value_new_from_border_corner_radius (&no_corner_radius));
1518 gtk_css_style_property_register ("border-top-right-radius",
1519 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1520 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1521 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1523 border_corner_radius_value_parse,
1524 border_corner_radius_value_print,
1527 _gtk_css_value_new_from_border_corner_radius (&no_corner_radius));
1528 gtk_css_style_property_register ("border-bottom-right-radius",
1529 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1530 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1531 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1533 border_corner_radius_value_parse,
1534 border_corner_radius_value_print,
1537 _gtk_css_value_new_from_border_corner_radius (&no_corner_radius));
1538 gtk_css_style_property_register ("border-bottom-left-radius",
1539 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1540 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1541 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1543 border_corner_radius_value_parse,
1544 border_corner_radius_value_print,
1547 _gtk_css_value_new_from_border_corner_radius (&no_corner_radius));
1549 gtk_css_style_property_register ("outline-style",
1550 GTK_TYPE_BORDER_STYLE,
1551 GTK_TYPE_BORDER_STYLE,
1552 GTK_TYPE_BORDER_STYLE,
1558 _gtk_css_value_new_from_border_style (GTK_BORDER_STYLE_NONE));
1559 gtk_css_style_property_register ("outline-width",
1560 GTK_TYPE_CSS_NUMBER,
1566 compute_border_width,
1568 _gtk_css_value_new_from_number (&number));
1569 gtk_css_style_property_register ("outline-offset",
1578 _gtk_css_value_new_from_int (0));
1580 gtk_css_style_property_register ("background-clip",
1589 _gtk_css_value_new_from_enum (GTK_TYPE_CSS_AREA, GTK_CSS_AREA_BORDER_BOX));
1590 gtk_css_style_property_register ("background-origin",
1599 _gtk_css_value_new_from_enum (GTK_TYPE_CSS_AREA, GTK_CSS_AREA_PADDING_BOX));
1600 gtk_css_style_property_register ("background-size",
1601 GTK_TYPE_CSS_BACKGROUND_SIZE,
1602 GTK_TYPE_CSS_BACKGROUND_SIZE,
1605 background_size_parse,
1606 background_size_print,
1607 background_size_compute,
1609 _gtk_css_value_new_from_background_size (&default_background_size));
1610 gtk_css_style_property_register ("background-position",
1611 GTK_TYPE_CSS_BACKGROUND_POSITION,
1612 GTK_TYPE_CSS_BACKGROUND_POSITION,
1615 background_position_parse,
1616 background_position_print,
1617 background_position_compute,
1619 _gtk_css_value_new_from_background_position (&default_background_position));
1621 gtk_css_style_property_register ("border-top-color",
1622 GTK_TYPE_SYMBOLIC_COLOR,
1630 _gtk_css_value_new_take_symbolic_color (
1631 gtk_symbolic_color_ref (
1632 _gtk_symbolic_color_get_current_color ())));
1633 gtk_css_style_property_register ("border-right-color",
1634 GTK_TYPE_SYMBOLIC_COLOR,
1642 _gtk_css_value_new_take_symbolic_color (
1643 gtk_symbolic_color_ref (
1644 _gtk_symbolic_color_get_current_color ())));
1645 gtk_css_style_property_register ("border-bottom-color",
1646 GTK_TYPE_SYMBOLIC_COLOR,
1654 _gtk_css_value_new_take_symbolic_color (
1655 gtk_symbolic_color_ref (
1656 _gtk_symbolic_color_get_current_color ())));
1657 gtk_css_style_property_register ("border-left-color",
1658 GTK_TYPE_SYMBOLIC_COLOR,
1666 _gtk_css_value_new_take_symbolic_color (
1667 gtk_symbolic_color_ref (
1668 _gtk_symbolic_color_get_current_color ())));
1669 gtk_css_style_property_register ("outline-color",
1670 GTK_TYPE_SYMBOLIC_COLOR,
1678 _gtk_css_value_new_take_symbolic_color (
1679 gtk_symbolic_color_ref (
1680 _gtk_symbolic_color_get_current_color ())));
1682 gtk_css_style_property_register ("background-repeat",
1683 GTK_TYPE_CSS_BACKGROUND_REPEAT,
1684 GTK_TYPE_CSS_BACKGROUND_REPEAT,
1685 GTK_TYPE_CSS_BACKGROUND_REPEAT,
1687 background_repeat_value_parse,
1688 background_repeat_value_print,
1691 _gtk_css_value_new_from_enum (GTK_TYPE_CSS_BACKGROUND_REPEAT,
1692 GTK_CSS_BACKGROUND_REPEAT |
1693 (GTK_CSS_BACKGROUND_REPEAT << GTK_CSS_BACKGROUND_REPEAT_SHIFT)));
1694 gtk_css_style_property_register ("background-image",
1697 CAIRO_GOBJECT_TYPE_PATTERN,
1699 css_image_value_parse,
1700 css_image_value_print,
1701 css_image_value_compute,
1703 _gtk_css_value_new_take_image (NULL));
1705 gtk_css_style_property_register ("border-image-source",
1708 CAIRO_GOBJECT_TYPE_PATTERN,
1710 css_image_value_parse,
1711 css_image_value_print,
1712 css_image_value_compute,
1714 _gtk_css_value_new_take_image (NULL));
1715 gtk_css_style_property_register ("border-image-repeat",
1716 GTK_TYPE_CSS_BORDER_IMAGE_REPEAT,
1717 GTK_TYPE_CSS_BORDER_IMAGE_REPEAT,
1718 GTK_TYPE_CSS_BORDER_IMAGE_REPEAT,
1720 border_image_repeat_parse,
1724 _gtk_css_value_new_from_border_image_repeat (&border_image_repeat));
1726 /* XXX: The initial value is wrong, it should be 100% */
1727 gtk_css_style_property_register ("border-image-slice",
1732 border_image_slice_parse,
1736 _gtk_css_value_new_from_boxed (GTK_TYPE_BORDER, &border_of_ones));
1737 gtk_css_style_property_register ("border-image-width",
1742 border_image_width_parse,
1746 _gtk_css_value_new_from_boxed (GTK_TYPE_BORDER, NULL));
1747 gtk_css_style_property_register ("engine",
1748 GTK_TYPE_THEMING_ENGINE,
1749 GTK_TYPE_THEMING_ENGINE,
1750 GTK_TYPE_THEMING_ENGINE,
1756 _gtk_css_value_new_from_theming_engine (gtk_theming_engine_load (NULL)));
1757 gtk_css_style_property_register ("transition",
1758 GTK_TYPE_ANIMATION_DESCRIPTION,
1759 GTK_TYPE_ANIMATION_DESCRIPTION,
1760 GTK_TYPE_ANIMATION_DESCRIPTION,
1766 _gtk_css_value_new_from_boxed (GTK_TYPE_ANIMATION_DESCRIPTION, NULL));
1768 /* Private property holding the binding sets */
1769 gtk_css_style_property_register ("gtk-key-bindings",
1774 bindings_value_parse,
1775 bindings_value_print,
1778 _gtk_css_value_new_take_binding_sets (NULL));