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, write to the
16 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
17 * Boston, MA 02111-1307, USA.
22 #include "gtkstylepropertyprivate.h"
24 #include <gobject/gvaluecollector.h>
25 #include <gdk-pixbuf/gdk-pixbuf.h>
26 #include <cairo-gobject.h>
29 #include "gtkcssparserprivate.h"
30 #include "gtkcssstylefuncsprivate.h"
31 #include "gtkcssstylepropertyprivate.h"
32 #include "gtkcsstypesprivate.h"
34 #include "gtkprivatetypebuiltins.h"
35 #include "gtkstylepropertiesprivate.h"
37 /* this is in case round() is not provided by the compiler,
38 * such as in the case of C89 compilers, like MSVC
40 #include "fallback-c89.c"
42 /* the actual parsers we have */
43 #include "gtkanimationdescription.h"
44 #include "gtkbindings.h"
45 #include "gtkcssimageprivate.h"
46 #include "gtkgradient.h"
47 #include "gtkshadowprivate.h"
48 #include "gtksymboliccolorprivate.h"
49 #include "gtkthemingengine.h"
50 #include "gtktypebuiltins.h"
51 #include "gtkwin32themeprivate.h"
53 /*** REGISTRATION ***/
56 gtk_css_style_property_register (const char * name,
60 GtkStylePropertyFlags flags,
61 GtkCssStylePropertyParseFunc parse_value,
62 GtkCssStylePropertyPrintFunc print_value,
63 GtkCssStylePropertyComputeFunc compute_value,
66 GtkCssStyleProperty *node;
67 GValue initial_value = G_VALUE_INIT;
71 va_start (args, compute_value);
72 G_VALUE_COLLECT_INIT (&initial_value, specified_type,
76 g_error ("property `%s' initial value is broken: %s", name, error);
77 g_value_unset (&initial_value);
83 node = g_object_new (GTK_TYPE_CSS_STYLE_PROPERTY,
84 "value-type", value_type,
85 "computed-type", computed_type,
86 "inherit", (flags & GTK_STYLE_PROPERTY_INHERIT) ? TRUE : FALSE,
87 "initial-value", &initial_value,
92 node->parse_value = parse_value;
94 node->print_value = print_value;
96 node->compute_value = compute_value;
98 g_value_unset (&initial_value);
104 string_append_string (GString *str,
109 g_string_append_c (str, '"');
112 len = strcspn (string, "\"\n\r\f");
113 g_string_append (str, string);
120 g_string_append (str, "\\A ");
123 g_string_append (str, "\\D ");
126 g_string_append (str, "\\C ");
129 g_string_append (str, "\\\"");
132 g_assert_not_reached ();
137 g_string_append_c (str, '"');
140 /*** IMPLEMENTATIONS ***/
143 color_compute (GtkCssStyleProperty *property,
145 GtkStyleContext *context,
146 const GValue *specified)
148 GtkSymbolicColor *symbolic = g_value_get_boxed (specified);
151 if (symbolic == _gtk_symbolic_color_get_current_color ())
153 /* The computed value of the ‘currentColor’ keyword is the computed
154 * value of the ‘color’ property. If the ‘currentColor’ keyword is
155 * set on the ‘color’ property itself, it is treated as ‘color: inherit’.
157 if (g_str_equal (_gtk_style_property_get_name (GTK_STYLE_PROPERTY (property)), "color"))
159 GtkStyleContext *parent = gtk_style_context_get_parent (context);
162 g_value_copy (_gtk_style_context_peek_property (parent, "color"), computed);
164 _gtk_css_style_compute_value (computed,
166 _gtk_css_style_property_get_initial_value (property));
170 g_value_copy (_gtk_style_context_peek_property (context, "color"), computed);
173 else if (_gtk_style_context_resolve_color (context,
177 g_value_set_boxed (computed, &rgba);
181 color_compute (property,
184 _gtk_css_style_property_get_initial_value (property));
189 font_family_parse (GtkCssStyleProperty *property,
191 GtkCssParser *parser,
197 /* We don't special case generic families. Pango should do
200 names = g_ptr_array_new ();
203 name = _gtk_css_parser_try_ident (parser, TRUE);
206 GString *string = g_string_new (name);
208 while ((name = _gtk_css_parser_try_ident (parser, TRUE)))
210 g_string_append_c (string, ' ');
211 g_string_append (string, name);
214 name = g_string_free (string, FALSE);
218 name = _gtk_css_parser_read_string (parser);
221 g_ptr_array_free (names, TRUE);
226 g_ptr_array_add (names, name);
227 } while (_gtk_css_parser_try (parser, ",", TRUE));
229 /* NULL-terminate array */
230 g_ptr_array_add (names, NULL);
231 g_value_set_boxed (value, g_ptr_array_free (names, FALSE));
236 font_family_value_print (GtkCssStyleProperty *property,
240 const char **names = g_value_get_boxed (value);
242 if (names == NULL || *names == NULL)
244 g_string_append (string, "none");
248 string_append_string (string, *names);
252 g_string_append (string, ", ");
253 string_append_string (string, *names);
259 bindings_value_parse (GtkCssStyleProperty *property,
261 GtkCssParser *parser,
265 GtkBindingSet *binding_set;
268 array = g_ptr_array_new ();
271 name = _gtk_css_parser_try_ident (parser, TRUE);
274 _gtk_css_parser_error (parser, "Not a valid binding name");
275 g_ptr_array_free (array, TRUE);
279 binding_set = gtk_binding_set_find (name);
283 _gtk_css_parser_error (parser, "No binding set named '%s'", name);
288 g_ptr_array_add (array, binding_set);
291 while (_gtk_css_parser_try (parser, ",", TRUE));
293 g_value_take_boxed (value, array);
299 bindings_value_print (GtkCssStyleProperty *property,
306 array = g_value_get_boxed (value);
308 for (i = 0; i < array->len; i++)
310 GtkBindingSet *binding_set = g_ptr_array_index (array, i);
313 g_string_append (string, ", ");
314 g_string_append (string, binding_set->set_name);
319 border_corner_radius_value_parse (GtkCssStyleProperty *property,
321 GtkCssParser *parser,
324 GtkCssBorderCornerRadius corner;
326 if (!_gtk_css_parser_read_number (parser,
328 GTK_CSS_POSITIVE_ONLY
329 | GTK_CSS_PARSE_PERCENT
330 | GTK_CSS_NUMBER_AS_PIXELS
331 | GTK_CSS_PARSE_LENGTH))
334 if (!_gtk_css_parser_has_number (parser))
335 corner.vertical = corner.horizontal;
336 else if (!_gtk_css_parser_read_number (parser,
338 GTK_CSS_POSITIVE_ONLY
339 | GTK_CSS_PARSE_PERCENT
340 | GTK_CSS_NUMBER_AS_PIXELS
341 | GTK_CSS_PARSE_LENGTH))
344 g_value_set_boxed (value, &corner);
349 border_corner_radius_value_print (GtkCssStyleProperty *property,
353 GtkCssBorderCornerRadius *corner;
355 corner = g_value_get_boxed (value);
357 _gtk_css_number_print (&corner->horizontal, string);
359 if (!_gtk_css_number_equal (&corner->horizontal, &corner->vertical))
361 g_string_append_c (string, ' ');
362 _gtk_css_number_print (&corner->vertical, string);
367 css_image_value_parse (GtkCssStyleProperty *property,
369 GtkCssParser *parser,
374 if (_gtk_css_parser_try (parser, "none", TRUE))
378 image = _gtk_css_image_new_parse (parser, base);
383 g_value_take_object (value, image);
388 css_image_value_print (GtkCssStyleProperty *property,
392 GtkCssImage *image = g_value_get_object (value);
395 _gtk_css_image_print (image, string);
397 g_string_append (string, "none");
401 css_image_value_compute (GtkCssStyleProperty *property,
403 GtkStyleContext *context,
404 const GValue *specified)
406 GtkCssImage *image = g_value_get_object (specified);
409 image = _gtk_css_image_compute (image, context);
411 g_value_take_object (computed, image);
415 parse_border_width (GtkCssStyleProperty *property,
417 GtkCssParser *parser,
422 if (!_gtk_css_parser_read_number (parser,
424 GTK_CSS_POSITIVE_ONLY
425 | GTK_CSS_NUMBER_AS_PIXELS
426 | GTK_CSS_PARSE_LENGTH))
429 g_value_set_boxed (value, &number);
434 compute_border_width (GtkCssStyleProperty *property,
436 GtkStyleContext *context,
437 const GValue *specified)
439 GtkCssStyleProperty *style;
440 GtkBorderStyle border_style;
443 /* The -1 is magic that is only true because we register the style
444 * properties directly after the width properties.
446 style = _gtk_css_style_property_lookup_by_id (_gtk_css_style_property_get_id (property) - 1);
447 border_style = g_value_get_enum (_gtk_style_context_peek_property (context, _gtk_style_property_get_name (GTK_STYLE_PROPERTY (style))));
449 if (border_style == GTK_BORDER_STYLE_NONE ||
450 border_style == GTK_BORDER_STYLE_HIDDEN)
452 g_value_set_int (computed, 0);
456 _gtk_css_number_compute (&number,
457 g_value_get_boxed (specified),
459 g_value_set_int (computed, round (number.value));
463 background_repeat_value_parse (GtkCssStyleProperty *property,
465 GtkCssParser *parser,
468 int repeat, vertical;
470 if (!_gtk_css_parser_try_enum (parser, GTK_TYPE_CSS_BACKGROUND_REPEAT, &repeat))
472 _gtk_css_parser_error (parser, "Not a valid value");
476 if (repeat <= GTK_CSS_BACKGROUND_REPEAT_MASK)
478 if (_gtk_css_parser_try_enum (parser, GTK_TYPE_CSS_BACKGROUND_REPEAT, &vertical))
480 if (vertical >= GTK_CSS_BACKGROUND_REPEAT_MASK)
482 _gtk_css_parser_error (parser, "Not a valid 2nd value");
486 repeat |= vertical << GTK_CSS_BACKGROUND_REPEAT_SHIFT;
489 repeat |= repeat << GTK_CSS_BACKGROUND_REPEAT_SHIFT;
492 g_value_set_enum (value, repeat);
497 background_repeat_value_print (GtkCssStyleProperty *property,
501 GEnumClass *enum_class;
502 GEnumValue *enum_value;
503 GtkCssBackgroundRepeat repeat;
505 repeat = g_value_get_enum (value);
506 enum_class = g_type_class_ref (GTK_TYPE_CSS_BACKGROUND_REPEAT);
507 enum_value = g_enum_get_value (enum_class, repeat);
509 /* only triggers for 'repeat-x' and 'repeat-y' */
511 g_string_append (string, enum_value->value_nick);
514 enum_value = g_enum_get_value (enum_class, GTK_CSS_BACKGROUND_HORIZONTAL (repeat));
515 g_string_append (string, enum_value->value_nick);
517 if (GTK_CSS_BACKGROUND_HORIZONTAL (repeat) != GTK_CSS_BACKGROUND_VERTICAL (repeat))
519 enum_value = g_enum_get_value (enum_class, GTK_CSS_BACKGROUND_VERTICAL (repeat));
520 g_string_append (string, " ");
521 g_string_append (string, enum_value->value_nick);
525 g_type_class_unref (enum_class);
529 background_size_parse (GtkCssStyleProperty *property,
531 GtkCssParser *parser,
534 GtkCssBackgroundSize size = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), FALSE, FALSE};
536 if (_gtk_css_parser_try (parser, "cover", TRUE))
538 else if (_gtk_css_parser_try (parser, "contain", TRUE))
542 if (_gtk_css_parser_try (parser, "auto", TRUE))
543 _gtk_css_number_init (&size.width, 0, GTK_CSS_PX);
544 else if (!_gtk_css_parser_read_number (parser,
546 GTK_CSS_POSITIVE_ONLY
547 | GTK_CSS_PARSE_PERCENT
548 | GTK_CSS_PARSE_LENGTH))
551 if (_gtk_css_parser_try (parser, "auto", TRUE))
552 _gtk_css_number_init (&size.height, 0, GTK_CSS_PX);
553 else if (_gtk_css_parser_has_number (parser))
555 if (!_gtk_css_parser_read_number (parser,
557 GTK_CSS_POSITIVE_ONLY
558 | GTK_CSS_PARSE_PERCENT
559 | GTK_CSS_PARSE_LENGTH))
563 _gtk_css_number_init (&size.height, 0, GTK_CSS_PX);
566 g_value_set_boxed (value, &size);
571 background_size_print (GtkCssStyleProperty *property,
575 GtkCssBackgroundSize *size = g_value_get_boxed (value);
578 g_string_append (string, "cover");
579 else if (size->contain)
580 g_string_append (string, "contain");
583 if (size->width.value == 0)
584 g_string_append (string, "auto");
586 _gtk_css_number_print (&size->width, string);
588 if (size->height.value != 0)
590 g_string_append (string, " ");
591 _gtk_css_number_print (&size->height, string);
597 background_size_compute (GtkCssStyleProperty *property,
599 GtkStyleContext *context,
600 const GValue *specified)
602 GtkCssBackgroundSize *ssize = g_value_get_boxed (specified);
603 GtkCssBackgroundSize csize;
605 csize.cover = ssize->cover;
606 csize.contain = ssize->contain;
607 _gtk_css_number_compute (&csize.width,
610 _gtk_css_number_compute (&csize.height,
614 g_value_set_boxed (computed, &csize);
617 /*** REGISTRATION ***/
619 static GtkSymbolicColor *
620 gtk_symbolic_color_new_rgba (double red,
625 GdkRGBA rgba = { red, green, blue, alpha };
627 return gtk_symbolic_color_new_literal (&rgba);
631 _gtk_css_style_property_init_properties (void)
633 char *default_font_family[] = { "Sans", NULL };
635 GtkSymbolicColor *symbolic;
636 GtkCssBackgroundSize default_background_size = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), FALSE, FALSE };
637 GtkCssBorderCornerRadius no_corner_radius = { GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX), GTK_CSS_NUMBER_INIT (0, GTK_CSS_PX) };
638 GtkBorder border_of_ones = { 1, 1, 1, 1 };
639 GtkCssBorderImageRepeat border_image_repeat = { GTK_CSS_REPEAT_STYLE_STRETCH, GTK_CSS_REPEAT_STYLE_STRETCH };
641 /* Initialize "color" and "font-size" first,
642 * so that when computing values later they are
643 * done first. That way, 'currentColor' and font
644 * sizes in em can be looked up properly */
645 symbolic = gtk_symbolic_color_new_rgba (1, 1, 1, 1);
646 gtk_css_style_property_register ("color",
647 GTK_TYPE_SYMBOLIC_COLOR,
650 GTK_STYLE_PROPERTY_INHERIT,
655 gtk_symbolic_color_unref (symbolic);
656 gtk_css_style_property_register ("font-size",
660 GTK_STYLE_PROPERTY_INHERIT,
666 /* properties that aren't referenced when computing values
668 symbolic = gtk_symbolic_color_new_rgba (0, 0, 0, 0);
669 gtk_css_style_property_register ("background-color",
670 GTK_TYPE_SYMBOLIC_COLOR,
678 gtk_symbolic_color_unref (symbolic);
680 gtk_css_style_property_register ("font-family",
684 GTK_STYLE_PROPERTY_INHERIT,
686 font_family_value_print,
688 default_font_family);
689 gtk_css_style_property_register ("font-style",
693 GTK_STYLE_PROPERTY_INHERIT,
698 gtk_css_style_property_register ("font-variant",
702 GTK_STYLE_PROPERTY_INHERIT,
706 PANGO_VARIANT_NORMAL);
707 /* xxx: need to parse this properly, ie parse the numbers */
708 gtk_css_style_property_register ("font-weight",
712 GTK_STYLE_PROPERTY_INHERIT,
716 PANGO_WEIGHT_NORMAL);
718 gtk_css_style_property_register ("text-shadow",
722 GTK_STYLE_PROPERTY_INHERIT,
728 gtk_css_style_property_register ("icon-shadow",
732 GTK_STYLE_PROPERTY_INHERIT,
738 gtk_css_style_property_register ("box-shadow",
748 gtk_css_style_property_register ("margin-top",
757 gtk_css_style_property_register ("margin-left",
766 gtk_css_style_property_register ("margin-bottom",
775 gtk_css_style_property_register ("margin-right",
784 gtk_css_style_property_register ("padding-top",
793 gtk_css_style_property_register ("padding-left",
802 gtk_css_style_property_register ("padding-bottom",
811 gtk_css_style_property_register ("padding-right",
820 _gtk_css_number_init (&number, 0, GTK_CSS_PX);
821 /* IMPORTANT: compute_border_width() requires that the border-width
822 * properties be immeditaly followed by the border-style properties
824 gtk_css_style_property_register ("border-top-style",
825 GTK_TYPE_BORDER_STYLE,
826 GTK_TYPE_BORDER_STYLE,
827 GTK_TYPE_BORDER_STYLE,
832 GTK_BORDER_STYLE_NONE);
833 gtk_css_style_property_register ("border-top-width",
840 compute_border_width,
842 gtk_css_style_property_register ("border-left-style",
843 GTK_TYPE_BORDER_STYLE,
844 GTK_TYPE_BORDER_STYLE,
845 GTK_TYPE_BORDER_STYLE,
850 GTK_BORDER_STYLE_NONE);
851 gtk_css_style_property_register ("border-left-width",
858 compute_border_width,
860 gtk_css_style_property_register ("border-bottom-style",
861 GTK_TYPE_BORDER_STYLE,
862 GTK_TYPE_BORDER_STYLE,
863 GTK_TYPE_BORDER_STYLE,
868 GTK_BORDER_STYLE_NONE);
869 gtk_css_style_property_register ("border-bottom-width",
876 compute_border_width,
878 gtk_css_style_property_register ("border-right-style",
879 GTK_TYPE_BORDER_STYLE,
880 GTK_TYPE_BORDER_STYLE,
881 GTK_TYPE_BORDER_STYLE,
886 GTK_BORDER_STYLE_NONE);
887 gtk_css_style_property_register ("border-right-width",
894 compute_border_width,
897 gtk_css_style_property_register ("border-top-left-radius",
898 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
899 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
900 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
902 border_corner_radius_value_parse,
903 border_corner_radius_value_print,
906 gtk_css_style_property_register ("border-top-right-radius",
907 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
908 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
909 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
911 border_corner_radius_value_parse,
912 border_corner_radius_value_print,
915 gtk_css_style_property_register ("border-bottom-right-radius",
916 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
917 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
918 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
920 border_corner_radius_value_parse,
921 border_corner_radius_value_print,
924 gtk_css_style_property_register ("border-bottom-left-radius",
925 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
926 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
927 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
929 border_corner_radius_value_parse,
930 border_corner_radius_value_print,
934 gtk_css_style_property_register ("outline-style",
935 GTK_TYPE_BORDER_STYLE,
936 GTK_TYPE_BORDER_STYLE,
937 GTK_TYPE_BORDER_STYLE,
942 GTK_BORDER_STYLE_NONE);
943 gtk_css_style_property_register ("outline-width",
950 compute_border_width,
952 gtk_css_style_property_register ("outline-offset",
962 gtk_css_style_property_register ("background-clip",
970 GTK_CSS_AREA_BORDER_BOX);
971 gtk_css_style_property_register ("background-origin",
979 GTK_CSS_AREA_PADDING_BOX);
980 gtk_css_style_property_register ("background-size",
981 GTK_TYPE_CSS_BACKGROUND_SIZE,
982 GTK_TYPE_CSS_BACKGROUND_SIZE,
985 background_size_parse,
986 background_size_print,
987 background_size_compute,
988 &default_background_size);
990 gtk_css_style_property_register ("border-top-color",
991 GTK_TYPE_SYMBOLIC_COLOR,
998 _gtk_symbolic_color_get_current_color ());
999 gtk_css_style_property_register ("border-right-color",
1000 GTK_TYPE_SYMBOLIC_COLOR,
1007 _gtk_symbolic_color_get_current_color ());
1008 gtk_css_style_property_register ("border-bottom-color",
1009 GTK_TYPE_SYMBOLIC_COLOR,
1016 _gtk_symbolic_color_get_current_color ());
1017 gtk_css_style_property_register ("border-left-color",
1018 GTK_TYPE_SYMBOLIC_COLOR,
1025 _gtk_symbolic_color_get_current_color ());
1026 gtk_css_style_property_register ("outline-color",
1027 GTK_TYPE_SYMBOLIC_COLOR,
1034 _gtk_symbolic_color_get_current_color ());
1036 gtk_css_style_property_register ("background-repeat",
1037 GTK_TYPE_CSS_BACKGROUND_REPEAT,
1038 GTK_TYPE_CSS_BACKGROUND_REPEAT,
1039 GTK_TYPE_CSS_BACKGROUND_REPEAT,
1041 background_repeat_value_parse,
1042 background_repeat_value_print,
1044 GTK_CSS_BACKGROUND_REPEAT | (GTK_CSS_BACKGROUND_REPEAT << GTK_CSS_BACKGROUND_REPEAT_SHIFT));
1045 gtk_css_style_property_register ("background-image",
1048 CAIRO_GOBJECT_TYPE_PATTERN,
1050 css_image_value_parse,
1051 css_image_value_print,
1052 css_image_value_compute,
1055 gtk_css_style_property_register ("border-image-source",
1058 CAIRO_GOBJECT_TYPE_PATTERN,
1060 css_image_value_parse,
1061 css_image_value_print,
1062 css_image_value_compute,
1064 gtk_css_style_property_register ("border-image-repeat",
1065 GTK_TYPE_CSS_BORDER_IMAGE_REPEAT,
1066 GTK_TYPE_CSS_BORDER_IMAGE_REPEAT,
1067 GTK_TYPE_CSS_BORDER_IMAGE_REPEAT,
1072 &border_image_repeat);
1074 /* XXX: The initial value is wrong, it should be 100% */
1075 gtk_css_style_property_register ("border-image-slice",
1084 gtk_css_style_property_register ("border-image-width",
1093 gtk_css_style_property_register ("engine",
1094 GTK_TYPE_THEMING_ENGINE,
1095 GTK_TYPE_THEMING_ENGINE,
1096 GTK_TYPE_THEMING_ENGINE,
1101 gtk_theming_engine_load (NULL));
1102 gtk_css_style_property_register ("transition",
1103 GTK_TYPE_ANIMATION_DESCRIPTION,
1104 GTK_TYPE_ANIMATION_DESCRIPTION,
1105 GTK_TYPE_ANIMATION_DESCRIPTION,
1112 /* Private property holding the binding sets */
1113 gtk_css_style_property_register ("gtk-key-bindings",
1118 bindings_value_parse,
1119 bindings_value_print,