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 "gtkcssnumbervalueprivate.h"
47 #include "gtkcssrgbavalueprivate.h"
48 #include "gtkcssshadowvalueprivate.h"
49 #include "gtksymboliccolorprivate.h"
50 #include "gtkthemingengine.h"
51 #include "gtktypebuiltins.h"
52 #include "gtkwin32themeprivate.h"
54 /*** REGISTRATION ***/
57 gtk_css_style_property_register (const char * name,
59 GtkStylePropertyFlags flags,
60 GtkCssStylePropertyParseFunc parse_value,
61 GtkCssStylePropertyPrintFunc print_value,
62 GtkCssStylePropertyComputeFunc compute_value,
63 GtkCssStylePropertyQueryFunc query_value,
64 GtkCssStylePropertyAssignFunc assign_value,
65 GtkCssStylePropertyEqualFunc equal_func,
66 GtkCssValue * initial_value)
68 GtkCssStyleProperty *node;
70 g_assert (initial_value != NULL);
71 g_assert (parse_value != NULL);
72 g_assert (value_type == G_TYPE_NONE || query_value != NULL);
73 g_assert (value_type == G_TYPE_NONE || assign_value != NULL);
75 node = g_object_new (GTK_TYPE_CSS_STYLE_PROPERTY,
76 "value-type", value_type,
77 "inherit", (flags & GTK_STYLE_PROPERTY_INHERIT) ? TRUE : FALSE,
78 "initial-value", initial_value,
82 node->parse_value = parse_value;
84 node->print_value = print_value;
86 node->compute_value = compute_value;
87 node->query_value = query_value;
88 node->assign_value = assign_value;
90 node->equal_func = equal_func;
92 _gtk_css_value_unref (initial_value);
98 string_append_string (GString *str,
103 g_string_append_c (str, '"');
106 len = strcspn (string, "\"\n\r\f");
107 g_string_append (str, string);
114 g_string_append (str, "\\A ");
117 g_string_append (str, "\\D ");
120 g_string_append (str, "\\C ");
123 g_string_append (str, "\\\"");
126 g_assert_not_reached ();
131 g_string_append_c (str, '"');
134 /*** IMPLEMENTATIONS ***/
137 query_simple (GtkCssStyleProperty *property,
138 const GtkCssValue *css_value,
141 _gtk_css_value_init_gvalue (css_value, value);
145 assign_simple (GtkCssStyleProperty *property,
148 return _gtk_css_value_new_from_gvalue (value);
152 query_length_as_int (GtkCssStyleProperty *property,
153 const GtkCssValue *css_value,
156 g_value_init (value, G_TYPE_INT);
157 g_value_set_int (value, round (_gtk_css_number_value_get (css_value, 100)));
161 assign_length_from_int (GtkCssStyleProperty *property,
164 return _gtk_css_number_value_new (g_value_get_int (value), GTK_CSS_PX);
168 color_parse (GtkCssStyleProperty *property,
169 GtkCssParser *parser,
172 GtkSymbolicColor *symbolic;
174 if (_gtk_css_parser_try (parser, "currentcolor", TRUE))
176 symbolic = gtk_symbolic_color_ref (_gtk_symbolic_color_get_current_color ());
180 symbolic = _gtk_css_parser_read_symbolic_color (parser);
181 if (symbolic == NULL)
185 return _gtk_css_value_new_take_symbolic_color (symbolic);
189 color_compute (GtkCssStyleProperty *property,
190 GtkStyleContext *context,
191 GtkCssValue *specified)
193 return _gtk_css_rgba_value_compute_from_symbolic (specified,
194 _gtk_css_style_property_get_initial_value (property),
200 color_property_compute (GtkCssStyleProperty *property,
201 GtkStyleContext *context,
202 GtkCssValue *specified)
206 value = _gtk_css_rgba_value_compute_from_symbolic (specified,
207 _gtk_css_style_property_get_initial_value (property),
210 _gtk_css_rgba_value_get_rgba (value);
215 color_query (GtkCssStyleProperty *property,
216 const GtkCssValue *css_value,
219 g_value_init (value, GDK_TYPE_RGBA);
220 g_value_set_boxed (value, _gtk_css_rgba_value_get_rgba (css_value));
224 color_assign (GtkCssStyleProperty *property,
227 return _gtk_css_rgba_value_new_from_rgba (g_value_get_boxed (value));
231 font_family_parse (GtkCssStyleProperty *property,
232 GtkCssParser *parser,
238 /* We don't special case generic families. Pango should do
241 names = g_ptr_array_new ();
244 name = _gtk_css_parser_try_ident (parser, TRUE);
247 GString *string = g_string_new (name);
249 while ((name = _gtk_css_parser_try_ident (parser, TRUE)))
251 g_string_append_c (string, ' ');
252 g_string_append (string, name);
255 name = g_string_free (string, FALSE);
259 name = _gtk_css_parser_read_string (parser);
262 g_ptr_array_free (names, TRUE);
267 g_ptr_array_add (names, name);
268 } while (_gtk_css_parser_try (parser, ",", TRUE));
270 /* NULL-terminate array */
271 g_ptr_array_add (names, NULL);
272 return _gtk_css_value_new_take_strv ((char **) g_ptr_array_free (names, FALSE));
276 font_family_value_print (GtkCssStyleProperty *property,
277 const GtkCssValue *value,
280 const char **names = _gtk_css_value_get_strv (value);
282 if (names == NULL || *names == NULL)
284 g_string_append (string, "none");
288 string_append_string (string, *names);
292 g_string_append (string, ", ");
293 string_append_string (string, *names);
299 parse_pango_style (GtkCssStyleProperty *property,
300 GtkCssParser *parser,
305 if (!_gtk_css_parser_try_enum (parser, PANGO_TYPE_STYLE, &value))
307 _gtk_css_parser_error (parser, "unknown value for property");
311 return _gtk_css_value_new_from_enum (PANGO_TYPE_STYLE, value);
315 parse_pango_weight (GtkCssStyleProperty *property,
316 GtkCssParser *parser,
321 if (!_gtk_css_parser_try_enum (parser, PANGO_TYPE_WEIGHT, &value))
323 _gtk_css_parser_error (parser, "unknown value for property");
327 return _gtk_css_value_new_from_enum (PANGO_TYPE_WEIGHT, value);
331 parse_pango_variant (GtkCssStyleProperty *property,
332 GtkCssParser *parser,
337 if (!_gtk_css_parser_try_enum (parser, PANGO_TYPE_VARIANT, &value))
339 _gtk_css_parser_error (parser, "unknown value for property");
343 return _gtk_css_value_new_from_enum (PANGO_TYPE_VARIANT, value);
347 parse_border_style (GtkCssStyleProperty *property,
348 GtkCssParser *parser,
353 if (!_gtk_css_parser_try_enum (parser, GTK_TYPE_BORDER_STYLE, &value))
355 _gtk_css_parser_error (parser, "unknown value for property");
359 return _gtk_css_value_new_from_enum (GTK_TYPE_BORDER_STYLE, value);
363 parse_css_area (GtkCssStyleProperty *property,
364 GtkCssParser *parser,
369 if (!_gtk_css_parser_try_enum (parser, GTK_TYPE_CSS_AREA, &value))
371 _gtk_css_parser_error (parser, "unknown value for property");
375 return _gtk_css_value_new_from_enum (GTK_TYPE_CSS_AREA, value);
379 bindings_value_parse (GtkCssStyleProperty *property,
380 GtkCssParser *parser,
384 GtkBindingSet *binding_set;
387 array = g_ptr_array_new ();
390 name = _gtk_css_parser_try_ident (parser, TRUE);
393 _gtk_css_parser_error (parser, "Not a valid binding name");
394 g_ptr_array_free (array, TRUE);
398 binding_set = gtk_binding_set_find (name);
402 _gtk_css_parser_error (parser, "No binding set named '%s'", name);
407 g_ptr_array_add (array, binding_set);
410 while (_gtk_css_parser_try (parser, ",", TRUE));
412 return _gtk_css_value_new_take_binding_sets (array);
416 bindings_value_print (GtkCssStyleProperty *property,
417 const GtkCssValue *value,
423 array = _gtk_css_value_get_boxed (value);
425 for (i = 0; i < array->len; i++)
427 GtkBindingSet *binding_set = g_ptr_array_index (array, i);
430 g_string_append (string, ", ");
431 g_string_append (string, binding_set->set_name);
436 shadow_value_parse (GtkCssStyleProperty *property,
437 GtkCssParser *parser,
440 return _gtk_css_shadow_value_parse (parser);
444 shadow_value_compute (GtkCssStyleProperty *property,
445 GtkStyleContext *context,
446 GtkCssValue *specified)
448 return _gtk_css_shadow_value_compute (specified, context);
452 border_corner_radius_value_parse (GtkCssStyleProperty *property,
453 GtkCssParser *parser,
456 GtkCssBorderCornerRadius corner;
458 if (!_gtk_css_parser_read_number (parser,
460 GTK_CSS_POSITIVE_ONLY
461 | GTK_CSS_PARSE_PERCENT
462 | GTK_CSS_NUMBER_AS_PIXELS
463 | GTK_CSS_PARSE_LENGTH))
466 if (!_gtk_css_parser_has_number (parser))
467 corner.vertical = corner.horizontal;
468 else if (!_gtk_css_parser_read_number (parser,
470 GTK_CSS_POSITIVE_ONLY
471 | GTK_CSS_PARSE_PERCENT
472 | GTK_CSS_NUMBER_AS_PIXELS
473 | GTK_CSS_PARSE_LENGTH))
476 return _gtk_css_value_new_from_border_corner_radius (&corner);
480 border_corner_radius_value_print (GtkCssStyleProperty *property,
481 const GtkCssValue *value,
484 const GtkCssBorderCornerRadius *corner;
486 corner = _gtk_css_value_get_border_corner_radius (value);
488 _gtk_css_number_print (&corner->horizontal, string);
490 if (!_gtk_css_number_equal (&corner->horizontal, &corner->vertical))
492 g_string_append_c (string, ' ');
493 _gtk_css_number_print (&corner->vertical, string);
498 css_image_value_parse (GtkCssStyleProperty *property,
499 GtkCssParser *parser,
504 if (_gtk_css_parser_try (parser, "none", TRUE))
508 image = _gtk_css_image_new_parse (parser, base);
513 return _gtk_css_value_new_take_image (image);
517 css_image_value_print (GtkCssStyleProperty *property,
518 const GtkCssValue *value,
521 GtkCssImage *image = _gtk_css_value_get_image (value);
524 _gtk_css_image_print (image, string);
526 g_string_append (string, "none");
530 css_image_value_compute (GtkCssStyleProperty *property,
531 GtkStyleContext *context,
532 GtkCssValue *specified)
534 GtkCssImage *image, *computed;
536 image = _gtk_css_value_get_image (specified);
539 return _gtk_css_value_ref (specified);
541 computed = _gtk_css_image_compute (image, context);
543 if (computed == image)
545 g_object_unref (computed);
546 return _gtk_css_value_ref (specified);
549 return _gtk_css_value_new_take_image (computed);
553 css_image_value_query (GtkCssStyleProperty *property,
554 const GtkCssValue *css_value,
557 GtkCssImage *image = _gtk_css_value_get_image (css_value);
558 cairo_pattern_t *pattern;
559 cairo_surface_t *surface;
560 cairo_matrix_t matrix;
562 g_value_init (value, CAIRO_GOBJECT_TYPE_PATTERN);
564 if (GTK_IS_CSS_IMAGE_GRADIENT (image))
565 g_value_set_boxed (value, GTK_CSS_IMAGE_GRADIENT (image)->pattern);
566 else if (image != NULL)
568 double width, height;
570 /* the 100, 100 is rather random */
571 _gtk_css_image_get_concrete_size (image, 0, 0, 100, 100, &width, &height);
572 surface = _gtk_css_image_get_surface (image, NULL, width, height);
573 pattern = cairo_pattern_create_for_surface (surface);
574 cairo_matrix_init_scale (&matrix, width, height);
575 cairo_pattern_set_matrix (pattern, &matrix);
576 cairo_surface_destroy (surface);
577 g_value_take_boxed (value, pattern);
582 css_image_value_assign (GtkCssStyleProperty *property,
585 g_warning ("FIXME: assigning images is not implemented");
586 return _gtk_css_value_new_take_image (NULL);
590 font_size_parse (GtkCssStyleProperty *property,
591 GtkCssParser *parser,
596 if (!_gtk_css_parser_try_double (parser, &d))
598 _gtk_css_parser_error (parser, "Expected a number");
602 return _gtk_css_value_new_from_double (d);
606 outline_parse (GtkCssStyleProperty *property,
607 GtkCssParser *parser,
612 if (!_gtk_css_parser_try_int (parser, &i))
614 _gtk_css_parser_error (parser, "Expected an integer");
618 return _gtk_css_value_new_from_int (i);
622 border_image_repeat_parse (GtkCssStyleProperty *property,
623 GtkCssParser *parser,
626 GValue value = G_VALUE_INIT;
629 g_value_init (&value, GTK_TYPE_CSS_BORDER_IMAGE_REPEAT);
630 if (!_gtk_css_style_parse_value (&value, parser, base))
632 g_value_unset (&value);
636 result = _gtk_css_value_new_from_gvalue (&value);
637 g_value_unset (&value);
643 border_image_slice_parse (GtkCssStyleProperty *property,
644 GtkCssParser *parser,
647 GValue value = G_VALUE_INIT;
650 g_value_init (&value, GTK_TYPE_BORDER);
651 if (!_gtk_css_style_parse_value (&value, parser, base))
653 g_value_unset (&value);
657 result = _gtk_css_value_new_from_gvalue (&value);
658 g_value_unset (&value);
664 border_image_width_parse (GtkCssStyleProperty *property,
665 GtkCssParser *parser,
668 GValue value = G_VALUE_INIT;
671 g_value_init (&value, GTK_TYPE_BORDER);
672 if (!_gtk_css_style_parse_value (&value, parser, base))
674 g_value_unset (&value);
678 result = _gtk_css_value_new_from_gvalue (&value);
679 g_value_unset (&value);
685 engine_parse (GtkCssStyleProperty *property,
686 GtkCssParser *parser,
689 GtkThemingEngine *engine;
692 if (_gtk_css_parser_try (parser, "none", TRUE))
693 return _gtk_css_value_new_from_theming_engine (gtk_theming_engine_load (NULL));
695 str = _gtk_css_parser_try_ident (parser, TRUE);
698 _gtk_css_parser_error (parser, "Expected a valid theme name");
702 engine = gtk_theming_engine_load (str);
706 _gtk_css_parser_error (parser, "Theming engine '%s' not found", str);
713 return _gtk_css_value_new_from_theming_engine (engine);
717 transition_parse (GtkCssStyleProperty *property,
718 GtkCssParser *parser,
721 GValue value = G_VALUE_INIT;
724 g_value_init (&value, GTK_TYPE_ANIMATION_DESCRIPTION);
725 if (!_gtk_css_style_parse_value (&value, parser, base))
727 g_value_unset (&value);
731 result = _gtk_css_value_new_from_gvalue (&value);
732 g_value_unset (&value);
738 parse_margin (GtkCssStyleProperty *property,
739 GtkCssParser *parser,
742 return _gtk_css_number_value_parse (parser,
743 GTK_CSS_NUMBER_AS_PIXELS
744 | GTK_CSS_PARSE_LENGTH);
748 compute_margin (GtkCssStyleProperty *property,
749 GtkStyleContext *context,
750 GtkCssValue *specified)
752 return _gtk_css_number_value_compute (specified, context);
756 parse_padding (GtkCssStyleProperty *property,
757 GtkCssParser *parser,
760 return _gtk_css_number_value_parse (parser,
761 GTK_CSS_POSITIVE_ONLY
762 | GTK_CSS_NUMBER_AS_PIXELS
763 | GTK_CSS_PARSE_LENGTH);
767 compute_padding (GtkCssStyleProperty *property,
768 GtkStyleContext *context,
769 GtkCssValue *specified)
771 return _gtk_css_number_value_compute (specified, context);
775 parse_border_width (GtkCssStyleProperty *property,
776 GtkCssParser *parser,
779 return _gtk_css_number_value_parse (parser,
780 GTK_CSS_POSITIVE_ONLY
781 | GTK_CSS_NUMBER_AS_PIXELS
782 | GTK_CSS_PARSE_LENGTH);
786 compute_border_width (GtkCssStyleProperty *property,
787 GtkStyleContext *context,
788 GtkCssValue *specified)
790 GtkCssStyleProperty *style;
791 GtkBorderStyle border_style;
793 /* The -1 is magic that is only true because we register the style
794 * properties directly after the width properties.
796 style = _gtk_css_style_property_lookup_by_id (_gtk_css_style_property_get_id (property) - 1);
798 border_style = _gtk_css_value_get_border_style (_gtk_style_context_peek_property (context, _gtk_style_property_get_name (GTK_STYLE_PROPERTY (style))));
800 if (border_style == GTK_BORDER_STYLE_NONE ||
801 border_style == GTK_BORDER_STYLE_HIDDEN)
802 return _gtk_css_number_value_new (0, GTK_CSS_PX);
804 return _gtk_css_number_value_compute (specified, context);
808 background_repeat_value_parse (GtkCssStyleProperty *property,
809 GtkCssParser *parser,
812 int repeat, vertical;
814 if (!_gtk_css_parser_try_enum (parser, GTK_TYPE_CSS_BACKGROUND_REPEAT, &repeat))
816 _gtk_css_parser_error (parser, "Not a valid value");
820 if (repeat <= GTK_CSS_BACKGROUND_REPEAT_MASK)
822 if (_gtk_css_parser_try_enum (parser, GTK_TYPE_CSS_BACKGROUND_REPEAT, &vertical))
824 if (vertical >= GTK_CSS_BACKGROUND_REPEAT_MASK)
826 _gtk_css_parser_error (parser, "Not a valid 2nd value");
830 repeat |= vertical << GTK_CSS_BACKGROUND_REPEAT_SHIFT;
833 repeat |= repeat << GTK_CSS_BACKGROUND_REPEAT_SHIFT;
836 return _gtk_css_value_new_from_enum (GTK_TYPE_CSS_BACKGROUND_REPEAT, repeat);
840 background_repeat_value_print (GtkCssStyleProperty *property,
841 const GtkCssValue *value,
844 GEnumClass *enum_class;
845 GEnumValue *enum_value;
846 GtkCssBackgroundRepeat repeat;
848 repeat = _gtk_css_value_get_enum (value);
849 enum_class = g_type_class_ref (GTK_TYPE_CSS_BACKGROUND_REPEAT);
850 enum_value = g_enum_get_value (enum_class, repeat);
852 /* only triggers for 'repeat-x' and 'repeat-y' */
854 g_string_append (string, enum_value->value_nick);
857 enum_value = g_enum_get_value (enum_class, GTK_CSS_BACKGROUND_HORIZONTAL (repeat));
858 g_string_append (string, enum_value->value_nick);
860 if (GTK_CSS_BACKGROUND_HORIZONTAL (repeat) != GTK_CSS_BACKGROUND_VERTICAL (repeat))
862 enum_value = g_enum_get_value (enum_class, GTK_CSS_BACKGROUND_VERTICAL (repeat));
863 g_string_append (string, " ");
864 g_string_append (string, enum_value->value_nick);
868 g_type_class_unref (enum_class);
872 background_size_parse (GtkCssStyleProperty *property,
873 GtkCssParser *parser,
876 GtkCssBackgroundSize size = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), FALSE, FALSE};
878 if (_gtk_css_parser_try (parser, "cover", TRUE))
880 else if (_gtk_css_parser_try (parser, "contain", TRUE))
884 if (_gtk_css_parser_try (parser, "auto", TRUE))
885 _gtk_css_number_init (&size.width, 0, GTK_CSS_PX);
886 else if (!_gtk_css_parser_read_number (parser,
888 GTK_CSS_POSITIVE_ONLY
889 | GTK_CSS_PARSE_PERCENT
890 | GTK_CSS_PARSE_LENGTH))
893 if (_gtk_css_parser_try (parser, "auto", TRUE))
894 _gtk_css_number_init (&size.height, 0, GTK_CSS_PX);
895 else if (_gtk_css_parser_has_number (parser))
897 if (!_gtk_css_parser_read_number (parser,
899 GTK_CSS_POSITIVE_ONLY
900 | GTK_CSS_PARSE_PERCENT
901 | GTK_CSS_PARSE_LENGTH))
905 _gtk_css_number_init (&size.height, 0, GTK_CSS_PX);
908 return _gtk_css_value_new_from_background_size (&size);
912 background_size_print (GtkCssStyleProperty *property,
913 const GtkCssValue *value,
916 const GtkCssBackgroundSize *size = _gtk_css_value_get_background_size (value);
919 g_string_append (string, "cover");
920 else if (size->contain)
921 g_string_append (string, "contain");
924 if (size->width.value == 0)
925 g_string_append (string, "auto");
927 _gtk_css_number_print (&size->width, string);
929 if (size->height.value != 0)
931 g_string_append (string, " ");
932 _gtk_css_number_print (&size->height, string);
938 background_size_compute (GtkCssStyleProperty *property,
939 GtkStyleContext *context,
940 GtkCssValue *specified)
942 const GtkCssBackgroundSize *ssize = _gtk_css_value_get_background_size (specified);
943 GtkCssBackgroundSize csize;
946 csize.cover = ssize->cover;
947 csize.contain = ssize->contain;
948 changed = _gtk_css_number_compute (&csize.width,
951 changed |= _gtk_css_number_compute (&csize.height,
955 return _gtk_css_value_new_from_background_size (&csize);
956 return _gtk_css_value_ref (specified);
960 background_position_parse (GtkCssStyleProperty *property,
961 GtkCssParser *parser,
964 static const struct {
970 { "left", 0, TRUE, FALSE },
971 { "right", 100, TRUE, FALSE },
972 { "center", 50, TRUE, TRUE },
973 { "top", 0, FALSE, TRUE },
974 { "bottom", 100, FALSE, TRUE },
975 { NULL , 0, TRUE, FALSE }, /* used for numbers */
976 { NULL , 50, TRUE, TRUE } /* used for no value */
978 GtkCssBackgroundPosition pos;
979 GtkCssNumber *missing;
982 for (first = 0; names[first].name != NULL; first++)
984 if (_gtk_css_parser_try (parser, names[first].name, TRUE))
986 if (names[first].horizontal)
988 _gtk_css_number_init (&pos.x, names[first].percentage, GTK_CSS_PERCENT);
993 _gtk_css_number_init (&pos.y, names[first].percentage, GTK_CSS_PERCENT);
999 if (names[first].name == NULL)
1002 if (!_gtk_css_parser_read_number (parser,
1004 GTK_CSS_PARSE_PERCENT
1005 | GTK_CSS_PARSE_LENGTH))
1009 for (second = 0; names[second].name != NULL; second++)
1011 if (_gtk_css_parser_try (parser, names[second].name, TRUE))
1013 _gtk_css_number_init (missing, names[second].percentage, GTK_CSS_PERCENT);
1018 if (names[second].name == NULL)
1020 if (_gtk_css_parser_has_number (parser))
1022 if (missing != &pos.y)
1024 _gtk_css_parser_error (parser, "Invalid combination of values");
1027 if (!_gtk_css_parser_read_number (parser,
1029 GTK_CSS_PARSE_PERCENT
1030 | GTK_CSS_PARSE_LENGTH))
1036 _gtk_css_number_init (missing, 50, GTK_CSS_PERCENT);
1041 if ((names[first].horizontal && !names[second].vertical) ||
1042 (!names[first].horizontal && !names[second].horizontal))
1044 _gtk_css_parser_error (parser, "Invalid combination of values");
1049 return _gtk_css_value_new_from_background_position (&pos);
1053 background_position_print (GtkCssStyleProperty *property,
1054 const GtkCssValue *value,
1057 const GtkCssBackgroundPosition *pos = _gtk_css_value_get_background_position (value);
1058 static const GtkCssNumber center = GTK_CSS_NUMBER_INIT (50, GTK_CSS_PERCENT);
1059 static const struct {
1062 GtkCssNumber number;
1064 { "left", "top", GTK_CSS_NUMBER_INIT (0, GTK_CSS_PERCENT) },
1065 { "right", "bottom", GTK_CSS_NUMBER_INIT (100, GTK_CSS_PERCENT) }
1069 if (_gtk_css_number_equal (&pos->x, ¢er))
1071 if (_gtk_css_number_equal (&pos->y, ¢er))
1073 g_string_append (string, "center");
1079 for (i = 0; i < G_N_ELEMENTS (values); i++)
1081 if (_gtk_css_number_equal (&pos->x, &values[i].number))
1083 g_string_append (string, values[i].x_name);
1087 if (i == G_N_ELEMENTS (values))
1088 _gtk_css_number_print (&pos->x, string);
1090 if (_gtk_css_number_equal (&pos->y, ¢er))
1093 g_string_append_c (string, ' ');
1096 for (i = 0; i < G_N_ELEMENTS (values); i++)
1098 if (_gtk_css_number_equal (&pos->y, &values[i].number))
1100 g_string_append (string, values[i].y_name);
1104 if (i == G_N_ELEMENTS (values))
1106 if (_gtk_css_number_equal (&pos->x, ¢er))
1107 g_string_append (string, "center ");
1108 _gtk_css_number_print (&pos->y, string);
1112 static GtkCssValue *
1113 background_position_compute (GtkCssStyleProperty *property,
1114 GtkStyleContext *context,
1115 GtkCssValue *specified)
1117 const GtkCssBackgroundPosition *spos = _gtk_css_value_get_background_position (specified);
1118 GtkCssBackgroundPosition cpos;
1121 changed = _gtk_css_number_compute (&cpos.x,
1124 changed |= _gtk_css_number_compute (&cpos.y,
1128 return _gtk_css_value_new_from_background_position (&cpos);
1129 return _gtk_css_value_ref (specified);
1132 /*** REGISTRATION ***/
1134 static GtkSymbolicColor *
1135 gtk_symbolic_color_new_rgba (double red,
1140 GdkRGBA rgba = { red, green, blue, alpha };
1142 return gtk_symbolic_color_new_literal (&rgba);
1146 _gtk_css_style_property_init_properties (void)
1148 char *default_font_family[] = { "Sans", NULL };
1149 GtkCssBackgroundSize default_background_size = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), FALSE, FALSE };
1150 GtkCssBackgroundPosition default_background_position = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PERCENT), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PERCENT)};
1151 GtkCssBorderCornerRadius no_corner_radius = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX) };
1152 GtkBorder border_of_ones = { 1, 1, 1, 1 };
1153 GtkCssBorderImageRepeat border_image_repeat = { GTK_CSS_REPEAT_STYLE_STRETCH, GTK_CSS_REPEAT_STYLE_STRETCH };
1155 /* Initialize "color" and "font-size" first,
1156 * so that when computing values later they are
1157 * done first. That way, 'currentColor' and font
1158 * sizes in em can be looked up properly */
1159 gtk_css_style_property_register ("color",
1161 GTK_STYLE_PROPERTY_INHERIT,
1164 color_property_compute,
1168 _gtk_css_value_new_take_symbolic_color (
1169 gtk_symbolic_color_new_rgba (1, 1, 1, 1)));
1170 gtk_css_style_property_register ("font-size",
1172 GTK_STYLE_PROPERTY_INHERIT,
1179 _gtk_css_value_new_from_double (10.0));
1181 /* properties that aren't referenced when computing values
1183 gtk_css_style_property_register ("background-color",
1192 _gtk_css_value_new_take_symbolic_color (
1193 gtk_symbolic_color_new_rgba (0, 0, 0, 0)));
1195 gtk_css_style_property_register ("font-family",
1197 GTK_STYLE_PROPERTY_INHERIT,
1199 font_family_value_print,
1204 _gtk_css_value_new_take_strv (g_strdupv (default_font_family)));
1205 gtk_css_style_property_register ("font-style",
1207 GTK_STYLE_PROPERTY_INHERIT,
1214 _gtk_css_value_new_from_enum (PANGO_TYPE_STYLE,
1215 PANGO_STYLE_NORMAL));
1216 gtk_css_style_property_register ("font-variant",
1218 GTK_STYLE_PROPERTY_INHERIT,
1219 parse_pango_variant,
1225 _gtk_css_value_new_from_enum (PANGO_TYPE_VARIANT,
1226 PANGO_VARIANT_NORMAL));
1227 /* xxx: need to parse this properly, ie parse the numbers */
1228 gtk_css_style_property_register ("font-weight",
1230 GTK_STYLE_PROPERTY_INHERIT,
1237 _gtk_css_value_new_from_enum (PANGO_TYPE_WEIGHT,
1238 PANGO_WEIGHT_NORMAL));
1240 gtk_css_style_property_register ("text-shadow",
1242 GTK_STYLE_PROPERTY_INHERIT,
1245 shadow_value_compute,
1249 _gtk_css_shadow_value_new_none ());
1251 gtk_css_style_property_register ("icon-shadow",
1253 GTK_STYLE_PROPERTY_INHERIT,
1256 shadow_value_compute,
1260 _gtk_css_shadow_value_new_none ());
1262 gtk_css_style_property_register ("box-shadow",
1267 shadow_value_compute,
1271 _gtk_css_shadow_value_new_none ());
1273 gtk_css_style_property_register ("margin-top",
1279 query_length_as_int,
1280 assign_length_from_int,
1282 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1283 gtk_css_style_property_register ("margin-left",
1289 query_length_as_int,
1290 assign_length_from_int,
1292 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1293 gtk_css_style_property_register ("margin-bottom",
1299 query_length_as_int,
1300 assign_length_from_int,
1302 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1303 gtk_css_style_property_register ("margin-right",
1309 query_length_as_int,
1310 assign_length_from_int,
1312 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1313 gtk_css_style_property_register ("padding-top",
1319 query_length_as_int,
1320 assign_length_from_int,
1322 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1323 gtk_css_style_property_register ("padding-left",
1329 query_length_as_int,
1330 assign_length_from_int,
1332 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1333 gtk_css_style_property_register ("padding-bottom",
1339 query_length_as_int,
1340 assign_length_from_int,
1342 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1343 gtk_css_style_property_register ("padding-right",
1349 query_length_as_int,
1350 assign_length_from_int,
1352 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1353 /* IMPORTANT: compute_border_width() requires that the border-width
1354 * properties be immeditaly followed by the border-style properties
1356 gtk_css_style_property_register ("border-top-style",
1357 GTK_TYPE_BORDER_STYLE,
1365 _gtk_css_value_new_from_border_style (GTK_BORDER_STYLE_NONE));
1366 gtk_css_style_property_register ("border-top-width",
1371 compute_border_width,
1372 query_length_as_int,
1373 assign_length_from_int,
1375 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1376 gtk_css_style_property_register ("border-left-style",
1377 GTK_TYPE_BORDER_STYLE,
1385 _gtk_css_value_new_from_border_style (GTK_BORDER_STYLE_NONE));
1386 gtk_css_style_property_register ("border-left-width",
1391 compute_border_width,
1392 query_length_as_int,
1393 assign_length_from_int,
1395 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1396 gtk_css_style_property_register ("border-bottom-style",
1397 GTK_TYPE_BORDER_STYLE,
1405 _gtk_css_value_new_from_border_style (GTK_BORDER_STYLE_NONE));
1406 gtk_css_style_property_register ("border-bottom-width",
1411 compute_border_width,
1412 query_length_as_int,
1413 assign_length_from_int,
1415 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1416 gtk_css_style_property_register ("border-right-style",
1417 GTK_TYPE_BORDER_STYLE,
1425 _gtk_css_value_new_from_border_style (GTK_BORDER_STYLE_NONE));
1426 gtk_css_style_property_register ("border-right-width",
1431 compute_border_width,
1432 query_length_as_int,
1433 assign_length_from_int,
1435 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1437 gtk_css_style_property_register ("border-top-left-radius",
1438 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1440 border_corner_radius_value_parse,
1441 border_corner_radius_value_print,
1446 _gtk_css_value_new_from_border_corner_radius (&no_corner_radius));
1447 gtk_css_style_property_register ("border-top-right-radius",
1448 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1450 border_corner_radius_value_parse,
1451 border_corner_radius_value_print,
1456 _gtk_css_value_new_from_border_corner_radius (&no_corner_radius));
1457 gtk_css_style_property_register ("border-bottom-right-radius",
1458 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1460 border_corner_radius_value_parse,
1461 border_corner_radius_value_print,
1466 _gtk_css_value_new_from_border_corner_radius (&no_corner_radius));
1467 gtk_css_style_property_register ("border-bottom-left-radius",
1468 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1470 border_corner_radius_value_parse,
1471 border_corner_radius_value_print,
1476 _gtk_css_value_new_from_border_corner_radius (&no_corner_radius));
1478 gtk_css_style_property_register ("outline-style",
1479 GTK_TYPE_BORDER_STYLE,
1487 _gtk_css_value_new_from_border_style (GTK_BORDER_STYLE_NONE));
1488 gtk_css_style_property_register ("outline-width",
1493 compute_border_width,
1494 query_length_as_int,
1495 assign_length_from_int,
1497 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1498 gtk_css_style_property_register ("outline-offset",
1507 _gtk_css_value_new_from_int (0));
1509 gtk_css_style_property_register ("background-clip",
1518 _gtk_css_value_new_from_enum (GTK_TYPE_CSS_AREA, GTK_CSS_AREA_BORDER_BOX));
1519 gtk_css_style_property_register ("background-origin",
1528 _gtk_css_value_new_from_enum (GTK_TYPE_CSS_AREA, GTK_CSS_AREA_PADDING_BOX));
1529 gtk_css_style_property_register ("background-size",
1532 background_size_parse,
1533 background_size_print,
1534 background_size_compute,
1538 _gtk_css_value_new_from_background_size (&default_background_size));
1539 gtk_css_style_property_register ("background-position",
1542 background_position_parse,
1543 background_position_print,
1544 background_position_compute,
1548 _gtk_css_value_new_from_background_position (&default_background_position));
1550 gtk_css_style_property_register ("border-top-color",
1559 _gtk_css_value_new_take_symbolic_color (
1560 gtk_symbolic_color_ref (
1561 _gtk_symbolic_color_get_current_color ())));
1562 gtk_css_style_property_register ("border-right-color",
1571 _gtk_css_value_new_take_symbolic_color (
1572 gtk_symbolic_color_ref (
1573 _gtk_symbolic_color_get_current_color ())));
1574 gtk_css_style_property_register ("border-bottom-color",
1583 _gtk_css_value_new_take_symbolic_color (
1584 gtk_symbolic_color_ref (
1585 _gtk_symbolic_color_get_current_color ())));
1586 gtk_css_style_property_register ("border-left-color",
1595 _gtk_css_value_new_take_symbolic_color (
1596 gtk_symbolic_color_ref (
1597 _gtk_symbolic_color_get_current_color ())));
1598 gtk_css_style_property_register ("outline-color",
1607 _gtk_css_value_new_take_symbolic_color (
1608 gtk_symbolic_color_ref (
1609 _gtk_symbolic_color_get_current_color ())));
1611 gtk_css_style_property_register ("background-repeat",
1612 GTK_TYPE_CSS_BACKGROUND_REPEAT,
1614 background_repeat_value_parse,
1615 background_repeat_value_print,
1620 _gtk_css_value_new_from_enum (GTK_TYPE_CSS_BACKGROUND_REPEAT,
1621 GTK_CSS_BACKGROUND_REPEAT |
1622 (GTK_CSS_BACKGROUND_REPEAT << GTK_CSS_BACKGROUND_REPEAT_SHIFT)));
1623 gtk_css_style_property_register ("background-image",
1624 CAIRO_GOBJECT_TYPE_PATTERN,
1626 css_image_value_parse,
1627 css_image_value_print,
1628 css_image_value_compute,
1629 css_image_value_query,
1630 css_image_value_assign,
1632 _gtk_css_value_new_take_image (NULL));
1634 gtk_css_style_property_register ("border-image-source",
1635 CAIRO_GOBJECT_TYPE_PATTERN,
1637 css_image_value_parse,
1638 css_image_value_print,
1639 css_image_value_compute,
1640 css_image_value_query,
1641 css_image_value_assign,
1643 _gtk_css_value_new_take_image (NULL));
1644 gtk_css_style_property_register ("border-image-repeat",
1645 GTK_TYPE_CSS_BORDER_IMAGE_REPEAT,
1647 border_image_repeat_parse,
1653 _gtk_css_value_new_from_border_image_repeat (&border_image_repeat));
1655 /* XXX: The initial value is wrong, it should be 100% */
1656 gtk_css_style_property_register ("border-image-slice",
1659 border_image_slice_parse,
1665 _gtk_css_value_new_from_boxed (GTK_TYPE_BORDER, &border_of_ones));
1666 gtk_css_style_property_register ("border-image-width",
1669 border_image_width_parse,
1675 _gtk_css_value_new_from_boxed (GTK_TYPE_BORDER, NULL));
1676 gtk_css_style_property_register ("engine",
1677 GTK_TYPE_THEMING_ENGINE,
1685 _gtk_css_value_new_from_theming_engine (gtk_theming_engine_load (NULL)));
1686 gtk_css_style_property_register ("transition",
1687 GTK_TYPE_ANIMATION_DESCRIPTION,
1695 _gtk_css_value_new_from_boxed (GTK_TYPE_ANIMATION_DESCRIPTION, NULL));
1697 /* Private property holding the binding sets */
1698 gtk_css_style_property_register ("gtk-key-bindings",
1701 bindings_value_parse,
1702 bindings_value_print,
1707 _gtk_css_value_new_take_binding_sets (NULL));