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 "gtkshadowprivate.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 GtkCssStylePropertyEqualFunc equal_func,
65 GtkCssValue * initial_value)
67 GtkCssStyleProperty *node;
69 g_assert (initial_value != NULL);
70 g_assert (parse_value != NULL);
71 g_assert (value_type == G_TYPE_NONE || query_value != NULL);
73 node = g_object_new (GTK_TYPE_CSS_STYLE_PROPERTY,
74 "value-type", value_type,
75 "inherit", (flags & GTK_STYLE_PROPERTY_INHERIT) ? TRUE : FALSE,
76 "initial-value", initial_value,
80 node->parse_value = parse_value;
82 node->print_value = print_value;
84 node->compute_value = compute_value;
85 node->query_value = query_value;
87 node->equal_func = equal_func;
89 _gtk_css_value_unref (initial_value);
95 string_append_string (GString *str,
100 g_string_append_c (str, '"');
103 len = strcspn (string, "\"\n\r\f");
104 g_string_append (str, string);
111 g_string_append (str, "\\A ");
114 g_string_append (str, "\\D ");
117 g_string_append (str, "\\C ");
120 g_string_append (str, "\\\"");
123 g_assert_not_reached ();
128 g_string_append_c (str, '"');
131 /*** IMPLEMENTATIONS ***/
134 query_simple (GtkCssStyleProperty *property,
135 const GtkCssValue *css_value,
138 _gtk_css_value_init_gvalue (css_value, value);
142 query_length_as_int (GtkCssStyleProperty *property,
143 const GtkCssValue *css_value,
146 g_value_init (value, G_TYPE_INT);
147 g_value_set_int (value, round (_gtk_css_number_value_get (css_value, 100)));
151 color_parse (GtkCssStyleProperty *property,
152 GtkCssParser *parser,
155 GtkSymbolicColor *symbolic;
157 if (_gtk_css_parser_try (parser, "currentcolor", TRUE))
159 symbolic = gtk_symbolic_color_ref (_gtk_symbolic_color_get_current_color ());
163 symbolic = _gtk_css_parser_read_symbolic_color (parser);
164 if (symbolic == NULL)
168 return _gtk_css_value_new_take_symbolic_color (symbolic);
172 color_compute (GtkCssStyleProperty *property,
173 GtkStyleContext *context,
174 GtkCssValue *specified)
176 GtkSymbolicColor *symbolic = _gtk_css_value_get_symbolic_color (specified);
177 GtkCssValue *resolved;
179 if (symbolic == _gtk_symbolic_color_get_current_color ())
181 /* The computed value of the ‘currentColor’ keyword is the computed
182 * value of the ‘color’ property. If the ‘currentColor’ keyword is
183 * set on the ‘color’ property itself, it is treated as ‘color: inherit’.
185 if (g_str_equal (_gtk_style_property_get_name (GTK_STYLE_PROPERTY (property)), "color"))
187 GtkStyleContext *parent = gtk_style_context_get_parent (context);
190 return _gtk_css_value_ref (_gtk_style_context_peek_property (parent, "color"));
192 return _gtk_css_style_compute_value (context,
194 _gtk_css_style_property_get_initial_value (property));
198 return _gtk_css_value_ref (_gtk_style_context_peek_property (context, "color"));
201 else if ((resolved = _gtk_style_context_resolve_color_value (context,
208 return color_compute (property,
210 _gtk_css_style_property_get_initial_value (property));
215 font_family_parse (GtkCssStyleProperty *property,
216 GtkCssParser *parser,
222 /* We don't special case generic families. Pango should do
225 names = g_ptr_array_new ();
228 name = _gtk_css_parser_try_ident (parser, TRUE);
231 GString *string = g_string_new (name);
233 while ((name = _gtk_css_parser_try_ident (parser, TRUE)))
235 g_string_append_c (string, ' ');
236 g_string_append (string, name);
239 name = g_string_free (string, FALSE);
243 name = _gtk_css_parser_read_string (parser);
246 g_ptr_array_free (names, TRUE);
251 g_ptr_array_add (names, name);
252 } while (_gtk_css_parser_try (parser, ",", TRUE));
254 /* NULL-terminate array */
255 g_ptr_array_add (names, NULL);
256 return _gtk_css_value_new_take_strv ((char **) g_ptr_array_free (names, FALSE));
260 font_family_value_print (GtkCssStyleProperty *property,
261 const GtkCssValue *value,
264 const char **names = _gtk_css_value_get_strv (value);
266 if (names == NULL || *names == NULL)
268 g_string_append (string, "none");
272 string_append_string (string, *names);
276 g_string_append (string, ", ");
277 string_append_string (string, *names);
283 parse_pango_style (GtkCssStyleProperty *property,
284 GtkCssParser *parser,
289 if (!_gtk_css_parser_try_enum (parser, PANGO_TYPE_STYLE, &value))
291 _gtk_css_parser_error (parser, "unknown value for property");
295 return _gtk_css_value_new_from_enum (PANGO_TYPE_STYLE, value);
299 parse_pango_weight (GtkCssStyleProperty *property,
300 GtkCssParser *parser,
305 if (!_gtk_css_parser_try_enum (parser, PANGO_TYPE_WEIGHT, &value))
307 _gtk_css_parser_error (parser, "unknown value for property");
311 return _gtk_css_value_new_from_enum (PANGO_TYPE_WEIGHT, value);
315 parse_pango_variant (GtkCssStyleProperty *property,
316 GtkCssParser *parser,
321 if (!_gtk_css_parser_try_enum (parser, PANGO_TYPE_VARIANT, &value))
323 _gtk_css_parser_error (parser, "unknown value for property");
327 return _gtk_css_value_new_from_enum (PANGO_TYPE_VARIANT, value);
331 parse_border_style (GtkCssStyleProperty *property,
332 GtkCssParser *parser,
337 if (!_gtk_css_parser_try_enum (parser, GTK_TYPE_BORDER_STYLE, &value))
339 _gtk_css_parser_error (parser, "unknown value for property");
343 return _gtk_css_value_new_from_enum (GTK_TYPE_BORDER_STYLE, value);
347 parse_css_area (GtkCssStyleProperty *property,
348 GtkCssParser *parser,
353 if (!_gtk_css_parser_try_enum (parser, GTK_TYPE_CSS_AREA, &value))
355 _gtk_css_parser_error (parser, "unknown value for property");
359 return _gtk_css_value_new_from_enum (GTK_TYPE_CSS_AREA, value);
363 bindings_value_parse (GtkCssStyleProperty *property,
364 GtkCssParser *parser,
368 GtkBindingSet *binding_set;
371 array = g_ptr_array_new ();
374 name = _gtk_css_parser_try_ident (parser, TRUE);
377 _gtk_css_parser_error (parser, "Not a valid binding name");
378 g_ptr_array_free (array, TRUE);
382 binding_set = gtk_binding_set_find (name);
386 _gtk_css_parser_error (parser, "No binding set named '%s'", name);
391 g_ptr_array_add (array, binding_set);
394 while (_gtk_css_parser_try (parser, ",", TRUE));
396 return _gtk_css_value_new_take_binding_sets (array);
400 bindings_value_print (GtkCssStyleProperty *property,
401 const GtkCssValue *value,
407 array = _gtk_css_value_get_boxed (value);
409 for (i = 0; i < array->len; i++)
411 GtkBindingSet *binding_set = g_ptr_array_index (array, i);
414 g_string_append (string, ", ");
415 g_string_append (string, binding_set->set_name);
420 shadow_value_parse (GtkCssStyleProperty *property,
421 GtkCssParser *parser,
424 return _gtk_css_shadow_value_parse (parser);
428 shadow_value_compute (GtkCssStyleProperty *property,
429 GtkStyleContext *context,
430 GtkCssValue *specified)
432 return _gtk_css_shadow_value_compute (specified, context);
436 border_corner_radius_value_parse (GtkCssStyleProperty *property,
437 GtkCssParser *parser,
440 GtkCssBorderCornerRadius corner;
442 if (!_gtk_css_parser_read_number (parser,
444 GTK_CSS_POSITIVE_ONLY
445 | GTK_CSS_PARSE_PERCENT
446 | GTK_CSS_NUMBER_AS_PIXELS
447 | GTK_CSS_PARSE_LENGTH))
450 if (!_gtk_css_parser_has_number (parser))
451 corner.vertical = corner.horizontal;
452 else if (!_gtk_css_parser_read_number (parser,
454 GTK_CSS_POSITIVE_ONLY
455 | GTK_CSS_PARSE_PERCENT
456 | GTK_CSS_NUMBER_AS_PIXELS
457 | GTK_CSS_PARSE_LENGTH))
460 return _gtk_css_value_new_from_border_corner_radius (&corner);
464 border_corner_radius_value_print (GtkCssStyleProperty *property,
465 const GtkCssValue *value,
468 const GtkCssBorderCornerRadius *corner;
470 corner = _gtk_css_value_get_border_corner_radius (value);
472 _gtk_css_number_print (&corner->horizontal, string);
474 if (!_gtk_css_number_equal (&corner->horizontal, &corner->vertical))
476 g_string_append_c (string, ' ');
477 _gtk_css_number_print (&corner->vertical, string);
482 css_image_value_parse (GtkCssStyleProperty *property,
483 GtkCssParser *parser,
488 if (_gtk_css_parser_try (parser, "none", TRUE))
492 image = _gtk_css_image_new_parse (parser, base);
497 return _gtk_css_value_new_take_image (image);
501 css_image_value_print (GtkCssStyleProperty *property,
502 const GtkCssValue *value,
505 GtkCssImage *image = _gtk_css_value_get_image (value);
508 _gtk_css_image_print (image, string);
510 g_string_append (string, "none");
514 css_image_value_compute (GtkCssStyleProperty *property,
515 GtkStyleContext *context,
516 GtkCssValue *specified)
518 GtkCssImage *image, *computed;
520 image = _gtk_css_value_get_image (specified);
523 return _gtk_css_value_ref (specified);
525 computed = _gtk_css_image_compute (image, context);
527 if (computed == image)
529 g_object_unref (computed);
530 return _gtk_css_value_ref (specified);
533 return _gtk_css_value_new_take_image (computed);
537 css_image_value_query (GtkCssStyleProperty *property,
538 const GtkCssValue *css_value,
541 GtkCssImage *image = _gtk_css_value_get_image (css_value);
542 cairo_pattern_t *pattern;
543 cairo_surface_t *surface;
544 cairo_matrix_t matrix;
546 g_value_init (value, CAIRO_GOBJECT_TYPE_PATTERN);
548 if (GTK_IS_CSS_IMAGE_GRADIENT (image))
549 g_value_set_boxed (value, GTK_CSS_IMAGE_GRADIENT (image)->pattern);
550 else if (image != NULL)
552 double width, height;
554 /* the 100, 100 is rather random */
555 _gtk_css_image_get_concrete_size (image, 0, 0, 100, 100, &width, &height);
556 surface = _gtk_css_image_get_surface (image, NULL, width, height);
557 pattern = cairo_pattern_create_for_surface (surface);
558 cairo_matrix_init_scale (&matrix, width, height);
559 cairo_pattern_set_matrix (pattern, &matrix);
560 cairo_surface_destroy (surface);
561 g_value_take_boxed (value, pattern);
566 font_size_parse (GtkCssStyleProperty *property,
567 GtkCssParser *parser,
572 if (!_gtk_css_parser_try_double (parser, &d))
574 _gtk_css_parser_error (parser, "Expected a number");
578 return _gtk_css_value_new_from_double (d);
582 outline_parse (GtkCssStyleProperty *property,
583 GtkCssParser *parser,
588 if (!_gtk_css_parser_try_int (parser, &i))
590 _gtk_css_parser_error (parser, "Expected an integer");
594 return _gtk_css_value_new_from_int (i);
598 border_image_repeat_parse (GtkCssStyleProperty *property,
599 GtkCssParser *parser,
602 GValue value = G_VALUE_INIT;
605 g_value_init (&value, GTK_TYPE_CSS_BORDER_IMAGE_REPEAT);
606 if (!_gtk_css_style_parse_value (&value, parser, base))
608 g_value_unset (&value);
612 result = _gtk_css_value_new_from_gvalue (&value);
613 g_value_unset (&value);
619 border_image_slice_parse (GtkCssStyleProperty *property,
620 GtkCssParser *parser,
623 GValue value = G_VALUE_INIT;
626 g_value_init (&value, GTK_TYPE_BORDER);
627 if (!_gtk_css_style_parse_value (&value, parser, base))
629 g_value_unset (&value);
633 result = _gtk_css_value_new_from_gvalue (&value);
634 g_value_unset (&value);
640 border_image_width_parse (GtkCssStyleProperty *property,
641 GtkCssParser *parser,
644 GValue value = G_VALUE_INIT;
647 g_value_init (&value, GTK_TYPE_BORDER);
648 if (!_gtk_css_style_parse_value (&value, parser, base))
650 g_value_unset (&value);
654 result = _gtk_css_value_new_from_gvalue (&value);
655 g_value_unset (&value);
661 engine_parse (GtkCssStyleProperty *property,
662 GtkCssParser *parser,
665 GtkThemingEngine *engine;
668 if (_gtk_css_parser_try (parser, "none", TRUE))
669 return _gtk_css_value_new_from_theming_engine (gtk_theming_engine_load (NULL));
671 str = _gtk_css_parser_try_ident (parser, TRUE);
674 _gtk_css_parser_error (parser, "Expected a valid theme name");
678 engine = gtk_theming_engine_load (str);
682 _gtk_css_parser_error (parser, "Theming engine '%s' not found", str);
689 return _gtk_css_value_new_from_theming_engine (engine);
693 transition_parse (GtkCssStyleProperty *property,
694 GtkCssParser *parser,
697 GValue value = G_VALUE_INIT;
700 g_value_init (&value, GTK_TYPE_ANIMATION_DESCRIPTION);
701 if (!_gtk_css_style_parse_value (&value, parser, base))
703 g_value_unset (&value);
707 result = _gtk_css_value_new_from_gvalue (&value);
708 g_value_unset (&value);
714 parse_margin (GtkCssStyleProperty *property,
715 GtkCssParser *parser,
718 return _gtk_css_number_value_parse (parser,
719 GTK_CSS_NUMBER_AS_PIXELS
720 | GTK_CSS_PARSE_LENGTH);
724 compute_margin (GtkCssStyleProperty *property,
725 GtkStyleContext *context,
726 GtkCssValue *specified)
728 return _gtk_css_number_value_compute (specified, context);
732 parse_padding (GtkCssStyleProperty *property,
733 GtkCssParser *parser,
736 return _gtk_css_number_value_parse (parser,
737 GTK_CSS_POSITIVE_ONLY
738 | GTK_CSS_NUMBER_AS_PIXELS
739 | GTK_CSS_PARSE_LENGTH);
743 compute_padding (GtkCssStyleProperty *property,
744 GtkStyleContext *context,
745 GtkCssValue *specified)
747 return _gtk_css_number_value_compute (specified, context);
751 parse_border_width (GtkCssStyleProperty *property,
752 GtkCssParser *parser,
755 return _gtk_css_number_value_parse (parser,
756 GTK_CSS_POSITIVE_ONLY
757 | GTK_CSS_NUMBER_AS_PIXELS
758 | GTK_CSS_PARSE_LENGTH);
762 compute_border_width (GtkCssStyleProperty *property,
763 GtkStyleContext *context,
764 GtkCssValue *specified)
766 GtkCssStyleProperty *style;
767 GtkBorderStyle border_style;
769 /* The -1 is magic that is only true because we register the style
770 * properties directly after the width properties.
772 style = _gtk_css_style_property_lookup_by_id (_gtk_css_style_property_get_id (property) - 1);
774 border_style = _gtk_css_value_get_border_style (_gtk_style_context_peek_property (context, _gtk_style_property_get_name (GTK_STYLE_PROPERTY (style))));
776 if (border_style == GTK_BORDER_STYLE_NONE ||
777 border_style == GTK_BORDER_STYLE_HIDDEN)
778 return _gtk_css_number_value_new (0, GTK_CSS_PX);
780 return _gtk_css_number_value_compute (specified, context);
784 background_repeat_value_parse (GtkCssStyleProperty *property,
785 GtkCssParser *parser,
788 int repeat, vertical;
790 if (!_gtk_css_parser_try_enum (parser, GTK_TYPE_CSS_BACKGROUND_REPEAT, &repeat))
792 _gtk_css_parser_error (parser, "Not a valid value");
796 if (repeat <= GTK_CSS_BACKGROUND_REPEAT_MASK)
798 if (_gtk_css_parser_try_enum (parser, GTK_TYPE_CSS_BACKGROUND_REPEAT, &vertical))
800 if (vertical >= GTK_CSS_BACKGROUND_REPEAT_MASK)
802 _gtk_css_parser_error (parser, "Not a valid 2nd value");
806 repeat |= vertical << GTK_CSS_BACKGROUND_REPEAT_SHIFT;
809 repeat |= repeat << GTK_CSS_BACKGROUND_REPEAT_SHIFT;
812 return _gtk_css_value_new_from_enum (GTK_TYPE_CSS_BACKGROUND_REPEAT, repeat);
816 background_repeat_value_print (GtkCssStyleProperty *property,
817 const GtkCssValue *value,
820 GEnumClass *enum_class;
821 GEnumValue *enum_value;
822 GtkCssBackgroundRepeat repeat;
824 repeat = _gtk_css_value_get_enum (value);
825 enum_class = g_type_class_ref (GTK_TYPE_CSS_BACKGROUND_REPEAT);
826 enum_value = g_enum_get_value (enum_class, repeat);
828 /* only triggers for 'repeat-x' and 'repeat-y' */
830 g_string_append (string, enum_value->value_nick);
833 enum_value = g_enum_get_value (enum_class, GTK_CSS_BACKGROUND_HORIZONTAL (repeat));
834 g_string_append (string, enum_value->value_nick);
836 if (GTK_CSS_BACKGROUND_HORIZONTAL (repeat) != GTK_CSS_BACKGROUND_VERTICAL (repeat))
838 enum_value = g_enum_get_value (enum_class, GTK_CSS_BACKGROUND_VERTICAL (repeat));
839 g_string_append (string, " ");
840 g_string_append (string, enum_value->value_nick);
844 g_type_class_unref (enum_class);
848 background_size_parse (GtkCssStyleProperty *property,
849 GtkCssParser *parser,
852 GtkCssBackgroundSize size = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), FALSE, FALSE};
854 if (_gtk_css_parser_try (parser, "cover", TRUE))
856 else if (_gtk_css_parser_try (parser, "contain", TRUE))
860 if (_gtk_css_parser_try (parser, "auto", TRUE))
861 _gtk_css_number_init (&size.width, 0, GTK_CSS_PX);
862 else if (!_gtk_css_parser_read_number (parser,
864 GTK_CSS_POSITIVE_ONLY
865 | GTK_CSS_PARSE_PERCENT
866 | GTK_CSS_PARSE_LENGTH))
869 if (_gtk_css_parser_try (parser, "auto", TRUE))
870 _gtk_css_number_init (&size.height, 0, GTK_CSS_PX);
871 else if (_gtk_css_parser_has_number (parser))
873 if (!_gtk_css_parser_read_number (parser,
875 GTK_CSS_POSITIVE_ONLY
876 | GTK_CSS_PARSE_PERCENT
877 | GTK_CSS_PARSE_LENGTH))
881 _gtk_css_number_init (&size.height, 0, GTK_CSS_PX);
884 return _gtk_css_value_new_from_background_size (&size);
888 background_size_print (GtkCssStyleProperty *property,
889 const GtkCssValue *value,
892 const GtkCssBackgroundSize *size = _gtk_css_value_get_background_size (value);
895 g_string_append (string, "cover");
896 else if (size->contain)
897 g_string_append (string, "contain");
900 if (size->width.value == 0)
901 g_string_append (string, "auto");
903 _gtk_css_number_print (&size->width, string);
905 if (size->height.value != 0)
907 g_string_append (string, " ");
908 _gtk_css_number_print (&size->height, string);
914 background_size_compute (GtkCssStyleProperty *property,
915 GtkStyleContext *context,
916 GtkCssValue *specified)
918 const GtkCssBackgroundSize *ssize = _gtk_css_value_get_background_size (specified);
919 GtkCssBackgroundSize csize;
922 csize.cover = ssize->cover;
923 csize.contain = ssize->contain;
924 changed = _gtk_css_number_compute (&csize.width,
927 changed |= _gtk_css_number_compute (&csize.height,
931 return _gtk_css_value_new_from_background_size (&csize);
932 return _gtk_css_value_ref (specified);
936 background_position_parse (GtkCssStyleProperty *property,
937 GtkCssParser *parser,
940 static const struct {
946 { "left", 0, TRUE, FALSE },
947 { "right", 100, TRUE, FALSE },
948 { "center", 50, TRUE, TRUE },
949 { "top", 0, FALSE, TRUE },
950 { "bottom", 100, FALSE, TRUE },
951 { NULL , 0, TRUE, FALSE }, /* used for numbers */
952 { NULL , 50, TRUE, TRUE } /* used for no value */
954 GtkCssBackgroundPosition pos;
955 GtkCssNumber *missing;
958 for (first = 0; names[first].name != NULL; first++)
960 if (_gtk_css_parser_try (parser, names[first].name, TRUE))
962 if (names[first].horizontal)
964 _gtk_css_number_init (&pos.x, names[first].percentage, GTK_CSS_PERCENT);
969 _gtk_css_number_init (&pos.y, names[first].percentage, GTK_CSS_PERCENT);
975 if (names[first].name == NULL)
978 if (!_gtk_css_parser_read_number (parser,
980 GTK_CSS_PARSE_PERCENT
981 | GTK_CSS_PARSE_LENGTH))
985 for (second = 0; names[second].name != NULL; second++)
987 if (_gtk_css_parser_try (parser, names[second].name, TRUE))
989 _gtk_css_number_init (missing, names[second].percentage, GTK_CSS_PERCENT);
994 if (names[second].name == NULL)
996 if (_gtk_css_parser_has_number (parser))
998 if (missing != &pos.y)
1000 _gtk_css_parser_error (parser, "Invalid combination of values");
1003 if (!_gtk_css_parser_read_number (parser,
1005 GTK_CSS_PARSE_PERCENT
1006 | GTK_CSS_PARSE_LENGTH))
1012 _gtk_css_number_init (missing, 50, GTK_CSS_PERCENT);
1017 if ((names[first].horizontal && !names[second].vertical) ||
1018 (!names[first].horizontal && !names[second].horizontal))
1020 _gtk_css_parser_error (parser, "Invalid combination of values");
1025 return _gtk_css_value_new_from_background_position (&pos);
1029 background_position_print (GtkCssStyleProperty *property,
1030 const GtkCssValue *value,
1033 const GtkCssBackgroundPosition *pos = _gtk_css_value_get_background_position (value);
1034 static const GtkCssNumber center = GTK_CSS_NUMBER_INIT (50, GTK_CSS_PERCENT);
1035 static const struct {
1038 GtkCssNumber number;
1040 { "left", "top", GTK_CSS_NUMBER_INIT (0, GTK_CSS_PERCENT) },
1041 { "right", "bottom", GTK_CSS_NUMBER_INIT (100, GTK_CSS_PERCENT) }
1045 if (_gtk_css_number_equal (&pos->x, ¢er))
1047 if (_gtk_css_number_equal (&pos->y, ¢er))
1049 g_string_append (string, "center");
1055 for (i = 0; i < G_N_ELEMENTS (values); i++)
1057 if (_gtk_css_number_equal (&pos->x, &values[i].number))
1059 g_string_append (string, values[i].x_name);
1063 if (i == G_N_ELEMENTS (values))
1064 _gtk_css_number_print (&pos->x, string);
1066 if (_gtk_css_number_equal (&pos->y, ¢er))
1069 g_string_append_c (string, ' ');
1072 for (i = 0; i < G_N_ELEMENTS (values); i++)
1074 if (_gtk_css_number_equal (&pos->y, &values[i].number))
1076 g_string_append (string, values[i].y_name);
1080 if (i == G_N_ELEMENTS (values))
1082 if (_gtk_css_number_equal (&pos->x, ¢er))
1083 g_string_append (string, "center ");
1084 _gtk_css_number_print (&pos->y, string);
1088 static GtkCssValue *
1089 background_position_compute (GtkCssStyleProperty *property,
1090 GtkStyleContext *context,
1091 GtkCssValue *specified)
1093 const GtkCssBackgroundPosition *spos = _gtk_css_value_get_background_position (specified);
1094 GtkCssBackgroundPosition cpos;
1097 changed = _gtk_css_number_compute (&cpos.x,
1100 changed |= _gtk_css_number_compute (&cpos.y,
1104 return _gtk_css_value_new_from_background_position (&cpos);
1105 return _gtk_css_value_ref (specified);
1108 /*** REGISTRATION ***/
1110 static GtkSymbolicColor *
1111 gtk_symbolic_color_new_rgba (double red,
1116 GdkRGBA rgba = { red, green, blue, alpha };
1118 return gtk_symbolic_color_new_literal (&rgba);
1122 _gtk_css_style_property_init_properties (void)
1124 char *default_font_family[] = { "Sans", NULL };
1125 GtkCssBackgroundSize default_background_size = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), FALSE, FALSE };
1126 GtkCssBackgroundPosition default_background_position = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PERCENT), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PERCENT)};
1127 GtkCssBorderCornerRadius no_corner_radius = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX) };
1128 GtkBorder border_of_ones = { 1, 1, 1, 1 };
1129 GtkCssBorderImageRepeat border_image_repeat = { GTK_CSS_REPEAT_STYLE_STRETCH, GTK_CSS_REPEAT_STYLE_STRETCH };
1131 /* Initialize "color" and "font-size" first,
1132 * so that when computing values later they are
1133 * done first. That way, 'currentColor' and font
1134 * sizes in em can be looked up properly */
1135 gtk_css_style_property_register ("color",
1137 GTK_STYLE_PROPERTY_INHERIT,
1143 _gtk_css_value_new_take_symbolic_color (
1144 gtk_symbolic_color_new_rgba (1, 1, 1, 1)));
1145 gtk_css_style_property_register ("font-size",
1147 GTK_STYLE_PROPERTY_INHERIT,
1153 _gtk_css_value_new_from_double (10.0));
1155 /* properties that aren't referenced when computing values
1157 gtk_css_style_property_register ("background-color",
1165 _gtk_css_value_new_take_symbolic_color (
1166 gtk_symbolic_color_new_rgba (0, 0, 0, 0)));
1168 gtk_css_style_property_register ("font-family",
1170 GTK_STYLE_PROPERTY_INHERIT,
1172 font_family_value_print,
1176 _gtk_css_value_new_take_strv (g_strdupv (default_font_family)));
1177 gtk_css_style_property_register ("font-style",
1179 GTK_STYLE_PROPERTY_INHERIT,
1185 _gtk_css_value_new_from_enum (PANGO_TYPE_STYLE,
1186 PANGO_STYLE_NORMAL));
1187 gtk_css_style_property_register ("font-variant",
1189 GTK_STYLE_PROPERTY_INHERIT,
1190 parse_pango_variant,
1195 _gtk_css_value_new_from_enum (PANGO_TYPE_VARIANT,
1196 PANGO_VARIANT_NORMAL));
1197 /* xxx: need to parse this properly, ie parse the numbers */
1198 gtk_css_style_property_register ("font-weight",
1200 GTK_STYLE_PROPERTY_INHERIT,
1206 _gtk_css_value_new_from_enum (PANGO_TYPE_WEIGHT,
1207 PANGO_WEIGHT_NORMAL));
1209 gtk_css_style_property_register ("text-shadow",
1211 GTK_STYLE_PROPERTY_INHERIT,
1214 shadow_value_compute,
1217 _gtk_css_shadow_value_new_none ());
1219 gtk_css_style_property_register ("icon-shadow",
1221 GTK_STYLE_PROPERTY_INHERIT,
1224 shadow_value_compute,
1227 _gtk_css_shadow_value_new_none ());
1229 gtk_css_style_property_register ("box-shadow",
1234 shadow_value_compute,
1237 _gtk_css_shadow_value_new_none ());
1239 gtk_css_style_property_register ("margin-top",
1245 query_length_as_int,
1247 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1248 gtk_css_style_property_register ("margin-left",
1254 query_length_as_int,
1256 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1257 gtk_css_style_property_register ("margin-bottom",
1263 query_length_as_int,
1265 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1266 gtk_css_style_property_register ("margin-right",
1272 query_length_as_int,
1274 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1275 gtk_css_style_property_register ("padding-top",
1281 query_length_as_int,
1283 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1284 gtk_css_style_property_register ("padding-left",
1290 query_length_as_int,
1292 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1293 gtk_css_style_property_register ("padding-bottom",
1299 query_length_as_int,
1301 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1302 gtk_css_style_property_register ("padding-right",
1308 query_length_as_int,
1310 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1311 /* IMPORTANT: compute_border_width() requires that the border-width
1312 * properties be immeditaly followed by the border-style properties
1314 gtk_css_style_property_register ("border-top-style",
1315 GTK_TYPE_BORDER_STYLE,
1322 _gtk_css_value_new_from_border_style (GTK_BORDER_STYLE_NONE));
1323 gtk_css_style_property_register ("border-top-width",
1328 compute_border_width,
1329 query_length_as_int,
1331 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1332 gtk_css_style_property_register ("border-left-style",
1333 GTK_TYPE_BORDER_STYLE,
1340 _gtk_css_value_new_from_border_style (GTK_BORDER_STYLE_NONE));
1341 gtk_css_style_property_register ("border-left-width",
1346 compute_border_width,
1347 query_length_as_int,
1349 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1350 gtk_css_style_property_register ("border-bottom-style",
1351 GTK_TYPE_BORDER_STYLE,
1358 _gtk_css_value_new_from_border_style (GTK_BORDER_STYLE_NONE));
1359 gtk_css_style_property_register ("border-bottom-width",
1364 compute_border_width,
1365 query_length_as_int,
1367 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1368 gtk_css_style_property_register ("border-right-style",
1369 GTK_TYPE_BORDER_STYLE,
1376 _gtk_css_value_new_from_border_style (GTK_BORDER_STYLE_NONE));
1377 gtk_css_style_property_register ("border-right-width",
1382 compute_border_width,
1383 query_length_as_int,
1385 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1387 gtk_css_style_property_register ("border-top-left-radius",
1388 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1390 border_corner_radius_value_parse,
1391 border_corner_radius_value_print,
1395 _gtk_css_value_new_from_border_corner_radius (&no_corner_radius));
1396 gtk_css_style_property_register ("border-top-right-radius",
1397 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1399 border_corner_radius_value_parse,
1400 border_corner_radius_value_print,
1404 _gtk_css_value_new_from_border_corner_radius (&no_corner_radius));
1405 gtk_css_style_property_register ("border-bottom-right-radius",
1406 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1408 border_corner_radius_value_parse,
1409 border_corner_radius_value_print,
1413 _gtk_css_value_new_from_border_corner_radius (&no_corner_radius));
1414 gtk_css_style_property_register ("border-bottom-left-radius",
1415 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1417 border_corner_radius_value_parse,
1418 border_corner_radius_value_print,
1422 _gtk_css_value_new_from_border_corner_radius (&no_corner_radius));
1424 gtk_css_style_property_register ("outline-style",
1425 GTK_TYPE_BORDER_STYLE,
1432 _gtk_css_value_new_from_border_style (GTK_BORDER_STYLE_NONE));
1433 gtk_css_style_property_register ("outline-width",
1438 compute_border_width,
1439 query_length_as_int,
1441 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1442 gtk_css_style_property_register ("outline-offset",
1450 _gtk_css_value_new_from_int (0));
1452 gtk_css_style_property_register ("background-clip",
1460 _gtk_css_value_new_from_enum (GTK_TYPE_CSS_AREA, GTK_CSS_AREA_BORDER_BOX));
1461 gtk_css_style_property_register ("background-origin",
1469 _gtk_css_value_new_from_enum (GTK_TYPE_CSS_AREA, GTK_CSS_AREA_PADDING_BOX));
1470 gtk_css_style_property_register ("background-size",
1473 background_size_parse,
1474 background_size_print,
1475 background_size_compute,
1478 _gtk_css_value_new_from_background_size (&default_background_size));
1479 gtk_css_style_property_register ("background-position",
1482 background_position_parse,
1483 background_position_print,
1484 background_position_compute,
1487 _gtk_css_value_new_from_background_position (&default_background_position));
1489 gtk_css_style_property_register ("border-top-color",
1497 _gtk_css_value_new_take_symbolic_color (
1498 gtk_symbolic_color_ref (
1499 _gtk_symbolic_color_get_current_color ())));
1500 gtk_css_style_property_register ("border-right-color",
1508 _gtk_css_value_new_take_symbolic_color (
1509 gtk_symbolic_color_ref (
1510 _gtk_symbolic_color_get_current_color ())));
1511 gtk_css_style_property_register ("border-bottom-color",
1519 _gtk_css_value_new_take_symbolic_color (
1520 gtk_symbolic_color_ref (
1521 _gtk_symbolic_color_get_current_color ())));
1522 gtk_css_style_property_register ("border-left-color",
1530 _gtk_css_value_new_take_symbolic_color (
1531 gtk_symbolic_color_ref (
1532 _gtk_symbolic_color_get_current_color ())));
1533 gtk_css_style_property_register ("outline-color",
1541 _gtk_css_value_new_take_symbolic_color (
1542 gtk_symbolic_color_ref (
1543 _gtk_symbolic_color_get_current_color ())));
1545 gtk_css_style_property_register ("background-repeat",
1546 GTK_TYPE_CSS_BACKGROUND_REPEAT,
1548 background_repeat_value_parse,
1549 background_repeat_value_print,
1553 _gtk_css_value_new_from_enum (GTK_TYPE_CSS_BACKGROUND_REPEAT,
1554 GTK_CSS_BACKGROUND_REPEAT |
1555 (GTK_CSS_BACKGROUND_REPEAT << GTK_CSS_BACKGROUND_REPEAT_SHIFT)));
1556 gtk_css_style_property_register ("background-image",
1557 CAIRO_GOBJECT_TYPE_PATTERN,
1559 css_image_value_parse,
1560 css_image_value_print,
1561 css_image_value_compute,
1562 css_image_value_query,
1564 _gtk_css_value_new_take_image (NULL));
1566 gtk_css_style_property_register ("border-image-source",
1567 CAIRO_GOBJECT_TYPE_PATTERN,
1569 css_image_value_parse,
1570 css_image_value_print,
1571 css_image_value_compute,
1572 css_image_value_query,
1574 _gtk_css_value_new_take_image (NULL));
1575 gtk_css_style_property_register ("border-image-repeat",
1576 GTK_TYPE_CSS_BORDER_IMAGE_REPEAT,
1578 border_image_repeat_parse,
1583 _gtk_css_value_new_from_border_image_repeat (&border_image_repeat));
1585 /* XXX: The initial value is wrong, it should be 100% */
1586 gtk_css_style_property_register ("border-image-slice",
1589 border_image_slice_parse,
1594 _gtk_css_value_new_from_boxed (GTK_TYPE_BORDER, &border_of_ones));
1595 gtk_css_style_property_register ("border-image-width",
1598 border_image_width_parse,
1603 _gtk_css_value_new_from_boxed (GTK_TYPE_BORDER, NULL));
1604 gtk_css_style_property_register ("engine",
1605 GTK_TYPE_THEMING_ENGINE,
1612 _gtk_css_value_new_from_theming_engine (gtk_theming_engine_load (NULL)));
1613 gtk_css_style_property_register ("transition",
1614 GTK_TYPE_ANIMATION_DESCRIPTION,
1621 _gtk_css_value_new_from_boxed (GTK_TYPE_ANIMATION_DESCRIPTION, NULL));
1623 /* Private property holding the binding sets */
1624 gtk_css_style_property_register ("gtk-key-bindings",
1627 bindings_value_parse,
1628 bindings_value_print,
1632 _gtk_css_value_new_take_binding_sets (NULL));