aboutsummaryrefslogtreecommitdiffstats
path: root/shell/e-sidebar.c
diff options
context:
space:
mode:
Diffstat (limited to 'shell/e-sidebar.c')
-rw-r--r--shell/e-sidebar.c196
1 files changed, 172 insertions, 24 deletions
diff --git a/shell/e-sidebar.c b/shell/e-sidebar.c
index fc1ca4dd90..7a7932eff0 100644
--- a/shell/e-sidebar.c
+++ b/shell/e-sidebar.c
@@ -33,6 +33,8 @@
#include <gtk/gtklabel.h>
#include <gtk/gtktogglebutton.h>
+#include <gconf/gconf-client.h>
+#include <libgnome/gnome-gconf.h>
typedef struct {
GtkWidget *button_widget;
@@ -44,10 +46,15 @@ typedef struct {
struct _ESidebarPrivate {
ESidebarMode mode;
+ ESidebarMode toolbar_mode;
+
+ gboolean show;
GtkWidget *selection_widget;
GSList *buttons;
+ guint style_changed_id;
+
gboolean in_toggle;
};
@@ -61,10 +68,10 @@ static unsigned int signals[NUM_SIGNALS] = { 0 };
G_DEFINE_TYPE (ESidebar, e_sidebar, GTK_TYPE_CONTAINER)
+#define INTERNAL_MODE(sidebar) (sidebar->priv->mode == E_SIDEBAR_MODE_TOOLBAR ? sidebar->priv->toolbar_mode : sidebar->priv->mode)
#define H_PADDING 6
#define V_PADDING 6
-
/* Utility functions. */
static Button *
@@ -154,7 +161,8 @@ static int
layout_buttons (ESidebar *sidebar)
{
GtkAllocation *allocation = & GTK_WIDGET (sidebar)->allocation;
- gboolean icons_only = (sidebar->priv->mode == E_SIDEBAR_MODE_ICON);
+ ESidebarMode mode;
+ gboolean icons_only;
int num_btns = g_slist_length (sidebar->priv->buttons), btns_per_row;
GSList **rows, *p;
Button *button;
@@ -169,6 +177,9 @@ layout_buttons (ESidebar *sidebar)
if (num_btns == 0)
return y;
+ mode = INTERNAL_MODE (sidebar);
+ icons_only = (mode == E_SIDEBAR_MODE_ICON);
+
/* Figure out the max width and height */
for (p = sidebar->priv->buttons; p != NULL; p = p->next) {
GtkRequisition requisition;
@@ -223,7 +234,7 @@ layout_buttons (ESidebar *sidebar)
y -= max_btn_height;
x = H_PADDING + allocation->x;
len = g_slist_length (rows[i]);
- if (sidebar->priv->mode == E_SIDEBAR_MODE_TEXT)
+ if (mode == E_SIDEBAR_MODE_TEXT || mode == E_SIDEBAR_MODE_BOTH)
extra_width = (allocation->width - (len * max_btn_width ) - (len * H_PADDING)) / len;
else
extra_width = 0;
@@ -257,7 +268,10 @@ do_layout (ESidebar *sidebar)
GtkAllocation child_allocation;
int y;
- y = layout_buttons (sidebar);
+ if (sidebar->priv->show)
+ y = layout_buttons (sidebar);
+ else
+ y = allocation->y + allocation->height;
/* Place the selection widget. */
child_allocation.x = allocation->x;
@@ -330,6 +344,9 @@ impl_size_request (GtkWidget *widget,
gtk_widget_size_request (sidebar->priv->selection_widget, requisition);
}
+ if (!sidebar->priv->show)
+ return;
+
for (p = sidebar->priv->buttons; p != NULL; p = p->next) {
Button *button = p->data;
GtkRequisition button_requisition;
@@ -357,11 +374,19 @@ static void
impl_dispose (GObject *object)
{
ESidebarPrivate *priv = E_SIDEBAR (object)->priv;
+ GConfClient *gconf_client = gconf_client_get_default ();
g_slist_foreach (priv->buttons, (GFunc) button_free, NULL);
g_slist_free (priv->buttons);
priv->buttons = NULL;
+ if (priv->style_changed_id) {
+ gconf_client_notify_remove (gconf_client, priv->style_changed_id);
+ priv->style_changed_id = 0;
+ }
+
+ g_object_unref (gconf_client);
+
(* G_OBJECT_CLASS (e_sidebar_parent_class)->dispose) (object);
}
@@ -418,7 +443,6 @@ e_sidebar_init (ESidebar *sidebar)
priv->mode = E_SIDEBAR_MODE_TEXT;
}
-
GtkWidget *
e_sidebar_new (void)
{
@@ -455,23 +479,40 @@ e_sidebar_add_button (ESidebar *sidebar,
GtkWidget *label_widget;
button_widget = gtk_toggle_button_new ();
+ if (sidebar->priv->show)
+ gtk_widget_show (button_widget);
g_signal_connect (button_widget, "toggled", G_CALLBACK (button_toggled_callback), sidebar);
hbox = gtk_hbox_new (FALSE, 3);
gtk_container_set_border_width (GTK_CONTAINER (hbox), 2);
+ gtk_widget_show (hbox);
+
icon_widget = gtk_image_new_from_pixbuf (icon);
+ gtk_widget_show (icon_widget);
+
label_widget = gtk_label_new (label);
gtk_misc_set_alignment (GTK_MISC (label_widget), 0.0, 0.5);
- gtk_box_pack_start (GTK_BOX (hbox), icon_widget, sidebar->priv->mode == E_SIDEBAR_MODE_ICON, TRUE, 0);
- gtk_box_pack_start (GTK_BOX (hbox), label_widget, TRUE, TRUE, 0);
+ gtk_widget_show (label_widget);
+
+ switch (INTERNAL_MODE (sidebar)) {
+ case E_SIDEBAR_MODE_TEXT:
+ gtk_box_pack_start (GTK_BOX (hbox), label_widget, TRUE, TRUE, 0);
+ break;
+ case E_SIDEBAR_MODE_ICON:
+ gtk_box_pack_start (GTK_BOX (hbox), icon_widget, TRUE, TRUE, 0);
+ break;
+ case E_SIDEBAR_MODE_BOTH:
+ default:
+ gtk_box_pack_start (GTK_BOX (hbox), icon_widget, FALSE, TRUE, 0);
+ gtk_box_pack_start (GTK_BOX (hbox), label_widget, TRUE, TRUE, 0);
+ break;
+ }
+
gtk_container_add (GTK_CONTAINER (button_widget), hbox);
sidebar->priv->buttons = g_slist_append (sidebar->priv->buttons, button_new (button_widget, label_widget, icon_widget, hbox, id));
gtk_widget_set_parent (button_widget, GTK_WIDGET (sidebar));
- if (sidebar->priv->mode == E_SIDEBAR_MODE_ICON)
- gtk_container_remove (GTK_CONTAINER (hbox), label_widget);
-
gtk_widget_queue_resize (GTK_WIDGET (sidebar));
}
@@ -490,12 +531,22 @@ e_sidebar_get_mode (ESidebar *sidebar)
return sidebar->priv->mode;
}
-void
-e_sidebar_set_mode (ESidebar *sidebar, ESidebarMode mode)
+
+static GConfEnumStringPair toolbar_styles[] = {
+ { E_SIDEBAR_MODE_TEXT, "text" },
+ { E_SIDEBAR_MODE_ICON, "icons" },
+ { E_SIDEBAR_MODE_BOTH, "both" },
+ { E_SIDEBAR_MODE_BOTH, "both-horiz" },
+ { E_SIDEBAR_MODE_BOTH, "both_horiz" },
+ { -1, NULL }
+};
+
+static void
+set_mode_internal (ESidebar *sidebar, ESidebarMode mode )
{
GSList *p;
-
- if (sidebar->priv->mode == mode)
+
+ if (mode == INTERNAL_MODE (sidebar))
return;
for (p = sidebar->priv->buttons; p != NULL; p = p->next) {
@@ -503,24 +554,121 @@ e_sidebar_set_mode (ESidebar *sidebar, ESidebarMode mode)
switch (mode) {
case E_SIDEBAR_MODE_TEXT:
- gtk_box_pack_start (GTK_BOX (button->hbox), button->label, TRUE, TRUE, 0);
- gtk_container_child_set (GTK_CONTAINER (button->hbox), button->icon,
- "expand", FALSE,
- NULL);
+ gtk_container_remove (GTK_CONTAINER (button->hbox), button->icon);
+ if (INTERNAL_MODE (sidebar) == E_SIDEBAR_MODE_ICON) {
+ gtk_box_pack_start (GTK_BOX (button->hbox), button->label, TRUE, TRUE, 0);
+ gtk_widget_show (button->label);
+ }
break;
case E_SIDEBAR_MODE_ICON:
- gtk_container_remove (GTK_CONTAINER (button->hbox), button->label);
- gtk_container_child_set (GTK_CONTAINER (button->hbox), button->icon,
- "expand", TRUE,
- NULL);
+ gtk_container_remove(GTK_CONTAINER (button->hbox), button->label);
+ if (INTERNAL_MODE (sidebar) == E_SIDEBAR_MODE_TEXT) {
+ gtk_box_pack_start (GTK_BOX (button->hbox), button->icon, TRUE, TRUE, 0);
+ gtk_widget_show (button->icon);
+ } else
+ gtk_container_child_set (GTK_CONTAINER (button->hbox), button->icon,
+ "expand", TRUE,
+ NULL);
+ break;
+ case E_SIDEBAR_MODE_BOTH:
+ if (INTERNAL_MODE (sidebar) == E_SIDEBAR_MODE_TEXT) {
+ gtk_container_remove (GTK_CONTAINER (button->hbox), button->label);
+ gtk_box_pack_start (GTK_BOX (button->hbox), button->icon, FALSE, TRUE, 0);
+ gtk_widget_show (button->icon);
+ } else {
+ gtk_container_child_set (GTK_CONTAINER (button->hbox), button->icon,
+ "expand", FALSE,
+ NULL);
+ }
+
+ gtk_box_pack_start (GTK_BOX (button->hbox), button->label, TRUE, TRUE, 0);
+ gtk_widget_show (button->label);
break;
default:
- g_assert_not_reached ();
- return;
+ break;
}
}
+}
+
+static void
+style_changed_notify (GConfClient *gconf, guint id, GConfEntry *entry, void *data)
+{
+ ESidebar *sidebar = data;
+ char *val;
+ int mode;
+
+ val = gconf_client_get_string (gconf, "/desktop/gnome/interface/toolbar_style", NULL);
+ if (val == NULL || !gconf_string_to_enum (toolbar_styles, val, &mode))
+ mode = E_SIDEBAR_MODE_BOTH;
+ g_free(val);
+
+ set_mode_internal (E_SIDEBAR (sidebar), mode);
+ sidebar->priv->toolbar_mode = mode;
+
+ gtk_widget_queue_resize (GTK_WIDGET (sidebar));
+}
+
+void
+e_sidebar_set_mode (ESidebar *sidebar, ESidebarMode mode)
+{
+ GConfClient *gconf_client = gconf_client_get_default ();
+
+ if (sidebar->priv->mode == mode)
+ return;
+
+ if (sidebar->priv->mode == E_SIDEBAR_MODE_TOOLBAR) {
+ if (sidebar->priv->style_changed_id) {
+ gconf_client_notify_remove (gconf_client, sidebar->priv->style_changed_id);
+ sidebar->priv->style_changed_id = 0;
+ }
+ }
+
+ if (mode != E_SIDEBAR_MODE_TOOLBAR) {
+ set_mode_internal (sidebar, mode);
+
+ gtk_widget_queue_resize (GTK_WIDGET (sidebar));
+ } else {
+ /* This is a little bit tricky, toolbar mode is more
+ * of a meta-mode where the actual mode is dictated by
+ * the gnome toolbar setting, so that is why we have
+ * the is_toolbar_mode bool - it tracks the toolbar
+ * mode while the mode member is the actual look and
+ * feel */
+ sidebar->priv->style_changed_id = gconf_client_notify_add (gconf_client,
+ "/desktop/gnome/interface/toolbar_style",
+ style_changed_notify, sidebar, NULL, NULL);
+ style_changed_notify (gconf_client, 0, NULL, sidebar);
+ }
+
+ g_object_unref (gconf_client);
sidebar->priv->mode = mode;
+}
+
+void
+e_sidebar_set_show_buttons (ESidebar *sidebar, gboolean show)
+{
+ GSList *p;
+
+ if (sidebar->priv->show == show)
+ return;
+
+ for (p = sidebar->priv->buttons; p != NULL; p = p->next) {
+ Button *button = p->data;
+
+ if (show)
+ gtk_widget_show (button->button_widget);
+ else
+ gtk_widget_hide (button->button_widget);
+ }
+
+ sidebar->priv->show = show;
gtk_widget_queue_resize (GTK_WIDGET (sidebar));
}
+
+gboolean
+e_sidebar_get_show_buttons (ESidebar *sidebar)
+{
+ return sidebar->priv->show;
+}