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 color_compute (GtkCssStyleProperty *property,
52 GtkStyleContext *context,
53 const GValue *specified)
55 g_value_init (computed, GDK_TYPE_RGBA);
57 /* for when resolvage fails */
60 if (G_VALUE_HOLDS (specified, GTK_TYPE_SYMBOLIC_COLOR))
62 GtkSymbolicColor *symbolic = g_value_get_boxed (specified);
65 if (symbolic == _gtk_symbolic_color_get_current_color ())
67 /* The computed value of the ‘currentColor’ keyword is the computed
68 * value of the ‘color’ property. If the ‘currentColor’ keyword is
69 * set on the ‘color’ property itself, it is treated as ‘color: inherit’.
71 if (g_str_equal (_gtk_style_property_get_name (GTK_STYLE_PROPERTY (property)), "color"))
73 GtkStyleContext *parent = gtk_style_context_get_parent (context);
76 g_value_copy (_gtk_style_context_peek_property (parent, "color"), computed);
78 _gtk_css_style_compute_value (computed,
80 _gtk_css_style_property_get_initial_value (property));
84 g_value_copy (_gtk_style_context_peek_property (context, "color"), computed);
87 else if (_gtk_style_context_resolve_color (context,
91 g_value_set_boxed (computed, &rgba);
95 specified = _gtk_css_style_property_get_initial_value (property);
101 g_value_copy (specified, computed);
105 _gtk_style_property_register (const char * name,
107 GtkStylePropertyFlags flags,
108 GtkCssStylePropertyParseFunc parse_value,
109 GtkCssStylePropertyPrintFunc print_value,
110 GtkCssStylePropertyComputeFunc compute_value,
111 const GValue * initial_value)
113 GtkCssStyleProperty *node;
115 node = g_object_new (GTK_TYPE_CSS_STYLE_PROPERTY,
116 "inherit", (flags & GTK_STYLE_PROPERTY_INHERIT) ? TRUE : FALSE,
117 "initial-value", initial_value,
119 "value-type", value_type,
123 node->parse_value = parse_value;
125 node->print_value = print_value;
127 node->compute_value = compute_value;
131 gtk_style_property_register (const char * name,
133 GtkStylePropertyFlags flags,
134 GtkCssStylePropertyParseFunc parse_value,
135 GtkCssStylePropertyPrintFunc print_value,
136 GtkCssStylePropertyComputeFunc compute_value,
139 GValue initial_value = G_VALUE_INIT;
143 va_start (args, compute_value);
144 G_VALUE_COLLECT_INIT (&initial_value, value_type,
148 g_error ("property `%s' initial value is broken: %s", name, error);
149 g_value_unset (&initial_value);
155 _gtk_style_property_register (name,
163 g_value_unset (&initial_value);
169 string_append_double (GString *string,
172 char buf[G_ASCII_DTOSTR_BUF_SIZE];
174 g_ascii_dtostr (buf, sizeof (buf), d);
175 g_string_append (string, buf);
179 string_append_string (GString *str,
184 g_string_append_c (str, '"');
187 len = strcspn (string, "\"\n\r\f");
188 g_string_append (str, string);
195 g_string_append (str, "\\A ");
198 g_string_append (str, "\\D ");
201 g_string_append (str, "\\C ");
204 g_string_append (str, "\\\"");
207 g_assert_not_reached ();
212 g_string_append_c (str, '"');
215 /*** IMPLEMENTATIONS ***/
218 font_family_parse (GtkCssStyleProperty *property,
220 GtkCssParser *parser,
226 /* We don't special case generic families. Pango should do
229 names = g_ptr_array_new ();
232 name = _gtk_css_parser_try_ident (parser, TRUE);
235 GString *string = g_string_new (name);
237 while ((name = _gtk_css_parser_try_ident (parser, TRUE)))
239 g_string_append_c (string, ' ');
240 g_string_append (string, name);
243 name = g_string_free (string, FALSE);
247 name = _gtk_css_parser_read_string (parser);
250 g_ptr_array_free (names, TRUE);
255 g_ptr_array_add (names, name);
256 } while (_gtk_css_parser_try (parser, ",", TRUE));
258 /* NULL-terminate array */
259 g_ptr_array_add (names, NULL);
260 g_value_set_boxed (value, g_ptr_array_free (names, FALSE));
265 font_family_value_print (GtkCssStyleProperty *property,
269 const char **names = g_value_get_boxed (value);
271 if (names == NULL || *names == NULL)
273 g_string_append (string, "none");
277 string_append_string (string, *names);
281 g_string_append (string, ", ");
282 string_append_string (string, *names);
288 bindings_value_parse (GtkCssStyleProperty *property,
290 GtkCssParser *parser,
294 GtkBindingSet *binding_set;
297 array = g_ptr_array_new ();
300 name = _gtk_css_parser_try_ident (parser, TRUE);
303 _gtk_css_parser_error (parser, "Not a valid binding name");
304 g_ptr_array_free (array, TRUE);
308 binding_set = gtk_binding_set_find (name);
312 _gtk_css_parser_error (parser, "No binding set named '%s'", name);
317 g_ptr_array_add (array, binding_set);
320 while (_gtk_css_parser_try (parser, ",", TRUE));
322 g_value_take_boxed (value, array);
328 bindings_value_print (GtkCssStyleProperty *property,
335 array = g_value_get_boxed (value);
337 for (i = 0; i < array->len; i++)
339 GtkBindingSet *binding_set = g_ptr_array_index (array, i);
342 g_string_append (string, ", ");
343 g_string_append (string, binding_set->set_name);
348 border_corner_radius_value_parse (GtkCssStyleProperty *property,
350 GtkCssParser *parser,
353 GtkCssBorderCornerRadius corner;
355 if (!_gtk_css_parser_try_double (parser, &corner.horizontal))
357 _gtk_css_parser_error (parser, "Expected a number");
360 else if (corner.horizontal < 0)
363 if (!_gtk_css_parser_try_double (parser, &corner.vertical))
364 corner.vertical = corner.horizontal;
365 else if (corner.vertical < 0)
368 g_value_set_boxed (value, &corner);
372 _gtk_css_parser_error (parser, "Border radius values cannot be negative");
377 border_corner_radius_value_print (GtkCssStyleProperty *property,
381 GtkCssBorderCornerRadius *corner;
383 corner = g_value_get_boxed (value);
387 g_string_append (string, "none");
391 string_append_double (string, corner->horizontal);
392 if (corner->horizontal != corner->vertical)
394 g_string_append_c (string, ' ');
395 string_append_double (string, corner->vertical);
400 css_image_value_parse (GtkCssStyleProperty *property,
402 GtkCssParser *parser,
407 if (_gtk_css_parser_try (parser, "none", TRUE))
411 image = _gtk_css_image_new_parse (parser, base);
416 g_value_unset (value);
417 g_value_init (value, GTK_TYPE_CSS_IMAGE);
418 g_value_take_object (value, image);
423 css_image_value_print (GtkCssStyleProperty *property,
427 GtkCssImage *image = g_value_get_object (value);
430 _gtk_css_image_print (image, string);
432 g_string_append (string, "none");
436 css_image_value_compute (GtkCssStyleProperty *property,
438 GtkStyleContext *context,
439 const GValue *specified)
441 GtkCssImage *image = g_value_get_object (specified);
444 image = _gtk_css_image_compute (image, context);
446 g_value_init (computed, GTK_TYPE_CSS_IMAGE);
447 g_value_take_object (computed, image);
451 compute_border_width (GtkCssStyleProperty *property,
453 GtkStyleContext *context,
454 const GValue *specified)
456 GtkCssStyleProperty *style;
457 GtkBorderStyle border_style;
459 /* The -1 is magic that is only true because we register the style
460 * properties directly after the width properties.
462 style = _gtk_css_style_property_lookup_by_id (_gtk_css_style_property_get_id (property) - 1);
463 border_style = g_value_get_enum (_gtk_style_context_peek_property (context, _gtk_style_property_get_name (GTK_STYLE_PROPERTY (style))));
465 g_value_init (computed, G_TYPE_INT);
466 if (border_style == GTK_BORDER_STYLE_NONE ||
467 border_style == GTK_BORDER_STYLE_HIDDEN)
468 g_value_set_int (computed, 0);
470 g_value_copy (specified, computed);
474 background_repeat_value_parse (GtkCssStyleProperty *property,
476 GtkCssParser *parser,
479 int repeat, vertical;
481 if (!_gtk_css_parser_try_enum (parser, GTK_TYPE_CSS_BACKGROUND_REPEAT, &repeat))
483 _gtk_css_parser_error (parser, "Not a valid value");
487 if (repeat <= GTK_CSS_BACKGROUND_REPEAT_MASK)
489 if (_gtk_css_parser_try_enum (parser, GTK_TYPE_CSS_BACKGROUND_REPEAT, &vertical))
491 if (vertical >= GTK_CSS_BACKGROUND_REPEAT_MASK)
493 _gtk_css_parser_error (parser, "Not a valid 2nd value");
497 repeat |= vertical << GTK_CSS_BACKGROUND_REPEAT_SHIFT;
500 repeat |= repeat << GTK_CSS_BACKGROUND_REPEAT_SHIFT;
503 g_value_set_enum (value, repeat);
508 background_repeat_value_print (GtkCssStyleProperty *property,
512 GEnumClass *enum_class;
513 GEnumValue *enum_value;
514 GtkCssBackgroundRepeat repeat;
516 repeat = g_value_get_enum (value);
517 enum_class = g_type_class_ref (GTK_TYPE_CSS_BACKGROUND_REPEAT);
518 enum_value = g_enum_get_value (enum_class, repeat);
520 /* only triggers for 'repeat-x' and 'repeat-y' */
522 g_string_append (string, enum_value->value_nick);
525 enum_value = g_enum_get_value (enum_class, GTK_CSS_BACKGROUND_HORIZONTAL (repeat));
526 g_string_append (string, enum_value->value_nick);
528 if (GTK_CSS_BACKGROUND_HORIZONTAL (repeat) != GTK_CSS_BACKGROUND_VERTICAL (repeat))
530 enum_value = g_enum_get_value (enum_class, GTK_CSS_BACKGROUND_VERTICAL (repeat));
531 g_string_append (string, " ");
532 g_string_append (string, enum_value->value_nick);
536 g_type_class_unref (enum_class);
539 /*** REGISTRATION ***/
541 #define rgba_init(rgba, r, g, b, a) G_STMT_START{ \
543 (rgba)->green = (g); \
544 (rgba)->blue = (b); \
545 (rgba)->alpha = (a); \
548 _gtk_css_style_property_init_properties (void)
550 GValue value = { 0, };
551 char *default_font_family[] = { "Sans", NULL };
553 GtkCssBorderCornerRadius no_corner_radius = { 0, };
554 GtkBorder border_of_ones = { 1, 1, 1, 1 };
555 GtkCssBorderImageRepeat border_image_repeat = { GTK_CSS_REPEAT_STYLE_STRETCH, GTK_CSS_REPEAT_STYLE_STRETCH };
557 /* Initialize "color" and "font-size" first,
558 * so that when computing values later they are
559 * done first. That way, 'currentColor' and font
560 * sizes in em can be looked up properly */
561 rgba_init (&rgba, 1, 1, 1, 1);
562 gtk_style_property_register ("color",
564 GTK_STYLE_PROPERTY_INHERIT,
569 gtk_style_property_register ("font-size",
571 GTK_STYLE_PROPERTY_INHERIT,
577 /* properties that aren't referenced when computing values
579 rgba_init (&rgba, 0, 0, 0, 0);
580 gtk_style_property_register ("background-color",
588 gtk_style_property_register ("font-family",
590 GTK_STYLE_PROPERTY_INHERIT,
592 font_family_value_print,
594 default_font_family);
595 gtk_style_property_register ("font-style",
597 GTK_STYLE_PROPERTY_INHERIT,
602 gtk_style_property_register ("font-variant",
604 GTK_STYLE_PROPERTY_INHERIT,
608 PANGO_VARIANT_NORMAL);
609 /* xxx: need to parse this properly, ie parse the numbers */
610 gtk_style_property_register ("font-weight",
612 GTK_STYLE_PROPERTY_INHERIT,
616 PANGO_WEIGHT_NORMAL);
618 gtk_style_property_register ("text-shadow",
620 GTK_STYLE_PROPERTY_INHERIT,
626 gtk_style_property_register ("icon-shadow",
628 GTK_STYLE_PROPERTY_INHERIT,
634 gtk_style_property_register ("box-shadow",
642 gtk_style_property_register ("margin-top",
649 gtk_style_property_register ("margin-left",
656 gtk_style_property_register ("margin-bottom",
663 gtk_style_property_register ("margin-right",
670 gtk_style_property_register ("padding-top",
677 gtk_style_property_register ("padding-left",
684 gtk_style_property_register ("padding-bottom",
691 gtk_style_property_register ("padding-right",
698 /* IMPORTANT: compute_border_width() requires that the border-width
699 * properties be immeditaly followed by the border-style properties
701 gtk_style_property_register ("border-top-style",
702 GTK_TYPE_BORDER_STYLE,
707 GTK_BORDER_STYLE_NONE);
708 gtk_style_property_register ("border-top-width",
713 compute_border_width,
715 gtk_style_property_register ("border-left-style",
716 GTK_TYPE_BORDER_STYLE,
721 GTK_BORDER_STYLE_NONE);
722 gtk_style_property_register ("border-left-width",
727 compute_border_width,
729 gtk_style_property_register ("border-bottom-style",
730 GTK_TYPE_BORDER_STYLE,
735 GTK_BORDER_STYLE_NONE);
736 gtk_style_property_register ("border-bottom-width",
741 compute_border_width,
743 gtk_style_property_register ("border-right-style",
744 GTK_TYPE_BORDER_STYLE,
749 GTK_BORDER_STYLE_NONE);
750 gtk_style_property_register ("border-right-width",
755 compute_border_width,
758 gtk_style_property_register ("border-top-left-radius",
759 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
761 border_corner_radius_value_parse,
762 border_corner_radius_value_print,
765 gtk_style_property_register ("border-top-right-radius",
766 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
768 border_corner_radius_value_parse,
769 border_corner_radius_value_print,
772 gtk_style_property_register ("border-bottom-right-radius",
773 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
775 border_corner_radius_value_parse,
776 border_corner_radius_value_print,
779 gtk_style_property_register ("border-bottom-left-radius",
780 GTK_TYPE_CSS_BORDER_CORNER_RADIUS,
782 border_corner_radius_value_parse,
783 border_corner_radius_value_print,
787 gtk_style_property_register ("outline-style",
788 GTK_TYPE_BORDER_STYLE,
793 GTK_BORDER_STYLE_NONE);
794 gtk_style_property_register ("outline-width",
799 compute_border_width,
801 gtk_style_property_register ("outline-offset",
809 gtk_style_property_register ("background-clip",
815 GTK_CSS_AREA_BORDER_BOX);
817 gtk_style_property_register ("background-origin",
823 GTK_CSS_AREA_PADDING_BOX);
825 g_value_init (&value, GTK_TYPE_SYMBOLIC_COLOR);
826 g_value_set_boxed (&value, _gtk_symbolic_color_get_current_color ());
827 _gtk_style_property_register ("border-top-color",
834 _gtk_style_property_register ("border-right-color",
841 _gtk_style_property_register ("border-bottom-color",
848 _gtk_style_property_register ("border-left-color",
855 _gtk_style_property_register ("outline-color",
862 g_value_unset (&value);
864 gtk_style_property_register ("background-repeat",
865 GTK_TYPE_CSS_BACKGROUND_REPEAT,
867 background_repeat_value_parse,
868 background_repeat_value_print,
870 GTK_CSS_BACKGROUND_REPEAT | (GTK_CSS_BACKGROUND_REPEAT << GTK_CSS_BACKGROUND_REPEAT_SHIFT));
871 g_value_init (&value, GTK_TYPE_CSS_IMAGE);
872 _gtk_style_property_register ("background-image",
873 CAIRO_GOBJECT_TYPE_PATTERN,
875 css_image_value_parse,
876 css_image_value_print,
877 css_image_value_compute,
880 _gtk_style_property_register ("border-image-source",
881 CAIRO_GOBJECT_TYPE_PATTERN,
883 css_image_value_parse,
884 css_image_value_print,
885 css_image_value_compute,
887 g_value_unset (&value);
888 gtk_style_property_register ("border-image-repeat",
889 GTK_TYPE_CSS_BORDER_IMAGE_REPEAT,
894 &border_image_repeat);
896 /* XXX: The initial vaue is wrong, it should be 100% */
897 gtk_style_property_register ("border-image-slice",
904 gtk_style_property_register ("border-image-width",
911 gtk_style_property_register ("engine",
912 GTK_TYPE_THEMING_ENGINE,
917 gtk_theming_engine_load (NULL));
918 gtk_style_property_register ("transition",
919 GTK_TYPE_ANIMATION_DESCRIPTION,
926 /* Private property holding the binding sets */
927 gtk_style_property_register ("gtk-key-bindings",
930 bindings_value_parse,
931 bindings_value_print,