[clutter/wip/apocalypses/apocalypse-1: 4/46] actor: Add :layout-manager



commit 2785395c3d32a89f91e9b808eb9dc1badf8cfd78
Author: Emmanuele Bassi <ebassi linux intel com>
Date:   Thu Nov 17 17:07:10 2011 +0000

    actor: Add :layout-manager
    
    Now that ClutterActor implements the Container contract we can actually
    defer the size negotiation to a ClutterLayoutManager directly from the
    default implementation of the Actor's virtual functions.

 clutter/clutter-actor.c          |  144 ++++++++++++++++++++++++++++++++++++++
 clutter/clutter-actor.h          |    4 +
 clutter/clutter-layout-manager.h |    1 -
 clutter/clutter-types.h          |    1 +
 4 files changed, 149 insertions(+), 1 deletions(-)
---
diff --git a/clutter/clutter-actor.c b/clutter/clutter-actor.c
index d493fc1..df88569 100644
--- a/clutter/clutter-actor.c
+++ b/clutter/clutter-actor.c
@@ -492,6 +492,8 @@ struct _ClutterActorPrivate
 
   ClutterStageQueueRedrawEntry *queue_redraw_entry;
 
+  ClutterLayoutManager *layout_manager;
+
   /* bitfields */
   guint position_set                : 1;
   guint min_width_set               : 1;
@@ -608,6 +610,8 @@ enum
   PROP_CONSTRAINTS,
   PROP_EFFECT,
 
+  PROP_LAYOUT_MANAGER,
+
   PROP_LAST
 };
 
@@ -1756,6 +1760,26 @@ clutter_actor_real_get_preferred_width (ClutterActor *self,
                                         gfloat       *min_width_p,
                                         gfloat       *natural_width_p)
 {
+  ClutterActorPrivate *priv = self->priv;
+
+  if (priv->children != NULL && priv->layout_manager != NULL)
+    {
+      ClutterContainer *container = CLUTTER_CONTAINER (self);
+
+      CLUTTER_NOTE (LAYOUT, "Querying the layout manager '%s'[%p] "
+                    "for the preferred width",
+                    G_OBJECT_TYPE_NAME (priv->layout_manager),
+                    priv->layout_manager);
+
+      clutter_layout_manager_get_preferred_width (priv->layout_manager,
+                                                  container,
+                                                  for_height,
+                                                  min_width_p,
+                                                  natural_width_p);
+
+      return;
+    }
+
   /* Default implementation is always 0x0, usually an actor
    * using this default is relying on someone to set the
    * request manually
@@ -1775,6 +1799,25 @@ clutter_actor_real_get_preferred_height (ClutterActor *self,
                                          gfloat       *min_height_p,
                                          gfloat       *natural_height_p)
 {
+  ClutterActorPrivate *priv = self->priv;
+
+  if (priv->children != NULL && priv->layout_manager != NULL)
+    {
+      ClutterContainer *container = CLUTTER_CONTAINER (self);
+
+      CLUTTER_NOTE (LAYOUT, "Querying the layout manager '%s'[%p] "
+                    "for the preferred height",
+                    G_OBJECT_TYPE_NAME (priv->layout_manager),
+                    priv->layout_manager);
+
+      clutter_layout_manager_get_preferred_height (priv->layout_manager,
+                                                   container,
+                                                   for_width,
+                                                   min_height_p,
+                                                   natural_height_p);
+
+      return;
+    }
   /* Default implementation is always 0x0, usually an actor
    * using this default is relying on someone to set the
    * request manually
@@ -1876,6 +1919,28 @@ clutter_actor_real_allocate (ClutterActor           *self,
   priv->allocation_flags = flags;
   priv->needs_allocation = FALSE;
 
+  /* we allocate our children before we notify changes in our geometry,
+   * so that people connecting to properties will be able to get valid
+   * data out of the sub-tree of the scene graph that has this actor at
+   * the root
+   */
+  if (priv->children != NULL && priv->layout_manager != NULL)
+    {
+      ClutterContainer *container = CLUTTER_CONTAINER (self);
+      gfloat width, height;
+      ClutterActorBox children_box;
+
+      clutter_actor_box_get_size (&priv->allocation, &width, &height);
+
+      clutter_actor_box_set_origin (&children_box, 0, 0);
+      clutter_actor_box_set_size (&children_box, width, height);
+
+      clutter_layout_manager_allocate (priv->layout_manager,
+                                       container,
+                                       &children_box,
+                                       priv->allocation_flags);
+    }
+
   g_object_freeze_notify (G_OBJECT (self));
 
   if (x1_changed || y1_changed || x2_changed || y2_changed || flags_changed)
@@ -3449,6 +3514,10 @@ clutter_actor_set_property (GObject      *object,
       clutter_actor_add_effect (actor, g_value_get_object (value));
       break;
 
+    case PROP_LAYOUT_MANAGER:
+      clutter_actor_set_layout_manager (actor, g_value_get_object (value));
+      break;
+
     default:
       G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
       break;
@@ -3714,6 +3783,10 @@ clutter_actor_get_property (GObject    *object,
       g_value_set_boolean (value, priv->has_pointer);
       break;
 
+    case PROP_LAYOUT_MANAGER:
+      g_value_set_object (value, priv->layout_manager);
+      break;
+
     default:
       G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
       break;
@@ -4766,6 +4839,15 @@ clutter_actor_class_init (ClutterActorClass *klass)
   obj_props[PROP_EFFECT] = pspec;
   g_object_class_install_property (object_class, PROP_EFFECT, pspec);
 
+  obj_props[PROP_LAYOUT_MANAGER] =
+    g_param_spec_object ("layout-manager",
+                         P_("Layout Manager"),
+                         P_("The object controlling the layout of an actor's children"),
+                         CLUTTER_TYPE_LAYOUT_MANAGER,
+                         CLUTTER_PARAM_READWRITE);
+  g_object_class_install_property (object_class, PROP_LAYOUT_MANAGER,
+                                   obj_props[PROP_LAYOUT_MANAGER]);
+
   /**
    * ClutterActor::destroy:
    * @actor: the #ClutterActor which emitted the signal
@@ -12828,3 +12910,65 @@ _clutter_actor_traverse (ClutterActor              *actor,
                                    0, /* start depth */
                                    user_data);
 }
+
+static void
+on_layout_manager_changed (ClutterLayoutManager *manager,
+                           ClutterActor         *self)
+{
+  clutter_actor_queue_relayout (self);
+}
+
+void
+clutter_actor_set_layout_manager (ClutterActor         *self,
+                                  ClutterLayoutManager *manager)
+{
+  ClutterActorPrivate *priv;
+
+  g_return_if_fail (CLUTTER_IS_ACTOR (self));
+  g_return_if_fail (manager == NULL || CLUTTER_IS_LAYOUT_MANAGER (manager));
+
+  priv = self->priv;
+
+  if (priv->layout_manager != NULL)
+    {
+      g_signal_handlers_disconnect_by_func (priv->layout_manager,
+                                            G_CALLBACK (on_layout_manager_changed),
+                                            self);
+      clutter_layout_manager_set_container (priv->layout_manager, NULL);
+      g_object_unref (priv->layout_manager);
+    }
+
+  priv->layout_manager = manager;
+
+  if (priv->layout_manager != NULL)
+    {
+      g_object_ref_sink (priv->layout_manager);
+      clutter_layout_manager_set_container (priv->layout_manager,
+                                            CLUTTER_CONTAINER (self));
+      g_signal_connect (priv->layout_manager, "layout-changed",
+                        G_CALLBACK (on_layout_manager_changed),
+                        self);
+    }
+
+  clutter_actor_queue_relayout (self);
+  g_object_notify_by_pspec (G_OBJECT (self), obj_props[PROP_LAYOUT_MANAGER]);
+}
+
+/**
+ * clutter_actor_get_layout_manager:
+ * @self: a #ClutterActor
+ *
+ * Retrieves the #ClutterLayoutManager used by @self.
+ *
+ * Return value: (transfer none): a pointer to the #ClutterLayoutManager,
+ *   or %NULL
+ *
+ * Since: 1.10
+ */
+ClutterLayoutManager *
+clutter_actor_get_layout_manager (ClutterActor *self)
+{
+  g_return_val_if_fail (CLUTTER_IS_ACTOR (self), NULL);
+
+  return self->priv->layout_manager;
+}
diff --git a/clutter/clutter-actor.h b/clutter/clutter-actor.h
index 6cef2cb..80d9f1b 100644
--- a/clutter/clutter-actor.h
+++ b/clutter/clutter-actor.h
@@ -361,6 +361,10 @@ void                  clutter_actor_set_x                     (ClutterActor
                                                                gfloat                 x);
 void                  clutter_actor_set_y                     (ClutterActor          *self,
                                                                gfloat                 y);
+void                  clutter_actor_set_layout_manager        (ClutterActor          *self,
+                                                               ClutterLayoutManager  *manager);
+ClutterLayoutManager *clutter_actor_get_layout_manager        (ClutterActor          *self);
+
 void                  clutter_actor_set_rotation              (ClutterActor          *self,
                                                                ClutterRotateAxis      axis,
                                                                gdouble                angle,
diff --git a/clutter/clutter-layout-manager.h b/clutter/clutter-layout-manager.h
index c735869..db6ce07 100644
--- a/clutter/clutter-layout-manager.h
+++ b/clutter/clutter-layout-manager.h
@@ -43,7 +43,6 @@ G_BEGIN_DECLS
 #define CLUTTER_IS_LAYOUT_MANAGER_CLASS(klass)  (G_TYPE_CHECK_CLASS_TYPE ((klass), CLUTTER_TYPE_LAYOUT_MANAGER))
 #define CLUTTER_LAYOUT_MANAGER_GET_CLASS(obj)   (G_TYPE_INSTANCE_GET_CLASS ((obj), CLUTTER_TYPE_LAYOUT_MANAGER, ClutterLayoutManagerClass))
 
-typedef struct _ClutterLayoutManager            ClutterLayoutManager;
 typedef struct _ClutterLayoutManagerClass       ClutterLayoutManagerClass;
 
 /**
diff --git a/clutter/clutter-types.h b/clutter/clutter-types.h
index 69665c0..f22e4aa 100644
--- a/clutter/clutter-types.h
+++ b/clutter/clutter-types.h
@@ -59,6 +59,7 @@ typedef struct _ClutterContainer        ClutterContainer; /* dummy */
 typedef struct _ClutterChildMeta        ClutterChildMeta;
 typedef struct _ClutterLayoutMeta       ClutterLayoutMeta;
 typedef struct _ClutterActorMeta        ClutterActorMeta;
+typedef struct _ClutterLayoutManager    ClutterLayoutManager;
 
 typedef struct _ClutterAlpha            ClutterAlpha;
 typedef struct _ClutterAnimatable       ClutterAnimatable; /* dummy */



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