[libadwaita/wip/exalm/view-switcher-cleanup: 10/12] view-switcher: Remove custom layout




commit 864ef720c4740a18585a0d7a1f9ffd10c75bd607
Author: Alexander Mikhaylenko <alexm gnome org>
Date:   Tue Aug 31 16:12:40 2021 +0500

    view-switcher: Remove custom layout
    
    Since there's no auto policy anymore, the only thing we need to be custom
    is a 100px natural width limit for view switcher buttons. That can be done
    from within the buttons themselves.

 src/adw-view-switcher-button-private.h |   6 --
 src/adw-view-switcher-button.c         |  69 +++++++-------
 src/adw-view-switcher.c                | 158 ++++-----------------------------
 3 files changed, 52 insertions(+), 181 deletions(-)
---
diff --git a/src/adw-view-switcher-button-private.h b/src/adw-view-switcher-button-private.h
index 9d2f2168..d80f43d7 100644
--- a/src/adw-view-switcher-button-private.h
+++ b/src/adw-view-switcher-button-private.h
@@ -41,10 +41,4 @@ const char *adw_view_switcher_button_get_label (AdwViewSwitcherButton *self);
 void        adw_view_switcher_button_set_label (AdwViewSwitcherButton *self,
                                                 const char            *label);
 
-void adw_view_switcher_button_get_size (AdwViewSwitcherButton *self,
-                                        int                   *h_min_width,
-                                        int                   *h_nat_width,
-                                        int                   *v_min_width,
-                                        int                   *v_nat_width);
-
 G_END_DECLS
diff --git a/src/adw-view-switcher-button.c b/src/adw-view-switcher-button.c
index a4736b35..1ba247b2 100644
--- a/src/adw-view-switcher-button.c
+++ b/src/adw-view-switcher-button.c
@@ -25,6 +25,8 @@ enum {
   LAST_PROP = PROP_BADGE_NUMBER + 1,
 };
 
+#define MIN_NAT_BUTTON_WIDTH 100
+
 struct _AdwViewSwitcherButton
 {
   GtkToggleButton parent_instance;
@@ -216,6 +218,35 @@ adw_view_switcher_button_finalize (GObject *object)
   G_OBJECT_CLASS (adw_view_switcher_button_parent_class)->finalize (object);
 }
 
+static void
+adw_view_switcher_button_measure (GtkWidget      *widget,
+                                  GtkOrientation  orientation,
+                                  int             for_size,
+                                  int            *minimum,
+                                  int            *natural,
+                                  int            *minimum_baseline,
+                                  int            *natural_baseline)
+{
+  AdwViewSwitcherButton *self = ADW_VIEW_SWITCHER_BUTTON (widget);
+
+  gtk_widget_measure (GTK_WIDGET (self->stack), orientation, for_size,
+                      minimum, natural, minimum_baseline, natural_baseline);
+
+  if (orientation == GTK_ORIENTATION_HORIZONTAL)
+    *natural = MAX (*natural, MIN_NAT_BUTTON_WIDTH);
+}
+
+static void
+adw_view_switcher_button_size_allocate (GtkWidget *widget,
+                                        int        width,
+                                        int        height,
+                                        int        baseline)
+{
+  AdwViewSwitcherButton *self = ADW_VIEW_SWITCHER_BUTTON (widget);
+
+  gtk_widget_allocate (GTK_WIDGET (self->stack), width, height, baseline, NULL);
+}
+
 static void
 adw_view_switcher_button_class_init (AdwViewSwitcherButtonClass *klass)
 {
@@ -227,6 +258,9 @@ adw_view_switcher_button_class_init (AdwViewSwitcherButtonClass *klass)
   object_class->dispose = adw_view_switcher_button_dispose;
   object_class->finalize = adw_view_switcher_button_finalize;
 
+  widget_class->measure = adw_view_switcher_button_measure;
+  widget_class->size_allocate = adw_view_switcher_button_size_allocate;
+
   g_object_class_override_property (object_class,
                                     PROP_LABEL,
                                     "label");
@@ -316,6 +350,8 @@ adw_view_switcher_button_class_init (AdwViewSwitcherButtonClass *klass)
 static void
 adw_view_switcher_button_init (AdwViewSwitcherButton *self)
 {
+  gtk_widget_set_layout_manager (GTK_WIDGET (self), NULL);
+
   self->icon_name = g_strdup ("image-missing");
 
   g_type_ensure (ADW_TYPE_INDICATOR_BIN);
@@ -520,36 +556,3 @@ adw_view_switcher_button_set_label (AdwViewSwitcherButton *self,
 
   g_object_notify (G_OBJECT (self), "label");
 }
-
-/**
- * adw_view_switcher_button_get_size:
- * @self: a `AdwViewSwitcherButton`
- * @h_min_width: (out) (nullable): the minimum width when horizontal
- * @h_nat_width: (out) (nullable): the natural width when horizontal
- * @v_min_width: (out) (nullable): the minimum width when vertical
- * @v_nat_width: (out) (nullable): the natural width when vertical
- *
- * Measure the size requests in both horizontal and vertical modes.
- *
- * Since: 1.0
- */
-void
-adw_view_switcher_button_get_size (AdwViewSwitcherButton *self,
-                                   int                   *h_min_width,
-                                   int                   *h_nat_width,
-                                   int                   *v_min_width,
-                                   int                   *v_nat_width)
-{
-  /* gtk_widget_get_preferred_width() doesn't accept both its out parameters to
-   * be NULL, so we must have guards.
-   */
-  if (h_min_width != NULL || h_nat_width != NULL)
-    gtk_widget_measure (GTK_WIDGET (self->horizontal_box),
-                        GTK_ORIENTATION_HORIZONTAL, -1,
-                        h_min_width, h_nat_width, NULL, NULL);
-
-  if (v_min_width != NULL || v_nat_width != NULL)
-    gtk_widget_measure (GTK_WIDGET (self->vertical_box),
-                        GTK_ORIENTATION_HORIZONTAL, -1,
-                        v_min_width, v_nat_width, NULL, NULL);
-}
diff --git a/src/adw-view-switcher.c b/src/adw-view-switcher.c
index 3e3b73fa..8170a328 100644
--- a/src/adw-view-switcher.c
+++ b/src/adw-view-switcher.c
@@ -48,8 +48,6 @@
  * Describes the adaptive modes of [class@Adw.ViewSwitcher].
  */
 
-#define MIN_NAT_BUTTON_WIDTH 100
-
 enum {
   PROP_0,
   PROP_POLICY,
@@ -64,7 +62,6 @@ struct _AdwViewSwitcher
   AdwViewStack *stack;
   GtkSelectionModel *pages;
   GHashTable *buttons;
-  GtkBox *box;
 
   AdwViewSwitcherPolicy policy;
 };
@@ -146,7 +143,7 @@ add_child (AdwViewSwitcher *self,
   page = g_list_model_get_item (G_LIST_MODEL (self->pages), position);
   update_button (self, page, GTK_WIDGET (button));
 
-  gtk_box_append (self->box, GTK_WIDGET (button));
+  gtk_widget_set_parent (GTK_WIDGET (button), GTK_WIDGET (self));
 
   g_object_set_data (G_OBJECT (button), "child-index", GUINT_TO_POINTER (position));
   selected = gtk_selection_model_is_selected (self->pages, position);
@@ -156,6 +153,9 @@ add_child (AdwViewSwitcher *self,
                                GTK_ACCESSIBLE_STATE_SELECTED, selected,
                                -1);
 
+  gtk_orientable_set_orientation (GTK_ORIENTABLE (button),
+                                  self->policy == ADW_VIEW_SWITCHER_POLICY_WIDE ? GTK_ORIENTATION_HORIZONTAL 
: GTK_ORIENTATION_VERTICAL);
+
   g_signal_connect (button, "notify::active", G_CALLBACK (on_button_toggled), self);
   g_signal_connect (page, "notify", G_CALLBACK (on_page_updated), self);
 
@@ -183,7 +183,7 @@ clear_switcher (AdwViewSwitcher *self)
 
   g_hash_table_iter_init (&iter, self->buttons);
   while (g_hash_table_iter_next (&iter, (gpointer *) &page, (gpointer *) &button)) {
-    gtk_box_remove (self->box, button);
+    gtk_widget_unparent (button);
     g_signal_handlers_disconnect_by_func (page, on_page_updated, self);
     g_hash_table_iter_remove (&iter);
   }
@@ -312,7 +312,6 @@ adw_view_switcher_dispose (GObject *object)
   AdwViewSwitcher *self = ADW_VIEW_SWITCHER (object);
 
   unset_stack (self);
-  g_clear_pointer ((GtkWidget **) &self->box, gtk_widget_unparent);
 
   G_OBJECT_CLASS (adw_view_switcher_parent_class)->dispose (object);
 }
@@ -327,136 +326,6 @@ adw_view_switcher_finalize (GObject *object)
   G_OBJECT_CLASS (adw_view_switcher_parent_class)->finalize (object);
 }
 
-static void
-adw_view_switcher_measure (GtkWidget      *widget,
-                           GtkOrientation  orientation,
-                           int             for_size,
-                           int            *minimum,
-                           int            *natural,
-                           int            *minimum_baseline,
-                           int            *natural_baseline)
-{
-  AdwViewSwitcher *self = ADW_VIEW_SWITCHER (widget);
-  GHashTableIter iter;
-  AdwViewStackPage *page;
-  AdwViewSwitcherButton *button;
-  int max_h_min = 0, max_h_nat = 0, max_v_min = 0, max_v_nat = 0;
-  int min = 0, nat = 0;
-  int n_children = 0;
-
-  g_hash_table_iter_init (&iter, self->buttons);
-
-  if (orientation == GTK_ORIENTATION_HORIZONTAL) {
-    while (g_hash_table_iter_next (&iter, (gpointer *) &page, (gpointer *) &button)) {
-      int h_min = 0, h_nat = 0, v_min = 0, v_nat = 0;
-
-      if (!adw_view_stack_page_get_visible (page))
-        continue;
-
-      adw_view_switcher_button_get_size (button, &h_min, &h_nat, &v_min, &v_nat);
-      max_h_min = MAX (h_min, max_h_min);
-      max_h_nat = MAX (h_nat, max_h_nat);
-      max_v_min = MAX (v_min, max_v_min);
-      max_v_nat = MAX (v_nat, max_v_nat);
-
-      n_children++;
-    }
-
-    /* Make the buttons ask at least a minimum arbitrary size for their natural
-     * width. This prevents them from looking terribly narrow in a very wide bar.
-     */
-    max_h_nat = MAX (max_h_nat, MIN_NAT_BUTTON_WIDTH);
-    max_v_nat = MAX (max_v_nat, MIN_NAT_BUTTON_WIDTH);
-
-    switch (self->policy) {
-    case ADW_VIEW_SWITCHER_POLICY_NARROW:
-    default:
-      min = max_v_min * n_children;
-      nat = max_v_nat * n_children;
-      break;
-    case ADW_VIEW_SWITCHER_POLICY_WIDE:
-      min = max_h_min * n_children;
-      nat = max_h_nat * n_children;
-      break;
-    }
-  } else {
-    while (g_hash_table_iter_next (&iter, (gpointer *) &page, (gpointer *) &button)) {
-      int child_min, child_nat;
-
-      if (!adw_view_stack_page_get_visible (page))
-        continue;
-
-      gtk_widget_measure (GTK_WIDGET (button), GTK_ORIENTATION_VERTICAL, -1,
-                          &child_min, &child_nat, NULL, NULL);
-
-      min = MAX (child_min, min);
-      nat = MAX (child_nat, nat);
-    }
-  }
-
-  if (minimum)
-    *minimum = min;
-  if (natural)
-    *natural = nat;
-  if (minimum_baseline)
-    *minimum_baseline = -1;
-  if (natural_baseline)
-    *natural_baseline = -1;
-}
-
-static int
-is_narrow (AdwViewSwitcher *self,
-           int              width)
-{
-  GHashTableIter iter;
-  AdwViewSwitcherButton *button;
-  int max_h_min = 0;
-  int n_children = 0;
-
-  if (self->policy == ADW_VIEW_SWITCHER_POLICY_NARROW)
-    return TRUE;
-
-  if (self->policy == ADW_VIEW_SWITCHER_POLICY_WIDE)
-    return FALSE;
-
-  g_hash_table_iter_init (&iter, self->buttons);
-  while (g_hash_table_iter_next (&iter, NULL, (gpointer *) &button)) {
-    int h_min = 0;
-
-    if (!gtk_widget_get_visible (GTK_WIDGET (button)))
-      continue;
-
-    adw_view_switcher_button_get_size (button, &h_min, NULL, NULL, NULL);
-    max_h_min = MAX (max_h_min, h_min);
-
-    n_children++;
-  }
-
-  return (max_h_min * n_children) > width;
-}
-
-static void
-adw_view_switcher_size_allocate (GtkWidget *widget,
-                                 int        width,
-                                 int        height,
-                                 int        baseline)
-{
-  AdwViewSwitcher *self = ADW_VIEW_SWITCHER (widget);
-  GtkOrientation orientation;
-  GHashTableIter iter;
-  AdwViewSwitcherButton *button;
-
-  orientation = is_narrow (ADW_VIEW_SWITCHER (widget), width) ?
-    GTK_ORIENTATION_VERTICAL :
-    GTK_ORIENTATION_HORIZONTAL;
-
-  g_hash_table_iter_init (&iter, self->buttons);
-  while (g_hash_table_iter_next (&iter, NULL, (gpointer *) &button))
-    gtk_orientable_set_orientation (GTK_ORIENTABLE (button), orientation);
-
-  gtk_widget_allocate (GTK_WIDGET (self->box), width, height, baseline, NULL);
-}
-
 static void
 adw_view_switcher_class_init (AdwViewSwitcherClass *klass)
 {
@@ -468,9 +337,6 @@ adw_view_switcher_class_init (AdwViewSwitcherClass *klass)
   object_class->dispose = adw_view_switcher_dispose;
   object_class->finalize = adw_view_switcher_finalize;
 
-  widget_class->size_allocate = adw_view_switcher_size_allocate;
-  widget_class->measure = adw_view_switcher_measure;
-
   /**
    * AdwViewSwitcher:policy: (attributes org.gtk.Property.get=adw_view_switcher_get_policy 
org.gtk.Property.set=adw_view_switcher_set_policy)
    *
@@ -503,16 +369,16 @@ adw_view_switcher_class_init (AdwViewSwitcherClass *klass)
   g_object_class_install_properties (object_class, LAST_PROP, props);
 
   gtk_widget_class_set_css_name (widget_class, "viewswitcher");
+  gtk_widget_class_set_layout_manager_type (widget_class, GTK_TYPE_BOX_LAYOUT);
   gtk_widget_class_set_accessible_role (widget_class, GTK_ACCESSIBLE_ROLE_TAB_LIST);
 }
 
 static void
 adw_view_switcher_init (AdwViewSwitcher *self)
 {
-  self->box = GTK_BOX (gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0));
-  gtk_box_set_homogeneous (self->box, TRUE);
+  GtkLayoutManager *layout = gtk_widget_get_layout_manager (GTK_WIDGET (self));
 
-  gtk_widget_set_parent (GTK_WIDGET (self->box), GTK_WIDGET (self));
+  gtk_box_layout_set_homogeneous (GTK_BOX_LAYOUT (layout), TRUE);
 
   self->buttons = g_hash_table_new_full (g_direct_hash, g_direct_equal, g_object_unref, NULL);
 }
@@ -563,6 +429,9 @@ void
 adw_view_switcher_set_policy (AdwViewSwitcher       *self,
                               AdwViewSwitcherPolicy  policy)
 {
+  GHashTableIter iter;
+  GtkWidget *button;
+
   g_return_if_fail (ADW_IS_VIEW_SWITCHER (self));
 
   if (self->policy == policy)
@@ -570,6 +439,11 @@ adw_view_switcher_set_policy (AdwViewSwitcher       *self,
 
   self->policy = policy;
 
+  g_hash_table_iter_init (&iter, self->buttons);
+  while (g_hash_table_iter_next (&iter, NULL, (gpointer *) &button))
+    gtk_orientable_set_orientation (GTK_ORIENTABLE (button),
+                                    self->policy == ADW_VIEW_SWITCHER_POLICY_WIDE ? 
GTK_ORIENTATION_HORIZONTAL : GTK_ORIENTATION_VERTICAL);
+
   g_object_notify_by_pspec (G_OBJECT (self), props[PROP_POLICY]);
 
   gtk_widget_queue_resize (GTK_WIDGET (self));


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