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>
28 #include "gtkcssparserprivate.h"
29 #include "gtkcssstylefuncsprivate.h"
30 #include "gtkcssstylepropertyprivate.h"
31 #include "gtkcsstypesprivate.h"
33 #include "gtkprivatetypebuiltins.h"
34 #include "gtkstylepropertiesprivate.h"
36 /* the actual parsers we have */
37 #include "gtkanimationdescription.h"
38 #include "gtkbindings.h"
39 #include "gtkcssimageprivate.h"
40 #include "gtkgradient.h"
41 #include "gtkshadowprivate.h"
42 #include "gtksymboliccolorprivate.h"
43 #include "gtkthemingengine.h"
44 #include "gtktypebuiltins.h"
45 #include "gtkwin32themeprivate.h"
47 /*** REGISTRATION ***/
50 gtk_css_style_property_register (const char * name,
54 GtkStylePropertyFlags flags,
55 GtkCssStylePropertyParseFunc parse_value,
56 GtkCssStylePropertyPrintFunc print_value,
57 GtkCssStylePropertyComputeFunc compute_value,
60 GtkCssStyleProperty *node;
61 GValue initial_value = G_VALUE_INIT;
65 va_start (args, compute_value);
66 G_VALUE_COLLECT_INIT (&initial_value, specified_type,
70 g_error ("property `%s' initial value is broken: %s", name, error);
71 g_value_unset (&initial_value);
77 node = g_object_new (GTK_TYPE_CSS_STYLE_PROPERTY,
78 "value-type", value_type,
79 "computed-type", computed_type,
80 "inherit", (flags & GTK_STYLE_PROPERTY_INHERIT) ? TRUE : FALSE,
81 "initial-value", &initial_value,
86 node->parse_value = parse_value;
88 node->print_value = print_value;
90 node->compute_value = compute_value;
92 g_value_unset (&initial_value);
98 string_append_double (GString *string,
101 char buf[G_ASCII_DTOSTR_BUF_SIZE];
103 g_ascii_dtostr (buf, sizeof (buf), d);
104 g_string_append (string, buf);
108 string_append_string (GString *str,
113 g_string_append_c (str, '"');
116 len = strcspn (string, "\"\n\r\f");
117 g_string_append (str, string);
124 g_string_append (str, "\\A ");
127 g_string_append (str, "\\D ");
130 g_string_append (str, "\\C ");
133 g_string_append (str, "\\\"");
136 g_assert_not_reached ();
141 g_string_append_c (str, '"');
144 /*** IMPLEMENTATIONS ***/
147 color_compute (GtkCssStyleProperty *property,
149 GtkStyleContext *context,
150 const GValue *specified)
152 GtkSymbolicColor *symbolic = g_value_get_boxed (specified);
155 if (symbolic == _gtk_symbolic_color_get_current_color ())
157 /* The computed value of the ‘currentColor’ keyword is the computed
158 * value of the ‘color’ property. If the ‘currentColor’ keyword is
159 * set on the ‘color’ property itself, it is treated as ‘color: inherit’.
161 if (g_str_equal (_gtk_style_property_get_name (GTK_STYLE_PROPERTY (property)), "color"))
163 GtkStyleContext *parent = gtk_style_context_get_parent (context);
166 g_value_copy (_gtk_style_context_peek_property (parent, "color"), computed);
168 _gtk_css_style_compute_value (computed,
170 _gtk_css_style_property_get_initial_value (property));
174 g_value_copy (_gtk_style_context_peek_property (context, "color"), computed);
177 else if (_gtk_style_context_resolve_color (context,
181 g_value_set_boxed (computed, &rgba);
185 color_compute (property,
188 _gtk_css_style_property_get_initial_value (property));
193 font_family_parse (GtkCssStyleProperty *property,
195 GtkCssParser *parser,
201 /* We don't special case generic families. Pango should do
204 names = g_ptr_array_new ();
207 name = _gtk_css_parser_try_ident (parser, TRUE);
210 GString *string = g_string_new (name);
212 while ((name = _gtk_css_parser_try_ident (parser, TRUE)))
214 g_string_append_c (string, ' ');
215 g_string_append (string, name);
218 name = g_string_free (string, FALSE);
222 name = _gtk_css_parser_read_string (parser);
225 g_ptr_array_free (names, TRUE);
230 g_ptr_array_add (names, name);
231 } while (_gtk_css_parser_try (parser, ",", TRUE));
233 /* NULL-terminate array */
234 g_ptr_array_add (names, NULL);
235 g_value_set_boxed (value, g_ptr_array_free (names, FALSE));
240 font_family_value_print (GtkCssStyleProperty *property,
244 const char **names = g_value_get_boxed (value);
246 if (names == NULL || *names == NULL)
248 g_string_append (string, "none");
252 string_append_string (string, *names);
256 g_string_append (string, ", ");
257 string_append_string (string, *names);
263 bindings_value_parse (GtkCssStyleProperty *property,
265 GtkCssParser *parser,
269 GtkBindingSet *binding_set;
272 array = g_ptr_array_new ();
275 name = _gtk_css_parser_try_ident (parser, TRUE);
278 _gtk_css_parser_error (parser, "Not a valid binding name");
279 g_ptr_array_free (array, TRUE);
283 binding_set = gtk_binding_set_find (name);
287 _gtk_css_parser_error (parser, "No binding set named '%s'", name);
292 g_ptr_array_add (array, binding_set);
295 while (_gtk_css_parser_try (parser, ",", TRUE));
297 g_value_take_boxed (value, array);
303 bindings_value_print (GtkCssStyleProperty *property,
310 array = g_value_get_boxed (value);
312 for (i = 0; i < array->len; i++)
314 GtkBindingSet *binding_set = g_ptr_array_index (array, i);
317 g_string_append (string, ", ");
318 g_string_append (string, binding_set->set_name);
323 border_corner_radius_value_parse (GtkCssStyleProperty *property,
325 GtkCssParser *parser,
328 GtkCssBorderCornerRadius corner;
330 if (!_gtk_css_parser_try_double (parser, &corner.horizontal))
332 _gtk_css_parser_error (parser, "Expected a number");
335 else if (corner.horizontal < 0)
338 if (!_gtk_css_parser_try_double (parser, &corner.vertical))
339 corner.vertical = corner.horizontal;
340 else if (corner.vertical < 0)
343 g_value_set_boxed (value, &corner);
347 _gtk_css_parser_error (parser, "Border radius values cannot be negative");
352 border_corner_radius_value_print (GtkCssStyleProperty *property,
356 GtkCssBorderCornerRadius *corner;
358 corner = g_value_get_boxed (value);
362 g_string_append (string, "none");
366 string_append_double (string, corner->horizontal);
367 if (corner->horizontal != corner->vertical)
369 g_string_append_c (string, ' ');
370 string_append_double (string, corner->vertical);
375 css_image_value_parse (GtkCssStyleProperty *property,
377 GtkCssParser *parser,
382 if (_gtk_css_parser_try (parser, "none", TRUE))
386 image = _gtk_css_image_new_parse (parser, base);
391 g_value_take_object (value, image);
396 css_image_value_print (GtkCssStyleProperty *property,
400 GtkCssImage *image = g_value_get_object (value);
403 _gtk_css_image_print (image, string);
405 g_string_append (string, "none");
409 css_image_value_compute (GtkCssStyleProperty *property,
411 GtkStyleContext *context,
412 const GValue *specified)
414 GtkCssImage *image = g_value_get_object (specified);
417 image = _gtk_css_image_compute (image, context);
419 g_value_take_object (computed, image);
423 compute_border_width (GtkCssStyleProperty *property,
425 GtkStyleContext *context,
426 const GValue *specified)
428 GtkCssStyleProperty *style;
429 GtkBorderStyle border_style;
431 /* The -1 is magic that is only true because we register the style
432 * properties directly after the width properties.
434 style = _gtk_css_style_property_lookup_by_id (_gtk_css_style_property_get_id (property) - 1);
435 border_style = g_value_get_enum (_gtk_style_context_peek_property (context, _gtk_style_property_get_name (GTK_STYLE_PROPERTY (style))));
437 if (border_style == GTK_BORDER_STYLE_NONE ||
438 border_style == GTK_BORDER_STYLE_HIDDEN)
439 g_value_set_int (computed, 0);
441 g_value_copy (specified, computed);
445 background_repeat_value_parse (GtkCssStyleProperty *property,
447 GtkCssParser *parser,
450 int repeat, vertical;
452 if (!_gtk_css_parser_try_enum (parser, GTK_TYPE_CSS_BACKGROUND_REPEAT, &repeat))
454 _gtk_css_parser_error (parser, "Not a valid value");
458 if (repeat <= GTK_CSS_BACKGROUND_REPEAT_MASK)
460 if (_gtk_css_parser_try_enum (parser, GTK_TYPE_CSS_BACKGROUND_REPEAT, &vertical))
462 if (vertical >= GTK_CSS_BACKGROUND_REPEAT_MASK)
464 _gtk_css_parser_error (parser, "Not a valid 2nd value");
468 repeat |= vertical << GTK_CSS_BACKGROUND_REPEAT_SHIFT;
471 repeat |= repeat << GTK_CSS_BACKGROUND_REPEAT_SHIFT;
474 g_value_set_enum (value, repeat);
479 background_repeat_value_print (GtkCssStyleProperty *property,
483 GEnumClass *enum_class;
484 GEnumValue *enum_value;
485 GtkCssBackgroundRepeat repeat;
487 repeat = g_value_get_enum (value);
488 enum_class = g_type_class_ref (GTK_TYPE_CSS_BACKGROUND_REPEAT);
489 enum_value = g_enum_get_value (enum_class, repeat);
491 /* only triggers for 'repeat-x' and 'repeat-y' */
493 g_string_append (string, enum_value->value_nick);
496 enum_value = g_enum_get_value (enum_class, GTK_CSS_BACKGROUND_HORIZONTAL (repeat));
497 g_string_append (string, enum_value->value_nick);
499 if (GTK_CSS_BACKGROUND_HORIZONTAL (repeat) != GTK_CSS_BACKGROUND_VERTICAL (repeat))
501 enum_value = g_enum_get_value (enum_class, GTK_CSS_BACKGROUND_VERTICAL (repeat));
502 g_string_append (string, " ");
503 g_string_append (string, enum_value->value_nick);
507 g_type_class_unref (enum_class);
510 /*** REGISTRATION ***/
512 static GtkSymbolicColor *
513 gtk_symbolic_color_new_rgba (double red,
518 GdkRGBA rgba = { red, green, blue, alpha };
520 return gtk_symbolic_color_new_literal (&rgba);
524 _gtk_css_style_property_init_properties (void)
526 char *default_font_family[] = { "Sans", NULL };
527 GtkSymbolicColor *symbolic;
528 GtkCssBorderCornerRadius no_corner_radius = { 0, };
529 GtkBorder border_of_ones = { 1, 1, 1, 1 };
530 GtkCssBorderImageRepeat border_image_repeat = { GTK_CSS_REPEAT_STYLE_STRETCH, GTK_CSS_REPEAT_STYLE_STRETCH };
532 /* Initialize "color" and "font-size" first,
533 * so that when computing values later they are
534 * done first. That way, 'currentColor' and font
535 * sizes in em can be looked up properly */
536 symbolic = gtk_symbolic_color_new_rgba (1, 1, 1, 1);
537 gtk_css_style_property_register ("color",
538 GTK_TYPE_SYMBOLIC_COLOR,
541 GTK_STYLE_PROPERTY_INHERIT,
546 gtk_symbolic_color_unref (symbolic);
547 gtk_css_style_property_register ("font-size",
551 GTK_STYLE_PROPERTY_INHERIT,
557 /* properties that aren't referenced when computing values
559 symbolic = gtk_symbolic_color_new_rgba (0, 0, 0, 0);
560 gtk_css_style_property_register ("background-color",
561 GTK_TYPE_SYMBOLIC_COLOR,
569 gtk_symbolic_color_unref (symbolic);
571 gtk_css_style_property_register ("font-family",
575 GTK_STYLE_PROPERTY_INHERIT,
577 font_family_value_print,
579 default_font_family);
580 gtk_css_style_property_register ("font-style",
584 GTK_STYLE_PROPERTY_INHERIT,
589 gtk_css_style_property_register ("font-variant",
593 GTK_STYLE_PROPERTY_INHERIT,
597 PANGO_VARIANT_NORMAL);
598 /* xxx: need to parse this properly, ie parse the numbers */
599 gtk_css_style_property_register ("font-weight",
603 GTK_STYLE_PROPERTY_INHERIT,
607 PANGO_WEIGHT_NORMAL);
609 gtk_css_style_property_register ("text-shadow",
613 GTK_STYLE_PROPERTY_INHERIT,
619 gtk_css_style_property_register ("icon-shadow",
623 GTK_STYLE_PROPERTY_INHERIT,
629 gtk_css_style_property_register ("box-shadow",
639 gtk_css_style_property_register ("margin-top",
648 gtk_css_style_property_register ("margin-left",
657 gtk_css_style_property_register ("margin-bottom",
666 gtk_css_style_property_register ("margin-right",
675 gtk_css_style_property_register ("padding-top",
684 gtk_css_style_property_register ("padding-left",
693 gtk_css_style_property_register ("padding-bottom",
702 gtk_css_style_property_register ("padding-right",
711 /* IMPORTANT: compute_border_width() requires that the border-width
712 * properties be immeditaly followed by the border-style properties
714 gtk_css_style_property_register ("border-top-style",
715 GTK_TYPE_BORDER_STYLE,
716 GTK_TYPE_BORDER_STYLE,
717 GTK_TYPE_BORDER_STYLE,
722 GTK_BORDER_STYLE_NONE);
723 gtk_css_style_property_register ("border-top-width",
730 compute_border_width,
732 gtk_css_style_property_register ("border-left-style",
733 GTK_TYPE_BORDER_STYLE,
734 GTK_TYPE_BORDER_STYLE,
735 GTK_TYPE_BORDER_STYLE,
740 GTK_BORDER_STYLE_NONE);
741 gtk_css_style_property_register ("border-left-width",
748 compute_border_width,
750 gtk_css_style_property_register ("border-bottom-style",
751 GTK_TYPE_BORDER_STYLE,
752 GTK_TYPE_BORDER_STYLE,
753 GTK_TYPE_BORDER_STYLE,
758 GTK_BORDER_STYLE_NONE);
759 gtk_css_style_property_register ("border-bottom-width",
766 compute_border_width,
768 gtk_css_style_property_register ("border-right-style",
769 GTK_TYPE_BORDER_STYLE,
770 GTK_TYPE_BORDER_STYLE,
771 GTK_TYPE_BORDER_STYLE,
776 GTK_BORDER_STYLE_NONE);
777 gtk_css_style_property_register ("border-right-width",
784 compute_border_width,
787 gtk_css_style_property_register ("border-top-left-radius",
788 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
789 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
790 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
792 border_corner_radius_value_parse,
793 border_corner_radius_value_print,
796 gtk_css_style_property_register ("border-top-right-radius",
797 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
798 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
799 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
801 border_corner_radius_value_parse,
802 border_corner_radius_value_print,
805 gtk_css_style_property_register ("border-bottom-right-radius",
806 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
807 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
808 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
810 border_corner_radius_value_parse,
811 border_corner_radius_value_print,
814 gtk_css_style_property_register ("border-bottom-left-radius",
815 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
816 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
817 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
819 border_corner_radius_value_parse,
820 border_corner_radius_value_print,
824 gtk_css_style_property_register ("outline-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 ("outline-width",
840 compute_border_width,
842 gtk_css_style_property_register ("outline-offset",
852 gtk_css_style_property_register ("background-clip",
860 GTK_CSS_AREA_BORDER_BOX);
862 gtk_css_style_property_register ("background-origin",
870 GTK_CSS_AREA_PADDING_BOX);
872 gtk_css_style_property_register ("border-top-color",
873 GTK_TYPE_SYMBOLIC_COLOR,
880 _gtk_symbolic_color_get_current_color ());
881 gtk_css_style_property_register ("border-right-color",
882 GTK_TYPE_SYMBOLIC_COLOR,
889 _gtk_symbolic_color_get_current_color ());
890 gtk_css_style_property_register ("border-bottom-color",
891 GTK_TYPE_SYMBOLIC_COLOR,
898 _gtk_symbolic_color_get_current_color ());
899 gtk_css_style_property_register ("border-left-color",
900 GTK_TYPE_SYMBOLIC_COLOR,
907 _gtk_symbolic_color_get_current_color ());
908 gtk_css_style_property_register ("outline-color",
909 GTK_TYPE_SYMBOLIC_COLOR,
916 _gtk_symbolic_color_get_current_color ());
918 gtk_css_style_property_register ("background-repeat",
919 GTK_TYPE_CSS_BACKGROUND_REPEAT,
920 GTK_TYPE_CSS_BACKGROUND_REPEAT,
921 GTK_TYPE_CSS_BACKGROUND_REPEAT,
923 background_repeat_value_parse,
924 background_repeat_value_print,
926 GTK_CSS_BACKGROUND_REPEAT | (GTK_CSS_BACKGROUND_REPEAT << GTK_CSS_BACKGROUND_REPEAT_SHIFT));
927 gtk_css_style_property_register ("background-image",
930 CAIRO_GOBJECT_TYPE_PATTERN,
932 css_image_value_parse,
933 css_image_value_print,
934 css_image_value_compute,
937 gtk_css_style_property_register ("border-image-source",
940 CAIRO_GOBJECT_TYPE_PATTERN,
942 css_image_value_parse,
943 css_image_value_print,
944 css_image_value_compute,
946 gtk_css_style_property_register ("border-image-repeat",
947 GTK_TYPE_CSS_BORDER_IMAGE_REPEAT,
948 GTK_TYPE_CSS_BORDER_IMAGE_REPEAT,
949 GTK_TYPE_CSS_BORDER_IMAGE_REPEAT,
954 &border_image_repeat);
956 /* XXX: The initial value is wrong, it should be 100% */
957 gtk_css_style_property_register ("border-image-slice",
966 gtk_css_style_property_register ("border-image-width",
975 gtk_css_style_property_register ("engine",
976 GTK_TYPE_THEMING_ENGINE,
977 GTK_TYPE_THEMING_ENGINE,
978 GTK_TYPE_THEMING_ENGINE,
983 gtk_theming_engine_load (NULL));
984 gtk_css_style_property_register ("transition",
985 GTK_TYPE_ANIMATION_DESCRIPTION,
986 GTK_TYPE_ANIMATION_DESCRIPTION,
987 GTK_TYPE_ANIMATION_DESCRIPTION,
994 /* Private property holding the binding sets */
995 gtk_css_style_property_register ("gtk-key-bindings",
1000 bindings_value_parse,
1001 bindings_value_print,