541 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			541 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 * Copyright (c) 2014 Intel Corporation
 | 
						|
 *
 | 
						|
 * This program is free software; you can redistribute it and/or modify
 | 
						|
 * it under the terms of the GNU Lesser General Public License as published by
 | 
						|
 * the Free Software Foundation; either version 2 of the License, or (at your
 | 
						|
 * option) any later version.
 | 
						|
 *
 | 
						|
 * This program is distributed in the hope that it will be useful, but
 | 
						|
 * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
 | 
						|
 * or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU Lesser General Public
 | 
						|
 * License for more details.
 | 
						|
 *
 | 
						|
 * You should have received a copy of the GNU Lesser General Public License
 | 
						|
 * along with this program; if not, write to the Free Software Foundation,
 | 
						|
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
 | 
						|
 *
 | 
						|
 * Author:
 | 
						|
 *      Ikey Doherty <michael.i.doherty@intel.com>
 | 
						|
 */
 | 
						|
 | 
						|
#include "config.h"
 | 
						|
 | 
						|
#include "gtkstacksidebar.h"
 | 
						|
 | 
						|
#include "gtklabel.h"
 | 
						|
#include "gtklistbox.h"
 | 
						|
#include "gtkscrolledwindow.h"
 | 
						|
#include "gtkseparator.h"
 | 
						|
#include "gtkstylecontext.h"
 | 
						|
#include "gtkprivate.h"
 | 
						|
#include "gtkintl.h"
 | 
						|
 | 
						|
/**
 | 
						|
 * SECTION:gtkstacksidebar
 | 
						|
 * @Title: GtkStackSidebar
 | 
						|
 * @Short_description: An automatic sidebar widget
 | 
						|
 *
 | 
						|
 * A GtkStackSidebar enables you to quickly and easily provide a
 | 
						|
 * consistent "sidebar" object for your user interface.
 | 
						|
 *
 | 
						|
 * In order to use a GtkStackSidebar, you simply use a GtkStack to
 | 
						|
 * organize your UI flow, and add the sidebar to your sidebar area. You
 | 
						|
 * can use gtk_stack_sidebar_set_stack() to connect the #GtkStackSidebar
 | 
						|
 * to the #GtkStack.
 | 
						|
 *
 | 
						|
 * # CSS nodes
 | 
						|
 *
 | 
						|
 * GtkStackSidebar has a single CSS node with name stacksidebar and
 | 
						|
 * style class .sidebar.
 | 
						|
 *
 | 
						|
 * When circumstances require it, GtkStackSidebar adds the
 | 
						|
 * .needs-attention style class to the widgets representing the stack
 | 
						|
 * pages.
 | 
						|
 *
 | 
						|
 * Since: 3.16
 | 
						|
 */
 | 
						|
 | 
						|
struct _GtkStackSidebarPrivate
 | 
						|
{
 | 
						|
  GtkListBox *list;
 | 
						|
  GtkStack *stack;
 | 
						|
  GHashTable *rows;
 | 
						|
  gboolean in_child_changed;
 | 
						|
};
 | 
						|
 | 
						|
G_DEFINE_TYPE_WITH_PRIVATE (GtkStackSidebar, gtk_stack_sidebar, GTK_TYPE_BIN)
 | 
						|
 | 
						|
enum
 | 
						|
{
 | 
						|
  PROP_0,
 | 
						|
  PROP_STACK,
 | 
						|
  N_PROPERTIES
 | 
						|
};
 | 
						|
static GParamSpec *obj_properties[N_PROPERTIES] = { NULL, };
 | 
						|
 | 
						|
static void
 | 
						|
gtk_stack_sidebar_set_property (GObject    *object,
 | 
						|
                                guint       prop_id,
 | 
						|
                                const       GValue *value,
 | 
						|
                                GParamSpec *pspec)
 | 
						|
{
 | 
						|
  switch (prop_id)
 | 
						|
    {
 | 
						|
    case PROP_STACK:
 | 
						|
      gtk_stack_sidebar_set_stack (GTK_STACK_SIDEBAR (object), g_value_get_object (value));
 | 
						|
      break;
 | 
						|
 | 
						|
    default:
 | 
						|
      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
 | 
						|
      break;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
gtk_stack_sidebar_get_property (GObject    *object,
 | 
						|
                                guint       prop_id,
 | 
						|
                                GValue     *value,
 | 
						|
                                GParamSpec *pspec)
 | 
						|
{
 | 
						|
  GtkStackSidebarPrivate *priv = gtk_stack_sidebar_get_instance_private (GTK_STACK_SIDEBAR (object));
 | 
						|
 | 
						|
  switch (prop_id)
 | 
						|
    {
 | 
						|
    case PROP_STACK:
 | 
						|
      g_value_set_object (value, priv->stack);
 | 
						|
      break;
 | 
						|
 | 
						|
    default:
 | 
						|
      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
 | 
						|
      break;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
update_header (GtkListBoxRow *row,
 | 
						|
               GtkListBoxRow *before,
 | 
						|
               gpointer       userdata)
 | 
						|
{
 | 
						|
  GtkWidget *ret = NULL;
 | 
						|
 | 
						|
  if (before && !gtk_list_box_row_get_header (row))
 | 
						|
    {
 | 
						|
      ret = gtk_separator_new (GTK_ORIENTATION_HORIZONTAL);
 | 
						|
      gtk_list_box_row_set_header (row, ret);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static gint
 | 
						|
sort_list (GtkListBoxRow *row1,
 | 
						|
           GtkListBoxRow *row2,
 | 
						|
           gpointer       userdata)
 | 
						|
{
 | 
						|
  GtkStackSidebar *sidebar = GTK_STACK_SIDEBAR (userdata);
 | 
						|
  GtkStackSidebarPrivate *priv = gtk_stack_sidebar_get_instance_private (sidebar);
 | 
						|
  GtkWidget *item;
 | 
						|
  GtkWidget *widget;
 | 
						|
  gint left = 0; gint right = 0;
 | 
						|
 | 
						|
 | 
						|
  if (row1)
 | 
						|
    {
 | 
						|
      item = gtk_bin_get_child (GTK_BIN (row1));
 | 
						|
      widget = g_object_get_data (G_OBJECT (item), "stack-child");
 | 
						|
      gtk_container_child_get (GTK_CONTAINER (priv->stack), widget,
 | 
						|
                               "position", &left,
 | 
						|
                               NULL);
 | 
						|
    }
 | 
						|
 | 
						|
  if (row2)
 | 
						|
    {
 | 
						|
      item = gtk_bin_get_child (GTK_BIN (row2));
 | 
						|
      widget = g_object_get_data (G_OBJECT (item), "stack-child");
 | 
						|
      gtk_container_child_get (GTK_CONTAINER (priv->stack), widget,
 | 
						|
                               "position", &right,
 | 
						|
                               NULL);
 | 
						|
    }
 | 
						|
 | 
						|
  if (left < right)
 | 
						|
    return  -1;
 | 
						|
 | 
						|
  if (left == right)
 | 
						|
    return 0;
 | 
						|
 | 
						|
  return 1;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
gtk_stack_sidebar_row_selected (GtkListBox    *box,
 | 
						|
                                GtkListBoxRow *row,
 | 
						|
                                gpointer       userdata)
 | 
						|
{
 | 
						|
  GtkStackSidebar *sidebar = GTK_STACK_SIDEBAR (userdata);
 | 
						|
  GtkStackSidebarPrivate *priv = gtk_stack_sidebar_get_instance_private (sidebar);
 | 
						|
  GtkWidget *item;
 | 
						|
  GtkWidget *widget;
 | 
						|
 | 
						|
  if (priv->in_child_changed)
 | 
						|
    return;
 | 
						|
 | 
						|
  if (!row)
 | 
						|
    return;
 | 
						|
 | 
						|
  item = gtk_bin_get_child (GTK_BIN (row));
 | 
						|
  widget = g_object_get_data (G_OBJECT (item), "stack-child");
 | 
						|
  gtk_stack_set_visible_child (priv->stack, widget);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
gtk_stack_sidebar_init (GtkStackSidebar *sidebar)
 | 
						|
{
 | 
						|
  GtkStyleContext *style;
 | 
						|
  GtkStackSidebarPrivate *priv;
 | 
						|
  GtkWidget *sw;
 | 
						|
 | 
						|
  priv = gtk_stack_sidebar_get_instance_private (sidebar);
 | 
						|
 | 
						|
  sw = gtk_scrolled_window_new (NULL, NULL);
 | 
						|
  gtk_widget_show (sw);
 | 
						|
  gtk_widget_set_no_show_all (sw, TRUE);
 | 
						|
  gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (sw),
 | 
						|
                                  GTK_POLICY_NEVER,
 | 
						|
                                  GTK_POLICY_AUTOMATIC);
 | 
						|
 | 
						|
  gtk_container_add (GTK_CONTAINER (sidebar), sw);
 | 
						|
 | 
						|
  priv->list = GTK_LIST_BOX (gtk_list_box_new ());
 | 
						|
  gtk_widget_show (GTK_WIDGET (priv->list));
 | 
						|
 | 
						|
  gtk_container_add (GTK_CONTAINER (sw), GTK_WIDGET (priv->list));
 | 
						|
 | 
						|
  gtk_list_box_set_header_func (priv->list, update_header, sidebar, NULL);
 | 
						|
  gtk_list_box_set_sort_func (priv->list, sort_list, sidebar, NULL);
 | 
						|
 | 
						|
  g_signal_connect (priv->list, "row-selected",
 | 
						|
                    G_CALLBACK (gtk_stack_sidebar_row_selected), sidebar);
 | 
						|
 | 
						|
  style = gtk_widget_get_style_context (GTK_WIDGET (sidebar));
 | 
						|
  gtk_style_context_add_class (style, "sidebar");
 | 
						|
 | 
						|
  priv->rows = g_hash_table_new (NULL, NULL);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
update_row (GtkStackSidebar *sidebar,
 | 
						|
            GtkWidget       *widget,
 | 
						|
            GtkWidget       *row)
 | 
						|
{
 | 
						|
  GtkStackSidebarPrivate *priv = gtk_stack_sidebar_get_instance_private (sidebar);
 | 
						|
  GtkWidget *item;
 | 
						|
  gchar *title;
 | 
						|
  gboolean needs_attention;
 | 
						|
  GtkStyleContext *context;
 | 
						|
 | 
						|
  gtk_container_child_get (GTK_CONTAINER (priv->stack), widget,
 | 
						|
                           "title", &title,
 | 
						|
                           "needs-attention", &needs_attention,
 | 
						|
                           NULL);
 | 
						|
 | 
						|
  item = gtk_bin_get_child (GTK_BIN (row));
 | 
						|
  gtk_label_set_text (GTK_LABEL (item), title);
 | 
						|
 | 
						|
  gtk_widget_set_visible (row, gtk_widget_get_visible (widget) && title != NULL);
 | 
						|
 | 
						|
  context = gtk_widget_get_style_context (row);
 | 
						|
  if (needs_attention)
 | 
						|
     gtk_style_context_add_class (context, GTK_STYLE_CLASS_NEEDS_ATTENTION);
 | 
						|
  else
 | 
						|
    gtk_style_context_remove_class (context, GTK_STYLE_CLASS_NEEDS_ATTENTION);
 | 
						|
 | 
						|
  g_free (title);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
on_position_updated (GtkWidget       *widget,
 | 
						|
                     GParamSpec      *pspec,
 | 
						|
                     GtkStackSidebar *sidebar)
 | 
						|
{
 | 
						|
  GtkStackSidebarPrivate *priv = gtk_stack_sidebar_get_instance_private (sidebar);
 | 
						|
 | 
						|
  gtk_list_box_invalidate_sort (priv->list);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
on_child_updated (GtkWidget       *widget,
 | 
						|
                  GParamSpec      *pspec,
 | 
						|
                  GtkStackSidebar *sidebar)
 | 
						|
{
 | 
						|
  GtkStackSidebarPrivate *priv = gtk_stack_sidebar_get_instance_private (sidebar);
 | 
						|
  GtkWidget *row;
 | 
						|
 | 
						|
  row = g_hash_table_lookup (priv->rows, widget);
 | 
						|
  update_row (sidebar, widget, row);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
add_child (GtkWidget       *widget,
 | 
						|
           GtkStackSidebar *sidebar)
 | 
						|
{
 | 
						|
  GtkStackSidebarPrivate *priv = gtk_stack_sidebar_get_instance_private (sidebar);
 | 
						|
  GtkWidget *item;
 | 
						|
  GtkWidget *row;
 | 
						|
 | 
						|
  /* Check we don't actually already know about this widget */
 | 
						|
  if (g_hash_table_lookup (priv->rows, widget))
 | 
						|
    return;
 | 
						|
 | 
						|
  /* Make a pretty item when we add kids */
 | 
						|
  item = gtk_label_new ("");
 | 
						|
  gtk_widget_set_halign (item, GTK_ALIGN_START);
 | 
						|
  gtk_widget_set_valign (item, GTK_ALIGN_CENTER);
 | 
						|
  row = gtk_list_box_row_new ();
 | 
						|
  gtk_container_add (GTK_CONTAINER (row), item);
 | 
						|
  gtk_widget_show (item);
 | 
						|
 | 
						|
  update_row (sidebar, widget, row);
 | 
						|
 | 
						|
  /* Hook up for events */
 | 
						|
  g_signal_connect (widget, "child-notify::title",
 | 
						|
                    G_CALLBACK (on_child_updated), sidebar);
 | 
						|
  g_signal_connect (widget, "child-notify::needs-attention",
 | 
						|
                    G_CALLBACK (on_child_updated), sidebar);
 | 
						|
  g_signal_connect (widget, "notify::visible",
 | 
						|
                    G_CALLBACK (on_child_updated), sidebar);
 | 
						|
  g_signal_connect (widget, "child-notify::position",
 | 
						|
                    G_CALLBACK (on_position_updated), sidebar);
 | 
						|
 | 
						|
  g_object_set_data (G_OBJECT (item), "stack-child", widget);
 | 
						|
  g_hash_table_insert (priv->rows, widget, row);
 | 
						|
  gtk_container_add (GTK_CONTAINER (priv->list), row);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
remove_child (GtkWidget       *widget,
 | 
						|
              GtkStackSidebar *sidebar)
 | 
						|
{
 | 
						|
  GtkStackSidebarPrivate *priv = gtk_stack_sidebar_get_instance_private (sidebar);
 | 
						|
  GtkWidget *row;
 | 
						|
 | 
						|
  row = g_hash_table_lookup (priv->rows, widget);
 | 
						|
  if (!row)
 | 
						|
    return;
 | 
						|
 | 
						|
  g_signal_handlers_disconnect_by_func (widget, on_child_updated, sidebar);
 | 
						|
  g_signal_handlers_disconnect_by_func (widget, on_position_updated, sidebar);
 | 
						|
 | 
						|
  gtk_container_remove (GTK_CONTAINER (priv->list), row);
 | 
						|
  g_hash_table_remove (priv->rows, widget);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
populate_sidebar (GtkStackSidebar *sidebar)
 | 
						|
{
 | 
						|
  GtkStackSidebarPrivate *priv = gtk_stack_sidebar_get_instance_private (sidebar);
 | 
						|
  GtkWidget *widget, *row;
 | 
						|
 | 
						|
  gtk_container_foreach (GTK_CONTAINER (priv->stack), (GtkCallback)add_child, sidebar);
 | 
						|
 | 
						|
  widget = gtk_stack_get_visible_child (priv->stack);
 | 
						|
  if (widget)
 | 
						|
    {
 | 
						|
      row = g_hash_table_lookup (priv->rows, widget);
 | 
						|
      gtk_list_box_select_row (priv->list, GTK_LIST_BOX_ROW (row));
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
clear_sidebar (GtkStackSidebar *sidebar)
 | 
						|
{
 | 
						|
  GtkStackSidebarPrivate *priv = gtk_stack_sidebar_get_instance_private (sidebar);
 | 
						|
 | 
						|
  gtk_container_foreach (GTK_CONTAINER (priv->stack), (GtkCallback)remove_child, sidebar);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
on_child_changed (GtkWidget       *widget,
 | 
						|
                  GParamSpec      *pspec,
 | 
						|
                  GtkStackSidebar *sidebar)
 | 
						|
{
 | 
						|
  GtkStackSidebarPrivate *priv = gtk_stack_sidebar_get_instance_private (sidebar);
 | 
						|
  GtkWidget *child;
 | 
						|
  GtkWidget *row;
 | 
						|
 | 
						|
  child = gtk_stack_get_visible_child (GTK_STACK (widget));
 | 
						|
  row = g_hash_table_lookup (priv->rows, child);
 | 
						|
  if (row != NULL)
 | 
						|
    {
 | 
						|
      priv->in_child_changed = TRUE;
 | 
						|
      gtk_list_box_select_row (priv->list, GTK_LIST_BOX_ROW (row));
 | 
						|
      priv->in_child_changed = FALSE;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
on_stack_child_added (GtkContainer    *container,
 | 
						|
                      GtkWidget       *widget,
 | 
						|
                      GtkStackSidebar *sidebar)
 | 
						|
{
 | 
						|
  add_child (widget, sidebar);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
on_stack_child_removed (GtkContainer    *container,
 | 
						|
                        GtkWidget       *widget,
 | 
						|
                        GtkStackSidebar *sidebar)
 | 
						|
{
 | 
						|
  remove_child (widget, sidebar);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
disconnect_stack_signals (GtkStackSidebar *sidebar)
 | 
						|
{
 | 
						|
  GtkStackSidebarPrivate *priv = gtk_stack_sidebar_get_instance_private (sidebar);
 | 
						|
 | 
						|
  g_signal_handlers_disconnect_by_func (priv->stack, on_stack_child_added, sidebar);
 | 
						|
  g_signal_handlers_disconnect_by_func (priv->stack, on_stack_child_removed, sidebar);
 | 
						|
  g_signal_handlers_disconnect_by_func (priv->stack, on_child_changed, sidebar);
 | 
						|
  g_signal_handlers_disconnect_by_func (priv->stack, disconnect_stack_signals, sidebar);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
connect_stack_signals (GtkStackSidebar *sidebar)
 | 
						|
{
 | 
						|
  GtkStackSidebarPrivate *priv = gtk_stack_sidebar_get_instance_private (sidebar);
 | 
						|
 | 
						|
  g_signal_connect_after (priv->stack, "add",
 | 
						|
                          G_CALLBACK (on_stack_child_added), sidebar);
 | 
						|
  g_signal_connect_after (priv->stack, "remove",
 | 
						|
                          G_CALLBACK (on_stack_child_removed), sidebar);
 | 
						|
  g_signal_connect (priv->stack, "notify::visible-child",
 | 
						|
                    G_CALLBACK (on_child_changed), sidebar);
 | 
						|
  g_signal_connect_swapped (priv->stack, "destroy",
 | 
						|
                            G_CALLBACK (disconnect_stack_signals), sidebar);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
gtk_stack_sidebar_dispose (GObject *object)
 | 
						|
{
 | 
						|
  GtkStackSidebar *sidebar = GTK_STACK_SIDEBAR (object);
 | 
						|
 | 
						|
  gtk_stack_sidebar_set_stack (sidebar, NULL);
 | 
						|
 | 
						|
  G_OBJECT_CLASS (gtk_stack_sidebar_parent_class)->dispose (object);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
gtk_stack_sidebar_finalize (GObject *object)
 | 
						|
{
 | 
						|
  GtkStackSidebar *sidebar = GTK_STACK_SIDEBAR (object);
 | 
						|
  GtkStackSidebarPrivate *priv = gtk_stack_sidebar_get_instance_private (sidebar);
 | 
						|
 | 
						|
  g_hash_table_destroy (priv->rows);
 | 
						|
 | 
						|
  G_OBJECT_CLASS (gtk_stack_sidebar_parent_class)->finalize (object);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
gtk_stack_sidebar_class_init (GtkStackSidebarClass *klass)
 | 
						|
{
 | 
						|
  GObjectClass *object_class = G_OBJECT_CLASS (klass);
 | 
						|
  GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
 | 
						|
 | 
						|
  object_class->dispose = gtk_stack_sidebar_dispose;
 | 
						|
  object_class->finalize = gtk_stack_sidebar_finalize;
 | 
						|
  object_class->set_property = gtk_stack_sidebar_set_property;
 | 
						|
  object_class->get_property = gtk_stack_sidebar_get_property;
 | 
						|
 | 
						|
  obj_properties[PROP_STACK] =
 | 
						|
      g_param_spec_object (I_("stack"), P_("Stack"),
 | 
						|
                           P_("Associated stack for this GtkStackSidebar"),
 | 
						|
                           GTK_TYPE_STACK,
 | 
						|
                           G_PARAM_READWRITE|G_PARAM_STATIC_STRINGS|G_PARAM_EXPLICIT_NOTIFY);
 | 
						|
 | 
						|
  g_object_class_install_properties (object_class, N_PROPERTIES, obj_properties);
 | 
						|
 | 
						|
  gtk_widget_class_set_css_name (widget_class, "stacksidebar");
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * gtk_stack_sidebar_new:
 | 
						|
 *
 | 
						|
 * Creates a new sidebar.
 | 
						|
 *
 | 
						|
 * Returns: the new #GtkStackSidebar
 | 
						|
 *
 | 
						|
 * Since: 3.16
 | 
						|
 */
 | 
						|
GtkWidget *
 | 
						|
gtk_stack_sidebar_new (void)
 | 
						|
{
 | 
						|
  return GTK_WIDGET (g_object_new (GTK_TYPE_STACK_SIDEBAR, NULL));
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * gtk_stack_sidebar_set_stack:
 | 
						|
 * @sidebar: a #GtkStackSidebar
 | 
						|
 * @stack: a #GtkStack
 | 
						|
 *
 | 
						|
 * Set the #GtkStack associated with this #GtkStackSidebar.
 | 
						|
 *
 | 
						|
 * The sidebar widget will automatically update according to the order
 | 
						|
 * (packing) and items within the given #GtkStack.
 | 
						|
 *
 | 
						|
 * Since: 3.16
 | 
						|
 */
 | 
						|
void
 | 
						|
gtk_stack_sidebar_set_stack (GtkStackSidebar *sidebar,
 | 
						|
                             GtkStack        *stack)
 | 
						|
{
 | 
						|
  GtkStackSidebarPrivate *priv;
 | 
						|
 | 
						|
  g_return_if_fail (GTK_IS_STACK_SIDEBAR (sidebar));
 | 
						|
  g_return_if_fail (GTK_IS_STACK (stack) || stack == NULL);
 | 
						|
 | 
						|
  priv = gtk_stack_sidebar_get_instance_private (sidebar);
 | 
						|
 | 
						|
  if (priv->stack == stack)
 | 
						|
    return;
 | 
						|
 | 
						|
  if (priv->stack)
 | 
						|
    {
 | 
						|
      disconnect_stack_signals (sidebar);
 | 
						|
      clear_sidebar (sidebar);
 | 
						|
      g_clear_object (&priv->stack);
 | 
						|
    }
 | 
						|
  if (stack)
 | 
						|
    {
 | 
						|
      priv->stack = g_object_ref (stack);
 | 
						|
      populate_sidebar (sidebar);
 | 
						|
      connect_stack_signals (sidebar);
 | 
						|
    }
 | 
						|
 | 
						|
  gtk_widget_queue_resize (GTK_WIDGET (sidebar));
 | 
						|
 | 
						|
  g_object_notify (G_OBJECT (sidebar), "stack");
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * gtk_stack_sidebar_get_stack:
 | 
						|
 * @sidebar: a #GtkStackSidebar
 | 
						|
 *
 | 
						|
 * Retrieves the stack.
 | 
						|
 * See gtk_stack_sidebar_set_stack().
 | 
						|
 *
 | 
						|
 * Returns: (nullable) (transfer none): the associated #GtkStack or
 | 
						|
 *     %NULL if none has been set explicitly
 | 
						|
 *
 | 
						|
 * Since: 3.16
 | 
						|
 */
 | 
						|
GtkStack *
 | 
						|
gtk_stack_sidebar_get_stack (GtkStackSidebar *sidebar)
 | 
						|
{
 | 
						|
  GtkStackSidebarPrivate *priv;
 | 
						|
 | 
						|
  g_return_val_if_fail (GTK_IS_STACK_SIDEBAR (sidebar), NULL);
 | 
						|
 | 
						|
  priv = gtk_stack_sidebar_get_instance_private (sidebar);
 | 
						|
 | 
						|
  return GTK_STACK (priv->stack);
 | 
						|
}
 |