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 "gtkgradient.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 GtkSymbolicColor *symbolic = _gtk_css_value_get_symbolic_color (specified);
194 GtkCssValue *resolved;
196 if (symbolic == _gtk_symbolic_color_get_current_color ())
198 /* The computed value of the ‘currentColor’ keyword is the computed
199 * value of the ‘color’ property. If the ‘currentColor’ keyword is
200 * set on the ‘color’ property itself, it is treated as ‘color: inherit’.
202 if (g_str_equal (_gtk_style_property_get_name (GTK_STYLE_PROPERTY (property)), "color"))
204 GtkStyleContext *parent = gtk_style_context_get_parent (context);
207 return _gtk_css_value_ref (_gtk_style_context_peek_property (parent, "color"));
209 return _gtk_css_style_compute_value (context,
211 _gtk_css_style_property_get_initial_value (property));
215 return _gtk_css_value_ref (_gtk_style_context_peek_property (context, "color"));
218 else if ((resolved = _gtk_style_context_resolve_color_value (context,
225 return color_compute (property,
227 _gtk_css_style_property_get_initial_value (property));
232 font_family_parse (GtkCssStyleProperty *property,
233 GtkCssParser *parser,
239 /* We don't special case generic families. Pango should do
242 names = g_ptr_array_new ();
245 name = _gtk_css_parser_try_ident (parser, TRUE);
248 GString *string = g_string_new (name);
250 while ((name = _gtk_css_parser_try_ident (parser, TRUE)))
252 g_string_append_c (string, ' ');
253 g_string_append (string, name);
256 name = g_string_free (string, FALSE);
260 name = _gtk_css_parser_read_string (parser);
263 g_ptr_array_free (names, TRUE);
268 g_ptr_array_add (names, name);
269 } while (_gtk_css_parser_try (parser, ",", TRUE));
271 /* NULL-terminate array */
272 g_ptr_array_add (names, NULL);
273 return _gtk_css_value_new_take_strv ((char **) g_ptr_array_free (names, FALSE));
277 font_family_value_print (GtkCssStyleProperty *property,
278 const GtkCssValue *value,
281 const char **names = _gtk_css_value_get_strv (value);
283 if (names == NULL || *names == NULL)
285 g_string_append (string, "none");
289 string_append_string (string, *names);
293 g_string_append (string, ", ");
294 string_append_string (string, *names);
300 parse_pango_style (GtkCssStyleProperty *property,
301 GtkCssParser *parser,
306 if (!_gtk_css_parser_try_enum (parser, PANGO_TYPE_STYLE, &value))
308 _gtk_css_parser_error (parser, "unknown value for property");
312 return _gtk_css_value_new_from_enum (PANGO_TYPE_STYLE, value);
316 parse_pango_weight (GtkCssStyleProperty *property,
317 GtkCssParser *parser,
322 if (!_gtk_css_parser_try_enum (parser, PANGO_TYPE_WEIGHT, &value))
324 _gtk_css_parser_error (parser, "unknown value for property");
328 return _gtk_css_value_new_from_enum (PANGO_TYPE_WEIGHT, value);
332 parse_pango_variant (GtkCssStyleProperty *property,
333 GtkCssParser *parser,
338 if (!_gtk_css_parser_try_enum (parser, PANGO_TYPE_VARIANT, &value))
340 _gtk_css_parser_error (parser, "unknown value for property");
344 return _gtk_css_value_new_from_enum (PANGO_TYPE_VARIANT, value);
348 parse_border_style (GtkCssStyleProperty *property,
349 GtkCssParser *parser,
354 if (!_gtk_css_parser_try_enum (parser, GTK_TYPE_BORDER_STYLE, &value))
356 _gtk_css_parser_error (parser, "unknown value for property");
360 return _gtk_css_value_new_from_enum (GTK_TYPE_BORDER_STYLE, value);
364 parse_css_area (GtkCssStyleProperty *property,
365 GtkCssParser *parser,
370 if (!_gtk_css_parser_try_enum (parser, GTK_TYPE_CSS_AREA, &value))
372 _gtk_css_parser_error (parser, "unknown value for property");
376 return _gtk_css_value_new_from_enum (GTK_TYPE_CSS_AREA, value);
380 bindings_value_parse (GtkCssStyleProperty *property,
381 GtkCssParser *parser,
385 GtkBindingSet *binding_set;
388 array = g_ptr_array_new ();
391 name = _gtk_css_parser_try_ident (parser, TRUE);
394 _gtk_css_parser_error (parser, "Not a valid binding name");
395 g_ptr_array_free (array, TRUE);
399 binding_set = gtk_binding_set_find (name);
403 _gtk_css_parser_error (parser, "No binding set named '%s'", name);
408 g_ptr_array_add (array, binding_set);
411 while (_gtk_css_parser_try (parser, ",", TRUE));
413 return _gtk_css_value_new_take_binding_sets (array);
417 bindings_value_print (GtkCssStyleProperty *property,
418 const GtkCssValue *value,
424 array = _gtk_css_value_get_boxed (value);
426 for (i = 0; i < array->len; i++)
428 GtkBindingSet *binding_set = g_ptr_array_index (array, i);
431 g_string_append (string, ", ");
432 g_string_append (string, binding_set->set_name);
437 shadow_value_parse (GtkCssStyleProperty *property,
438 GtkCssParser *parser,
441 return _gtk_css_shadow_value_parse (parser);
445 shadow_value_compute (GtkCssStyleProperty *property,
446 GtkStyleContext *context,
447 GtkCssValue *specified)
449 return _gtk_css_shadow_value_compute (specified, context);
453 border_corner_radius_value_parse (GtkCssStyleProperty *property,
454 GtkCssParser *parser,
457 GtkCssBorderCornerRadius corner;
459 if (!_gtk_css_parser_read_number (parser,
461 GTK_CSS_POSITIVE_ONLY
462 | GTK_CSS_PARSE_PERCENT
463 | GTK_CSS_NUMBER_AS_PIXELS
464 | GTK_CSS_PARSE_LENGTH))
467 if (!_gtk_css_parser_has_number (parser))
468 corner.vertical = corner.horizontal;
469 else if (!_gtk_css_parser_read_number (parser,
471 GTK_CSS_POSITIVE_ONLY
472 | GTK_CSS_PARSE_PERCENT
473 | GTK_CSS_NUMBER_AS_PIXELS
474 | GTK_CSS_PARSE_LENGTH))
477 return _gtk_css_value_new_from_border_corner_radius (&corner);
481 border_corner_radius_value_print (GtkCssStyleProperty *property,
482 const GtkCssValue *value,
485 const GtkCssBorderCornerRadius *corner;
487 corner = _gtk_css_value_get_border_corner_radius (value);
489 _gtk_css_number_print (&corner->horizontal, string);
491 if (!_gtk_css_number_equal (&corner->horizontal, &corner->vertical))
493 g_string_append_c (string, ' ');
494 _gtk_css_number_print (&corner->vertical, string);
499 css_image_value_parse (GtkCssStyleProperty *property,
500 GtkCssParser *parser,
505 if (_gtk_css_parser_try (parser, "none", TRUE))
509 image = _gtk_css_image_new_parse (parser, base);
514 return _gtk_css_value_new_take_image (image);
518 css_image_value_print (GtkCssStyleProperty *property,
519 const GtkCssValue *value,
522 GtkCssImage *image = _gtk_css_value_get_image (value);
525 _gtk_css_image_print (image, string);
527 g_string_append (string, "none");
531 css_image_value_compute (GtkCssStyleProperty *property,
532 GtkStyleContext *context,
533 GtkCssValue *specified)
535 GtkCssImage *image, *computed;
537 image = _gtk_css_value_get_image (specified);
540 return _gtk_css_value_ref (specified);
542 computed = _gtk_css_image_compute (image, context);
544 if (computed == image)
546 g_object_unref (computed);
547 return _gtk_css_value_ref (specified);
550 return _gtk_css_value_new_take_image (computed);
554 css_image_value_query (GtkCssStyleProperty *property,
555 const GtkCssValue *css_value,
558 GtkCssImage *image = _gtk_css_value_get_image (css_value);
559 cairo_pattern_t *pattern;
560 cairo_surface_t *surface;
561 cairo_matrix_t matrix;
563 g_value_init (value, CAIRO_GOBJECT_TYPE_PATTERN);
565 if (GTK_IS_CSS_IMAGE_GRADIENT (image))
566 g_value_set_boxed (value, GTK_CSS_IMAGE_GRADIENT (image)->pattern);
567 else if (image != NULL)
569 double width, height;
571 /* the 100, 100 is rather random */
572 _gtk_css_image_get_concrete_size (image, 0, 0, 100, 100, &width, &height);
573 surface = _gtk_css_image_get_surface (image, NULL, width, height);
574 pattern = cairo_pattern_create_for_surface (surface);
575 cairo_matrix_init_scale (&matrix, width, height);
576 cairo_pattern_set_matrix (pattern, &matrix);
577 cairo_surface_destroy (surface);
578 g_value_take_boxed (value, pattern);
583 css_image_value_assign (GtkCssStyleProperty *property,
586 g_warning ("FIXME: assigning images is not implemented");
587 return _gtk_css_value_new_take_image (NULL);
591 font_size_parse (GtkCssStyleProperty *property,
592 GtkCssParser *parser,
597 if (!_gtk_css_parser_try_double (parser, &d))
599 _gtk_css_parser_error (parser, "Expected a number");
603 return _gtk_css_value_new_from_double (d);
607 outline_parse (GtkCssStyleProperty *property,
608 GtkCssParser *parser,
613 if (!_gtk_css_parser_try_int (parser, &i))
615 _gtk_css_parser_error (parser, "Expected an integer");
619 return _gtk_css_value_new_from_int (i);
623 border_image_repeat_parse (GtkCssStyleProperty *property,
624 GtkCssParser *parser,
627 GValue value = G_VALUE_INIT;
630 g_value_init (&value, GTK_TYPE_CSS_BORDER_IMAGE_REPEAT);
631 if (!_gtk_css_style_parse_value (&value, parser, base))
633 g_value_unset (&value);
637 result = _gtk_css_value_new_from_gvalue (&value);
638 g_value_unset (&value);
644 border_image_slice_parse (GtkCssStyleProperty *property,
645 GtkCssParser *parser,
648 GValue value = G_VALUE_INIT;
651 g_value_init (&value, GTK_TYPE_BORDER);
652 if (!_gtk_css_style_parse_value (&value, parser, base))
654 g_value_unset (&value);
658 result = _gtk_css_value_new_from_gvalue (&value);
659 g_value_unset (&value);
665 border_image_width_parse (GtkCssStyleProperty *property,
666 GtkCssParser *parser,
669 GValue value = G_VALUE_INIT;
672 g_value_init (&value, GTK_TYPE_BORDER);
673 if (!_gtk_css_style_parse_value (&value, parser, base))
675 g_value_unset (&value);
679 result = _gtk_css_value_new_from_gvalue (&value);
680 g_value_unset (&value);
686 engine_parse (GtkCssStyleProperty *property,
687 GtkCssParser *parser,
690 GtkThemingEngine *engine;
693 if (_gtk_css_parser_try (parser, "none", TRUE))
694 return _gtk_css_value_new_from_theming_engine (gtk_theming_engine_load (NULL));
696 str = _gtk_css_parser_try_ident (parser, TRUE);
699 _gtk_css_parser_error (parser, "Expected a valid theme name");
703 engine = gtk_theming_engine_load (str);
707 _gtk_css_parser_error (parser, "Theming engine '%s' not found", str);
714 return _gtk_css_value_new_from_theming_engine (engine);
718 transition_parse (GtkCssStyleProperty *property,
719 GtkCssParser *parser,
722 GValue value = G_VALUE_INIT;
725 g_value_init (&value, GTK_TYPE_ANIMATION_DESCRIPTION);
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 parse_margin (GtkCssStyleProperty *property,
740 GtkCssParser *parser,
743 return _gtk_css_number_value_parse (parser,
744 GTK_CSS_NUMBER_AS_PIXELS
745 | GTK_CSS_PARSE_LENGTH);
749 compute_margin (GtkCssStyleProperty *property,
750 GtkStyleContext *context,
751 GtkCssValue *specified)
753 return _gtk_css_number_value_compute (specified, context);
757 parse_padding (GtkCssStyleProperty *property,
758 GtkCssParser *parser,
761 return _gtk_css_number_value_parse (parser,
762 GTK_CSS_POSITIVE_ONLY
763 | GTK_CSS_NUMBER_AS_PIXELS
764 | GTK_CSS_PARSE_LENGTH);
768 compute_padding (GtkCssStyleProperty *property,
769 GtkStyleContext *context,
770 GtkCssValue *specified)
772 return _gtk_css_number_value_compute (specified, context);
776 parse_border_width (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_border_width (GtkCssStyleProperty *property,
788 GtkStyleContext *context,
789 GtkCssValue *specified)
791 GtkCssStyleProperty *style;
792 GtkBorderStyle border_style;
794 /* The -1 is magic that is only true because we register the style
795 * properties directly after the width properties.
797 style = _gtk_css_style_property_lookup_by_id (_gtk_css_style_property_get_id (property) - 1);
799 border_style = _gtk_css_value_get_border_style (_gtk_style_context_peek_property (context, _gtk_style_property_get_name (GTK_STYLE_PROPERTY (style))));
801 if (border_style == GTK_BORDER_STYLE_NONE ||
802 border_style == GTK_BORDER_STYLE_HIDDEN)
803 return _gtk_css_number_value_new (0, GTK_CSS_PX);
805 return _gtk_css_number_value_compute (specified, context);
809 background_repeat_value_parse (GtkCssStyleProperty *property,
810 GtkCssParser *parser,
813 int repeat, vertical;
815 if (!_gtk_css_parser_try_enum (parser, GTK_TYPE_CSS_BACKGROUND_REPEAT, &repeat))
817 _gtk_css_parser_error (parser, "Not a valid value");
821 if (repeat <= GTK_CSS_BACKGROUND_REPEAT_MASK)
823 if (_gtk_css_parser_try_enum (parser, GTK_TYPE_CSS_BACKGROUND_REPEAT, &vertical))
825 if (vertical >= GTK_CSS_BACKGROUND_REPEAT_MASK)
827 _gtk_css_parser_error (parser, "Not a valid 2nd value");
831 repeat |= vertical << GTK_CSS_BACKGROUND_REPEAT_SHIFT;
834 repeat |= repeat << GTK_CSS_BACKGROUND_REPEAT_SHIFT;
837 return _gtk_css_value_new_from_enum (GTK_TYPE_CSS_BACKGROUND_REPEAT, repeat);
841 background_repeat_value_print (GtkCssStyleProperty *property,
842 const GtkCssValue *value,
845 GEnumClass *enum_class;
846 GEnumValue *enum_value;
847 GtkCssBackgroundRepeat repeat;
849 repeat = _gtk_css_value_get_enum (value);
850 enum_class = g_type_class_ref (GTK_TYPE_CSS_BACKGROUND_REPEAT);
851 enum_value = g_enum_get_value (enum_class, repeat);
853 /* only triggers for 'repeat-x' and 'repeat-y' */
855 g_string_append (string, enum_value->value_nick);
858 enum_value = g_enum_get_value (enum_class, GTK_CSS_BACKGROUND_HORIZONTAL (repeat));
859 g_string_append (string, enum_value->value_nick);
861 if (GTK_CSS_BACKGROUND_HORIZONTAL (repeat) != GTK_CSS_BACKGROUND_VERTICAL (repeat))
863 enum_value = g_enum_get_value (enum_class, GTK_CSS_BACKGROUND_VERTICAL (repeat));
864 g_string_append (string, " ");
865 g_string_append (string, enum_value->value_nick);
869 g_type_class_unref (enum_class);
873 background_size_parse (GtkCssStyleProperty *property,
874 GtkCssParser *parser,
877 GtkCssBackgroundSize size = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), FALSE, FALSE};
879 if (_gtk_css_parser_try (parser, "cover", TRUE))
881 else if (_gtk_css_parser_try (parser, "contain", TRUE))
885 if (_gtk_css_parser_try (parser, "auto", TRUE))
886 _gtk_css_number_init (&size.width, 0, GTK_CSS_PX);
887 else if (!_gtk_css_parser_read_number (parser,
889 GTK_CSS_POSITIVE_ONLY
890 | GTK_CSS_PARSE_PERCENT
891 | GTK_CSS_PARSE_LENGTH))
894 if (_gtk_css_parser_try (parser, "auto", TRUE))
895 _gtk_css_number_init (&size.height, 0, GTK_CSS_PX);
896 else if (_gtk_css_parser_has_number (parser))
898 if (!_gtk_css_parser_read_number (parser,
900 GTK_CSS_POSITIVE_ONLY
901 | GTK_CSS_PARSE_PERCENT
902 | GTK_CSS_PARSE_LENGTH))
906 _gtk_css_number_init (&size.height, 0, GTK_CSS_PX);
909 return _gtk_css_value_new_from_background_size (&size);
913 background_size_print (GtkCssStyleProperty *property,
914 const GtkCssValue *value,
917 const GtkCssBackgroundSize *size = _gtk_css_value_get_background_size (value);
920 g_string_append (string, "cover");
921 else if (size->contain)
922 g_string_append (string, "contain");
925 if (size->width.value == 0)
926 g_string_append (string, "auto");
928 _gtk_css_number_print (&size->width, string);
930 if (size->height.value != 0)
932 g_string_append (string, " ");
933 _gtk_css_number_print (&size->height, string);
939 background_size_compute (GtkCssStyleProperty *property,
940 GtkStyleContext *context,
941 GtkCssValue *specified)
943 const GtkCssBackgroundSize *ssize = _gtk_css_value_get_background_size (specified);
944 GtkCssBackgroundSize csize;
947 csize.cover = ssize->cover;
948 csize.contain = ssize->contain;
949 changed = _gtk_css_number_compute (&csize.width,
952 changed |= _gtk_css_number_compute (&csize.height,
956 return _gtk_css_value_new_from_background_size (&csize);
957 return _gtk_css_value_ref (specified);
961 background_position_parse (GtkCssStyleProperty *property,
962 GtkCssParser *parser,
965 static const struct {
971 { "left", 0, TRUE, FALSE },
972 { "right", 100, TRUE, FALSE },
973 { "center", 50, TRUE, TRUE },
974 { "top", 0, FALSE, TRUE },
975 { "bottom", 100, FALSE, TRUE },
976 { NULL , 0, TRUE, FALSE }, /* used for numbers */
977 { NULL , 50, TRUE, TRUE } /* used for no value */
979 GtkCssBackgroundPosition pos;
980 GtkCssNumber *missing;
983 for (first = 0; names[first].name != NULL; first++)
985 if (_gtk_css_parser_try (parser, names[first].name, TRUE))
987 if (names[first].horizontal)
989 _gtk_css_number_init (&pos.x, names[first].percentage, GTK_CSS_PERCENT);
994 _gtk_css_number_init (&pos.y, names[first].percentage, GTK_CSS_PERCENT);
1000 if (names[first].name == NULL)
1003 if (!_gtk_css_parser_read_number (parser,
1005 GTK_CSS_PARSE_PERCENT
1006 | GTK_CSS_PARSE_LENGTH))
1010 for (second = 0; names[second].name != NULL; second++)
1012 if (_gtk_css_parser_try (parser, names[second].name, TRUE))
1014 _gtk_css_number_init (missing, names[second].percentage, GTK_CSS_PERCENT);
1019 if (names[second].name == NULL)
1021 if (_gtk_css_parser_has_number (parser))
1023 if (missing != &pos.y)
1025 _gtk_css_parser_error (parser, "Invalid combination of values");
1028 if (!_gtk_css_parser_read_number (parser,
1030 GTK_CSS_PARSE_PERCENT
1031 | GTK_CSS_PARSE_LENGTH))
1037 _gtk_css_number_init (missing, 50, GTK_CSS_PERCENT);
1042 if ((names[first].horizontal && !names[second].vertical) ||
1043 (!names[first].horizontal && !names[second].horizontal))
1045 _gtk_css_parser_error (parser, "Invalid combination of values");
1050 return _gtk_css_value_new_from_background_position (&pos);
1054 background_position_print (GtkCssStyleProperty *property,
1055 const GtkCssValue *value,
1058 const GtkCssBackgroundPosition *pos = _gtk_css_value_get_background_position (value);
1059 static const GtkCssNumber center = GTK_CSS_NUMBER_INIT (50, GTK_CSS_PERCENT);
1060 static const struct {
1063 GtkCssNumber number;
1065 { "left", "top", GTK_CSS_NUMBER_INIT (0, GTK_CSS_PERCENT) },
1066 { "right", "bottom", GTK_CSS_NUMBER_INIT (100, GTK_CSS_PERCENT) }
1070 if (_gtk_css_number_equal (&pos->x, ¢er))
1072 if (_gtk_css_number_equal (&pos->y, ¢er))
1074 g_string_append (string, "center");
1080 for (i = 0; i < G_N_ELEMENTS (values); i++)
1082 if (_gtk_css_number_equal (&pos->x, &values[i].number))
1084 g_string_append (string, values[i].x_name);
1088 if (i == G_N_ELEMENTS (values))
1089 _gtk_css_number_print (&pos->x, string);
1091 if (_gtk_css_number_equal (&pos->y, ¢er))
1094 g_string_append_c (string, ' ');
1097 for (i = 0; i < G_N_ELEMENTS (values); i++)
1099 if (_gtk_css_number_equal (&pos->y, &values[i].number))
1101 g_string_append (string, values[i].y_name);
1105 if (i == G_N_ELEMENTS (values))
1107 if (_gtk_css_number_equal (&pos->x, ¢er))
1108 g_string_append (string, "center ");
1109 _gtk_css_number_print (&pos->y, string);
1113 static GtkCssValue *
1114 background_position_compute (GtkCssStyleProperty *property,
1115 GtkStyleContext *context,
1116 GtkCssValue *specified)
1118 const GtkCssBackgroundPosition *spos = _gtk_css_value_get_background_position (specified);
1119 GtkCssBackgroundPosition cpos;
1122 changed = _gtk_css_number_compute (&cpos.x,
1125 changed |= _gtk_css_number_compute (&cpos.y,
1129 return _gtk_css_value_new_from_background_position (&cpos);
1130 return _gtk_css_value_ref (specified);
1133 /*** REGISTRATION ***/
1135 static GtkSymbolicColor *
1136 gtk_symbolic_color_new_rgba (double red,
1141 GdkRGBA rgba = { red, green, blue, alpha };
1143 return gtk_symbolic_color_new_literal (&rgba);
1147 _gtk_css_style_property_init_properties (void)
1149 char *default_font_family[] = { "Sans", NULL };
1150 GtkCssBackgroundSize default_background_size = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), FALSE, FALSE };
1151 GtkCssBackgroundPosition default_background_position = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PERCENT), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PERCENT)};
1152 GtkCssBorderCornerRadius no_corner_radius = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX) };
1153 GtkBorder border_of_ones = { 1, 1, 1, 1 };
1154 GtkCssBorderImageRepeat border_image_repeat = { GTK_CSS_REPEAT_STYLE_STRETCH, GTK_CSS_REPEAT_STYLE_STRETCH };
1156 /* Initialize "color" and "font-size" first,
1157 * so that when computing values later they are
1158 * done first. That way, 'currentColor' and font
1159 * sizes in em can be looked up properly */
1160 gtk_css_style_property_register ("color",
1162 GTK_STYLE_PROPERTY_INHERIT,
1169 _gtk_css_value_new_take_symbolic_color (
1170 gtk_symbolic_color_new_rgba (1, 1, 1, 1)));
1171 gtk_css_style_property_register ("font-size",
1173 GTK_STYLE_PROPERTY_INHERIT,
1180 _gtk_css_value_new_from_double (10.0));
1182 /* properties that aren't referenced when computing values
1184 gtk_css_style_property_register ("background-color",
1193 _gtk_css_value_new_take_symbolic_color (
1194 gtk_symbolic_color_new_rgba (0, 0, 0, 0)));
1196 gtk_css_style_property_register ("font-family",
1198 GTK_STYLE_PROPERTY_INHERIT,
1200 font_family_value_print,
1205 _gtk_css_value_new_take_strv (g_strdupv (default_font_family)));
1206 gtk_css_style_property_register ("font-style",
1208 GTK_STYLE_PROPERTY_INHERIT,
1215 _gtk_css_value_new_from_enum (PANGO_TYPE_STYLE,
1216 PANGO_STYLE_NORMAL));
1217 gtk_css_style_property_register ("font-variant",
1219 GTK_STYLE_PROPERTY_INHERIT,
1220 parse_pango_variant,
1226 _gtk_css_value_new_from_enum (PANGO_TYPE_VARIANT,
1227 PANGO_VARIANT_NORMAL));
1228 /* xxx: need to parse this properly, ie parse the numbers */
1229 gtk_css_style_property_register ("font-weight",
1231 GTK_STYLE_PROPERTY_INHERIT,
1238 _gtk_css_value_new_from_enum (PANGO_TYPE_WEIGHT,
1239 PANGO_WEIGHT_NORMAL));
1241 gtk_css_style_property_register ("text-shadow",
1243 GTK_STYLE_PROPERTY_INHERIT,
1246 shadow_value_compute,
1250 _gtk_css_shadow_value_new_none ());
1252 gtk_css_style_property_register ("icon-shadow",
1254 GTK_STYLE_PROPERTY_INHERIT,
1257 shadow_value_compute,
1261 _gtk_css_shadow_value_new_none ());
1263 gtk_css_style_property_register ("box-shadow",
1268 shadow_value_compute,
1272 _gtk_css_shadow_value_new_none ());
1274 gtk_css_style_property_register ("margin-top",
1280 query_length_as_int,
1281 assign_length_from_int,
1283 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1284 gtk_css_style_property_register ("margin-left",
1290 query_length_as_int,
1291 assign_length_from_int,
1293 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1294 gtk_css_style_property_register ("margin-bottom",
1300 query_length_as_int,
1301 assign_length_from_int,
1303 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1304 gtk_css_style_property_register ("margin-right",
1310 query_length_as_int,
1311 assign_length_from_int,
1313 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1314 gtk_css_style_property_register ("padding-top",
1320 query_length_as_int,
1321 assign_length_from_int,
1323 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1324 gtk_css_style_property_register ("padding-left",
1330 query_length_as_int,
1331 assign_length_from_int,
1333 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1334 gtk_css_style_property_register ("padding-bottom",
1340 query_length_as_int,
1341 assign_length_from_int,
1343 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1344 gtk_css_style_property_register ("padding-right",
1350 query_length_as_int,
1351 assign_length_from_int,
1353 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1354 /* IMPORTANT: compute_border_width() requires that the border-width
1355 * properties be immeditaly followed by the border-style properties
1357 gtk_css_style_property_register ("border-top-style",
1358 GTK_TYPE_BORDER_STYLE,
1366 _gtk_css_value_new_from_border_style (GTK_BORDER_STYLE_NONE));
1367 gtk_css_style_property_register ("border-top-width",
1372 compute_border_width,
1373 query_length_as_int,
1374 assign_length_from_int,
1376 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1377 gtk_css_style_property_register ("border-left-style",
1378 GTK_TYPE_BORDER_STYLE,
1386 _gtk_css_value_new_from_border_style (GTK_BORDER_STYLE_NONE));
1387 gtk_css_style_property_register ("border-left-width",
1392 compute_border_width,
1393 query_length_as_int,
1394 assign_length_from_int,
1396 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1397 gtk_css_style_property_register ("border-bottom-style",
1398 GTK_TYPE_BORDER_STYLE,
1406 _gtk_css_value_new_from_border_style (GTK_BORDER_STYLE_NONE));
1407 gtk_css_style_property_register ("border-bottom-width",
1412 compute_border_width,
1413 query_length_as_int,
1414 assign_length_from_int,
1416 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1417 gtk_css_style_property_register ("border-right-style",
1418 GTK_TYPE_BORDER_STYLE,
1426 _gtk_css_value_new_from_border_style (GTK_BORDER_STYLE_NONE));
1427 gtk_css_style_property_register ("border-right-width",
1432 compute_border_width,
1433 query_length_as_int,
1434 assign_length_from_int,
1436 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1438 gtk_css_style_property_register ("border-top-left-radius",
1439 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1441 border_corner_radius_value_parse,
1442 border_corner_radius_value_print,
1447 _gtk_css_value_new_from_border_corner_radius (&no_corner_radius));
1448 gtk_css_style_property_register ("border-top-right-radius",
1449 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1451 border_corner_radius_value_parse,
1452 border_corner_radius_value_print,
1457 _gtk_css_value_new_from_border_corner_radius (&no_corner_radius));
1458 gtk_css_style_property_register ("border-bottom-right-radius",
1459 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1461 border_corner_radius_value_parse,
1462 border_corner_radius_value_print,
1467 _gtk_css_value_new_from_border_corner_radius (&no_corner_radius));
1468 gtk_css_style_property_register ("border-bottom-left-radius",
1469 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1471 border_corner_radius_value_parse,
1472 border_corner_radius_value_print,
1477 _gtk_css_value_new_from_border_corner_radius (&no_corner_radius));
1479 gtk_css_style_property_register ("outline-style",
1480 GTK_TYPE_BORDER_STYLE,
1488 _gtk_css_value_new_from_border_style (GTK_BORDER_STYLE_NONE));
1489 gtk_css_style_property_register ("outline-width",
1494 compute_border_width,
1495 query_length_as_int,
1496 assign_length_from_int,
1498 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1499 gtk_css_style_property_register ("outline-offset",
1508 _gtk_css_value_new_from_int (0));
1510 gtk_css_style_property_register ("background-clip",
1519 _gtk_css_value_new_from_enum (GTK_TYPE_CSS_AREA, GTK_CSS_AREA_BORDER_BOX));
1520 gtk_css_style_property_register ("background-origin",
1529 _gtk_css_value_new_from_enum (GTK_TYPE_CSS_AREA, GTK_CSS_AREA_PADDING_BOX));
1530 gtk_css_style_property_register ("background-size",
1533 background_size_parse,
1534 background_size_print,
1535 background_size_compute,
1539 _gtk_css_value_new_from_background_size (&default_background_size));
1540 gtk_css_style_property_register ("background-position",
1543 background_position_parse,
1544 background_position_print,
1545 background_position_compute,
1549 _gtk_css_value_new_from_background_position (&default_background_position));
1551 gtk_css_style_property_register ("border-top-color",
1560 _gtk_css_value_new_take_symbolic_color (
1561 gtk_symbolic_color_ref (
1562 _gtk_symbolic_color_get_current_color ())));
1563 gtk_css_style_property_register ("border-right-color",
1572 _gtk_css_value_new_take_symbolic_color (
1573 gtk_symbolic_color_ref (
1574 _gtk_symbolic_color_get_current_color ())));
1575 gtk_css_style_property_register ("border-bottom-color",
1584 _gtk_css_value_new_take_symbolic_color (
1585 gtk_symbolic_color_ref (
1586 _gtk_symbolic_color_get_current_color ())));
1587 gtk_css_style_property_register ("border-left-color",
1596 _gtk_css_value_new_take_symbolic_color (
1597 gtk_symbolic_color_ref (
1598 _gtk_symbolic_color_get_current_color ())));
1599 gtk_css_style_property_register ("outline-color",
1608 _gtk_css_value_new_take_symbolic_color (
1609 gtk_symbolic_color_ref (
1610 _gtk_symbolic_color_get_current_color ())));
1612 gtk_css_style_property_register ("background-repeat",
1613 GTK_TYPE_CSS_BACKGROUND_REPEAT,
1615 background_repeat_value_parse,
1616 background_repeat_value_print,
1621 _gtk_css_value_new_from_enum (GTK_TYPE_CSS_BACKGROUND_REPEAT,
1622 GTK_CSS_BACKGROUND_REPEAT |
1623 (GTK_CSS_BACKGROUND_REPEAT << GTK_CSS_BACKGROUND_REPEAT_SHIFT)));
1624 gtk_css_style_property_register ("background-image",
1625 CAIRO_GOBJECT_TYPE_PATTERN,
1627 css_image_value_parse,
1628 css_image_value_print,
1629 css_image_value_compute,
1630 css_image_value_query,
1631 css_image_value_assign,
1633 _gtk_css_value_new_take_image (NULL));
1635 gtk_css_style_property_register ("border-image-source",
1636 CAIRO_GOBJECT_TYPE_PATTERN,
1638 css_image_value_parse,
1639 css_image_value_print,
1640 css_image_value_compute,
1641 css_image_value_query,
1642 css_image_value_assign,
1644 _gtk_css_value_new_take_image (NULL));
1645 gtk_css_style_property_register ("border-image-repeat",
1646 GTK_TYPE_CSS_BORDER_IMAGE_REPEAT,
1648 border_image_repeat_parse,
1654 _gtk_css_value_new_from_border_image_repeat (&border_image_repeat));
1656 /* XXX: The initial value is wrong, it should be 100% */
1657 gtk_css_style_property_register ("border-image-slice",
1660 border_image_slice_parse,
1666 _gtk_css_value_new_from_boxed (GTK_TYPE_BORDER, &border_of_ones));
1667 gtk_css_style_property_register ("border-image-width",
1670 border_image_width_parse,
1676 _gtk_css_value_new_from_boxed (GTK_TYPE_BORDER, NULL));
1677 gtk_css_style_property_register ("engine",
1678 GTK_TYPE_THEMING_ENGINE,
1686 _gtk_css_value_new_from_theming_engine (gtk_theming_engine_load (NULL)));
1687 gtk_css_style_property_register ("transition",
1688 GTK_TYPE_ANIMATION_DESCRIPTION,
1696 _gtk_css_value_new_from_boxed (GTK_TYPE_ANIMATION_DESCRIPTION, NULL));
1698 /* Private property holding the binding sets */
1699 gtk_css_style_property_register ("gtk-key-bindings",
1702 bindings_value_parse,
1703 bindings_value_print,
1708 _gtk_css_value_new_take_binding_sets (NULL));