534 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			534 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.
 | |
|  *
 | |
|  * 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);
 | |
|   GtkStyleContext *style;
 | |
|   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);
 | |
| 
 | |
|   /* Fix up styling */
 | |
|   style = gtk_widget_get_style_context (row);
 | |
|   gtk_style_context_add_class (style, "sidebar-item");
 | |
| 
 | |
|   /* 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);
 | |
| 
 | |
|   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_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: (transfer full): 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);
 | |
| }
 | 
