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 "gtkcssenumvalueprivate.h"
47 #include "gtkcssnumbervalueprivate.h"
48 #include "gtkcssrgbavalueprivate.h"
49 #include "gtkcssshadowvalueprivate.h"
50 #include "gtksymboliccolorprivate.h"
51 #include "gtkthemingengine.h"
52 #include "gtktypebuiltins.h"
53 #include "gtkwin32themeprivate.h"
55 /*** REGISTRATION ***/
58 gtk_css_style_property_register (const char * name,
60 GtkStylePropertyFlags flags,
61 GtkCssStylePropertyParseFunc parse_value,
62 GtkCssStylePropertyPrintFunc print_value,
63 GtkCssStylePropertyComputeFunc compute_value,
64 GtkCssStylePropertyQueryFunc query_value,
65 GtkCssStylePropertyAssignFunc assign_value,
66 GtkCssStylePropertyEqualFunc equal_func,
67 GtkCssValue * initial_value)
69 GtkCssStyleProperty *node;
71 g_assert (initial_value != NULL);
72 g_assert (parse_value != NULL);
73 g_assert (value_type == G_TYPE_NONE || query_value != NULL);
74 g_assert (value_type == G_TYPE_NONE || assign_value != NULL);
76 node = g_object_new (GTK_TYPE_CSS_STYLE_PROPERTY,
77 "value-type", value_type,
78 "inherit", (flags & GTK_STYLE_PROPERTY_INHERIT) ? TRUE : FALSE,
79 "initial-value", initial_value,
83 node->parse_value = parse_value;
85 node->print_value = print_value;
87 node->compute_value = compute_value;
88 node->query_value = query_value;
89 node->assign_value = assign_value;
91 node->equal_func = equal_func;
93 _gtk_css_value_unref (initial_value);
99 string_append_string (GString *str,
104 g_string_append_c (str, '"');
107 len = strcspn (string, "\"\n\r\f");
108 g_string_append (str, string);
115 g_string_append (str, "\\A ");
118 g_string_append (str, "\\D ");
121 g_string_append (str, "\\C ");
124 g_string_append (str, "\\\"");
127 g_assert_not_reached ();
132 g_string_append_c (str, '"');
135 /*** IMPLEMENTATIONS ***/
138 query_simple (GtkCssStyleProperty *property,
139 const GtkCssValue *css_value,
142 _gtk_css_value_init_gvalue (css_value, value);
146 assign_simple (GtkCssStyleProperty *property,
149 return _gtk_css_value_new_from_gvalue (value);
153 query_length_as_int (GtkCssStyleProperty *property,
154 const GtkCssValue *css_value,
157 g_value_init (value, G_TYPE_INT);
158 g_value_set_int (value, round (_gtk_css_number_value_get (css_value, 100)));
162 assign_length_from_int (GtkCssStyleProperty *property,
165 return _gtk_css_number_value_new (g_value_get_int (value), GTK_CSS_PX);
169 color_parse (GtkCssStyleProperty *property,
170 GtkCssParser *parser,
173 GtkSymbolicColor *symbolic;
175 if (_gtk_css_parser_try (parser, "currentcolor", TRUE))
177 symbolic = gtk_symbolic_color_ref (_gtk_symbolic_color_get_current_color ());
181 symbolic = _gtk_css_parser_read_symbolic_color (parser);
182 if (symbolic == NULL)
186 return _gtk_css_value_new_take_symbolic_color (symbolic);
190 color_compute (GtkCssStyleProperty *property,
191 GtkStyleContext *context,
192 GtkCssValue *specified)
194 return _gtk_css_rgba_value_compute_from_symbolic (specified,
195 _gtk_css_style_property_get_initial_value (property),
201 color_property_compute (GtkCssStyleProperty *property,
202 GtkStyleContext *context,
203 GtkCssValue *specified)
207 value = _gtk_css_rgba_value_compute_from_symbolic (specified,
208 _gtk_css_style_property_get_initial_value (property),
211 _gtk_css_rgba_value_get_rgba (value);
216 color_query (GtkCssStyleProperty *property,
217 const GtkCssValue *css_value,
220 g_value_init (value, GDK_TYPE_RGBA);
221 g_value_set_boxed (value, _gtk_css_rgba_value_get_rgba (css_value));
225 color_assign (GtkCssStyleProperty *property,
228 return _gtk_css_rgba_value_new_from_rgba (g_value_get_boxed (value));
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,
304 GtkCssValue *value = _gtk_css_font_style_value_try_parse (parser);
307 _gtk_css_parser_error (parser, "unknown value for property");
313 query_pango_style (GtkCssStyleProperty *property,
314 const GtkCssValue *css_value,
317 g_value_init (value, PANGO_TYPE_STYLE);
318 g_value_set_enum (value, _gtk_css_font_style_value_get (css_value));
322 assign_pango_style (GtkCssStyleProperty *property,
325 return _gtk_css_font_style_value_new (g_value_get_enum (value));
329 parse_pango_weight (GtkCssStyleProperty *property,
330 GtkCssParser *parser,
333 GtkCssValue *value = _gtk_css_font_weight_value_try_parse (parser);
336 _gtk_css_parser_error (parser, "unknown value for property");
342 query_pango_weight (GtkCssStyleProperty *property,
343 const GtkCssValue *css_value,
346 g_value_init (value, PANGO_TYPE_WEIGHT);
347 g_value_set_enum (value, _gtk_css_font_weight_value_get (css_value));
351 assign_pango_weight (GtkCssStyleProperty *property,
354 return _gtk_css_font_weight_value_new (g_value_get_enum (value));
358 parse_pango_variant (GtkCssStyleProperty *property,
359 GtkCssParser *parser,
362 GtkCssValue *value = _gtk_css_font_variant_value_try_parse (parser);
365 _gtk_css_parser_error (parser, "unknown value for property");
371 query_pango_variant (GtkCssStyleProperty *property,
372 const GtkCssValue *css_value,
375 g_value_init (value, PANGO_TYPE_VARIANT);
376 g_value_set_enum (value, _gtk_css_font_variant_value_get (css_value));
380 assign_pango_variant (GtkCssStyleProperty *property,
383 return _gtk_css_font_variant_value_new (g_value_get_enum (value));
387 parse_border_style (GtkCssStyleProperty *property,
388 GtkCssParser *parser,
391 GtkCssValue *value = _gtk_css_border_style_value_try_parse (parser);
394 _gtk_css_parser_error (parser, "unknown value for property");
400 query_border_style (GtkCssStyleProperty *property,
401 const GtkCssValue *css_value,
404 g_value_init (value, GTK_TYPE_BORDER_STYLE);
405 g_value_set_enum (value, _gtk_css_border_style_value_get (css_value));
409 assign_border_style (GtkCssStyleProperty *property,
412 return _gtk_css_border_style_value_new (g_value_get_enum (value));
416 parse_css_area (GtkCssStyleProperty *property,
417 GtkCssParser *parser,
422 if (!_gtk_css_parser_try_enum (parser, GTK_TYPE_CSS_AREA, &value))
424 _gtk_css_parser_error (parser, "unknown value for property");
428 return _gtk_css_value_new_from_enum (GTK_TYPE_CSS_AREA, value);
432 bindings_value_parse (GtkCssStyleProperty *property,
433 GtkCssParser *parser,
437 GtkBindingSet *binding_set;
440 array = g_ptr_array_new ();
443 name = _gtk_css_parser_try_ident (parser, TRUE);
446 _gtk_css_parser_error (parser, "Not a valid binding name");
447 g_ptr_array_free (array, TRUE);
451 binding_set = gtk_binding_set_find (name);
455 _gtk_css_parser_error (parser, "No binding set named '%s'", name);
460 g_ptr_array_add (array, binding_set);
463 while (_gtk_css_parser_try (parser, ",", TRUE));
465 return _gtk_css_value_new_take_binding_sets (array);
469 bindings_value_print (GtkCssStyleProperty *property,
470 const GtkCssValue *value,
476 array = _gtk_css_value_get_boxed (value);
478 for (i = 0; i < array->len; i++)
480 GtkBindingSet *binding_set = g_ptr_array_index (array, i);
483 g_string_append (string, ", ");
484 g_string_append (string, binding_set->set_name);
489 shadow_value_parse (GtkCssStyleProperty *property,
490 GtkCssParser *parser,
493 return _gtk_css_shadow_value_parse (parser);
497 shadow_value_compute (GtkCssStyleProperty *property,
498 GtkStyleContext *context,
499 GtkCssValue *specified)
501 return _gtk_css_shadow_value_compute (specified, context);
505 border_corner_radius_value_parse (GtkCssStyleProperty *property,
506 GtkCssParser *parser,
509 GtkCssBorderCornerRadius corner;
511 if (!_gtk_css_parser_read_number (parser,
513 GTK_CSS_POSITIVE_ONLY
514 | GTK_CSS_PARSE_PERCENT
515 | GTK_CSS_NUMBER_AS_PIXELS
516 | GTK_CSS_PARSE_LENGTH))
519 if (!_gtk_css_parser_has_number (parser))
520 corner.vertical = corner.horizontal;
521 else if (!_gtk_css_parser_read_number (parser,
523 GTK_CSS_POSITIVE_ONLY
524 | GTK_CSS_PARSE_PERCENT
525 | GTK_CSS_NUMBER_AS_PIXELS
526 | GTK_CSS_PARSE_LENGTH))
529 return _gtk_css_value_new_from_border_corner_radius (&corner);
533 border_corner_radius_value_print (GtkCssStyleProperty *property,
534 const GtkCssValue *value,
537 const GtkCssBorderCornerRadius *corner;
539 corner = _gtk_css_value_get_border_corner_radius (value);
541 _gtk_css_number_print (&corner->horizontal, string);
543 if (!_gtk_css_number_equal (&corner->horizontal, &corner->vertical))
545 g_string_append_c (string, ' ');
546 _gtk_css_number_print (&corner->vertical, string);
551 css_image_value_parse (GtkCssStyleProperty *property,
552 GtkCssParser *parser,
557 if (_gtk_css_parser_try (parser, "none", TRUE))
561 image = _gtk_css_image_new_parse (parser, base);
566 return _gtk_css_value_new_take_image (image);
570 css_image_value_print (GtkCssStyleProperty *property,
571 const GtkCssValue *value,
574 GtkCssImage *image = _gtk_css_value_get_image (value);
577 _gtk_css_image_print (image, string);
579 g_string_append (string, "none");
583 css_image_value_compute (GtkCssStyleProperty *property,
584 GtkStyleContext *context,
585 GtkCssValue *specified)
587 GtkCssImage *image, *computed;
589 image = _gtk_css_value_get_image (specified);
592 return _gtk_css_value_ref (specified);
594 computed = _gtk_css_image_compute (image, context);
596 if (computed == image)
598 g_object_unref (computed);
599 return _gtk_css_value_ref (specified);
602 return _gtk_css_value_new_take_image (computed);
606 css_image_value_query (GtkCssStyleProperty *property,
607 const GtkCssValue *css_value,
610 GtkCssImage *image = _gtk_css_value_get_image (css_value);
611 cairo_pattern_t *pattern;
612 cairo_surface_t *surface;
613 cairo_matrix_t matrix;
615 g_value_init (value, CAIRO_GOBJECT_TYPE_PATTERN);
617 if (GTK_IS_CSS_IMAGE_GRADIENT (image))
618 g_value_set_boxed (value, GTK_CSS_IMAGE_GRADIENT (image)->pattern);
619 else if (image != NULL)
621 double width, height;
623 /* the 100, 100 is rather random */
624 _gtk_css_image_get_concrete_size (image, 0, 0, 100, 100, &width, &height);
625 surface = _gtk_css_image_get_surface (image, NULL, width, height);
626 pattern = cairo_pattern_create_for_surface (surface);
627 cairo_matrix_init_scale (&matrix, width, height);
628 cairo_pattern_set_matrix (pattern, &matrix);
629 cairo_surface_destroy (surface);
630 g_value_take_boxed (value, pattern);
635 css_image_value_assign (GtkCssStyleProperty *property,
638 g_warning ("FIXME: assigning images is not implemented");
639 return _gtk_css_value_new_take_image (NULL);
643 font_size_parse (GtkCssStyleProperty *property,
644 GtkCssParser *parser,
649 if (!_gtk_css_parser_try_double (parser, &d))
651 _gtk_css_parser_error (parser, "Expected a number");
655 return _gtk_css_value_new_from_double (d);
659 outline_parse (GtkCssStyleProperty *property,
660 GtkCssParser *parser,
665 if (!_gtk_css_parser_try_int (parser, &i))
667 _gtk_css_parser_error (parser, "Expected an integer");
671 return _gtk_css_value_new_from_int (i);
675 border_image_repeat_parse (GtkCssStyleProperty *property,
676 GtkCssParser *parser,
679 GValue value = G_VALUE_INIT;
682 g_value_init (&value, GTK_TYPE_CSS_BORDER_IMAGE_REPEAT);
683 if (!_gtk_css_style_parse_value (&value, parser, base))
685 g_value_unset (&value);
689 result = _gtk_css_value_new_from_gvalue (&value);
690 g_value_unset (&value);
696 border_image_slice_parse (GtkCssStyleProperty *property,
697 GtkCssParser *parser,
700 GValue value = G_VALUE_INIT;
703 g_value_init (&value, GTK_TYPE_BORDER);
704 if (!_gtk_css_style_parse_value (&value, parser, base))
706 g_value_unset (&value);
710 result = _gtk_css_value_new_from_gvalue (&value);
711 g_value_unset (&value);
717 border_image_width_parse (GtkCssStyleProperty *property,
718 GtkCssParser *parser,
721 GValue value = G_VALUE_INIT;
724 g_value_init (&value, GTK_TYPE_BORDER);
725 if (!_gtk_css_style_parse_value (&value, parser, base))
727 g_value_unset (&value);
731 result = _gtk_css_value_new_from_gvalue (&value);
732 g_value_unset (&value);
738 engine_parse (GtkCssStyleProperty *property,
739 GtkCssParser *parser,
742 GtkThemingEngine *engine;
745 if (_gtk_css_parser_try (parser, "none", TRUE))
746 return _gtk_css_value_new_from_theming_engine (gtk_theming_engine_load (NULL));
748 str = _gtk_css_parser_try_ident (parser, TRUE);
751 _gtk_css_parser_error (parser, "Expected a valid theme name");
755 engine = gtk_theming_engine_load (str);
759 _gtk_css_parser_error (parser, "Theming engine '%s' not found", str);
766 return _gtk_css_value_new_from_theming_engine (engine);
770 transition_parse (GtkCssStyleProperty *property,
771 GtkCssParser *parser,
774 GValue value = G_VALUE_INIT;
777 g_value_init (&value, GTK_TYPE_ANIMATION_DESCRIPTION);
778 if (!_gtk_css_style_parse_value (&value, parser, base))
780 g_value_unset (&value);
784 result = _gtk_css_value_new_from_gvalue (&value);
785 g_value_unset (&value);
791 parse_margin (GtkCssStyleProperty *property,
792 GtkCssParser *parser,
795 return _gtk_css_number_value_parse (parser,
796 GTK_CSS_NUMBER_AS_PIXELS
797 | GTK_CSS_PARSE_LENGTH);
801 compute_margin (GtkCssStyleProperty *property,
802 GtkStyleContext *context,
803 GtkCssValue *specified)
805 return _gtk_css_number_value_compute (specified, context);
809 parse_padding (GtkCssStyleProperty *property,
810 GtkCssParser *parser,
813 return _gtk_css_number_value_parse (parser,
814 GTK_CSS_POSITIVE_ONLY
815 | GTK_CSS_NUMBER_AS_PIXELS
816 | GTK_CSS_PARSE_LENGTH);
820 compute_padding (GtkCssStyleProperty *property,
821 GtkStyleContext *context,
822 GtkCssValue *specified)
824 return _gtk_css_number_value_compute (specified, context);
828 parse_border_width (GtkCssStyleProperty *property,
829 GtkCssParser *parser,
832 return _gtk_css_number_value_parse (parser,
833 GTK_CSS_POSITIVE_ONLY
834 | GTK_CSS_NUMBER_AS_PIXELS
835 | GTK_CSS_PARSE_LENGTH);
839 compute_border_width (GtkCssStyleProperty *property,
840 GtkStyleContext *context,
841 GtkCssValue *specified)
843 GtkCssStyleProperty *style;
844 GtkBorderStyle border_style;
846 /* The -1 is magic that is only true because we register the style
847 * properties directly after the width properties.
849 style = _gtk_css_style_property_lookup_by_id (_gtk_css_style_property_get_id (property) - 1);
851 border_style = _gtk_css_border_style_value_get (_gtk_style_context_peek_property (context, _gtk_style_property_get_name (GTK_STYLE_PROPERTY (style))));
853 if (border_style == GTK_BORDER_STYLE_NONE ||
854 border_style == GTK_BORDER_STYLE_HIDDEN)
855 return _gtk_css_number_value_new (0, GTK_CSS_PX);
857 return _gtk_css_number_value_compute (specified, context);
861 background_repeat_value_parse (GtkCssStyleProperty *property,
862 GtkCssParser *parser,
865 int repeat, vertical;
867 if (!_gtk_css_parser_try_enum (parser, GTK_TYPE_CSS_BACKGROUND_REPEAT, &repeat))
869 _gtk_css_parser_error (parser, "Not a valid value");
873 if (repeat <= GTK_CSS_BACKGROUND_REPEAT_MASK)
875 if (_gtk_css_parser_try_enum (parser, GTK_TYPE_CSS_BACKGROUND_REPEAT, &vertical))
877 if (vertical >= GTK_CSS_BACKGROUND_REPEAT_MASK)
879 _gtk_css_parser_error (parser, "Not a valid 2nd value");
883 repeat |= vertical << GTK_CSS_BACKGROUND_REPEAT_SHIFT;
886 repeat |= repeat << GTK_CSS_BACKGROUND_REPEAT_SHIFT;
889 return _gtk_css_value_new_from_enum (GTK_TYPE_CSS_BACKGROUND_REPEAT, repeat);
893 background_repeat_value_print (GtkCssStyleProperty *property,
894 const GtkCssValue *value,
897 GEnumClass *enum_class;
898 GEnumValue *enum_value;
899 GtkCssBackgroundRepeat repeat;
901 repeat = _gtk_css_value_get_enum (value);
902 enum_class = g_type_class_ref (GTK_TYPE_CSS_BACKGROUND_REPEAT);
903 enum_value = g_enum_get_value (enum_class, repeat);
905 /* only triggers for 'repeat-x' and 'repeat-y' */
907 g_string_append (string, enum_value->value_nick);
910 enum_value = g_enum_get_value (enum_class, GTK_CSS_BACKGROUND_HORIZONTAL (repeat));
911 g_string_append (string, enum_value->value_nick);
913 if (GTK_CSS_BACKGROUND_HORIZONTAL (repeat) != GTK_CSS_BACKGROUND_VERTICAL (repeat))
915 enum_value = g_enum_get_value (enum_class, GTK_CSS_BACKGROUND_VERTICAL (repeat));
916 g_string_append (string, " ");
917 g_string_append (string, enum_value->value_nick);
921 g_type_class_unref (enum_class);
925 background_size_parse (GtkCssStyleProperty *property,
926 GtkCssParser *parser,
929 GtkCssBackgroundSize size = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), FALSE, FALSE};
931 if (_gtk_css_parser_try (parser, "cover", TRUE))
933 else if (_gtk_css_parser_try (parser, "contain", TRUE))
937 if (_gtk_css_parser_try (parser, "auto", TRUE))
938 _gtk_css_number_init (&size.width, 0, GTK_CSS_PX);
939 else if (!_gtk_css_parser_read_number (parser,
941 GTK_CSS_POSITIVE_ONLY
942 | GTK_CSS_PARSE_PERCENT
943 | GTK_CSS_PARSE_LENGTH))
946 if (_gtk_css_parser_try (parser, "auto", TRUE))
947 _gtk_css_number_init (&size.height, 0, GTK_CSS_PX);
948 else if (_gtk_css_parser_has_number (parser))
950 if (!_gtk_css_parser_read_number (parser,
952 GTK_CSS_POSITIVE_ONLY
953 | GTK_CSS_PARSE_PERCENT
954 | GTK_CSS_PARSE_LENGTH))
958 _gtk_css_number_init (&size.height, 0, GTK_CSS_PX);
961 return _gtk_css_value_new_from_background_size (&size);
965 background_size_print (GtkCssStyleProperty *property,
966 const GtkCssValue *value,
969 const GtkCssBackgroundSize *size = _gtk_css_value_get_background_size (value);
972 g_string_append (string, "cover");
973 else if (size->contain)
974 g_string_append (string, "contain");
977 if (size->width.value == 0)
978 g_string_append (string, "auto");
980 _gtk_css_number_print (&size->width, string);
982 if (size->height.value != 0)
984 g_string_append (string, " ");
985 _gtk_css_number_print (&size->height, string);
991 background_size_compute (GtkCssStyleProperty *property,
992 GtkStyleContext *context,
993 GtkCssValue *specified)
995 const GtkCssBackgroundSize *ssize = _gtk_css_value_get_background_size (specified);
996 GtkCssBackgroundSize csize;
999 csize.cover = ssize->cover;
1000 csize.contain = ssize->contain;
1001 changed = _gtk_css_number_compute (&csize.width,
1004 changed |= _gtk_css_number_compute (&csize.height,
1008 return _gtk_css_value_new_from_background_size (&csize);
1009 return _gtk_css_value_ref (specified);
1012 static GtkCssValue *
1013 background_position_parse (GtkCssStyleProperty *property,
1014 GtkCssParser *parser,
1017 static const struct {
1020 gboolean horizontal;
1023 { "left", 0, TRUE, FALSE },
1024 { "right", 100, TRUE, FALSE },
1025 { "center", 50, TRUE, TRUE },
1026 { "top", 0, FALSE, TRUE },
1027 { "bottom", 100, FALSE, TRUE },
1028 { NULL , 0, TRUE, FALSE }, /* used for numbers */
1029 { NULL , 50, TRUE, TRUE } /* used for no value */
1031 GtkCssBackgroundPosition pos;
1032 GtkCssNumber *missing;
1033 guint first, second;
1035 for (first = 0; names[first].name != NULL; first++)
1037 if (_gtk_css_parser_try (parser, names[first].name, TRUE))
1039 if (names[first].horizontal)
1041 _gtk_css_number_init (&pos.x, names[first].percentage, GTK_CSS_PERCENT);
1046 _gtk_css_number_init (&pos.y, names[first].percentage, GTK_CSS_PERCENT);
1052 if (names[first].name == NULL)
1055 if (!_gtk_css_parser_read_number (parser,
1057 GTK_CSS_PARSE_PERCENT
1058 | GTK_CSS_PARSE_LENGTH))
1062 for (second = 0; names[second].name != NULL; second++)
1064 if (_gtk_css_parser_try (parser, names[second].name, TRUE))
1066 _gtk_css_number_init (missing, names[second].percentage, GTK_CSS_PERCENT);
1071 if (names[second].name == NULL)
1073 if (_gtk_css_parser_has_number (parser))
1075 if (missing != &pos.y)
1077 _gtk_css_parser_error (parser, "Invalid combination of values");
1080 if (!_gtk_css_parser_read_number (parser,
1082 GTK_CSS_PARSE_PERCENT
1083 | GTK_CSS_PARSE_LENGTH))
1089 _gtk_css_number_init (missing, 50, GTK_CSS_PERCENT);
1094 if ((names[first].horizontal && !names[second].vertical) ||
1095 (!names[first].horizontal && !names[second].horizontal))
1097 _gtk_css_parser_error (parser, "Invalid combination of values");
1102 return _gtk_css_value_new_from_background_position (&pos);
1106 background_position_print (GtkCssStyleProperty *property,
1107 const GtkCssValue *value,
1110 const GtkCssBackgroundPosition *pos = _gtk_css_value_get_background_position (value);
1111 static const GtkCssNumber center = GTK_CSS_NUMBER_INIT (50, GTK_CSS_PERCENT);
1112 static const struct {
1115 GtkCssNumber number;
1117 { "left", "top", GTK_CSS_NUMBER_INIT (0, GTK_CSS_PERCENT) },
1118 { "right", "bottom", GTK_CSS_NUMBER_INIT (100, GTK_CSS_PERCENT) }
1122 if (_gtk_css_number_equal (&pos->x, ¢er))
1124 if (_gtk_css_number_equal (&pos->y, ¢er))
1126 g_string_append (string, "center");
1132 for (i = 0; i < G_N_ELEMENTS (values); i++)
1134 if (_gtk_css_number_equal (&pos->x, &values[i].number))
1136 g_string_append (string, values[i].x_name);
1140 if (i == G_N_ELEMENTS (values))
1141 _gtk_css_number_print (&pos->x, string);
1143 if (_gtk_css_number_equal (&pos->y, ¢er))
1146 g_string_append_c (string, ' ');
1149 for (i = 0; i < G_N_ELEMENTS (values); i++)
1151 if (_gtk_css_number_equal (&pos->y, &values[i].number))
1153 g_string_append (string, values[i].y_name);
1157 if (i == G_N_ELEMENTS (values))
1159 if (_gtk_css_number_equal (&pos->x, ¢er))
1160 g_string_append (string, "center ");
1161 _gtk_css_number_print (&pos->y, string);
1165 static GtkCssValue *
1166 background_position_compute (GtkCssStyleProperty *property,
1167 GtkStyleContext *context,
1168 GtkCssValue *specified)
1170 const GtkCssBackgroundPosition *spos = _gtk_css_value_get_background_position (specified);
1171 GtkCssBackgroundPosition cpos;
1174 changed = _gtk_css_number_compute (&cpos.x,
1177 changed |= _gtk_css_number_compute (&cpos.y,
1181 return _gtk_css_value_new_from_background_position (&cpos);
1182 return _gtk_css_value_ref (specified);
1185 /*** REGISTRATION ***/
1187 static GtkSymbolicColor *
1188 gtk_symbolic_color_new_rgba (double red,
1193 GdkRGBA rgba = { red, green, blue, alpha };
1195 return gtk_symbolic_color_new_literal (&rgba);
1199 _gtk_css_style_property_init_properties (void)
1201 char *default_font_family[] = { "Sans", NULL };
1202 GtkCssBackgroundSize default_background_size = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), FALSE, FALSE };
1203 GtkCssBackgroundPosition default_background_position = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PERCENT), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PERCENT)};
1204 GtkCssBorderCornerRadius no_corner_radius = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX) };
1205 GtkBorder border_of_ones = { 1, 1, 1, 1 };
1206 GtkCssBorderImageRepeat border_image_repeat = { GTK_CSS_REPEAT_STYLE_STRETCH, GTK_CSS_REPEAT_STYLE_STRETCH };
1208 /* Initialize "color" and "font-size" first,
1209 * so that when computing values later they are
1210 * done first. That way, 'currentColor' and font
1211 * sizes in em can be looked up properly */
1212 gtk_css_style_property_register ("color",
1214 GTK_STYLE_PROPERTY_INHERIT,
1217 color_property_compute,
1221 _gtk_css_value_new_take_symbolic_color (
1222 gtk_symbolic_color_new_rgba (1, 1, 1, 1)));
1223 gtk_css_style_property_register ("font-size",
1225 GTK_STYLE_PROPERTY_INHERIT,
1232 _gtk_css_value_new_from_double (10.0));
1234 /* properties that aren't referenced when computing values
1236 gtk_css_style_property_register ("background-color",
1245 _gtk_css_value_new_take_symbolic_color (
1246 gtk_symbolic_color_new_rgba (0, 0, 0, 0)));
1248 gtk_css_style_property_register ("font-family",
1250 GTK_STYLE_PROPERTY_INHERIT,
1252 font_family_value_print,
1257 _gtk_css_value_new_take_strv (g_strdupv (default_font_family)));
1258 gtk_css_style_property_register ("font-style",
1260 GTK_STYLE_PROPERTY_INHERIT,
1267 _gtk_css_font_style_value_new (PANGO_STYLE_NORMAL));
1268 gtk_css_style_property_register ("font-variant",
1270 GTK_STYLE_PROPERTY_INHERIT,
1271 parse_pango_variant,
1274 query_pango_variant,
1275 assign_pango_variant,
1277 _gtk_css_font_variant_value_new (PANGO_VARIANT_NORMAL));
1278 gtk_css_style_property_register ("font-weight",
1280 GTK_STYLE_PROPERTY_INHERIT,
1285 assign_pango_weight,
1287 _gtk_css_font_weight_value_new (PANGO_WEIGHT_NORMAL));
1289 gtk_css_style_property_register ("text-shadow",
1291 GTK_STYLE_PROPERTY_INHERIT,
1294 shadow_value_compute,
1298 _gtk_css_shadow_value_new_none ());
1300 gtk_css_style_property_register ("icon-shadow",
1302 GTK_STYLE_PROPERTY_INHERIT,
1305 shadow_value_compute,
1309 _gtk_css_shadow_value_new_none ());
1311 gtk_css_style_property_register ("box-shadow",
1316 shadow_value_compute,
1320 _gtk_css_shadow_value_new_none ());
1322 gtk_css_style_property_register ("margin-top",
1328 query_length_as_int,
1329 assign_length_from_int,
1331 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1332 gtk_css_style_property_register ("margin-left",
1338 query_length_as_int,
1339 assign_length_from_int,
1341 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1342 gtk_css_style_property_register ("margin-bottom",
1348 query_length_as_int,
1349 assign_length_from_int,
1351 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1352 gtk_css_style_property_register ("margin-right",
1358 query_length_as_int,
1359 assign_length_from_int,
1361 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1362 gtk_css_style_property_register ("padding-top",
1368 query_length_as_int,
1369 assign_length_from_int,
1371 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1372 gtk_css_style_property_register ("padding-left",
1378 query_length_as_int,
1379 assign_length_from_int,
1381 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1382 gtk_css_style_property_register ("padding-bottom",
1388 query_length_as_int,
1389 assign_length_from_int,
1391 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1392 gtk_css_style_property_register ("padding-right",
1398 query_length_as_int,
1399 assign_length_from_int,
1401 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1402 /* IMPORTANT: compute_border_width() requires that the border-width
1403 * properties be immeditaly followed by the border-style properties
1405 gtk_css_style_property_register ("border-top-style",
1406 GTK_TYPE_BORDER_STYLE,
1412 assign_border_style,
1414 _gtk_css_border_style_value_new (GTK_BORDER_STYLE_NONE));
1415 gtk_css_style_property_register ("border-top-width",
1420 compute_border_width,
1421 query_length_as_int,
1422 assign_length_from_int,
1424 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1425 gtk_css_style_property_register ("border-left-style",
1426 GTK_TYPE_BORDER_STYLE,
1432 assign_border_style,
1434 _gtk_css_border_style_value_new (GTK_BORDER_STYLE_NONE));
1435 gtk_css_style_property_register ("border-left-width",
1440 compute_border_width,
1441 query_length_as_int,
1442 assign_length_from_int,
1444 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1445 gtk_css_style_property_register ("border-bottom-style",
1446 GTK_TYPE_BORDER_STYLE,
1452 assign_border_style,
1454 _gtk_css_border_style_value_new (GTK_BORDER_STYLE_NONE));
1455 gtk_css_style_property_register ("border-bottom-width",
1460 compute_border_width,
1461 query_length_as_int,
1462 assign_length_from_int,
1464 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1465 gtk_css_style_property_register ("border-right-style",
1466 GTK_TYPE_BORDER_STYLE,
1472 assign_border_style,
1474 _gtk_css_border_style_value_new (GTK_BORDER_STYLE_NONE));
1475 gtk_css_style_property_register ("border-right-width",
1480 compute_border_width,
1481 query_length_as_int,
1482 assign_length_from_int,
1484 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1486 gtk_css_style_property_register ("border-top-left-radius",
1487 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1489 border_corner_radius_value_parse,
1490 border_corner_radius_value_print,
1495 _gtk_css_value_new_from_border_corner_radius (&no_corner_radius));
1496 gtk_css_style_property_register ("border-top-right-radius",
1497 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1499 border_corner_radius_value_parse,
1500 border_corner_radius_value_print,
1505 _gtk_css_value_new_from_border_corner_radius (&no_corner_radius));
1506 gtk_css_style_property_register ("border-bottom-right-radius",
1507 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1509 border_corner_radius_value_parse,
1510 border_corner_radius_value_print,
1515 _gtk_css_value_new_from_border_corner_radius (&no_corner_radius));
1516 gtk_css_style_property_register ("border-bottom-left-radius",
1517 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
1519 border_corner_radius_value_parse,
1520 border_corner_radius_value_print,
1525 _gtk_css_value_new_from_border_corner_radius (&no_corner_radius));
1527 gtk_css_style_property_register ("outline-style",
1528 GTK_TYPE_BORDER_STYLE,
1534 assign_border_style,
1536 _gtk_css_border_style_value_new (GTK_BORDER_STYLE_NONE));
1537 gtk_css_style_property_register ("outline-width",
1542 compute_border_width,
1543 query_length_as_int,
1544 assign_length_from_int,
1546 _gtk_css_number_value_new (0.0, GTK_CSS_PX));
1547 gtk_css_style_property_register ("outline-offset",
1556 _gtk_css_value_new_from_int (0));
1558 gtk_css_style_property_register ("background-clip",
1567 _gtk_css_value_new_from_enum (GTK_TYPE_CSS_AREA, GTK_CSS_AREA_BORDER_BOX));
1568 gtk_css_style_property_register ("background-origin",
1577 _gtk_css_value_new_from_enum (GTK_TYPE_CSS_AREA, GTK_CSS_AREA_PADDING_BOX));
1578 gtk_css_style_property_register ("background-size",
1581 background_size_parse,
1582 background_size_print,
1583 background_size_compute,
1587 _gtk_css_value_new_from_background_size (&default_background_size));
1588 gtk_css_style_property_register ("background-position",
1591 background_position_parse,
1592 background_position_print,
1593 background_position_compute,
1597 _gtk_css_value_new_from_background_position (&default_background_position));
1599 gtk_css_style_property_register ("border-top-color",
1608 _gtk_css_value_new_take_symbolic_color (
1609 gtk_symbolic_color_ref (
1610 _gtk_symbolic_color_get_current_color ())));
1611 gtk_css_style_property_register ("border-right-color",
1620 _gtk_css_value_new_take_symbolic_color (
1621 gtk_symbolic_color_ref (
1622 _gtk_symbolic_color_get_current_color ())));
1623 gtk_css_style_property_register ("border-bottom-color",
1632 _gtk_css_value_new_take_symbolic_color (
1633 gtk_symbolic_color_ref (
1634 _gtk_symbolic_color_get_current_color ())));
1635 gtk_css_style_property_register ("border-left-color",
1644 _gtk_css_value_new_take_symbolic_color (
1645 gtk_symbolic_color_ref (
1646 _gtk_symbolic_color_get_current_color ())));
1647 gtk_css_style_property_register ("outline-color",
1656 _gtk_css_value_new_take_symbolic_color (
1657 gtk_symbolic_color_ref (
1658 _gtk_symbolic_color_get_current_color ())));
1660 gtk_css_style_property_register ("background-repeat",
1661 GTK_TYPE_CSS_BACKGROUND_REPEAT,
1663 background_repeat_value_parse,
1664 background_repeat_value_print,
1669 _gtk_css_value_new_from_enum (GTK_TYPE_CSS_BACKGROUND_REPEAT,
1670 GTK_CSS_BACKGROUND_REPEAT |
1671 (GTK_CSS_BACKGROUND_REPEAT << GTK_CSS_BACKGROUND_REPEAT_SHIFT)));
1672 gtk_css_style_property_register ("background-image",
1673 CAIRO_GOBJECT_TYPE_PATTERN,
1675 css_image_value_parse,
1676 css_image_value_print,
1677 css_image_value_compute,
1678 css_image_value_query,
1679 css_image_value_assign,
1681 _gtk_css_value_new_take_image (NULL));
1683 gtk_css_style_property_register ("border-image-source",
1684 CAIRO_GOBJECT_TYPE_PATTERN,
1686 css_image_value_parse,
1687 css_image_value_print,
1688 css_image_value_compute,
1689 css_image_value_query,
1690 css_image_value_assign,
1692 _gtk_css_value_new_take_image (NULL));
1693 gtk_css_style_property_register ("border-image-repeat",
1694 GTK_TYPE_CSS_BORDER_IMAGE_REPEAT,
1696 border_image_repeat_parse,
1702 _gtk_css_value_new_from_border_image_repeat (&border_image_repeat));
1704 /* XXX: The initial value is wrong, it should be 100% */
1705 gtk_css_style_property_register ("border-image-slice",
1708 border_image_slice_parse,
1714 _gtk_css_value_new_from_boxed (GTK_TYPE_BORDER, &border_of_ones));
1715 gtk_css_style_property_register ("border-image-width",
1718 border_image_width_parse,
1724 _gtk_css_value_new_from_boxed (GTK_TYPE_BORDER, NULL));
1725 gtk_css_style_property_register ("engine",
1726 GTK_TYPE_THEMING_ENGINE,
1734 _gtk_css_value_new_from_theming_engine (gtk_theming_engine_load (NULL)));
1735 gtk_css_style_property_register ("transition",
1736 GTK_TYPE_ANIMATION_DESCRIPTION,
1744 _gtk_css_value_new_from_boxed (GTK_TYPE_ANIMATION_DESCRIPTION, NULL));
1746 /* Private property holding the binding sets */
1747 gtk_css_style_property_register ("gtk-key-bindings",
1750 bindings_value_parse,
1751 bindings_value_print,
1756 _gtk_css_value_new_take_binding_sets (NULL));