Add GtkSidebar
GtkSidebar behaves internally much like GtkStackSwitcher, providing a vertical sidebar like widget. It is virtually identical in appearance to the widget currently used in GNOME Tweak Tool. This widget is connected to a GtkStack, and builds its own contents as a GtkListBox subclass, using the "title" child property to provide a consistent navigatable widget. Being a subclass of GtkListBox it benefits immediately from strong keyboard navigation, and minimal changes are required for theming. https://bugzilla.gnome.org/show_bug.cgi?id=735293 Signed-off-by: Ikey Doherty <michael.i.doherty@intel.com>
This commit is contained in:
committed by
Matthias Clasen
parent
3b08ba24b3
commit
cdd2651db0
@@ -48,6 +48,7 @@ demos = \
|
||||
rotated_text.c \
|
||||
search_entry.c \
|
||||
search_entry2.c \
|
||||
sidebar.c \
|
||||
sizegroup.c \
|
||||
spinner.c \
|
||||
stack.c \
|
||||
|
||||
@@ -118,6 +118,7 @@
|
||||
<file>search_entry.c</file>
|
||||
<file>search_entry2.c</file>
|
||||
<file>sizegroup.c</file>
|
||||
<file>sidebar.c</file>
|
||||
<file>stack.c</file>
|
||||
<file>spinner.c</file>
|
||||
<file>textview.c</file>
|
||||
|
||||
77
demos/gtk-demo/sidebar.c
Normal file
77
demos/gtk-demo/sidebar.c
Normal file
@@ -0,0 +1,77 @@
|
||||
/* Sidebar
|
||||
*
|
||||
* GtkSidebar provides an automatic sidebar widget to control navigation
|
||||
* of a GtkStack object. This widget automatically updates it content
|
||||
* based on what is presently available in the GtkStack object, and
|
||||
* using the "title" child property to set the display labels.
|
||||
*/
|
||||
|
||||
#include <glib/gi18n.h>
|
||||
#include <gtk/gtk.h>
|
||||
|
||||
static GtkWidget *window = NULL;
|
||||
|
||||
GtkWidget *
|
||||
do_sidebar (GtkWidget *do_widget)
|
||||
{
|
||||
GtkWidget *sidebar;
|
||||
GtkWidget *stack;
|
||||
GtkWidget *box;
|
||||
GtkWidget *widget;
|
||||
GtkWidget *header;
|
||||
const gchar* pages[] = { "Welcome to GTK+", "GtkSidebar Widget", "Automatic navigation", "Consistent appearance", NULL };
|
||||
const gchar *c = NULL;
|
||||
guint i;
|
||||
|
||||
if (!window)
|
||||
{
|
||||
window = gtk_window_new (GTK_WINDOW_TOPLEVEL);
|
||||
gtk_window_set_resizable (GTK_WINDOW (window), TRUE);
|
||||
gtk_widget_set_size_request (window, 500, 350);
|
||||
|
||||
header = gtk_header_bar_new ();
|
||||
gtk_header_bar_set_show_close_button (GTK_HEADER_BAR(header), TRUE);
|
||||
gtk_window_set_titlebar (GTK_WINDOW(window), header);
|
||||
gtk_window_set_title (GTK_WINDOW(window), "Sidebar demo");
|
||||
|
||||
g_signal_connect (window, "destroy",
|
||||
G_CALLBACK (gtk_widget_destroyed), &window);
|
||||
|
||||
box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
|
||||
sidebar = gtk_sidebar_new ();
|
||||
gtk_box_pack_start (GTK_BOX (box), sidebar, FALSE, FALSE, 0);
|
||||
|
||||
stack = gtk_stack_new ();
|
||||
gtk_stack_set_transition_type (GTK_STACK (stack), GTK_STACK_TRANSITION_TYPE_SLIDE_UP_DOWN);
|
||||
gtk_sidebar_set_stack (GTK_SIDEBAR (sidebar), GTK_STACK (stack));
|
||||
|
||||
/* Separator between sidebar and stack */
|
||||
widget = gtk_separator_new (GTK_ORIENTATION_VERTICAL);
|
||||
gtk_box_pack_start (GTK_BOX(box), widget, FALSE, FALSE, 0);
|
||||
|
||||
gtk_box_pack_start (GTK_BOX (box), stack, TRUE, TRUE, 0);
|
||||
|
||||
for (i=0; (c = *(pages+i)) != NULL; i++ )
|
||||
{
|
||||
if (i == 0)
|
||||
{
|
||||
widget = gtk_image_new_from_icon_name ("help-about", GTK_ICON_SIZE_INVALID);
|
||||
gtk_image_set_pixel_size (GTK_IMAGE (widget), 256);
|
||||
} else
|
||||
{
|
||||
widget = gtk_label_new (c);
|
||||
}
|
||||
gtk_stack_add_named (GTK_STACK (stack), widget, c);
|
||||
gtk_container_child_set (GTK_CONTAINER (stack), widget, "title", c, NULL);
|
||||
}
|
||||
|
||||
gtk_container_add (GTK_CONTAINER (window), box);
|
||||
}
|
||||
|
||||
if (!gtk_widget_get_visible (window))
|
||||
gtk_widget_show_all (window);
|
||||
else
|
||||
gtk_widget_destroy (window);
|
||||
|
||||
return window;
|
||||
}
|
||||
@@ -76,6 +76,7 @@
|
||||
<xi:include href="xml/gtkflowbox.xml" />
|
||||
<xi:include href="xml/gtkstack.xml" />
|
||||
<xi:include href="xml/gtkstackswitcher.xml" />
|
||||
<xi:include href="xml/gtksidebar.xml" />
|
||||
<xi:include href="xml/gtkactionbar.xml" />
|
||||
<xi:include href="xml/gtkheaderbar.xml" />
|
||||
<xi:include href="xml/gtkoverlay.xml" />
|
||||
|
||||
@@ -8157,3 +8157,23 @@ GTK_GESTURE_ZOOM_GET_CLASS
|
||||
<SUBSECTION Private>
|
||||
gtk_gesture_zoom_get_type
|
||||
</SECTION>
|
||||
|
||||
|
||||
<SECTION>
|
||||
<FILE>gtksidebar</FILE>
|
||||
GtkSidebar
|
||||
GtkSidebarClass
|
||||
gtk_sidebar_new
|
||||
|
||||
<SUBSECTION Standard>
|
||||
GTK_TYPE_SIDEBAR
|
||||
GTK_SIDEBAR
|
||||
GTK_SIDEBAR_CLASS
|
||||
GTK_IS_SIDEBAR
|
||||
GTK_IS_SIDEBAR_CLASS
|
||||
GTK_SIDEBAR_GET_CLASS
|
||||
|
||||
<SUBSECTION Private>
|
||||
GtkSidebarPrivate
|
||||
gtk_sidebar_get_type
|
||||
</SECTION>
|
||||
|
||||
@@ -170,6 +170,7 @@ gtk_separator_get_type
|
||||
gtk_separator_menu_item_get_type
|
||||
gtk_separator_tool_item_get_type
|
||||
gtk_settings_get_type
|
||||
gtk_sidebar_get_type
|
||||
gtk_size_group_get_type
|
||||
@ENABLE_ON_X11@gtk_socket_get_type
|
||||
gtk_spin_button_get_type
|
||||
|
||||
BIN
docs/reference/gtk/images/sidebar.png
Normal file
BIN
docs/reference/gtk/images/sidebar.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 8.3 KiB |
@@ -141,6 +141,9 @@
|
||||
<link linkend="GtkStackSwitcher">
|
||||
<inlinegraphic fileref="stackswitcher.png" format="PNG"></inlinegraphic>
|
||||
</link>
|
||||
<link linkend="GtkSidebar">
|
||||
<inlinegraphic fileref="sidebar.png" format="PNG"></inlinegraphic>
|
||||
</link>
|
||||
<link linkend="GtkTreeView">
|
||||
<inlinegraphic fileref="list-and-tree.png" format="PNG"></inlinegraphic>
|
||||
</link>
|
||||
|
||||
@@ -431,6 +431,7 @@ gtk_public_h_sources = \
|
||||
gtkseparatortoolitem.h \
|
||||
gtksettings.h \
|
||||
gtkshow.h \
|
||||
gtksidebar.h \
|
||||
gtksizegroup.h \
|
||||
gtksizerequest.h \
|
||||
gtksocket.h \
|
||||
@@ -1027,6 +1028,7 @@ gtk_base_c_sources = \
|
||||
gtksizerequest.c \
|
||||
gtksizerequestcache.c \
|
||||
gtkshow.c \
|
||||
gtksidebar.c \
|
||||
gtkspinbutton.c \
|
||||
gtkspinner.c \
|
||||
gtkstack.c \
|
||||
|
||||
@@ -180,6 +180,7 @@
|
||||
#include <gtk/gtkseparatortoolitem.h>
|
||||
#include <gtk/gtksettings.h>
|
||||
#include <gtk/gtkshow.h>
|
||||
#include <gtk/gtksidebar.h>
|
||||
#include <gtk/gtksizegroup.h>
|
||||
#include <gtk/gtksizerequest.h>
|
||||
#include <gtk/gtkspinbutton.h>
|
||||
|
||||
499
gtk/gtksidebar.c
Normal file
499
gtk/gtksidebar.c
Normal file
@@ -0,0 +1,499 @@
|
||||
/*
|
||||
* 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 "gtksidebar.h"
|
||||
#include "gtklistbox.h"
|
||||
#include "gtkseparator.h"
|
||||
#include "gtkstylecontext.h"
|
||||
#include "gtklabel.h"
|
||||
#include "gtkprivate.h"
|
||||
#include "gtkintl.h"
|
||||
|
||||
/**
|
||||
* SECTION:gtk-sidebar
|
||||
* @short_description: An automatic sidebar widget
|
||||
* @title: GtkSidebar
|
||||
*
|
||||
* A GtkSidebarWindow enables you to quickly and easily provide a consistent
|
||||
* "sidebar" object for your user interface.
|
||||
*
|
||||
* In order to use a GtkSidebar, you simply use a GtkStack to organise
|
||||
* your UI flow, and add the sidebar to your sidebar area. You can use
|
||||
* gtk_sidebar_set_stack() to connect the #GtkSidebar to the #GtkStack.
|
||||
*
|
||||
* Since: 3.16
|
||||
*/
|
||||
|
||||
struct _GtkSidebarPrivate
|
||||
{
|
||||
GtkStack *stack;
|
||||
GHashTable *rows;
|
||||
gboolean in_child_changed;
|
||||
};
|
||||
|
||||
G_DEFINE_TYPE_WITH_PRIVATE (GtkSidebar, gtk_sidebar, GTK_TYPE_LIST_BOX)
|
||||
|
||||
enum
|
||||
{
|
||||
PROP_0,
|
||||
PROP_STACK,
|
||||
N_PROPERTIES
|
||||
};
|
||||
static GParamSpec *obj_properties[N_PROPERTIES] = { NULL, };
|
||||
|
||||
static void
|
||||
gtk_sidebar_set_property (GObject *object,
|
||||
guint prop_id,
|
||||
const GValue *value,
|
||||
GParamSpec *pspec)
|
||||
{
|
||||
switch (prop_id)
|
||||
{
|
||||
case PROP_STACK:
|
||||
gtk_sidebar_set_stack (GTK_SIDEBAR (object), g_value_get_object (value));
|
||||
break;
|
||||
|
||||
default:
|
||||
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
static void
|
||||
gtk_sidebar_get_property (GObject *object,
|
||||
guint prop_id,
|
||||
GValue *value,
|
||||
GParamSpec *pspec)
|
||||
{
|
||||
GtkSidebarPrivate *priv = gtk_sidebar_get_instance_private (GTK_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)
|
||||
{
|
||||
GtkSidebar *sidebar = GTK_SIDEBAR (userdata);
|
||||
GtkSidebarPrivate *priv = gtk_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_sidebar_row_selected (GtkListBox *box,
|
||||
GtkListBoxRow *row)
|
||||
{
|
||||
GtkSidebar *sidebar = GTK_SIDEBAR (box);
|
||||
GtkSidebarPrivate *priv = gtk_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_sidebar_init (GtkSidebar *sidebar)
|
||||
{
|
||||
GtkStyleContext *style;
|
||||
GtkSidebarPrivate *priv;
|
||||
|
||||
priv = gtk_sidebar_get_instance_private (sidebar);
|
||||
|
||||
gtk_list_box_set_header_func (GTK_LIST_BOX (sidebar), update_header, sidebar, NULL);
|
||||
gtk_list_box_set_sort_func (GTK_LIST_BOX (sidebar), sort_list, sidebar, NULL);
|
||||
|
||||
style = gtk_widget_get_style_context (GTK_WIDGET (sidebar));
|
||||
gtk_style_context_add_class (style, "sidebar");
|
||||
|
||||
/* Store this for later use */
|
||||
priv->rows = g_hash_table_new (NULL, NULL);
|
||||
}
|
||||
|
||||
static void
|
||||
update_row (GtkSidebar *sidebar,
|
||||
GtkWidget *widget,
|
||||
GtkWidget *row)
|
||||
{
|
||||
GtkSidebarPrivate *priv = gtk_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,
|
||||
GtkSidebar *sidebar)
|
||||
{
|
||||
gtk_list_box_invalidate_sort (GTK_LIST_BOX (sidebar));
|
||||
}
|
||||
|
||||
static void
|
||||
on_child_updated (GtkWidget *widget,
|
||||
GParamSpec *pspec,
|
||||
GtkSidebar *sidebar)
|
||||
{
|
||||
GtkSidebarPrivate *priv = gtk_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,
|
||||
GtkSidebar *sidebar)
|
||||
{
|
||||
GtkSidebarPrivate *priv = gtk_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 (sidebar), row);
|
||||
}
|
||||
|
||||
static void
|
||||
remove_child (GtkWidget *widget,
|
||||
GtkSidebar *sidebar)
|
||||
{
|
||||
GtkSidebarPrivate *priv = gtk_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 (sidebar), row);
|
||||
g_hash_table_remove (priv->rows, widget);
|
||||
}
|
||||
|
||||
static void
|
||||
populate_sidebar (GtkSidebar *sidebar)
|
||||
{
|
||||
GtkSidebarPrivate *priv = gtk_sidebar_get_instance_private (sidebar);
|
||||
|
||||
gtk_container_foreach (GTK_CONTAINER (priv->stack), (GtkCallback)add_child, sidebar);
|
||||
}
|
||||
|
||||
static void
|
||||
clear_sidebar (GtkSidebar *sidebar)
|
||||
{
|
||||
GtkSidebarPrivate *priv = gtk_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,
|
||||
GtkSidebar *sidebar)
|
||||
{
|
||||
GtkSidebarPrivate *priv = gtk_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 (GTK_LIST_BOX (sidebar), GTK_LIST_BOX_ROW (row));
|
||||
priv->in_child_changed = FALSE;
|
||||
}
|
||||
}
|
||||
|
||||
static void
|
||||
on_stack_child_added (GtkContainer *container,
|
||||
GtkWidget *widget,
|
||||
GtkSidebar *sidebar)
|
||||
{
|
||||
add_child (widget, sidebar);
|
||||
}
|
||||
|
||||
static void
|
||||
on_stack_child_removed (GtkContainer *container,
|
||||
GtkWidget *widget,
|
||||
GtkSidebar *sidebar)
|
||||
{
|
||||
remove_child (widget, sidebar);
|
||||
}
|
||||
|
||||
static void
|
||||
disconnect_stack_signals (GtkSidebar *sidebar)
|
||||
{
|
||||
GtkSidebarPrivate *priv = gtk_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 (GtkSidebar *sidebar)
|
||||
{
|
||||
GtkSidebarPrivate *priv = gtk_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_sidebar_dispose (GObject *object)
|
||||
{
|
||||
GtkSidebar *sidebar = GTK_SIDEBAR (object);
|
||||
|
||||
gtk_sidebar_set_stack (sidebar, NULL);
|
||||
|
||||
G_OBJECT_CLASS (gtk_sidebar_parent_class)->dispose (object);
|
||||
}
|
||||
|
||||
static void
|
||||
gtk_sidebar_finalize (GObject *object)
|
||||
{
|
||||
GtkSidebar *sidebar = GTK_SIDEBAR (object);
|
||||
GtkSidebarPrivate *priv = gtk_sidebar_get_instance_private (sidebar);
|
||||
|
||||
g_hash_table_destroy (priv->rows);
|
||||
|
||||
G_OBJECT_CLASS (gtk_sidebar_parent_class)->finalize (object);
|
||||
}
|
||||
|
||||
static void
|
||||
gtk_sidebar_class_init (GtkSidebarClass *klass)
|
||||
{
|
||||
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
||||
GtkListBoxClass *list_box_class = GTK_LIST_BOX_CLASS (klass);
|
||||
|
||||
object_class->dispose = gtk_sidebar_dispose;
|
||||
object_class->finalize = gtk_sidebar_finalize;
|
||||
object_class->set_property = gtk_sidebar_set_property;
|
||||
object_class->get_property = gtk_sidebar_get_property;
|
||||
|
||||
list_box_class->row_selected = gtk_sidebar_row_selected;
|
||||
|
||||
obj_properties[PROP_STACK] =
|
||||
g_param_spec_pointer ("stack", P_("Stack"),
|
||||
P_("Associated stack for this GtkSidebar"),
|
||||
G_PARAM_READWRITE);
|
||||
|
||||
g_object_class_install_properties (object_class, N_PROPERTIES, obj_properties);
|
||||
}
|
||||
|
||||
/**
|
||||
* gtk_sidebar_new:
|
||||
*
|
||||
* Creates a new sidebar.
|
||||
*
|
||||
* Returns: the new #GtkSidebar
|
||||
*
|
||||
* Since: 3.16
|
||||
*/
|
||||
GtkWidget *
|
||||
gtk_sidebar_new (void)
|
||||
{
|
||||
return GTK_WIDGET (g_object_new (GTK_TYPE_SIDEBAR, NULL));
|
||||
}
|
||||
|
||||
/**
|
||||
* gtk_sidebar_set_stack:
|
||||
* @sidebar: a #GtkSidebar
|
||||
* @stack: a #GtkStack
|
||||
*
|
||||
* Set the #GtkStack associated with this #GtkSidebar.
|
||||
*
|
||||
* The sidebar widget will automatically update according to the order
|
||||
* (packing) and items within the given #GtkStack.
|
||||
*
|
||||
* Since: 3.16
|
||||
*/
|
||||
void
|
||||
gtk_sidebar_set_stack (GtkSidebar *sidebar,
|
||||
GtkStack *stack)
|
||||
{
|
||||
GtkSidebarPrivate *priv;
|
||||
|
||||
g_return_if_fail (GTK_IS_SIDEBAR (sidebar));
|
||||
g_return_if_fail (GTK_IS_STACK (stack) || stack == NULL);
|
||||
|
||||
priv = gtk_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_sidebar_get_stack:
|
||||
* @sidebar: a #GtkSidebar
|
||||
*
|
||||
* Returns: (transfer full): the associated #GtkStack
|
||||
*
|
||||
* Since: 3.16
|
||||
*/
|
||||
GtkStack *
|
||||
gtk_sidebar_get_stack (GtkSidebar *sidebar)
|
||||
{
|
||||
GtkSidebarPrivate *priv;
|
||||
|
||||
g_return_if_fail (GTK_IS_SIDEBAR (sidebar));
|
||||
|
||||
priv = gtk_sidebar_get_instance_private (sidebar);
|
||||
|
||||
return GTK_STACK (priv->stack);
|
||||
}
|
||||
74
gtk/gtksidebar.h
Normal file
74
gtk/gtksidebar.h
Normal file
@@ -0,0 +1,74 @@
|
||||
/*
|
||||
* 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>
|
||||
*/
|
||||
|
||||
#ifndef __GTK_SIDEBAR_H__
|
||||
#define __GTK_SIDEBAR_H__
|
||||
|
||||
#if !defined (__GTK_H_INSIDE__) && !defined (GTK_COMPILATION)
|
||||
#error "Only <gtk/gtk.h> can be included directly."
|
||||
#endif
|
||||
|
||||
#include <gtk/gtkwidget.h>
|
||||
#include <gtk/gtklistbox.h>
|
||||
#include <gtk/gtkstack.h>
|
||||
|
||||
G_BEGIN_DECLS
|
||||
|
||||
#define GTK_TYPE_SIDEBAR (gtk_sidebar_get_type ())
|
||||
#define GTK_SIDEBAR(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), GTK_TYPE_SIDEBAR, GtkSidebar))
|
||||
#define GTK_IS_SIDEBAR(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), GTK_TYPE_SIDEBAR))
|
||||
#define GTK_SIDEBAR_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), GTK_TYPE_SIDEBAR, GtkSidebarClass))
|
||||
#define GTK_IS_SIDEBAR_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), GTK_TYPE_SIDEBAR))
|
||||
#define GTK_SIDEBAR_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), GTK_TYPE_SIDEBAR, GtkSidebarClass))
|
||||
|
||||
typedef struct _GtkSidebar GtkSidebar;
|
||||
typedef struct _GtkSidebarPrivate GtkSidebarPrivate;
|
||||
typedef struct _GtkSidebarClass GtkSidebarClass;
|
||||
|
||||
struct _GtkSidebar
|
||||
{
|
||||
GtkListBox parent;
|
||||
};
|
||||
|
||||
struct _GtkSidebarClass
|
||||
{
|
||||
GtkListBoxClass parent_class;
|
||||
|
||||
/* Padding for future expansion */
|
||||
void (*_gtk_reserved1) (void);
|
||||
void (*_gtk_reserved2) (void);
|
||||
void (*_gtk_reserved3) (void);
|
||||
void (*_gtk_reserved4) (void);
|
||||
};
|
||||
|
||||
GDK_AVAILABLE_IN_3_16
|
||||
GType gtk_sidebar_get_type (void) G_GNUC_CONST;
|
||||
GDK_AVAILABLE_IN_3_16
|
||||
GtkWidget * gtk_sidebar_new (void);
|
||||
GDK_AVAILABLE_IN_3_16
|
||||
void gtk_sidebar_set_stack (GtkSidebar *sidebar,
|
||||
GtkStack *stack);
|
||||
GDK_AVAILABLE_IN_3_16
|
||||
GtkStack * gtk_sidebar_get_stack (GtkSidebar *sidebar);
|
||||
|
||||
G_END_DECLS
|
||||
|
||||
#endif /* __GTK_SIDEBAR_H__ */
|
||||
@@ -283,6 +283,7 @@ gtk_OBJECTS = \
|
||||
gtkseparatortoolitem.obj \
|
||||
gtksettings.obj \
|
||||
gtkshow.obj \
|
||||
gtksidebar.obj \
|
||||
gtksizegroup.obj \
|
||||
gtksizerequest.obj \
|
||||
gtkspinbutton.obj \
|
||||
@@ -475,6 +476,7 @@ gtk_public_h_sources = \
|
||||
gtkseparatortoolitem.h \
|
||||
gtksettings.h \
|
||||
gtkshow.h \
|
||||
gtksidebar.h \
|
||||
gtksizegroup.h \
|
||||
gtksizerequest.h \
|
||||
gtksocket.h \
|
||||
|
||||
@@ -2,6 +2,7 @@
|
||||
|
||||
GtkWidget *stack;
|
||||
GtkWidget *switcher;
|
||||
GtkWidget *sidebar;
|
||||
GtkWidget *w1;
|
||||
|
||||
static void
|
||||
@@ -99,7 +100,7 @@ gint
|
||||
main (gint argc,
|
||||
gchar ** argv)
|
||||
{
|
||||
GtkWidget *window, *box, *button, *hbox, *combo;
|
||||
GtkWidget *window, *box, *button, *hbox, *combo, *layout;
|
||||
GtkWidget *w2, *w3;
|
||||
GtkListStore* store;
|
||||
GtkWidget *tree_view;
|
||||
@@ -127,7 +128,15 @@ main (gint argc,
|
||||
gtk_stack_set_transition_duration (GTK_STACK (stack), 1500);
|
||||
|
||||
gtk_widget_set_halign (stack, GTK_ALIGN_START);
|
||||
gtk_container_add (GTK_CONTAINER (box), stack);
|
||||
|
||||
/* Add sidebar before stack */
|
||||
sidebar = gtk_sidebar_new ();
|
||||
gtk_sidebar_set_stack (GTK_SIDEBAR (sidebar), GTK_STACK (stack));
|
||||
layout = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
|
||||
gtk_box_pack_start (GTK_BOX (layout), sidebar, FALSE, FALSE, 0);
|
||||
gtk_box_pack_start (GTK_BOX (layout), stack, TRUE, TRUE, 0);
|
||||
|
||||
gtk_container_add (GTK_CONTAINER (box), layout);
|
||||
|
||||
gtk_stack_switcher_set_stack (GTK_STACK_SWITCHER (switcher), GTK_STACK (stack));
|
||||
|
||||
|
||||
Reference in New Issue
Block a user