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 "gtkcssarrayvalueprivate.h"
44 #include "gtkcssimagegradientprivate.h"
45 #include "gtkcssimageprivate.h"
46 #include "gtkcssimagevalueprivate.h"
47 #include "gtkcssenumvalueprivate.h"
48 #include "gtkcssnumbervalueprivate.h"
49 #include "gtkcssrgbavalueprivate.h"
50 #include "gtkcssshadowvalueprivate.h"
51 #include "gtkcssstringvalueprivate.h"
52 #include "gtksymboliccolorprivate.h"
53 #include "gtkthemingengine.h"
54 #include "gtktypebuiltins.h"
55 #include "gtkwin32themeprivate.h"
57 /*** REGISTRATION ***/
60 gtk_css_style_property_register (const char * name,
62 GtkStylePropertyFlags flags,
63 GtkCssStylePropertyParseFunc parse_value,
64 GtkCssStylePropertyPrintFunc print_value,
65 GtkCssStylePropertyComputeFunc compute_value,
66 GtkCssStylePropertyQueryFunc query_value,
67 GtkCssStylePropertyAssignFunc assign_value,
68 GtkCssStylePropertyEqualFunc equal_func,
69 GtkCssValue * initial_value)
71 GtkCssStyleProperty *node;
73 g_assert (initial_value != NULL);
74 g_assert (parse_value != NULL);
75 g_assert (value_type == G_TYPE_NONE || query_value != NULL);
76 g_assert (value_type == G_TYPE_NONE || assign_value != NULL);
78 node = g_object_new (GTK_TYPE_CSS_STYLE_PROPERTY,
79 "value-type", value_type,
80 "inherit", (flags & GTK_STYLE_PROPERTY_INHERIT) ? TRUE : FALSE,
81 "initial-value", initial_value,
85 node->parse_value = parse_value;
87 node->print_value = print_value;
89 node->compute_value = compute_value;
90 node->query_value = query_value;
91 node->assign_value = assign_value;
93 node->equal_func = equal_func;
95 _gtk_css_value_unref (initial_value);
98 /*** IMPLEMENTATIONS ***/
101 query_simple (GtkCssStyleProperty *property,
102 const GtkCssValue *css_value,
105 _gtk_css_value_init_gvalue (css_value, value);
109 assign_simple (GtkCssStyleProperty *property,
112 return _gtk_css_value_new_from_gvalue (value);
116 query_length_as_int (GtkCssStyleProperty *property,
117 const GtkCssValue *css_value,
120 g_value_init (value, G_TYPE_INT);
121 g_value_set_int (value, round (_gtk_css_number_value_get (css_value, 100)));
125 assign_length_from_int (GtkCssStyleProperty *property,
128 return _gtk_css_number_value_new (g_value_get_int (value), GTK_CSS_PX);
132 color_parse (GtkCssStyleProperty *property,
133 GtkCssParser *parser,
136 GtkSymbolicColor *symbolic;
138 if (_gtk_css_parser_try (parser, "currentcolor", TRUE))
140 symbolic = gtk_symbolic_color_ref (_gtk_symbolic_color_get_current_color ());
144 symbolic = _gtk_css_parser_read_symbolic_color (parser);
145 if (symbolic == NULL)
149 return _gtk_css_value_new_take_symbolic_color (symbolic);
153 color_compute (GtkCssStyleProperty *property,
154 GtkStyleContext *context,
155 GtkCssValue *specified)
157 return _gtk_css_rgba_value_compute_from_symbolic (specified,
158 _gtk_css_style_property_get_initial_value (property),
164 color_property_compute (GtkCssStyleProperty *property,
165 GtkStyleContext *context,
166 GtkCssValue *specified)
170 value = _gtk_css_rgba_value_compute_from_symbolic (specified,
171 _gtk_css_style_property_get_initial_value (property),
174 _gtk_css_rgba_value_get_rgba (value);
179 color_query (GtkCssStyleProperty *property,
180 const GtkCssValue *css_value,
183 g_value_init (value, GDK_TYPE_RGBA);
184 g_value_set_boxed (value, _gtk_css_rgba_value_get_rgba (css_value));
188 color_assign (GtkCssStyleProperty *property,
191 return _gtk_css_rgba_value_new_from_rgba (g_value_get_boxed (value));
195 font_family_parse (GtkCssStyleProperty *property,
196 GtkCssParser *parser,
203 /* We don't special case generic families. Pango should do
206 names = g_ptr_array_new ();
209 name = _gtk_css_parser_try_ident (parser, TRUE);
212 GString *string = g_string_new (name);
214 while ((name = _gtk_css_parser_try_ident (parser, TRUE)))
216 g_string_append_c (string, ' ');
217 g_string_append (string, name);
220 name = g_string_free (string, FALSE);
224 name = _gtk_css_parser_read_string (parser);
227 g_ptr_array_free (names, TRUE);
232 g_ptr_array_add (names, _gtk_css_string_value_new_take (name));
233 } while (_gtk_css_parser_try (parser, ",", TRUE));
235 result = _gtk_css_array_value_new ((GtkCssValue **) names->pdata, names->len);
236 g_ptr_array_free (names, TRUE);
241 font_family_query (GtkCssStyleProperty *property,
242 const GtkCssValue *css_value,
248 array = g_ptr_array_new ();
250 for (i = 0; i < _gtk_css_array_value_get_n_values (css_value); i++)
252 g_ptr_array_add (array, g_strdup (_gtk_css_string_value_get (_gtk_css_array_value_get_nth (css_value, i))));
256 g_ptr_array_add (array, NULL);
258 g_value_init (value, G_TYPE_STRV);
259 g_value_set_boxed (value, g_ptr_array_free (array, FALSE));
263 font_family_assign (GtkCssStyleProperty *property,
266 const char **names = g_value_get_boxed (value);
270 array = g_ptr_array_new ();
272 for (names = g_value_get_boxed (value); *names; names++)
274 g_ptr_array_add (array, _gtk_css_string_value_new (*names));
277 result = _gtk_css_array_value_new ((GtkCssValue **) array->pdata, array->len);
278 g_ptr_array_free (array, TRUE);
283 parse_pango_style (GtkCssStyleProperty *property,
284 GtkCssParser *parser,
287 GtkCssValue *value = _gtk_css_font_style_value_try_parse (parser);
290 _gtk_css_parser_error (parser, "unknown value for property");
296 query_pango_style (GtkCssStyleProperty *property,
297 const GtkCssValue *css_value,
300 g_value_init (value, PANGO_TYPE_STYLE);
301 g_value_set_enum (value, _gtk_css_font_style_value_get (css_value));
305 assign_pango_style (GtkCssStyleProperty *property,
308 return _gtk_css_font_style_value_new (g_value_get_enum (value));
312 parse_pango_weight (GtkCssStyleProperty *property,
313 GtkCssParser *parser,
316 GtkCssValue *value = _gtk_css_font_weight_value_try_parse (parser);
319 _gtk_css_parser_error (parser, "unknown value for property");
325 query_pango_weight (GtkCssStyleProperty *property,
326 const GtkCssValue *css_value,
329 g_value_init (value, PANGO_TYPE_WEIGHT);
330 g_value_set_enum (value, _gtk_css_font_weight_value_get (css_value));
334 assign_pango_weight (GtkCssStyleProperty *property,
337 return _gtk_css_font_weight_value_new (g_value_get_enum (value));
341 parse_pango_variant (GtkCssStyleProperty *property,
342 GtkCssParser *parser,
345 GtkCssValue *value = _gtk_css_font_variant_value_try_parse (parser);
348 _gtk_css_parser_error (parser, "unknown value for property");
354 query_pango_variant (GtkCssStyleProperty *property,
355 const GtkCssValue *css_value,
358 g_value_init (value, PANGO_TYPE_VARIANT);
359 g_value_set_enum (value, _gtk_css_font_variant_value_get (css_value));
363 assign_pango_variant (GtkCssStyleProperty *property,
366 return _gtk_css_font_variant_value_new (g_value_get_enum (value));
370 parse_border_style (GtkCssStyleProperty *property,
371 GtkCssParser *parser,
374 GtkCssValue *value = _gtk_css_border_style_value_try_parse (parser);
377 _gtk_css_parser_error (parser, "unknown value for property");
383 query_border_style (GtkCssStyleProperty *property,
384 const GtkCssValue *css_value,
387 g_value_init (value, GTK_TYPE_BORDER_STYLE);
388 g_value_set_enum (value, _gtk_css_border_style_value_get (css_value));
392 assign_border_style (GtkCssStyleProperty *property,
395 return _gtk_css_border_style_value_new (g_value_get_enum (value));
399 parse_css_area (GtkCssStyleProperty *property,
400 GtkCssParser *parser,
403 GtkCssValue *value = _gtk_css_area_value_try_parse (parser);
406 _gtk_css_parser_error (parser, "unknown value for property");
412 bindings_value_parse (GtkCssStyleProperty *property,
413 GtkCssParser *parser,
417 GtkBindingSet *binding_set;
420 array = g_ptr_array_new ();
423 name = _gtk_css_parser_try_ident (parser, TRUE);
426 _gtk_css_parser_error (parser, "Not a valid binding name");
427 g_ptr_array_free (array, TRUE);
431 binding_set = gtk_binding_set_find (name);
435 _gtk_css_parser_error (parser, "No binding set named '%s'", name);
440 g_ptr_array_add (array, binding_set);
443 while (_gtk_css_parser_try (parser, ",", TRUE));
445 return _gtk_css_value_new_take_binding_sets (array);
449 bindings_value_print (GtkCssStyleProperty *property,
450 const GtkCssValue *value,
456 array = _gtk_css_value_get_boxed (value);
458 for (i = 0; i < array->len; i++)
460 GtkBindingSet *binding_set = g_ptr_array_index (array, i);
463 g_string_append (string, ", ");
464 g_string_append (string, binding_set->set_name);
469 shadow_value_parse (GtkCssStyleProperty *property,
470 GtkCssParser *parser,
473 return _gtk_css_shadow_value_parse (parser);
477 shadow_value_compute (GtkCssStyleProperty *property,
478 GtkStyleContext *context,
479 GtkCssValue *specified)
481 return _gtk_css_shadow_value_compute (specified, context);
485 border_corner_radius_value_parse (GtkCssStyleProperty *property,
486 GtkCssParser *parser,
489 GtkCssBorderCornerRadius corner;
491 if (!_gtk_css_parser_read_number (parser,
493 GTK_CSS_POSITIVE_ONLY
494 | GTK_CSS_PARSE_PERCENT
495 | GTK_CSS_NUMBER_AS_PIXELS
496 | GTK_CSS_PARSE_LENGTH))
499 if (!_gtk_css_parser_has_number (parser))
500 corner.vertical = corner.horizontal;
501 else if (!_gtk_css_parser_read_number (parser,
503 GTK_CSS_POSITIVE_ONLY
504 | GTK_CSS_PARSE_PERCENT
505 | GTK_CSS_NUMBER_AS_PIXELS
506 | GTK_CSS_PARSE_LENGTH))
509 return _gtk_css_value_new_from_border_corner_radius (&corner);
513 border_corner_radius_value_print (GtkCssStyleProperty *property,
514 const GtkCssValue *value,
517 const GtkCssBorderCornerRadius *corner;
519 corner = _gtk_css_value_get_border_corner_radius (value);
521 _gtk_css_number_print (&corner->horizontal, string);
523 if (!_gtk_css_number_equal (&corner->horizontal, &corner->vertical))
525 g_string_append_c (string, ' ');
526 _gtk_css_number_print (&corner->vertical, string);
531 css_image_value_parse (GtkCssStyleProperty *property,
532 GtkCssParser *parser,
537 if (_gtk_css_parser_try (parser, "none", TRUE))
541 image = _gtk_css_image_new_parse (parser, base);
546 return _gtk_css_image_value_new (image);
550 css_image_value_compute (GtkCssStyleProperty *property,
551 GtkStyleContext *context,
552 GtkCssValue *specified)
554 GtkCssImage *image, *computed;
556 image = _gtk_css_image_value_get_image (specified);
559 return _gtk_css_value_ref (specified);
561 computed = _gtk_css_image_compute (image, context);
563 if (computed == image)
565 g_object_unref (computed);
566 return _gtk_css_value_ref (specified);
569 return _gtk_css_image_value_new (computed);
573 css_image_value_query (GtkCssStyleProperty *property,
574 const GtkCssValue *css_value,
577 GtkCssImage *image = _gtk_css_image_value_get_image (css_value);
578 cairo_pattern_t *pattern;
579 cairo_surface_t *surface;
580 cairo_matrix_t matrix;
582 g_value_init (value, CAIRO_GOBJECT_TYPE_PATTERN);
584 if (GTK_IS_CSS_IMAGE_GRADIENT (image))
585 g_value_set_boxed (value, GTK_CSS_IMAGE_GRADIENT (image)->pattern);
586 else if (image != NULL)
588 double width, height;
590 /* the 100, 100 is rather random */
591 _gtk_css_image_get_concrete_size (image, 0, 0, 100, 100, &width, &height);
592 surface = _gtk_css_image_get_surface (image, NULL, width, height);
593 pattern = cairo_pattern_create_for_surface (surface);
594 cairo_matrix_init_scale (&matrix, width, height);
595 cairo_pattern_set_matrix (pattern, &matrix);
596 cairo_surface_destroy (surface);
597 g_value_take_boxed (value, pattern);
602 css_image_value_assign (GtkCssStyleProperty *property,
605 g_warning ("FIXME: assigning images is not implemented");
606 return _gtk_css_image_value_new (NULL);
610 font_size_parse (GtkCssStyleProperty *property,
611 GtkCssParser *parser,
616 if (!_gtk_css_parser_try_double (parser, &d))
618 _gtk_css_parser_error (parser, "Expected a number");
622 return _gtk_css_value_new_from_double (d);
626 outline_parse (GtkCssStyleProperty *property,
627 GtkCssParser *parser,
632 if (!_gtk_css_parser_try_int (parser, &i))
634 _gtk_css_parser_error (parser, "Expected an integer");
638 return _gtk_css_value_new_from_int (i);
642 border_image_repeat_parse (GtkCssStyleProperty *property,
643 GtkCssParser *parser,
646 GValue value = G_VALUE_INIT;
649 g_value_init (&value, GTK_TYPE_CSS_BORDER_IMAGE_REPEAT);
650 if (!_gtk_css_style_parse_value (&value, parser, base))
652 g_value_unset (&value);
656 result = _gtk_css_value_new_from_gvalue (&value);
657 g_value_unset (&value);
663 border_image_slice_parse (GtkCssStyleProperty *property,
664 GtkCssParser *parser,
667 GValue value = G_VALUE_INIT;
670 g_value_init (&value, GTK_TYPE_BORDER);
671 if (!_gtk_css_style_parse_value (&value, parser, base))
673 g_value_unset (&value);
677 result = _gtk_css_value_new_from_gvalue (&value);
678 g_value_unset (&value);
684 border_image_width_parse (GtkCssStyleProperty *property,
685 GtkCssParser *parser,
688 GValue value = G_VALUE_INIT;
691 g_value_init (&value, GTK_TYPE_BORDER);
692 if (!_gtk_css_style_parse_value (&value, parser, base))
694 g_value_unset (&value);
698 result = _gtk_css_value_new_from_gvalue (&value);
699 g_value_unset (&value);
705 engine_parse (GtkCssStyleProperty *property,
706 GtkCssParser *parser,
709 GtkThemingEngine *engine;
712 if (_gtk_css_parser_try (parser, "none", TRUE))
713 return _gtk_css_value_new_from_theming_engine (gtk_theming_engine_load (NULL));
715 str = _gtk_css_parser_try_ident (parser, TRUE);
718 _gtk_css_parser_error (parser, "Expected a valid theme name");
722 engine = gtk_theming_engine_load (str);
726 _gtk_css_parser_error (parser, "Theming engine '%s' not found", str);
733 return _gtk_css_value_new_from_theming_engine (engine);
737 transition_parse (GtkCssStyleProperty *property,
738 GtkCssParser *parser,
741 GValue value = G_VALUE_INIT;
744 g_value_init (&value, GTK_TYPE_ANIMATION_DESCRIPTION);
745 if (!_gtk_css_style_parse_value (&value, parser, base))
747 g_value_unset (&value);
751 result = _gtk_css_value_new_from_gvalue (&value);
752 g_value_unset (&value);
758 parse_margin (GtkCssStyleProperty *property,
759 GtkCssParser *parser,
762 return _gtk_css_number_value_parse (parser,
763 GTK_CSS_NUMBER_AS_PIXELS
764 | GTK_CSS_PARSE_LENGTH);
768 compute_margin (GtkCssStyleProperty *property,
769 GtkStyleContext *context,
770 GtkCssValue *specified)
772 return _gtk_css_number_value_compute (specified, context);
776 parse_padding (GtkCssStyleProperty *property,
777 GtkCssParser *parser,
780 return _gtk_css_number_value_parse (parser,
781 GTK_CSS_POSITIVE_ONLY
782 | GTK_CSS_NUMBER_AS_PIXELS
783 | GTK_CSS_PARSE_LENGTH);
787 compute_padding (GtkCssStyleProperty *property,
788 GtkStyleContext *context,
789 GtkCssValue *specified)
791 return _gtk_css_number_value_compute (specified, context);
795 parse_border_width (GtkCssStyleProperty *property,
796 GtkCssParser *parser,
799 return _gtk_css_number_value_parse (parser,
800 GTK_CSS_POSITIVE_ONLY
801 | GTK_CSS_NUMBER_AS_PIXELS
802 | GTK_CSS_PARSE_LENGTH);
806 compute_border_width (GtkCssStyleProperty *property,
807 GtkStyleContext *context,
808 GtkCssValue *specified)
810 GtkCssStyleProperty *style;
811 GtkBorderStyle border_style;
813 /* The -1 is magic that is only true because we register the style
814 * properties directly after the width properties.
816 style = _gtk_css_style_property_lookup_by_id (_gtk_css_style_property_get_id (property) - 1);
818 border_style = _gtk_css_border_style_value_get (_gtk_style_context_peek_property (context, _gtk_style_property_get_name (GTK_STYLE_PROPERTY (style))));
820 if (border_style == GTK_BORDER_STYLE_NONE ||
821 border_style == GTK_BORDER_STYLE_HIDDEN)
822 return _gtk_css_number_value_new (0, GTK_CSS_PX);
824 return _gtk_css_number_value_compute (specified, context);
828 background_repeat_value_parse (GtkCssStyleProperty *property,
829 GtkCssParser *parser,
832 int repeat, vertical;
834 if (!_gtk_css_parser_try_enum (parser, GTK_TYPE_CSS_BACKGROUND_REPEAT, &repeat))
836 _gtk_css_parser_error (parser, "Not a valid value");
840 if (repeat <= GTK_CSS_BACKGROUND_REPEAT_MASK)
842 if (_gtk_css_parser_try_enum (parser, GTK_TYPE_CSS_BACKGROUND_REPEAT, &vertical))
844 if (vertical >= GTK_CSS_BACKGROUND_REPEAT_MASK)
846 _gtk_css_parser_error (parser, "Not a valid 2nd value");
850 repeat |= vertical << GTK_CSS_BACKGROUND_REPEAT_SHIFT;
853 repeat |= repeat << GTK_CSS_BACKGROUND_REPEAT_SHIFT;
856 return _gtk_css_value_new_from_enum (GTK_TYPE_CSS_BACKGROUND_REPEAT, repeat);
860 background_repeat_value_print (GtkCssStyleProperty *property,
861 const GtkCssValue *value,
864 GEnumClass *enum_class;
865 GEnumValue *enum_value;
866 GtkCssBackgroundRepeat repeat;
868 repeat = _gtk_css_value_get_enum (value);
869 enum_class = g_type_class_ref (GTK_TYPE_CSS_BACKGROUND_REPEAT);
870 enum_value = g_enum_get_value (enum_class, repeat);
872 /* only triggers for 'repeat-x' and 'repeat-y' */
874 g_string_append (string, enum_value->value_nick);
877 enum_value = g_enum_get_value (enum_class, GTK_CSS_BACKGROUND_HORIZONTAL (repeat));
878 g_string_append (string, enum_value->value_nick);
880 if (GTK_CSS_BACKGROUND_HORIZONTAL (repeat) != GTK_CSS_BACKGROUND_VERTICAL (repeat))
882 enum_value = g_enum_get_value (enum_class, GTK_CSS_BACKGROUND_VERTICAL (repeat));
883 g_string_append (string, " ");
884 g_string_append (string, enum_value->value_nick);
888 g_type_class_unref (enum_class);
892 background_size_parse (GtkCssStyleProperty *property,
893 GtkCssParser *parser,
896 GtkCssBackgroundSize size = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), FALSE, FALSE};
898 if (_gtk_css_parser_try (parser, "cover", TRUE))
900 else if (_gtk_css_parser_try (parser, "contain", TRUE))
904 if (_gtk_css_parser_try (parser, "auto", TRUE))
905 _gtk_css_number_init (&size.width, 0, GTK_CSS_PX);
906 else if (!_gtk_css_parser_read_number (parser,
908 GTK_CSS_POSITIVE_ONLY
909 | GTK_CSS_PARSE_PERCENT
910 | GTK_CSS_PARSE_LENGTH))
913 if (_gtk_css_parser_try (parser, "auto", TRUE))
914 _gtk_css_number_init (&size.height, 0, GTK_CSS_PX);
915 else if (_gtk_css_parser_has_number (parser))
917 if (!_gtk_css_parser_read_number (parser,
919 GTK_CSS_POSITIVE_ONLY
920 | GTK_CSS_PARSE_PERCENT
921 | GTK_CSS_PARSE_LENGTH))
925 _gtk_css_number_init (&size.height, 0, GTK_CSS_PX);
928 return _gtk_css_value_new_from_background_size (&size);
932 background_size_print (GtkCssStyleProperty *property,
933 const GtkCssValue *value,
936 const GtkCssBackgroundSize *size = _gtk_css_value_get_background_size (value);
939 g_string_append (string, "cover");
940 else if (size->contain)
941 g_string_append (string, "contain");
944 if (size->width.value == 0)
945 g_string_append (string, "auto");
947 _gtk_css_number_print (&size->width, string);
949 if (size->height.value != 0)
951 g_string_append (string, " ");
952 _gtk_css_number_print (&size->height, string);
958 background_size_compute (GtkCssStyleProperty *property,
959 GtkStyleContext *context,
960 GtkCssValue *specified)
962 const GtkCssBackgroundSize *ssize = _gtk_css_value_get_background_size (specified);
963 GtkCssBackgroundSize csize;
966 csize.cover = ssize->cover;
967 csize.contain = ssize->contain;
968 changed = _gtk_css_number_compute (&csize.width,
971 changed |= _gtk_css_number_compute (&csize.height,
975 return _gtk_css_value_new_from_background_size (&csize);
976 return _gtk_css_value_ref (specified);
980 background_position_parse (GtkCssStyleProperty *property,
981 GtkCssParser *parser,
984 static const struct {
990 { "left", 0, TRUE, FALSE },
991 { "right", 100, TRUE, FALSE },
992 { "center", 50, TRUE, TRUE },
993 { "top", 0, FALSE, TRUE },
994 { "bottom", 100, FALSE, TRUE },
995 { NULL , 0, TRUE, FALSE }, /* used for numbers */
996 { NULL , 50, TRUE, TRUE } /* used for no value */
998 GtkCssBackgroundPosition pos;
999 GtkCssNumber *missing;
1000 guint first, second;
1002 for (first = 0; names[first].name != NULL; first++)
1004 if (_gtk_css_parser_try (parser, names[first].name, TRUE))
1006 if (names[first].horizontal)
1008 _gtk_css_number_init (&pos.x, names[first].percentage, GTK_CSS_PERCENT);
1013 _gtk_css_number_init (&pos.y, names[first].percentage, GTK_CSS_PERCENT);
1019 if (names[first].name == NULL)
1022 if (!_gtk_css_parser_read_number (parser,
1024 GTK_CSS_PARSE_PERCENT
1025 | GTK_CSS_PARSE_LENGTH))
1029 for (second = 0; names[second].name != NULL; second++)
1031 if (_gtk_css_parser_try (parser, names[second].name, TRUE))
1033 _gtk_css_number_init (missing, names[second].percentage, GTK_CSS_PERCENT);
1038 if (names[second].name == NULL)
1040 if (_gtk_css_parser_has_number (parser))
1042 if (missing != &pos.y)
1044 _gtk_css_parser_error (parser, "Invalid combination of values");
1047 if (!_gtk_css_parser_read_number (parser,
1049 GTK_CSS_PARSE_PERCENT
1050 | GTK_CSS_PARSE_LENGTH))
1056 _gtk_css_number_init (missing, 50, GTK_CSS_PERCENT);
1061 if ((names[first].horizontal && !names[second].vertical) ||
1062 (!names[first].horizontal && !names[second].horizontal))
1064 _gtk_css_parser_error (parser, "Invalid combination of values");
1069 return _gtk_css_value_new_from_background_position (&pos);
1073 background_position_print (GtkCssStyleProperty *property,
1074 const GtkCssValue *value,
1077 const GtkCssBackgroundPosition *pos = _gtk_css_value_get_background_position (value);
1078 static const GtkCssNumber center = GTK_CSS_NUMBER_INIT (50, GTK_CSS_PERCENT);
1079 static const struct {
1082 GtkCssNumber number;
1084 { "left", "top", GTK_CSS_NUMBER_INIT (0, GTK_CSS_PERCENT) },
1085 { "right", "bottom", GTK_CSS_NUMBER_INIT (100, GTK_CSS_PERCENT) }
1089 if (_gtk_css_number_equal (&pos->x, ¢er))
1091 if (_gtk_css_number_equal (&pos->y, ¢er))
1093 g_string_append (string, "center");
1099 for (i = 0; i < G_N_ELEMENTS (values); i++)
1101 if (_gtk_css_number_equal (&pos->x, &values[i].number))
1103 g_string_append (string, values[i].x_name);
1107 if (i == G_N_ELEMENTS (values))
1108 _gtk_css_number_print (&pos->x, string);
1110 if (_gtk_css_number_equal (&pos->y, ¢er))
1113 g_string_append_c (string, ' ');
1116 for (i = 0; i < G_N_ELEMENTS (values); i++)
1118 if (_gtk_css_number_equal (&pos->y, &values[i].number))
1120 g_string_append (string, values[i].y_name);
1124 if (i == G_N_ELEMENTS (values))
1126 if (_gtk_css_number_equal (&pos->x, ¢er))
1127 g_string_append (string, "center ");
1128 _gtk_css_number_print (&pos->y, string);
1132 static GtkCssValue *
1133 background_position_compute (GtkCssStyleProperty *property,
1134 GtkStyleContext *context,
1135 GtkCssValue *specified)
1137 const GtkCssBackgroundPosition *spos = _gtk_css_value_get_background_position (specified);
1138 GtkCssBackgroundPosition cpos;
1141 changed = _gtk_css_number_compute (&cpos.x,
1144 changed |= _gtk_css_number_compute (&cpos.y,
1148 return _gtk_css_value_new_from_background_position (&cpos);
1149 return _gtk_css_value_ref (specified);
1152 /*** REGISTRATION ***/
1154 static GtkSymbolicColor *
1155 gtk_symbolic_color_new_rgba (double red,
1160 GdkRGBA rgba = { red, green, blue, alpha };
1162 return gtk_symbolic_color_new_literal (&rgba);
1166 _gtk_css_style_property_init_properties (void)
1169 GtkCssBackgroundSize default_background_size = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), FALSE, FALSE };
1170 GtkCssBackgroundPosition default_background_position = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PERCENT), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PERCENT)};
1171 GtkCssBorderCornerRadius no_corner_radius = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX) };
1172 GtkBorder border_of_ones = { 1, 1, 1, 1 };
1173 GtkCssBorderImageRepeat border_image_repeat = { GTK_CSS_REPEAT_STYLE_STRETCH, GTK_CSS_REPEAT_STYLE_STRETCH };
1175 /* Initialize "color" and "font-size" first,
1176 * so that when computing values later they are
1177 * done first. That way, 'currentColor' and font
1178 * sizes in em can be looked up properly */
1179 gtk_css_style_property_register ("color",
1181 GTK_STYLE_PROPERTY_INHERIT,
1184 color_property_compute,
1188 _gtk_css_value_new_take_symbolic_color (
1189 gtk_symbolic_color_new_rgba (1, 1, 1, 1)));
1190 gtk_css_style_property_register ("font-size",
1192 GTK_STYLE_PROPERTY_INHERIT,
1199 _gtk_css_value_new_from_double (10.0));
1201 /* properties that aren't referenced when computing values
1203 gtk_css_style_property_register ("background-color",
1212 _gtk_css_value_new_take_symbolic_color (
1213 gtk_symbolic_color_new_rgba (0, 0, 0, 0)));
1215 value = _gtk_css_string_value_new ("Sans");
1216 gtk_css_style_property_register ("font-family",
1218 GTK_STYLE_PROPERTY_INHERIT,
1225 _gtk_css_array_value_new (&value, 1));
1226 gtk_css_style_property_register ("font-style",
1228 GTK_STYLE_PROPERTY_INHERIT,
1235 _gtk_css_font_style_value_new (PANGO_STYLE_NORMAL));
1236 gtk_css_style_property_register ("font-variant",
1238 GTK_STYLE_PROPERTY_INHERIT,
1239 parse_pango_variant,
1242 query_pango_variant,
1243 assign_pango_variant,
1245 _gtk_css_font_variant_value_new (PANGO_VARIANT_NORMAL));
1246 gtk_css_style_property_register ("font-weight",
1248 GTK_STYLE_PROPERTY_INHERIT,
1253 assign_pango_weight,
1255 _gtk_css_font_weight_value_new (PANGO_WEIGHT_NORMAL));
1257 gtk_css_style_property_register ("text-shadow",
1259 GTK_STYLE_PROPERTY_INHERIT,
1262 shadow_value_compute,
1266 _gtk_css_shadow_value_new_none ());
1268 gtk_css_style_property_register ("icon-shadow",
1270 GTK_STYLE_PROPERTY_INHERIT,
1273 shadow_value_compute,
1277 _gtk_css_shadow_value_new_none ());
1279 gtk_css_style_property_register ("box-shadow",
1284 shadow_value_compute,
1288 _gtk_css_shadow_value_new_none ());
1290 gtk_css_style_property_register ("margin-top",
1296 query_length_as_int,
1297 assign_length_from_int,
1299 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1300 gtk_css_style_property_register ("margin-left",
1306 query_length_as_int,
1307 assign_length_from_int,
1309 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1310 gtk_css_style_property_register ("margin-bottom",
1316 query_length_as_int,
1317 assign_length_from_int,
1319 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1320 gtk_css_style_property_register ("margin-right",
1326 query_length_as_int,
1327 assign_length_from_int,
1329 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1330 gtk_css_style_property_register ("padding-top",
1336 query_length_as_int,
1337 assign_length_from_int,
1339 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1340 gtk_css_style_property_register ("padding-left",
1346 query_length_as_int,
1347 assign_length_from_int,
1349 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1350 gtk_css_style_property_register ("padding-bottom",
1356 query_length_as_int,
1357 assign_length_from_int,
1359 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1360 gtk_css_style_property_register ("padding-right",
1366 query_length_as_int,
1367 assign_length_from_int,
1369 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1370 /* IMPORTANT: compute_border_width() requires that the border-width
1371 * properties be immeditaly followed by the border-style properties
1373 gtk_css_style_property_register ("border-top-style",
1374 GTK_TYPE_BORDER_STYLE,
1380 assign_border_style,
1382 _gtk_css_border_style_value_new (GTK_BORDER_STYLE_NONE));
1383 gtk_css_style_property_register ("border-top-width",
1388 compute_border_width,
1389 query_length_as_int,
1390 assign_length_from_int,
1392 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1393 gtk_css_style_property_register ("border-left-style",
1394 GTK_TYPE_BORDER_STYLE,
1400 assign_border_style,
1402 _gtk_css_border_style_value_new (GTK_BORDER_STYLE_NONE));
1403 gtk_css_style_property_register ("border-left-width",
1408 compute_border_width,
1409 query_length_as_int,
1410 assign_length_from_int,
1412 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1413 gtk_css_style_property_register ("border-bottom-style",
1414 GTK_TYPE_BORDER_STYLE,
1420 assign_border_style,
1422 _gtk_css_border_style_value_new (GTK_BORDER_STYLE_NONE));
1423 gtk_css_style_property_register ("border-bottom-width",
1428 compute_border_width,
1429 query_length_as_int,
1430 assign_length_from_int,
1432 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1433 gtk_css_style_property_register ("border-right-style",
1434 GTK_TYPE_BORDER_STYLE,
1440 assign_border_style,
1442 _gtk_css_border_style_value_new (GTK_BORDER_STYLE_NONE));
1443 gtk_css_style_property_register ("border-right-width",
1448 compute_border_width,
1449 query_length_as_int,
1450 assign_length_from_int,
1452 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1454 gtk_css_style_property_register ("border-top-left-radius",
1455 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1457 border_corner_radius_value_parse,
1458 border_corner_radius_value_print,
1463 _gtk_css_value_new_from_border_corner_radius (&no_corner_radius));
1464 gtk_css_style_property_register ("border-top-right-radius",
1465 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1467 border_corner_radius_value_parse,
1468 border_corner_radius_value_print,
1473 _gtk_css_value_new_from_border_corner_radius (&no_corner_radius));
1474 gtk_css_style_property_register ("border-bottom-right-radius",
1475 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1477 border_corner_radius_value_parse,
1478 border_corner_radius_value_print,
1483 _gtk_css_value_new_from_border_corner_radius (&no_corner_radius));
1484 gtk_css_style_property_register ("border-bottom-left-radius",
1485 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1487 border_corner_radius_value_parse,
1488 border_corner_radius_value_print,
1493 _gtk_css_value_new_from_border_corner_radius (&no_corner_radius));
1495 gtk_css_style_property_register ("outline-style",
1496 GTK_TYPE_BORDER_STYLE,
1502 assign_border_style,
1504 _gtk_css_border_style_value_new (GTK_BORDER_STYLE_NONE));
1505 gtk_css_style_property_register ("outline-width",
1510 compute_border_width,
1511 query_length_as_int,
1512 assign_length_from_int,
1514 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1515 gtk_css_style_property_register ("outline-offset",
1524 _gtk_css_value_new_from_int (0));
1526 gtk_css_style_property_register ("background-clip",
1535 _gtk_css_area_value_new (GTK_CSS_AREA_BORDER_BOX));
1536 gtk_css_style_property_register ("background-origin",
1545 _gtk_css_area_value_new (GTK_CSS_AREA_PADDING_BOX));
1546 gtk_css_style_property_register ("background-size",
1549 background_size_parse,
1550 background_size_print,
1551 background_size_compute,
1555 _gtk_css_value_new_from_background_size (&default_background_size));
1556 gtk_css_style_property_register ("background-position",
1559 background_position_parse,
1560 background_position_print,
1561 background_position_compute,
1565 _gtk_css_value_new_from_background_position (&default_background_position));
1567 gtk_css_style_property_register ("border-top-color",
1576 _gtk_css_value_new_take_symbolic_color (
1577 gtk_symbolic_color_ref (
1578 _gtk_symbolic_color_get_current_color ())));
1579 gtk_css_style_property_register ("border-right-color",
1588 _gtk_css_value_new_take_symbolic_color (
1589 gtk_symbolic_color_ref (
1590 _gtk_symbolic_color_get_current_color ())));
1591 gtk_css_style_property_register ("border-bottom-color",
1600 _gtk_css_value_new_take_symbolic_color (
1601 gtk_symbolic_color_ref (
1602 _gtk_symbolic_color_get_current_color ())));
1603 gtk_css_style_property_register ("border-left-color",
1612 _gtk_css_value_new_take_symbolic_color (
1613 gtk_symbolic_color_ref (
1614 _gtk_symbolic_color_get_current_color ())));
1615 gtk_css_style_property_register ("outline-color",
1624 _gtk_css_value_new_take_symbolic_color (
1625 gtk_symbolic_color_ref (
1626 _gtk_symbolic_color_get_current_color ())));
1628 gtk_css_style_property_register ("background-repeat",
1629 GTK_TYPE_CSS_BACKGROUND_REPEAT,
1631 background_repeat_value_parse,
1632 background_repeat_value_print,
1637 _gtk_css_value_new_from_enum (GTK_TYPE_CSS_BACKGROUND_REPEAT,
1638 GTK_CSS_BACKGROUND_REPEAT |
1639 (GTK_CSS_BACKGROUND_REPEAT << GTK_CSS_BACKGROUND_REPEAT_SHIFT)));
1640 gtk_css_style_property_register ("background-image",
1641 CAIRO_GOBJECT_TYPE_PATTERN,
1643 css_image_value_parse,
1645 css_image_value_compute,
1646 css_image_value_query,
1647 css_image_value_assign,
1649 _gtk_css_image_value_new (NULL));
1651 gtk_css_style_property_register ("border-image-source",
1652 CAIRO_GOBJECT_TYPE_PATTERN,
1654 css_image_value_parse,
1656 css_image_value_compute,
1657 css_image_value_query,
1658 css_image_value_assign,
1660 _gtk_css_image_value_new (NULL));
1661 gtk_css_style_property_register ("border-image-repeat",
1662 GTK_TYPE_CSS_BORDER_IMAGE_REPEAT,
1664 border_image_repeat_parse,
1670 _gtk_css_value_new_from_border_image_repeat (&border_image_repeat));
1672 /* XXX: The initial value is wrong, it should be 100% */
1673 gtk_css_style_property_register ("border-image-slice",
1676 border_image_slice_parse,
1682 _gtk_css_value_new_from_boxed (GTK_TYPE_BORDER, &border_of_ones));
1683 gtk_css_style_property_register ("border-image-width",
1686 border_image_width_parse,
1692 _gtk_css_value_new_from_boxed (GTK_TYPE_BORDER, NULL));
1693 gtk_css_style_property_register ("engine",
1694 GTK_TYPE_THEMING_ENGINE,
1702 _gtk_css_value_new_from_theming_engine (gtk_theming_engine_load (NULL)));
1703 gtk_css_style_property_register ("transition",
1704 GTK_TYPE_ANIMATION_DESCRIPTION,
1712 _gtk_css_value_new_from_boxed (GTK_TYPE_ANIMATION_DESCRIPTION, NULL));
1714 /* Private property holding the binding sets */
1715 gtk_css_style_property_register ("gtk-key-bindings",
1718 bindings_value_parse,
1719 bindings_value_print,
1724 _gtk_css_value_new_take_binding_sets (NULL));