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 "gtkcssimagegradientprivate.h"
44 #include "gtkcssimageprivate.h"
45 #include "gtkcssimageprivate.h"
46 #include "gtkgradient.h"
47 #include "gtkshadowprivate.h"
48 #include "gtksymboliccolorprivate.h"
49 #include "gtkthemingengine.h"
50 #include "gtktypebuiltins.h"
51 #include "gtkwin32themeprivate.h"
53 /*** REGISTRATION ***/
56 gtk_css_style_property_register (const char * name,
58 GtkStylePropertyFlags flags,
59 GtkCssStylePropertyParseFunc parse_value,
60 GtkCssStylePropertyPrintFunc print_value,
61 GtkCssStylePropertyComputeFunc compute_value,
62 GtkCssStylePropertyQueryFunc query_value,
63 GtkCssStylePropertyEqualFunc equal_func,
64 GtkCssValue * initial_value)
66 GtkCssStyleProperty *node;
68 g_assert (initial_value != NULL);
69 g_assert (parse_value != NULL);
70 g_assert (value_type == G_TYPE_NONE || query_value != NULL);
72 node = g_object_new (GTK_TYPE_CSS_STYLE_PROPERTY,
73 "value-type", value_type,
74 "inherit", (flags & GTK_STYLE_PROPERTY_INHERIT) ? TRUE : FALSE,
75 "initial-value", initial_value,
79 node->parse_value = parse_value;
81 node->print_value = print_value;
83 node->compute_value = compute_value;
84 node->query_value = query_value;
86 node->equal_func = equal_func;
88 _gtk_css_value_unref (initial_value);
94 string_append_string (GString *str,
99 g_string_append_c (str, '"');
102 len = strcspn (string, "\"\n\r\f");
103 g_string_append (str, string);
110 g_string_append (str, "\\A ");
113 g_string_append (str, "\\D ");
116 g_string_append (str, "\\C ");
119 g_string_append (str, "\\\"");
122 g_assert_not_reached ();
127 g_string_append_c (str, '"');
130 /*** IMPLEMENTATIONS ***/
133 query_simple (GtkCssStyleProperty *property,
134 const GtkCssValue *css_value,
137 _gtk_css_value_init_gvalue (css_value, value);
141 query_length_as_int (GtkCssStyleProperty *property,
142 const GtkCssValue *css_value,
145 g_value_init (value, G_TYPE_INT);
146 g_value_set_int (value, round (_gtk_css_number_get (_gtk_css_value_get_number (css_value), 100)));
150 color_parse (GtkCssStyleProperty *property,
151 GtkCssParser *parser,
154 GtkSymbolicColor *symbolic;
156 if (_gtk_css_parser_try (parser, "currentcolor", TRUE))
158 symbolic = gtk_symbolic_color_ref (_gtk_symbolic_color_get_current_color ());
162 symbolic = _gtk_css_parser_read_symbolic_color (parser);
163 if (symbolic == NULL)
167 return _gtk_css_value_new_take_symbolic_color (symbolic);
171 color_compute (GtkCssStyleProperty *property,
172 GtkStyleContext *context,
173 GtkCssValue *specified)
175 GtkSymbolicColor *symbolic = _gtk_css_value_get_symbolic_color (specified);
176 GtkCssValue *resolved;
178 if (symbolic == _gtk_symbolic_color_get_current_color ())
180 /* The computed value of the ‘currentColor’ keyword is the computed
181 * value of the ‘color’ property. If the ‘currentColor’ keyword is
182 * set on the ‘color’ property itself, it is treated as ‘color: inherit’.
184 if (g_str_equal (_gtk_style_property_get_name (GTK_STYLE_PROPERTY (property)), "color"))
186 GtkStyleContext *parent = gtk_style_context_get_parent (context);
189 return _gtk_css_value_ref (_gtk_style_context_peek_property (parent, "color"));
191 return _gtk_css_style_compute_value (context,
193 _gtk_css_style_property_get_initial_value (property));
197 return _gtk_css_value_ref (_gtk_style_context_peek_property (context, "color"));
200 else if ((resolved = _gtk_style_context_resolve_color_value (context,
207 return color_compute (property,
209 _gtk_css_style_property_get_initial_value (property));
214 font_family_parse (GtkCssStyleProperty *property,
215 GtkCssParser *parser,
221 /* We don't special case generic families. Pango should do
224 names = g_ptr_array_new ();
227 name = _gtk_css_parser_try_ident (parser, TRUE);
230 GString *string = g_string_new (name);
232 while ((name = _gtk_css_parser_try_ident (parser, TRUE)))
234 g_string_append_c (string, ' ');
235 g_string_append (string, name);
238 name = g_string_free (string, FALSE);
242 name = _gtk_css_parser_read_string (parser);
245 g_ptr_array_free (names, TRUE);
250 g_ptr_array_add (names, name);
251 } while (_gtk_css_parser_try (parser, ",", TRUE));
253 /* NULL-terminate array */
254 g_ptr_array_add (names, NULL);
255 return _gtk_css_value_new_take_strv ((char **) g_ptr_array_free (names, FALSE));
259 font_family_value_print (GtkCssStyleProperty *property,
260 const GtkCssValue *value,
263 const char **names = _gtk_css_value_get_strv (value);
265 if (names == NULL || *names == NULL)
267 g_string_append (string, "none");
271 string_append_string (string, *names);
275 g_string_append (string, ", ");
276 string_append_string (string, *names);
282 parse_pango_style (GtkCssStyleProperty *property,
283 GtkCssParser *parser,
288 if (!_gtk_css_parser_try_enum (parser, PANGO_TYPE_STYLE, &value))
290 _gtk_css_parser_error (parser, "unknown value for property");
294 return _gtk_css_value_new_from_enum (PANGO_TYPE_STYLE, value);
298 parse_pango_weight (GtkCssStyleProperty *property,
299 GtkCssParser *parser,
304 if (!_gtk_css_parser_try_enum (parser, PANGO_TYPE_WEIGHT, &value))
306 _gtk_css_parser_error (parser, "unknown value for property");
310 return _gtk_css_value_new_from_enum (PANGO_TYPE_WEIGHT, value);
314 parse_pango_variant (GtkCssStyleProperty *property,
315 GtkCssParser *parser,
320 if (!_gtk_css_parser_try_enum (parser, PANGO_TYPE_VARIANT, &value))
322 _gtk_css_parser_error (parser, "unknown value for property");
326 return _gtk_css_value_new_from_enum (PANGO_TYPE_VARIANT, value);
330 parse_border_style (GtkCssStyleProperty *property,
331 GtkCssParser *parser,
336 if (!_gtk_css_parser_try_enum (parser, GTK_TYPE_BORDER_STYLE, &value))
338 _gtk_css_parser_error (parser, "unknown value for property");
342 return _gtk_css_value_new_from_enum (GTK_TYPE_BORDER_STYLE, value);
346 parse_css_area (GtkCssStyleProperty *property,
347 GtkCssParser *parser,
352 if (!_gtk_css_parser_try_enum (parser, GTK_TYPE_CSS_AREA, &value))
354 _gtk_css_parser_error (parser, "unknown value for property");
358 return _gtk_css_value_new_from_enum (GTK_TYPE_CSS_AREA, value);
362 bindings_value_parse (GtkCssStyleProperty *property,
363 GtkCssParser *parser,
367 GtkBindingSet *binding_set;
370 array = g_ptr_array_new ();
373 name = _gtk_css_parser_try_ident (parser, TRUE);
376 _gtk_css_parser_error (parser, "Not a valid binding name");
377 g_ptr_array_free (array, TRUE);
381 binding_set = gtk_binding_set_find (name);
385 _gtk_css_parser_error (parser, "No binding set named '%s'", name);
390 g_ptr_array_add (array, binding_set);
393 while (_gtk_css_parser_try (parser, ",", TRUE));
395 return _gtk_css_value_new_take_binding_sets (array);
399 bindings_value_print (GtkCssStyleProperty *property,
400 const GtkCssValue *value,
406 array = _gtk_css_value_get_boxed (value);
408 for (i = 0; i < array->len; i++)
410 GtkBindingSet *binding_set = g_ptr_array_index (array, i);
413 g_string_append (string, ", ");
414 g_string_append (string, binding_set->set_name);
419 shadow_value_parse (GtkCssStyleProperty *property,
420 GtkCssParser *parser,
423 gboolean have_inset, have_color, have_lengths;
424 gdouble hoffset, voffset, blur, spread;
425 GtkSymbolicColor *color;
429 if (_gtk_css_parser_try (parser, "none", TRUE))
430 return _gtk_css_value_new_take_shadow (NULL);
432 shadow = _gtk_shadow_new ();
436 have_inset = have_lengths = have_color = FALSE;
438 for (i = 0; i < 3; i++)
441 _gtk_css_parser_try (parser, "inset", TRUE))
448 _gtk_css_parser_try_double (parser, &hoffset))
452 if (!_gtk_css_parser_try_double (parser, &voffset))
454 _gtk_css_parser_error (parser, "Horizontal and vertical offsets are required");
455 _gtk_shadow_unref (shadow);
459 if (!_gtk_css_parser_try_double (parser, &blur))
462 if (!_gtk_css_parser_try_double (parser, &spread))
472 /* XXX: the color is optional and UA-defined if it's missing,
473 * but it doesn't really make sense for us...
475 color = _gtk_css_parser_read_symbolic_color (parser);
479 _gtk_shadow_unref (shadow);
485 if (!have_color || !have_lengths)
487 _gtk_css_parser_error (parser, "Must specify at least color and offsets");
488 _gtk_shadow_unref (shadow);
492 _gtk_shadow_append (shadow,
497 gtk_symbolic_color_unref (color);
500 while (_gtk_css_parser_try (parser, ",", TRUE));
502 return _gtk_css_value_new_take_shadow (shadow);
506 shadow_value_print (GtkCssStyleProperty *property,
507 const GtkCssValue *value,
512 shadow = _gtk_css_value_get_shadow (value);
515 g_string_append (string, "none");
517 _gtk_shadow_print (shadow, string);
521 shadow_value_compute (GtkCssStyleProperty *property,
522 GtkStyleContext *context,
523 GtkCssValue *specified)
527 shadow = _gtk_css_value_get_shadow (specified);
529 shadow = _gtk_shadow_resolve (shadow, context);
531 return _gtk_css_value_new_take_shadow (shadow);
535 border_corner_radius_value_parse (GtkCssStyleProperty *property,
536 GtkCssParser *parser,
539 GtkCssBorderCornerRadius corner;
541 if (!_gtk_css_parser_read_number (parser,
543 GTK_CSS_POSITIVE_ONLY
544 | GTK_CSS_PARSE_PERCENT
545 | GTK_CSS_NUMBER_AS_PIXELS
546 | GTK_CSS_PARSE_LENGTH))
549 if (!_gtk_css_parser_has_number (parser))
550 corner.vertical = corner.horizontal;
551 else if (!_gtk_css_parser_read_number (parser,
553 GTK_CSS_POSITIVE_ONLY
554 | GTK_CSS_PARSE_PERCENT
555 | GTK_CSS_NUMBER_AS_PIXELS
556 | GTK_CSS_PARSE_LENGTH))
559 return _gtk_css_value_new_from_border_corner_radius (&corner);
563 border_corner_radius_value_print (GtkCssStyleProperty *property,
564 const GtkCssValue *value,
567 const GtkCssBorderCornerRadius *corner;
569 corner = _gtk_css_value_get_border_corner_radius (value);
571 _gtk_css_number_print (&corner->horizontal, string);
573 if (!_gtk_css_number_equal (&corner->horizontal, &corner->vertical))
575 g_string_append_c (string, ' ');
576 _gtk_css_number_print (&corner->vertical, string);
581 css_image_value_parse (GtkCssStyleProperty *property,
582 GtkCssParser *parser,
587 if (_gtk_css_parser_try (parser, "none", TRUE))
591 image = _gtk_css_image_new_parse (parser, base);
596 return _gtk_css_value_new_take_image (image);
600 css_image_value_print (GtkCssStyleProperty *property,
601 const GtkCssValue *value,
604 GtkCssImage *image = _gtk_css_value_get_image (value);
607 _gtk_css_image_print (image, string);
609 g_string_append (string, "none");
613 css_image_value_compute (GtkCssStyleProperty *property,
614 GtkStyleContext *context,
615 GtkCssValue *specified)
617 GtkCssImage *image, *computed;
619 image = _gtk_css_value_get_image (specified);
622 return _gtk_css_value_ref (specified);
624 computed = _gtk_css_image_compute (image, context);
626 if (computed == image)
628 g_object_unref (computed);
629 return _gtk_css_value_ref (specified);
632 return _gtk_css_value_new_take_image (computed);
636 css_image_value_query (GtkCssStyleProperty *property,
637 const GtkCssValue *css_value,
640 GtkCssImage *image = _gtk_css_value_get_image (css_value);
641 cairo_pattern_t *pattern;
642 cairo_surface_t *surface;
643 cairo_matrix_t matrix;
645 g_value_init (value, CAIRO_GOBJECT_TYPE_PATTERN);
647 if (GTK_IS_CSS_IMAGE_GRADIENT (image))
648 g_value_set_boxed (value, GTK_CSS_IMAGE_GRADIENT (image)->pattern);
649 else if (image != NULL)
651 double width, height;
653 /* the 100, 100 is rather random */
654 _gtk_css_image_get_concrete_size (image, 0, 0, 100, 100, &width, &height);
655 surface = _gtk_css_image_get_surface (image, NULL, width, height);
656 pattern = cairo_pattern_create_for_surface (surface);
657 cairo_matrix_init_scale (&matrix, width, height);
658 cairo_pattern_set_matrix (pattern, &matrix);
659 cairo_surface_destroy (surface);
660 g_value_take_boxed (value, pattern);
665 font_size_parse (GtkCssStyleProperty *property,
666 GtkCssParser *parser,
671 if (!_gtk_css_parser_try_double (parser, &d))
673 _gtk_css_parser_error (parser, "Expected a number");
677 return _gtk_css_value_new_from_double (d);
681 outline_parse (GtkCssStyleProperty *property,
682 GtkCssParser *parser,
687 if (!_gtk_css_parser_try_int (parser, &i))
689 _gtk_css_parser_error (parser, "Expected an integer");
693 return _gtk_css_value_new_from_int (i);
697 border_image_repeat_parse (GtkCssStyleProperty *property,
698 GtkCssParser *parser,
701 GValue value = G_VALUE_INIT;
704 g_value_init (&value, GTK_TYPE_CSS_BORDER_IMAGE_REPEAT);
705 if (!_gtk_css_style_parse_value (&value, parser, base))
707 g_value_unset (&value);
711 result = _gtk_css_value_new_from_gvalue (&value);
712 g_value_unset (&value);
718 border_image_slice_parse (GtkCssStyleProperty *property,
719 GtkCssParser *parser,
722 GValue value = G_VALUE_INIT;
725 g_value_init (&value, GTK_TYPE_BORDER);
726 if (!_gtk_css_style_parse_value (&value, parser, base))
728 g_value_unset (&value);
732 result = _gtk_css_value_new_from_gvalue (&value);
733 g_value_unset (&value);
739 border_image_width_parse (GtkCssStyleProperty *property,
740 GtkCssParser *parser,
743 GValue value = G_VALUE_INIT;
746 g_value_init (&value, GTK_TYPE_BORDER);
747 if (!_gtk_css_style_parse_value (&value, parser, base))
749 g_value_unset (&value);
753 result = _gtk_css_value_new_from_gvalue (&value);
754 g_value_unset (&value);
760 engine_parse (GtkCssStyleProperty *property,
761 GtkCssParser *parser,
764 GtkThemingEngine *engine;
767 if (_gtk_css_parser_try (parser, "none", TRUE))
768 return _gtk_css_value_new_from_theming_engine (gtk_theming_engine_load (NULL));
770 str = _gtk_css_parser_try_ident (parser, TRUE);
773 _gtk_css_parser_error (parser, "Expected a valid theme name");
777 engine = gtk_theming_engine_load (str);
781 _gtk_css_parser_error (parser, "Theming engine '%s' not found", str);
788 return _gtk_css_value_new_from_theming_engine (engine);
792 transition_parse (GtkCssStyleProperty *property,
793 GtkCssParser *parser,
796 GValue value = G_VALUE_INIT;
799 g_value_init (&value, GTK_TYPE_ANIMATION_DESCRIPTION);
800 if (!_gtk_css_style_parse_value (&value, parser, base))
802 g_value_unset (&value);
806 result = _gtk_css_value_new_from_gvalue (&value);
807 g_value_unset (&value);
813 parse_margin (GtkCssStyleProperty *property,
814 GtkCssParser *parser,
819 if (!_gtk_css_parser_read_number (parser,
821 GTK_CSS_NUMBER_AS_PIXELS
822 | GTK_CSS_PARSE_LENGTH))
825 return _gtk_css_value_new_from_number (&number);
829 compute_margin (GtkCssStyleProperty *property,
830 GtkStyleContext *context,
831 GtkCssValue *specified)
835 if (_gtk_css_number_compute (&number,
836 _gtk_css_value_get_number (specified),
839 return _gtk_css_value_new_from_number (&number);
841 return _gtk_css_value_ref (specified);
845 parse_padding (GtkCssStyleProperty *property,
846 GtkCssParser *parser,
851 if (!_gtk_css_parser_read_number (parser,
853 GTK_CSS_POSITIVE_ONLY
854 | GTK_CSS_NUMBER_AS_PIXELS
855 | GTK_CSS_PARSE_LENGTH))
858 return _gtk_css_value_new_from_number (&number);
862 compute_padding (GtkCssStyleProperty *property,
863 GtkStyleContext *context,
864 GtkCssValue *specified)
868 if (_gtk_css_number_compute (&number,
869 _gtk_css_value_get_number (specified),
871 return _gtk_css_value_new_from_number (&number);
872 return _gtk_css_value_ref (specified);
876 parse_border_width (GtkCssStyleProperty *property,
877 GtkCssParser *parser,
882 if (!_gtk_css_parser_read_number (parser,
884 GTK_CSS_POSITIVE_ONLY
885 | GTK_CSS_NUMBER_AS_PIXELS
886 | GTK_CSS_PARSE_LENGTH))
889 return _gtk_css_value_new_from_number (&number);
893 compute_border_width (GtkCssStyleProperty *property,
894 GtkStyleContext *context,
895 GtkCssValue *specified)
897 GtkCssStyleProperty *style;
898 GtkBorderStyle border_style;
901 /* The -1 is magic that is only true because we register the style
902 * properties directly after the width properties.
904 style = _gtk_css_style_property_lookup_by_id (_gtk_css_style_property_get_id (property) - 1);
906 border_style = _gtk_css_value_get_border_style (_gtk_style_context_peek_property (context, _gtk_style_property_get_name (GTK_STYLE_PROPERTY (style))));
908 if (border_style == GTK_BORDER_STYLE_NONE ||
909 border_style == GTK_BORDER_STYLE_HIDDEN)
911 _gtk_css_number_init (&number, 0, GTK_CSS_PX);
915 _gtk_css_number_compute (&number,
916 _gtk_css_value_get_number (specified),
919 return _gtk_css_value_new_from_number (&number);
923 background_repeat_value_parse (GtkCssStyleProperty *property,
924 GtkCssParser *parser,
927 int repeat, vertical;
929 if (!_gtk_css_parser_try_enum (parser, GTK_TYPE_CSS_BACKGROUND_REPEAT, &repeat))
931 _gtk_css_parser_error (parser, "Not a valid value");
935 if (repeat <= GTK_CSS_BACKGROUND_REPEAT_MASK)
937 if (_gtk_css_parser_try_enum (parser, GTK_TYPE_CSS_BACKGROUND_REPEAT, &vertical))
939 if (vertical >= GTK_CSS_BACKGROUND_REPEAT_MASK)
941 _gtk_css_parser_error (parser, "Not a valid 2nd value");
945 repeat |= vertical << GTK_CSS_BACKGROUND_REPEAT_SHIFT;
948 repeat |= repeat << GTK_CSS_BACKGROUND_REPEAT_SHIFT;
951 return _gtk_css_value_new_from_enum (GTK_TYPE_CSS_BACKGROUND_REPEAT, repeat);
955 background_repeat_value_print (GtkCssStyleProperty *property,
956 const GtkCssValue *value,
959 GEnumClass *enum_class;
960 GEnumValue *enum_value;
961 GtkCssBackgroundRepeat repeat;
963 repeat = _gtk_css_value_get_enum (value);
964 enum_class = g_type_class_ref (GTK_TYPE_CSS_BACKGROUND_REPEAT);
965 enum_value = g_enum_get_value (enum_class, repeat);
967 /* only triggers for 'repeat-x' and 'repeat-y' */
969 g_string_append (string, enum_value->value_nick);
972 enum_value = g_enum_get_value (enum_class, GTK_CSS_BACKGROUND_HORIZONTAL (repeat));
973 g_string_append (string, enum_value->value_nick);
975 if (GTK_CSS_BACKGROUND_HORIZONTAL (repeat) != GTK_CSS_BACKGROUND_VERTICAL (repeat))
977 enum_value = g_enum_get_value (enum_class, GTK_CSS_BACKGROUND_VERTICAL (repeat));
978 g_string_append (string, " ");
979 g_string_append (string, enum_value->value_nick);
983 g_type_class_unref (enum_class);
987 background_size_parse (GtkCssStyleProperty *property,
988 GtkCssParser *parser,
991 GtkCssBackgroundSize size = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), FALSE, FALSE};
993 if (_gtk_css_parser_try (parser, "cover", TRUE))
995 else if (_gtk_css_parser_try (parser, "contain", TRUE))
999 if (_gtk_css_parser_try (parser, "auto", TRUE))
1000 _gtk_css_number_init (&size.width, 0, GTK_CSS_PX);
1001 else if (!_gtk_css_parser_read_number (parser,
1003 GTK_CSS_POSITIVE_ONLY
1004 | GTK_CSS_PARSE_PERCENT
1005 | GTK_CSS_PARSE_LENGTH))
1008 if (_gtk_css_parser_try (parser, "auto", TRUE))
1009 _gtk_css_number_init (&size.height, 0, GTK_CSS_PX);
1010 else if (_gtk_css_parser_has_number (parser))
1012 if (!_gtk_css_parser_read_number (parser,
1014 GTK_CSS_POSITIVE_ONLY
1015 | GTK_CSS_PARSE_PERCENT
1016 | GTK_CSS_PARSE_LENGTH))
1020 _gtk_css_number_init (&size.height, 0, GTK_CSS_PX);
1023 return _gtk_css_value_new_from_background_size (&size);
1027 background_size_print (GtkCssStyleProperty *property,
1028 const GtkCssValue *value,
1031 const GtkCssBackgroundSize *size = _gtk_css_value_get_background_size (value);
1034 g_string_append (string, "cover");
1035 else if (size->contain)
1036 g_string_append (string, "contain");
1039 if (size->width.value == 0)
1040 g_string_append (string, "auto");
1042 _gtk_css_number_print (&size->width, string);
1044 if (size->height.value != 0)
1046 g_string_append (string, " ");
1047 _gtk_css_number_print (&size->height, string);
1052 static GtkCssValue *
1053 background_size_compute (GtkCssStyleProperty *property,
1054 GtkStyleContext *context,
1055 GtkCssValue *specified)
1057 const GtkCssBackgroundSize *ssize = _gtk_css_value_get_background_size (specified);
1058 GtkCssBackgroundSize csize;
1061 csize.cover = ssize->cover;
1062 csize.contain = ssize->contain;
1063 changed = _gtk_css_number_compute (&csize.width,
1066 changed |= _gtk_css_number_compute (&csize.height,
1070 return _gtk_css_value_new_from_background_size (&csize);
1071 return _gtk_css_value_ref (specified);
1074 static GtkCssValue *
1075 background_position_parse (GtkCssStyleProperty *property,
1076 GtkCssParser *parser,
1079 static const struct {
1082 gboolean horizontal;
1085 { "left", 0, TRUE, FALSE },
1086 { "right", 100, TRUE, FALSE },
1087 { "center", 50, TRUE, TRUE },
1088 { "top", 0, FALSE, TRUE },
1089 { "bottom", 100, FALSE, TRUE },
1090 { NULL , 0, TRUE, FALSE }, /* used for numbers */
1091 { NULL , 50, TRUE, TRUE } /* used for no value */
1093 GtkCssBackgroundPosition pos;
1094 GtkCssNumber *missing;
1095 guint first, second;
1097 for (first = 0; names[first].name != NULL; first++)
1099 if (_gtk_css_parser_try (parser, names[first].name, TRUE))
1101 if (names[first].horizontal)
1103 _gtk_css_number_init (&pos.x, names[first].percentage, GTK_CSS_PERCENT);
1108 _gtk_css_number_init (&pos.y, names[first].percentage, GTK_CSS_PERCENT);
1114 if (names[first].name == NULL)
1117 if (!_gtk_css_parser_read_number (parser,
1119 GTK_CSS_PARSE_PERCENT
1120 | GTK_CSS_PARSE_LENGTH))
1124 for (second = 0; names[second].name != NULL; second++)
1126 if (_gtk_css_parser_try (parser, names[second].name, TRUE))
1128 _gtk_css_number_init (missing, names[second].percentage, GTK_CSS_PERCENT);
1133 if (names[second].name == NULL)
1135 if (_gtk_css_parser_has_number (parser))
1137 if (missing != &pos.y)
1139 _gtk_css_parser_error (parser, "Invalid combination of values");
1142 if (!_gtk_css_parser_read_number (parser,
1144 GTK_CSS_PARSE_PERCENT
1145 | GTK_CSS_PARSE_LENGTH))
1151 _gtk_css_number_init (missing, 50, GTK_CSS_PERCENT);
1156 if ((names[first].horizontal && !names[second].vertical) ||
1157 (!names[first].horizontal && !names[second].horizontal))
1159 _gtk_css_parser_error (parser, "Invalid combination of values");
1164 return _gtk_css_value_new_from_background_position (&pos);
1168 background_position_print (GtkCssStyleProperty *property,
1169 const GtkCssValue *value,
1172 const GtkCssBackgroundPosition *pos = _gtk_css_value_get_background_position (value);
1173 static const GtkCssNumber center = GTK_CSS_NUMBER_INIT (50, GTK_CSS_PERCENT);
1174 static const struct {
1177 GtkCssNumber number;
1179 { "left", "top", GTK_CSS_NUMBER_INIT (0, GTK_CSS_PERCENT) },
1180 { "right", "bottom", GTK_CSS_NUMBER_INIT (100, GTK_CSS_PERCENT) }
1184 if (_gtk_css_number_equal (&pos->x, ¢er))
1186 if (_gtk_css_number_equal (&pos->y, ¢er))
1188 g_string_append (string, "center");
1194 for (i = 0; i < G_N_ELEMENTS (values); i++)
1196 if (_gtk_css_number_equal (&pos->x, &values[i].number))
1198 g_string_append (string, values[i].x_name);
1202 if (i == G_N_ELEMENTS (values))
1203 _gtk_css_number_print (&pos->x, string);
1205 if (_gtk_css_number_equal (&pos->y, ¢er))
1208 g_string_append_c (string, ' ');
1211 for (i = 0; i < G_N_ELEMENTS (values); i++)
1213 if (_gtk_css_number_equal (&pos->y, &values[i].number))
1215 g_string_append (string, values[i].y_name);
1219 if (i == G_N_ELEMENTS (values))
1221 if (_gtk_css_number_equal (&pos->x, ¢er))
1222 g_string_append (string, "center ");
1223 _gtk_css_number_print (&pos->y, string);
1227 static GtkCssValue *
1228 background_position_compute (GtkCssStyleProperty *property,
1229 GtkStyleContext *context,
1230 GtkCssValue *specified)
1232 const GtkCssBackgroundPosition *spos = _gtk_css_value_get_background_position (specified);
1233 GtkCssBackgroundPosition cpos;
1236 changed = _gtk_css_number_compute (&cpos.x,
1239 changed |= _gtk_css_number_compute (&cpos.y,
1243 return _gtk_css_value_new_from_background_position (&cpos);
1244 return _gtk_css_value_ref (specified);
1247 /*** REGISTRATION ***/
1249 static GtkSymbolicColor *
1250 gtk_symbolic_color_new_rgba (double red,
1255 GdkRGBA rgba = { red, green, blue, alpha };
1257 return gtk_symbolic_color_new_literal (&rgba);
1261 _gtk_css_style_property_init_properties (void)
1263 char *default_font_family[] = { "Sans", NULL };
1264 GtkCssNumber number;
1265 GtkCssBackgroundSize default_background_size = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), FALSE, FALSE };
1266 GtkCssBackgroundPosition default_background_position = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PERCENT), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PERCENT)};
1267 GtkCssBorderCornerRadius no_corner_radius = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX) };
1268 GtkBorder border_of_ones = { 1, 1, 1, 1 };
1269 GtkCssBorderImageRepeat border_image_repeat = { GTK_CSS_REPEAT_STYLE_STRETCH, GTK_CSS_REPEAT_STYLE_STRETCH };
1271 /* Initialize "color" and "font-size" first,
1272 * so that when computing values later they are
1273 * done first. That way, 'currentColor' and font
1274 * sizes in em can be looked up properly */
1275 gtk_css_style_property_register ("color",
1277 GTK_STYLE_PROPERTY_INHERIT,
1283 _gtk_css_value_new_take_symbolic_color (
1284 gtk_symbolic_color_new_rgba (1, 1, 1, 1)));
1285 gtk_css_style_property_register ("font-size",
1287 GTK_STYLE_PROPERTY_INHERIT,
1293 _gtk_css_value_new_from_double (10.0));
1295 /* properties that aren't referenced when computing values
1297 gtk_css_style_property_register ("background-color",
1305 _gtk_css_value_new_take_symbolic_color (
1306 gtk_symbolic_color_new_rgba (0, 0, 0, 0)));
1308 gtk_css_style_property_register ("font-family",
1310 GTK_STYLE_PROPERTY_INHERIT,
1312 font_family_value_print,
1316 _gtk_css_value_new_take_strv (g_strdupv (default_font_family)));
1317 gtk_css_style_property_register ("font-style",
1319 GTK_STYLE_PROPERTY_INHERIT,
1325 _gtk_css_value_new_from_enum (PANGO_TYPE_STYLE,
1326 PANGO_STYLE_NORMAL));
1327 gtk_css_style_property_register ("font-variant",
1329 GTK_STYLE_PROPERTY_INHERIT,
1330 parse_pango_variant,
1335 _gtk_css_value_new_from_enum (PANGO_TYPE_VARIANT,
1336 PANGO_VARIANT_NORMAL));
1337 /* xxx: need to parse this properly, ie parse the numbers */
1338 gtk_css_style_property_register ("font-weight",
1340 GTK_STYLE_PROPERTY_INHERIT,
1346 _gtk_css_value_new_from_enum (PANGO_TYPE_WEIGHT,
1347 PANGO_WEIGHT_NORMAL));
1349 gtk_css_style_property_register ("text-shadow",
1351 GTK_STYLE_PROPERTY_INHERIT,
1354 shadow_value_compute,
1357 _gtk_css_value_new_take_shadow (NULL));
1359 gtk_css_style_property_register ("icon-shadow",
1361 GTK_STYLE_PROPERTY_INHERIT,
1364 shadow_value_compute,
1367 _gtk_css_value_new_take_shadow (NULL));
1369 gtk_css_style_property_register ("box-shadow",
1374 shadow_value_compute,
1377 _gtk_css_value_new_take_shadow (NULL));
1379 _gtk_css_number_init (&number, 0, GTK_CSS_PX);
1380 gtk_css_style_property_register ("margin-top",
1386 query_length_as_int,
1388 _gtk_css_value_new_from_number (&number));
1389 gtk_css_style_property_register ("margin-left",
1395 query_length_as_int,
1397 _gtk_css_value_new_from_number (&number));
1398 gtk_css_style_property_register ("margin-bottom",
1404 query_length_as_int,
1406 _gtk_css_value_new_from_number (&number));
1407 gtk_css_style_property_register ("margin-right",
1413 query_length_as_int,
1415 _gtk_css_value_new_from_number (&number));
1416 gtk_css_style_property_register ("padding-top",
1422 query_length_as_int,
1424 _gtk_css_value_new_from_number (&number));
1425 gtk_css_style_property_register ("padding-left",
1431 query_length_as_int,
1433 _gtk_css_value_new_from_number (&number));
1434 gtk_css_style_property_register ("padding-bottom",
1440 query_length_as_int,
1442 _gtk_css_value_new_from_number (&number));
1443 gtk_css_style_property_register ("padding-right",
1449 query_length_as_int,
1451 _gtk_css_value_new_from_number (&number));
1452 /* IMPORTANT: compute_border_width() requires that the border-width
1453 * properties be immeditaly followed by the border-style properties
1455 gtk_css_style_property_register ("border-top-style",
1456 GTK_TYPE_BORDER_STYLE,
1463 _gtk_css_value_new_from_border_style (GTK_BORDER_STYLE_NONE));
1464 gtk_css_style_property_register ("border-top-width",
1469 compute_border_width,
1470 query_length_as_int,
1472 _gtk_css_value_new_from_number (&number));
1473 gtk_css_style_property_register ("border-left-style",
1474 GTK_TYPE_BORDER_STYLE,
1481 _gtk_css_value_new_from_border_style (GTK_BORDER_STYLE_NONE));
1482 gtk_css_style_property_register ("border-left-width",
1487 compute_border_width,
1488 query_length_as_int,
1490 _gtk_css_value_new_from_number (&number));
1491 gtk_css_style_property_register ("border-bottom-style",
1492 GTK_TYPE_BORDER_STYLE,
1499 _gtk_css_value_new_from_border_style (GTK_BORDER_STYLE_NONE));
1500 gtk_css_style_property_register ("border-bottom-width",
1505 compute_border_width,
1506 query_length_as_int,
1508 _gtk_css_value_new_from_number (&number));
1509 gtk_css_style_property_register ("border-right-style",
1510 GTK_TYPE_BORDER_STYLE,
1517 _gtk_css_value_new_from_border_style (GTK_BORDER_STYLE_NONE));
1518 gtk_css_style_property_register ("border-right-width",
1523 compute_border_width,
1524 query_length_as_int,
1526 _gtk_css_value_new_from_number (&number));
1528 gtk_css_style_property_register ("border-top-left-radius",
1529 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1531 border_corner_radius_value_parse,
1532 border_corner_radius_value_print,
1536 _gtk_css_value_new_from_border_corner_radius (&no_corner_radius));
1537 gtk_css_style_property_register ("border-top-right-radius",
1538 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1540 border_corner_radius_value_parse,
1541 border_corner_radius_value_print,
1545 _gtk_css_value_new_from_border_corner_radius (&no_corner_radius));
1546 gtk_css_style_property_register ("border-bottom-right-radius",
1547 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1549 border_corner_radius_value_parse,
1550 border_corner_radius_value_print,
1554 _gtk_css_value_new_from_border_corner_radius (&no_corner_radius));
1555 gtk_css_style_property_register ("border-bottom-left-radius",
1556 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1558 border_corner_radius_value_parse,
1559 border_corner_radius_value_print,
1563 _gtk_css_value_new_from_border_corner_radius (&no_corner_radius));
1565 gtk_css_style_property_register ("outline-style",
1566 GTK_TYPE_BORDER_STYLE,
1573 _gtk_css_value_new_from_border_style (GTK_BORDER_STYLE_NONE));
1574 gtk_css_style_property_register ("outline-width",
1579 compute_border_width,
1580 query_length_as_int,
1582 _gtk_css_value_new_from_number (&number));
1583 gtk_css_style_property_register ("outline-offset",
1591 _gtk_css_value_new_from_int (0));
1593 gtk_css_style_property_register ("background-clip",
1601 _gtk_css_value_new_from_enum (GTK_TYPE_CSS_AREA, GTK_CSS_AREA_BORDER_BOX));
1602 gtk_css_style_property_register ("background-origin",
1610 _gtk_css_value_new_from_enum (GTK_TYPE_CSS_AREA, GTK_CSS_AREA_PADDING_BOX));
1611 gtk_css_style_property_register ("background-size",
1614 background_size_parse,
1615 background_size_print,
1616 background_size_compute,
1619 _gtk_css_value_new_from_background_size (&default_background_size));
1620 gtk_css_style_property_register ("background-position",
1623 background_position_parse,
1624 background_position_print,
1625 background_position_compute,
1628 _gtk_css_value_new_from_background_position (&default_background_position));
1630 gtk_css_style_property_register ("border-top-color",
1638 _gtk_css_value_new_take_symbolic_color (
1639 gtk_symbolic_color_ref (
1640 _gtk_symbolic_color_get_current_color ())));
1641 gtk_css_style_property_register ("border-right-color",
1649 _gtk_css_value_new_take_symbolic_color (
1650 gtk_symbolic_color_ref (
1651 _gtk_symbolic_color_get_current_color ())));
1652 gtk_css_style_property_register ("border-bottom-color",
1660 _gtk_css_value_new_take_symbolic_color (
1661 gtk_symbolic_color_ref (
1662 _gtk_symbolic_color_get_current_color ())));
1663 gtk_css_style_property_register ("border-left-color",
1671 _gtk_css_value_new_take_symbolic_color (
1672 gtk_symbolic_color_ref (
1673 _gtk_symbolic_color_get_current_color ())));
1674 gtk_css_style_property_register ("outline-color",
1682 _gtk_css_value_new_take_symbolic_color (
1683 gtk_symbolic_color_ref (
1684 _gtk_symbolic_color_get_current_color ())));
1686 gtk_css_style_property_register ("background-repeat",
1687 GTK_TYPE_CSS_BACKGROUND_REPEAT,
1689 background_repeat_value_parse,
1690 background_repeat_value_print,
1694 _gtk_css_value_new_from_enum (GTK_TYPE_CSS_BACKGROUND_REPEAT,
1695 GTK_CSS_BACKGROUND_REPEAT |
1696 (GTK_CSS_BACKGROUND_REPEAT << GTK_CSS_BACKGROUND_REPEAT_SHIFT)));
1697 gtk_css_style_property_register ("background-image",
1698 CAIRO_GOBJECT_TYPE_PATTERN,
1700 css_image_value_parse,
1701 css_image_value_print,
1702 css_image_value_compute,
1703 css_image_value_query,
1705 _gtk_css_value_new_take_image (NULL));
1707 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,
1713 css_image_value_query,
1715 _gtk_css_value_new_take_image (NULL));
1716 gtk_css_style_property_register ("border-image-repeat",
1717 GTK_TYPE_CSS_BORDER_IMAGE_REPEAT,
1719 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",
1730 border_image_slice_parse,
1735 _gtk_css_value_new_from_boxed (GTK_TYPE_BORDER, &border_of_ones));
1736 gtk_css_style_property_register ("border-image-width",
1739 border_image_width_parse,
1744 _gtk_css_value_new_from_boxed (GTK_TYPE_BORDER, NULL));
1745 gtk_css_style_property_register ("engine",
1746 GTK_TYPE_THEMING_ENGINE,
1753 _gtk_css_value_new_from_theming_engine (gtk_theming_engine_load (NULL)));
1754 gtk_css_style_property_register ("transition",
1755 GTK_TYPE_ANIMATION_DESCRIPTION,
1762 _gtk_css_value_new_from_boxed (GTK_TYPE_ANIMATION_DESCRIPTION, NULL));
1764 /* Private property holding the binding sets */
1765 gtk_css_style_property_register ("gtk-key-bindings",
1768 bindings_value_parse,
1769 bindings_value_print,
1773 _gtk_css_value_new_take_binding_sets (NULL));