[gtk+/wip/css: 69/125] shorthand: Redo shorthand value parsing



commit 5138c736605017de19dffe89b05c259865343afc
Author: Benjamin Otte <otte redhat com>
Date:   Mon Jan 2 02:05:05 2012 +0100

    shorthand: Redo shorthand value parsing
    
    The new approach does not need unpack functions anymore.

 gtk/gtkcssprovider.c                 |   21 +++----
 gtk/gtkcssshorthandproperty.c        |  111 ++++++++++++++++++++++++++++++++++
 gtk/gtkcssshorthandpropertyprivate.h |    6 ++
 3 files changed, 126 insertions(+), 12 deletions(-)
---
diff --git a/gtk/gtkcssprovider.c b/gtk/gtkcssprovider.c
index 485edb2..03b3dbf 100644
--- a/gtk/gtkcssprovider.c
+++ b/gtk/gtkcssprovider.c
@@ -1193,24 +1193,21 @@ gtk_css_ruleset_add (GtkCssRuleset    *ruleset,
 
   if (GTK_IS_CSS_SHORTHAND_PROPERTY (prop))
     {
-      GParameter *parameters;
-      guint i, n_parameters;
+      GtkCssShorthandProperty *shorthand = GTK_CSS_SHORTHAND_PROPERTY (prop);
+      GValueArray *array = g_value_get_boxed (&value->value);
+      guint i;
 
-      parameters = _gtk_style_property_unpack (prop, &value->value, &n_parameters);
-
-      for (i = 0; i < n_parameters; i++)
+      for (i = 0; i < _gtk_css_shorthand_property_get_n_subproperties (shorthand); i++)
         {
-          GtkStyleProperty *child;
+          GtkCssStyleProperty *child = _gtk_css_shorthand_property_get_subproperty (shorthand, i);
           PropertyValue *val;
-
-          child = _gtk_style_property_lookup (parameters[i].name);
+          
           val = property_value_new (value->section);
-          memcpy (&val->value, &parameters[i].value, sizeof (GValue));
-          gtk_css_ruleset_add (ruleset, child, val);
+          g_value_init (&val->value, G_VALUE_TYPE (g_value_array_get_nth (array, i)));
+          g_value_copy (g_value_array_get_nth (array, i), &val->value);
+          gtk_css_ruleset_add (ruleset, GTK_STYLE_PROPERTY (child), val);
         }
-      g_free (parameters);
       property_value_free (value);
-      return;
     }
   else if (GTK_IS_CSS_STYLE_PROPERTY (prop))
     {
diff --git a/gtk/gtkcssshorthandproperty.c b/gtk/gtkcssshorthandproperty.c
index 6b85a98..bc0e802 100644
--- a/gtk/gtkcssshorthandproperty.c
+++ b/gtk/gtkcssshorthandproperty.c
@@ -22,7 +22,10 @@
 
 #include "gtkcssshorthandpropertyprivate.h"
 
+#include "gtkcssstylefuncsprivate.h"
+#include "gtkcsstypesprivate.h"
 #include "gtkintl.h"
+#include "gtkprivatetypebuiltins.h"
 
 enum {
   PROP_0,
@@ -91,6 +94,59 @@ _gtk_css_shorthand_property_query (GtkStyleProperty   *property,
   property->pack_func (value, props, state, context);
 }
 
+static gboolean
+gtk_css_shorthand_property_parse_value (GtkStyleProperty *property,
+                                        GValue           *value,
+                                        GtkCssParser     *parser,
+                                        GFile            *base)
+{
+  GtkCssShorthandProperty *shorthand = GTK_CSS_SHORTHAND_PROPERTY (property);
+  GValueArray *array;
+  guint i;
+
+  array = g_value_array_new (shorthand->subproperties->len);
+  for (i = 0; i < shorthand->subproperties->len; i++)
+    g_value_array_append (array, NULL);
+
+  if (_gtk_css_parser_try (parser, "initial", TRUE))
+    {
+      /* the initial value can be explicitly specified with the
+       * âinitialâ keyword which all properties accept.
+       */
+      for (i = 0; i < shorthand->subproperties->len; i++)
+        {
+          GValue *val = g_value_array_get_nth (array, i);
+          g_value_init (val, GTK_TYPE_CSS_SPECIAL_VALUE);
+          g_value_set_enum (val, GTK_CSS_INITIAL);
+        }
+    }
+  else if (_gtk_css_parser_try (parser, "inherit", TRUE))
+    {
+      /* All properties accept the âinheritâ value which
+       * explicitly specifies that the value will be determined
+       * by inheritance. The âinheritâ value can be used to
+       * strengthen inherited values in the cascade, and it can
+       * also be used on properties that are not normally inherited.
+       */
+      for (i = 0; i < shorthand->subproperties->len; i++)
+        {
+          GValue *val = g_value_array_get_nth (array, i);
+          g_value_init (val, GTK_TYPE_CSS_SPECIAL_VALUE);
+          g_value_set_enum (val, GTK_CSS_INHERIT);
+        }
+    }
+  else if (!shorthand->parse (shorthand, array->values, parser, base))
+    {
+      g_value_array_free (array);
+      return FALSE;
+    }
+
+  g_value_unset (value);
+  g_value_init (value, G_TYPE_VALUE_ARRAY);
+  g_value_set_boxed (value, array);
+  return TRUE;
+}
+
 static void
 _gtk_css_shorthand_property_class_init (GtkCssShorthandPropertyClass *klass)
 {
@@ -109,12 +165,67 @@ _gtk_css_shorthand_property_class_init (GtkCssShorthandPropertyClass *klass)
 
   property_class->assign = _gtk_css_shorthand_property_assign;
   property_class->query = _gtk_css_shorthand_property_query;
+  property_class->parse_value = gtk_css_shorthand_property_parse_value;
+}
+
+/* XXX: This function is compat only, don't read it */
+static gboolean
+gtk_css_shorthand_property_parse (GtkCssShorthandProperty *shorthand,
+                                  GValue                  *values,
+                                  GtkCssParser            *parser,
+                                  GFile                   *base)
+{
+  GtkStyleProperty *property = GTK_STYLE_PROPERTY (shorthand);
+  GParameter *parameters;
+  guint i, j, n_parameters;
+
+  GValue val = G_VALUE_INIT;
+
+  g_value_init (&val, _gtk_style_property_get_value_type (property));
+  if (property->parse_func)
+    {
+      if (!(* property->parse_func) (parser, base, &val))
+        {
+          g_value_unset (&val);
+          return FALSE;
+        }
+    }
+  else if (!_gtk_css_style_parse_value (&val, parser, base))
+    {
+      g_value_unset (&val);
+      return FALSE;
+    }
+
+  parameters = _gtk_style_property_unpack (property, &val, &n_parameters);
+  g_value_unset (&val);
+
+  for (i = 0; i < shorthand->subproperties->len; i++)
+    {
+      for (j = 0; j < n_parameters; j++)
+        {
+          if (GTK_STYLE_PROPERTY (_gtk_css_shorthand_property_get_subproperty (shorthand, i))
+              == _gtk_style_property_lookup (parameters[j].name))
+            {
+              g_value_init (&values[i], G_VALUE_TYPE (&parameters[j].value));
+              g_value_copy (&parameters[j].value, &values[i]);
+              g_value_unset (&parameters[j].value);
+              break;
+            }
+        }
+      g_assert (j < n_parameters);
+    }
+  
+  g_free (parameters);
+
+  return TRUE;
 }
 
 static void
 _gtk_css_shorthand_property_init (GtkCssShorthandProperty *shorthand)
 {
   shorthand->subproperties = g_ptr_array_new_with_free_func (g_object_unref);
+
+  shorthand->parse = gtk_css_shorthand_property_parse;
 }
 
 GtkCssStyleProperty *
diff --git a/gtk/gtkcssshorthandpropertyprivate.h b/gtk/gtkcssshorthandpropertyprivate.h
index c53bed7..c07d36c 100644
--- a/gtk/gtkcssshorthandpropertyprivate.h
+++ b/gtk/gtkcssshorthandpropertyprivate.h
@@ -39,11 +39,17 @@ G_BEGIN_DECLS
 typedef struct _GtkCssShorthandProperty           GtkCssShorthandProperty;
 typedef struct _GtkCssShorthandPropertyClass      GtkCssShorthandPropertyClass;
 
+typedef gboolean              (* GtkCssShorthandPropertyParseFunc)      (GtkCssShorthandProperty *shorthand,
+                                                                         GValue                  *values,
+                                                                         GtkCssParser            *parser,
+                                                                         GFile                   *base);
 struct _GtkCssShorthandProperty
 {
   GtkStyleProperty parent;
 
   GPtrArray *subproperties;
+
+  GtkCssShorthandPropertyParseFunc parse;
 };
 
 struct _GtkCssShorthandPropertyClass



[Date Prev][Date Next]   [Thread Prev][Thread Next]   [Thread Index] [Date Index] [Author Index]