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 "gtkthemingengine.h"
43 #include "gtktypebuiltins.h"
44 #include "gtkwin32themeprivate.h"
46 /*** REGISTRATION ***/
49 color_compute (GtkCssStyleProperty *property,
51 GtkStyleContext *context,
52 const GValue *specified)
54 g_value_init (computed, GDK_TYPE_RGBA);
56 /* for when resolvage fails */
59 if (G_VALUE_HOLDS (specified, GTK_TYPE_CSS_SPECIAL_VALUE))
61 g_assert (g_value_get_enum (specified) == GTK_CSS_CURRENT_COLOR);
63 /* The computed value of the ‘currentColor’ keyword is the computed
64 * value of the ‘color’ property. If the ‘currentColor’ keyword is
65 * set on the ‘color’ property itself, it is treated as ‘color: inherit’.
67 if (g_str_equal (_gtk_style_property_get_name (GTK_STYLE_PROPERTY (property)), "color"))
69 GtkStyleContext *parent = gtk_style_context_get_parent (context);
72 g_value_copy (_gtk_style_context_peek_property (parent, "color"), computed);
74 _gtk_css_style_compute_value (computed,
76 _gtk_css_style_property_get_initial_value (property));
80 g_value_copy (_gtk_style_context_peek_property (context, "color"), computed);
83 else if (G_VALUE_HOLDS (specified, GTK_TYPE_SYMBOLIC_COLOR))
87 if (!_gtk_style_context_resolve_color (context,
88 g_value_get_boxed (specified),
91 specified = _gtk_css_style_property_get_initial_value (property);
95 g_value_set_boxed (computed, &rgba);
98 g_value_copy (specified, computed);
102 _gtk_style_property_register (const char * name,
104 GtkStylePropertyFlags flags,
105 GtkCssStylePropertyParseFunc parse_value,
106 GtkCssStylePropertyPrintFunc print_value,
107 GtkCssStylePropertyComputeFunc compute_value,
108 const GValue * initial_value)
110 GtkCssStyleProperty *node;
112 node = g_object_new (GTK_TYPE_CSS_STYLE_PROPERTY,
113 "inherit", (flags & GTK_STYLE_PROPERTY_INHERIT) ? TRUE : FALSE,
114 "initial-value", initial_value,
116 "value-type", value_type,
120 node->parse_value = parse_value;
122 node->print_value = print_value;
124 node->compute_value = compute_value;
128 gtk_style_property_register (const char * name,
130 GtkStylePropertyFlags flags,
131 GtkCssStylePropertyParseFunc parse_value,
132 GtkCssStylePropertyPrintFunc print_value,
133 GtkCssStylePropertyComputeFunc compute_value,
136 GValue initial_value = G_VALUE_INIT;
140 va_start (args, compute_value);
141 G_VALUE_COLLECT_INIT (&initial_value, value_type,
145 g_error ("property `%s' initial value is broken: %s", name, error);
146 g_value_unset (&initial_value);
152 _gtk_style_property_register (name,
160 g_value_unset (&initial_value);
166 string_append_double (GString *string,
169 char buf[G_ASCII_DTOSTR_BUF_SIZE];
171 g_ascii_dtostr (buf, sizeof (buf), d);
172 g_string_append (string, buf);
176 string_append_string (GString *str,
181 g_string_append_c (str, '"');
184 len = strcspn (string, "\"\n\r\f");
185 g_string_append (str, string);
192 g_string_append (str, "\\A ");
195 g_string_append (str, "\\D ");
198 g_string_append (str, "\\C ");
201 g_string_append (str, "\\\"");
204 g_assert_not_reached ();
209 g_string_append_c (str, '"');
212 /*** IMPLEMENTATIONS ***/
215 font_family_parse (GtkCssStyleProperty *property,
217 GtkCssParser *parser,
223 /* We don't special case generic families. Pango should do
226 names = g_ptr_array_new ();
229 name = _gtk_css_parser_try_ident (parser, TRUE);
232 GString *string = g_string_new (name);
234 while ((name = _gtk_css_parser_try_ident (parser, TRUE)))
236 g_string_append_c (string, ' ');
237 g_string_append (string, name);
240 name = g_string_free (string, FALSE);
244 name = _gtk_css_parser_read_string (parser);
247 g_ptr_array_free (names, TRUE);
252 g_ptr_array_add (names, name);
253 } while (_gtk_css_parser_try (parser, ",", TRUE));
255 /* NULL-terminate array */
256 g_ptr_array_add (names, NULL);
257 g_value_set_boxed (value, g_ptr_array_free (names, FALSE));
262 font_family_value_print (GtkCssStyleProperty *property,
266 const char **names = g_value_get_boxed (value);
268 if (names == NULL || *names == NULL)
270 g_string_append (string, "none");
274 string_append_string (string, *names);
278 g_string_append (string, ", ");
279 string_append_string (string, *names);
285 bindings_value_parse (GtkCssStyleProperty *property,
287 GtkCssParser *parser,
291 GtkBindingSet *binding_set;
294 array = g_ptr_array_new ();
297 name = _gtk_css_parser_try_ident (parser, TRUE);
300 _gtk_css_parser_error (parser, "Not a valid binding name");
301 g_ptr_array_free (array, TRUE);
305 binding_set = gtk_binding_set_find (name);
309 _gtk_css_parser_error (parser, "No binding set named '%s'", name);
314 g_ptr_array_add (array, binding_set);
317 while (_gtk_css_parser_try (parser, ",", TRUE));
319 g_value_take_boxed (value, array);
325 bindings_value_print (GtkCssStyleProperty *property,
332 array = g_value_get_boxed (value);
334 for (i = 0; i < array->len; i++)
336 GtkBindingSet *binding_set = g_ptr_array_index (array, i);
339 g_string_append (string, ", ");
340 g_string_append (string, binding_set->set_name);
345 border_corner_radius_value_parse (GtkCssStyleProperty *property,
347 GtkCssParser *parser,
350 GtkCssBorderCornerRadius corner;
352 if (!_gtk_css_parser_try_double (parser, &corner.horizontal))
354 _gtk_css_parser_error (parser, "Expected a number");
357 else if (corner.horizontal < 0)
360 if (!_gtk_css_parser_try_double (parser, &corner.vertical))
361 corner.vertical = corner.horizontal;
362 else if (corner.vertical < 0)
365 g_value_set_boxed (value, &corner);
369 _gtk_css_parser_error (parser, "Border radius values cannot be negative");
374 border_corner_radius_value_print (GtkCssStyleProperty *property,
378 GtkCssBorderCornerRadius *corner;
380 corner = g_value_get_boxed (value);
384 g_string_append (string, "none");
388 string_append_double (string, corner->horizontal);
389 if (corner->horizontal != corner->vertical)
391 g_string_append_c (string, ' ');
392 string_append_double (string, corner->vertical);
397 css_image_value_parse (GtkCssStyleProperty *property,
399 GtkCssParser *parser,
404 if (_gtk_css_parser_try (parser, "none", TRUE))
408 image = _gtk_css_image_new_parse (parser, base);
413 g_value_unset (value);
414 g_value_init (value, GTK_TYPE_CSS_IMAGE);
415 g_value_take_object (value, image);
420 css_image_value_print (GtkCssStyleProperty *property,
424 GtkCssImage *image = g_value_get_object (value);
427 _gtk_css_image_print (image, string);
429 g_string_append (string, "none");
433 css_image_value_compute (GtkCssStyleProperty *property,
435 GtkStyleContext *context,
436 const GValue *specified)
438 GtkCssImage *image = g_value_get_object (specified);
441 image = _gtk_css_image_compute (image, context);
443 g_value_init (computed, GTK_TYPE_CSS_IMAGE);
444 g_value_take_object (computed, image);
448 compute_border_width (GtkCssStyleProperty *property,
450 GtkStyleContext *context,
451 const GValue *specified)
453 GtkCssStyleProperty *style;
454 GtkBorderStyle border_style;
456 /* The -1 is magic that is only true because we register the style
457 * properties directly after the width properties.
459 style = _gtk_css_style_property_lookup_by_id (_gtk_css_style_property_get_id (property) - 1);
460 border_style = g_value_get_enum (_gtk_style_context_peek_property (context, _gtk_style_property_get_name (GTK_STYLE_PROPERTY (style))));
462 g_value_init (computed, G_TYPE_INT);
463 if (border_style == GTK_BORDER_STYLE_NONE ||
464 border_style == GTK_BORDER_STYLE_HIDDEN)
465 g_value_set_int (computed, 0);
467 g_value_copy (specified, computed);
471 background_repeat_value_parse (GtkCssStyleProperty *property,
473 GtkCssParser *parser,
476 int repeat, vertical;
478 if (!_gtk_css_parser_try_enum (parser, GTK_TYPE_CSS_BACKGROUND_REPEAT, &repeat))
480 _gtk_css_parser_error (parser, "Not a valid value");
484 if (repeat <= GTK_CSS_BACKGROUND_REPEAT_MASK)
486 if (_gtk_css_parser_try_enum (parser, GTK_TYPE_CSS_BACKGROUND_REPEAT, &vertical))
488 if (vertical >= GTK_CSS_BACKGROUND_REPEAT_MASK)
490 _gtk_css_parser_error (parser, "Not a valid 2nd value");
494 repeat |= vertical << GTK_CSS_BACKGROUND_REPEAT_SHIFT;
497 repeat |= repeat << GTK_CSS_BACKGROUND_REPEAT_SHIFT;
500 g_value_set_enum (value, repeat);
505 background_repeat_value_print (GtkCssStyleProperty *property,
509 GEnumClass *enum_class;
510 GEnumValue *enum_value;
511 GtkCssBackgroundRepeat repeat;
513 repeat = g_value_get_enum (value);
514 enum_class = g_type_class_ref (GTK_TYPE_CSS_BACKGROUND_REPEAT);
515 enum_value = g_enum_get_value (enum_class, repeat);
517 /* only triggers for 'repeat-x' and 'repeat-y' */
519 g_string_append (string, enum_value->value_nick);
522 enum_value = g_enum_get_value (enum_class, GTK_CSS_BACKGROUND_HORIZONTAL (repeat));
523 g_string_append (string, enum_value->value_nick);
525 if (GTK_CSS_BACKGROUND_HORIZONTAL (repeat) != GTK_CSS_BACKGROUND_VERTICAL (repeat))
527 enum_value = g_enum_get_value (enum_class, GTK_CSS_BACKGROUND_VERTICAL (repeat));
528 g_string_append (string, " ");
529 g_string_append (string, enum_value->value_nick);
533 g_type_class_unref (enum_class);
536 /*** REGISTRATION ***/
538 #define rgba_init(rgba, r, g, b, a) G_STMT_START{ \
540 (rgba)->green = (g); \
541 (rgba)->blue = (b); \
542 (rgba)->alpha = (a); \
545 _gtk_css_style_property_init_properties (void)
547 GValue value = { 0, };
548 char *default_font_family[] = { "Sans", NULL };
550 GtkCssBorderCornerRadius no_corner_radius = { 0, };
551 GtkBorder border_of_ones = { 1, 1, 1, 1 };
552 GtkCssBorderImageRepeat border_image_repeat = { GTK_CSS_REPEAT_STYLE_STRETCH, GTK_CSS_REPEAT_STYLE_STRETCH };
554 /* Initialize "color" and "font-size" first,
555 * so that when computing values later they are
556 * done first. That way, 'currentColor' and font
557 * sizes in em can be looked up properly */
558 rgba_init (&rgba, 1, 1, 1, 1);
559 gtk_style_property_register ("color",
561 GTK_STYLE_PROPERTY_INHERIT,
566 gtk_style_property_register ("font-size",
568 GTK_STYLE_PROPERTY_INHERIT,
574 /* properties that aren't referenced when computing values
576 rgba_init (&rgba, 0, 0, 0, 0);
577 gtk_style_property_register ("background-color",
585 gtk_style_property_register ("font-family",
587 GTK_STYLE_PROPERTY_INHERIT,
589 font_family_value_print,
591 default_font_family);
592 gtk_style_property_register ("font-style",
594 GTK_STYLE_PROPERTY_INHERIT,
599 gtk_style_property_register ("font-variant",
601 GTK_STYLE_PROPERTY_INHERIT,
605 PANGO_VARIANT_NORMAL);
606 /* xxx: need to parse this properly, ie parse the numbers */
607 gtk_style_property_register ("font-weight",
609 GTK_STYLE_PROPERTY_INHERIT,
613 PANGO_WEIGHT_NORMAL);
615 gtk_style_property_register ("text-shadow",
617 GTK_STYLE_PROPERTY_INHERIT,
623 gtk_style_property_register ("icon-shadow",
625 GTK_STYLE_PROPERTY_INHERIT,
631 gtk_style_property_register ("box-shadow",
639 gtk_style_property_register ("margin-top",
646 gtk_style_property_register ("margin-left",
653 gtk_style_property_register ("margin-bottom",
660 gtk_style_property_register ("margin-right",
667 gtk_style_property_register ("padding-top",
674 gtk_style_property_register ("padding-left",
681 gtk_style_property_register ("padding-bottom",
688 gtk_style_property_register ("padding-right",
695 /* IMPORTANT: compute_border_width() requires that the border-width
696 * properties be immeditaly followed by the border-style properties
698 gtk_style_property_register ("border-top-style",
699 GTK_TYPE_BORDER_STYLE,
704 GTK_BORDER_STYLE_NONE);
705 gtk_style_property_register ("border-top-width",
710 compute_border_width,
712 gtk_style_property_register ("border-left-style",
713 GTK_TYPE_BORDER_STYLE,
718 GTK_BORDER_STYLE_NONE);
719 gtk_style_property_register ("border-left-width",
724 compute_border_width,
726 gtk_style_property_register ("border-bottom-style",
727 GTK_TYPE_BORDER_STYLE,
732 GTK_BORDER_STYLE_NONE);
733 gtk_style_property_register ("border-bottom-width",
738 compute_border_width,
740 gtk_style_property_register ("border-right-style",
741 GTK_TYPE_BORDER_STYLE,
746 GTK_BORDER_STYLE_NONE);
747 gtk_style_property_register ("border-right-width",
752 compute_border_width,
755 gtk_style_property_register ("border-top-left-radius",
756 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
758 border_corner_radius_value_parse,
759 border_corner_radius_value_print,
762 gtk_style_property_register ("border-top-right-radius",
763 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
765 border_corner_radius_value_parse,
766 border_corner_radius_value_print,
769 gtk_style_property_register ("border-bottom-right-radius",
770 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
772 border_corner_radius_value_parse,
773 border_corner_radius_value_print,
776 gtk_style_property_register ("border-bottom-left-radius",
777 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
779 border_corner_radius_value_parse,
780 border_corner_radius_value_print,
784 gtk_style_property_register ("outline-style",
785 GTK_TYPE_BORDER_STYLE,
790 GTK_BORDER_STYLE_NONE);
791 gtk_style_property_register ("outline-width",
796 compute_border_width,
798 gtk_style_property_register ("outline-offset",
806 gtk_style_property_register ("background-clip",
812 GTK_CSS_AREA_BORDER_BOX);
814 gtk_style_property_register ("background-origin",
820 GTK_CSS_AREA_PADDING_BOX);
822 g_value_init (&value, GTK_TYPE_CSS_SPECIAL_VALUE);
823 g_value_set_enum (&value, GTK_CSS_CURRENT_COLOR);
824 _gtk_style_property_register ("border-top-color",
831 _gtk_style_property_register ("border-right-color",
838 _gtk_style_property_register ("border-bottom-color",
845 _gtk_style_property_register ("border-left-color",
852 _gtk_style_property_register ("outline-color",
859 g_value_unset (&value);
861 gtk_style_property_register ("background-repeat",
862 GTK_TYPE_CSS_BACKGROUND_REPEAT,
864 background_repeat_value_parse,
865 background_repeat_value_print,
867 GTK_CSS_BACKGROUND_REPEAT | (GTK_CSS_BACKGROUND_REPEAT << GTK_CSS_BACKGROUND_REPEAT_SHIFT));
868 g_value_init (&value, GTK_TYPE_CSS_IMAGE);
869 _gtk_style_property_register ("background-image",
870 CAIRO_GOBJECT_TYPE_PATTERN,
872 css_image_value_parse,
873 css_image_value_print,
874 css_image_value_compute,
877 _gtk_style_property_register ("border-image-source",
878 CAIRO_GOBJECT_TYPE_PATTERN,
880 css_image_value_parse,
881 css_image_value_print,
882 css_image_value_compute,
884 g_value_unset (&value);
885 gtk_style_property_register ("border-image-repeat",
886 GTK_TYPE_CSS_BORDER_IMAGE_REPEAT,
891 &border_image_repeat);
893 /* XXX: The initial vaue is wrong, it should be 100% */
894 gtk_style_property_register ("border-image-slice",
901 gtk_style_property_register ("border-image-width",
908 gtk_style_property_register ("engine",
909 GTK_TYPE_THEMING_ENGINE,
914 gtk_theming_engine_load (NULL));
915 gtk_style_property_register ("transition",
916 GTK_TYPE_ANIMATION_DESCRIPTION,
923 /* Private property holding the binding sets */
924 gtk_style_property_register ("gtk-key-bindings",
927 bindings_value_parse,
928 bindings_value_print,